173 lines
5.6 KiB
JavaScript
Raw Normal View History

2018-05-20 12:23:48 -04:00
const stringReplaceAsync = require('string-replace-async');
const async = require('async');
class SpecialConfigs {
constructor(embark, options) {
this.logger = embark.logger;
this.events = embark.events;
this.buildDir = options.buildDir;
this.embark = embark;
2018-10-18 08:37:28 -04:00
this.config = embark.config;
this.registerAfterDeployAction();
this.registerOnDeployAction();
2018-08-23 18:00:52 -04:00
this.registerDeployIfAction();
}
2018-10-05 14:49:10 +05:30
replaceWithENSAddress(cmd, cb) {
const self = this;
let regex = /\'[a-zA-Z0-9.]+\.eth\'/g;
return stringReplaceAsync.seq(cmd, regex, (ensDomain) => {
ensDomain = ensDomain.slice(1, ensDomain.length - 1);
return (new Promise((resolve, reject) => {
self.events.request("ens:resolve", ensDomain, (err, address) => {
if(err) {
return reject(new Error(err));
}
address = `'${address}'`;
return resolve(address);
});
}));
}).then((address) => {
cb(null, address);
}).catch(cb);
}
2018-05-20 12:23:48 -04:00
replaceWithAddresses(cmd, cb) {
const self = this;
2018-10-05 14:49:10 +05:30
2018-05-20 12:23:48 -04:00
let regex = /\$\w+/g;
stringReplaceAsync.seq(cmd, regex, (match) => {
return (new Promise((resolve, reject) => {
let referedContractName = match.slice(1);
self.events.request('contracts:contract', referedContractName, (referedContract) => {
if (!referedContract) {
self.logger.error(referedContractName + ' does not exist');
self.logger.error("error running cmd: " + cmd);
2018-05-20 12:23:48 -04:00
return reject(new Error("ReferedContractDoesNotExist"));
}
if (referedContract && referedContract.deploy === false) {
self.logger.error(referedContractName + " exists but has been set to not deploy");
self.logger.error("error running cmd: " + cmd);
2018-05-20 12:23:48 -04:00
return reject(new Error("ReferedContracSetToNotdeploy"));
}
if (referedContract && !referedContract.deployedAddress) {
self.logger.error("couldn't find a valid address for " + referedContractName + ". has it been deployed?");
self.logger.error("error running cmd: " + cmd);
2018-05-20 12:23:48 -04:00
return reject(new Error("ReferedContractAddressNotFound"));
}
2018-05-20 12:23:48 -04:00
return resolve(referedContract.deployedAddress);
});
2018-05-20 12:23:48 -04:00
}));
2018-10-05 14:49:10 +05:30
}).then((address) => {
cb(null, address);
2018-05-20 12:23:48 -04:00
}).catch(cb);
}
registerAfterDeployAction() {
const self = this;
this.embark.registerActionForEvent("contracts:deploy:afterAll", (cb) => {
2018-10-18 08:37:28 -04:00
let afterDeployCmds = self.config.contractsConfig.afterDeploy || [];
2018-05-20 12:23:48 -04:00
async.mapLimit(afterDeployCmds, 1, (cmd, nextMapCb) => {
2018-10-05 14:49:10 +05:30
async.waterfall([
function replaceWithAddresses(next) {
self.replaceWithAddresses(cmd, next);
},
self.replaceWithENSAddress.bind(self)
], nextMapCb);
}, (err, onDeployCode) => {
2018-05-20 12:23:48 -04:00
if (err) {
2018-06-01 14:06:02 +10:00
self.logger.trace(err);
2018-05-20 12:23:48 -04:00
return cb(new Error("error running afterDeploy"));
}
self.runOnDeployCode(onDeployCode, cb);
2018-05-20 12:23:48 -04:00
});
});
}
2018-08-07 16:04:26 -04:00
runOnDeployCode(onDeployCode, callback, silent) {
const self = this;
2018-08-07 16:04:26 -04:00
const logFunction = silent ? self.logger.trace.bind(self.logger) : self.logger.info.bind(self.logger);
async.each(onDeployCode, (cmd, eachCb) => {
if (!cmd) {
return eachCb();
}
2018-08-07 16:04:26 -04:00
logFunction("==== executing: " + cmd);
self.events.request('runcode:eval', cmd, (err) => {
if (err && err.message.indexOf("invalid opcode") >= 0) {
self.logger.error('the transaction was rejected; this usually happens due to a throw or a require, it can also happen due to an invalid operation');
}
eachCb(err);
});
}, callback);
}
registerOnDeployAction() {
const self = this;
2018-05-28 19:40:55 -04:00
this.embark.registerActionForEvent("deploy:contract:deployed", (params, cb) => {
let contract = params.contract;
2018-09-03 14:01:26 -04:00
if (!contract.onDeploy || contract.deploy === false) {
return cb();
}
2018-10-05 14:49:10 +05:30
2018-08-07 16:04:26 -04:00
if (!contract.silent) {
self.logger.info(__('executing onDeploy commands'));
}
let onDeployCmds = contract.onDeploy;
async.mapLimit(onDeployCmds, 1, (cmd, nextMapCb) => {
2018-10-05 14:49:10 +05:30
async.waterfall([
function replaceWithAddresses(next) {
self.replaceWithAddresses(cmd, next);
},
self.replaceWithENSAddress.bind(self)
], (err, code) => {
if (err) {
self.logger.error(err.message || err);
return nextMapCb(); // Don't return error as we just skip the failing command
}
nextMapCb(null, code);
});
}, (err, onDeployCode) => {
if (err) {
return cb(new Error("error running onDeploy for " + contract.className.cyan));
}
2018-08-07 16:04:26 -04:00
self.runOnDeployCode(onDeployCode, cb, contract.silent);
});
});
}
registerDeployIfAction() {
const self = this;
self.embark.registerActionForEvent("deploy:contract:shouldDeploy", (params, cb) => {
2018-08-23 18:00:52 -04:00
let cmd = params.contract.deployIf;
if (!cmd) {
return cb(params);
}
2018-08-23 18:00:52 -04:00
self.events.request('runcode:eval', cmd, (err, result) => {
if (err) {
self.logger.error(params.contract.className + ' deployIf directive has an error; contract will not deploy');
self.logger.error(err);
params.shouldDeploy = false;
} else if (!result) {
self.logger.info(params.contract.className + ' deployIf directive returned false; contract will not deploy');
params.shouldDeploy = false;
}
cb(params);
});
});
}
}
module.exports = SpecialConfigs;