stacker.news/lib/cln.js

169 lines
4.3 KiB
JavaScript
Raw Normal View History

import fetch from 'cross-fetch'
import crypto from 'crypto'
import { getAgent } from '@/lib/proxy'
import { assertContentTypeJson, assertResponseOk } from './url'
export const createInvoice = async ({ msats, description, expiry }, { socket, rune, cert }) => {
const agent = getAgent({ hostname: socket, cert })
const url = `${agent.protocol}//${socket}/v1/invoice`
const res = await fetch(url, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
Rune: rune,
// can be any node id, only required for CLN v23.08 and below
// see https://docs.corelightning.org/docs/rest#server
nodeId: '02cb2e2d5a6c5b17fa67b1a883e2973c82e328fb9bd08b2b156a9e23820c87a490'
},
agent,
body: JSON.stringify({
// CLN requires a unique label for every invoice
// see https://docs.corelightning.org/reference/lightning-invoice
label: crypto.randomBytes(16).toString('hex'),
description,
amount_msat: msats,
expiry
})
})
supercharged wallet logs (#1516) * Inject wallet logger interface * Include method in NWC logs * Fix wrong page total * Poll for new logs every second * Fix overlapping pagination * Remove unused total * Better logs for incoming payments * Use _setLogs instead of wrapper * Remove inconsistent receive log * Remove console.log from wallet logger on server * Fix missing 'wallet detached' log * Fix confirm_withdrawl code * Remove duplicate autowithdrawal log * Add context to log * Add more context * Better table styling * Move CSS for wallet logs into one file * remove unused logNav class * rename classes * Align key with second column * Fix TypeError if context empty * Check content-type header before calling res.json() * Fix duplicate 'failed to create invoice' * Parse details from LND error * Fix invalid DOM property 'colspan' * P2P zap logs with context * Remove unnecessary withdrawal error log * the code assignment was broken anyway * we already log withdrawal errors using .catch on payViaPaymentRequest * Don't show outgoing fee to receiver to avoid confusion * Fix typo in comment * Log if invoice was canceled by payer * Automatically populate context from bolt11 * Fix missing context * Fix wrap errors not logged * Only log cancel if client canceled * Remove unused imports * Log withdrawal/forward success/error in payment flow * Fix boss not passed to checkInvoice * Fix TypeError * Fix database timeouts caused by logger The logger shares the same connection pool with any currently running transaction. This means that we enter a classic deadlock when we await logger calls: the logger call is waiting for a connection but the currently running transaction is waiting for the logger call to finish before it can release a connection. * Fix cache returning undefined * Fix typo in comment * Add padding-right to key in log context * Always use 'incoming payment failed:' --------- Co-authored-by: Keyan <34140557+huumn@users.noreply.github.com>
2024-11-08 19:26:40 +00:00
assertResponseOk(res)
assertContentTypeJson(res)
supercharged wallet logs (#1516) * Inject wallet logger interface * Include method in NWC logs * Fix wrong page total * Poll for new logs every second * Fix overlapping pagination * Remove unused total * Better logs for incoming payments * Use _setLogs instead of wrapper * Remove inconsistent receive log * Remove console.log from wallet logger on server * Fix missing 'wallet detached' log * Fix confirm_withdrawl code * Remove duplicate autowithdrawal log * Add context to log * Add more context * Better table styling * Move CSS for wallet logs into one file * remove unused logNav class * rename classes * Align key with second column * Fix TypeError if context empty * Check content-type header before calling res.json() * Fix duplicate 'failed to create invoice' * Parse details from LND error * Fix invalid DOM property 'colspan' * P2P zap logs with context * Remove unnecessary withdrawal error log * the code assignment was broken anyway * we already log withdrawal errors using .catch on payViaPaymentRequest * Don't show outgoing fee to receiver to avoid confusion * Fix typo in comment * Log if invoice was canceled by payer * Automatically populate context from bolt11 * Fix missing context * Fix wrap errors not logged * Only log cancel if client canceled * Remove unused imports * Log withdrawal/forward success/error in payment flow * Fix boss not passed to checkInvoice * Fix TypeError * Fix database timeouts caused by logger The logger shares the same connection pool with any currently running transaction. This means that we enter a classic deadlock when we await logger calls: the logger call is waiting for a connection but the currently running transaction is waiting for the logger call to finish before it can release a connection. * Fix cache returning undefined * Fix typo in comment * Add padding-right to key in log context * Always use 'incoming payment failed:' --------- Co-authored-by: Keyan <34140557+huumn@users.noreply.github.com>
2024-11-08 19:26:40 +00:00
const inv = await res.json()
if (inv.error) {
throw new Error(inv.error.message)
}
return inv
}
// https://github.com/clams-tech/rune-decoder/blob/57c2e76d1ef9ab7336f565b99de300da1c7b67ce/src/index.ts
export const decodeRune = (rune) => {
const runeBinary = Base64Binary.decode(rune)
const hashBinary = runeBinary.slice(0, 32)
const hash = binaryHashToHex(hashBinary)
const restBinary = runeBinary.slice(32)
const [uniqueId, ...restrictionStrings] = new TextDecoder().decode(restBinary).split('&')
const id = uniqueId.split('=')[1]
// invalid rune checks
if (!id) return null
if (restrictionStrings.some(invalidAscii)) return null
const restrictions = restrictionStrings.map((restriction) => {
const alternatives = restriction.split('|')
const summary = alternatives.reduce((str, alternative) => {
const [operator] = alternative.match(runeOperatorRegex) || []
if (!operator) return str
const [name, value] = alternative.split(operator)
return `${str ? `${str} OR ` : ''}${name} ${operatorToDescription(operator)} ${value}`
}, '')
return {
alternatives,
summary
}
})
return {
id,
hash,
restrictions
}
}
const Base64Binary = {
_keyStr: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=',
removePaddingChars: function (input) {
const lkey = this._keyStr.indexOf(input.charAt(input.length - 1))
if (lkey === 64) {
return input.substring(0, input.length - 1)
}
return input
},
decode: function (input) {
// get last chars to see if are valid
input = this.removePaddingChars(input)
input = this.removePaddingChars(input)
const bytes = parseInt(((input.length / 4) * 3).toString(), 10)
let chr1, chr2, chr3
let enc1, enc2, enc3, enc4
let i = 0
let j = 0
const uarray = new Uint8Array(bytes)
for (i = 0; i < bytes; i += 3) {
// get the 3 octects in 4 ascii chars
enc1 = this._keyStr.indexOf(input.charAt(j++))
enc2 = this._keyStr.indexOf(input.charAt(j++))
enc3 = this._keyStr.indexOf(input.charAt(j++))
enc4 = this._keyStr.indexOf(input.charAt(j++))
chr1 = (enc1 << 2) | (enc2 >> 4)
chr2 = ((enc2 & 15) << 4) | (enc3 >> 2)
chr3 = ((enc3 & 3) << 6) | enc4
uarray[i] = chr1
if (enc3 !== 64) uarray[i + 1] = chr2
if (enc4 !== 64) uarray[i + 2] = chr3
}
return uarray
}
}
function i2hex (i) {
return ('0' + i.toString(16)).slice(-2)
}
const binaryHashToHex = (hash) => {
return hash.reduce(function (memo, i) {
return memo + i2hex(i)
}, '')
}
const runeOperatorRegex = /[=^$/~<>{}#!]/g
const operatorToDescription = (operator) => {
switch (operator) {
case '=':
return 'is equal to'
case '^':
return 'starts with'
case '$':
return 'ends with'
case '/':
return 'is not equal to'
case '~':
return 'contains'
case '<':
return 'is less than'
case '>':
return 'is greater than'
case '{':
return 'sorts before'
case '}':
return 'sorts after'
case '#':
return 'comment'
case '!':
return 'is missing'
default:
return ''
}
}
const invalidAscii = (str) => !![...str].some((char) => char.charCodeAt(0) > 127)