Browse Source

Merge pull request #719 from mrfelton/fix/lnd-graceful-shutdown

fix(lnd): ensure graceful shutdown
renovate/lint-staged-8.x
JimmyMow 7 years ago
committed by GitHub
parent
commit
f1ad88e951
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 69
      app/lib/lnd/lightning.js
  2. 8
      app/lib/lnd/neutrino.js
  3. 148
      app/lib/zap/controller.js
  4. 1
      package.json
  5. 50
      test/unit/lnd/neutrino.spec.js
  6. 8
      yarn.lock

69
app/lib/lnd/lightning.js

@ -3,6 +3,7 @@
import grpc from 'grpc' import grpc from 'grpc'
import { loadSync } from '@grpc/proto-loader' import { loadSync } from '@grpc/proto-loader'
import { BrowserWindow } from 'electron' import { BrowserWindow } from 'electron'
import StateMachine from 'javascript-state-machine'
import LndConfig from './config' import LndConfig from './config'
import { getDeadline, validateHost, createSslCreds, createMacaroonCreds } from './util' import { getDeadline, validateHost, createSslCreds, createMacaroonCreds } from './util'
import methods from './methods' import methods from './methods'
@ -25,24 +26,43 @@ type LightningSubscriptionsType = {
class Lightning { class Lightning {
mainWindow: BrowserWindow mainWindow: BrowserWindow
lnd: any lnd: any
lndConfig: LndConfig
subscriptions: LightningSubscriptionsType subscriptions: LightningSubscriptionsType
_fsm: StateMachine
constructor() { // Transitions provided by the state machine.
connect: any
disconnect: any
terminate: any
is: any
can: any
state: string
constructor(lndConfig: LndConfig) {
this.mainWindow = null this.mainWindow = null
this.lnd = null this.lnd = null
this.lndConfig = lndConfig
this.subscriptions = { this.subscriptions = {
channelGraph: null, channelGraph: null,
invoices: null, invoices: null,
transactions: null transactions: null
} }
// Initialize the state machine.
this._fsm()
} }
// ------------------------------------
// FSM Callbacks
// ------------------------------------
/** /**
* Connect to the gRPC interface and verify it is functional. * Connect to the gRPC interface and verify it is functional.
* @return {Promise<rpc.lnrpc.Lightning>} * @return {Promise<rpc.lnrpc.Lightning>}
*/ */
async connect(lndConfig: LndConfig) { async onBeforeConnect() {
const { rpcProtoPath, host, cert, macaroon } = lndConfig mainLog.info('Connecting to Lightning gRPC service')
const { rpcProtoPath, host, cert, macaroon } = this.lndConfig
// Verify that the host is valid before creating a gRPC client that is connected to it. // Verify that the host is valid before creating a gRPC client that is connected to it.
return await validateHost(host).then(async () => { return await validateHost(host).then(async () => {
@ -69,16 +89,15 @@ class Lightning {
const credentials = grpc.credentials.combineChannelCredentials(sslCreds, macaroonCreds) const credentials = grpc.credentials.combineChannelCredentials(sslCreds, macaroonCreds)
// Create a new gRPC client instance. // Create a new gRPC client instance.
const lnd = new rpc.lnrpc.Lightning(host, credentials) this.lnd = new rpc.lnrpc.Lightning(host, credentials)
// Call the getInfo method to ensure that we can make successful calls to the gRPC interface. // Wait for the gRPC connection to be established.
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
lnd.getInfo({}, { deadline: getDeadline(2) }, err => { grpc.waitForClientReady(this.lnd, getDeadline(2), err => {
if (err) { if (err) {
return reject(err) return reject(err)
} }
this.lnd = lnd return resolve()
return resolve(lnd)
}) })
}) })
}) })
@ -87,13 +106,34 @@ class Lightning {
/** /**
* Discomnnect the gRPC service. * Discomnnect the gRPC service.
*/ */
disconnect() { onBeforeDisconnect() {
mainLog.info('Disconnecting from Lightning gRPC service')
this.unsubscribe() this.unsubscribe()
if (this.lnd) { if (this.lnd) {
this.lnd.close() this.lnd.close()
} }
} }
/**
* Gracefully shutdown the gRPC service.
*/
async onBeforeTerminate() {
mainLog.info('Shutting down Lightning daemon')
this.unsubscribe()
return new Promise((resolve, reject) => {
this.lnd.stopDaemon({}, (err, data) => {
if (err) {
return reject(err)
}
resolve(data)
})
})
}
// ------------------------------------
// Helpers
// ------------------------------------
/** /**
* Hook up lnd restful methods. * Hook up lnd restful methods.
*/ */
@ -105,6 +145,7 @@ class Lightning {
* Subscribe to all bi-directional streams. * Subscribe to all bi-directional streams.
*/ */
subscribe(mainWindow: BrowserWindow) { subscribe(mainWindow: BrowserWindow) {
mainLog.info('Subscribing to Lightning gRPC streams')
this.mainWindow = mainWindow this.mainWindow = mainWindow
this.subscriptions.channelGraph = subscribeToChannelGraph.call(this) this.subscriptions.channelGraph = subscribeToChannelGraph.call(this)
@ -116,6 +157,7 @@ class Lightning {
* Unsubscribe from all bi-directional streams. * Unsubscribe from all bi-directional streams.
*/ */
unsubscribe() { unsubscribe() {
mainLog.info('Unsubscribing from Lightning gRPC streams')
this.mainWindow = null this.mainWindow = null
Object.keys(this.subscriptions).forEach(subscription => { Object.keys(this.subscriptions).forEach(subscription => {
if (this.subscriptions[subscription]) { if (this.subscriptions[subscription]) {
@ -126,4 +168,13 @@ class Lightning {
} }
} }
StateMachine.factory(Lightning, {
init: 'ready',
transitions: [
{ name: 'connect', from: 'ready', to: 'connected' },
{ name: 'disconnect', from: 'connected', to: 'ready' },
{ name: 'terminate', from: 'connected', to: 'ready' }
]
})
export default Lightning export default Lightning

8
app/lib/lnd/neutrino.js

@ -141,7 +141,10 @@ class Neutrino extends EventEmitter {
if (this.is(CHAIN_SYNC_PENDING) || this.is(CHAIN_SYNC_IN_PROGRESS)) { if (this.is(CHAIN_SYNC_PENDING) || this.is(CHAIN_SYNC_IN_PROGRESS)) {
// If we cant get a connectionn to the backend. // If we cant get a connectionn to the backend.
if (line.includes('Waiting for chain backend to finish sync')) { if (
line.includes('Waiting for chain backend to finish sync') ||
line.includes('Waiting for block headers to sync, then will start cfheaders sync')
) {
this.setState(CHAIN_SYNC_WAITING) this.setState(CHAIN_SYNC_WAITING)
} }
// If we are still waiting for the back end to finish synncing. // If we are still waiting for the back end to finish synncing.
@ -223,8 +226,9 @@ class Neutrino extends EventEmitter {
/** /**
* Stop the Lnd process. * Stop the Lnd process.
*/ */
stop() { kill() {
if (this.process) { if (this.process) {
mainLog.info('Killing Neutrino process...')
this.process.kill() this.process.kill()
this.process = null this.process = null
} }

148
app/lib/zap/controller.js

@ -51,10 +51,9 @@ const grpcSslCipherSuites = connectionType =>
*/ */
class ZapController { class ZapController {
mainWindow: BrowserWindow mainWindow: BrowserWindow
neutrino: any neutrino: Neutrino
lightning: any lightning: Lightning
splashScreenTime: number splashScreenTime: number
lightningGrpcConnected: boolean
lndConfig: LndConfig lndConfig: LndConfig
_fsm: StateMachine _fsm: StateMachine
@ -73,18 +72,9 @@ class ZapController {
// Variable to hold the main window instance. // Variable to hold the main window instance.
this.mainWindow = mainWindow this.mainWindow = mainWindow
// Keep a reference any neutrino process started by us.
this.neutrino = undefined
// Keep a reference to the lightning gRPC instance.
this.lightning = undefined
// Time for the splash screen to remain visible. // Time for the splash screen to remain visible.
this.splashScreenTime = 500 this.splashScreenTime = 500
// Boolean indicating wether the lightning grpc is connected ot not.
this.lightningGrpcConnected = false
// Initialize the state machine. // Initialize the state machine.
this._fsm() this._fsm()
@ -127,7 +117,7 @@ class ZapController {
// FSM Callbacks // FSM Callbacks
// ------------------------------------ // ------------------------------------
onOnboarding() { async onOnboarding(lifecycle: any) {
mainLog.debug('[FSM] onOnboarding...') mainLog.debug('[FSM] onOnboarding...')
// Remove any existing IPC listeners so that we can start fresh. // Remove any existing IPC listeners so that we can start fresh.
@ -136,12 +126,14 @@ class ZapController {
// Register IPC listeners so that we can react to instructions coming from the app. // Register IPC listeners so that we can react to instructions coming from the app.
this._registerIpcListeners() this._registerIpcListeners()
// Ensure wallet is disconnected. // Disconnect any pre-existing lightning wallet connection.
this.disconnectLightningWallet() if (lifecycle.from === 'connected' && this.lightning && this.lightning.can('disconnect')) {
this.lightning.disconnect()
}
// If Neutrino is running, kill it. // If we are comming from a running state, stop the Neutrino process.
if (this.neutrino) { else if (lifecycle.from === 'running') {
this.neutrino.stop() await this.shutdownNeutrino()
} }
// Give the grpc connections a chance to be properly closed out. // Give the grpc connections a chance to be properly closed out.
@ -209,15 +201,16 @@ class ZapController {
}) })
} }
onTerminated() { async onTerminated(lifecycle: any) {
mainLog.debug('[FSM] onTerminated...') mainLog.debug('[FSM] onTerminated...')
// Unsubscribe the gRPC streams before thhe window closes. This ensures that we can properly reestablish a fresh
// connection when a new window is opened.
this.disconnectLightningWallet()
// If Neutrino is running, kill it. // Disconnect from any existing lightning wallet connection.
if (this.neutrino) { if (lifecycle.from === 'connected' && this.lightning && this.lightning.can('disconnect')) {
this.neutrino.stop() this.lightning.disconnect()
}
// If we are comming from a running state, stop the Neutrino process.
else if (lifecycle.from === 'running') {
await this.shutdownNeutrino()
} }
} }
@ -251,7 +244,7 @@ class ZapController {
* Start the wallet unlocker. * Start the wallet unlocker.
*/ */
startWalletUnlocker() { startWalletUnlocker() {
mainLog.info('Starting wallet unlocker...') mainLog.info('Establishing connection to Wallet Unlocker gRPC interface...')
try { try {
const walletUnlockerMethods = initWalletUnlocker(this.lndConfig) const walletUnlockerMethods = initWalletUnlocker(this.lndConfig)
@ -275,42 +268,23 @@ class ZapController {
* Create and subscribe to the Lightning service. * Create and subscribe to the Lightning service.
*/ */
async startLightningWallet() { async startLightningWallet() {
if (this.lightningGrpcConnected) { mainLog.info('Establishing connection to Lightning gRPC interface...')
return this.lightning = new Lightning(this.lndConfig)
}
mainLog.info('Starting lightning wallet...')
this.lightning = new Lightning()
// Connect to the Lightning interface. // Connect to the Lightning interface.
await this.lightning.connect(this.lndConfig) try {
await this.lightning.connect()
// Subscribe the main window to receive streams.
this.lightning.subscribe(this.mainWindow)
// Listen for all gRPC restful methods and pass to gRPC.
ipcMain.on('lnd', (event, { msg, data }) => this.lightning.lndMethods(event, msg, data))
// Let the renderer know that we are connected. this.lightning.subscribe(this.mainWindow)
this.sendMessage('lightningGrpcActive')
// Update our internal state. // Listen for all gRPC restful methods and pass to gRPC.
this.lightningGrpcConnected = true ipcMain.on('lnd', (event, { msg, data }) => this.lightning.lndMethods(event, msg, data))
}
/** // Let the renderer know that we are connected.
* Unsubscribe from the Lightning service. this.sendMessage('lightningGrpcActive')
*/ } catch (err) {
disconnectLightningWallet() { mainLog.warn('Unable to connect to Lighitnng gRPC interface: %o', err)
if (!this.lightningGrpcConnected) {
return
} }
mainLog.info('Disconnecting lightning Wallet...')
// Disconnect streams.
this.lightning.disconnect()
// Update our internal state.
this.lightningGrpcConnected = false
} }
/** /**
@ -337,7 +311,7 @@ class ZapController {
if (this.is('running') || this.is('connected')) { if (this.is('running') || this.is('connected')) {
dialog.showMessageBox({ dialog.showMessageBox({
type: 'error', type: 'error',
message: `Lnd has unexpectadly quit: ${lastError}` message: `Lnd has unexpectedly quit: ${lastError}`
}) })
this.terminate() this.terminate()
} }
@ -383,6 +357,66 @@ class ZapController {
this.neutrino.start() 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') {
return Promise.resolve()
}
// Attempt a graceful shutdown if we can.
if (this.lightning && this.lightning.can('terminate')) {
mainLog.info('Shutting down Neutrino...')
return new Promise(resolve => {
// HACK: Sometimes there are errors during the shutdown process that prevent the daeming 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('close', closeHandler)
if (this.neutrino) {
mainLog.warn('Graceful shutdown failed to complete within 30 seconds.')
this.neutrino.kill()
resolve()
}
}, 1000 * 10)
// HACK: The Lightning.stopDaemon` call returns before lnd has actually fully completed the shutdown process
// so we add a listener on the close event so that we can wrap things up once the process has been fully closed
// out.
const closeHandler = function() {
mainLog.info('Neutrino shutdown complete.')
clearTimeout(shutdownTimeout)
resolve()
}
this.neutrino.once('close', closeHandler)
this.lightning
.terminate()
.then(() => mainLog.info('Neutrino Daemon shutdown complete'))
.catch(err => {
mainLog.error('Unable to gracefully shutdown LND: %o', err)
// Kill the process ourselves here to ensure that we don't leave hanging processes.
if (this.neutrino) {
this.neutrino.kill()
resolve()
}
})
})
}
// The Lightning service is only active after the wallet has been unlocked and a gRPC connection has been
// established. In this case, kill the Neutrino process to ensure that we don't leave hanging process.
// FIXME: This currencly doesn't do a graceful shutdown as LND does not properly handle SIGTERM.
// See https://github.com/lightningnetwork/lnd/issues/1028
else if (this.neutrino) {
this.neutrino.kill()
}
}
finishOnboarding(options: onboardingOptions) { finishOnboarding(options: onboardingOptions) {
mainLog.info('Finishing onboarding') mainLog.info('Finishing onboarding')
// Save the lnd config options that we got from the renderer. // Save the lnd config options that we got from the renderer.

1
package.json

@ -258,6 +258,7 @@
"lint-staged": "^7.2.0", "lint-staged": "^7.2.0",
"lnd-binary": "^0.3.5", "lnd-binary": "^0.3.5",
"minimist": "^1.2.0", "minimist": "^1.2.0",
"mock-spawn": "^0.2.6",
"node-sass": "^4.9.0", "node-sass": "^4.9.0",
"prettier": "^1.13.5", "prettier": "^1.13.5",
"react-addons-test-utils": "^15.6.2", "react-addons-test-utils": "^15.6.2",

50
test/unit/lnd/neutrino.spec.js

@ -1,10 +1,22 @@
// @flow
import Neutrino from 'lib/lnd/neutrino' import Neutrino from 'lib/lnd/neutrino'
import LndConfig from 'lib/lnd/config'
import mockSpawn from 'mock-spawn'
jest.mock('electron-store') jest.mock('electron-store')
jest.mock('child_process', () => {
var mockSpawn = require('mock-spawn')
return {
spawn: mockSpawn()
}
})
describe('Neutrino', function() { describe('Neutrino', function() {
describe('Constructor', () => { describe('Constructor', () => {
beforeAll(() => (this.neutrino = new Neutrino())) beforeAll(() => {
this.neutrino = new Neutrino(new LndConfig())
})
describe('initial values', () => { describe('initial values', () => {
it('should set the "process" property to null', () => { it('should set the "process" property to null', () => {
@ -37,7 +49,7 @@ describe('Neutrino', function() {
describe('.setState', () => { describe('.setState', () => {
describe('called with new state', () => { describe('called with new state', () => {
beforeEach(() => { beforeEach(() => {
this.neutrino = new Neutrino() this.neutrino = new Neutrino(new LndConfig())
this.callback = jest.fn() this.callback = jest.fn()
this.newVal = 'chain-sync-finished' this.newVal = 'chain-sync-finished'
this.neutrino.on('chain-sync-finished', this.callback) this.neutrino.on('chain-sync-finished', this.callback)
@ -53,7 +65,7 @@ describe('Neutrino', function() {
}) })
describe('called with current state', () => { describe('called with current state', () => {
beforeEach(() => { beforeEach(() => {
this.neutrino = new Neutrino() this.neutrino = new Neutrino(new LndConfig())
this.callback = jest.fn() this.callback = jest.fn()
this.newVal = 'chain-sync-pending' this.newVal = 'chain-sync-pending'
this.neutrino.on('chain-sync-pending', this.callback) this.neutrino.on('chain-sync-pending', this.callback)
@ -72,7 +84,7 @@ describe('Neutrino', function() {
describe('.setCurrentBlockHeight', () => { describe('.setCurrentBlockHeight', () => {
describe('called with higher height', () => { describe('called with higher height', () => {
beforeEach(() => { beforeEach(() => {
this.neutrino = new Neutrino() this.neutrino = new Neutrino(new LndConfig())
this.callback = jest.fn() this.callback = jest.fn()
this.newVal = 100 this.newVal = 100
this.neutrino.on('got-current-block-height', this.callback) this.neutrino.on('got-current-block-height', this.callback)
@ -89,7 +101,7 @@ describe('Neutrino', function() {
}) })
describe('called with lower height', () => { describe('called with lower height', () => {
beforeEach(() => { beforeEach(() => {
this.neutrino = new Neutrino() this.neutrino = new Neutrino(new LndConfig())
this.callback = jest.fn() this.callback = jest.fn()
this.newVal = -1 this.newVal = -1
this.neutrino.on('got-current-block-height', this.callback) this.neutrino.on('got-current-block-height', this.callback)
@ -108,7 +120,7 @@ describe('Neutrino', function() {
describe('.setLndBlockHeight', () => { describe('.setLndBlockHeight', () => {
describe('called with higher height', () => { describe('called with higher height', () => {
beforeEach(() => { beforeEach(() => {
this.neutrino = new Neutrino() this.neutrino = new Neutrino(new LndConfig())
this.callback = jest.fn() this.callback = jest.fn()
this.newVal = 100 this.newVal = 100
this.neutrino.on('got-lnd-block-height', this.callback) this.neutrino.on('got-lnd-block-height', this.callback)
@ -130,7 +142,7 @@ describe('Neutrino', function() {
}) })
describe('called with lower height', () => { describe('called with lower height', () => {
beforeEach(() => { beforeEach(() => {
this.neutrino = new Neutrino() this.neutrino = new Neutrino(new LndConfig())
this.callback = jest.fn() this.callback = jest.fn()
this.newVal = -1 this.newVal = -1
this.neutrino.on('got-lnd-block-height', this.callback) this.neutrino.on('got-lnd-block-height', this.callback)
@ -152,7 +164,9 @@ describe('Neutrino', function() {
describe('.is', () => { describe('.is', () => {
describe('called with current state', () => { describe('called with current state', () => {
beforeEach(() => (this.neutrino = new Neutrino())) beforeEach(() => {
this.neutrino = new Neutrino(new LndConfig())
})
it('should returnn true if the current state matches', () => { it('should returnn true if the current state matches', () => {
expect(this.neutrino.is('chain-sync-pending')).toEqual(true) expect(this.neutrino.is('chain-sync-pending')).toEqual(true)
@ -164,10 +178,20 @@ describe('Neutrino', function() {
}) })
describe('.start', () => { describe('.start', () => {
describe('called when neutrino is not running', () => {
beforeEach(() => {
this.neutrino = new Neutrino(new LndConfig())
this.neutrino.start()
})
it('should set the subprocess object on the `process` property', () => {
expect(this.neutrino.process.pid).toBeDefined()
})
})
describe('called when neutrino is already running', () => { describe('called when neutrino is already running', () => {
beforeEach(() => { beforeEach(() => {
this.neutrino = new Neutrino() this.neutrino = new Neutrino(new LndConfig())
this.neutrino.process = 123 this.neutrino.process = mockSpawn()
}) })
it('should throw an error', () => { it('should throw an error', () => {
expect(() => { expect(() => {
@ -177,14 +201,14 @@ describe('Neutrino', function() {
}) })
}) })
describe('.stop', () => { describe('.kill', () => {
describe('called when neutrino is already running', () => { describe('called when neutrino is already running', () => {
beforeEach(() => { beforeEach(() => {
this.neutrino = new Neutrino() this.neutrino = new Neutrino(new LndConfig())
this.neutrino.process = { this.neutrino.process = {
kill: jest.fn() kill: jest.fn()
} }
this.neutrino.stop() this.neutrino.kill()
}) })
it('should kill the neutrino process', () => { it('should kill the neutrino process', () => {
expect(this.neutrino.process).toBeNull() expect(this.neutrino.process).toBeNull()

8
yarn.lock

@ -8084,6 +8084,12 @@ mkdirp@0.5.0:
dependencies: dependencies:
minimist "0.0.8" minimist "0.0.8"
mock-spawn@^0.2.6:
version "0.2.6"
resolved "https://registry.yarnpkg.com/mock-spawn/-/mock-spawn-0.2.6.tgz#b39c15a1c067504310144151f2c1de344d03937f"
dependencies:
through "2.3.x"
moment@^2.22.2: moment@^2.22.2:
version "2.22.2" version "2.22.2"
resolved "https://registry.yarnpkg.com/moment/-/moment-2.22.2.tgz#3c257f9839fc0e93ff53149632239eb90783ff66" resolved "https://registry.yarnpkg.com/moment/-/moment-2.22.2.tgz#3c257f9839fc0e93ff53149632239eb90783ff66"
@ -11648,7 +11654,7 @@ through2@~0.2.3:
readable-stream "~1.1.9" readable-stream "~1.1.9"
xtend "~2.1.1" xtend "~2.1.1"
through@2, "through@>=2.2.7 <3", through@^2.3.6, through@^2.3.8: through@2, through@2.3.x, "through@>=2.2.7 <3", through@^2.3.6, through@^2.3.8:
version "2.3.8" version "2.3.8"
resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5" resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5"

Loading…
Cancel
Save