cc289089cf
* not-custodial zap scaffolding * invoice forward state machine * small refinements to state machine * make wrap invoice work * get state machine working end to end * untested logic layout for paidAction invoice wraps * perform pessimisitic actions before outgoing payment * working end to end * remove unneeded params from wallets/server/createInvoice * fix cltv relative/absolute confusion + cancelling forwards * small refinements * add p2p wrap info to paidAction docs * fallback to SN invoice when wrap fails * fix paidAction retry description * consistent naming scheme for state machine * refinements * have sn pay bounded outbound fee * remove debug logging * reenable lnc permissions checks * don't p2p zap on item forward splits * make createInvoice params json encodeable * direct -> p2p badge on notifications * allow no tls in dev for core lightning * fix autowithdraw to create invoice with msats * fix autowithdraw msats/sats inconsitency * label p2p zaps properly in satistics * add fees to autowithdrawal notifications * add RETRYING as terminal paid action state * Update api/paidAction/README.md Co-authored-by: ekzyis <ek@stacker.news> * Update api/paidAction/README.md Co-authored-by: ekzyis <ek@stacker.news> * Update api/lnd/index.js Co-authored-by: ekzyis <ek@stacker.news> * ek suggestions * add bugetable to nwc card * get paranoid with numbers * better finalize retries and better max timeout height * refine forward failure transitions * more accurate satistics p2p status * make sure paidaction cancel in state machine only * dont drop bolt11s unless status is not null * only allow PENDING_HELD to transition to FORWARDING * add mermaid state machine diagrams to paid action doc * fix cancel transition name * cleanup readme * move forwarding outside of transition * refine testServerConnect and make sure ensureB64 transforms * remove unused params from testServerConnect --------- Co-authored-by: ekzyis <ek@stacker.news> Co-authored-by: k00b <k00b@stacker.news>
96 lines
2.9 KiB
JavaScript
96 lines
2.9 KiB
JavaScript
import { toPositiveNumber } from '@/lib/validate'
|
|
import lndService from 'ln-service'
|
|
|
|
const { lnd } = lndService.authenticatedLndGrpc({
|
|
cert: process.env.LND_CERT,
|
|
macaroon: process.env.LND_MACAROON,
|
|
socket: process.env.LND_SOCKET
|
|
})
|
|
|
|
// Check LND GRPC connection
|
|
lndService.getWalletInfo({ lnd }, (err, result) => {
|
|
if (err) {
|
|
console.error('LND GRPC connection error')
|
|
return
|
|
}
|
|
console.log('LND GRPC connection successful')
|
|
})
|
|
|
|
export async function estimateRouteFee ({ lnd, destination, tokens, mtokens, request, timeout }) {
|
|
return await new Promise((resolve, reject) => {
|
|
lnd.router.estimateRouteFee({
|
|
dest: Buffer.from(destination, 'hex'),
|
|
amt_sat: tokens ? toPositiveNumber(tokens) : toPositiveNumber(BigInt(mtokens) / BigInt(1e3)),
|
|
payment_request: request,
|
|
timeout
|
|
}, (err, res) => {
|
|
if (err) {
|
|
reject(err)
|
|
return
|
|
}
|
|
|
|
if (res?.failure_reason) {
|
|
reject(new Error(`Unable to estimate route: ${res.failure_reason}`))
|
|
return
|
|
}
|
|
|
|
if (res.routing_fee_msat < 0 || res.time_lock_delay <= 0) {
|
|
reject(new Error('Unable to estimate route, excessive values: ' + JSON.stringify(res)))
|
|
return
|
|
}
|
|
|
|
resolve({
|
|
routingFeeMsat: toPositiveNumber(res.routing_fee_msat),
|
|
timeLockDelay: toPositiveNumber(res.time_lock_delay)
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
// created_height is the accepted_height, timeout is the expiry height
|
|
// ln-service remaps the `htlcs` field of lookupInvoice to `payments` and
|
|
// see: https://github.com/alexbosworth/lightning/blob/master/lnd_responses/htlc_as_payment.js
|
|
// and: https://lightning.engineering/api-docs/api/lnd/lightning/lookup-invoice/index.html#lnrpcinvoicehtlc
|
|
export function hodlInvoiceCltvDetails (inv) {
|
|
if (!inv.payments) {
|
|
throw new Error('No payments found')
|
|
}
|
|
if (!inv.is_held) {
|
|
throw new Error('Invoice is not held')
|
|
}
|
|
|
|
const acceptHeight = inv.payments.reduce((max, htlc) => {
|
|
const createdHeight = toPositiveNumber(htlc.created_height)
|
|
return createdHeight > max ? createdHeight : max
|
|
}, 0)
|
|
const expiryHeight = inv.payments.reduce((min, htlc) => {
|
|
const timeout = toPositiveNumber(htlc.timeout)
|
|
return timeout < min ? timeout : min
|
|
}, Number.MAX_SAFE_INTEGER)
|
|
|
|
return {
|
|
expiryHeight: toPositiveNumber(expiryHeight),
|
|
acceptHeight: toPositiveNumber(acceptHeight)
|
|
}
|
|
}
|
|
|
|
export function getPaymentFailureStatus (withdrawal) {
|
|
if (withdrawal && !withdrawal.is_failed) {
|
|
throw new Error('withdrawal is not failed')
|
|
}
|
|
|
|
if (withdrawal?.failed.is_insufficient_balance) {
|
|
return 'INSUFFICIENT_BALANCE'
|
|
} else if (withdrawal?.failed.is_invalid_payment) {
|
|
return 'INVALID_PAYMENT'
|
|
} else if (withdrawal?.failed.is_pathfinding_timeout) {
|
|
return 'PATHFINDING_TIMEOUT'
|
|
} else if (withdrawal?.failed.is_route_not_found) {
|
|
return 'ROUTE_NOT_FOUND'
|
|
}
|
|
|
|
return 'UNKNOWN_FAILURE'
|
|
}
|
|
|
|
export default lnd
|