2018-05-07 14:30:30 -04:00
|
|
|
const Web3 = require('web3');
|
|
|
|
const Events = require('./events.js');
|
|
|
|
const Logger = require('./logger.js');
|
|
|
|
const Config = require('./config.js');
|
|
|
|
const ContractsManager = require('../contracts/contracts.js');
|
|
|
|
const DeployManager = require('../contracts/deploy_manager.js');
|
|
|
|
const CodeGenerator = require('../contracts/code_generator.js');
|
|
|
|
const ServicesMonitor = require('./services_monitor.js');
|
|
|
|
const Watch = require('../pipeline/watch.js');
|
|
|
|
const LibraryManager = require('../versions/library_manager.js');
|
2018-05-10 13:28:12 -04:00
|
|
|
const Pipeline = require('../pipeline/pipeline.js');
|
|
|
|
const async = require('async');
|
2018-05-10 14:52:30 -04:00
|
|
|
const Provider = require('./provider');
|
2017-03-30 20:12:39 +09:00
|
|
|
|
|
|
|
class Engine {
|
|
|
|
constructor(options) {
|
|
|
|
this.env = options.env;
|
|
|
|
this.embarkConfig = options.embarkConfig;
|
|
|
|
this.interceptLogs = options.interceptLogs;
|
2017-03-31 07:34:43 -04:00
|
|
|
this.version = options.version;
|
2018-04-19 14:25:43 +10:00
|
|
|
this.logFile = options.logFile;
|
2018-04-17 16:17:59 +10:00
|
|
|
this.logLevel = options.logLevel;
|
2018-04-19 14:25:43 +10:00
|
|
|
this.events = options.events;
|
2018-04-25 10:34:17 -04:00
|
|
|
this.context = options.context;
|
2017-03-30 20:12:39 +09:00
|
|
|
}
|
2017-03-03 01:22:12 -05:00
|
|
|
|
2017-03-30 20:38:14 +09:00
|
|
|
init(_options) {
|
|
|
|
let self = this;
|
|
|
|
let options = _options || {};
|
2018-04-19 14:25:43 +10:00
|
|
|
this.events = options.events || this.events || new Events();
|
|
|
|
this.logger = options.logger || new Logger({logLevel: options.logLevel || this.logLevel || 'debug', events: this.events, logFile: this.logFile});
|
2018-04-25 10:34:17 -04:00
|
|
|
this.config = new Config({env: this.env, logger: this.logger, events: this.events, context: this.context});
|
2017-03-30 20:38:14 +09:00
|
|
|
this.config.loadConfigFiles({embarkConfig: this.embarkConfig, interceptLogs: this.interceptLogs});
|
|
|
|
this.plugins = this.config.plugins;
|
|
|
|
|
|
|
|
this.servicesMonitor = new ServicesMonitor({events: this.events, logger: this.logger});
|
|
|
|
this.servicesMonitor.addCheck('embarkVersion', function (cb) {
|
2017-04-01 23:22:43 -04:00
|
|
|
return cb({name: 'Embark ' + self.version, status: 'on'});
|
2017-03-30 20:38:14 +09:00
|
|
|
}, 0);
|
2018-04-30 15:56:43 +10:00
|
|
|
|
|
|
|
if (this.interceptLogs || this.interceptLogs === undefined) {
|
|
|
|
this.doInterceptLogs();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-07 14:30:30 -04:00
|
|
|
normalizeInput(input) {
|
|
|
|
let args = Object.values(input);
|
|
|
|
if (args.length === 0) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
if (args.length === 1) {
|
|
|
|
if (Array.isArray(args[0])) { return args[0].join(','); }
|
|
|
|
return args[0] || "";
|
|
|
|
}
|
|
|
|
return ('[' + args.map((x) => {
|
|
|
|
if (x === null) { return "null"; }
|
|
|
|
if (x === undefined) { return "undefined"; }
|
|
|
|
if (Array.isArray(x)) { return x.join(','); }
|
|
|
|
return x;
|
|
|
|
}).toString() + ']');
|
|
|
|
}
|
|
|
|
|
2018-04-30 15:56:43 +10:00
|
|
|
doInterceptLogs() {
|
|
|
|
var self = this;
|
|
|
|
let context = {};
|
|
|
|
context.console = console;
|
|
|
|
|
|
|
|
context.console.log = function() {
|
2018-05-07 14:30:30 -04:00
|
|
|
self.logger.info(self.normalizeInput(arguments));
|
2018-04-30 15:56:43 +10:00
|
|
|
};
|
|
|
|
context.console.warn = function() {
|
2018-05-07 14:30:30 -04:00
|
|
|
self.logger.warn(self.normalizeInput(arguments));
|
2018-04-30 15:56:43 +10:00
|
|
|
};
|
|
|
|
context.console.info = function() {
|
2018-05-07 14:30:30 -04:00
|
|
|
self.logger.info(self.normalizeInput(arguments));
|
2018-04-30 15:56:43 +10:00
|
|
|
};
|
|
|
|
context.console.debug = function() {
|
|
|
|
// TODO: ue JSON.stringify
|
2018-05-07 14:30:30 -04:00
|
|
|
self.logger.debug(self.normalizeInput(arguments));
|
2018-04-30 15:56:43 +10:00
|
|
|
};
|
|
|
|
context.console.trace = function() {
|
2018-05-07 14:30:30 -04:00
|
|
|
self.logger.trace(self.normalizeInput(arguments));
|
2018-04-30 15:56:43 +10:00
|
|
|
};
|
|
|
|
context.console.dir = function() {
|
2018-05-07 14:30:30 -04:00
|
|
|
self.logger.dir(self.normalizeInput(arguments));
|
2018-04-30 15:56:43 +10:00
|
|
|
};
|
2017-03-30 20:38:14 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
startMonitor() {
|
|
|
|
let self = this;
|
|
|
|
if (this.plugins) {
|
2017-12-18 09:37:16 -05:00
|
|
|
// --------
|
|
|
|
// TODO: this only works for services done on startup
|
|
|
|
// --------
|
2017-03-30 20:38:14 +09:00
|
|
|
let servicePlugins = this.plugins.getPluginsFor('serviceChecks');
|
|
|
|
servicePlugins.forEach(function (plugin) {
|
|
|
|
plugin.serviceChecks.forEach(function (pluginCheck) {
|
|
|
|
self.servicesMonitor.addCheck(pluginCheck.checkName, pluginCheck.checkFn, pluginCheck.time);
|
|
|
|
});
|
2017-03-16 07:31:52 -04:00
|
|
|
});
|
2017-03-30 20:38:14 +09:00
|
|
|
}
|
|
|
|
this.servicesMonitor.startMonitor();
|
|
|
|
}
|
|
|
|
|
2017-12-16 15:39:30 -05:00
|
|
|
registerModule(moduleName, options) {
|
|
|
|
this.plugins.loadInternalPlugin(moduleName, options);
|
|
|
|
}
|
|
|
|
|
2017-03-30 20:38:14 +09:00
|
|
|
startService(serviceName, _options) {
|
|
|
|
let options = _options || {};
|
|
|
|
|
|
|
|
let services = {
|
|
|
|
"pipeline": this.pipelineService,
|
2017-08-03 19:29:09 -04:00
|
|
|
"codeGenerator": this.codeGeneratorService,
|
2017-03-30 20:38:14 +09:00
|
|
|
"deployment": this.deploymentService,
|
|
|
|
"fileWatcher": this.fileWatchService,
|
|
|
|
"webServer": this.webServerService,
|
|
|
|
"ipfs": this.ipfsService,
|
2017-12-30 15:52:51 -05:00
|
|
|
"web3": this.web3Service,
|
2018-04-24 10:27:11 +10:00
|
|
|
"libraryManager": this.libraryManagerService,
|
|
|
|
"swarm": this.swarmService
|
2017-03-30 20:38:14 +09:00
|
|
|
};
|
|
|
|
|
|
|
|
let service = services[serviceName];
|
|
|
|
|
|
|
|
if (!service) {
|
|
|
|
throw new Error("unknown service: " + serviceName);
|
|
|
|
}
|
|
|
|
|
|
|
|
// need to be careful with circular references due to passing the web3 object
|
|
|
|
//this.logger.trace("calling: " + serviceName + "(" + JSON.stringify(options) + ")");
|
|
|
|
return service.apply(this, [options]);
|
|
|
|
}
|
|
|
|
|
2017-10-14 10:13:30 -04:00
|
|
|
pipelineService(_options) {
|
2018-05-10 10:10:09 -04:00
|
|
|
const self = this;
|
|
|
|
this.events.emit("status", "Building Assets");
|
|
|
|
const pipeline = new Pipeline({
|
|
|
|
buildDir: this.config.buildDir,
|
|
|
|
contractsFiles: this.config.contractsFiles,
|
|
|
|
assetFiles: this.config.assetFiles,
|
|
|
|
events: this.events,
|
|
|
|
logger: this.logger,
|
2018-05-10 11:14:25 -04:00
|
|
|
normalizeInput: this.normalizeInput,
|
2018-05-10 10:10:09 -04:00
|
|
|
plugins: this.plugins
|
|
|
|
});
|
2018-05-10 13:28:12 -04:00
|
|
|
|
|
|
|
const queue = async.queue((task, callback) => {
|
|
|
|
pipeline.build(task.abi, task.contractsJSON, task.path, callback);
|
|
|
|
}, 1);
|
|
|
|
|
2018-05-10 10:10:09 -04:00
|
|
|
this.events.on('code-generator-ready', function () {
|
|
|
|
self.events.request('code', function (abi, contractsJSON) {
|
|
|
|
self.currentAbi = abi;
|
|
|
|
self.contractsJSON = contractsJSON;
|
2018-05-10 13:28:12 -04:00
|
|
|
|
|
|
|
queue.push({abi, contractsJSON, path: null}, () => {
|
2018-04-30 09:29:31 -04:00
|
|
|
if (self.watch) {
|
|
|
|
self.watch.restart(); // Necessary because changing a file while it is writing can stop it from being watched
|
|
|
|
}
|
2018-05-10 10:10:09 -04:00
|
|
|
self.events.emit('outputDone');
|
2017-07-06 18:48:20 -04:00
|
|
|
});
|
2017-07-05 08:35:51 -04:00
|
|
|
});
|
2017-03-30 20:38:14 +09:00
|
|
|
});
|
2017-03-16 07:31:52 -04:00
|
|
|
}
|
2017-03-30 20:38:14 +09:00
|
|
|
|
2017-10-14 10:13:30 -04:00
|
|
|
codeGeneratorService(_options) {
|
2017-03-30 20:38:14 +09:00
|
|
|
let self = this;
|
2018-01-12 18:06:51 -05:00
|
|
|
|
2017-08-03 19:29:09 -04:00
|
|
|
let generateCode = function (contractsManager) {
|
|
|
|
let codeGenerator = new CodeGenerator({
|
2017-03-30 20:38:14 +09:00
|
|
|
blockchainConfig: self.config.blockchainConfig,
|
2017-07-06 18:48:20 -04:00
|
|
|
contractsConfig: self.config.contractsConfig,
|
2017-03-30 20:38:14 +09:00
|
|
|
contractsManager: contractsManager,
|
|
|
|
plugins: self.plugins,
|
|
|
|
storageConfig: self.config.storageConfig,
|
2017-07-06 18:48:20 -04:00
|
|
|
communicationConfig: self.config.communicationConfig,
|
|
|
|
events: self.events
|
2017-03-30 20:38:14 +09:00
|
|
|
});
|
2017-08-03 19:29:09 -04:00
|
|
|
codeGenerator.listenToCommands();
|
2018-01-10 10:43:25 -05:00
|
|
|
codeGenerator.buildEmbarkJS(function() {
|
|
|
|
self.events.emit('code-generator-ready');
|
|
|
|
});
|
2017-03-30 20:38:14 +09:00
|
|
|
};
|
2017-08-03 19:29:09 -04:00
|
|
|
this.events.on('contractsDeployed', generateCode);
|
|
|
|
this.events.on('blockchainDisabled', generateCode);
|
2018-02-21 18:43:34 -05:00
|
|
|
this.events.on('asset-changed', generateCode);
|
2017-03-03 01:22:12 -05:00
|
|
|
}
|
2017-03-03 21:48:32 -05:00
|
|
|
|
2017-03-30 20:38:14 +09:00
|
|
|
deploymentService(options) {
|
|
|
|
let self = this;
|
2017-12-16 15:39:30 -05:00
|
|
|
|
|
|
|
this.registerModule('solidity', {
|
|
|
|
contractDirectories: self.config.contractDirectories
|
|
|
|
});
|
2018-04-12 13:24:54 -04:00
|
|
|
this.registerModule('vyper', {
|
|
|
|
contractDirectories: self.config.contractDirectories
|
|
|
|
});
|
2018-05-08 10:30:46 -05:00
|
|
|
this.registerModule('profiler', {
|
|
|
|
events: this.events,
|
2018-05-08 10:53:34 -05:00
|
|
|
logger: this.logger
|
2018-05-08 10:30:46 -05:00
|
|
|
});
|
2017-12-16 15:39:30 -05:00
|
|
|
|
2018-02-21 18:43:34 -05:00
|
|
|
this.contractsManager = new ContractsManager({
|
|
|
|
contractFiles: this.config.contractsFiles,
|
|
|
|
contractsConfig: this.config.contractsConfig,
|
|
|
|
logger: this.logger,
|
|
|
|
plugins: this.plugins,
|
2018-04-30 15:56:43 +10:00
|
|
|
gasLimit: false,
|
|
|
|
events: this.events
|
2018-02-21 18:43:34 -05:00
|
|
|
});
|
|
|
|
|
2017-03-30 20:38:14 +09:00
|
|
|
this.deployManager = new DeployManager({
|
|
|
|
web3: options.web3 || self.web3,
|
|
|
|
trackContracts: options.trackContracts,
|
|
|
|
config: this.config,
|
|
|
|
logger: this.logger,
|
|
|
|
plugins: this.plugins,
|
2018-02-21 18:43:34 -05:00
|
|
|
events: this.events,
|
2018-03-22 15:09:01 -04:00
|
|
|
contractsManager: this.contractsManager,
|
|
|
|
onlyCompile: options.onlyCompile
|
2017-03-03 21:48:32 -05:00
|
|
|
});
|
2017-03-30 20:38:14 +09:00
|
|
|
|
2018-04-30 15:56:43 +10:00
|
|
|
this.events.on('file-event', function (fileType) {
|
|
|
|
// TODO: still need to redeploy contracts because the original contracts
|
|
|
|
// config is being corrupted
|
|
|
|
if (fileType === 'asset') {
|
|
|
|
self.events.emit('asset-changed', self.contractsManager);
|
|
|
|
}
|
2017-03-30 20:38:14 +09:00
|
|
|
// TODO: for now need to deploy on asset chanes as well
|
|
|
|
// because the contractsManager config is corrupted after a deploy
|
2018-02-21 18:43:34 -05:00
|
|
|
if (fileType === 'contract' || fileType === 'config') {
|
|
|
|
self.config.reloadConfig();
|
|
|
|
self.deployManager.deployContracts(function () {
|
|
|
|
});
|
|
|
|
}
|
2017-03-30 20:12:39 +09:00
|
|
|
});
|
|
|
|
}
|
2017-03-10 22:00:30 -05:00
|
|
|
|
2017-10-14 10:13:30 -04:00
|
|
|
fileWatchService(_options) {
|
2018-02-27 15:49:21 -05:00
|
|
|
this.events.emit("status", "Watching for changes");
|
2018-04-30 15:56:43 +10:00
|
|
|
this.watch = new Watch({logger: this.logger, events: this.events});
|
|
|
|
this.watch.start();
|
2017-03-30 20:38:14 +09:00
|
|
|
}
|
|
|
|
|
2017-12-19 09:50:29 -05:00
|
|
|
webServerService() {
|
2017-12-18 09:37:16 -05:00
|
|
|
this.registerModule('webserver', {
|
|
|
|
addCheck: this.servicesMonitor.addCheck.bind(this.servicesMonitor)
|
2017-03-30 20:38:14 +09:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-10-14 10:13:30 -04:00
|
|
|
ipfsService(_options) {
|
2017-12-26 20:32:51 -05:00
|
|
|
this.registerModule('ipfs', {
|
|
|
|
addCheck: this.servicesMonitor.addCheck.bind(this.servicesMonitor),
|
|
|
|
storageConfig: this.config.storageConfig,
|
|
|
|
host: _options.host,
|
|
|
|
port: _options.port
|
2017-03-10 22:00:30 -05:00
|
|
|
});
|
2017-03-11 10:29:45 -05:00
|
|
|
}
|
|
|
|
|
2018-04-24 10:27:11 +10:00
|
|
|
swarmService(_options) {
|
|
|
|
this.registerModule('swarm', {
|
|
|
|
addCheck: this.servicesMonitor.addCheck.bind(this.servicesMonitor),
|
|
|
|
storageConfig: this.config.storageConfig,
|
2018-04-30 15:56:43 +10:00
|
|
|
bzz: _options.bzz
|
2018-04-24 10:27:11 +10:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-03-30 20:38:14 +09:00
|
|
|
web3Service(options) {
|
|
|
|
let self = this;
|
|
|
|
this.web3 = options.web3;
|
|
|
|
if (this.web3 === undefined) {
|
|
|
|
this.web3 = new Web3();
|
2017-07-05 20:24:28 -04:00
|
|
|
if (this.config.contractsConfig.deployment.type === "rpc") {
|
2018-05-10 14:52:30 -04:00
|
|
|
const web3Endpoint = 'http://' + this.config.contractsConfig.deployment.host + ':' + this.config.contractsConfig.deployment.port;
|
2018-05-10 14:52:51 -04:00
|
|
|
const providerOptions = {
|
|
|
|
web3: this.web3,
|
|
|
|
accountsConfig: this.config.contractsConfig.deployment.accounts,
|
|
|
|
logger: this.logger,
|
|
|
|
web3Endpoint
|
|
|
|
};
|
|
|
|
this.web3.setProvider(new Provider(providerOptions));
|
2017-07-05 20:24:28 -04:00
|
|
|
} else {
|
|
|
|
throw new Error("contracts config error: unknown deployment type " + this.config.contractsConfig.deployment.type);
|
|
|
|
}
|
2017-03-11 10:29:45 -05:00
|
|
|
}
|
|
|
|
|
2017-03-30 20:38:14 +09:00
|
|
|
self.servicesMonitor.addCheck('Ethereum', function (cb) {
|
2018-01-05 15:30:52 -05:00
|
|
|
if (self.web3.currentProvider === undefined) {
|
2018-05-08 17:49:46 -04:00
|
|
|
return cb({name: __("No Blockchain node found"), status: 'off'});
|
2018-01-05 15:30:52 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
self.web3.eth.getAccounts(function(err, _accounts) {
|
|
|
|
if (err) {
|
2018-05-08 17:49:46 -04:00
|
|
|
return cb({name: __("No Blockchain node found"), status: 'off'});
|
2018-01-05 15:30:52 -05:00
|
|
|
}
|
2018-04-10 15:08:08 -04:00
|
|
|
|
|
|
|
// TODO: web3_clientVersion method is currently not implemented in web3.js 1.0
|
|
|
|
self.web3._requestManager.send({method: 'web3_clientVersion', params: []}, (err, version) => {
|
|
|
|
if (err) {
|
2018-05-08 17:49:46 -04:00
|
|
|
return cb({name: __("Ethereum node (version unknown)"), status: 'on'});
|
2018-04-10 15:08:08 -04:00
|
|
|
}
|
2018-04-23 12:14:48 -04:00
|
|
|
if (version.indexOf("/") < 0) {
|
|
|
|
return cb({name: version, status: 'on'});
|
|
|
|
}
|
2018-04-10 15:14:00 -04:00
|
|
|
let nodeName = version.split("/")[0];
|
|
|
|
let versionNumber = version.split("/")[1].split("-")[0];
|
|
|
|
let name = nodeName + " " + versionNumber + " (Ethereum)";
|
|
|
|
|
|
|
|
return cb({name: name, status: 'on'});
|
2017-03-30 20:38:14 +09:00
|
|
|
});
|
2018-01-05 15:30:52 -05:00
|
|
|
});
|
2017-03-11 10:29:45 -05:00
|
|
|
});
|
|
|
|
|
2018-01-05 15:30:52 -05:00
|
|
|
this.registerModule('whisper', {
|
|
|
|
addCheck: this.servicesMonitor.addCheck.bind(this.servicesMonitor),
|
|
|
|
communicationConfig: this.config.communicationConfig,
|
|
|
|
web3: this.web3
|
|
|
|
});
|
2017-03-30 20:38:14 +09:00
|
|
|
}
|
2017-12-30 15:52:51 -05:00
|
|
|
|
2017-12-30 17:07:13 -05:00
|
|
|
libraryManagerService(_options) {
|
|
|
|
this.libraryManager = new LibraryManager({
|
2017-12-30 15:52:51 -05:00
|
|
|
plugins: this.plugins,
|
|
|
|
config: this.config
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-03-30 20:38:14 +09:00
|
|
|
}
|
2017-03-11 10:29:45 -05:00
|
|
|
|
2017-03-03 01:22:12 -05:00
|
|
|
module.exports = Engine;
|