189 lines
5.9 KiB
JavaScript
Raw Normal View History

import { __ } from 'embark-i18n';
const async = require('async');
const ContractDeployer = require('./contract_deployer.js');
2018-09-30 12:48:44 -04:00
const cloneDeep = require('clone-deep');
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) {
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;
this.deployOnlyOnConfig = false;
this.onlyCompile = options.onlyCompile !== undefined ? options.onlyCompile : false;
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;
});
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);
});
}
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) {
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
},
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;
if (contract.error === constants.blockchain.gasAllowanceError) {
self.logger.error(`[${contract.className}]: ${constants.blockchain.gasAllowanceErrorMessage}`);
} else {
self.logger.error(`[${contract.className}]: ${err.message || err}`);
}
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) {
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);
});
2018-08-22 18:36:34 -04: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-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) {
self.events.request("contracts:build", self.deployOnlyOnConfig, (err) => {
callback(err);
});
2017-03-30 20:12:39 +09: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');
return done();
2018-08-09 12:58:41 -04:00
}
return callback();
},
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
},
// TODO: this can be done on the fly or as part of the initialization
function determineDefaultAccount(callback) {
self.blockchain.determineDefaultAccount((err) => {
callback(err);
2017-03-30 20:12:39 +09:00
});
},
2018-05-20 06:46:12 -04:00
function deployAllContracts(callback) {
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);
}
callback();
2017-03-30 20:12:39 +09:00
});
2017-12-21 11:21:36 -05: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
}
], 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;