|
|
|
// @flow
|
|
|
|
|
|
|
|
import { app, ipcMain, dialog, BrowserWindow } from 'electron'
|
|
|
|
import pick from 'lodash.pick'
|
|
|
|
import StateMachine from 'javascript-state-machine'
|
|
|
|
import { mainLog } from '../utils/log'
|
|
|
|
|
|
|
|
import LndConfig, { chains, networks, types } from '../lnd/config'
|
|
|
|
import Lightning from '../lnd/lightning'
|
|
|
|
import Neutrino from '../lnd/neutrino'
|
|
|
|
import WalletUnlocker from '../lnd/walletUnlocker'
|
|
|
|
|
|
|
|
type onboardingOptions = {
|
|
|
|
id?: number,
|
|
|
|
type: $Keys<typeof types>,
|
|
|
|
chain?: $Keys<typeof chains>,
|
|
|
|
network?: $Keys<typeof networks>,
|
|
|
|
host?: string,
|
|
|
|
cert?: string,
|
|
|
|
macaroon?: string,
|
|
|
|
alias?: string,
|
|
|
|
autopilot?: boolean
|
|
|
|
}
|
|
|
|
|
|
|
|
const grpcSslCipherSuites = connectionType =>
|
|
|
|
(connectionType === 'btcpayserver'
|
|
|
|
? [
|
|
|
|
// BTCPay Server serves lnd behind an nginx proxy with a trusted SSL cert from Lets Encrypt.
|
|
|
|
// These certs use an RSA TLS cipher suite.
|
|
|
|
'ECDHE-RSA-AES256-GCM-SHA384',
|
|
|
|
'ECDHE-RSA-AES128-GCM-SHA256'
|
|
|
|
]
|
|
|
|
: [
|
|
|
|
// Default is ECDHE-RSA-AES128-GCM-SHA256:ECDHE-RSA-AES256-GCM-SHA384
|
|
|
|
// https://github.com/grpc/grpc/blob/master/doc/environment_variables.md
|
|
|
|
//
|
|
|
|
// Current LND cipher suites here:
|
|
|
|
// https://github.com/lightningnetwork/lnd/blob/master/lnd.go#L80
|
|
|
|
//
|
|
|
|
// We order the suites by priority, based on the recommendations provided by SSL Labs here:
|
|
|
|
// https://github.com/ssllabs/research/wiki/SSL-and-TLS-Deployment-Best-Practices#23-use-secure-cipher-suites
|
|
|
|
'ECDHE-ECDSA-AES128-GCM-SHA256',
|
|
|
|
'ECDHE-ECDSA-AES256-GCM-SHA384',
|
|
|
|
'ECDHE-ECDSA-AES128-CBC-SHA256',
|
|
|
|
'ECDHE-ECDSA-CHACHA20-POLY1305'
|
|
|
|
]
|
|
|
|
).join(':')
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @class ZapController
|
|
|
|
*
|
|
|
|
* The ZapController class coordinates actions between the the main nand renderer processes.
|
|
|
|
*/
|
|
|
|
class ZapController {
|
|
|
|
mainWindow: BrowserWindow
|
|
|
|
neutrino: Neutrino
|
|
|
|
lightning: Lightning
|
|
|
|
walletUnlocker: WalletUnlocker
|
|
|
|
lndConfig: LndConfig
|
|
|
|
fsm: StateMachine
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new ZapController instance.
|
|
|
|
* @param {BrowserWindow} mainWindow BrowserWindow instance to interact with.
|
|
|
|
*/
|
|
|
|
constructor(mainWindow: BrowserWindow) {
|
|
|
|
this.fsm = new StateMachine({
|
|
|
|
transitions: [
|
|
|
|
{ name: 'startOnboarding', from: '*', to: 'onboarding' },
|
|
|
|
{ name: 'startLocalLnd', from: 'onboarding', to: 'running' },
|
|
|
|
{ name: 'startRemoteLnd', from: 'onboarding', to: 'connected' },
|
|
|
|
{ name: 'terminate', from: '*', to: 'terminated' }
|
|
|
|
],
|
|
|
|
methods: {
|
|
|
|
onOnboarding: this.onOnboarding.bind(this),
|
|
|
|
onStartOnboarding: this.onStartOnboarding.bind(this),
|
|
|
|
onBeforeStartLocalLnd: this.onBeforeStartLocalLnd.bind(this),
|
|
|
|
onBeforeStartRemoteLnd: this.onBeforeStartRemoteLnd.bind(this),
|
|
|
|
onTerminated: this.onTerminated.bind(this),
|
|
|
|
onTerminate: this.onTerminate.bind(this)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
// Variable to hold the main window instance.
|
|
|
|
this.mainWindow = mainWindow
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialize the controller.
|
|
|
|
*/
|
|
|
|
init() {
|
|
|
|
// Load the application into the main window.
|
|
|
|
if (process.env.HOT) {
|
|
|
|
const port = process.env.PORT || 1212
|
|
|
|
this.mainWindow.loadURL(`http://localhost:${port}/dist/index.html`)
|
|
|
|
} else {
|
|
|
|
this.mainWindow.loadURL(`file://${__dirname}/dist/index.html`)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Show the window as soon as the application has finished loading.
|
|
|
|
this.mainWindow.webContents.on('did-finish-load', () => {
|
|
|
|
this.mainWindow.show()
|
|
|
|
this.mainWindow.focus()
|
|
|
|
|
|
|
|
// Start the onboarding process.
|
|
|
|
this.startOnboarding()
|
|
|
|
})
|
|
|
|
|
|
|
|
// When the window is closed, just hide it unless we are force closing.
|
|
|
|
this.mainWindow.on('close', e => {
|
|
|
|
if (process.platform === 'darwin' && !this.mainWindow.forceClose) {
|
|
|
|
e.preventDefault()
|
|
|
|
this.mainWindow.hide()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
// Dereference the window object, usually you would store windows in an array if your app supports multi windows,
|
|
|
|
// this is the time when you should delete the corresponding element.
|
|
|
|
this.mainWindow.on('closed', () => {
|
|
|
|
this.mainWindow = null
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// FSM Proxies
|
|
|
|
// ------------------------------------
|
|
|
|
|
|
|
|
startOnboarding(...args: any[]) {
|
|
|
|
return this.fsm.startOnboarding(...args)
|
|
|
|
}
|
|
|
|
startLocalLnd(...args: any[]) {
|
|
|
|
return this.fsm.startLocalLnd(...args)
|
|
|
|
}
|
|
|
|
startRemoteLnd(...args: any[]) {
|
|
|
|
return this.fsm.startRemoteLnd(...args)
|
|
|
|
}
|
|
|
|
terminate(...args: any[]) {
|
|
|
|
return this.fsm.terminate(...args)
|
|
|
|
}
|
|
|
|
is(...args: any[]) {
|
|
|
|
return this.fsm.is(...args)
|
|
|
|
}
|
|
|
|
can(...args: any[]) {
|
|
|
|
return this.fsm.can(...args)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// FSM Callbacks
|
|
|
|
// ------------------------------------
|
|
|
|
|
|
|
|
async onOnboarding(lifecycle: any) {
|
|
|
|
mainLog.debug('[FSM] onOnboarding...')
|
|
|
|
|
|
|
|
// Remove any existing IPC listeners so that we can start fresh.
|
|
|
|
this._removeIpcListeners()
|
|
|
|
|
|
|
|
// Register IPC listeners so that we can react to instructions coming from the app.
|
|
|
|
this._registerIpcListeners()
|
|
|
|
|
|
|
|
// Disconnect any pre-existing lightning wallet connection.
|
|
|
|
if (lifecycle.from === 'connected' && this.lightning && this.lightning.can('disconnect')) {
|
|
|
|
this.lightning.disconnect()
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we are comming from a running state, stop the Neutrino process.
|
|
|
|
else if (lifecycle.from === 'running') {
|
|
|
|
await this.shutdownNeutrino()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Give the grpc connections a chance to be properly closed out.
|
|
|
|
return new Promise(resolve => setTimeout(resolve, 200))
|
|
|
|
}
|
|
|
|
|
|
|
|
onStartOnboarding() {
|
|
|
|
mainLog.debug('[FSM] onStartOnboarding...')
|
|
|
|
|
|
|
|
// Notify the app to start the onboarding process.
|
|
|
|
this.sendMessage('startOnboarding')
|
|
|
|
}
|
|
|
|
|
|
|
|
onBeforeStartLocalLnd() {
|
|
|
|
mainLog.debug('[FSM] onBeforeStartLocalLnd...')
|
|
|
|
|
|
|
|
mainLog.info('Starting new lnd instance')
|
|
|
|
mainLog.info(' > alias:', this.lndConfig.alias)
|
|
|
|
mainLog.info(' > autopilot:', this.lndConfig.autopilot)
|
|
|
|
|
|
|
|
return this.startNeutrino()
|
|
|
|
}
|
|
|
|
|
|
|
|
onBeforeStartRemoteLnd() {
|
|
|
|
mainLog.debug('[FSM] onBeforeStartRemoteLnd...')
|
|
|
|
mainLog.info('Connecting to custom lnd instance')
|
|
|
|
mainLog.info(' > host:', this.lndConfig.host)
|
|
|
|
mainLog.info(' > cert:', this.lndConfig.cert)
|
|
|
|
mainLog.info(' > macaroon:', this.lndConfig.macaroon)
|
|
|
|
|
|
|
|
return this.startLightningWallet()
|
|
|
|
.then(() => this.sendMessage('walletConnected'))
|
|
|
|
.catch(e => {
|
|
|
|
const errors = {}
|
|
|
|
// There was a problem connecting to the host.
|
|
|
|
if (e.code === 'LND_GRPC_HOST_ERROR') {
|
|
|
|
errors.host = e.message
|
|
|
|
}
|
|
|
|
// There was a problem accessing the ssl cert.
|
|
|
|
if (e.code === 'LND_GRPC_CERT_ERROR') {
|
|
|
|
errors.cert = e.message
|
|
|
|
}
|
|
|
|
// There was a problem accessing the macaroon file.
|
|
|
|
else if (e.code === 'LND_GRPC_MACAROON_ERROR') {
|
|
|
|
errors.macaroon = e.message
|
|
|
|
}
|
|
|
|
// Other error codes such as UNAVAILABLE most likely indicate that there is a problem with the host.
|
|
|
|
else {
|
|
|
|
errors.host = `Unable to connect to host: ${e.details || e.message}`
|
|
|
|
}
|
|
|
|
|
|
|
|
// The `startLightningWallet` call attempts to call the `getInfo` method on the Lightning service in order to
|
|
|
|
// verify that it is accessible. If it is not, an error 12 is thrown which is the gRPC code for `UNIMPLEMENTED`
|
|
|
|
// which indicates that the requested operation is not implemented or not supported/enabled in the service.
|
|
|
|
// See https://github.com/grpc/grpc-node/blob/master/packages/grpc-native-core/src/constants.js#L129
|
|
|
|
if (e.code === 12) {
|
|
|
|
return this.startWalletUnlocker()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Notify the app of errors.
|
|
|
|
this.sendMessage('startLndError', errors)
|
|
|
|
throw e
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
async onTerminated(lifecycle: any) {
|
|
|
|
mainLog.debug('[FSM] onTerminated...')
|
|
|
|
|
|
|
|
// Disconnect from any existing lightning wallet connection.
|
|
|
|
if (lifecycle.from === 'connected' && this.lightning && this.lightning.can('disconnect')) {
|
|
|
|
this.lightning.disconnect()
|
|
|
|
}
|
|
|
|
// If we are comming from a running state, stop the Neutrino process.
|
|
|
|
else if (lifecycle.from === 'running') {
|
|
|
|
await this.shutdownNeutrino()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onTerminate() {
|
|
|
|
mainLog.debug('[FSM] onTerminate...')
|
|
|
|
app.quit()
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// Helpers
|
|
|
|
// ------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send a message to the main window.
|
|
|
|
* @param {string} msg message to send.
|
|
|
|
* @param {[type]} data additional data to accompany the message.
|
|
|
|
*/
|
|
|
|
sendMessage(msg: string, data: any) {
|
|
|
|
if (this.mainWindow) {
|
|
|
|
mainLog.info('Sending message to renderer process: %o', { msg, data })
|
|
|
|
this.mainWindow.webContents.send(msg, data)
|
|
|
|
} else {
|
|
|
|
mainLog.warn('Unable to send message to renderer process (main window not available): %o', {
|
|
|
|
msg,
|
|
|
|
data
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Start the wallet unlocker.
|
|
|
|
*/
|
|
|
|
async startWalletUnlocker() {
|
|
|
|
mainLog.info('Establishing connection to Wallet Unlocker gRPC interface...')
|
|
|
|
this.walletUnlocker = new WalletUnlocker(this.lndConfig)
|
|
|
|
|
|
|
|
// Connect to the WalletUnlocker interface.
|
|
|
|
try {
|
|
|
|
await this.walletUnlocker.connect()
|
|
|
|
|
|
|
|
// Listen for all gRPC restful methods and pass to gRPC.
|
|
|
|
ipcMain.on('walletUnlocker', (event, { msg, data }) =>
|
|
|
|
this.walletUnlocker.registerMethods(event, msg, data)
|
|
|
|
)
|
|
|
|
|
|
|
|
// Notify the renderer that the wallet unlocker is active.
|
|
|
|
this.sendMessage('walletUnlockerGrpcActive', this.lndConfig)
|
|
|
|
} catch (err) {
|
|
|
|
mainLog.warn('Unable to connect to WalletUnlocker gRPC interface: %o', err)
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create and subscribe to the Lightning service.
|
|
|
|
*/
|
|
|
|
async startLightningWallet() {
|
|
|
|
mainLog.info('Establishing connection to Lightning gRPC interface...')
|
|
|
|
this.lightning = new Lightning(this.lndConfig)
|
|
|
|
|
|
|
|
// Connect to the Lightning interface.
|
|
|
|
try {
|
|
|
|
await this.lightning.connect()
|
|
|
|
|
|
|
|
this.lightning.subscribe(this.mainWindow)
|
|
|
|
|
|
|
|
// Listen for all gRPC restful methods and pass to gRPC.
|
|
|
|
ipcMain.on('lnd', (event, { msg, data }) => this.lightning.registerMethods(event, msg, data))
|
|
|
|
|
|
|
|
// Let the renderer know that we are connected.
|
|
|
|
this.sendMessage('lightningGrpcActive', this.lndConfig)
|
|
|
|
} catch (err) {
|
|
|
|
mainLog.warn('Unable to connect to Lightning gRPC interface: %o', err)
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Starts the LND node and attach event listeners.
|
|
|
|
* @return {Neutrino} Neutrino instance.
|
|
|
|
*/
|
|
|
|
startNeutrino() {
|
|
|
|
mainLog.info('Starting Neutrino...')
|
|
|
|
this.neutrino = new Neutrino(this.lndConfig)
|
|
|
|
|
|
|
|
this.neutrino.on('error', error => {
|
|
|
|
mainLog.error(`Got error from lnd process: ${error})`)
|
|
|
|
dialog.showMessageBox({
|
|
|
|
type: 'error',
|
|
|
|
message: `lnd error: ${error}`
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
this.neutrino.on('exit', (code, signal, lastError) => {
|
|
|
|
mainLog.info(`Lnd process has shut down (code: ${code}, signal: ${signal})`)
|
|
|
|
if (this.is('running') || this.is('connected')) {
|
|
|
|
dialog.showMessageBox({
|
|
|
|
type: 'error',
|
|
|
|
message: `Lnd has unexpectedly quit:\n\nError code: ${code}\nExit signal: ${signal}\nLast error: ${lastError}`
|
|
|
|
})
|
|
|
|
this.terminate()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
this.neutrino.on('wallet-unlocker-grpc-active', () => {
|
|
|
|
mainLog.info('Wallet unlocker gRPC active')
|
|
|
|
this.startWalletUnlocker()
|
|
|
|
})
|
|
|
|
|
|
|
|
this.neutrino.on('chain-sync-waiting', () => {
|
|
|
|
mainLog.info('Neutrino sync waiting')
|
|
|
|
this.sendMessage('lndSyncStatus', 'waiting')
|
|
|
|
})
|
|
|
|
|
|
|
|
this.neutrino.on('chain-sync-started', () => {
|
|
|
|
mainLog.info('Neutrino sync started')
|
|
|
|
this.sendMessage('lndSyncStatus', 'in-progress')
|
|
|
|
})
|
|
|
|
|
|
|
|
this.neutrino.on('chain-sync-finished', () => {
|
|
|
|
mainLog.info('Neutrino sync finished')
|
|
|
|
this.sendMessage('lndSyncStatus', 'complete')
|
|
|
|
})
|
|
|
|
|
|
|
|
this.neutrino.on('got-current-block-height', height => {
|
|
|
|
this.sendMessage('currentBlockHeight', Number(height))
|
|
|
|
})
|
|
|
|
|
|
|
|
this.neutrino.on('got-lnd-block-height', height => {
|
|
|
|
this.sendMessage('lndBlockHeight', Number(height))
|
|
|
|
})
|
|
|
|
|
|
|
|
this.neutrino.on('got-lnd-cfilter-height', height => {
|
|
|
|
this.sendMessage('lndCfilterHeight', Number(height))
|
|
|
|
})
|
|
|
|
|
|
|
|
return this.neutrino.start()
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gracefully shutdown LND.
|
|
|
|
*/
|
|
|
|
async shutdownNeutrino() {
|
|
|
|
// We only want to shut down LND if we are running it locally.
|
|
|
|
if (this.lndConfig.type !== 'local' || !this.neutrino || !this.neutrino.process) {
|
|
|
|
return Promise.resolve()
|
|
|
|
}
|
|
|
|
|
|
|
|
mainLog.info('Shutting down Neutrino...')
|
|
|
|
|
|
|
|
return new Promise(async resolve => {
|
|
|
|
// HACK: Sometimes there are errors during the shutdown process that prevent the daemon from shutting down at
|
|
|
|
// all. If we haven't received notification of the process closing within 10 seconds, kill it.
|
|
|
|
// See https://github.com/lightningnetwork/lnd/pull/1781
|
|
|
|
// See https://github.com/lightningnetwork/lnd/pull/1783
|
|
|
|
const shutdownTimeout = setTimeout(() => {
|
|
|
|
this.neutrino.removeListener('exit', exitHandler)
|
|
|
|
if (this.neutrino) {
|
|
|
|
mainLog.warn('Graceful shutdown failed to complete within 10 seconds.')
|
|
|
|
this.neutrino.kill('SIGTERM')
|
|
|
|
resolve()
|
|
|
|
}
|
|
|
|
}, 1000 * 10)
|
|
|
|
|
|
|
|
const exitHandler = () => {
|
|
|
|
clearTimeout(shutdownTimeout)
|
|
|
|
resolve()
|
|
|
|
}
|
|
|
|
this.neutrino.once('exit', exitHandler)
|
|
|
|
|
|
|
|
// The Lightning service is only active once the wallet has been unlocked and a gRPC connection has been made.
|
|
|
|
// If it is active, disconnect from it before we terminate neutrino.
|
|
|
|
if (this.lightning && this.lightning.can('terminate')) {
|
|
|
|
await this.lightning.disconnect()
|
|
|
|
}
|
|
|
|
// Kill the Neutrino process (sends SIGINT to Neutrino process)
|
|
|
|
this.neutrino.kill()
|
|
|
|
}).then(() => mainLog.info('Neutrino shutdown complete'))
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Start or connect to lnd process after onboarding has been completed by the app.
|
|
|
|
*/
|
|
|
|
async startLnd(options: onboardingOptions) {
|
|
|
|
mainLog.info('Starting lnd with options: %o', options)
|
|
|
|
|
|
|
|
// Save the lnd config options that we got from the renderer.
|
|
|
|
this.lndConfig = new LndConfig({
|
|
|
|
id: options.id,
|
|
|
|
type: options.type || 'local',
|
|
|
|
chain: options.chain || 'bitcoin',
|
|
|
|
network: options.network || 'testnet',
|
|
|
|
settings: pick(options, LndConfig.SETTINGS_PROPS[options.type])
|
|
|
|
})
|
|
|
|
|
|
|
|
// Set up SSL with the cypher suits that we need based on the connection type.
|
|
|
|
process.env.GRPC_SSL_CIPHER_SUITES =
|
|
|
|
process.env.GRPC_SSL_CIPHER_SUITES || grpcSslCipherSuites(options.type)
|
|
|
|
|
|
|
|
// If the requested connection type is a local one then start up a new lnd instance.
|
|
|
|
// Otherwise attempt to connect to an lnd instance using user supplied connection details.
|
|
|
|
return options.type === 'local' ? this.startLocalLnd() : this.startRemoteLnd()
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add IPC event listeners...
|
|
|
|
*/
|
|
|
|
_registerIpcListeners() {
|
|
|
|
ipcMain.on('startLnd', (event, options: onboardingOptions) => this.startLnd(options))
|
|
|
|
ipcMain.on('startLightningWallet', () =>
|
|
|
|
this.startLightningWallet().catch(e => {
|
|
|
|
// Notify the app of errors.
|
|
|
|
this.sendMessage('startLndError', e.message)
|
|
|
|
|
|
|
|
// Return back to the start of the onboarding process.
|
|
|
|
return this.startOnboarding()
|
|
|
|
})
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add IPC event listeners...
|
|
|
|
*/
|
|
|
|
_removeIpcListeners() {
|
|
|
|
ipcMain.removeAllListeners('startLnd')
|
|
|
|
ipcMain.removeAllListeners('startLightningWallet')
|
|
|
|
ipcMain.removeAllListeners('walletUnlocker')
|
|
|
|
ipcMain.removeAllListeners('lnd')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default ZapController
|