2017-03-30 00:37:30 +09:00
let async = require ( 'async' ) ;
2018-04-25 10:34:17 -04:00
const constants = require ( './constants' ) ;
2018-04-30 15:56:43 +10:00
const _ = require ( 'underscore' ) ;
2017-03-13 00:21:19 +09:00
// require("./utils/debug_util.js")(__filename, async);
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 ) => {
if ( ! process . env . PWD ) {
process . env . PWD = process . cwd ( ) ;
}
realChdir ( ... args ) ;
} ;
2017-03-30 22:16:46 +09:00
let Engine = require ( './core/engine.js' ) ;
let version = require ( '../package.json' ) . version ;
2017-03-31 07:34:43 -04:00
class Embark {
2017-03-30 22:16:46 +09:00
constructor ( options ) {
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-05-14 14:32:19 -04:00
isDev ( env ) {
if ( this . config && this . config . blockchainConfig && this . config . blockchainConfig . isDev ) {
return true ;
} else if ( this . config && this . config . blockchainConfig && this . config . blockchainConfig . isDev === false ) {
return false ;
}
return ( env === 'development' ) ;
}
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-04-26 21:37:00 +10:00
let blockchainConfig = this . config . blockchainConfig ;
let storageConfig = this . config . storageConfig ;
let webServerConfig = this . config . webServerConfig ;
if ( blockchainConfig . rpcCorsDomain === 'auto' ) {
if ( webServerConfig ) blockchainConfig . rpcCorsDomain = ` http:// ${ webServerConfig . host } : ${ webServerConfig . port } ` ;
2018-04-30 15:56:43 +10:00
if ( storageConfig ) blockchainConfig . rpcCorsDomain += ` ${ blockchainConfig . rpcCorsDomain . length ? ',' : '' } ${ storageConfig . protocol } :// ${ storageConfig . host } : ${ storageConfig . port } ` ;
2018-04-26 21:37:00 +10:00
}
if ( blockchainConfig . wsOrigins === 'auto' ) {
if ( webServerConfig ) blockchainConfig . wsOrigins = ` http:// ${ webServerConfig . host } : ${ webServerConfig . port } ` ;
2018-04-30 15:56:43 +10:00
if ( storageConfig ) blockchainConfig . wsOrigins += ` ${ blockchainConfig . wsOrigins . length ? ',' : '' } ${ storageConfig . protocol } :// ${ storageConfig . host } : ${ storageConfig . port } ` ;
2018-04-26 21:37:00 +10:00
}
2018-05-14 14:32:19 -04:00
return require ( './cmds/blockchain/blockchain.js' ) ( blockchainConfig , client , env , this . isDev ( env ) ) . 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-04-27 09:16:29 -04:00
let simulator = new Simulator ( { blockchainConfig : this . config . blockchainConfig , logger : this . logger } ) ;
2017-03-30 22:16:46 +09:00
simulator . run ( options ) ;
}
2017-01-15 18:46:40 -05:00
2017-03-30 22:31:23 +09:00
generateTemplate ( templateName , destinationFolder , name ) {
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 ) ;
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-12-19 15:14:09 -05:00
let windowSize = require ( 'window-size' ) ;
2017-03-10 22:00:30 -05:00
2017-03-30 22:16:46 +09:00
let engine = new Engine ( {
env : options . env ,
2018-05-18 15:48:28 -04:00
client : options . client ,
locale : options . locale ,
2018-05-14 14:32:19 -04:00
isDev : this . isDev ( options . env ) ,
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 ,
context : self . context
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
engine . startMonitor ( ) ;
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-23 11:16:13 -04:00
engine . startService ( "codeGenerator" ) ;
2018-05-25 12:25:02 -05:00
engine . startService ( "namingSystem" ) ;
2018-05-18 21:05:39 -04:00
// TODO: this should be just 'storage' and the storage should figure out the module
engine . startService ( engine . config . storageConfig . provider ) ;
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 ( ) ;
engine . deployManager . deployContracts ( function ( ) {
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-23 10:06:12 -04:00
engine . deployManager . deployContracts ( function ( err ) {
engine . startService ( "fileWatcher" ) ;
if ( options . runWebserver ) {
engine . startService ( "webServer" , {
host : options . serverHost ,
port : options . serverPort
} ) ;
}
callback ( err ) ;
2017-03-30 20:12:39 +09:00
} ) ;
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' ) ;
let size = windowSize . get ( ) ;
if ( size . height < 40 || size . width < 118 ) {
2018-05-08 17:49:46 -04:00
engine . logger . warn ( _ _ ( "tip: you can resize the terminal or disable the dashboard with" ) + " embark run --nodashboard" . bold . underline ) ;
2017-12-19 15:14:09 -05:00
}
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-04-19 14:25:43 +10:00
let engine = new Engine ( {
env : options . env ,
2018-05-22 15:46:02 -04:00
client : options . client ,
locale : options . locale ,
2018-05-14 14:32:19 -04:00
isDev : this . isDev ( options . env ) ,
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" ) ;
engine . startService ( "pipeline" ) ;
2018-05-17 15:37:57 -05:00
engine . startService ( "deployment" , { onlyCompile : options . onlyCompile } ) ;
2018-05-23 11:16:13 -04:00
engine . startService ( "codeGenerator" ) ;
2018-05-18 21:05:39 -04:00
// TODO: this should be just 'storage' and the storage should figure out the modules to load
2017-12-30 15:52:51 -05:00
engine . startService ( "ipfs" ) ;
2018-05-18 21:05:39 -04:00
engine . startService ( "swarm" ) ;
2017-03-30 22:16:46 +09:00
callback ( ) ;
} ,
function deploy ( callback ) {
2018-05-23 10:06:12 -04:00
engine . deployManager . deployContracts ( function ( err ) {
callback ( err ) ;
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 . debug ( err . stack ) ;
} else {
2018-05-08 17:49:46 -04:00
engine . logger . info ( _ _ ( "finished building" ) . underline ) ;
2017-03-30 22:16:46 +09:00
}
// needed due to child processes
2018-04-20 17:39:45 +10:00
process . exit ( ) ;
2017-03-30 22:16:46 +09:00
} ) ;
}
2017-03-30 22:31:23 +09:00
initTests ( options ) {
2018-04-25 10:34:17 -04:00
this . context = options . context || [ constants . contexts . test ] ;
2018-01-19 21:13:20 -05:00
let Test = require ( './tests/test.js' ) ;
2018-04-25 10:34:17 -04:00
options . context = this . context ;
2017-03-30 22:16:46 +09:00
return new Test ( options ) ;
}
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 ;
let engine = new Engine ( {
env : options . env ,
2018-05-14 14:32:19 -04:00
isDev : this . isDev ( options . env ) ,
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 ( ) ;
async . parallel ( [
2018-05-08 17:49:46 -04:00
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
}
engine . startMonitor ( ) ;
engine . startService ( "libraryManager" ) ;
engine . startService ( "pipeline" ) ;
engine . startService ( "deployment" , { onlyCompile : true } ) ;
2018-05-23 11:16:13 -04:00
engine . startService ( "codeGenerator" ) ;
2018-03-22 15:09:01 -04:00
engine . deployManager . deployContracts ( function ( err ) {
callback ( err ) ;
} ) ;
}
2018-05-04 16:17:12 -04:00
] , ( err , _result ) => {
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-05-08 17:49:46 -04:00
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-04-20 17:39:45 +10:00
let engine = new Engine ( {
env : options . env ,
2018-05-22 15:46:58 -04:00
client : options . client ,
locale : options . locale ,
2018-05-14 14:32:19 -04:00
isDev : this . isDev ( options . env ) ,
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 ,
plugins : options . plugins
} ) ;
engine . init ( ) ;
2018-05-17 17:38:17 +10:00
let platform = engine . config . storageConfig . provider ;
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-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-23 11:16:13 -04:00
engine . startService ( "codeGenerator" ) ;
2018-05-18 21:05:39 -04:00
// TODO: this should be just 'storage' and the storage should figure out the modules to load
engine . startService ( platform . toLowerCase ( ) ) ;
2018-04-30 15:56:43 +10:00
engine . startMonitor ( ) ;
2018-04-24 10:27:11 +10:00
callback ( ) ;
} ,
2018-04-30 15:56:43 +10:00
function checkStorageService ( callback ) {
let checkFn ;
_ . find ( engine . servicesMonitor . checkList , ( value , key ) => {
if ( key . toLowerCase ( ) === platform . toLowerCase ( ) ) {
checkFn = value ;
return true ;
}
} ) ;
2018-05-01 09:38:13 -04:00
if ( ! checkFn || typeof checkFn . fn !== 'function' ) {
return callback ( ) ;
2018-04-30 15:56:43 +10:00
}
2018-05-01 09:38:13 -04:00
checkFn . fn ( function ( serviceCheckResult ) {
if ( ! serviceCheckResult . status || serviceCheckResult . status === 'off' ) {
2018-05-17 17:38:17 +10:00
return callback ( { message : _ _ ( 'Cannot upload: {{platform}} node is not running on {{protocol}}://{{host}}:{{port}}.' , { platform : platform , protocol : engine . config . storageConfig . protocol , host : engine . config . storageConfig . host , port : engine . config . storageConfig . port } ) } ) ;
2018-05-01 09:38:13 -04:00
}
callback ( ) ;
} ) ;
2018-04-30 15:56:43 +10:00
} ,
2018-04-15 18:41:50 +10: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-08 17:49:46 -04:00
engine . logger . info ( _ _ ( 'try "{{ipfs}}" or "{{swarm}}"' , { ipfs : 'embark upload ipfs' , swarm : 'embark upload swarm' } ) . green ) ;
2018-05-01 09:38:13 -04:00
return callback ( { message : 'unknown 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-15 18:41:50 +10:00
// 2. upload to storage (outputDone event triggered after webpack finished)
2018-04-24 10:27:11 +10:00
engine . events . on ( 'outputDone' , function ( ) {
cmdPlugin . uploadCmds [ 0 ] . cb ( )
2018-04-15 18:41:50 +10:00
. then ( ( success ) => {
callback ( null , success ) ;
} )
. catch ( callback ) ;
} ) ;
2018-05-22 15:46:58 -04:00
2018-05-23 10:06:12 -04:00
// 1. build the contracts and dapp webpack
engine . deployManager . deployContracts ( function ( err ) {
engine . logger . info ( _ _ ( "finished deploying" ) . underline ) ;
if ( err ) {
callback ( err ) ;
}
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
}
2017-07-02 11:32:16 -04:00
runTests ( file ) {
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' ) ;
2017-07-02 11:32:16 -04:00
RunTests . run ( file ) ;
2017-07-01 23:11:42 -04:00
}
2017-03-30 22:16:46 +09:00
}
2017-03-11 21:49:12 -05:00
2017-06-26 09:01:54 -04:00
// temporary until next refactor
Embark . initTests = function ( options ) {
2018-01-19 21:13:20 -05:00
let Test = require ( './tests/test.js' ) ;
2018-04-25 10:34:17 -04:00
options . context = [ constants . contexts . test ] ;
2017-06-26 09:01:54 -04:00
return new Test ( options ) ;
2017-06-26 09:02:58 -04:00
} ;
2017-06-26 09:01:54 -04:00
2016-08-17 20:29:41 -04:00
module . exports = Embark ;