embark/lib/core/modules/coderunner/codeRunner.js

110 lines
2.8 KiB
JavaScript
Raw Normal View History

2018-08-27 16:12:28 +01:00
const RunCode = require('./runCode.js');
2018-09-17 09:57:51 +01:00
const Utils = require('../../../utils/utils');
2018-08-28 11:47:40 +01:00
2018-10-02 11:52:52 +03:00
const WEB3_INVALID_RESPONSE_ERROR = 'Invalid JSON RPC response';
2018-05-21 17:22:19 -04:00
class CodeRunner {
constructor(options) {
2018-08-28 14:43:52 +01:00
this.config = options.config;
2018-05-21 17:22:19 -04:00
this.plugins = options.plugins;
this.logger = options.logger;
this.events = options.events;
2018-08-08 13:42:45 +01:00
this.ipc = options.ipc;
this.commands = [];
2018-09-10 09:26:37 +01:00
this.runCode = new RunCode({logger: this.logger});
2018-08-29 10:49:53 +01:00
this.registerIpcEvents();
this.IpcClientListen();
this.registerEvents();
this.registerCommands();
}
2018-05-23 11:16:56 -04:00
2018-08-29 10:49:53 +01:00
registerIpcEvents() {
if (!this.ipc.isServer()) {
return;
2018-08-08 13:42:45 +01:00
}
2018-08-29 10:49:53 +01:00
this.ipc.on('runcode:getCommands', (_err, callback) => {
let result = {web3Config: this.runCode.getWeb3Config(), commands: this.commands};
callback(null, result);
});
}
2018-08-28 12:16:36 +01:00
2018-08-29 10:49:53 +01:00
IpcClientListen() {
if (!this.ipc.isClient() || !this.ipc.connected) {
return;
2018-08-08 13:42:45 +01:00
}
2018-08-29 10:49:53 +01:00
this.ipc.listenTo('runcode:newCommand', (command) => {
if (command.varName) {
this.events.emit("runcode:register", command.varName, command.code);
} else {
this.events.request("runcode:eval", command.code);
2018-08-08 13:42:45 +01:00
}
2018-08-24 14:05:19 +01:00
});
2018-08-29 10:49:53 +01:00
}
registerEvents() {
this.events.on("runcode:register", this.registerVar.bind(this));
}
2018-08-24 14:05:19 +01:00
2018-08-29 10:49:53 +01:00
registerCommands() {
2018-08-24 14:05:19 +01:00
this.events.setCommandHandler('runcode:getContext', (cb) => {
2018-08-29 10:49:53 +01:00
cb(this.runCode.context);
2018-05-23 11:16:56 -04:00
});
2018-08-29 10:49:53 +01:00
this.events.setCommandHandler('runcode:eval', this.evalCode.bind(this));
}
2018-05-21 17:22:19 -04:00
2018-08-29 11:03:06 +01:00
registerVar(varName, code, toRecord = true) {
if (this.ipc.isServer() && toRecord) {
2018-08-29 10:49:53 +01:00
this.commands.push({varName, code});
this.ipc.broadcast("runcode:newCommand", {varName, code});
}
this.runCode.registerVar(varName, code);
}
2018-09-17 09:57:51 +01:00
async evalCode(code, cb, forConsoleOnly = false) {
2018-08-30 10:41:13 +01:00
cb = cb || function() {};
2018-08-29 10:49:53 +01:00
const awaitIdx = code.indexOf('await');
2018-09-17 09:57:51 +01:00
let awaiting = false;
2018-08-29 10:49:53 +01:00
if (awaitIdx > -1) {
2018-09-17 09:57:51 +01:00
awaiting = true;
const instructions = Utils.compact(code.split(';'));
const last = instructions.pop();
if (!last.trim().startsWith('return')) {
instructions.push(`return ${last}`);
2018-08-29 10:49:53 +01:00
} else {
2018-09-17 09:57:51 +01:00
instructions.push(last);
2018-08-22 11:32:15 -04:00
}
2018-09-17 09:57:51 +01:00
code = `(async function() {${instructions.join(';')}})();`;
2018-08-29 10:49:53 +01:00
}
let result = this.runCode.doEval(code);
2018-08-24 14:05:19 +01:00
2018-08-29 10:49:53 +01:00
if (forConsoleOnly && this.ipc.isServer()) {
this.commands.push({code});
this.ipc.broadcast("runcode:newCommand", {code});
}
2018-08-24 14:05:19 +01:00
2018-09-17 09:57:51 +01:00
if (!awaiting) {
return cb(null, result);
2018-08-29 10:49:53 +01:00
}
2018-08-24 14:05:19 +01:00
2018-09-17 09:57:51 +01:00
try {
const value = await result;
cb(null, value);
} catch (error) {
// Improve error message when there's no connection to node
2018-10-02 11:52:52 +03:00
if (error.message.indexOf(WEB3_INVALID_RESPONSE_ERROR) !== -1) {
error.message += '. Are you connected to an Ethereum node?';
}
2018-09-17 09:57:51 +01:00
cb(error);
}
2018-05-21 17:22:19 -04:00
}
2018-08-23 11:36:08 -04:00
2018-05-21 17:22:19 -04:00
}
module.exports = CodeRunner;