2017-03-30 00:37:30 +09:00
let async = require ( 'async' ) ;
2018-01-05 15:10:47 -05:00
//require("../utils/debug_util.js")(__filename, async);
2017-03-30 00:37:30 +09:00
let Deploy = require ( './deploy.js' ) ;
2017-12-21 11:21:36 -05:00
let RunCode = require ( '../core/runCode.js' ) ;
2017-02-28 20:42:03 -05:00
2017-03-30 20:12:39 +09:00
class DeployManager {
constructor ( options ) {
this . config = options . config ;
this . logger = options . logger ;
this . blockchainConfig = this . config . blockchainConfig ;
2017-12-21 11:21:36 -05:00
2017-03-31 07:34:43 -04:00
this . events = options . events ;
2017-03-30 20:12:39 +09:00
this . plugins = options . plugins ;
this . web3 = options . web3 ;
this . chainConfig = ( options . trackContracts !== false ) ? this . config . chainTracker : false ;
2018-02-21 18:43:34 -05:00
this . contractsManager = options . contractsManager ;
2018-01-13 11:38:10 -05:00
this . gasLimit = false ;
this . fatalErrors = false ;
2018-03-11 08:28:03 -04:00
this . deployOnlyOnConfig = false ;
2018-03-22 15:09:01 -04:00
this . onlyCompile = options . onlyCompile !== undefined ? options . onlyCompile : false ;
2017-02-28 20:42:03 -05:00
}
2017-03-30 20:12:39 +09:00
deployContracts ( done ) {
let self = this ;
if ( self . blockchainConfig === { } || self . blockchainConfig . enabled === false ) {
2018-05-08 17:49:46 -04:00
self . logger . info ( _ _ ( "Blockchain component is disabled in the config" ) . underline ) ;
2017-03-31 07:34:43 -04:00
this . events . emit ( 'blockchainDisabled' , { } ) ;
2017-03-30 20:12:39 +09:00
return done ( ) ;
2017-03-11 10:29:45 -05:00
}
2017-03-30 20:12:39 +09:00
async . waterfall ( [
function buildContracts ( callback ) {
2018-03-11 08:28:03 -04:00
self . contractsManager . deployOnlyOnConfig = self . deployOnlyOnConfig ; // temporary, should refactor
2018-02-21 18:43:34 -05:00
self . contractsManager . build ( callback ) ;
2017-03-30 20:12:39 +09:00
} ,
2018-03-22 15:09:01 -04:00
function checkCompileOnly ( contractsManager , callback ) {
if ( self . onlyCompile ) {
self . events . emit ( 'contractsDeployed' , contractsManager ) ;
return done ( ) ;
}
2018-04-13 14:54:46 -04:00
return callback ( null , contractsManager ) ;
2018-03-22 15:09:01 -04:00
} ,
2017-03-30 20:12:39 +09:00
function checkWeb3IsConnected ( contractsManager , callback ) {
if ( ! self . web3 ) {
return callback ( Error ( "no web3 instance found" ) ) ;
}
2018-01-05 15:10:47 -05:00
if ( self . web3 . currentProvider === undefined ) {
2018-05-08 17:49:46 -04:00
self . logger . error ( _ _ ( "Couldn't connect to an Ethereum node are you sure it's on?" ) . red ) ;
self . logger . info ( _ _ ( "make sure you have an Ethereum node or simulator running. e.g '%s'" , 'embark blockchain' ) . magenta ) ;
2017-03-11 10:29:45 -05:00
return callback ( Error ( "error connecting to blockchain node" ) ) ;
}
2018-01-05 15:10:47 -05:00
self . web3 . eth . getAccounts ( function ( err , _accounts ) {
if ( err ) {
2018-05-08 17:49:46 -04:00
self . logger . error ( _ _ ( "Couldn't connect to an Ethereum node are you sure it's on?" ) . red ) ;
self . logger . info ( _ _ ( "make sure you have an Ethereum node or simulator running. e.g '%s'" , 'embark blockchain' ) . magenta ) ;
2018-01-05 15:10:47 -05:00
return callback ( Error ( "error connecting to blockchain node" ) ) ;
}
2017-03-30 20:12:39 +09:00
return callback ( null , contractsManager , self . web3 ) ;
2018-01-05 15:10:47 -05:00
} ) ;
2017-03-30 20:12:39 +09:00
} ,
function setDefaultAccount ( contractsManager , web3 , callback ) {
web3 . eth . getAccounts ( function ( err , accounts ) {
if ( err ) {
2017-12-20 14:54:47 -05:00
self . logger . error ( err ) ;
2017-03-30 20:12:39 +09:00
return callback ( new Error ( err ) ) ;
}
let accountConfig = self . config . blockchainConfig . account ;
let selectedAccount = accountConfig && accountConfig . address ;
web3 . eth . defaultAccount = ( selectedAccount || accounts [ 0 ] ) ;
callback ( null , contractsManager , web3 ) ;
} ) ;
} ,
function deployAllContracts ( contractsManager , web3 , callback ) {
let deploy = new Deploy ( {
web3 : web3 ,
contractsManager : contractsManager ,
logger : self . logger ,
2018-02-27 15:40:05 -05:00
events : self . events ,
2017-03-30 20:12:39 +09:00
chainConfig : self . chainConfig ,
2018-01-13 11:38:10 -05:00
env : self . config . env ,
2018-01-17 23:04:19 +00:00
plugins : self . plugins ,
2018-01-13 11:38:10 -05:00
gasLimit : self . gasLimit
2017-03-30 20:12:39 +09:00
} ) ;
2018-01-05 15:10:47 -05:00
deploy . initTracker ( function ( ) {
deploy . deployAll ( function ( err ) {
if ( ! err ) {
self . events . emit ( 'contractsDeployed' , contractsManager ) ;
}
2018-01-13 11:38:10 -05:00
if ( err && self . fatalErrors ) {
return callback ( err ) ;
}
2018-01-05 15:10:47 -05:00
callback ( null , contractsManager , web3 ) ;
} ) ;
2017-03-30 20:12:39 +09:00
} ) ;
2017-12-21 11:21:36 -05:00
} ,
function runAfterDeployCommands ( contractsManager , web3 , callback ) {
let afterDeployCmds = self . config . contractsConfig . afterDeploy || [ ] ;
let withErrors = false ;
let regex = /\$\w+/g ;
let onDeployCode = afterDeployCmds . map ( ( cmd ) => {
let realCmd = cmd . replace ( regex , ( match ) => {
let referedContractName = match . slice ( 1 ) ;
let referedContract = contractsManager . getContract ( referedContractName ) ;
if ( ! referedContract ) {
self . logger . error ( referedContractName + ' does not exist' ) ;
2018-05-08 17:49:46 -04:00
self . logger . error ( _ _ ( "error running afterDeploy: " ) + cmd ) ;
2017-12-21 11:21:36 -05:00
withErrors = true ;
return ;
}
if ( referedContract && referedContract . deploy === false ) {
self . logger . error ( referedContractName + " exists but has been set to not deploy" ) ;
2018-05-08 17:49:46 -04:00
self . logger . error ( _ _ ( "error running afterDeploy: " ) + cmd ) ;
2017-12-21 11:21:36 -05:00
withErrors = true ;
return ;
}
if ( referedContract && ! referedContract . deployedAddress ) {
self . logger . error ( "couldn't find a valid address for " + referedContractName + ". has it been deployed?" ) ;
2018-05-08 17:49:46 -04:00
self . logger . error ( _ _ ( "error running afterDeploy: " ) + cmd ) ;
2017-12-21 11:21:36 -05:00
withErrors = true ;
return ;
}
return referedContract . deployedAddress ;
} ) ;
return realCmd ;
} ) ;
if ( withErrors ) {
return callback ( new Error ( "error running afterDeploy" ) ) ;
}
2017-12-27 13:07:13 -05:00
// TODO: convert to for to avoid repeated callback
2017-12-28 07:30:41 -05:00
for ( let cmd of onDeployCode ) {
2018-05-08 17:49:46 -04:00
self . logger . info ( _ _ ( "executing" ) + ": " + cmd ) ;
2017-12-27 13:07:13 -05:00
try {
RunCode . doEval ( cmd , web3 ) ;
} catch ( e ) {
if ( e . message . indexOf ( "invalid opcode" ) >= 0 ) {
2018-05-08 17:49:46 -04:00
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' ) ) ;
2017-12-27 13:07:13 -05:00
}
return callback ( new Error ( e ) ) ;
}
2017-12-28 07:30:41 -05:00
}
2017-12-21 11:21:36 -05:00
callback ( null , contractsManager ) ;
2017-03-30 20:12:39 +09:00
}
] , function ( err , result ) {
2017-02-24 08:20:03 -05:00
if ( err ) {
2017-03-30 20:12:39 +09:00
done ( err , null ) ;
} else {
done ( null , result ) ;
2017-02-24 08:20:03 -05:00
}
} ) ;
}
2017-03-30 20:12:39 +09:00
}
2017-02-24 08:20:03 -05:00
module . exports = DeployManager ;