|
|
|
// @flow
|
|
|
|
|
|
|
|
import grpc from 'grpc'
|
|
|
|
import { loadSync } from '@grpc/proto-loader'
|
|
|
|
import { BrowserWindow } from 'electron'
|
|
|
|
import StateMachine from 'javascript-state-machine'
|
|
|
|
import LndConfig from './config'
|
|
|
|
import { getDeadline, validateHost, createSslCreds, createMacaroonCreds } from './util'
|
|
|
|
import methods from './methods'
|
|
|
|
import { mainLog } from '../utils/log'
|
|
|
|
import subscribeToTransactions from './subscribe/transactions'
|
|
|
|
import subscribeToInvoices from './subscribe/invoices'
|
|
|
|
import subscribeToChannelGraph from './subscribe/channelgraph'
|
|
|
|
import { getInfo } from './methods/networkController'
|
|
|
|
|
|
|
|
// Type definition for subscriptions property.
|
|
|
|
type LightningSubscriptionsType = {
|
|
|
|
channelGraph: any,
|
|
|
|
invoices: any,
|
|
|
|
transactions: any
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates an LND grpc client lightning service.
|
|
|
|
* @returns {Lightning}
|
|
|
|
*/
|
|
|
|
class Lightning {
|
|
|
|
mainWindow: BrowserWindow
|
|
|
|
service: any
|
|
|
|
lndConfig: LndConfig
|
|
|
|
subscriptions: LightningSubscriptionsType
|
|
|
|
_fsm: StateMachine
|
|
|
|
|
|
|
|
// 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.service = null
|
|
|
|
this.lndConfig = lndConfig
|
|
|
|
this.subscriptions = {
|
|
|
|
channelGraph: null,
|
|
|
|
invoices: null,
|
|
|
|
transactions: null
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize the state machine.
|
|
|
|
this._fsm()
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// FSM Callbacks
|
|
|
|
// ------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Connect to the gRPC interface and verify it is functional.
|
|
|
|
* @return {Promise<rpc.lnrpc.Lightning>}
|
|
|
|
*/
|
|
|
|
async onBeforeConnect() {
|
|
|
|
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.
|
|
|
|
return validateHost(host)
|
|
|
|
.then(async () => {
|
|
|
|
// Load the gRPC proto file.
|
|
|
|
// The following options object closely approximates the existing behavior of grpc.load.
|
|
|
|
// See https://github.com/grpc/grpc-node/blob/master/packages/grpc-protobufjs/README.md
|
|
|
|
const options = {
|
|
|
|
keepCase: true,
|
|
|
|
longs: Number,
|
|
|
|
enums: String,
|
|
|
|
defaults: true,
|
|
|
|
oneofs: true
|
|
|
|
}
|
|
|
|
const packageDefinition = loadSync(rpcProtoPath, options)
|
|
|
|
|
|
|
|
// Load gRPC package definition as a gRPC object hierarchy.
|
|
|
|
const rpc = grpc.loadPackageDefinition(packageDefinition)
|
|
|
|
|
|
|
|
// Create ssl and macaroon credentials to use with the gRPC client.
|
|
|
|
const [sslCreds, macaroonCreds] = await Promise.all([
|
|
|
|
createSslCreds(cert),
|
|
|
|
createMacaroonCreds(macaroon)
|
|
|
|
])
|
|
|
|
const credentials = grpc.credentials.combineChannelCredentials(sslCreds, macaroonCreds)
|
|
|
|
|
|
|
|
// Create a new gRPC client instance.
|
|
|
|
this.service = new rpc.lnrpc.Lightning(host, credentials)
|
|
|
|
|
|
|
|
// Wait for the gRPC connection to be established.
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
this.service.waitForReady(getDeadline(5), err => {
|
|
|
|
if (err) {
|
|
|
|
return reject(err)
|
|
|
|
}
|
|
|
|
return resolve()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
.then(() => getInfo(this.service))
|
|
|
|
.catch(err => {
|
|
|
|
this.service.close()
|
|
|
|
throw err
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Discomnnect the gRPC service.
|
|
|
|
*/
|
|
|
|
onBeforeDisconnect() {
|
|
|
|
mainLog.info('Disconnecting from Lightning gRPC service')
|
|
|
|
this.unsubscribe()
|
|
|
|
if (this.service) {
|
|
|
|
this.service.close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gracefully shutdown the gRPC service.
|
|
|
|
*/
|
|
|
|
async onBeforeTerminate() {
|
|
|
|
mainLog.info('Shutting down Lightning daemon')
|
|
|
|
this.unsubscribe()
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
this.service.stopDaemon({}, (err, data) => {
|
|
|
|
if (err) {
|
|
|
|
return reject(err)
|
|
|
|
}
|
|
|
|
resolve(data)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// Helpers
|
|
|
|
// ------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Hook up lnd restful methods.
|
|
|
|
*/
|
|
|
|
registerMethods(event: Event, msg: string, data: any) {
|
|
|
|
return methods(this.service, mainLog, event, msg, data)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Subscribe to all bi-directional streams.
|
|
|
|
*/
|
|
|
|
subscribe(mainWindow: BrowserWindow) {
|
|
|
|
mainLog.info('Subscribing to Lightning gRPC streams')
|
|
|
|
this.mainWindow = mainWindow
|
|
|
|
|
|
|
|
this.subscriptions.channelGraph = subscribeToChannelGraph.call(this)
|
|
|
|
this.subscriptions.invoices = subscribeToInvoices.call(this)
|
|
|
|
this.subscriptions.transactions = subscribeToTransactions.call(this)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unsubscribe from all bi-directional streams.
|
|
|
|
*/
|
|
|
|
unsubscribe() {
|
|
|
|
mainLog.info('Unsubscribing from Lightning gRPC streams')
|
|
|
|
this.mainWindow = null
|
|
|
|
Object.keys(this.subscriptions).forEach(subscription => {
|
|
|
|
if (this.subscriptions[subscription]) {
|
|
|
|
this.subscriptions[subscription].cancel()
|
|
|
|
this.subscriptions[subscription] = null
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|