You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
113 lines
3.4 KiB
113 lines
3.4 KiB
import grpc from 'grpc'
|
|
import { loadSync } from '@grpc/proto-loader'
|
|
import config 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'
|
|
|
|
/**
|
|
* Creates an LND grpc client lightning service.
|
|
* @returns {Lightning}
|
|
*/
|
|
class Lightning {
|
|
constructor() {
|
|
this.mainWindow = null
|
|
this.lnd = null
|
|
this.subscriptions = {
|
|
channelGraph: null,
|
|
invoices: null,
|
|
transactions: null
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Connect to the gRPC interface and verify it is functional.
|
|
* @return {Promise<rpc.lnrpc.Lightning>}
|
|
*/
|
|
async connect() {
|
|
const lndConfig = config.lnd()
|
|
const { host, rpcProtoPath, cert, macaroon } = lndConfig
|
|
|
|
// Verify that the host is valid before creating a gRPC client that is connected to it.
|
|
return await 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: String,
|
|
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.
|
|
const lnd = new rpc.lnrpc.Lightning(host, credentials)
|
|
|
|
// Call the getInfo method to ensure that we can make successful calls to the gRPC interface.
|
|
return new Promise((resolve, reject) => {
|
|
lnd.getInfo({}, { deadline: getDeadline(2) }, err => {
|
|
if (err) {
|
|
return reject(err)
|
|
}
|
|
this.lnd = lnd
|
|
return resolve(lnd)
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Discomnnect the gRPC service.
|
|
*/
|
|
disconnect() {
|
|
this.unsubscribe()
|
|
this.lnd.close()
|
|
}
|
|
|
|
/**
|
|
* Hook up lnd restful methods.
|
|
*/
|
|
lndMethods(event, msg, data) {
|
|
return methods(this.lnd, mainLog, event, msg, data)
|
|
}
|
|
|
|
/**
|
|
* Subscribe to all bi-directional streams.
|
|
*/
|
|
subscribe(mainWindow) {
|
|
this.mainWindow = mainWindow
|
|
this.subscriptions.channelGraph = subscribeToChannelGraph(this.mainWindow, this.lnd, mainLog)
|
|
this.subscriptions.invoices = subscribeToInvoices(this.mainWindow, this.lnd, mainLog)
|
|
this.subscriptions.transactions = subscribeToTransactions(this.mainWindow, this.lnd, mainLog)
|
|
}
|
|
|
|
/**
|
|
* Unsubscribe from all bi-directional streams.
|
|
*/
|
|
unsubscribe() {
|
|
Object.keys(this.subscriptions).forEach(subscription => {
|
|
if (this.subscriptions[subscription]) {
|
|
this.subscriptions[subscription].cancel()
|
|
this.subscriptions[subscription] = null
|
|
}
|
|
})
|
|
this.mainWindow = null
|
|
}
|
|
}
|
|
|
|
export default Lightning
|
|
|