normalized wallet logs (#1826)
* Add invoiceId, withdrawalId to wallet logs * Truncate wallet logs * Fix extra db dips per log line * Fix leak of invoice for sender
This commit is contained in:
parent
e7eece744f
commit
b54268a88f
@ -430,6 +430,10 @@ const resolvers = {
|
|||||||
lte: to ? new Date(Number(to)) : undefined
|
lte: to ? new Date(Number(to)) : undefined
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
include: {
|
||||||
|
invoice: true,
|
||||||
|
withdrawal: true
|
||||||
|
},
|
||||||
orderBy: [
|
orderBy: [
|
||||||
{ createdAt: 'desc' },
|
{ createdAt: 'desc' },
|
||||||
{ id: 'desc' }
|
{ id: 'desc' }
|
||||||
@ -445,6 +449,10 @@ const resolvers = {
|
|||||||
lte: decodedCursor.time
|
lte: decodedCursor.time
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
include: {
|
||||||
|
invoice: true,
|
||||||
|
withdrawal: true
|
||||||
|
},
|
||||||
orderBy: [
|
orderBy: [
|
||||||
{ createdAt: 'desc' },
|
{ createdAt: 'desc' },
|
||||||
{ id: 'desc' }
|
{ id: 'desc' }
|
||||||
@ -745,11 +753,42 @@ const resolvers = {
|
|||||||
return item
|
return item
|
||||||
},
|
},
|
||||||
sats: fact => msatsToSatsDecimal(fact.msats)
|
sats: fact => msatsToSatsDecimal(fact.msats)
|
||||||
|
},
|
||||||
|
|
||||||
|
WalletLogEntry: {
|
||||||
|
context: async ({ level, context, invoice, withdrawal }, args, { models }) => {
|
||||||
|
const isError = ['error', 'warn'].includes(level.toLowerCase())
|
||||||
|
|
||||||
|
if (withdrawal) {
|
||||||
|
return {
|
||||||
|
...await logContextFromBolt11(withdrawal.bolt11),
|
||||||
|
...(withdrawal.preimage ? { preimage: withdrawal.preimage } : {}),
|
||||||
|
...(isError ? { max_fee: formatMsats(withdrawal.msatsFeePaying) } : {})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// XXX never return invoice as context because it might leak sensitive sender details
|
||||||
|
// if (invoice) { ... }
|
||||||
|
|
||||||
|
return context
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default injectResolvers(resolvers)
|
export default injectResolvers(resolvers)
|
||||||
|
|
||||||
|
const logContextFromBolt11 = async (bolt11) => {
|
||||||
|
const decoded = await parsePaymentRequest({ request: bolt11 })
|
||||||
|
return {
|
||||||
|
bolt11,
|
||||||
|
amount: formatMsats(decoded.mtokens),
|
||||||
|
payment_hash: decoded.id,
|
||||||
|
created_at: decoded.created_at,
|
||||||
|
expires_at: decoded.expires_at,
|
||||||
|
description: decoded.description
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export const walletLogger = ({ wallet, models }) => {
|
export const walletLogger = ({ wallet, models }) => {
|
||||||
// no-op logger if wallet is not provided
|
// no-op logger if wallet is not provided
|
||||||
if (!wallet) {
|
if (!wallet) {
|
||||||
@ -762,23 +801,17 @@ export const walletLogger = ({ wallet, models }) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// server implementation of wallet logger interface on client
|
// server implementation of wallet logger interface on client
|
||||||
const log = (level) => async (message, context = {}) => {
|
const log = (level) => async (message, ctx = {}) => {
|
||||||
try {
|
try {
|
||||||
if (context?.bolt11) {
|
let { invoiceId, withdrawalId, ...context } = ctx
|
||||||
|
|
||||||
|
if (context.bolt11) {
|
||||||
// automatically populate context from bolt11 to avoid duplicating this code
|
// automatically populate context from bolt11 to avoid duplicating this code
|
||||||
const decoded = await parsePaymentRequest({ request: context.bolt11 })
|
|
||||||
context = {
|
context = {
|
||||||
...context,
|
...context,
|
||||||
amount: formatMsats(decoded.mtokens),
|
...await logContextFromBolt11(context.bolt11)
|
||||||
payment_hash: decoded.id,
|
|
||||||
created_at: decoded.created_at,
|
|
||||||
expires_at: decoded.expires_at,
|
|
||||||
description: decoded.description,
|
|
||||||
// payments should affect wallet status
|
|
||||||
status: true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
context.recv = true
|
|
||||||
|
|
||||||
await models.walletLog.create({
|
await models.walletLog.create({
|
||||||
data: {
|
data: {
|
||||||
@ -786,7 +819,9 @@ export const walletLogger = ({ wallet, models }) => {
|
|||||||
wallet: wallet.type,
|
wallet: wallet.type,
|
||||||
level,
|
level,
|
||||||
message,
|
message,
|
||||||
context
|
context,
|
||||||
|
invoiceId,
|
||||||
|
withdrawalId
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
@ -239,7 +239,13 @@ export function useWalletLogs (wallet, initialPage = 1, logsPerPage = 10) {
|
|||||||
const newLogs = data.walletLogs.entries.map(({ createdAt, wallet: walletType, ...log }) => ({
|
const newLogs = data.walletLogs.entries.map(({ createdAt, wallet: walletType, ...log }) => ({
|
||||||
ts: +new Date(createdAt),
|
ts: +new Date(createdAt),
|
||||||
wallet: walletTag(getWalletByType(walletType)),
|
wallet: walletTag(getWalletByType(walletType)),
|
||||||
...log
|
...log,
|
||||||
|
// required to resolve recv status
|
||||||
|
context: {
|
||||||
|
recv: true,
|
||||||
|
status: !!log.context?.bolt11 && ['warn', 'error', 'success'].includes(log.level.toLowerCase()),
|
||||||
|
...log.context
|
||||||
|
}
|
||||||
}))
|
}))
|
||||||
const combinedLogs = uniqueSort([...result.data, ...newLogs])
|
const combinedLogs = uniqueSort([...result.data, ...newLogs])
|
||||||
|
|
||||||
|
@ -0,0 +1,8 @@
|
|||||||
|
ALTER TABLE "WalletLog"
|
||||||
|
ADD COLUMN "invoiceId" INTEGER,
|
||||||
|
ADD COLUMN "withdrawalId" INTEGER;
|
||||||
|
|
||||||
|
ALTER TABLE "WalletLog" ADD CONSTRAINT "WalletLog_invoiceId_fkey" FOREIGN KEY ("invoiceId") REFERENCES "Invoice"("id") ON DELETE SET NULL ON UPDATE CASCADE;
|
||||||
|
ALTER TABLE "WalletLog" ADD CONSTRAINT "WalletLog_withdrawalId_fkey" FOREIGN KEY ("withdrawalId") REFERENCES "Withdrawl"("id") ON DELETE SET NULL ON UPDATE CASCADE;
|
||||||
|
|
||||||
|
TRUNCATE "WalletLog" RESTRICT;
|
@ -271,6 +271,10 @@ model WalletLog {
|
|||||||
wallet WalletType
|
wallet WalletType
|
||||||
level LogLevel
|
level LogLevel
|
||||||
message String
|
message String
|
||||||
|
invoiceId Int?
|
||||||
|
invoice Invoice? @relation(fields: [invoiceId], references: [id])
|
||||||
|
withdrawalId Int?
|
||||||
|
withdrawal Withdrawl? @relation(fields: [withdrawalId], references: [id])
|
||||||
context Json? @db.JsonB
|
context Json? @db.JsonB
|
||||||
|
|
||||||
@@index([userId, createdAt])
|
@@index([userId, createdAt])
|
||||||
@ -971,6 +975,7 @@ model Invoice {
|
|||||||
Upload Upload[]
|
Upload Upload[]
|
||||||
PollVote PollVote[]
|
PollVote PollVote[]
|
||||||
PollBlindVote PollBlindVote[]
|
PollBlindVote PollBlindVote[]
|
||||||
|
WalletLog WalletLog[]
|
||||||
|
|
||||||
@@index([createdAt], map: "Invoice.created_at_index")
|
@@index([createdAt], map: "Invoice.created_at_index")
|
||||||
@@index([userId], map: "Invoice.userId_index")
|
@@index([userId], map: "Invoice.userId_index")
|
||||||
@ -1048,6 +1053,7 @@ model Withdrawl {
|
|||||||
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
||||||
wallet Wallet? @relation(fields: [walletId], references: [id], onDelete: SetNull)
|
wallet Wallet? @relation(fields: [walletId], references: [id], onDelete: SetNull)
|
||||||
invoiceForward InvoiceForward?
|
invoiceForward InvoiceForward?
|
||||||
|
WalletLog WalletLog[]
|
||||||
|
|
||||||
@@index([createdAt], map: "Withdrawl.created_at_index")
|
@@index([createdAt], map: "Withdrawl.created_at_index")
|
||||||
@@index([userId], map: "Withdrawl.userId_index")
|
@@index([userId], map: "Withdrawl.userId_index")
|
||||||
|
@ -39,8 +39,7 @@ export async function * createUserInvoice (userId, { msats, description, descrip
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
logger.info(
|
logger.info(
|
||||||
`↙ incoming payment: ${formatSats(msatsToSats(msats))}`,
|
`↙ incoming payment: ${formatSats(msatsToSats(msats))}`, {
|
||||||
{
|
|
||||||
amount: formatMsats(msats)
|
amount: formatMsats(msats)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -2,7 +2,7 @@ import { getPaymentFailureStatus, hodlInvoiceCltvDetails, getPaymentOrNotSent }
|
|||||||
import { paidActions } from '@/api/paidAction'
|
import { paidActions } from '@/api/paidAction'
|
||||||
import { walletLogger } from '@/api/resolvers/wallet'
|
import { walletLogger } from '@/api/resolvers/wallet'
|
||||||
import { LND_PATHFINDING_TIME_PREF_PPM, LND_PATHFINDING_TIMEOUT_MS, PAID_ACTION_TERMINAL_STATES } from '@/lib/constants'
|
import { LND_PATHFINDING_TIME_PREF_PPM, LND_PATHFINDING_TIMEOUT_MS, PAID_ACTION_TERMINAL_STATES } from '@/lib/constants'
|
||||||
import { formatMsats, formatSats, msatsToSats, toPositiveNumber } from '@/lib/format'
|
import { formatSats, msatsToSats, toPositiveNumber } from '@/lib/format'
|
||||||
import { datePivot } from '@/lib/time'
|
import { datePivot } from '@/lib/time'
|
||||||
import { Prisma } from '@prisma/client'
|
import { Prisma } from '@prisma/client'
|
||||||
import {
|
import {
|
||||||
@ -317,17 +317,13 @@ export async function paidActionForwarded ({ data: { invoiceId, withdrawal, ...a
|
|||||||
}, { models, lnd, boss })
|
}, { models, lnd, boss })
|
||||||
|
|
||||||
if (transitionedInvoice) {
|
if (transitionedInvoice) {
|
||||||
const { bolt11, msatsPaid } = transitionedInvoice.invoiceForward.withdrawl
|
const withdrawal = transitionedInvoice.invoiceForward.withdrawl
|
||||||
|
|
||||||
const logger = walletLogger({ wallet: transitionedInvoice.invoiceForward.wallet, models })
|
const logger = walletLogger({ wallet: transitionedInvoice.invoiceForward.wallet, models })
|
||||||
logger.ok(
|
logger.ok(
|
||||||
`↙ payment received: ${formatSats(msatsToSats(Number(msatsPaid)))}`,
|
`↙ payment received: ${formatSats(msatsToSats(Number(withdrawal.msatsPaid)))}`, {
|
||||||
{
|
invoiceId: transitionedInvoice.id,
|
||||||
bolt11,
|
withdrawalId: withdrawal.id
|
||||||
preimage: transitionedInvoice.preimage
|
|
||||||
// we could show the outgoing fee that we paid from the incoming amount to the receiver
|
|
||||||
// but we don't since it might look like the receiver paid the fee but that's not the case.
|
|
||||||
// fee: formatMsats(msatsFeePaid)
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -376,12 +372,11 @@ export async function paidActionFailedForward ({ data: { invoiceId, withdrawal:
|
|||||||
}, { models, lnd, boss })
|
}, { models, lnd, boss })
|
||||||
|
|
||||||
if (transitionedInvoice) {
|
if (transitionedInvoice) {
|
||||||
const { bolt11, msatsFeePaying } = transitionedInvoice.invoiceForward.withdrawl
|
const fwd = transitionedInvoice.invoiceForward
|
||||||
const logger = walletLogger({ wallet: transitionedInvoice.invoiceForward.wallet, models })
|
const logger = walletLogger({ wallet: fwd.wallet, models })
|
||||||
logger.warn(
|
logger.warn(
|
||||||
`incoming payment failed: ${message}`, {
|
`incoming payment failed: ${message}`, {
|
||||||
bolt11,
|
withdrawalId: fwd.withdrawl.id
|
||||||
max_fee: formatMsats(msatsFeePaying)
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -446,7 +441,11 @@ export async function paidActionCanceling ({ data: { invoiceId, ...args }, model
|
|||||||
const { wallet, bolt11 } = transitionedInvoice.invoiceForward
|
const { wallet, bolt11 } = transitionedInvoice.invoiceForward
|
||||||
const logger = walletLogger({ wallet, models })
|
const logger = walletLogger({ wallet, models })
|
||||||
const decoded = await parsePaymentRequest({ request: bolt11 })
|
const decoded = await parsePaymentRequest({ request: bolt11 })
|
||||||
logger.info(`invoice for ${formatSats(msatsToSats(decoded.mtokens))} canceled by payer`, { bolt11 })
|
logger.info(
|
||||||
|
`invoice for ${formatSats(msatsToSats(decoded.mtokens))} canceled by payer`, {
|
||||||
|
bolt11,
|
||||||
|
invoiceId: transitionedInvoice.id
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -125,11 +125,8 @@ export async function payingActionConfirmed ({ data: args, models, lnd, boss })
|
|||||||
|
|
||||||
const logger = walletLogger({ models, wallet: transitionedWithdrawal.wallet })
|
const logger = walletLogger({ models, wallet: transitionedWithdrawal.wallet })
|
||||||
logger?.ok(
|
logger?.ok(
|
||||||
`↙ payment received: ${formatSats(msatsToSats(transitionedWithdrawal.msatsPaid))}`,
|
`↙ payment received: ${formatSats(msatsToSats(transitionedWithdrawal.msatsPaid))}`, {
|
||||||
{
|
withdrawalId: transitionedWithdrawal.id
|
||||||
bolt11: transitionedWithdrawal.bolt11,
|
|
||||||
preimage: transitionedWithdrawal.preimage,
|
|
||||||
fee: formatMsats(transitionedWithdrawal.msatsFeePaid)
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user