import { WALLET_CREATE_INVOICE_TIMEOUT_MS } from '@/lib/constants' import { FetchTimeoutError } from '@/lib/fetch' import { msatsToSats } from '@/lib/format' import { getAgent } from '@/lib/proxy' import { assertContentTypeJson } from '@/lib/url' import fetch from 'cross-fetch' export * from '@/wallets/lnbits' export async function testCreateInvoice ({ url, invoiceKey }, { signal }) { return await createInvoice({ msats: 1000, expiry: 1 }, { url, invoiceKey }, { signal }) } export async function createInvoice ( { msats, description, descriptionHash, expiry }, { url, invoiceKey }, { signal }) { const path = '/api/v1/payments' const headers = new Headers() headers.append('Accept', 'application/json') headers.append('Content-Type', 'application/json') headers.append('X-Api-Key', invoiceKey) // lnbits doesn't support msats so we have to floor to nearest sat const sats = msatsToSats(msats) const body = JSON.stringify({ amount: sats, unit: 'sat', expiry, memo: description, out: false }) let hostname = url.replace(/^https?:\/\//, '') const agent = getAgent({ hostname }) if (process.env.NODE_ENV !== 'production' && hostname.startsWith('localhost:')) { // to make it possible to attach LNbits for receives during local dev hostname = 'lnbits:5000' } let res try { res = await fetch(`${agent.protocol}//${hostname}${path}`, { method: 'POST', headers, agent, body, signal }) } catch (err) { if (err.name === 'AbortError') { // XXX node-fetch doesn't throw our custom TimeoutError but throws a generic error so we have to handle that manually. // see https://github.com/node-fetch/node-fetch/issues/1462 throw new FetchTimeoutError('POST', url, WALLET_CREATE_INVOICE_TIMEOUT_MS) } throw err } assertContentTypeJson(res) if (!res.ok) { const errBody = await res.json() throw new Error(errBody.detail) } const payment = await res.json() return payment.payment_request }