2024-04-16 00:25:30 +00:00
|
|
|
import fetch from 'cross-fetch'
|
2024-04-14 22:34:21 +00:00
|
|
|
import https from 'https'
|
2024-04-15 14:16:32 +00:00
|
|
|
import crypto from 'crypto'
|
2024-05-03 22:00:28 +00:00
|
|
|
import { HttpProxyAgent, HttpsProxyAgent } from '@/lib/proxy'
|
2024-05-16 13:41:49 +00:00
|
|
|
import { TOR_REGEXP } from '@/lib/url'
|
2024-04-14 22:34:21 +00:00
|
|
|
|
|
|
|
export const createInvoice = async ({ socket, rune, cert, label, description, msats, expiry }) => {
|
2024-05-03 22:00:28 +00:00
|
|
|
let protocol, agent
|
|
|
|
const httpsAgentOptions = { ca: cert ? Buffer.from(cert, 'base64') : undefined }
|
2024-05-16 13:41:49 +00:00
|
|
|
const isOnion = TOR_REGEXP.test(socket)
|
2024-05-03 22:00:28 +00:00
|
|
|
if (isOnion) {
|
|
|
|
// we support HTTP and HTTPS over Tor
|
|
|
|
protocol = cert ? 'https:' : 'http:'
|
|
|
|
// we need to use our Tor proxy to resolve onion addresses
|
2024-05-10 02:07:47 +00:00
|
|
|
const proxyOptions = { proxy: process.env.TOR_PROXY }
|
2024-05-03 22:00:28 +00:00
|
|
|
agent = protocol === 'https:'
|
2024-05-10 02:07:47 +00:00
|
|
|
? new HttpsProxyAgent({ ...proxyOptions, ...httpsAgentOptions, rejectUnauthorized: false })
|
2024-05-03 22:00:28 +00:00
|
|
|
: new HttpProxyAgent(proxyOptions)
|
|
|
|
} else {
|
|
|
|
// we only support HTTPS over clearnet
|
|
|
|
agent = new https.Agent(httpsAgentOptions)
|
|
|
|
protocol = 'https:'
|
|
|
|
}
|
|
|
|
|
|
|
|
const url = `${protocol}//${socket}/v1/invoice`
|
2024-04-14 22:34:21 +00:00
|
|
|
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({
|
2024-04-15 14:16:32 +00:00
|
|
|
// CLN requires a unique label for every invoice
|
|
|
|
// see https://docs.corelightning.org/reference/lightning-invoice
|
|
|
|
label: crypto.randomBytes(16).toString('hex'),
|
2024-04-14 22:34:21 +00:00
|
|
|
description,
|
|
|
|
amount_msat: msats,
|
|
|
|
expiry
|
|
|
|
})
|
|
|
|
})
|
|
|
|
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)
|