embark-area-51/lib/deploy.js

220 lines
7.1 KiB
JavaScript
Raw Normal View History

var web3 = require('web3');
var fs = require('fs');
var grunt = require('grunt');
var readYaml = require('read-yaml');
2015-06-28 13:51:58 +00:00
var Config = require('./config/config.js');
// Ugly, but sleep lib has issues on osx
sleep = function sleep(ms) {
var start = new Date().getTime();
while (new Date().getTime() < start + ms);
}
Deploy = function(env, contractFiles, blockchainConfig, contractsConfig, chainManager) {
2015-07-03 08:59:33 +00:00
//this.blockchainConfig = (new Config.Blockchain()).loadConfigFile('config/blockchain.yml').config(env);
this.blockchainConfig = blockchainConfig;
this.chainManager = chainManager;
this.chainManager.init(env, this.blockchainConfig);
2015-07-03 08:59:33 +00:00
//this.contractsManager = (new Config.Contracts(contractFiles, blockchainConfig)).loadConfigFile('config/contracts.yml');
this.contractsManager = contractsConfig;
this.contractsConfig = this.contractsManager.config(env);
this.deployedContracts = {};
try {
2015-07-03 08:59:33 +00:00
web3.setProvider(new web3.providers.HttpProvider("http://" + this.blockchainConfig.rpcHost + ":" + this.blockchainConfig.rpcPort));
primaryAddress = web3.eth.coinbase;
web3.eth.defaultAccount = primaryAddress;
2015-07-04 02:41:39 +00:00
} catch (e) {
throw new Error("==== can't connect to " + this.blockchainConfig.rpcHost + ":" + this.blockchainConfig.rpcPort + " check if an ethereum node is running");
}
2015-08-01 15:19:21 +00:00
console.log("primary account address is : " + primaryAddress);
2015-07-04 02:41:39 +00:00
};
2015-10-08 19:30:47 +00:00
Deploy.prototype.deploy_contract = function(contractObject, contractParams, cb) {
console.log("called deploy_contract");
var callback = function(e, contract) {
console.log("got receipt");
console.log(arguments);
if(!e && contract.address !== undefined) {
console.log("Contract mined! Address: " + contract.address);
cb(contract.address);
}
2015-10-08 19:30:47 +00:00
else {
console.log("error deploying");
exit();
}
};
contractParams.push(callback);
contractObject["new"].apply(contractObject, contractParams);
//var transactionHash = contractObject["new"].apply(contractObject, contractParams).transactionHash;
//var receipt = null;
//var time = 0;
//while ((receipt = web3.eth.getTransactionReceipt(transactionHash)) === null || receipt.contractAddress === null) {
// sleep(1000);
// time += 1;
// if (time >= this.blockchainConfig.deployTimeout) {
// return false;
// }
//}
//return receipt;
}
2015-10-08 19:30:47 +00:00
Deploy.prototype.deploy_contracts = function(env, cb) {
2015-07-03 08:59:33 +00:00
this.contractsManager.compileContracts(env);
2015-10-08 19:30:47 +00:00
var all_contracts = this.contractsManager.all_contracts;
2015-07-03 08:59:33 +00:00
this.contractDB = this.contractsManager.contractDB;
this.deployedContracts = {};
2015-10-08 19:30:47 +00:00
this.deploy_contract_list(all_contracts.length, env, all_contracts, cb);
}
Deploy.prototype.deploy_contract_list = function(index, env, all_contracts, cb) {
if(index === 0) {
cb();
}
else {
var _this = this;
this.deploy_contract_list(index - 1, env, all_contracts, function() {
var className = all_contracts[index - 1];
_this.deploy_a_contract(env, className, cb);
});
}
}
Deploy.prototype.deploy_a_contract = function(env, className, cb) {
var contractDependencies = this.contractsManager.contractDependencies;
var contract = this.contractDB[className];
2015-07-03 08:59:33 +00:00
2015-09-01 00:53:17 +00:00
if (contract.deploy === false) {
console.log("skipping " + className);
2015-10-08 19:30:47 +00:00
cb();
return;
2015-09-01 00:53:17 +00:00
}
var realArgs = [];
for (var l = 0; l < contract.args.length; l++) {
arg = contract.args[l];
if (arg[0] === "$") {
realArgs.push(this.deployedContracts[arg.substr(1)]);
} else {
realArgs.push(arg);
}
}
2015-08-04 12:18:04 +00:00
2015-07-06 12:19:25 +00:00
if (contract.address !== undefined) {
this.deployedContracts[className] = contract.address;
2015-07-09 11:31:37 +00:00
//console.log("contract " + className + " at " + contractAddress);
console.log("contract " + className + " at " + contract.address);
2015-10-08 19:30:47 +00:00
cb();
}
2015-07-06 12:19:25 +00:00
else {
var chainContract = this.chainManager.getContract(className, contract.compiled.code, realArgs);
2015-08-04 12:18:04 +00:00
2015-09-01 01:14:12 +00:00
if (chainContract != undefined && web3.eth.getCode(chainContract.address) !== "0x") {
2015-08-04 12:18:04 +00:00
console.log("contract " + className + " is unchanged and already deployed at " + chainContract.address);
2015-08-05 14:11:32 +00:00
this.deployedContracts[className] = chainContract.address;
this.execute_cmds(contract.onDeploy);
2015-10-08 19:30:47 +00:00
cb();
2015-07-06 12:19:25 +00:00
}
2015-08-04 12:18:04 +00:00
else {
2015-07-03 08:59:33 +00:00
2015-08-04 12:18:04 +00:00
contractObject = web3.eth.contract(contract.compiled.info.abiDefinition);
2015-07-03 08:59:33 +00:00
contractParams = realArgs.slice();
2015-08-04 12:18:04 +00:00
contractParams.push({
from: primaryAddress,
data: contract.compiled.code,
gas: contract.gasLimit,
gasPrice: contract.gasPrice
});
2015-08-04 12:18:04 +00:00
console.log('trying to obtain ' + className + ' address...');
2015-10-08 19:30:47 +00:00
var _this = this;
this.deploy_contract(contractObject, contractParams, function(contractAddress) {
console.log("response!");
2015-08-04 12:18:04 +00:00
2015-10-08 19:30:47 +00:00
if (web3.eth.getCode(contractAddress) === "0x") {
console.log("=========");
console.log("contract was deployed at " + contractAddress + " but doesn't seem to be working");
console.log("try adjusting your gas values");
console.log("=========");
}
else {
console.log("deployed " + className + " at " + contractAddress);
_this.chainManager.addContract(className, contract.compiled.code, realArgs, contractAddress);
_this.chainManager.save();
}
2015-08-04 12:18:04 +00:00
2015-10-08 19:30:47 +00:00
_this.deployedContracts[className] = contractAddress;
2015-10-08 19:30:47 +00:00
_this.execute_cmds(contract.onDeploy);
2015-10-08 19:30:47 +00:00
cb();
});
//while((receipt = this.deploy_contract(contractObject, contractParams)) === false) {
// console.log("timeout... failed to deploy contract.. retrying...");
//}
2015-08-04 12:18:04 +00:00
}
2015-07-09 11:31:37 +00:00
}
2015-07-04 02:41:39 +00:00
};
2015-07-03 08:59:33 +00:00
Deploy.prototype.execute_cmds = function(cmds) {
if (cmds == undefined || cmds.length === 0) return;
eval(this.generate_abi_file());
for (var i = 0; i < cmds.length; i++) {
var cmd = cmds[i];
for(className in this.deployedContracts) {
var contractAddress = this.deployedContracts[className];
var re = new RegExp("\\$" + className, 'g');
cmd = cmd.replace(re, '"' + contractAddress + '"');
}
console.log("executing: " + cmd);
eval(cmd);
}
}
2015-10-08 19:30:47 +00:00
Deploy.prototype.generate_abi_file = function(web3) {
2015-07-03 08:59:33 +00:00
var result;
2015-10-08 19:30:47 +00:00
result = "";
if (web3 === undefined) {
result = "web3.setProvider(new web3.providers.HttpProvider('http://" + this.blockchainConfig.rpcHost + ":" + this.blockchainConfig.rpcPort + "'));";
result += "web3.eth.defaultAccount = web3.eth.accounts[0];";
}
2015-07-03 08:59:33 +00:00
for(className in this.deployedContracts) {
var deployedContract = this.deployedContracts[className];
var contract = this.contractDB[className];
2015-07-04 21:02:01 +00:00
var abi = JSON.stringify(contract.compiled.info.abiDefinition);
var contractAddress = deployedContract;
console.log('address is ' + contractAddress);
2015-07-07 10:51:03 +00:00
result += className + "Abi = " + abi + ";";
result += className + "Contract = web3.eth.contract(" + className + "Abi);";
result += className + " = " + className + "Contract.at('" + contractAddress + "');";
}
2015-07-03 08:59:33 +00:00
return result;
2015-07-04 02:41:39 +00:00
};
2015-07-03 08:59:33 +00:00
Deploy.prototype.generate_and_write_abi_file = function(destFile) {
var result = this.generate_abi_file();
grunt.file.write(destFile, result);
2015-07-04 02:41:39 +00:00
};
2015-07-04 02:41:39 +00:00
module.exports = Deploy;