refactor contract build method

This commit is contained in:
Iuri Matias 2016-10-29 12:02:07 -04:00
parent 215a823c35
commit 9a09e206ae
2 changed files with 142 additions and 120 deletions

View File

@ -1,5 +1,6 @@
var Compiler = require('./compiler.js'); var Compiler = require('./compiler.js');
var toposort = require('toposort'); var toposort = require('toposort');
var async = require('async');
// TODO: create a contract object // TODO: create a contract object
@ -17,60 +18,73 @@ ContractsManager.prototype.compileContracts = function() {
return compiler.compile_solidity(this.contractFiles); return compiler.compile_solidity(this.contractFiles);
}; };
ContractsManager.prototype.build = function() { ContractsManager.prototype.build = function(done) {
this.compiledContracts = this.compileContracts(); var self = this;
var className; async.waterfall([
var contract; function compileContracts(callback) {
var compiler = new Compiler();
// go through config file first try {
for(className in this.contractsConfig.contracts) { self.compiledContracts = compiler.compile_solidity(self.contractFiles);
contract = this.contractsConfig.contracts[className]; callback();
} catch(err) {
callback(new Error(err.message));
}
},
function prepareContractsFromConfig(callback) {
var className, contract;
for(className in self.contractsConfig.contracts) {
contract = self.contractsConfig.contracts[className];
contract.className = className; contract.className = className;
contract.args = contract.args || []; contract.args = contract.args || [];
this.contracts[className] = contract; self.contracts[className] = contract;
} }
callback();
},
function prepareContractsFromCompilation(callback) {
var className, compiledContract, contractConfig, contract;
var maxGas, adjustedGas;
for(className in self.compiledContracts) {
compiledContract = self.compiledContracts[className];
contractConfig = self.contractsConfig.contracts[className];
// compile contracts contract = self.contracts[className] || {className: className, args: []};
for(className in this.compiledContracts) {
var compiledContract = this.compiledContracts[className];
var contractConfig = this.contractsConfig.contracts[className];
contract = this.contracts[className] || {className: className, args: []};
contract.code = compiledContract.code; contract.code = compiledContract.code;
contract.runtimeBytecode = compiledContract.runtimeBytecode; contract.runtimeBytecode = compiledContract.runtimeBytecode;
contract.gasEstimates = compiledContract.gasEstimates; contract.gasEstimates = compiledContract.gasEstimates;
contract.functionHashes = compiledContract.functionHashes; contract.functionHashes = compiledContract.functionHashes;
contract.abiDefinition = compiledContract.abiDefinition; contract.abiDefinition = compiledContract.abiDefinition;
contract.gas = (contractConfig && contractConfig.gas) || this.contractsConfig.gas; contract.gas = (contractConfig && contractConfig.gas) || self.contractsConfig.gas;
if (contract.deploy === undefined) { if (contract.deploy === undefined) {
contract.deploy = true; contract.deploy = true;
} }
if (contract.gas === 'auto') { if (contract.gas === 'auto') {
var maxGas;
if (contract.deploy) { if (contract.deploy) {
maxGas = Math.max(contract.gasEstimates.creation[0], contract.gasEstimates.creation[1], 500000); maxGas = Math.max(contract.gasEstimates.creation[0], contract.gasEstimates.creation[1], 500000);
} else { } else {
maxGas = 500000; maxGas = 500000;
} }
// TODO: put a check so it doesn't go over the block limit // TODO: put a check so it doesn't go over the block limit
var adjustedGas = Math.round(maxGas * 1.40); adjustedGas = Math.round(maxGas * 1.40);
contract.gas = adjustedGas; contract.gas = adjustedGas;
} }
contract.gasPrice = contract.gasPrice || this.contractsConfig.gasPrice; contract.gasPrice = contract.gasPrice || self.contractsConfig.gasPrice;
contract.type = 'file'; contract.type = 'file';
contract.className = className; contract.className = className;
this.contracts[className] = contract; self.contracts[className] = contract;
} }
callback();
},
function dealWithSpecialConfigs(callback) {
var className, contract, parentContractName, parentContract;
// deal with special configs for(className in self.contracts) {
for(className in this.contracts) { contract = self.contracts[className];
contract = this.contracts[className];
// if deploy intention is not specified default is true // if deploy intention is not specified default is true
if (contract.deploy === undefined) { if (contract.deploy === undefined) {
@ -78,16 +92,16 @@ ContractsManager.prototype.build = function() {
} }
if (contract.instanceOf !== undefined) { if (contract.instanceOf !== undefined) {
var parentContractName = contract.instanceOf; parentContractName = contract.instanceOf;
var parentContract = this.contracts[parentContractName]; parentContract = self.contracts[parentContractName];
if (parentContract === className) { if (parentContract === className) {
this.logger.error(className + ": instanceOf is set to itself"); self.logger.error(className + ": instanceOf is set to itself");
continue; continue;
} }
if (parentContract === undefined) { if (parentContract === undefined) {
this.logger.error(className + ": couldn't find instanceOf contract " + parentContractName); slef.logger.error(className + ": couldn't find instanceOf contract " + parentContractName);
continue; continue;
} }
@ -96,7 +110,7 @@ ContractsManager.prototype.build = function() {
} }
if (contract.code !== undefined) { if (contract.code !== undefined) {
this.logger.error(className + " has code associated to it but it's configured as an instanceOf " + parentContractName); self.logger.error(className + " has code associated to it but it's configured as an instanceOf " + parentContractName);
} }
contract.code = parentContract.code; contract.code = parentContract.code;
@ -109,22 +123,25 @@ ContractsManager.prototype.build = function() {
contract.gasPrice = contract.gasPrice || parentContract.gasPrice; contract.gasPrice = contract.gasPrice || parentContract.gasPrice;
} }
} }
callback();
// remove contracts that don't have code },
for(className in this.contracts) { function removeContractsWithNoCode(callback) {
contract = this.contracts[className]; var className, contract;
for(className in self.contracts) {
contract = self.contracts[className];
if (contract.code === undefined) { if (contract.code === undefined) {
this.logger.error(className + " has no code associated"); self.logger.error(className + " has no code associated");
delete this.contracts[className]; delete self.contracts[className];
} }
} }
self.logger.trace(self.contracts);
this.logger.trace(this.contracts); callback();
},
// determine dependencies function determineDependencies(callback) {
for(className in this.contracts) { var className, contract;
contract = this.contracts[className]; for(className in self.contracts) {
contract = self.contracts[className];
if (contract.args === []) continue; if (contract.args === []) continue;
@ -132,14 +149,24 @@ ContractsManager.prototype.build = function() {
for (var j = 0; j < ref.length; j++) { for (var j = 0; j < ref.length; j++) {
var arg = ref[j]; var arg = ref[j];
if (arg[0] === "$") { if (arg[0] === "$") {
if (this.contractDependencies[className] === void 0) { if (self.contractDependencies[className] === void 0) {
this.contractDependencies[className] = []; self.contractDependencies[className] = [];
} }
this.contractDependencies[className].push(arg.substr(1)); self.contractDependencies[className].push(arg.substr(1));
} }
} }
} }
callback();
}
], function(err, result) {
self.logger.trace("finished".underline);
if (err) {
//self.logger.debug(err.stack);
done(err);
} else {
done(null, self);
}
});
}; };
ContractsManager.prototype.getContract = function(className) { ContractsManager.prototype.getContract = function(className) {

View File

@ -143,12 +143,7 @@ var Embark = {
contractsConfig: self.config.contractsConfig, contractsConfig: self.config.contractsConfig,
logger: Embark.logger logger: Embark.logger
}); });
try { contractsManager.build(callback);
contractsManager.build();
callback(null, contractsManager);
} catch(err) {
callback(new Error(err.message));
}
}, },
function deployContracts(contractsManager, callback) { function deployContracts(contractsManager, callback) {