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} */ 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