const UploadIPFS = require('./upload.js'); const utils = require('../../utils/utils.js'); const fs = require('../../core/fs.js'); const IpfsApi = require('ipfs-api'); // TODO: not great, breaks module isolation const StorageProcessesLauncher = require('../storage/storageProcessesLauncher'); class IPFS { constructor(embark, options) { const self = this; this.logger = embark.logger; this.events = embark.events; this.buildDir = options.buildDir; this.storageConfig = embark.config.storageConfig; this.namesystemConfig = embark.config.namesystemConfig; this.embark = embark; this.webServerConfig = embark.config.webServerConfig; this.blockchainConfig = embark.config.blockchainConfig; this.events.request('processes:register', 'ipfs', (cb) => { self.startProcess(cb); }); if (this.isIpfsStorageEnabledInTheConfig() || this.isIpfsNameEnabledInTheConfig()) { this.downloadIpfsApi(); this.addDefaultToEmbarkJS(); } if (this.isIpfsStorageEnabledInTheConfig()) { this.setServiceCheck(); this.addStorageProviderToEmbarkJS(); this.addObjectToConsole(); this.registerUploadCommand(); this._checkService((err) => { if (!err) { return; } self.logger.info("IPFS node not found, attempting to start own node"); this.events.request("processes:launch", "ipfs", () => {}); }); } if (this.isIpfsNameEnabledInTheConfig()) { this.addNamesystemProviderToEmbarkJS(); this.setNamesystemProvider(); } } downloadIpfsApi() { const self = this; self.events.request("version:get:ipfs-api", function(ipfsApiVersion) { let currentIpfsApiVersion = require('../../../package.json').dependencies["ipfs-api"]; if (ipfsApiVersion !== currentIpfsApiVersion) { self.events.request("version:getPackageLocation", "ipfs-api", ipfsApiVersion, function(err, location) { self.embark.registerImportFile("ipfs-api", fs.dappPath(location)); }); } }); } setServiceCheck() { let self = this; self.events.on('check:backOnline:IPFS', function () { self.logger.info(__('IPFS node detected') + '..'); }); self.events.on('check:wentOffline:IPFS', function () { self.logger.info(__('IPFS node is offline') + '..'); }); self.events.request("services:register", 'IPFS', function (cb) { self._checkService((err, body) => { if (err) { self.logger.trace("IPFS unavailable"); return cb({name: "IPFS ", status: 'off'}); } if (body.Version) { self.logger.trace("IPFS available"); return cb({name: ("IPFS " + body.Version), status: 'on'}); } self.logger.trace("IPFS available"); return cb({name: "IPFS ", status: 'on'}); }); }); } _getNodeUrl() { if (this.storageConfig.upload.provider === 'ipfs') { return utils.buildUrlFromConfig(this.storageConfig.upload) + '/api/v0/version'; } for (let connection of this.storageConfig.dappConnection) { if (connection.provider === 'ipfs') { return utils.buildUrlFromConfig(connection) + '/api/v0/version'; } } } _checkService(cb) { let url = this._getNodeUrl(); utils.getJson(url, cb); } addDefaultToEmbarkJS() { let code = ""; code += "\n" + fs.readFileSync(utils.joinPath(__dirname, 'embarkjs' , 'default.js')).toString(); this.embark.addCodeToEmbarkJS(code); } addStorageProviderToEmbarkJS() { let code = ""; code += "\n" + fs.readFileSync(utils.joinPath(__dirname, 'embarkjs', 'storage.js')).toString(); code += "\nEmbarkJS.Storage.registerProvider('ipfs', __embarkIPFS);"; this.embark.addCodeToEmbarkJS(code); } addNamesystemProviderToEmbarkJS() { let code = ""; code += "\n" + fs.readFileSync(utils.joinPath(__dirname, 'embarkjs', 'name.js')).toString(); code += "\nEmbarkJS.Names.registerProvider('ipns', __embarkIPFS);"; this.embark.addCodeToEmbarkJS(code); } addNamesystemProviderToEmbarkJS() { let code = ""; code += "\n" + fs.readFileSync(utils.joinPath(__dirname, 'embarkjs', 'name.js')).toString(); code += "\nEmbarkJS.Names.registerProvider('ipns', __embarkIPFS);"; this.embark.addCodeToEmbarkJS(code); } addObjectToConsole() { let ipfs = IpfsApi(this.host, this.port); this.events.emit("runcode:register", "ipfs", ipfs); } startProcess(callback) { let self = this; const storageProcessesLauncher = new StorageProcessesLauncher({ logger: self.logger, events: self.events, storageConfig: self.storageConfig, webServerConfig: self.webServerConfig, blockchainConfig: self.blockchainConfig, embark: self.embark }); self.logger.trace(`Storage module: Launching ipfs process...`); return storageProcessesLauncher.launchProcess('ipfs', callback); } registerUploadCommand() { const self = this; this.embark.registerUploadCommand('ipfs', (cb) => { let upload_ipfs = new UploadIPFS({ buildDir: self.buildDir || 'dist/', storageConfig: self.storageConfig, configIpfsBin: self.storageConfig.ipfs_bin || "ipfs" }); upload_ipfs.deploy(cb); }); } isIpfsStorageEnabledInTheConfig() { let {enabled, available_providers, dappConnection} = this.storageConfig; return enabled && (available_providers.indexOf('ipfs') > 0 || dappConnection.find(c => c.provider === 'ipfs')); } isIpfsNameEnabledInTheConfig() { let {enabled, available_providers} = this.namesystemConfig; return enabled && available_providers.indexOf('ipns') > 0; } setNamesystemProvider() { let code = `\nEmbarkJS.Names.setProvider('ipns', ${JSON.stringify(this.storageConfig.dappConnection[0] || {})});`; let shouldInit = (namesystemConfig) => { return (namesystemConfig.provider === 'ipns' && namesystemConfig.enabled === true); }; this.embark.addProviderInit('names', code, shouldInit); } } module.exports = IPFS;