2018-05-23 15:10:53 -04:00
|
|
|
const fs = require('../../core/fs');
|
2018-06-08 10:40:01 -04:00
|
|
|
const shellJs = require('shelljs');
|
2018-05-23 15:10:53 -04:00
|
|
|
const utils = require('../../utils/utils');
|
2018-07-27 17:33:50 -04:00
|
|
|
const ProcessLauncher = require('../../core/processes/processLauncher');
|
2018-05-23 15:10:53 -04:00
|
|
|
const constants = require('../../constants');
|
2018-07-15 15:35:01 -05:00
|
|
|
const {canonicalHost} = require('../../utils/host');
|
2018-10-19 00:13:03 +11:00
|
|
|
const cloneDeep = require('lodash.clonedeep');
|
2018-05-23 15:10:53 -04:00
|
|
|
|
2018-07-20 17:54:02 +03:00
|
|
|
let References = {
|
|
|
|
ipfs: 'https://ipfs.io/docs/install/',
|
|
|
|
swarm: 'http://swarm-guide.readthedocs.io/en/latest/installation.html'
|
2018-07-26 12:37:33 -04:00
|
|
|
};
|
2018-07-20 17:54:02 +03:00
|
|
|
|
2018-05-23 15:10:53 -04:00
|
|
|
class StorageProcessesLauncher {
|
|
|
|
constructor(options) {
|
|
|
|
this.logger = options.logger;
|
|
|
|
this.events = options.events;
|
2018-05-28 11:50:01 -04:00
|
|
|
this.storageConfig = options.storageConfig;
|
2018-05-24 12:47:10 -04:00
|
|
|
this.webServerConfig = options.webServerConfig;
|
2018-05-31 20:18:25 +10:00
|
|
|
this.blockchainConfig = options.blockchainConfig;
|
2018-08-01 11:14:02 -04:00
|
|
|
this.embark = options.embark;
|
2018-05-23 15:10:53 -04:00
|
|
|
this.processes = {};
|
2018-10-19 00:13:03 +11:00
|
|
|
this.corsParts = options.corsParts || [];
|
2018-05-30 10:52:15 -04:00
|
|
|
|
2018-05-31 20:18:25 +10:00
|
|
|
this.cors = this.buildCors();
|
|
|
|
|
2018-05-30 10:52:15 -04:00
|
|
|
this.events.on('exit', () => {
|
|
|
|
Object.keys(this.processes).forEach(processName => {
|
|
|
|
this.processes[processName].send('exit');
|
|
|
|
});
|
|
|
|
});
|
2018-05-23 15:10:53 -04:00
|
|
|
}
|
|
|
|
|
2018-09-24 21:07:02 -05:00
|
|
|
buildCors()
|
2018-05-31 20:18:25 +10:00
|
|
|
{
|
2018-10-19 00:13:03 +11:00
|
|
|
let corsParts = cloneDeep(this.corsParts);
|
2018-05-31 20:18:25 +10:00
|
|
|
// add our webserver CORS
|
|
|
|
if(this.webServerConfig.enabled){
|
|
|
|
if (this.webServerConfig && this.webServerConfig.host) {
|
2018-06-01 13:12:17 +10:00
|
|
|
corsParts.push(utils.buildUrlFromConfig(this.webServerConfig));
|
2018-05-31 20:18:25 +10:00
|
|
|
}
|
|
|
|
else corsParts.push('http://localhost:8000');
|
|
|
|
}
|
|
|
|
|
2018-09-24 20:50:27 -05:00
|
|
|
// add all dapp connection storage
|
2018-05-31 20:18:25 +10:00
|
|
|
if(this.storageConfig.enabled) {
|
|
|
|
this.storageConfig.dappConnection.forEach(dappConn => {
|
|
|
|
if(dappConn.getUrl || dappConn.host){
|
2018-09-24 20:50:27 -05:00
|
|
|
|
2018-05-31 20:18:25 +10:00
|
|
|
// if getUrl is specified in the config, that needs to be included in cors
|
|
|
|
// instead of the concatenated protocol://host:port
|
|
|
|
if(dappConn.getUrl) {
|
|
|
|
// remove /ipfs or /bzz: from getUrl if it's there
|
|
|
|
let getUrlParts = dappConn.getUrl.split('/');
|
|
|
|
getUrlParts = getUrlParts.slice(0, 3);
|
2018-07-15 15:35:01 -05:00
|
|
|
let host = canonicalHost(getUrlParts[2].split(':')[0]);
|
|
|
|
let port = getUrlParts[2].split(':')[1];
|
|
|
|
getUrlParts[2] = port ? [host, port].join(':') : host;
|
2018-05-31 20:18:25 +10:00
|
|
|
corsParts.push(getUrlParts.join('/'));
|
|
|
|
}
|
2018-06-01 13:12:17 +10:00
|
|
|
// in case getUrl wasn't specified, use a built url
|
2018-05-31 20:18:25 +10:00
|
|
|
else{
|
2018-06-01 13:12:17 +10:00
|
|
|
corsParts.push(utils.buildUrlFromConfig(dappConn));
|
2018-05-31 20:18:25 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if(this.blockchainConfig.enabled) {
|
|
|
|
// add our rpc endpoints to CORS
|
|
|
|
if(this.blockchainConfig.rpcHost && this.blockchainConfig.rpcPort){
|
2018-07-15 15:35:01 -05:00
|
|
|
corsParts.push(`http://${canonicalHost(this.blockchainConfig.rpcHost)}:${this.blockchainConfig.rpcPort}`);
|
2018-05-31 20:18:25 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
// add our ws endpoints to CORS
|
|
|
|
if(this.blockchainConfig.wsRPC && this.blockchainConfig.wsHost && this.blockchainConfig.wsPort){
|
2018-07-15 15:35:01 -05:00
|
|
|
corsParts.push(`ws://${canonicalHost(this.blockchainConfig.wsHost)}:${this.blockchainConfig.wsPort}`);
|
2018-05-31 20:18:25 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return corsParts;
|
|
|
|
}
|
|
|
|
|
2018-05-23 15:10:53 -04:00
|
|
|
processExited(storageName, code) {
|
2018-06-08 10:40:01 -04:00
|
|
|
this.logger.error(__(`Storage process for {{storageName}} ended before the end of this process. Code: {{code}}`, {storageName, code}));
|
2018-05-23 15:10:53 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
launchProcess(storageName, callback) {
|
|
|
|
const self = this;
|
2018-10-11 12:25:09 +02:00
|
|
|
callback = callback || function () {};
|
|
|
|
|
2018-05-23 15:10:53 -04:00
|
|
|
if (self.processes[storageName]) {
|
|
|
|
return callback(__('Storage process already started'));
|
|
|
|
}
|
2018-07-20 18:55:17 +03:00
|
|
|
const filePath = utils.joinPath(__dirname, `../${storageName}/process.js`);
|
2018-05-23 15:10:53 -04:00
|
|
|
fs.access(filePath, (err) => {
|
|
|
|
if (err) {
|
2018-06-08 10:40:01 -04:00
|
|
|
return callback(__('No process file for this storage type (%s) exists. Please start the process locally.', storageName));
|
2018-05-23 15:10:53 -04:00
|
|
|
}
|
2018-06-08 10:40:01 -04:00
|
|
|
|
2018-07-20 17:54:02 +03:00
|
|
|
let cmd = (storageName === 'swarm' ? (self.storageConfig.swarmPath || 'swarm') : 'ipfs');
|
|
|
|
|
|
|
|
const program = shellJs.which(cmd);
|
2018-06-08 10:40:01 -04:00
|
|
|
if (!program) {
|
|
|
|
self.logger.warn(__('{{storageName}} is not installed or your configuration is not right', {storageName}).yellow);
|
2018-07-20 17:54:02 +03:00
|
|
|
self.logger.info(__('You can install and get more information here: ').yellow + References[storageName].underline);
|
2018-06-08 10:40:01 -04:00
|
|
|
return callback(__('%s not installed', storageName));
|
|
|
|
}
|
|
|
|
|
|
|
|
self.logger.info(__(`Starting %s process`, storageName).cyan);
|
2018-05-23 15:10:53 -04:00
|
|
|
self.processes[storageName] = new ProcessLauncher({
|
|
|
|
modulePath: filePath,
|
2018-08-01 11:14:02 -04:00
|
|
|
name: storageName,
|
2018-05-23 15:10:53 -04:00
|
|
|
logger: self.logger,
|
|
|
|
events: self.events,
|
2018-08-01 11:14:02 -04:00
|
|
|
embark: self.embark,
|
2018-05-31 20:18:25 +10:00
|
|
|
silent: self.logger.logLevel !== 'trace',
|
2018-05-23 15:10:53 -04:00
|
|
|
exitCallback: self.processExited.bind(this, storageName)
|
|
|
|
});
|
2018-10-22 14:49:38 -04:00
|
|
|
this.events.once("blockchain:ready", () => {
|
2018-10-17 16:14:55 +11:00
|
|
|
this.events.request("blockchain:object", (blockchain) => {
|
|
|
|
blockchain.determineDefaultAccount((err, defaultAccount) => {
|
2018-10-19 17:02:27 +11:00
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2018-10-17 16:14:55 +11:00
|
|
|
self.processes[storageName].send({
|
|
|
|
action: constants.storage.init, options: {
|
|
|
|
storageConfig: self.storageConfig,
|
|
|
|
blockchainConfig: self.blockchainConfig,
|
|
|
|
cors: self.buildCors(),
|
|
|
|
defaultAccount: defaultAccount
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2018-05-24 12:47:10 -04:00
|
|
|
});
|
2018-10-17 16:14:55 +11:00
|
|
|
|
2018-05-23 15:10:53 -04:00
|
|
|
|
2018-05-28 11:50:01 -04:00
|
|
|
self.processes[storageName].on('result', constants.storage.initiated, (msg) => {
|
|
|
|
if (msg.error) {
|
|
|
|
self.processes[storageName].disconnect();
|
|
|
|
delete self.processes[storageName];
|
|
|
|
return callback(msg.error);
|
|
|
|
}
|
2018-05-24 10:51:05 -04:00
|
|
|
self.logger.info(__(`${storageName} process started`).cyan);
|
2018-05-28 11:50:01 -04:00
|
|
|
callback();
|
|
|
|
});
|
2018-09-16 00:18:53 +05:30
|
|
|
|
|
|
|
self.events.on('logs:swarm:enable', () => {
|
|
|
|
self.processes[storageName].silent = false;
|
|
|
|
});
|
|
|
|
|
|
|
|
self.events.on('logs:swarm:disable', () => {
|
|
|
|
self.processes[storageName].silent = true;
|
|
|
|
});
|
|
|
|
|
2018-05-23 15:10:53 -04:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = StorageProcessesLauncher;
|