2022-05-06 19:32:20 +00:00
|
|
|
import { createHash } from 'crypto'
|
2021-06-27 03:18:32 +00:00
|
|
|
import { bech32 } from 'bech32'
|
2023-10-03 23:22:56 +00:00
|
|
|
import { lnAddrSchema } from './validate'
|
2024-12-16 20:05:31 +00:00
|
|
|
import { FetchTimeoutError } from '@/lib/fetch'
|
|
|
|
import { WALLET_CREATE_INVOICE_TIMEOUT_MS } from './constants'
|
2021-06-27 03:18:32 +00:00
|
|
|
|
2022-05-06 19:32:20 +00:00
|
|
|
export function encodeLNUrl (url) {
|
2021-06-27 03:18:32 +00:00
|
|
|
const words = bech32.toWords(Buffer.from(url.toString(), 'utf8'))
|
2022-05-06 19:32:20 +00:00
|
|
|
return bech32.encode('lnurl', words, 1023)
|
2021-06-27 03:18:32 +00:00
|
|
|
}
|
2021-10-07 21:03:54 +00:00
|
|
|
|
|
|
|
export function lnurlPayMetadataString (username) {
|
|
|
|
return JSON.stringify([[
|
|
|
|
'text/plain',
|
|
|
|
`Funding @${username} on stacker.news`
|
2022-05-19 13:18:28 +00:00
|
|
|
], [
|
|
|
|
'text/identifier',
|
|
|
|
`${username}@stacker.news`
|
2021-10-07 21:03:54 +00:00
|
|
|
]])
|
|
|
|
}
|
|
|
|
|
2022-05-19 14:22:25 +00:00
|
|
|
export function lnurlPayDescriptionHashForUser (username) {
|
|
|
|
return lnurlPayDescriptionHash(lnurlPayMetadataString(username))
|
|
|
|
}
|
|
|
|
|
|
|
|
export function lnurlPayDescriptionHash (data) {
|
|
|
|
return createHash('sha256').update(data).digest('hex')
|
2021-10-07 21:03:54 +00:00
|
|
|
}
|
2023-10-03 23:22:56 +00:00
|
|
|
|
2024-12-16 20:05:31 +00:00
|
|
|
export async function lnAddrOptions (addr, { signal } = {}) {
|
2023-10-03 23:22:56 +00:00
|
|
|
await lnAddrSchema().fields.addr.validate(addr)
|
|
|
|
const [name, domain] = addr.split('@')
|
2024-01-13 04:02:17 +00:00
|
|
|
let protocol = 'https'
|
|
|
|
if (process.env.NODE_ENV === 'development') {
|
|
|
|
// support HTTP and HTTPS during development
|
2024-04-08 22:54:39 +00:00
|
|
|
protocol = process.env.NEXT_PUBLIC_URL.split('://')[0]
|
2024-01-13 04:02:17 +00:00
|
|
|
}
|
2024-03-24 18:25:14 +00:00
|
|
|
const unexpectedErrorMessage = `An unexpected error occurred fetching the Lightning Address metadata for ${addr}. Check the address and try again.`
|
|
|
|
let res
|
2024-12-16 20:05:31 +00:00
|
|
|
const url = `${protocol}://${domain}/.well-known/lnurlp/${name}`
|
2024-03-24 18:25:14 +00:00
|
|
|
try {
|
2024-12-16 20:05:31 +00:00
|
|
|
const req = await fetch(url, { signal })
|
2024-03-24 18:25:14 +00:00
|
|
|
res = await req.json()
|
|
|
|
} catch (err) {
|
|
|
|
console.log('Error fetching lnurlp', err)
|
2024-12-16 20:05:31 +00:00
|
|
|
if (err.name === 'TimeoutError') {
|
|
|
|
throw new FetchTimeoutError('GET', url, WALLET_CREATE_INVOICE_TIMEOUT_MS)
|
|
|
|
}
|
|
|
|
// If `fetch` fails, or if `req.json` fails, catch it here and surface a reasonable error
|
2024-03-24 18:25:14 +00:00
|
|
|
throw new Error(unexpectedErrorMessage)
|
|
|
|
}
|
2023-10-03 23:22:56 +00:00
|
|
|
if (res.status === 'ERROR') {
|
2024-03-24 18:25:14 +00:00
|
|
|
// if the response doesn't adhere to spec by providing a `reason` entry, returns a default error message
|
|
|
|
throw new Error(res.reason ?? unexpectedErrorMessage)
|
2023-10-03 23:22:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const { minSendable, maxSendable, ...leftOver } = res
|
|
|
|
return { min: minSendable / 1000, max: maxSendable / 1000, ...leftOver }
|
|
|
|
}
|