embark/lib/deploy.js

169 lines
5.5 KiB
JavaScript
Raw Normal View History

2016-08-13 10:48:00 -04:00
var async = require('async');
2016-08-14 08:04:34 -04:00
var Compiler = require('./compiler.js');
2016-09-24 21:10:47 -04:00
var DeployTracker = require('./deploy_tracker.js');
2016-10-02 17:57:33 -04:00
var ABIGenerator = require('./abi.js');
var web3;
2016-09-24 21:10:47 -04:00
var Deploy = function(options) {
this.web3 = options.web3;
this.contractsManager = options.contractsManager;
this.logger = options.logger;
this.env = options.env;
this.deployTracker = new DeployTracker({
logger: options.logger, chainConfig: options.chainConfig, web3: options.web3, env: this.env
});
};
Deploy.prototype.checkAndDeployContract = function(contract, params, callback) {
var self = this;
2016-10-22 13:29:41 -06:00
var suppliedArgs;
var realArgs;
var arg;
var l;
var contractName;
var referedContract;
2016-10-21 07:16:15 -04:00
contract.error = false;
if (contract.deploy === false) {
self.logger.contractsState(self.contractsManager.contractsState());
return callback();
}
if (contract.address !== undefined) {
// determine arguments
2016-10-22 13:29:41 -06:00
suppliedArgs = (params || contract.args);
realArgs = [];
2016-10-22 13:29:41 -06:00
for (l = 0; l < suppliedArgs.length; l++) {
arg = suppliedArgs[l];
if (arg[0] === "$") {
2016-10-22 13:29:41 -06:00
contractName = arg.substr(1);
referedContract = this.contractsManager.getContract(contractName);
realArgs.push(referedContract.deployedAddress);
} else {
realArgs.push(arg);
}
}
contract.deployedAddress = contract.address;
self.deployTracker.trackContract(contract.className, contract.code, realArgs, contract.address);
self.deployTracker.save();
self.logger.contractsState(self.contractsManager.contractsState());
return callback();
}
2016-09-25 02:30:03 -04:00
var trackedContract = self.deployTracker.getContract(contract.className, contract.code, contract.args);
2016-09-24 21:10:47 -04:00
2016-09-25 02:30:03 -04:00
if (trackedContract && this.web3.eth.getCode(trackedContract.address) !== "0x") {
self.logger.info(contract.className + " already deployed " + trackedContract.address);
contract.deployedAddress = trackedContract.address;
2016-09-24 21:10:47 -04:00
self.logger.contractsState(self.contractsManager.contractsState());
callback();
} else {
2016-09-27 00:55:35 -04:00
// determine arguments
2016-10-22 13:29:41 -06:00
suppliedArgs = (params || contract.args);
realArgs = [];
2016-09-27 00:55:35 -04:00
2016-10-22 13:29:41 -06:00
for (l = 0; l < suppliedArgs.length; l++) {
arg = suppliedArgs[l];
2016-09-27 00:55:35 -04:00
if (arg[0] === "$") {
2016-10-22 13:29:41 -06:00
contractName = arg.substr(1);
referedContract = this.contractsManager.getContract(contractName);
2016-09-27 00:55:35 -04:00
realArgs.push(referedContract.deployedAddress);
} else {
realArgs.push(arg);
}
}
this.deployContract(contract, realArgs, function(err, address) {
self.deployTracker.trackContract(contract.className, contract.code, realArgs, address);
2016-09-24 21:10:47 -04:00
self.deployTracker.save();
self.logger.contractsState(self.contractsManager.contractsState());
2016-10-02 17:57:33 -04:00
2016-10-22 17:29:06 -04:00
// TODO: replace evals with separate process so it's isolated and with
2016-10-02 17:57:33 -04:00
// a callback
if (contract.onDeploy !== undefined) {
self.logger.info('executing onDeploy commands');
var abiGenerator = new ABIGenerator({}, self.contractsManager);
web3 = self.web3;
var abi = abiGenerator.generateContracts(false);
2016-10-22 17:29:06 -04:00
eval(abi); // jshint ignore:line
2016-10-02 17:57:33 -04:00
var cmds = contract.onDeploy.join(';\n');
2016-10-22 17:29:06 -04:00
eval(cmds); // jshint ignore:line
2016-10-02 17:57:33 -04:00
}
2016-09-24 21:10:47 -04:00
callback();
});
}
2016-08-13 10:48:00 -04:00
2016-08-14 08:04:34 -04:00
};
Deploy.prototype.deployContract = function(contract, params, callback) {
2016-09-16 23:56:25 -04:00
var self = this;
2016-08-14 08:04:34 -04:00
var contractObject = this.web3.eth.contract(contract.abiDefinition);
2015-10-09 13:20:35 -04:00
2016-09-24 21:10:47 -04:00
var contractParams = (params || contract.args).slice();
2016-10-22 13:29:41 -06:00
this.web3.eth.getAccounts(function(err, accounts) {
2016-10-02 16:57:13 -04:00
//console.log("using address" + this.web3.eth.accounts[0]);
// TODO: probably needs to be defaultAccount
// TODO: it wouldn't necessary be the first address
// use defined blockchain address or first address
contractParams.push({
//from: this.web3.eth.coinbase,
from: accounts[0],
data: contract.code,
gas: contract.gas,
gasPrice: contract.gasPrice
});
2016-08-14 08:04:34 -04:00
self.logger.info("deploying " + contract.className + " with " + contract.gas + " gas");
2016-10-02 16:57:13 -04:00
contractParams.push(function(err, transaction) {
self.logger.contractsState(self.contractsManager.contractsState());
2016-10-02 16:57:13 -04:00
if (err) {
self.logger.error("error deploying contract: " + contract.className);
2016-10-22 11:48:47 -04:00
var errMsg = err.toString();
2016-10-22 11:57:36 -04:00
if (errMsg === 'Error: The contract code couldn\'t be stored, please check your gas amount.') {
errMsg = 'The contract code couldn\'t be stored, out of gas or constructor error';
2016-10-22 11:48:47 -04:00
}
self.logger.error(errMsg);
contract.error = errMsg;
2016-10-02 16:57:13 -04:00
callback(new Error(err));
} else if (transaction.address !== undefined) {
self.logger.info(contract.className + " deployed at " + transaction.address);
contract.deployedAddress = transaction.address;
contract.transactionHash = transaction.transactionHash;
2016-10-02 16:57:13 -04:00
callback(null, transaction.address);
}
});
2015-08-04 08:18:04 -04:00
2016-10-02 16:57:13 -04:00
contractObject["new"].apply(contractObject, contractParams);
});
};
2016-08-14 08:04:34 -04:00
Deploy.prototype.deployAll = function(done) {
var self = this;
2016-09-17 12:28:26 -04:00
this.logger.info("deploying contracts");
2016-08-14 08:04:34 -04:00
async.eachOfSeries(this.contractsManager.listContracts(),
function(contract, key, callback) {
2016-09-17 12:28:26 -04:00
self.logger.trace(arguments);
2016-09-24 21:10:47 -04:00
self.checkAndDeployContract(contract, null, callback);
2016-08-14 08:04:34 -04:00
},
function(err, results) {
2016-09-16 23:56:25 -04:00
self.logger.info("finished");
2016-09-17 12:28:26 -04:00
self.logger.trace(arguments);
2016-08-14 08:04:34 -04:00
done();
}
);
};
module.exports = Deploy;