103 lines
3.0 KiB
JavaScript
103 lines
3.0 KiB
JavaScript
import { BALANCE_LIMIT_MSATS, PAID_ACTION_TERMINAL_STATES, USER_ID, SN_ADMIN_IDS } from '@/lib/constants'
|
|
import { msatsToSats, numWithUnits } from '@/lib/format'
|
|
import { datePivot } from '@/lib/time'
|
|
|
|
const MAX_PENDING_PAID_ACTIONS_PER_USER = 100
|
|
const MAX_PENDING_DIRECT_INVOICES_PER_USER_MINUTES = 10
|
|
const MAX_PENDING_DIRECT_INVOICES_PER_USER = 100
|
|
const USER_IDS_BALANCE_NO_LIMIT = [...SN_ADMIN_IDS, USER_ID.anon, USER_ID.ad]
|
|
|
|
export async function assertBelowMaxPendingInvoices (context) {
|
|
const { models, me } = context
|
|
const pendingInvoices = await models.invoice.count({
|
|
where: {
|
|
userId: me?.id ?? USER_ID.anon,
|
|
actionState: {
|
|
notIn: PAID_ACTION_TERMINAL_STATES
|
|
}
|
|
}
|
|
})
|
|
|
|
if (pendingInvoices >= MAX_PENDING_PAID_ACTIONS_PER_USER) {
|
|
throw new Error('You have too many pending paid actions, cancel some or wait for them to expire')
|
|
}
|
|
}
|
|
|
|
export async function assertBelowMaxPendingDirectPayments (userId, context) {
|
|
const { models, me } = context
|
|
|
|
if (me?.id !== userId) {
|
|
const pendingSenderInvoices = await models.directPayment.count({
|
|
where: {
|
|
senderId: me?.id ?? USER_ID.anon,
|
|
createdAt: {
|
|
gt: datePivot(new Date(), { minutes: -MAX_PENDING_DIRECT_INVOICES_PER_USER_MINUTES })
|
|
}
|
|
}
|
|
})
|
|
|
|
if (pendingSenderInvoices >= MAX_PENDING_DIRECT_INVOICES_PER_USER) {
|
|
throw new Error('You\'ve sent too many direct payments')
|
|
}
|
|
}
|
|
|
|
if (!userId) return
|
|
|
|
const pendingReceiverInvoices = await models.directPayment.count({
|
|
where: {
|
|
receiverId: userId,
|
|
createdAt: {
|
|
gt: datePivot(new Date(), { minutes: -MAX_PENDING_DIRECT_INVOICES_PER_USER_MINUTES })
|
|
}
|
|
}
|
|
})
|
|
|
|
if (pendingReceiverInvoices >= MAX_PENDING_DIRECT_INVOICES_PER_USER) {
|
|
throw new Error('Receiver has too many direct payments')
|
|
}
|
|
}
|
|
|
|
export async function assertBelowBalanceLimit (context) {
|
|
const { me, tx } = context
|
|
if (!me || USER_IDS_BALANCE_NO_LIMIT.includes(me.id)) return
|
|
|
|
// we need to prevent this invoice (and any other pending invoices and withdrawls)
|
|
// from causing the user's balance to exceed the balance limit
|
|
const pendingInvoices = await tx.invoice.aggregate({
|
|
where: {
|
|
userId: me.id,
|
|
// p2p invoices are never in state PENDING
|
|
actionState: 'PENDING',
|
|
actionType: 'RECEIVE'
|
|
},
|
|
_sum: {
|
|
msatsRequested: true
|
|
}
|
|
})
|
|
|
|
// Get pending withdrawals total
|
|
const pendingWithdrawals = await tx.withdrawl.aggregate({
|
|
where: {
|
|
userId: me.id,
|
|
status: null
|
|
},
|
|
_sum: {
|
|
msatsPaying: true,
|
|
msatsFeePaying: true
|
|
}
|
|
})
|
|
|
|
// Calculate total pending amount
|
|
const pendingMsats = (pendingInvoices._sum.msatsRequested ?? 0n) +
|
|
((pendingWithdrawals._sum.msatsPaying ?? 0n) + (pendingWithdrawals._sum.msatsFeePaying ?? 0n))
|
|
|
|
// Check balance limit
|
|
if (pendingMsats + me.msats > BALANCE_LIMIT_MSATS) {
|
|
throw new Error(
|
|
`pending invoices and withdrawals must not cause balance to exceed ${
|
|
numWithUnits(msatsToSats(BALANCE_LIMIT_MSATS))
|
|
}`
|
|
)
|
|
}
|
|
}
|