2017-03-30 02:50:05 +09:00
let async = require ( 'async' ) ;
2018-01-05 15:10:47 -05:00
//require("../utils/debug_util.js")(__filename, async);
2018-05-18 16:51:03 -04:00
let utils = require ( '../utils/utils.js' ) ;
2017-02-19 13:17:28 -05:00
2017-03-30 20:12:39 +09:00
class Deploy {
constructor ( options ) {
2018-05-18 18:31:47 -04:00
this . blockchain = options . blockchain ;
2017-03-30 20:12:39 +09:00
this . logger = options . logger ;
2018-02-27 15:40:05 -05:00
this . events = options . events ;
2018-01-17 23:04:19 +00:00
this . plugins = options . plugins ;
2018-01-13 11:38:10 -05:00
this . gasLimit = options . gasLimit ;
2018-01-05 15:10:47 -05:00
}
2017-03-30 20:12:39 +09:00
2018-05-21 12:29:18 -04:00
// TODO: determining the arguments could also be in a module since it's not
// part of ta 'normal' contract deployment
determineArguments ( suppliedArgs , contract , callback ) {
2018-05-21 13:11:13 -04:00
const self = this ;
2017-03-30 20:12:39 +09:00
2018-03-04 20:07:39 -05:00
let args = suppliedArgs ;
if ( ! Array . isArray ( args ) ) {
args = [ ] ;
let abi = contract . abiDefinition . find ( ( abi ) => abi . type === 'constructor' ) ;
for ( let input of abi . inputs ) {
let inputValue = suppliedArgs [ input . name ] ;
if ( ! inputValue ) {
2018-05-08 17:49:46 -04:00
this . logger . error ( _ _ ( "{{inputName}} has not been defined for {{className}} constructor" , { inputName : input . name , className : contract . className } ) ) ;
2018-03-04 20:07:39 -05:00
}
args . push ( inputValue || "" ) ;
}
}
2018-05-21 13:30:45 -04:00
let realArgs = [ ] ;
2018-05-21 12:29:18 -04:00
async . eachLimit ( args , 1 , ( arg , nextEachCb ) => {
2017-03-30 20:12:39 +09:00
if ( arg [ 0 ] === "$" ) {
2018-05-21 13:30:45 -04:00
let contractName = arg . substr ( 1 ) ;
2018-05-21 13:11:13 -04:00
self . events . request ( 'contracts:contract' , contractName , ( referedContract ) => {
realArgs . push ( referedContract . deployedAddress ) ;
nextEachCb ( ) ;
} ) ;
2018-03-04 18:46:12 -05:00
} else if ( Array . isArray ( arg ) ) {
let subRealArgs = [ ] ;
2018-05-21 13:30:45 -04:00
async . eachLimit ( arg , 1 , ( sub _arg , nextSubEachCb ) => {
2018-03-04 18:46:12 -05:00
if ( sub _arg [ 0 ] === "$" ) {
2018-05-21 13:30:45 -04:00
let contractName = sub _arg . substr ( 1 ) ;
self . events . request ( 'contracts:contract' , contractName , ( referedContract ) => {
subRealArgs . push ( referedContract . deployedAddress ) ;
nextSubEachCb ( ) ;
} ) ;
2018-03-04 18:46:12 -05:00
} else {
subRealArgs . push ( sub _arg ) ;
2018-05-21 13:30:45 -04:00
nextSubEachCb ( ) ;
2018-03-04 18:46:12 -05:00
}
2018-05-21 13:30:45 -04:00
} , ( ) => {
realArgs . push ( subRealArgs ) ;
nextEachCb ( ) ;
} ) ;
2017-03-30 20:12:39 +09:00
} else {
realArgs . push ( arg ) ;
2018-05-21 13:11:13 -04:00
nextEachCb ( ) ;
2017-03-30 20:12:39 +09:00
}
2018-05-21 12:29:18 -04:00
} , ( ) => {
callback ( realArgs ) ;
} ) ;
2016-09-27 21:13:54 -04:00
}
2017-03-30 20:12:39 +09:00
checkAndDeployContract ( contract , params , callback ) {
let self = this ;
contract . error = false ;
if ( contract . deploy === false ) {
2018-05-20 20:26:15 -04:00
self . events . emit ( "deploy:contract:undeployed" , contract ) ;
2017-03-30 20:12:39 +09:00
return callback ( ) ;
}
2016-09-24 21:10:47 -04:00
2018-05-20 20:59:55 -04:00
async . waterfall ( [
2018-05-21 12:29:18 -04:00
function _determineArguments ( next ) {
self . determineArguments ( params || contract . args , contract , ( realArgs ) => {
contract . realArgs = realArgs ;
next ( ) ;
} ) ;
2018-05-20 20:59:55 -04:00
} ,
function deployIt ( next ) {
if ( contract . address !== undefined ) {
try {
utils . toChecksumAddress ( contract . address ) ;
} catch ( e ) {
self . logger . error ( _ _ ( "error deploying %s" , contract . className ) ) ;
self . logger . error ( e . message ) ;
contract . error = e . message ;
self . events . emit ( "deploy:contract:error" , contract ) ;
return next ( e . message ) ;
}
contract . deployedAddress = contract . address ;
self . logger . info ( contract . className . bold . cyan + _ _ ( " already deployed at " ) . green + contract . address . bold . cyan ) ;
self . events . emit ( "deploy:contract:deployed" , contract ) ;
return next ( ) ;
}
2016-10-02 17:57:33 -04:00
2018-05-20 20:59:55 -04:00
// TODO: this should be a plugin API instead, if not existing, it should by default deploy the contract
self . events . request ( "deploy:contract:shouldDeploy" , contract , function ( trackedContract ) {
if ( ! trackedContract ) {
return self . contractToDeploy ( contract , params , next ) ;
}
2018-05-18 22:40:47 -04:00
2018-05-20 20:59:55 -04:00
self . blockchain . getCode ( trackedContract . address , function ( _getCodeErr , codeInChain ) {
if ( codeInChain !== "0x" ) {
self . contractAlreadyDeployed ( contract , trackedContract , next ) ;
} else {
self . contractToDeploy ( contract , params , next ) ;
}
} ) ;
} ) ;
}
] , callback ) ;
2018-01-05 15:10:47 -05:00
}
2017-02-18 16:53:49 -05:00
2018-01-05 15:10:47 -05:00
contractAlreadyDeployed ( contract , trackedContract , callback ) {
const self = this ;
2018-05-08 17:49:46 -04:00
self . logger . info ( contract . className . bold . cyan + _ _ ( " already deployed at " ) . green + trackedContract . address . bold . cyan ) ;
2018-01-05 15:10:47 -05:00
contract . deployedAddress = trackedContract . address ;
2018-05-20 19:59:35 -04:00
self . events . emit ( "deploy:contract:deployed" , contract ) ;
2018-01-05 15:10:47 -05:00
2018-05-23 11:16:13 -04:00
// TODO: can be moved into a afterDeploy event
// just need to figure out the gasLimit coupling issue
self . events . request ( 'code-generator:contract:vanilla' , contract , self . gasLimit , ( contractCode ) => {
2018-05-23 11:16:56 -04:00
self . events . request ( 'runcode:eval' , contractCode ) ;
2018-05-23 11:16:13 -04:00
return callback ( ) ;
} ) ;
2018-01-05 15:10:47 -05:00
}
contractToDeploy ( contract , params , callback ) {
const self = this ;
2018-05-21 12:29:18 -04:00
// TODO: refactor to async
self . determineArguments ( params || contract . args , contract , ( realArgs ) => {
contract . realArgs = realArgs ;
2017-12-27 11:40:21 -05:00
2018-05-21 12:29:18 -04:00
this . deployContract ( contract , contract . realArgs , function ( err , address ) {
if ( err ) {
self . events . emit ( "deploy:contract:error" , contract ) ;
return callback ( new Error ( err ) ) ;
}
contract . address = address ;
self . events . emit ( "deploy:contract:deployed" , contract ) ;
2017-12-27 11:40:21 -05:00
2018-05-23 11:16:13 -04:00
// TODO: can be moved into a afterDeploy event
// just need to figure out the gasLimit coupling issue
self . events . request ( 'code-generator:contract:vanilla' , contract , self . gasLimit , ( contractCode ) => {
2018-05-23 11:16:56 -04:00
self . events . request ( 'runcode:eval' , contractCode ) ;
2015-06-27 22:20:07 -04:00
2018-05-23 11:16:13 -04:00
let onDeployPlugins = self . plugins . getPluginsProperty ( 'onDeployActions' , 'onDeployActions' ) ;
2018-05-21 12:29:18 -04:00
2018-05-23 11:16:13 -04:00
async . eachLimit ( onDeployPlugins , 1 , function ( plugin , nextEach ) {
plugin . call ( plugin , contract , nextEach ) ;
} , ( ) => {
callback ( ) ;
} ) ;
2018-05-21 12:29:18 -04:00
} ) ;
2018-05-23 11:16:13 -04:00
2018-05-20 19:29:26 -04:00
} ) ;
2018-01-05 15:10:47 -05:00
} ) ;
2017-03-30 20:12:39 +09:00
}
2016-08-14 08:04:34 -04:00
2017-03-30 20:12:39 +09:00
deployContract ( contract , params , callback ) {
let self = this ;
2018-01-18 14:41:33 -05:00
let accounts = [ ] ;
2017-03-30 20:12:39 +09:00
let contractParams = ( params || contract . args ) . slice ( ) ;
2018-01-18 14:41:33 -05:00
let contractCode = contract . code ;
2018-05-18 18:31:47 -04:00
let deploymentAccount = self . blockchain . defaultAccount ( ) ;
2018-01-18 14:41:33 -05:00
let deployObject ;
async . waterfall ( [
2018-05-20 19:29:26 -04:00
// TODO: can potentially go to a beforeDeploy plugin
2018-01-18 14:41:33 -05:00
function getAccounts ( next ) {
2018-05-18 18:31:47 -04:00
self . blockchain . getAccounts ( function ( err , _accounts ) {
2018-01-18 14:41:33 -05:00
if ( err ) {
2018-01-18 14:46:53 -05:00
return next ( new Error ( err ) ) ;
2018-01-18 14:41:33 -05:00
}
accounts = _accounts ;
2018-01-20 00:56:05 +00:00
// applying deployer account configuration, if any
if ( typeof contract . fromIndex == 'number' ) {
deploymentAccount = accounts [ contract . fromIndex ] ;
2018-01-19 20:38:28 -05:00
if ( deploymentAccount === undefined ) {
2018-05-08 17:49:46 -04:00
return next ( _ _ ( "error deploying" ) + " " + contract . className + ": " + _ _ ( "no account found at index" ) + " " + contract . fromIndex + _ _ ( " check the config" ) ) ;
2018-01-19 20:38:28 -05:00
}
2018-01-20 00:56:05 +00:00
}
if ( typeof contract . from == 'string' && typeof contract . fromIndex != 'undefined' ) {
2018-05-08 17:49:46 -04:00
self . logger . warn ( _ _ ( 'Both "from" and "fromIndex" are defined for contract' ) + ' "' + contract . className + '". ' + _ _ ( 'Using "from" as deployer account.' ) ) ;
2018-01-20 00:56:05 +00:00
}
if ( typeof contract . from == 'string' ) {
deploymentAccount = contract . from ;
}
2018-01-18 14:41:33 -05:00
deploymentAccount = deploymentAccount || accounts [ 0 ] ;
next ( ) ;
} ) ;
} ,
function doLinking ( next ) {
2018-05-20 20:46:05 -04:00
self . events . request ( 'contracts:list' , ( contracts ) => {
for ( let contractObj of contracts ) {
let filename = contractObj . filename ;
let deployedAddress = contractObj . deployedAddress ;
if ( deployedAddress ) {
deployedAddress = deployedAddress . substr ( 2 ) ;
}
let linkReference = '__' + filename + ":" + contractObj . className ;
if ( contractCode . indexOf ( linkReference ) < 0 ) {
continue ;
}
if ( linkReference . length > 40 ) {
return next ( new Error ( _ _ ( "{{linkReference}} is too long, try reducing the path of the contract ({{filename}}) and/or its name {{contractName}}" , { linkReference : linkReference , filename : filename , contractName : contractObj . className } ) ) ) ;
}
let toReplace = linkReference + "_" . repeat ( 40 - linkReference . length ) ;
if ( deployedAddress === undefined ) {
let libraryName = contractObj . className ;
return next ( new Error ( _ _ ( "{{contractName}} needs {{libraryName}} but an address was not found, did you deploy it or configured an address?" , { contractName : contract . className , libraryName : libraryName } ) ) ) ;
}
contractCode = contractCode . replace ( new RegExp ( toReplace , "g" ) , deployedAddress ) ;
2018-01-18 14:41:33 -05:00
}
2018-05-20 20:46:05 -04:00
// saving code changes back to contract object
contract . code = contractCode ;
next ( ) ;
} ) ;
2018-01-18 14:41:33 -05:00
} ,
function applyBeforeDeploy ( next ) {
let beforeDeployPlugins = self . plugins . getPluginsFor ( 'beforeDeploy' ) ;
//self.logger.info("applying beforeDeploy plugins...", beforeDeployPlugins.length);
async . eachSeries ( beforeDeployPlugins , ( plugin , eachPluginCb ) => {
2018-05-08 17:49:46 -04:00
self . logger . info ( _ _ ( "running beforeDeploy plugin %s ." , plugin . name ) ) ;
2018-01-18 14:41:33 -05:00
// calling each beforeDeploy handler declared by the plugin
async . eachSeries ( plugin . beforeDeploy , ( beforeDeployFn , eachCb ) => {
2018-04-25 15:01:01 -04:00
function beforeDeployCb ( resObj ) {
contract . code = resObj . contractCode ;
eachCb ( ) ;
}
2018-01-18 14:41:33 -05:00
beforeDeployFn ( {
embarkDeploy : self ,
pluginConfig : plugin . pluginConfig ,
deploymentAccount : deploymentAccount ,
contract : contract ,
2018-04-25 15:01:01 -04:00
callback : beforeDeployCb
} , beforeDeployCb ) ;
2018-01-17 23:34:38 +00:00
} , ( ) => {
2018-01-18 14:41:33 -05:00
//self.logger.info('All beforeDeploy handlers of the plugin has processed.');
eachPluginCb ( ) ;
2018-01-17 23:04:19 +00:00
} ) ;
2018-01-18 14:41:33 -05:00
} , ( ) => {
//self.logger.info('All beforeDeploy plugins has been processed.');
contractCode = contract . code ;
next ( ) ;
} ) ;
} ,
function createDeployObject ( next ) {
2018-05-18 19:00:36 -04:00
let contractObject = self . blockchain . ContractObject ( { abi : contract . abiDefinition } ) ;
2018-01-18 14:41:33 -05:00
try {
2018-04-13 15:48:19 -04:00
const dataCode = contractCode . startsWith ( '0x' ) ? contractCode : "0x" + contractCode ;
2018-05-18 20:26:21 -04:00
deployObject = self . blockchain . deployContractObject ( contractObject , { arguments : contractParams , data : dataCode } ) ;
2018-01-18 14:41:33 -05:00
} catch ( e ) {
2018-03-02 17:48:30 -05:00
if ( e . message . indexOf ( 'Invalid number of parameters for "undefined"' ) >= 0 ) {
2018-05-08 17:49:46 -04:00
return next ( new Error ( _ _ ( "attempted to deploy %s without specifying parameters" , contract . className ) ) ) ;
2018-01-18 14:41:33 -05:00
} else {
return next ( new Error ( e ) ) ;
2018-01-17 23:04:19 +00:00
}
2018-01-05 15:10:47 -05:00
}
2018-01-18 14:41:33 -05:00
next ( ) ;
} ,
2018-01-19 13:57:35 -05:00
function estimateCorrectGas ( next ) {
if ( contract . gas === 'auto' ) {
return deployObject . estimateGas ( ) . then ( ( gasValue ) => {
contract . gas = gasValue ;
next ( ) ;
} ) . catch ( next ) ;
}
next ( ) ;
} ,
2018-01-18 14:41:33 -05:00
function deployTheContract ( next ) {
2018-05-08 17:49:46 -04:00
self . logger . info ( _ _ ( "deploying" ) + " " + contract . className . bold . cyan + " " + _ _ ( "with" ) . green + " " + contract . gas + " " + _ _ ( "gas" ) . green ) ;
2018-01-18 14:41:33 -05:00
2018-05-18 20:26:21 -04:00
self . blockchain . deployContractFromObject ( deployObject , {
2018-01-18 14:41:33 -05:00
from : deploymentAccount ,
gas : contract . gas ,
gasPrice : contract . gasPrice
2018-05-18 20:26:21 -04:00
} , function ( error , receipt ) {
if ( error ) {
return next ( new Error ( "error deploying =" + contract . className + "= due to error: " + error . message ) ) ;
2018-01-18 14:41:33 -05:00
}
2018-05-18 20:26:21 -04:00
self . logger . info ( contract . className . bold . cyan + " " + _ _ ( "deployed at" ) . green + " " + receipt . contractAddress . bold . cyan ) ;
contract . deployedAddress = receipt . contractAddress ;
contract . transactionHash = receipt . transactionHash ;
return next ( null , receipt . contractAddress ) ;
2018-01-18 14:41:33 -05:00
} ) ;
}
] , callback ) ;
2017-03-30 20:12:39 +09:00
}
deployAll ( done ) {
let self = this ;
2018-05-08 17:49:46 -04:00
this . logger . info ( _ _ ( "deploying contracts" ) ) ;
2018-05-18 22:40:47 -04:00
this . events . emit ( "deploy:beforeAll" ) ;
2017-03-30 20:12:39 +09:00
2018-05-20 20:46:05 -04:00
self . events . request ( 'contracts:list' , ( contracts ) => {
async . eachOfSeries ( contracts ,
function ( contract , key , callback ) {
self . logger . trace ( arguments ) ;
self . checkAndDeployContract ( contract , null , callback ) ;
} ,
function ( err , _results ) {
if ( err ) {
self . logger . error ( _ _ ( "error deploying contracts" ) ) ;
self . logger . error ( err . message ) ;
self . logger . debug ( err . stack ) ;
}
if ( contracts . length === 0 ) {
self . logger . info ( _ _ ( "no contracts found" ) ) ;
return done ( ) ;
}
self . logger . info ( _ _ ( "finished deploying contracts" ) ) ;
self . logger . trace ( arguments ) ;
done ( err ) ;
2018-04-26 14:15:43 -04:00
}
2018-05-20 20:46:05 -04:00
) ;
} ) ;
2015-08-04 08:18:04 -04:00
2017-03-30 20:38:14 +09:00
}
2017-03-30 20:12:39 +09:00
}
2016-08-14 08:04:34 -04:00
module . exports = Deploy ;