2017-03-30 00:37:30 +09:00
let async = require ( 'async' ) ;
2018-04-25 10:34:17 -04:00
const constants = require ( './constants' ) ;
2017-02-28 23:29:16 -05:00
2017-12-05 18:14:46 -05:00
require ( 'colors' ) ;
2016-08-14 08:04:34 -04:00
2018-04-09 16:14:07 -04:00
// Override process.chdir so that we have a partial-implementation PWD for Windows
const realChdir = process . chdir ;
process . chdir = ( ... args ) => {
2018-06-19 10:03:46 -04:00
if ( ! process . env . PWD ) {
process . env . PWD = process . cwd ( ) ;
}
realChdir ( ... args ) ;
2018-04-09 16:14:07 -04:00
} ;
2017-03-30 22:16:46 +09:00
let version = require ( '../package.json' ) . version ;
2017-03-31 07:34:43 -04:00
class Embark {
2017-03-30 22:16:46 +09:00
2018-06-19 10:03:46 -04:00
constructor ( options ) {
2017-03-30 22:16:46 +09:00
this . version = version ;
this . options = options || { } ;
}
2017-03-30 20:12:39 +09:00
2017-03-30 22:16:46 +09:00
initConfig ( env , options ) {
let Events = require ( './core/events.js' ) ;
let Logger = require ( './core/logger.js' ) ;
let Config = require ( './core/config.js' ) ;
2017-03-30 20:12:39 +09:00
2017-03-30 22:16:46 +09:00
this . events = new Events ( ) ;
2018-03-15 17:18:20 -04:00
this . logger = new Logger ( { logLevel : 'debug' , events : this . events } ) ;
2017-03-30 22:16:46 +09:00
2018-04-25 10:34:17 -04:00
this . config = new Config ( { env : env , logger : this . logger , events : this . events , context : this . context } ) ;
2017-03-30 20:12:39 +09:00
this . config . loadConfigFiles ( options ) ;
this . plugins = this . config . plugins ;
2017-03-30 22:16:46 +09:00
}
2017-03-30 20:12:39 +09:00
2018-06-27 14:29:31 -04:00
isDev ( ) {
return this . config && this . config . blockchainConfig && this . config . blockchainConfig . isDev ;
2018-05-14 14:32:19 -04:00
}
2017-03-30 22:16:46 +09:00
blockchain ( env , client ) {
2018-04-25 10:34:17 -04:00
this . context = [ constants . contexts . blockchain ] ;
2018-06-27 14:29:31 -04:00
return require ( './cmds/blockchain/blockchain.js' ) ( this . config . blockchainConfig , client , env , this . isDev ( ) ) . run ( ) ;
2017-03-30 22:16:46 +09:00
}
2017-02-24 22:49:34 -05:00
2017-03-30 22:16:46 +09:00
simulator ( options ) {
2018-04-25 10:34:17 -04:00
this . context = options . context || [ constants . contexts . simulator , constants . contexts . blockchain ] ;
2017-03-30 22:16:46 +09:00
let Simulator = require ( './cmds/simulator.js' ) ;
2018-06-07 12:50:22 -04:00
let simulator = new Simulator ( {
2018-06-19 10:03:46 -04:00
blockchainConfig : this . config . blockchainConfig ,
2018-06-07 15:13:35 -04:00
logger : this . logger
2018-06-07 12:50:22 -04:00
} ) ;
2017-03-30 22:16:46 +09:00
simulator . run ( options ) ;
}
2017-01-15 18:46:40 -05:00
2018-07-06 11:38:09 +03:00
generateTemplate ( templateName , destinationFolder , name , url ) {
2018-04-25 10:34:17 -04:00
this . context = [ constants . contexts . templateGeneration ] ;
2017-03-30 22:16:46 +09:00
let TemplateGenerator = require ( './cmds/template_generator.js' ) ;
let templateGenerator = new TemplateGenerator ( templateName ) ;
2018-07-06 11:38:09 +03:00
if ( url ) {
return templateGenerator . downloadAndGenerate ( url , destinationFolder , name ) ;
}
2017-03-30 22:16:46 +09:00
templateGenerator . generate ( destinationFolder , name ) ;
}
2017-02-24 22:49:34 -05:00
2017-03-30 22:16:46 +09:00
run ( options ) {
2017-03-31 07:34:43 -04:00
let self = this ;
2018-04-25 10:34:17 -04:00
self . context = options . context || [ constants . contexts . run , constants . contexts . build ] ;
2017-03-30 22:16:46 +09:00
let Dashboard = require ( './dashboard/dashboard.js' ) ;
2017-03-10 22:00:30 -05:00
2018-06-02 10:06:58 -04:00
const Engine = require ( './core/engine.js' ) ;
const engine = new Engine ( {
2017-03-30 22:16:46 +09:00
env : options . env ,
2018-05-18 15:48:28 -04:00
client : options . client ,
locale : options . locale ,
2018-06-27 14:29:31 -04:00
isDev : this . isDev ( ) ,
2017-03-31 07:34:43 -04:00
version : this . version ,
2018-03-10 13:45:56 -05:00
embarkConfig : options . embarkConfig || 'embark.json' ,
2018-04-19 14:25:43 +10:00
logFile : options . logFile ,
2018-04-25 10:34:17 -04:00
logLevel : options . logLevel ,
2018-06-15 09:37:52 +10:00
context : self . context ,
useDashboard : options . useDashboard
2017-03-30 22:16:46 +09:00
} ) ;
engine . init ( ) ;
if ( ! options . useDashboard ) {
2018-04-24 14:42:56 -04:00
engine . logger . info ( '========================' . bold . green ) ;
2018-05-08 17:49:46 -04:00
engine . logger . info ( ( _ _ ( 'Welcome to Embark' ) + ' ' + this . version ) . yellow . bold ) ;
2018-04-24 14:42:56 -04:00
engine . logger . info ( '========================' . bold . green ) ;
2017-03-30 22:16:46 +09:00
}
async . parallel ( [
function startDashboard ( callback ) {
if ( ! options . useDashboard ) {
return callback ( ) ;
}
let dashboard = new Dashboard ( {
2017-12-17 18:34:41 -05:00
events : engine . events ,
2017-03-30 22:16:46 +09:00
logger : engine . logger ,
plugins : engine . plugins ,
2017-03-31 07:34:43 -04:00
version : self . version ,
2017-07-06 19:50:36 -04:00
env : engine . env ,
contractsConfig : engine . config . contractsConfig
2017-03-30 22:16:46 +09:00
} ) ;
dashboard . start ( function ( ) {
2018-05-08 17:49:46 -04:00
engine . logger . info ( _ _ ( 'dashboard start' ) ) ;
2017-02-25 15:47:35 -05:00
callback ( ) ;
2017-03-30 22:16:46 +09:00
} ) ;
} ,
function ( callback ) {
let pluginList = engine . plugins . listPlugins ( ) ;
if ( pluginList . length > 0 ) {
2018-05-08 17:49:46 -04:00
engine . logger . info ( _ _ ( "loaded plugins" ) + ": " + pluginList . join ( ", " ) ) ;
2017-03-30 20:12:39 +09:00
}
2017-03-30 22:16:46 +09:00
2018-06-01 12:27:12 -04:00
engine . startService ( "serviceMonitor" ) ;
2017-12-30 15:52:51 -05:00
engine . startService ( "libraryManager" ) ;
2018-05-23 11:16:56 -04:00
engine . startService ( "codeRunner" ) ;
2017-03-30 22:16:46 +09:00
engine . startService ( "web3" ) ;
engine . startService ( "pipeline" ) ;
engine . startService ( "deployment" ) ;
2018-05-31 20:18:25 +10:00
engine . startService ( "storage" ) ;
2018-05-23 11:16:13 -04:00
engine . startService ( "codeGenerator" ) ;
2018-05-25 12:25:02 -05:00
engine . startService ( "namingSystem" ) ;
2018-06-08 07:07:27 -04:00
2017-03-30 22:16:46 +09:00
engine . events . on ( 'check:backOnline:Ethereum' , function ( ) {
2018-05-08 17:49:46 -04:00
engine . logger . info ( _ _ ( 'Ethereum node detected' ) + '..' ) ;
2017-03-30 22:16:46 +09:00
engine . config . reloadConfig ( ) ;
2018-06-19 10:03:46 -04:00
engine . events . request ( 'deploy:contracts' , function ( err ) {
2018-06-08 07:07:27 -04:00
if ( err ) {
return ;
}
2018-05-08 17:49:46 -04:00
engine . logger . info ( _ _ ( 'Deployment Done' ) ) ;
2017-03-30 22:16:46 +09:00
} ) ;
2016-11-28 17:14:39 -03:00
} ) ;
2017-03-30 22:16:46 +09:00
2018-04-17 14:11:22 +10:00
engine . events . on ( 'outputDone' , function ( ) {
2018-05-08 17:49:46 -04:00
engine . logger . info ( ( _ _ ( "Looking for documentation? You can find it at" ) + " " ) . cyan + "http://embark.status.im/docs/" . green . underline + "." . cyan ) ;
engine . logger . info ( _ _ ( "Ready" ) . underline ) ;
engine . events . emit ( "status" , _ _ ( "Ready" ) . green ) ;
2018-04-17 14:11:22 +10:00
} ) ;
2018-05-29 17:23:29 -04:00
2018-05-30 12:00:44 -04:00
if ( options . runWebserver ) {
engine . startService ( "webServer" , {
host : options . serverHost ,
port : options . serverPort
} ) ;
}
engine . startService ( "fileWatcher" ) ;
callback ( ) ;
2017-03-30 22:16:46 +09:00
}
2017-12-05 18:14:46 -05:00
] , function ( err , _result ) {
2017-03-30 22:16:46 +09:00
if ( err ) {
engine . logger . error ( err . message ) ;
engine . logger . info ( err . stack ) ;
2016-10-30 22:04:25 -04:00
} else {
2017-12-19 15:14:09 -05:00
engine . events . emit ( 'firstDeploymentDone' ) ;
2016-10-30 22:04:25 -04:00
}
2017-03-30 22:16:46 +09:00
} ) ;
}
2018-04-20 17:39:45 +10:00
build ( options ) {
2018-04-25 10:34:17 -04:00
this . context = options . context || [ constants . contexts . build ] ;
2018-04-26 19:05:56 +10:00
2018-06-02 10:06:58 -04:00
const Engine = require ( './core/engine.js' ) ;
const engine = new Engine ( {
2018-04-19 14:25:43 +10:00
env : options . env ,
2018-05-22 15:46:02 -04:00
client : options . client ,
locale : options . locale ,
2018-06-27 14:29:31 -04:00
isDev : this . isDev ( ) ,
2018-04-19 14:25:43 +10:00
version : this . version ,
embarkConfig : 'embark.json' ,
interceptLogs : false ,
logFile : options . logFile ,
logLevel : options . logLevel ,
events : options . events ,
logger : options . logger ,
config : options . config ,
2018-04-25 10:34:17 -04:00
plugins : options . plugins ,
context : this . context
2018-04-19 14:25:43 +10:00
} ) ;
engine . init ( ) ;
2016-10-02 17:26:57 -04:00
2017-03-30 22:16:46 +09:00
async . waterfall ( [
function startServices ( callback ) {
let pluginList = engine . plugins . listPlugins ( ) ;
if ( pluginList . length > 0 ) {
2018-05-08 17:49:46 -04:00
engine . logger . info ( _ _ ( "loaded plugins" ) + ": " + pluginList . join ( ", " ) ) ;
2017-03-30 22:16:46 +09:00
}
2017-12-30 15:52:51 -05:00
engine . startService ( "libraryManager" ) ;
2018-05-23 11:16:56 -04:00
engine . startService ( "codeRunner" ) ;
2017-03-30 22:16:46 +09:00
engine . startService ( "web3" ) ;
2018-06-22 11:12:34 -04:00
if ( ! options . onlyCompile ) {
2018-06-22 15:26:43 +10:00
engine . startService ( "pipeline" ) ;
}
2018-05-17 15:37:57 -05:00
engine . startService ( "deployment" , { onlyCompile : options . onlyCompile } ) ;
2018-05-25 17:13:57 +10:00
engine . startService ( "storage" ) ;
2018-05-23 11:16:13 -04:00
engine . startService ( "codeGenerator" ) ;
2017-03-30 22:16:46 +09:00
callback ( ) ;
} ,
function deploy ( callback ) {
2018-06-19 10:03:46 -04:00
engine . events . request ( 'deploy:contracts' , function ( err ) {
2018-05-23 10:06:12 -04:00
callback ( err ) ;
2017-03-30 22:16:46 +09:00
} ) ;
2018-06-14 10:30:34 -04:00
} ,
function waitForWriteFinish ( callback ) {
2018-06-22 11:12:34 -04:00
if ( options . onlyCompile ) {
engine . logger . info ( "Finished compiling" . underline ) ;
return callback ( null , true ) ;
}
2018-06-14 10:30:34 -04:00
engine . logger . info ( "Finished deploying" . underline ) ;
2018-06-22 11:12:34 -04:00
if ( ! engine . config . assetFiles || ! Object . keys ( engine . config . assetFiles ) . length ) {
2018-06-22 15:26:43 +10:00
return callback ( ) ;
}
2018-06-22 11:12:34 -04:00
engine . events . on ( 'outputDone' , ( err ) => {
engine . logger . info ( _ _ ( "finished building" ) . underline ) ;
callback ( err , true ) ;
} ) ;
2017-03-30 22:16:46 +09:00
}
2018-06-22 11:12:34 -04:00
] , function ( err , canExit ) {
2017-03-30 22:16:46 +09:00
if ( err ) {
engine . logger . error ( err . message ) ;
engine . logger . debug ( err . stack ) ;
}
2018-06-22 11:12:34 -04:00
if ( canExit || ! engine . config . contractsConfig . afterDeploy || ! engine . config . contractsConfig . afterDeploy . length ) {
process . exit ( ) ;
}
engine . logger . info ( _ _ ( 'Waiting for after deploy to finish...' ) ) ;
engine . logger . info ( _ _ ( 'You can exit with CTRL+C when after deploy completes' ) ) ;
2017-03-30 22:16:46 +09:00
} ) ;
}
2018-03-22 15:09:01 -04:00
graph ( options ) {
2018-04-25 10:34:17 -04:00
this . context = options . context || [ constants . contexts . graph ] ;
2018-03-22 15:09:01 -04:00
options . onlyCompile = true ;
2018-06-19 10:03:46 -04:00
2018-06-02 10:06:58 -04:00
const Engine = require ( './core/engine.js' ) ;
const engine = new Engine ( {
2018-03-22 15:09:01 -04:00
env : options . env ,
2018-06-27 14:29:31 -04:00
isDev : this . isDev ( ) ,
2018-03-22 15:09:01 -04:00
version : this . version ,
embarkConfig : options . embarkConfig || 'embark.json' ,
2018-04-25 10:34:17 -04:00
logFile : options . logFile ,
context : this . context
2018-03-22 15:09:01 -04:00
} ) ;
engine . init ( ) ;
2018-06-19 10:03:46 -04:00
async . waterfall ( [
2018-03-22 15:09:01 -04:00
function ( callback ) {
let pluginList = engine . plugins . listPlugins ( ) ;
if ( pluginList . length > 0 ) {
2018-05-08 17:49:46 -04:00
engine . logger . info ( _ _ ( "loaded plugins" ) + ": " + pluginList . join ( ", " ) ) ;
2018-03-22 15:09:01 -04:00
}
2018-06-01 12:27:12 -04:00
engine . startService ( "serviceMonitor" ) ;
2018-03-22 15:09:01 -04:00
engine . startService ( "libraryManager" ) ;
engine . startService ( "pipeline" ) ;
engine . startService ( "deployment" , { onlyCompile : true } ) ;
2018-06-19 10:03:46 -04:00
engine . startService ( "web3" ) ;
2018-05-23 11:16:13 -04:00
engine . startService ( "codeGenerator" ) ;
2018-03-22 15:09:01 -04:00
2018-06-19 10:03:46 -04:00
engine . events . request ( 'deploy:contracts' , callback ) ;
2018-03-22 15:09:01 -04:00
}
2018-06-19 10:03:46 -04:00
] , ( err ) => {
2018-03-22 15:09:01 -04:00
if ( err ) {
engine . logger . error ( err . message ) ;
engine . logger . info ( err . stack ) ;
} else {
const GraphGenerator = require ( './cmds/graph.js' ) ;
2018-03-22 16:18:13 -04:00
let graphGen = new GraphGenerator ( engine ) ;
2018-05-04 16:17:12 -04:00
graphGen . generate ( options ) ;
2018-03-22 15:09:01 -04:00
2018-05-08 17:49:46 -04:00
engine . logger . info ( _ _ ( "Done. %s generated" , "./diagram.svg" ) . underline ) ;
2018-03-22 15:09:01 -04:00
}
2018-05-23 11:13:51 -04:00
process . exit ( ) ;
2018-03-22 15:09:01 -04:00
} ) ;
2018-03-22 10:43:29 -04:00
}
2018-01-11 09:22:58 -05:00
reset ( ) {
2018-04-25 10:34:17 -04:00
this . context = [ constants . contexts . reset ] ;
2018-01-11 09:22:58 -05:00
let resetCmd = require ( './cmds/reset.js' ) ;
resetCmd ( ) ;
}
2018-05-17 17:38:17 +10:00
upload ( options ) {
2018-04-26 19:05:56 +10:00
this . context = options . context || [ constants . contexts . upload , constants . contexts . build ] ;
2018-04-19 14:25:43 +10:00
2018-06-02 10:06:58 -04:00
const Engine = require ( './core/engine.js' ) ;
const engine = new Engine ( {
2018-04-20 17:39:45 +10:00
env : options . env ,
2018-05-22 15:46:58 -04:00
client : options . client ,
locale : options . locale ,
2018-06-27 14:29:31 -04:00
isDev : this . isDev ( ) ,
2018-04-20 17:39:45 +10:00
version : this . version ,
embarkConfig : 'embark.json' ,
interceptLogs : false ,
logFile : options . logFile ,
logLevel : options . logLevel ,
events : options . events ,
logger : options . logger ,
config : options . config ,
2018-06-01 13:12:17 +10:00
plugins : options . plugins ,
context : this . context
2018-04-20 17:39:45 +10:00
} ) ;
engine . init ( ) ;
2018-05-25 17:13:57 +10:00
let platform = engine . config . storageConfig . upload . provider ;
2018-05-17 17:38:17 +10:00
2017-12-26 19:55:42 -05:00
let cmdPlugin ;
2018-04-15 18:41:50 +10:00
async . waterfall ( [
2018-05-08 17:49:46 -04:00
2018-04-24 10:27:11 +10:00
function startServices ( callback ) {
2018-05-08 17:49:46 -04:00
2018-06-01 12:27:12 -04:00
engine . startService ( "serviceMonitor" ) ;
2018-04-24 10:27:11 +10:00
engine . startService ( "libraryManager" ) ;
2018-05-23 11:16:56 -04:00
engine . startService ( "codeRunner" ) ;
2018-04-24 10:27:11 +10:00
engine . startService ( "web3" ) ;
engine . startService ( "pipeline" ) ;
engine . startService ( "deployment" ) ;
2018-05-31 20:18:25 +10:00
engine . startService ( "storage" ) ;
2018-05-23 11:16:13 -04:00
engine . startService ( "codeGenerator" ) ;
2018-04-24 10:27:11 +10:00
callback ( ) ;
} ,
2018-06-19 10:03:46 -04:00
function setupStoragePlugin ( callback ) {
2018-04-24 10:27:11 +10:00
let pluginList = engine . plugins . listPlugins ( ) ;
if ( pluginList . length > 0 ) {
2018-05-08 17:49:46 -04:00
engine . logger . info ( _ _ ( "loaded plugins" ) + ": " + pluginList . join ( ", " ) ) ;
2018-04-24 10:27:11 +10:00
}
2018-05-08 17:49:46 -04:00
2018-04-15 18:41:50 +10:00
// check use has input existing storage plugin
2018-04-24 10:27:11 +10:00
let cmdPlugins = engine . plugins . getPluginsFor ( 'uploadCmds' ) ;
2018-05-08 17:49:46 -04:00
2018-04-15 18:41:50 +10:00
if ( cmdPlugins . length > 0 ) {
cmdPlugin = cmdPlugins . find ( ( pluginCmd ) => {
2018-04-26 14:00:41 -04:00
return pluginCmd . uploadCmds . some ( uploadCmd => {
return uploadCmd . cmd === platform ;
} ) ;
2018-04-12 20:46:04 +10:00
} ) ;
}
2018-04-15 18:41:50 +10:00
if ( ! cmdPlugin ) {
2018-05-25 17:13:57 +10:00
return callback ( { message : _ _ ( 'platform "{{platform}}" is specified as the upload provider, however no plugins have registered an upload command for "{{platform}}".' , { platform : platform } ) } ) ;
2018-04-12 20:46:04 +10:00
}
2018-05-01 09:38:13 -04:00
callback ( ) ;
2018-04-15 18:41:50 +10:00
} ,
2018-04-20 17:39:45 +10:00
function deploy ( callback ) {
2018-04-24 10:27:11 +10:00
engine . events . on ( 'outputDone' , function ( ) {
cmdPlugin . uploadCmds [ 0 ] . cb ( )
2018-06-19 10:03:46 -04:00
. then ( ( success ) => {
callback ( null , success ) ;
} )
. catch ( callback ) ;
2018-04-15 18:41:50 +10:00
} ) ;
2018-06-22 15:17:51 +10:00
engine . events . on ( 'check:backOnline:Ethereum' , function ( ) {
engine . logger . info ( _ _ ( 'Ethereum node detected' ) + '..' ) ;
engine . config . reloadConfig ( ) ;
engine . events . request ( 'deploy:contracts' , function ( err ) {
if ( err ) {
return ;
}
engine . logger . info ( _ _ ( 'Deployment Done' ) ) ;
} ) ;
2018-04-20 17:39:45 +10:00
} ) ;
2018-04-15 18:41:50 +10:00
}
] , function ( err , _result ) {
if ( err ) {
2018-04-24 10:27:11 +10:00
engine . logger . error ( err . message ) ;
engine . logger . debug ( err . stack ) ;
2018-04-15 18:41:50 +10:00
} else {
2018-05-08 17:49:46 -04:00
engine . logger . info ( ( _ _ ( "finished building DApp and deploying to" ) + " " + platform ) . underline ) ;
2018-04-15 18:41:50 +10:00
}
2018-04-12 20:46:04 +10:00
2018-04-15 18:41:50 +10:00
// needed due to child processes
process . exit ( ) ;
} ) ;
2017-03-30 00:37:30 +09:00
}
2018-06-13 09:44:19 -04:00
runTests ( options ) {
2018-04-25 10:34:17 -04:00
this . context = [ constants . contexts . test ] ;
2018-01-19 21:13:20 -05:00
let RunTests = require ( './tests/run_tests.js' ) ;
2018-06-13 09:44:19 -04:00
RunTests . run ( options ) ;
2017-07-01 23:11:42 -04:00
}
2017-03-30 22:16:46 +09:00
}
2017-03-11 21:49:12 -05:00
2016-08-17 20:29:41 -04:00
module . exports = Embark ;