rank views
This commit is contained in:
parent
85ea1f0a07
commit
28263bc6c7
|
@ -99,6 +99,17 @@ export async function orderByNumerator (me, models) {
|
||||||
+ "Item"."weightedComments"/2)`
|
+ "Item"."weightedComments"/2)`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function joinSatRankView (me, models) {
|
||||||
|
if (me) {
|
||||||
|
const user = await models.user.findUnique({ where: { id: me.id } })
|
||||||
|
if (user.wildWestMode) {
|
||||||
|
return 'JOIN sat_rank_wwm_view ON "Item".id = sat_rank_wwm_view.id'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'JOIN sat_rank_tender_view ON "Item".id = sat_rank_tender_view.id'
|
||||||
|
}
|
||||||
|
|
||||||
export async function filterClause (me, models) {
|
export async function filterClause (me, models) {
|
||||||
// by default don't include freebies unless they have upvotes
|
// by default don't include freebies unless they have upvotes
|
||||||
let clause = ' AND (NOT "Item".freebie OR "Item"."weightedVotes" - "Item"."weightedDownVotes" > 0'
|
let clause = ' AND (NOT "Item".freebie OR "Item"."weightedVotes" - "Item"."weightedDownVotes" > 0'
|
||||||
|
@ -180,8 +191,8 @@ async function itemQueryWithMeta ({ me, models, query, orderBy = '' }, ...args)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const subClause = (sub, num, table) => {
|
const subClause = (sub, num, table, solo) => {
|
||||||
return sub ? ` AND ${table ? `${table}.` : ''}"subName" = $${num} ` : ''
|
return sub ? ` ${solo ? 'WHERE' : 'AND'} ${table ? `"${table}".` : ''}"subName" = $${num} ` : ''
|
||||||
}
|
}
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
@ -342,47 +353,19 @@ export default {
|
||||||
}, decodedCursor.time, decodedCursor.offset, ...subArr)
|
}, decodedCursor.time, decodedCursor.offset, ...subArr)
|
||||||
break
|
break
|
||||||
default:
|
default:
|
||||||
// HACK we can speed hack the first hot page, by limiting our query to only
|
|
||||||
// the most recently created items so that the tables doesn't have to
|
|
||||||
// fully be computed
|
|
||||||
// if the offset is 0, we limit our search to posts from the last week
|
|
||||||
// if there are 21 items, return them ... if not do the unrestricted query
|
|
||||||
// instead of doing this we should materialize a view ... but this is easier for now
|
|
||||||
if (decodedCursor.offset === 0) {
|
|
||||||
items = await itemQueryWithMeta({
|
items = await itemQueryWithMeta({
|
||||||
me,
|
me,
|
||||||
models,
|
models,
|
||||||
query: `
|
query: `
|
||||||
${SELECT}
|
${SELECT}, rank
|
||||||
FROM "Item"
|
FROM "Item"
|
||||||
WHERE "parentId" IS NULL AND "Item".created_at <= $1 AND "Item".created_at > $3
|
${await joinSatRankView(me, models)}
|
||||||
AND "pinId" IS NULL AND NOT bio AND "deletedAt" IS NULL
|
${subClause(sub, 2, 'Item', true)}
|
||||||
${subClause(sub, 4)}
|
ORDER BY rank DESC NULLS LAST, id DESC
|
||||||
${await filterClause(me, models)}
|
OFFSET $1
|
||||||
${await newTimedOrderByWeightedSats(me, models, 1)}
|
|
||||||
OFFSET $2
|
|
||||||
LIMIT ${LIMIT}`,
|
LIMIT ${LIMIT}`,
|
||||||
orderBy: await newTimedOrderByWeightedSats(me, models, 1)
|
orderBy: 'ORDER BY rank DESC NULLS LAST, id DESC'
|
||||||
}, decodedCursor.time, decodedCursor.offset, new Date(new Date().setDate(new Date().getDate() - 5)), ...subArr)
|
}, decodedCursor.offset, ...subArr)
|
||||||
}
|
|
||||||
|
|
||||||
if (decodedCursor.offset !== 0 || items?.length < LIMIT) {
|
|
||||||
items = await itemQueryWithMeta({
|
|
||||||
me,
|
|
||||||
models,
|
|
||||||
query: `
|
|
||||||
${SELECT}
|
|
||||||
FROM "Item"
|
|
||||||
WHERE "parentId" IS NULL AND "Item".created_at <= $1
|
|
||||||
AND "pinId" IS NULL AND NOT bio AND "deletedAt" IS NULL
|
|
||||||
${subClause(sub, 3)}
|
|
||||||
${await filterClause(me, models)}
|
|
||||||
${await newTimedOrderByWeightedSats(me, models, 1)}
|
|
||||||
OFFSET $2
|
|
||||||
LIMIT ${LIMIT}`,
|
|
||||||
orderBy: await newTimedOrderByWeightedSats(me, models, 1)
|
|
||||||
}, decodedCursor.time, decodedCursor.offset, ...subArr)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (decodedCursor.offset === 0) {
|
if (decodedCursor.offset === 0) {
|
||||||
// get pins for the page and return those separately
|
// get pins for the page and return those separately
|
||||||
|
@ -1249,12 +1232,6 @@ export const SELECT =
|
||||||
"Item"."weightedDownVotes", "Item".freebie, "Item"."otsHash", "Item"."bountyPaidTo",
|
"Item"."weightedDownVotes", "Item".freebie, "Item"."otsHash", "Item"."bountyPaidTo",
|
||||||
ltree2text("Item"."path") AS "path", "Item"."weightedComments"`
|
ltree2text("Item"."path") AS "path", "Item"."weightedComments"`
|
||||||
|
|
||||||
async function newTimedOrderByWeightedSats (me, models, num) {
|
|
||||||
return `
|
|
||||||
ORDER BY (${await orderByNumerator(me, models)}/POWER(GREATEST(3, EXTRACT(EPOCH FROM ($${num} - "Item".created_at))/3600), 1.3) +
|
|
||||||
("Item".boost/${BOOST_MIN}::float)/POWER(EXTRACT(EPOCH FROM ($${num} - "Item".created_at))/3600+2, 2.6)) DESC NULLS LAST, "Item".id DESC`
|
|
||||||
}
|
|
||||||
|
|
||||||
async function topOrderByWeightedSats (me, models) {
|
async function topOrderByWeightedSats (me, models) {
|
||||||
return `ORDER BY ${await orderByNumerator(me, models)} DESC NULLS LAST, "Item".id DESC`
|
return `ORDER BY ${await orderByNumerator(me, models)} DESC NULLS LAST, "Item".id DESC`
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,38 @@
|
||||||
|
-- basically we want to recreate the frontpage every 5 minutes and avoid the computation
|
||||||
|
-- on every page load
|
||||||
|
CREATE MATERIALIZED VIEW IF NOT EXISTS sat_rank_wwm_view AS
|
||||||
|
SELECT *,
|
||||||
|
((GREATEST("weightedVotes", POWER("weightedVotes", 1.2)) + "weightedComments"/2)
|
||||||
|
/
|
||||||
|
POWER(GREATEST(3, EXTRACT(EPOCH FROM (now_utc() - created_at))/3600), 1.3)
|
||||||
|
+
|
||||||
|
(boost/5000::float)
|
||||||
|
/
|
||||||
|
POWER(EXTRACT(EPOCH FROM (now_utc() - created_at))/3600+2, 2.6)) AS rank
|
||||||
|
FROM "Item"
|
||||||
|
WHERE "parentId" IS NULL AND NOT bio AND "pinId" IS NULL AND "deletedAt" IS NULL
|
||||||
|
ORDER BY rank DESC NULLS LAST, id DESC;
|
||||||
|
|
||||||
|
CREATE UNIQUE INDEX IF NOT EXISTS sat_rank_wwm_view_idx ON sat_rank_wwm_view(id);
|
||||||
|
CREATE INDEX IF NOT EXISTS sat_rank_wwm_view_subname_idx ON sat_rank_wwm_view("subName");
|
||||||
|
CREATE INDEX IF NOT EXISTS sat_rank_wwm_view_rank_idx ON sat_rank_wwm_view(rank DESC NULLS LAST, id DESC);
|
||||||
|
|
||||||
|
-- we do the same for the tender view
|
||||||
|
CREATE MATERIALIZED VIEW IF NOT EXISTS sat_rank_tender_view AS
|
||||||
|
SELECT *,
|
||||||
|
((GREATEST(ABS("weightedVotes" - "weightedDownVotes"), POWER(ABS("weightedVotes" - "weightedDownVotes"), 1.2))
|
||||||
|
+ "weightedComments"/2)
|
||||||
|
/
|
||||||
|
POWER(GREATEST(3, EXTRACT(EPOCH FROM (now_utc() - created_at))/3600), 1.3)
|
||||||
|
+
|
||||||
|
(boost/5000::float)
|
||||||
|
/
|
||||||
|
POWER(EXTRACT(EPOCH FROM (now_utc() - created_at))/3600+2, 2.6)) AS rank
|
||||||
|
FROM "Item"
|
||||||
|
WHERE "parentId" IS NULL AND NOT bio AND "pinId" IS NULL AND "deletedAt" IS NULL
|
||||||
|
AND "weightedVotes" > "weightedDownVotes"
|
||||||
|
ORDER BY rank DESC NULLS LAST, id DESC;
|
||||||
|
|
||||||
|
CREATE UNIQUE INDEX IF NOT EXISTS sat_rank_tender_view_idx ON sat_rank_tender_view(id);
|
||||||
|
CREATE INDEX IF NOT EXISTS sat_rank_tender_view_subname_idx ON sat_rank_tender_view("subName");
|
||||||
|
CREATE INDEX IF NOT EXISTS sat_rank_tender_view_rank_idx ON sat_rank_tender_view(rank DESC NULLS LAST, id DESC);
|
|
@ -0,0 +1,22 @@
|
||||||
|
-- hack ... prisma doesn't know about our other schemas (e.g. pgboss)
|
||||||
|
-- and this is only really a problem on their "shadow database"
|
||||||
|
-- so we catch the exception it throws and ignore it
|
||||||
|
CREATE OR REPLACE FUNCTION create_scheduled_jobs()
|
||||||
|
RETURNS INTEGER
|
||||||
|
LANGUAGE plpgsql
|
||||||
|
AS $$
|
||||||
|
DECLARE
|
||||||
|
BEGIN
|
||||||
|
INSERT INTO pgboss.schedule (name, cron, timezone) VALUES ('trust', '0 2 * * *', 'America/Chicago') ON CONFLICT DO NOTHING;
|
||||||
|
INSERT INTO pgboss.schedule (name, cron, timezone) VALUES ('auction', '* * * * *', 'America/Chicago') ON CONFLICT DO NOTHING;
|
||||||
|
INSERT INTO pgboss.schedule (name, cron, timezone) VALUES ('earn', '0 0 * * *', 'America/Chicago') ON CONFLICT DO NOTHING;
|
||||||
|
INSERT INTO pgboss.schedule (name, cron, timezone) VALUES ('streak', '15 0 * * *','America/Chicago') ON CONFLICT DO NOTHING;
|
||||||
|
INSERT INTO pgboss.schedule (name, cron, timezone) VALUES ('views', '0 0 * * *', 'America/Chicago') ON CONFLICT DO NOTHING;
|
||||||
|
INSERT INTO pgboss.schedule (name, cron, timezone) VALUES ('rankViews', '* * * * *', 'America/Chicago') ON CONFLICT DO NOTHING;
|
||||||
|
return 0;
|
||||||
|
EXCEPTION WHEN OTHERS THEN
|
||||||
|
return 0;
|
||||||
|
END;
|
||||||
|
$$;
|
||||||
|
|
||||||
|
SELECT create_scheduled_jobs();
|
|
@ -15,7 +15,7 @@ const { nip57 } = require('./nostr')
|
||||||
|
|
||||||
const fetch = require('cross-fetch')
|
const fetch = require('cross-fetch')
|
||||||
const { authenticatedLndGrpc } = require('ln-service')
|
const { authenticatedLndGrpc } = require('ln-service')
|
||||||
const { views } = require('./views')
|
const { views, rankViews } = require('./views')
|
||||||
|
|
||||||
async function work () {
|
async function work () {
|
||||||
const boss = new PgBoss(process.env.DATABASE_URL)
|
const boss = new PgBoss(process.env.DATABASE_URL)
|
||||||
|
@ -62,6 +62,7 @@ async function work () {
|
||||||
await boss.work('checkStreak', checkStreak(args))
|
await boss.work('checkStreak', checkStreak(args))
|
||||||
await boss.work('nip57', nip57(args))
|
await boss.work('nip57', nip57(args))
|
||||||
await boss.work('views', views(args))
|
await boss.work('views', views(args))
|
||||||
|
await boss.work('rankViews', rankViews(args))
|
||||||
|
|
||||||
console.log('working jobs')
|
console.log('working jobs')
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,4 +14,17 @@ function views ({ models }) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = { views }
|
// this should be run regularly ... like, every 1-5 minutes
|
||||||
|
function rankViews ({ models }) {
|
||||||
|
return async function () {
|
||||||
|
console.log('refreshing rank views')
|
||||||
|
|
||||||
|
for (const view of ['sat_rank_wwm_view', 'sat_rank_tender_view']) {
|
||||||
|
await models.$queryRaw(`REFRESH MATERIALIZED VIEW CONCURRENTLY ${view}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('done refreshing rank views')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { views, rankViews }
|
||||||
|
|
Loading…
Reference in New Issue