stacker.news/worker/earn.js

190 lines
7.2 KiB
JavaScript

import { notifyEarner } from '@/lib/webPush'
import createPrisma from '@/lib/create-prisma'
import { SN_NO_REWARDS_IDS } from '@/lib/constants'
const TOTAL_UPPER_BOUND_MSATS = 1_000_000_000
export async function earn ({ name }) {
// grab a greedy connection
const models = createPrisma({ connectionParams: { connection_limit: 1 } })
try {
// compute how much sn earned yesterday
const [{ sum: sumDecimal }] = await models.$queryRaw`
SELECT sum(total) as sum
FROM rewards(
date_trunc('day', now() AT TIME ZONE 'America/Chicago' - interval '1 day'),
date_trunc('day', now() AT TIME ZONE 'America/Chicago' - interval '1 day'), '1 day'::INTERVAL, 'day')`
// XXX primsa will return a Decimal (https://mikemcl.github.io/decimal.js)
// because sum of a BIGINT returns a NUMERIC type (https://www.postgresql.org/docs/13/functions-aggregate.html)
// and Decimal is what prisma maps it to
// https://www.prisma.io/docs/concepts/components/prisma-client/raw-database-access#raw-query-type-mapping
// so check it before coercing to Number
if (!sumDecimal || sumDecimal.lessThanOrEqualTo(0)) {
console.log('done', name, 'no sats to award today')
return
}
// extra sanity check on rewards ... if it's more than upper bound, we
// probably have a bug somewhere or we've grown A LOT
if (sumDecimal.greaterThan(TOTAL_UPPER_BOUND_MSATS)) {
console.log('done', name, 'error: too many sats to award today', sumDecimal)
return
}
const sum = Number(sumDecimal)
console.log(name, 'giving away', sum, 'msats', 'rewarding all')
/*
How earnings (used to) work:
1/3: top 50% posts over last 36 hours, scored on a relative basis
1/3: top 50% comments over last 36 hours, scored on a relative basis
1/3: top upvoters of top posts/comments, scored on:
- their trust
- how much they tipped
- how early they upvoted it
- how the post/comment scored
Now: 80% of earnings go to top stackers by relative value, and 10% each to their forever and one day referrers
*/
// get earners { userId, id, type, rank, proportion, foreverReferrerId, oneDayReferrerId }
// has to earn at least 125000 msats to be eligible (so that they get at least 1 sat after referrals)
const earners = await models.$queryRaw`
WITH earners AS (
SELECT users.id AS "userId", users."referrerId" AS "foreverReferrerId",
proportion, (ROW_NUMBER() OVER (ORDER BY proportion DESC))::INTEGER AS rank
FROM user_values(
date_trunc('day', now() AT TIME ZONE 'America/Chicago' - interval '1 day'),
date_trunc('day', now() AT TIME ZONE 'America/Chicago' - interval '1 day'),
'1 day'::INTERVAL,
'day') uv
JOIN users ON users.id = uv.id
WHERE NOT (users.id = ANY (${SN_NO_REWARDS_IDS}))
AND uv.proportion >= 0.0000125
ORDER BY proportion DESC
)
SELECT earners.*,
COALESCE(
mode() WITHIN GROUP (ORDER BY "OneDayReferral"."referrerId"),
earners."foreverReferrerId") AS "oneDayReferrerId"
FROM earners
LEFT JOIN "OneDayReferral" ON "OneDayReferral"."refereeId" = earners."userId"
WHERE "OneDayReferral".created_at >= date_trunc('day', now() AT TIME ZONE 'America/Chicago' - interval '1 day')
GROUP BY earners."userId", earners."foreverReferrerId", earners.proportion, earners.rank
ORDER BY rank ASC`
// in order to group earnings for users we use the same createdAt time for
// all earnings
const createdAt = new Date(new Date().getTime())
// stmts is an array of prisma promises we'll call after the loop
const stmts = []
// this is just a sanity check because it seems like a good idea
let total = 0
const notifications = {}
for (const [, earner] of earners.entries()) {
const foreverReferrerEarnings = Math.floor(parseFloat(earner.proportion * sum * 0.1)) // 10% of earnings
let oneDayReferrerEarnings = Math.floor(parseFloat(earner.proportion * sum * 0.1)) // 10% of earnings
const earnerEarnings = Math.floor(parseFloat(earner.proportion * sum)) - foreverReferrerEarnings - oneDayReferrerEarnings
total += earnerEarnings + foreverReferrerEarnings + oneDayReferrerEarnings
if (total > sum) {
console.log(name, 'total exceeds sum', total, '>', sum)
return
}
console.log(
'stacker', earner.userId,
'earned', earnerEarnings,
'proportion', earner.proportion,
'rank', earner.rank,
'type', earner.type,
'foreverReferrer', earner.foreverReferrerId,
'foreverReferrerEarnings', foreverReferrerEarnings,
'oneDayReferrer', earner.oneDayReferrerId,
'oneDayReferrerEarnings', oneDayReferrerEarnings)
if (earnerEarnings > 1000) {
stmts.push(...earnStmts({
msats: earnerEarnings,
userId: earner.userId,
createdAt,
type: earner.type,
rank: earner.rank
}, { models }))
const userN = notifications[earner.userId] || {}
// sum total
const prevMsats = userN.msats || 0
const msats = earnerEarnings + prevMsats
// sum total per earn type (POST, COMMENT, TIP_COMMENT, TIP_POST)
const prevEarnTypeMsats = userN[earner.type]?.msats || 0
const earnTypeMsats = earnerEarnings + prevEarnTypeMsats
// best (=lowest) rank per earn type
const prevEarnTypeBestRank = userN[earner.type]?.bestRank
const earnTypeBestRank = prevEarnTypeBestRank
? Math.min(prevEarnTypeBestRank, Number(earner.rank))
: Number(earner.rank)
notifications[earner.userId] = {
...userN,
msats,
[earner.type]: { msats: earnTypeMsats, bestRank: earnTypeBestRank }
}
}
if (earner.foreverReferrerId && foreverReferrerEarnings > 1000) {
stmts.push(...earnStmts({
msats: foreverReferrerEarnings,
userId: earner.foreverReferrerId,
createdAt,
type: 'FOREVER_REFERRAL',
rank: earner.rank
}, { models }))
} else if (earner.oneDayReferrerId) {
// if the person doesn't have a forever referrer yet, they give double to their one day referrer
oneDayReferrerEarnings += foreverReferrerEarnings
}
if (earner.oneDayReferrerId && oneDayReferrerEarnings > 1000) {
stmts.push(...earnStmts({
msats: oneDayReferrerEarnings,
userId: earner.oneDayReferrerId,
createdAt,
type: 'ONE_DAY_REFERRAL',
rank: earner.rank
}, { models }))
}
}
// execute all the transactions
await models.$transaction(stmts)
Promise.allSettled(
Object.entries(notifications).map(([userId, earnings]) => notifyEarner(parseInt(userId, 10), earnings))
).catch(console.error)
} finally {
models.$disconnect().catch(console.error)
}
}
function earnStmts (data, { models }) {
const { msats, userId } = data
return [
models.earn.create({ data }),
models.user.update({
where: { id: userId },
data: {
msats: { increment: msats },
stackedMsats: { increment: msats }
}
})]
}