diff --git a/api/resolvers/rewards.js b/api/resolvers/rewards.js
index 3dcfd2c25..a65ebc43e 100644
--- a/api/resolvers/rewards.js
+++ b/api/resolvers/rewards.js
@@ -157,7 +157,7 @@ export default {
const [{ to, from }] = await models.$queryRaw`
SELECT date_trunc('day', (now() AT TIME ZONE 'America/Chicago')) AT TIME ZONE 'America/Chicago' as from,
(date_trunc('day', (now() AT TIME ZONE 'America/Chicago')) AT TIME ZONE 'America/Chicago') + interval '1 day - 1 second' as to`
- return await topUsers(parent, { when: 'custom', to: new Date(to).getTime().toString(), from: new Date(from).getTime().toString(), limit: 100 }, { models, ...context })
+ return await topUsers(parent, { when: 'custom', to: new Date(to).getTime().toString(), from: new Date(from).getTime().toString(), limit: 500 }, { models, ...context })
},
total: async (parent, args, { models }) => {
if (!parent.total) {
diff --git a/api/resolvers/user.js b/api/resolvers/user.js
index e836b22be..03f6a8d13 100644
--- a/api/resolvers/user.js
+++ b/api/resolvers/user.js
@@ -66,11 +66,12 @@ export async function topUsers (parent, { cursor, when, by, from, to, limit = LI
case 'comments': column = 'ncomments'; break
case 'referrals': column = 'referrals'; break
case 'stacking': column = 'stacked'; break
+ case 'value':
default: column = 'proportion'; break
}
const users = (await models.$queryRawUnsafe(`
- SELECT *
+ SELECT * ${column === 'proportion' ? ', proportion' : ''}
FROM
(SELECT users.*,
COALESCE(floor(sum(msats_spent)/1000), 0) as spent,
diff --git a/api/typeDefs/user.js b/api/typeDefs/user.js
index 8b100170f..e61cb4b76 100644
--- a/api/typeDefs/user.js
+++ b/api/typeDefs/user.js
@@ -59,6 +59,11 @@ export default gql`
photoId: Int
since: Int
+ """
+ this is only returned when we sort stackers by value
+ """
+ proportion: Float
+
optional: UserOptional!
privates: UserPrivates
diff --git a/components/user-list.js b/components/user-list.js
index 40f77286f..90be34f8e 100644
--- a/components/user-list.js
+++ b/components/user-list.js
@@ -107,13 +107,13 @@ export function User ({ user, rank, statComps, className = 'mb-2', Embellish, ny
{statComps.map((Comp, i) => )}
}
- {Embellish && }
+ {Embellish && }
>
)
}
-function UserHidden ({ rank, Embellish }) {
+function UserHidden ({ rank, user, Embellish }) {
return (
<>
{rank
@@ -133,7 +133,7 @@ function UserHidden ({ rank, Embellish }) {
stacker is in hiding
- {Embellish && }
+ {Embellish && }
>
@@ -148,7 +148,7 @@ export function ListUsers ({ users, rank, statComps = DEFAULT_STAT_COMPONENTS, E
{users.map((user, i) => (
user
?
- :
+ :
))}
)
diff --git a/docker-compose.yml b/docker-compose.yml
index ca0154480..40406609c 100644
--- a/docker-compose.yml
+++ b/docker-compose.yml
@@ -680,7 +680,6 @@ services:
- bitcoin
- sn_lnd
- lnd
- - cln
restart: unless-stopped
command: daemon --docker -f label=com.docker.compose.project=${COMPOSE_PROJECT_NAME}
volumes:
diff --git a/fragments/users.js b/fragments/users.js
index da2078618..b96ec9327 100644
--- a/fragments/users.js
+++ b/fragments/users.js
@@ -254,7 +254,7 @@ export const TOP_USERS = gql`
photoId
ncomments(when: $when, from: $from, to: $to)
nposts(when: $when, from: $from, to: $to)
-
+ proportion
optional {
stacked(when: $when, from: $from, to: $to)
spent(when: $when, from: $from, to: $to)
diff --git a/pages/rewards/index.js b/pages/rewards/index.js
index 164710f4c..a81d88437 100644
--- a/pages/rewards/index.js
+++ b/pages/rewards/index.js
@@ -16,7 +16,6 @@ import { useToast } from '@/components/toast'
import { useLightning } from '@/components/lightning'
import { ListUsers } from '@/components/user-list'
import { Col, Row } from 'react-bootstrap'
-import { proportions } from '@/lib/madness'
import { useData } from '@/components/use-data'
import { GrowthPieChartSkeleton } from '@/components/charts-skeletons'
import { useMemo } from 'react'
@@ -50,6 +49,7 @@ ${ITEM_FULL_FIELDS}
photoId
ncomments
nposts
+ proportion
optional {
streak
@@ -117,9 +117,10 @@ export default function Rewards ({ ssrData }) {
if (!dat) return
- function EstimatedReward ({ rank }) {
- const referrerReward = Math.floor(total * proportions[rank - 1] * 0.2)
- const reward = Math.floor(total * proportions[rank - 1]) - referrerReward
+ function EstimatedReward ({ rank, user }) {
+ if (!user) return null
+ const referrerReward = Math.max(Math.floor(total * user.proportion * 0.2), 0)
+ const reward = Math.max(Math.floor(total * user.proportion) - referrerReward, 0)
return (
diff --git a/prisma/migrations/20241217163642_user_values_improve/migration.sql b/prisma/migrations/20241217163642_user_values_improve/migration.sql
new file mode 100644
index 000000000..47f21be36
--- /dev/null
+++ b/prisma/migrations/20241217163642_user_values_improve/migration.sql
@@ -0,0 +1,93 @@
+CREATE OR REPLACE FUNCTION user_values(
+ min TIMESTAMP(3), max TIMESTAMP(3), ival INTERVAL, date_part TEXT,
+ percentile_cutoff INTEGER DEFAULT 50,
+ each_upvote_portion FLOAT DEFAULT 4.0,
+ each_item_portion FLOAT DEFAULT 4.0,
+ handicap_ids INTEGER[] DEFAULT '{616, 6030, 4502, 27}',
+ handicap_zap_mult FLOAT DEFAULT 0.3)
+RETURNS TABLE (
+ t TIMESTAMP(3), id INTEGER, proportion FLOAT
+)
+LANGUAGE plpgsql
+AS $$
+DECLARE
+ min_utc TIMESTAMP(3) := timezone('utc', min AT TIME ZONE 'America/Chicago');
+BEGIN
+ RETURN QUERY
+ SELECT period.t, u."userId", u.total_proportion
+ FROM generate_series(min, max, ival) period(t),
+ LATERAL
+ (WITH item_ratios AS (
+ SELECT *,
+ CASE WHEN "parentId" IS NULL THEN 'POST' ELSE 'COMMENT' END as type,
+ CASE WHEN "weightedVotes" > 0 THEN "weightedVotes"/(sum("weightedVotes") OVER (PARTITION BY "parentId" IS NULL)) ELSE 0 END AS ratio
+ FROM (
+ SELECT *,
+ NTILE(100) OVER (PARTITION BY "parentId" IS NULL ORDER BY ("weightedVotes"-"weightedDownVotes") desc) AS percentile,
+ ROW_NUMBER() OVER (PARTITION BY "parentId" IS NULL ORDER BY ("weightedVotes"-"weightedDownVotes") desc) AS rank
+ FROM
+ "Item"
+ WHERE date_trunc(date_part, created_at AT TIME ZONE 'UTC' AT TIME ZONE 'America/Chicago') = period.t
+ AND "weightedVotes" > 0
+ AND "deletedAt" IS NULL
+ AND NOT bio
+ AND ("invoiceActionState" IS NULL OR "invoiceActionState" = 'PAID')
+ ) x
+ WHERE x.percentile <= percentile_cutoff
+ ),
+ -- get top upvoters of top posts and comments
+ upvoter_islands AS (
+ SELECT "ItemAct"."userId", item_ratios.id, item_ratios.ratio, item_ratios."parentId",
+ "ItemAct".msats as tipped, "ItemAct".created_at as acted_at,
+ ROW_NUMBER() OVER (partition by item_ratios.id order by "ItemAct".created_at asc)
+ - ROW_NUMBER() OVER (partition by item_ratios.id, "ItemAct"."userId" order by "ItemAct".created_at asc) AS island
+ FROM item_ratios
+ JOIN "ItemAct" on "ItemAct"."itemId" = item_ratios.id
+ WHERE act = 'TIP'
+ AND date_trunc(date_part, "ItemAct".created_at AT TIME ZONE 'UTC' AT TIME ZONE 'America/Chicago') = period.t
+ AND ("ItemAct"."invoiceActionState" IS NULL OR "ItemAct"."invoiceActionState" = 'PAID')
+ ),
+ -- isolate contiguous upzaps from the same user on the same item so that when we take the log
+ -- of the upzaps it accounts for successive zaps and does not disproportionately reward them
+ -- quad root of the total tipped
+ upvoters AS (
+ SELECT "userId", upvoter_islands.id, ratio, "parentId", GREATEST(power(sum(tipped) / 1000, 0.25), 0) as tipped, min(acted_at) as acted_at
+ FROM upvoter_islands
+ GROUP BY "userId", upvoter_islands.id, ratio, "parentId", island
+ ),
+ -- the relative contribution of each upvoter to the post/comment
+ -- early component: 1/ln(early_rank + e - 1)
+ -- tipped component: how much they tipped relative to the total tipped for the item
+ -- multiplied by the relative rank of the item to the total items
+ -- multiplied by the trust of the user
+ upvoter_ratios AS (
+ SELECT "userId", sum((early_multiplier+tipped_ratio)*ratio*CASE WHEN users.id = ANY (handicap_ids) THEN handicap_zap_mult ELSE users.trust+0.1 END) as upvoter_ratio,
+ "parentId" IS NULL as "isPost", CASE WHEN "parentId" IS NULL THEN 'TIP_POST' ELSE 'TIP_COMMENT' END as type
+ FROM (
+ SELECT *,
+ 1.0/LN(ROW_NUMBER() OVER (partition by upvoters.id order by acted_at asc) + EXP(1.0) - 1) AS early_multiplier,
+ tipped::float/(sum(tipped) OVER (partition by upvoters.id)) tipped_ratio
+ FROM upvoters
+ WHERE tipped > 2.1
+ ) u
+ JOIN users on "userId" = users.id
+ GROUP BY "userId", "parentId" IS NULL
+ ),
+ proportions AS (
+ SELECT "userId", NULL as id, type, ROW_NUMBER() OVER (PARTITION BY "isPost" ORDER BY upvoter_ratio DESC) as rank,
+ upvoter_ratio/(sum(upvoter_ratio) OVER (PARTITION BY "isPost"))/each_upvote_portion as proportion
+ FROM upvoter_ratios
+ WHERE upvoter_ratio > 0
+ UNION ALL
+ SELECT "userId", item_ratios.id, type, rank, ratio/each_item_portion as proportion
+ FROM item_ratios
+ )
+ SELECT "userId", sum(proportions.proportion) AS total_proportion
+ FROM proportions
+ GROUP BY "userId"
+ HAVING sum(proportions.proportion) > 0.000001) u;
+END;
+$$;
+
+REFRESH MATERIALIZED VIEW CONCURRENTLY user_values_today;
+REFRESH MATERIALIZED VIEW CONCURRENTLY user_values_days;
\ No newline at end of file
diff --git a/worker/earn.js b/worker/earn.js
index 630342597..019cf9bb1 100644
--- a/worker/earn.js
+++ b/worker/earn.js
@@ -1,6 +1,5 @@
import { notifyEarner } from '@/lib/webPush'
import createPrisma from '@/lib/create-prisma'
-import { proportions } from '@/lib/madness'
import { SN_NO_REWARDS_IDS } from '@/lib/constants'
const TOTAL_UPPER_BOUND_MSATS = 1_000_000_000
@@ -40,18 +39,19 @@ export async function earn ({ name }) {
/*
How earnings (used to) work:
- 1/3: top 21% posts over last 36 hours, scored on a relative basis
- 1/3: top 21% comments over last 36 hours, scored on a relative basis
+ 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 100 stackers by value, and 10% each to their forever and one day referrers
+ 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",
@@ -63,8 +63,8 @@ export async function earn ({ name }) {
'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
- LIMIT 100
)
SELECT earners.*,
COALESCE(
@@ -86,10 +86,10 @@ export async function earn ({ name }) {
let total = 0
const notifications = {}
- for (const [i, earner] of earners.entries()) {
+ 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(proportions[i] * sum)) - foreverReferrerEarnings - oneDayReferrerEarnings
+ const earnerEarnings = Math.floor(parseFloat(earner.proportion * sum)) - foreverReferrerEarnings - oneDayReferrerEarnings
total += earnerEarnings + foreverReferrerEarnings + oneDayReferrerEarnings
if (total > sum) {
@@ -108,7 +108,7 @@ export async function earn ({ name }) {
'oneDayReferrer', earner.oneDayReferrerId,
'oneDayReferrerEarnings', oneDayReferrerEarnings)
- if (earnerEarnings > 0) {
+ if (earnerEarnings > 1000) {
stmts.push(...earnStmts({
msats: earnerEarnings,
userId: earner.userId,
@@ -140,7 +140,7 @@ export async function earn ({ name }) {
}
}
- if (earner.foreverReferrerId && foreverReferrerEarnings > 0) {
+ if (earner.foreverReferrerId && foreverReferrerEarnings > 1000) {
stmts.push(...earnStmts({
msats: foreverReferrerEarnings,
userId: earner.foreverReferrerId,
@@ -153,7 +153,7 @@ export async function earn ({ name }) {
oneDayReferrerEarnings += foreverReferrerEarnings
}
- if (earner.oneDayReferrerId && oneDayReferrerEarnings > 0) {
+ if (earner.oneDayReferrerId && oneDayReferrerEarnings > 1000) {
stmts.push(...earnStmts({
msats: oneDayReferrerEarnings,
userId: earner.oneDayReferrerId,