2024-11-16 00:38:14 +00:00
|
|
|
import { getScopes, SCOPE_READ, SCOPE_WRITE, getWallet, request } from '@/wallets/blink/common'
|
|
|
|
export * from '@/wallets/blink'
|
2024-08-18 16:36:55 +00:00
|
|
|
|
2024-12-16 20:05:31 +00:00
|
|
|
export async function testSendPayment ({ apiKey, currency }, { logger, signal }) {
|
2024-08-18 16:36:55 +00:00
|
|
|
logger.info('trying to fetch ' + currency + ' wallet')
|
2024-12-14 14:56:45 +00:00
|
|
|
|
2024-12-16 20:05:31 +00:00
|
|
|
const scopes = await getScopes({ apiKey }, { signal })
|
2024-10-16 17:46:33 +00:00
|
|
|
if (!scopes.includes(SCOPE_READ)) {
|
|
|
|
throw new Error('missing READ scope')
|
|
|
|
}
|
|
|
|
if (!scopes.includes(SCOPE_WRITE)) {
|
|
|
|
throw new Error('missing WRITE scope')
|
|
|
|
}
|
|
|
|
|
|
|
|
currency = currency ? currency.toUpperCase() : 'BTC'
|
2024-12-16 20:05:31 +00:00
|
|
|
await getWallet({ apiKey, currency }, { signal })
|
2024-10-16 17:46:33 +00:00
|
|
|
|
2024-08-18 16:36:55 +00:00
|
|
|
logger.ok(currency + ' wallet found')
|
|
|
|
}
|
|
|
|
|
2024-12-16 20:05:31 +00:00
|
|
|
export async function sendPayment (bolt11, { apiKey, currency }, { signal }) {
|
|
|
|
const wallet = await getWallet({ apiKey, currency }, { signal })
|
|
|
|
return await payInvoice(bolt11, { apiKey, wallet }, { signal })
|
2024-08-18 16:36:55 +00:00
|
|
|
}
|
|
|
|
|
2024-12-16 20:05:31 +00:00
|
|
|
async function payInvoice (bolt11, { apiKey, wallet }, { signal }) {
|
2024-12-14 14:56:45 +00:00
|
|
|
const out = await request({
|
|
|
|
apiKey,
|
|
|
|
query: `
|
|
|
|
mutation LnInvoicePaymentSend($input: LnInvoicePaymentInput!) {
|
2024-08-18 16:36:55 +00:00
|
|
|
lnInvoicePaymentSend(input: $input) {
|
2024-12-14 14:56:45 +00:00
|
|
|
status
|
|
|
|
errors {
|
|
|
|
message
|
|
|
|
path
|
|
|
|
code
|
|
|
|
}
|
|
|
|
transaction {
|
|
|
|
settlementVia {
|
|
|
|
... on SettlementViaIntraLedger {
|
|
|
|
preImage
|
|
|
|
}
|
|
|
|
... on SettlementViaLn {
|
|
|
|
preImage
|
|
|
|
}
|
2024-08-18 16:36:55 +00:00
|
|
|
}
|
2024-12-14 14:56:45 +00:00
|
|
|
}
|
2024-08-18 16:36:55 +00:00
|
|
|
}
|
2024-12-14 14:56:45 +00:00
|
|
|
}`,
|
|
|
|
variables: {
|
|
|
|
input: {
|
|
|
|
paymentRequest: bolt11,
|
|
|
|
walletId: wallet.id
|
|
|
|
}
|
2024-08-18 16:36:55 +00:00
|
|
|
}
|
2024-12-16 20:05:31 +00:00
|
|
|
}, { signal })
|
2024-12-14 14:56:45 +00:00
|
|
|
|
2024-08-18 16:36:55 +00:00
|
|
|
const status = out.data.lnInvoicePaymentSend.status
|
|
|
|
const errors = out.data.lnInvoicePaymentSend.errors
|
|
|
|
if (errors && errors.length > 0) {
|
|
|
|
throw new Error('failed to pay invoice ' + errors.map(e => e.code + ' ' + e.message).join(', '))
|
|
|
|
}
|
|
|
|
|
|
|
|
// payment was settled immediately
|
|
|
|
if (status === 'SUCCESS') {
|
|
|
|
const preimage = out.data.lnInvoicePaymentSend.transaction.settlementVia.preImage
|
|
|
|
if (!preimage) throw new Error('no preimage')
|
|
|
|
return preimage
|
|
|
|
}
|
|
|
|
|
|
|
|
// payment failed immediately
|
|
|
|
if (status === 'FAILED') {
|
|
|
|
throw new Error('failed to pay invoice')
|
|
|
|
}
|
|
|
|
|
|
|
|
// payment couldn't be settled (or fail) immediately, so we wait for a result
|
|
|
|
if (status === 'PENDING') {
|
|
|
|
while (true) {
|
|
|
|
// at some point it should either be settled or fail on the backend, so the loop will exit
|
|
|
|
await new Promise(resolve => setTimeout(resolve, 100))
|
|
|
|
|
2024-12-16 20:05:31 +00:00
|
|
|
const txInfo = await getTxInfo(bolt11, { apiKey, wallet }, { signal })
|
2024-08-18 16:36:55 +00:00
|
|
|
// settled
|
|
|
|
if (txInfo.status === 'SUCCESS') {
|
|
|
|
if (!txInfo.preImage) throw new Error('no preimage')
|
|
|
|
return txInfo.preImage
|
|
|
|
}
|
|
|
|
// failed
|
|
|
|
if (txInfo.status === 'FAILED') {
|
|
|
|
throw new Error(txInfo.error || 'failed to pay invoice')
|
|
|
|
}
|
|
|
|
// still pending
|
|
|
|
// retry later
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// this should never happen
|
|
|
|
throw new Error('unexpected error')
|
|
|
|
}
|
|
|
|
|
2024-12-16 20:05:31 +00:00
|
|
|
async function getTxInfo (bolt11, { apiKey, wallet }, { signal }) {
|
2024-08-18 16:36:55 +00:00
|
|
|
let out
|
|
|
|
try {
|
2024-12-14 14:56:45 +00:00
|
|
|
out = await request({
|
|
|
|
apiKey,
|
|
|
|
query: `
|
|
|
|
query GetTxInfo($walletId: WalletId!, $paymentRequest: LnPaymentRequest!) {
|
|
|
|
me {
|
|
|
|
defaultAccount {
|
|
|
|
walletById(walletId: $walletId) {
|
|
|
|
transactionsByPaymentRequest(paymentRequest: $paymentRequest) {
|
|
|
|
status
|
|
|
|
direction
|
|
|
|
settlementVia {
|
2024-08-18 16:36:55 +00:00
|
|
|
... on SettlementViaIntraLedger {
|
2024-12-14 14:56:45 +00:00
|
|
|
preImage
|
2024-08-18 16:36:55 +00:00
|
|
|
}
|
|
|
|
... on SettlementViaLn {
|
2024-12-14 14:56:45 +00:00
|
|
|
preImage
|
2024-08-18 16:36:55 +00:00
|
|
|
}
|
2024-12-14 14:56:45 +00:00
|
|
|
}
|
2024-08-18 16:36:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-12-14 14:56:45 +00:00
|
|
|
}`,
|
|
|
|
variables: {
|
|
|
|
paymentRequest: bolt11,
|
|
|
|
walletId: wallet.Id
|
2024-08-18 16:36:55 +00:00
|
|
|
}
|
2024-12-16 20:05:31 +00:00
|
|
|
}, { signal })
|
2024-08-18 16:36:55 +00:00
|
|
|
} catch (e) {
|
|
|
|
// something went wrong during the query,
|
|
|
|
// maybe the connection was lost, so we just return
|
|
|
|
// a pending status, the caller can retry later
|
|
|
|
return {
|
|
|
|
status: 'PENDING',
|
|
|
|
preImage: null,
|
|
|
|
error: ''
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const tx = out.data.me.defaultAccount.walletById.transactionsByPaymentRequest.find(t => t.direction === 'SEND')
|
|
|
|
if (!tx) {
|
|
|
|
// the transaction was not found, something went wrong
|
|
|
|
return {
|
|
|
|
status: 'FAILED',
|
|
|
|
preImage: null,
|
|
|
|
error: 'transaction not found'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const status = tx.status
|
|
|
|
const preImage = tx.settlementVia.preImage
|
|
|
|
return {
|
|
|
|
status,
|
|
|
|
preImage,
|
|
|
|
error: ''
|
|
|
|
}
|
|
|
|
}
|