/* eslint-disable */ /* eslint global-require: 1, flowtype-errors/show-errors: 0 */ /** * This module executes inside of electron's main process. You can start * electron renderer process from here and communicate with the other processes * through IPC. * * When running `npm run build` or `npm run build-main`, this file is compiled to * `./app/main.prod.js` using webpack. This gives us some performance wins. * * @flow */ import { app, BrowserWindow, ipcMain } from 'electron' import fs from 'fs' import path from 'path' import { spawn, exec } from 'child_process' import { lookup } from 'ps-node' import { platform } from 'os' import MenuBuilder from './menu' import lnd from './lnd' const plat = platform() let mainWindow = null let neutrino = null let syncing = false if (process.env.NODE_ENV === 'production') { const sourceMapSupport = require('source-map-support'); sourceMapSupport.install(); } if (process.env.NODE_ENV === 'development' || process.env.DEBUG_PROD === 'true') { require('electron-debug')(); const path = require('path'); const p = path.join(__dirname, '..', 'app', 'node_modules'); require('module').globalPaths.push(p); } const installExtensions = async () => { const installer = require('electron-devtools-installer'); const forceDownload = !!process.env.UPGRADE_EXTENSIONS; const extensions = [ 'REACT_DEVELOPER_TOOLS', 'REDUX_DEVTOOLS' ]; return Promise .all(extensions.map(name => installer.default(installer[name], forceDownload))) .catch(console.log); }; /** * Add event listeners... */ app.on('window-all-closed', () => { // Respect the OSX convention of having the application in memory even // after all windows have been closed if (process.platform !== 'darwin') { app.quit(); } }); app.on('ready', async () => { if (process.env.NODE_ENV === 'development' || process.env.DEBUG_PROD === 'true') { await installExtensions(); } mainWindow = new BrowserWindow({ show: false, frame: false }); mainWindow.maximize(); mainWindow.loadURL(`file://${__dirname}/app.html`); // @TODO: Use 'ready-to-show' event // https://github.com/electron/electron/blob/master/docs/api/browser-window.md#using-ready-to-show-event mainWindow.webContents.on('did-finish-load', () => { if (!mainWindow) { throw new Error('"mainWindow" is not defined') } mainWindow.show() mainWindow.focus() if (syncing) { mainWindow.webContents.send('lndSyncing') } }) mainWindow.on('closed', () => { mainWindow = null; }) const menuBuilder = new MenuBuilder(mainWindow); menuBuilder.buildMenu() // Check to see if and LND process is running lookup({ command: 'lnd' }, (err, results) => { // There was an error checking for the LND process if (err) { throw new Error( err ) } // No LND process was found if (!results.length) { // Let the front end know we have started syncing LND syncing = true // Run a bash script that checks for the LND folder and generates Node.js compatible certs console.log('CHECKING/GENERATING CERTS') exec(`sh ${path.join(__dirname, '..', 'resources', 'scripts', `${plat}_generate_certs.sh`)}`) // After the certs are generated, it's time to start LND console.log('STARTING LND') const lndPath = path.join(__dirname, '..', 'resources', 'bin', plat, plat === 'win32' ? 'lnd.exe' : 'lnd') neutrino = spawn(lndPath, [ '--bitcoin.active', '--bitcoin.testnet', '--neutrino.active', '--neutrino.connect=faucet.lightning.community:18333', '--autopilot.active', '--debuglevel=debug', '--no-macaroons' ] ) .on('error', error => console.log(`lnd error: ${error}`)) .on('close', code => console.log(`lnd shutting down ${code}`)) // Listen for when neutrino prints out data neutrino.stdout.on('data', data => { // Data stored in variable line, log line to the console let line = data.toString('utf8') console.log(line) // Pass current clock height progress to front end for loading state UX if (line.includes('Caught up to height')) { const blockHeight = line.slice(line.indexOf('Caught up to height') + 'Caught up to height'.length).trim() mainWindow.webContents.send('lndStdout', blockHeight) } // When LND is all caught up to the blockchain if (line.includes('Done catching up block hashes')) { // Log that LND is caught up to the current block height console.log('DONE CATCHING UP BLOCK HASHES') // Call lnd lnd((lndSubscribe, lndMethods) => { // Subscribe to bi-directional streams lndSubscribe(mainWindow) // Listen for all gRPC restful methods ipcMain.on('lnd', (event, { msg, data }) => { lndMethods(event, msg, data) }) // Let the front end know we have stopped syncing LND syncing = false mainWindow.webContents.send('lndSynced') }) } }) } else { // An LND process was found, no need to start our own console.log('LND ALREADY RUNNING') // Call lnd lnd((lndSubscribe, lndMethods) => { // Subscribe to bi-directional streams lndSubscribe(mainWindow) // Listen for all gRPC restful methods ipcMain.on('lnd', (event, { msg, data }) => { lndMethods(event, msg, data) }) }) } }) }); app.setAsDefaultProtocolClient('lightning') app.on('open-url', function (event, url) { event.preventDefault() const payreq = url.split(':')[1] mainWindow.webContents.send('lightningPaymentUri', { payreq }) mainWindow.show() })