Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
|
|
|
import { AuthenticationError, UserInputError } from 'apollo-server-micro'
|
2021-09-06 22:36:08 +00:00
|
|
|
import { decodeCursor, LIMIT, nextCursorEncoded } from '../../lib/cursor'
|
2022-09-21 19:57:36 +00:00
|
|
|
import { getItem, filterClause } from './item'
|
2022-03-23 18:54:39 +00:00
|
|
|
import { getInvoice } from './wallet'
|
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
|
|
|
import { pushSubscriptionSchema, ssValidate } from '../../lib/validate'
|
|
|
|
import { replyToSubscription } from '../webPush'
|
2021-08-17 18:15:24 +00:00
|
|
|
|
|
|
|
export default {
|
|
|
|
Query: {
|
2022-04-21 17:48:27 +00:00
|
|
|
notifications: async (parent, { cursor, inc }, { me, models }) => {
|
2021-08-17 18:15:24 +00:00
|
|
|
const decodedCursor = decodeCursor(cursor)
|
2021-08-17 23:59:22 +00:00
|
|
|
if (!me) {
|
|
|
|
throw new AuthenticationError('you must be logged in')
|
|
|
|
}
|
2021-08-17 18:15:24 +00:00
|
|
|
|
2022-05-02 01:01:33 +00:00
|
|
|
const meFull = await models.user.findUnique({ where: { id: me.id } })
|
|
|
|
|
2021-08-17 18:15:24 +00:00
|
|
|
/*
|
|
|
|
So that we can cursor over results, we union notifications together ...
|
|
|
|
this requires we have the same number of columns in all results
|
|
|
|
|
|
|
|
select "Item".id, NULL as earnedSats, "Item".created_at as created_at from
|
|
|
|
"Item" JOIN "Item" p ON "Item"."parentId" = p.id AND p."userId" = 622 AND
|
|
|
|
"Item"."userId" <> 622 UNION ALL select "Item".id, "Vote".sats as earnedSats,
|
|
|
|
"Vote".created_at as created_at FROM "Item" LEFT JOIN "Vote" on
|
|
|
|
"Vote"."itemId" = "Item".id AND "Vote"."userId" <> 622 AND "Vote".boost = false
|
|
|
|
WHERE "Item"."userId" = 622 ORDER BY created_at DESC;
|
|
|
|
|
|
|
|
Because we want to "collapse" time adjacent votes in the result
|
|
|
|
|
|
|
|
select vote.id, sum(vote."earnedSats") as "earnedSats", max(vote.voted_at)
|
|
|
|
as "createdAt" from (select "Item".*, "Vote".sats as "earnedSats",
|
|
|
|
"Vote".created_at as voted_at, ROW_NUMBER() OVER(ORDER BY "Vote".created_at) -
|
|
|
|
ROW_NUMBER() OVER(PARTITION BY "Item".id ORDER BY "Vote".created_at) as island
|
|
|
|
FROM "Item" LEFT JOIN "Vote" on "Vote"."itemId" = "Item".id AND
|
|
|
|
"Vote"."userId" <> 622 AND "Vote".boost = false WHERE "Item"."userId" = 622)
|
|
|
|
as vote group by vote.id, vote.island order by max(vote.voted_at) desc;
|
|
|
|
|
|
|
|
We can also "collapse" votes occuring within 1 hour intervals of each other
|
|
|
|
(I haven't yet combined with the above collapsing method .. but might be
|
|
|
|
overkill)
|
|
|
|
|
|
|
|
select "Item".id, sum("Vote".sats) as earnedSats, max("Vote".created_at)
|
|
|
|
as created_at, ROW_NUMBER() OVER(ORDER BY max("Vote".created_at)) - ROW_NUMBER()
|
|
|
|
OVER(PARTITION BY "Item".id ORDER BY max("Vote".created_at)) as island FROM
|
|
|
|
"Item" LEFT JOIN "Vote" on "Vote"."itemId" = "Item".id AND "Vote"."userId" <> 622
|
|
|
|
AND "Vote".boost = false WHERE "Item"."userId" = 622 group by "Item".id,
|
|
|
|
date_trunc('hour', "Vote".created_at) order by created_at desc;
|
|
|
|
|
2021-10-28 22:22:19 +00:00
|
|
|
island approach we used to take
|
2021-08-20 00:13:32 +00:00
|
|
|
(SELECT ${ITEM_SUBQUERY_FIELDS}, max(subquery.voted_at) as "sortTime",
|
2021-09-02 22:22:00 +00:00
|
|
|
sum(subquery.sats) as "earnedSats", false as mention
|
|
|
|
FROM
|
2021-09-08 21:51:23 +00:00
|
|
|
(SELECT ${ITEM_FIELDS}, "ItemAct".created_at as voted_at, "ItemAct".sats,
|
|
|
|
ROW_NUMBER() OVER(ORDER BY "ItemAct".created_at) -
|
|
|
|
ROW_NUMBER() OVER(PARTITION BY "Item".id ORDER BY "ItemAct".created_at) as island
|
|
|
|
FROM "ItemAct"
|
|
|
|
JOIN "Item" on "ItemAct"."itemId" = "Item".id
|
|
|
|
WHERE "ItemAct"."userId" <> $1
|
|
|
|
AND "ItemAct".created_at <= $2
|
|
|
|
AND "ItemAct".act <> 'BOOST'
|
2021-09-02 22:22:00 +00:00
|
|
|
AND "Item"."userId" = $1) subquery
|
2021-10-07 03:20:59 +00:00
|
|
|
GROUP BY ${ITEM_SUBQUERY_FIELDS}, subquery.island
|
|
|
|
ORDER BY max(subquery.voted_at) desc
|
|
|
|
LIMIT ${LIMIT}+$3)
|
2021-10-28 22:22:19 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
// HACK to make notifications faster, we only return a limited sub set of the unioned
|
|
|
|
// queries ... we only ever need at most LIMIT+current offset in the child queries to
|
|
|
|
// have enough items to return in the union
|
2022-04-21 22:50:02 +00:00
|
|
|
|
|
|
|
const queries = []
|
|
|
|
|
2023-06-01 00:44:06 +00:00
|
|
|
queries.push(
|
|
|
|
`(SELECT DISTINCT "Item".id::TEXT, "Item".created_at AS "sortTime", NULL::BIGINT as "earnedSats",
|
|
|
|
'Reply' AS type
|
|
|
|
FROM "Item"
|
|
|
|
JOIN "Item" p ON ${meFull.noteAllDescendants ? '"Item".path <@ p.path' : '"Item"."parentId" = p.id'}
|
|
|
|
WHERE p."userId" = $1 AND "Item"."userId" <> $1 AND "Item".created_at <= $2
|
|
|
|
${await filterClause(me, models)}
|
|
|
|
ORDER BY "sortTime" DESC
|
2023-06-02 21:48:39 +00:00
|
|
|
LIMIT ${LIMIT}+$3)
|
|
|
|
UNION DISTINCT
|
|
|
|
(SELECT DISTINCT "Item".id::TEXT, "Item".created_at AS "sortTime", NULL::BIGINT as "earnedSats",
|
2023-06-01 00:44:06 +00:00
|
|
|
'Reply' AS type
|
|
|
|
FROM "ThreadSubscription"
|
|
|
|
JOIN "Item" p ON "ThreadSubscription"."itemId" = p.id
|
|
|
|
JOIN "Item" ON ${meFull.noteAllDescendants ? '"Item".path <@ p.path' : '"Item"."parentId" = p.id'}
|
|
|
|
WHERE
|
|
|
|
"ThreadSubscription"."userId" = $1
|
|
|
|
AND "Item"."userId" <> $1 AND "Item".created_at <= $2
|
|
|
|
${await filterClause(me, models)}
|
|
|
|
ORDER BY "sortTime" DESC
|
|
|
|
LIMIT ${LIMIT}+$3)`
|
|
|
|
)
|
|
|
|
|
2023-06-02 21:48:39 +00:00
|
|
|
if (meFull.noteMentions) {
|
|
|
|
queries.push(
|
|
|
|
`(SELECT "Item".id::TEXT, "Mention".created_at AS "sortTime", NULL as "earnedSats",
|
|
|
|
'Mention' AS type
|
|
|
|
FROM "Mention"
|
|
|
|
JOIN "Item" ON "Mention"."itemId" = "Item".id
|
|
|
|
LEFT JOIN "Item" p ON "Item"."parentId" = p.id
|
|
|
|
WHERE "Mention"."userId" = $1
|
|
|
|
AND "Mention".created_at <= $2
|
|
|
|
AND "Item"."userId" <> $1
|
|
|
|
AND (p."userId" IS NULL OR p."userId" <> $1)
|
|
|
|
${await filterClause(me, models)}
|
|
|
|
ORDER BY "sortTime" DESC
|
|
|
|
LIMIT ${LIMIT}+$3)`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2023-06-01 00:44:06 +00:00
|
|
|
queries.push(
|
|
|
|
`(SELECT "Item".id::text, "Item"."statusUpdatedAt" AS "sortTime", NULL as "earnedSats",
|
|
|
|
'JobChanged' AS type
|
|
|
|
FROM "Item"
|
|
|
|
WHERE "Item"."userId" = $1
|
|
|
|
AND "maxBid" IS NOT NULL
|
|
|
|
AND "statusUpdatedAt" <= $2 AND "statusUpdatedAt" <> created_at
|
|
|
|
ORDER BY "sortTime" DESC
|
|
|
|
LIMIT ${LIMIT}+$3)`
|
|
|
|
)
|
|
|
|
|
|
|
|
if (meFull.noteItemSats) {
|
2022-04-21 22:50:02 +00:00
|
|
|
queries.push(
|
2023-06-01 00:44:06 +00:00
|
|
|
`(SELECT "Item".id::TEXT, MAX("ItemAct".created_at) AS "sortTime",
|
|
|
|
MAX("Item".msats/1000) as "earnedSats", 'Votification' AS type
|
|
|
|
FROM "Item"
|
|
|
|
JOIN "ItemAct" ON "ItemAct"."itemId" = "Item".id
|
|
|
|
WHERE "ItemAct"."userId" <> $1
|
|
|
|
AND "ItemAct".created_at <= $2
|
|
|
|
AND "ItemAct".act IN ('TIP', 'FEE')
|
|
|
|
AND "Item"."userId" = $1
|
|
|
|
GROUP BY "Item".id
|
|
|
|
ORDER BY "sortTime" DESC
|
|
|
|
LIMIT ${LIMIT}+$3)`
|
2022-04-21 22:50:02 +00:00
|
|
|
)
|
2023-06-01 00:44:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (meFull.noteDeposits) {
|
2022-04-21 22:50:02 +00:00
|
|
|
queries.push(
|
2023-06-01 00:44:06 +00:00
|
|
|
`(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
|
2022-03-22 19:53:48 +00:00
|
|
|
ORDER BY "sortTime" DESC
|
2022-04-21 22:50:02 +00:00
|
|
|
LIMIT ${LIMIT}+$3)`
|
|
|
|
)
|
2023-06-01 00:44:06 +00:00
|
|
|
}
|
2022-04-21 22:50:02 +00:00
|
|
|
|
2023-06-01 00:44:06 +00:00
|
|
|
if (meFull.noteInvites) {
|
|
|
|
queries.push(
|
|
|
|
`(SELECT "Invite".id, MAX(users.created_at) AS "sortTime", NULL as "earnedSats",
|
|
|
|
'Invitification' AS type
|
|
|
|
FROM users JOIN "Invite" on users."inviteId" = "Invite".id
|
|
|
|
WHERE "Invite"."userId" = $1
|
|
|
|
AND users.created_at <= $2
|
|
|
|
GROUP BY "Invite".id
|
|
|
|
ORDER BY "sortTime" DESC
|
|
|
|
LIMIT ${LIMIT}+$3)`
|
|
|
|
)
|
|
|
|
queries.push(
|
|
|
|
`(SELECT users.id::text, users.created_at AS "sortTime", NULL as "earnedSats",
|
|
|
|
'Referral' AS type
|
|
|
|
FROM users
|
|
|
|
WHERE "users"."referrerId" = $1
|
|
|
|
AND "inviteId" IS NULL
|
|
|
|
AND users.created_at <= $2
|
|
|
|
LIMIT ${LIMIT}+$3)`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (meFull.noteEarning) {
|
|
|
|
queries.push(
|
|
|
|
`SELECT min(id)::text, created_at AS "sortTime", FLOOR(sum(msats) / 1000) as "earnedSats",
|
|
|
|
'Earn' AS type
|
|
|
|
FROM "Earn"
|
|
|
|
WHERE "userId" = $1
|
|
|
|
AND created_at <= $2
|
|
|
|
GROUP BY "userId", created_at`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (meFull.noteCowboyHat) {
|
|
|
|
queries.push(
|
|
|
|
`SELECT id::text, updated_at AS "sortTime", 0 as "earnedSats", 'Streak' AS type
|
|
|
|
FROM "Streak"
|
|
|
|
WHERE "userId" = $1
|
|
|
|
AND updated_at <= $2`
|
|
|
|
)
|
2022-04-21 22:50:02 +00:00
|
|
|
}
|
|
|
|
|
2022-07-05 20:18:59 +00:00
|
|
|
// we do all this crazy subquery stuff to make 'reward' islands
|
2022-04-21 22:50:02 +00:00
|
|
|
const notifications = await models.$queryRaw(
|
2022-09-12 18:55:34 +00:00
|
|
|
`SELECT MAX(id) AS id, MAX("sortTime") AS "sortTime", sum("earnedSats") AS "earnedSats", type,
|
|
|
|
MIN("sortTime") AS "minSortTime"
|
2022-07-05 20:18:59 +00:00
|
|
|
FROM
|
|
|
|
(SELECT *,
|
|
|
|
CASE
|
|
|
|
WHEN type = 'Earn' THEN
|
|
|
|
ROW_NUMBER() OVER(ORDER BY "sortTime" DESC) -
|
|
|
|
ROW_NUMBER() OVER(PARTITION BY type = 'Earn' ORDER BY "sortTime" DESC)
|
|
|
|
ELSE
|
|
|
|
ROW_NUMBER() OVER(ORDER BY "sortTime" DESC)
|
|
|
|
END as island
|
|
|
|
FROM
|
|
|
|
(${queries.join(' UNION ALL ')}) u
|
|
|
|
) sub
|
|
|
|
GROUP BY type, island
|
2021-10-07 03:20:59 +00:00
|
|
|
ORDER BY "sortTime" DESC
|
|
|
|
OFFSET $3
|
|
|
|
LIMIT ${LIMIT}`, me.id, decodedCursor.time, decodedCursor.offset)
|
2021-08-17 18:15:24 +00:00
|
|
|
|
2021-09-06 22:36:08 +00:00
|
|
|
if (decodedCursor.offset === 0) {
|
|
|
|
await models.user.update({ where: { id: me.id }, data: { checkedNotesAt: new Date() } })
|
|
|
|
}
|
2021-08-17 23:59:22 +00:00
|
|
|
|
2021-08-17 18:15:24 +00:00
|
|
|
return {
|
2022-05-02 01:01:33 +00:00
|
|
|
lastChecked: meFull.checkedNotesAt,
|
2021-08-17 18:15:24 +00:00
|
|
|
cursor: notifications.length === LIMIT ? nextCursorEncoded(decodedCursor) : null,
|
|
|
|
notifications
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
|
|
|
Mutation: {
|
|
|
|
savePushSubscription: async (parent, { endpoint, p256dh, auth, oldEndpoint }, { me, models }) => {
|
|
|
|
if (!me) {
|
|
|
|
throw new AuthenticationError('you must be logged in')
|
|
|
|
}
|
|
|
|
|
|
|
|
await ssValidate(pushSubscriptionSchema, { endpoint, p256dh, auth })
|
|
|
|
|
|
|
|
let dbPushSubscription
|
|
|
|
if (oldEndpoint) {
|
|
|
|
dbPushSubscription = await models.pushSubscription.update({
|
|
|
|
data: { userId: me.id, endpoint, p256dh, auth }, where: { endpoint: oldEndpoint }
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
dbPushSubscription = await models.pushSubscription.create({
|
|
|
|
data: { userId: me.id, endpoint, p256dh, auth }
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
await replyToSubscription(dbPushSubscription.id, { title: 'Stacker News notifications are now active' })
|
|
|
|
|
|
|
|
return dbPushSubscription
|
|
|
|
},
|
|
|
|
deletePushSubscription: async (parent, { endpoint }, { me, models }) => {
|
|
|
|
if (!me) {
|
|
|
|
throw new AuthenticationError('you must be logged in')
|
|
|
|
}
|
|
|
|
|
|
|
|
const subscription = await models.pushSubscription.findFirst({ where: { endpoint, userId: Number(me.id) } })
|
|
|
|
if (!subscription) {
|
|
|
|
throw new UserInputError('endpoint not found', {
|
|
|
|
argumentName: 'endpoint'
|
|
|
|
})
|
|
|
|
}
|
|
|
|
await models.pushSubscription.delete({ where: { id: subscription.id } })
|
|
|
|
return subscription
|
|
|
|
}
|
|
|
|
},
|
2021-08-17 18:15:24 +00:00
|
|
|
Notification: {
|
2022-01-19 21:02:38 +00:00
|
|
|
__resolveType: async (n, args, { models }) => n.type
|
|
|
|
},
|
|
|
|
Votification: {
|
2023-05-07 20:21:58 +00:00
|
|
|
item: async (n, args, { models, me }) => getItem(n, { id: n.id }, { models, me })
|
2022-01-19 21:02:38 +00:00
|
|
|
},
|
|
|
|
Reply: {
|
2023-05-07 20:21:58 +00:00
|
|
|
item: async (n, args, { models, me }) => getItem(n, { id: n.id }, { models, me })
|
2022-01-19 21:02:38 +00:00
|
|
|
},
|
2022-02-28 20:09:21 +00:00
|
|
|
JobChanged: {
|
2023-05-07 20:21:58 +00:00
|
|
|
item: async (n, args, { models, me }) => getItem(n, { id: n.id }, { models, me })
|
2022-02-28 20:09:21 +00:00
|
|
|
},
|
2023-02-01 14:44:35 +00:00
|
|
|
Streak: {
|
|
|
|
days: async (n, args, { models }) => {
|
|
|
|
const res = await models.$queryRaw`
|
|
|
|
SELECT "endedAt" - "startedAt" AS days
|
|
|
|
FROM "Streak"
|
|
|
|
WHERE id = ${Number(n.id)} AND "endedAt" IS NOT NULL
|
|
|
|
`
|
|
|
|
|
|
|
|
return res.length ? res[0].days : null
|
|
|
|
}
|
|
|
|
},
|
2022-09-12 18:55:34 +00:00
|
|
|
Earn: {
|
|
|
|
sources: async (n, args, { me, models }) => {
|
|
|
|
const [sources] = await models.$queryRaw(`
|
|
|
|
SELECT
|
|
|
|
FLOOR(sum(msats) FILTER(WHERE type = 'POST') / 1000) AS posts,
|
|
|
|
FLOOR(sum(msats) FILTER(WHERE type = 'COMMENT') / 1000) AS comments,
|
2023-07-09 17:21:11 +00:00
|
|
|
FLOOR(sum(msats) FILTER(WHERE type = 'TIP_POST') / 1000) AS "tipPosts",
|
|
|
|
FLOOR(sum(msats) FILTER(WHERE type = 'TIP_COMMENT') / 1000) AS "tipComments"
|
2022-09-12 18:55:34 +00:00
|
|
|
FROM "Earn"
|
|
|
|
WHERE "userId" = $1 AND created_at <= $2 AND created_at >= $3
|
|
|
|
`, Number(me.id), new Date(n.sortTime), new Date(n.minSortTime))
|
|
|
|
sources.posts ||= 0
|
|
|
|
sources.comments ||= 0
|
2023-07-09 17:21:11 +00:00
|
|
|
sources.tipPosts ||= 0
|
|
|
|
sources.tipComments ||= 0
|
|
|
|
if (sources.posts + sources.comments + sources.tipPosts + sources.tipComments > 0) {
|
2022-09-12 18:55:34 +00:00
|
|
|
return sources
|
|
|
|
}
|
|
|
|
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
},
|
2022-01-19 21:02:38 +00:00
|
|
|
Mention: {
|
|
|
|
mention: async (n, args, { models }) => true,
|
2023-05-07 20:21:58 +00:00
|
|
|
item: async (n, args, { models, me }) => getItem(n, { id: n.id }, { models, me })
|
2022-01-19 21:02:38 +00:00
|
|
|
},
|
2022-03-23 18:54:39 +00:00
|
|
|
InvoicePaid: {
|
|
|
|
invoice: async (n, args, { me, models }) => getInvoice(n, { id: n.id }, { me, models })
|
|
|
|
},
|
2022-01-19 21:02:38 +00:00
|
|
|
Invitification: {
|
|
|
|
invite: async (n, args, { models }) => {
|
|
|
|
return await models.invite.findUnique({
|
|
|
|
where: {
|
|
|
|
id: n.id
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2021-08-17 18:15:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-28 22:22:19 +00:00
|
|
|
// const ITEM_SUBQUERY_FIELDS =
|
|
|
|
// `subquery.id, subquery."createdAt", subquery."updatedAt", subquery.title, subquery.text,
|
|
|
|
// subquery.url, subquery."userId", subquery."parentId", subquery.path`
|
|
|
|
|
2022-03-15 16:30:11 +00:00
|
|
|
// const ITEM_GROUP_FIELDS =
|
|
|
|
// `"Item".id, "Item".created_at, "Item".updated_at, "Item".title,
|
|
|
|
// "Item".text, "Item".url, "Item"."userId", "Item"."parentId", ltree2text("Item"."path")`
|
2021-08-17 18:15:24 +00:00
|
|
|
|
2022-01-19 21:02:38 +00:00
|
|
|
// const ITEM_FIELDS =
|
|
|
|
// `"Item".id, "Item".created_at as "createdAt", "Item".updated_at as "updatedAt", "Item".title,
|
|
|
|
// "Item".text, "Item".url, "Item"."userId", "Item"."parentId", ltree2text("Item"."path") AS path`
|