notify user when invoice is paid

This commit is contained in:
keyan 2022-03-23 13:54:39 -05:00
parent 69155139e6
commit 987a5ed3a3
7 changed files with 104 additions and 53 deletions

View File

@ -1,6 +1,7 @@
import { AuthenticationError } from 'apollo-server-micro'
import { decodeCursor, LIMIT, nextCursorEncoded } from '../../lib/cursor'
import { getItem } from './item'
import { getInvoice } from './wallet'
export default {
Query: {
@ -70,6 +71,7 @@ export default {
// at most 25 ancesestors belonging to the same user for a given reply, see: (LIMIT+OFFSET)*25 which is
// undoubtably the case today ... this probably won't hold indefinitely though
// One other less HACKy way to do this is to store in each reply, an set of users it's in response to
// or to simply denormalize the replies (simply the ids which we wouldn't have to be concerned about consitency)
const notifications = await models.$queryRaw(`
SELECT DISTINCT *
FROM
@ -128,8 +130,17 @@ export default {
(SELECT "Earn".id::text, "Earn".created_at AS "sortTime", FLOOR(msats / 1000) as "earnedSats",
'Earn' AS type
FROM "Earn"
WHERE "Earn"."userId" = $1 AND
FLOOR(msats / 1000) > 0
WHERE "Earn"."userId" = $1
AND FLOOR(msats / 1000) > 0
AND created_at <= $2
ORDER BY "sortTime" DESC
LIMIT ${LIMIT}+$3)
UNION ALL
(SELECT "Invoice".id::text, "Invoice"."confirmedAt" AS "sortTime", FLOOR("msatsReceived" / 1000) as "earnedSats",
'InvoicePaid' AS type
FROM "Invoice"
WHERE "Invoice"."userId" = $1
AND "confirmedAt" IS NOT NULL
AND created_at <= $2
ORDER BY "sortTime" DESC
LIMIT ${LIMIT}+$3)) AS n
@ -165,6 +176,9 @@ export default {
mention: async (n, args, { models }) => true,
item: async (n, args, { models }) => getItem(n, { id: n.id }, { models })
},
InvoicePaid: {
invoice: async (n, args, { me, models }) => getInvoice(n, { id: n.id }, { me, models })
},
Invitification: {
invite: async (n, args, { models }) => {
return await models.invite.findUnique({

View File

@ -260,6 +260,18 @@ export default {
return true
}
const invoice = await models.invoice.findFirst({
where: {
userId: user.id,
confirmedAt: {
gt: user.checkedNotesAt || new Date(0)
}
}
})
if (invoice) {
return true
}
// check if new invites have been redeemed
const newInvitees = await models.$queryRaw(`
SELECT "Invite".id

View File

@ -5,9 +5,7 @@ import { decodeCursor, LIMIT, nextCursorEncoded } from '../../lib/cursor'
import lnpr from 'bolt11'
import { SELECT } from './item'
export default {
Query: {
invoice: async (parent, { id }, { me, models, lnd }) => {
export async function getInvoice (parent, { id }, { me, models }) {
if (!me) {
throw new AuthenticationError('you must be logged in')
}
@ -26,7 +24,11 @@ export default {
}
return inv
},
}
export default {
Query: {
invoice: getInvoice,
withdrawl: async (parent, { id }, { me, models, lnd }) => {
if (!me) {
throw new AuthenticationError('you must be logged in')

View File

@ -37,8 +37,14 @@ export default gql`
sortTime: String!
}
type InvoicePaid {
earnedSats: Int!
invoice: Invoice!
sortTime: String!
}
union Notification = Reply | Votification | Mention
| Invitification | JobChanged | Earn
| Invitification | JobChanged | Earn | InvoicePaid
type Notifications {
lastChecked: String

View File

@ -7,7 +7,9 @@ import MoreFooter from './more-footer'
import Invite from './invite'
import { ignoreClick } from '../lib/clicks'
import Link from 'next/link'
import Check from '../svgs/check-double-line.svg'
// TODO: oh man, this is a mess ... each notification type should just be a component ...
function Notification ({ n }) {
const router = useRouter()
return (
@ -22,7 +24,9 @@ function Notification ({ n }) {
return
}
if (n.__typename === 'Invitification') {
if (n.__typename === 'InvoicePaid') {
router.push(`/invoices/${n.invoice.id}`)
} else if (n.__typename === 'Invitification') {
router.push('/invites')
} else if (!n.item.title) {
router.push({
@ -64,6 +68,8 @@ function Notification ({ n }) {
</div>
</>
)
: n.__typename === 'InvoicePaid'
? <div className='font-weight-bold text-info ml-2 py-1'><Check className='fill-info mr-1' />{n.earnedSats} sats were deposited in your account</div>
: (
<>
{n.__typename === 'Votification' &&

View File

@ -51,6 +51,13 @@ export const NOTIFICATIONS = gql`
sortTime
earnedSats
}
... on InvoicePaid {
sortTime
earnedSats
invoice {
id
}
}
}
}
} `

View File

@ -380,11 +380,15 @@ textarea.form-control {
}
.fill-success {
fill: #5c8001;
fill: var(--success);
}
.fill-info {
fill: var(--info);
}
.fill-danger {
fill: #c03221;
fill: var(--danger);
}
.fill-theme-color {