mirror of
https://github.com/ajayyy/SponsorBlockServer.git
synced 2024-11-13 02:14:32 +01:00
Add reputation system
This commit is contained in:
parent
799aef0b65
commit
cfcb6c6b64
3 changed files with 52 additions and 2 deletions
|
@ -3,7 +3,7 @@ import { Logger } from "../utils/logger";
|
||||||
import { skipSegmentsHashKey, skipSegmentsKey } from "./redisKeys";
|
import { skipSegmentsHashKey, skipSegmentsKey } from "./redisKeys";
|
||||||
import { Service, VideoID, VideoIDHash } from "../types/segments.model";
|
import { Service, VideoID, VideoIDHash } from "../types/segments.model";
|
||||||
|
|
||||||
async function get<T>(fetchFromDB: () => Promise<T[]>, key: string): Promise<T[]> {
|
async function get<T>(fetchFromDB: () => Promise<T>, key: string): Promise<T> {
|
||||||
const {err, reply} = await redis.getAsync(key);
|
const {err, reply} = await redis.getAsync(key);
|
||||||
|
|
||||||
if (!err && reply) {
|
if (!err && reply) {
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import { Service, VideoID, VideoIDHash } from "../types/segments.model";
|
import { Service, VideoID, VideoIDHash } from "../types/segments.model";
|
||||||
|
import { UserID } from "../types/user.model";
|
||||||
import { Logger } from "../utils/logger";
|
import { Logger } from "../utils/logger";
|
||||||
|
|
||||||
export function skipSegmentsKey(videoID: VideoID, service: Service): string {
|
export function skipSegmentsKey(videoID: VideoID, service: Service): string {
|
||||||
|
@ -10,4 +11,8 @@ export function skipSegmentsHashKey(hashedVideoIDPrefix: VideoIDHash, service: S
|
||||||
if (hashedVideoIDPrefix.length !== 4) Logger.warn("Redis skip segment hash-prefix key is not length 4! " + hashedVideoIDPrefix);
|
if (hashedVideoIDPrefix.length !== 4) Logger.warn("Redis skip segment hash-prefix key is not length 4! " + hashedVideoIDPrefix);
|
||||||
|
|
||||||
return "segments." + service + "." + hashedVideoIDPrefix;
|
return "segments." + service + "." + hashedVideoIDPrefix;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function userKey(userID: UserID): string {
|
||||||
|
return "user." + userID;
|
||||||
|
}
|
45
src/middleware/reputation.ts
Normal file
45
src/middleware/reputation.ts
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
import { db } from "../databases/databases";
|
||||||
|
import { UserID } from "../types/user.model";
|
||||||
|
import { QueryCacher } from "./queryCacher";
|
||||||
|
import { userKey } from "./redisKeys";
|
||||||
|
|
||||||
|
interface ReputationDBResult {
|
||||||
|
totalSubmissions: number,
|
||||||
|
downvotedSubmissions: number,
|
||||||
|
upvotedSum: number,
|
||||||
|
oldUpvotedSubmissions: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getReputation(userID: UserID) {
|
||||||
|
const pastDate = Date.now() - 1000 * 1000 * 60 * 60 * 24 * 45; // 45 days ago
|
||||||
|
const fetchFromDB = () => db.prepare("get",
|
||||||
|
`SELECT COUNT(*) AS "totalSubmissions",
|
||||||
|
SUM(CASE WHEN "votes" < 0 THEN 1 ELSE 0 END) AS "downvotedSubmissions",
|
||||||
|
SUM(CASE WHEN "votes" > 0 THEN "votes" ELSE 0 END) AS "upvotedSum",
|
||||||
|
SUM(CASE WHEN "timeSubmitted" < ? AND "votes" > 0 THEN 1 ELSE 0 END) AS "oldUpvotedSubmissions"
|
||||||
|
FROM "sponsorTimes" WHERE "userID" = ?`, [pastDate, userID]) as Promise<ReputationDBResult>;
|
||||||
|
|
||||||
|
const result = await QueryCacher.get(fetchFromDB, userKey(userID));
|
||||||
|
|
||||||
|
// Grace period
|
||||||
|
if (result.totalSubmissions < 5) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const downvoteRatio = result.downvotedSubmissions / result.totalSubmissions;
|
||||||
|
if (downvoteRatio > 0.3) {
|
||||||
|
return convertRange(downvoteRatio, 0.3, 1, -0.5, -1.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result.oldUpvotedSubmissions < 3 || result.upvotedSum < 5) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
return convertRange(Math.min(result.upvotedSum, 50), 5, 50, 0, 15);
|
||||||
|
}
|
||||||
|
|
||||||
|
function convertRange(value: number, currentMin: number, currentMax: number, targetMin: number, targetMax: number): number {
|
||||||
|
const currentRange = currentMax - currentMin;
|
||||||
|
const targetRange = targetMax - targetMin;
|
||||||
|
return ((value - currentMin) / currentRange) * targetRange + targetMin;
|
||||||
|
}
|
Loading…
Reference in a new issue