sendou.ink/app/features/top-search/XRankPlacementRepository.server.ts
Kalle 77978c450f
Some checks are pending
E2E Tests / e2e (push) Waiting to run
Tests and checks on push / run-checks-and-tests (push) Waiting to run
Updates translation progress / update-translation-progress-issue (push) Waiting to run
New user page (#2812)
Co-authored-by: hfcRed <hfcred@gmx.net>
2026-02-16 19:26:57 +02:00

143 lines
4.0 KiB
TypeScript

import type { InferResult } from "kysely";
import { sql } from "kysely";
import { db } from "~/db/sql";
import type { Tables } from "~/db/tables";
import { modesShort } from "~/modules/in-game-lists/modes";
import type { MainWeaponId } from "~/modules/in-game-lists/types";
export function unlinkPlayerByUserId(userId: number) {
return db
.updateTable("SplatoonPlayer")
.set({ userId: null })
.where("SplatoonPlayer.userId", "=", userId)
.execute();
}
function xRankPlacementsQueryBase() {
return db
.selectFrom("XRankPlacement")
.select([
"XRankPlacement.id",
"XRankPlacement.weaponSplId",
"XRankPlacement.name",
"XRankPlacement.power",
"XRankPlacement.rank",
"XRankPlacement.month",
"XRankPlacement.year",
"XRankPlacement.region",
"XRankPlacement.playerId",
"XRankPlacement.mode",
])
.innerJoin("SplatoonPlayer", "XRankPlacement.playerId", "SplatoonPlayer.id")
.leftJoin("User", "SplatoonPlayer.userId", "User.id")
.select(["User.discordId", "User.customUrl"]);
}
export async function findPlacementsOfMonth(
args: Pick<Tables["XRankPlacement"], "mode" | "region" | "month" | "year">,
) {
return await xRankPlacementsQueryBase()
.where("XRankPlacement.mode", "=", args.mode)
.where("XRankPlacement.region", "=", args.region)
.where("XRankPlacement.month", "=", args.month)
.where("XRankPlacement.year", "=", args.year)
.orderBy("XRankPlacement.rank", "asc")
.execute();
}
export async function findPlacementsByPlayerId(
playerId: Tables["XRankPlacement"]["playerId"],
) {
const result = await xRankPlacementsQueryBase()
.where("XRankPlacement.playerId", "=", playerId)
.orderBy("XRankPlacement.year", "desc")
.orderBy("XRankPlacement.month", "desc")
.orderBy("XRankPlacement.rank", "asc")
.execute();
return result.length ? result : null;
}
export async function findPlacementsByUserId(
userId: Tables["User"]["id"],
options?: { limit?: number; weaponId?: MainWeaponId },
) {
let query = xRankPlacementsQueryBase()
.where("SplatoonPlayer.userId", "=", userId)
.orderBy("XRankPlacement.power", "desc");
if (options?.weaponId) {
query = query.where("XRankPlacement.weaponSplId", "=", options.weaponId);
}
if (options?.limit) {
query = query.limit(options.limit);
}
const result = await query.execute();
return result.length ? result : null;
}
export async function monthYears() {
return await db
.selectFrom("XRankPlacement")
.select(["month", "year"])
.distinct()
.orderBy("year", "desc")
.orderBy("month", "desc")
.execute();
}
export async function findPeaksByUserId(
userId: Tables["User"]["id"],
division?: "both" | "tentatek" | "takoroka",
) {
let innerQuery = db
.selectFrom("XRankPlacement")
.innerJoin("SplatoonPlayer", "XRankPlacement.playerId", "SplatoonPlayer.id")
.where("SplatoonPlayer.userId", "=", userId)
.select([
"XRankPlacement.mode",
"XRankPlacement.rank",
"XRankPlacement.power",
"XRankPlacement.region",
"XRankPlacement.playerId",
sql<number>`ROW_NUMBER() OVER (PARTITION BY "XRankPlacement"."mode" ORDER BY "XRankPlacement"."power" DESC)`.as(
"rn",
),
]);
if (division === "tentatek") {
innerQuery = innerQuery.where("XRankPlacement.region", "=", "WEST");
} else if (division === "takoroka") {
innerQuery = innerQuery.where("XRankPlacement.region", "=", "JPN");
}
const rows = await db
.selectFrom(innerQuery.as("ranked"))
.selectAll()
.where("rn", "=", 1)
.execute();
const peaksByMode = new Map(rows.map((row) => [row.mode, row]));
return modesShort
.map((mode) => peaksByMode.get(mode))
.filter((p): p is NonNullable<typeof p> => p !== undefined);
}
export type FindPlacement = InferResult<
ReturnType<typeof xRankPlacementsQueryBase>
>[number];
export async function refreshAllPeakXp() {
await db
.updateTable("SplatoonPlayer")
.set((eb) => ({
peakXp: eb
.selectFrom("XRankPlacement")
.select((eb) => eb.fn.max("XRankPlacement.power").as("peakXp"))
.whereRef("XRankPlacement.playerId", "=", "SplatoonPlayer.id"),
}))
.execute();
}