mirror of
https://github.com/embarklabs/embark.git
synced 2025-01-11 14:24:24 +00:00
d76a82a30a
When running tests against non-simulated blockchain nodes, even for simplex
Smart Contracts, deployment transactions would exceed the block gas limit.
E.g. running `embark blockchain` in one process and `embark test --node embark`
in another, inside our demo application, will throw an error when Embark attempts
to deploy its `SimpleStorage`:
```
Compiling contracts
Compilation done
[SimpleStorage]: error deploying =SimpleStorage= due to error: Returned error: exceeds block gas limit
Error deploying contracts. Please fix errors to continue.
Error deploying contracts. Please fix errors to continue.
terminating due to error
Error deploying contracts. Please fix errors to continue.
```
The reason for that is because in https://github.com/embark-framework/embark/pull/1650, we've introduced a static
gas estimation for Smart Contract deployment that is just right below Ganache's
maximum gas limit of `6721975`, since Ganache tends to underestimate gas for
complex Smart Contracts due to its [low base fee](8ad1ab29de/lib/utils/gasEstimation.js (L33-L39)
).
The static gas estimation would apply any time we're in a test context, but we
didn't take into account the case where tests are executed against nodes
other than the simulated environment.
As mentioned in the comments in the linked PR:
> If this is not spec'ed at all, I wonder what complications it could cause when
> at some point we maybe switch to not using Ganache anymore for tests, or even
> the user itself (which I think is a reasonable thing to do).
This causes the error described above because we easily reach the block gas limit
with just two Smart Contracts and Embark already deploys a few Smart Contracts for
ENS.
So basically what we want is to use the static gas estimation when we know
the node we're connecting to is Ganache. In all other cases we can rely on the
standardized gas estimation offered by the node.
189 lines
5.9 KiB
JavaScript
189 lines
5.9 KiB
JavaScript
import { __ } from 'embark-i18n';
|
|
const async = require('async');
|
|
const ContractDeployer = require('./contract_deployer.js');
|
|
const cloneDeep = require('clone-deep');
|
|
const constants = require('embark-core/constants');
|
|
|
|
class DeployManager {
|
|
constructor(embark, options) {
|
|
const self = this;
|
|
this.config = embark.config;
|
|
this.logger = embark.logger;
|
|
this.blockchainConfig = this.config.blockchainConfig;
|
|
|
|
this.events = embark.events;
|
|
this.plugins = options.plugins;
|
|
this.blockchain = options.blockchain;
|
|
this.gasLimit = 6000000;
|
|
this.fatalErrors = false;
|
|
this.deployOnlyOnConfig = false;
|
|
this.onlyCompile = options.onlyCompile !== undefined ? options.onlyCompile : false;
|
|
|
|
this.contractDeployer = new ContractDeployer({
|
|
logger: this.logger,
|
|
events: this.events,
|
|
plugins: this.plugins
|
|
});
|
|
|
|
this.events.setCommandHandler('deploy:setGasLimit', (gasLimit) => {
|
|
self.gasLimit = gasLimit;
|
|
});
|
|
|
|
this.events.setCommandHandler('deploy:contracts', (cb) => {
|
|
self.deployContracts(cb);
|
|
});
|
|
|
|
this.events.setCommandHandler('deploy:contracts:test', (cb) => {
|
|
self.fatalErrors = true;
|
|
self.deployOnlyOnConfig = true;
|
|
self.deployContracts(cb);
|
|
});
|
|
}
|
|
|
|
deployAll(done) {
|
|
let self = this;
|
|
|
|
self.events.request('contracts:dependencies', (err, contractDependencies) => {
|
|
self.events.request('contracts:list', (err, contracts) => {
|
|
if (err) {
|
|
return done(err);
|
|
}
|
|
|
|
self.logger.info(__("deploying contracts"));
|
|
async.waterfall([
|
|
function (next) {
|
|
self.logger.info(__('Executing pre-deploy actions...'));
|
|
self.plugins.emitAndRunActionsForEvent("deploy:beforeAll", (err) => {
|
|
if (err) {
|
|
return next(err);
|
|
}
|
|
self.logger.info(__('Pre-deploy actions done. Deploying contracts'));
|
|
next();
|
|
});
|
|
},
|
|
function (next) {
|
|
const contractDeploys = {};
|
|
const errors = [];
|
|
contracts.forEach(contract => {
|
|
function deploy(result, callback) {
|
|
if (typeof result === 'function') {
|
|
callback = result;
|
|
}
|
|
contract._gasLimit = self.gasLimit;
|
|
self.events.request('deploy:contract', contract, (err) => {
|
|
if (err) {
|
|
contract.error = err.message || err;
|
|
if (contract.error === constants.blockchain.gasAllowanceError) {
|
|
self.logger.error(`[${contract.className}]: ${constants.blockchain.gasAllowanceErrorMessage}`);
|
|
} else {
|
|
self.logger.error(`[${contract.className}]: ${err.message || err}`);
|
|
}
|
|
errors.push(err);
|
|
}
|
|
callback();
|
|
});
|
|
}
|
|
|
|
const className = contract.className;
|
|
if (!contractDependencies[className] || contractDependencies[className].length === 0) {
|
|
contractDeploys[className] = deploy;
|
|
return;
|
|
}
|
|
contractDeploys[className] = cloneDeep(contractDependencies[className]);
|
|
contractDeploys[className].push(deploy);
|
|
});
|
|
|
|
async.auto(contractDeploys, function(_err, _results) {
|
|
if (errors.length) {
|
|
_err = __("Error deploying contracts. Please fix errors to continue.");
|
|
self.logger.error(_err);
|
|
self.events.emit("outputError", __("Error deploying contracts, please check console"));
|
|
return next(_err);
|
|
}
|
|
if (contracts.length === 0) {
|
|
self.logger.info(__("no contracts found"));
|
|
return next();
|
|
}
|
|
self.logger.info(__("finished deploying contracts"));
|
|
next(err);
|
|
});
|
|
}
|
|
], (err) => {
|
|
if (err) {
|
|
self.logger.error(err);
|
|
}
|
|
done(err);
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
deployContracts(done) {
|
|
let self = this;
|
|
|
|
if (self.blockchainConfig === {} || self.blockchainConfig.enabled === false) {
|
|
self.logger.info(__("Blockchain component is disabled in the config").underline);
|
|
this.events.emit('blockchainDisabled', {});
|
|
return done();
|
|
}
|
|
|
|
async.waterfall([
|
|
function requestBlockchainConnector(callback) {
|
|
self.events.request("blockchain:object", (blockchain) => {
|
|
self.blockchain = blockchain;
|
|
callback();
|
|
});
|
|
},
|
|
|
|
function buildContracts(callback) {
|
|
self.events.request("contracts:build", self.deployOnlyOnConfig, (err) => {
|
|
callback(err);
|
|
});
|
|
},
|
|
|
|
// TODO: shouldn't be necessary
|
|
function checkCompileOnly(callback) {
|
|
if (self.onlyCompile) {
|
|
self.events.emit('contractsDeployed');
|
|
return done();
|
|
}
|
|
return callback();
|
|
},
|
|
|
|
// TODO: could be implemented as an event (beforeDeployAll)
|
|
function checkIsConnectedToBlockchain(callback) {
|
|
self.blockchain.onReady((err) => {
|
|
callback(err);
|
|
});
|
|
},
|
|
|
|
// TODO: this can be done on the fly or as part of the initialization
|
|
function determineDefaultAccount(callback) {
|
|
self.blockchain.determineDefaultAccount((err) => {
|
|
callback(err);
|
|
});
|
|
},
|
|
|
|
function deployAllContracts(callback) {
|
|
self.deployAll(function (err) {
|
|
if (!err) {
|
|
self.events.emit('contractsDeployed');
|
|
}
|
|
if (err && self.fatalErrors) {
|
|
return callback(err);
|
|
}
|
|
callback();
|
|
});
|
|
},
|
|
function runAfterDeploy(callback) {
|
|
self.plugins.emitAndRunActionsForEvent('contracts:deploy:afterAll', callback);
|
|
}
|
|
], function (err, _result) {
|
|
done(err);
|
|
});
|
|
}
|
|
|
|
}
|
|
|
|
module.exports = DeployManager;
|