2019-05-07 14:49:22 -05:00
|
|
|
import { __ } from 'embark-i18n';
|
2019-05-02 12:09:12 +02:00
|
|
|
const async = require('async');
|
2018-07-27 12:48:06 -04:00
|
|
|
const ContractDeployer = require('./contract_deployer.js');
|
2018-09-30 12:48:44 -04:00
|
|
|
const cloneDeep = require('clone-deep');
|
2019-04-29 16:39:48 -05:00
|
|
|
const constants = require('embark-core/constants');
|
2018-07-24 13:29:06 +01:00
|
|
|
|
2017-03-30 20:12:39 +09:00
|
|
|
class DeployManager {
|
2018-09-30 12:48:44 -04:00
|
|
|
constructor(embark, options) {
|
2018-05-29 17:23:29 -04:00
|
|
|
const self = this;
|
2018-09-30 12:48:44 -04:00
|
|
|
this.config = embark.config;
|
|
|
|
this.logger = embark.logger;
|
2017-03-30 20:12:39 +09:00
|
|
|
this.blockchainConfig = this.config.blockchainConfig;
|
2017-12-21 11:21:36 -05:00
|
|
|
|
2018-09-30 12:48:44 -04:00
|
|
|
this.events = embark.events;
|
2017-03-30 20:12:39 +09:00
|
|
|
this.plugins = options.plugins;
|
2018-05-18 18:31:47 -04:00
|
|
|
this.blockchain = options.blockchain;
|
2018-11-13 13:58:30 +00:00
|
|
|
this.gasLimit = 6000000;
|
2018-01-13 11:38:10 -05:00
|
|
|
this.fatalErrors = false;
|
2018-03-11 08:28:03 -04:00
|
|
|
this.deployOnlyOnConfig = false;
|
2018-03-22 15:09:01 -04:00
|
|
|
this.onlyCompile = options.onlyCompile !== undefined ? options.onlyCompile : false;
|
2018-05-29 17:23:29 -04:00
|
|
|
|
2018-09-30 12:48:44 -04:00
|
|
|
this.contractDeployer = new ContractDeployer({
|
|
|
|
logger: this.logger,
|
|
|
|
events: this.events,
|
|
|
|
plugins: this.plugins
|
|
|
|
});
|
|
|
|
|
|
|
|
this.events.setCommandHandler('deploy:setGasLimit', (gasLimit) => {
|
|
|
|
self.gasLimit = gasLimit;
|
|
|
|
});
|
|
|
|
|
2018-05-29 17:23:29 -04:00
|
|
|
this.events.setCommandHandler('deploy:contracts', (cb) => {
|
|
|
|
self.deployContracts(cb);
|
|
|
|
});
|
2018-09-30 12:48:44 -04:00
|
|
|
|
|
|
|
this.events.setCommandHandler('deploy:contracts:test', (cb) => {
|
|
|
|
self.fatalErrors = true;
|
|
|
|
self.deployOnlyOnConfig = true;
|
|
|
|
self.deployContracts(cb);
|
|
|
|
});
|
2018-05-29 17:23:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
deployAll(done) {
|
|
|
|
let self = this;
|
|
|
|
|
2018-08-22 18:36:34 -04:00
|
|
|
self.events.request('contracts:dependencies', (err, contractDependencies) => {
|
|
|
|
self.events.request('contracts:list', (err, contracts) => {
|
|
|
|
if (err) {
|
|
|
|
return done(err);
|
|
|
|
}
|
2018-09-22 14:43:10 +01:00
|
|
|
|
2018-08-22 18:36:34 -04:00
|
|
|
self.logger.info(__("deploying contracts"));
|
2018-09-29 21:13:55 -04:00
|
|
|
async.waterfall([
|
|
|
|
function (next) {
|
2018-12-12 11:09:05 -05:00
|
|
|
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();
|
|
|
|
});
|
2018-09-29 21:13:55 -04:00
|
|
|
},
|
2019-04-16 14:32:26 +02:00
|
|
|
function (next) {
|
2018-09-29 21:13:55 -04:00
|
|
|
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;
|
2018-11-15 14:19:44 -05:00
|
|
|
if (contract.error === constants.blockchain.gasAllowanceError) {
|
2019-04-22 10:37:56 -04:00
|
|
|
self.logger.error(`[${contract.className}]: ${constants.blockchain.gasAllowanceErrorMessage}`);
|
|
|
|
} else {
|
|
|
|
self.logger.error(`[${contract.className}]: ${err.message || err}`);
|
2018-11-15 14:19:44 -05:00
|
|
|
}
|
2018-09-29 21:13:55 -04:00
|
|
|
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);
|
|
|
|
});
|
2018-08-09 12:58:41 -04:00
|
|
|
|
2018-11-13 13:58:30 +00:00
|
|
|
async.auto(contractDeploys, function(_err, _results) {
|
2018-11-08 14:14:31 +01:00
|
|
|
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"));
|
2019-04-16 14:32:26 +02:00
|
|
|
return next(_err);
|
2018-11-08 14:14:31 +01:00
|
|
|
}
|
|
|
|
if (contracts.length === 0) {
|
|
|
|
self.logger.info(__("no contracts found"));
|
2019-04-16 14:32:26 +02:00
|
|
|
return next();
|
2018-11-08 14:14:31 +01:00
|
|
|
}
|
|
|
|
self.logger.info(__("finished deploying contracts"));
|
2019-04-16 14:32:26 +02:00
|
|
|
next(err);
|
2018-11-08 14:14:31 +01:00
|
|
|
});
|
2018-09-22 15:08:50 +01:00
|
|
|
}
|
2019-04-16 14:32:26 +02:00
|
|
|
], (err) => {
|
2019-04-17 18:26:38 +02:00
|
|
|
if (err) {
|
|
|
|
self.logger.error(err);
|
|
|
|
}
|
2019-04-16 14:32:26 +02:00
|
|
|
done(err);
|
|
|
|
});
|
2018-08-22 18:36:34 -04:00
|
|
|
});
|
2018-09-22 15:08:50 +01:00
|
|
|
});
|
2017-02-28 20:42:03 -05:00
|
|
|
}
|
|
|
|
|
2017-03-30 20:12:39 +09:00
|
|
|
deployContracts(done) {
|
|
|
|
let self = this;
|
|
|
|
|
|
|
|
if (self.blockchainConfig === {} || self.blockchainConfig.enabled === false) {
|
2018-05-08 17:49:46 -04:00
|
|
|
self.logger.info(__("Blockchain component is disabled in the config").underline);
|
2017-03-31 07:34:43 -04:00
|
|
|
this.events.emit('blockchainDisabled', {});
|
2017-03-30 20:12:39 +09:00
|
|
|
return done();
|
2017-03-11 10:29:45 -05:00
|
|
|
}
|
2017-03-30 20:12:39 +09:00
|
|
|
|
|
|
|
async.waterfall([
|
2018-09-30 12:48:44 -04:00
|
|
|
function requestBlockchainConnector(callback) {
|
|
|
|
self.events.request("blockchain:object", (blockchain) => {
|
|
|
|
self.blockchain = blockchain;
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2017-03-30 20:12:39 +09:00
|
|
|
function buildContracts(callback) {
|
2018-06-08 07:07:27 -04:00
|
|
|
self.events.request("contracts:build", self.deployOnlyOnConfig, (err) => {
|
|
|
|
callback(err);
|
2018-05-20 09:15:28 -04:00
|
|
|
});
|
2017-03-30 20:12:39 +09:00
|
|
|
},
|
2018-05-20 10:46:36 -04:00
|
|
|
|
|
|
|
// TODO: shouldn't be necessary
|
2018-09-30 12:48:44 -04:00
|
|
|
function checkCompileOnly(callback) {
|
|
|
|
if (self.onlyCompile) {
|
2018-05-30 06:56:51 -04:00
|
|
|
self.events.emit('contractsDeployed');
|
2018-03-22 15:09:01 -04:00
|
|
|
return done();
|
2018-08-09 12:58:41 -04:00
|
|
|
}
|
2018-05-20 09:15:28 -04:00
|
|
|
return callback();
|
2018-03-22 15:09:01 -04:00
|
|
|
},
|
2018-05-20 06:46:12 -04:00
|
|
|
|
2018-07-26 13:15:20 -04:00
|
|
|
// TODO: could be implemented as an event (beforeDeployAll)
|
|
|
|
function checkIsConnectedToBlockchain(callback) {
|
2018-09-30 12:48:44 -04:00
|
|
|
self.blockchain.onReady((err) => {
|
|
|
|
callback(err);
|
|
|
|
});
|
2018-07-26 13:15:20 -04:00
|
|
|
},
|
2018-05-20 09:08:03 -04:00
|
|
|
|
|
|
|
// TODO: this can be done on the fly or as part of the initialization
|
2018-05-20 09:15:28 -04:00
|
|
|
function determineDefaultAccount(callback) {
|
2018-05-20 09:08:03 -04:00
|
|
|
self.blockchain.determineDefaultAccount((err) => {
|
2018-05-20 09:15:28 -04:00
|
|
|
callback(err);
|
2017-03-30 20:12:39 +09:00
|
|
|
});
|
|
|
|
},
|
2018-05-20 06:46:12 -04:00
|
|
|
|
2018-05-20 09:15:28 -04:00
|
|
|
function deployAllContracts(callback) {
|
2018-05-29 17:23:29 -04:00
|
|
|
self.deployAll(function (err) {
|
2018-05-18 22:40:47 -04:00
|
|
|
if (!err) {
|
2018-05-30 06:56:51 -04:00
|
|
|
self.events.emit('contractsDeployed');
|
2018-05-18 22:40:47 -04:00
|
|
|
}
|
|
|
|
if (err && self.fatalErrors) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2018-05-20 09:15:28 -04:00
|
|
|
callback();
|
2017-03-30 20:12:39 +09:00
|
|
|
});
|
2017-12-21 11:21:36 -05:00
|
|
|
},
|
2018-05-20 10:46:36 -04:00
|
|
|
function runAfterDeploy(callback) {
|
2018-05-30 08:00:31 -04:00
|
|
|
self.plugins.emitAndRunActionsForEvent('contracts:deploy:afterAll', callback);
|
2017-02-24 08:20:03 -05:00
|
|
|
}
|
2018-05-20 09:15:28 -04:00
|
|
|
], function (err, _result) {
|
|
|
|
done(err);
|
2017-02-24 08:20:03 -05:00
|
|
|
});
|
|
|
|
}
|
2017-03-30 20:12:39 +09:00
|
|
|
|
|
|
|
}
|
2017-02-24 08:20:03 -05:00
|
|
|
|
|
|
|
module.exports = DeployManager;
|