2024-09-28 05:57:35 +01:00
|
|
|
import { PlayerHistory } from "@/common/player/player-history";
|
|
|
|
import { IPlayer } from "@/common/schema/player-schema";
|
|
|
|
import ScoreSaberPlayerToken from "@/common/model/token/scoresaber/score-saber-player-token";
|
|
|
|
import ScoreSaberPlayer from "@/common/model/player/impl/scoresaber-player";
|
|
|
|
import { getDaysAgoDate, getMidnightAlignedDate } from "@/common/time-utils";
|
|
|
|
import { scoresaberService } from "@/common/service/impl/scoresaber";
|
|
|
|
import { IO } from "@trigger.dev/sdk";
|
|
|
|
|
|
|
|
const INACTIVE_CHECK_AGAIN_TIME = 3 * 24 * 60 * 60 * 1000; // 3 days
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sorts the player history based on date,
|
|
|
|
* so the most recent date is first
|
|
|
|
*
|
|
|
|
* @param history the player history
|
|
|
|
*/
|
|
|
|
export function sortPlayerHistory(history: Map<string, PlayerHistory>) {
|
|
|
|
return Array.from(history.entries()).sort(
|
2024-09-30 22:16:55 +01:00
|
|
|
(a, b) => Date.parse(b[0]) - Date.parse(a[0]) // Sort in descending order
|
2024-09-28 05:57:35 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-10-02 10:22:10 +01:00
|
|
|
/**
|
|
|
|
* Gets a value from an {@link PlayerHistory}
|
|
|
|
* based on the field
|
|
|
|
*
|
|
|
|
* @param history the history to get the value from
|
|
|
|
* @param field the field to get
|
|
|
|
*/
|
|
|
|
export function getValueFromHistory(history: PlayerHistory, field: string): number | null {
|
|
|
|
const keys = field.split(".");
|
|
|
|
let value: any = history;
|
|
|
|
|
|
|
|
// Navigate through the keys safely
|
|
|
|
for (const key of keys) {
|
|
|
|
if (value && key in value) {
|
|
|
|
value = value[key];
|
|
|
|
} else {
|
|
|
|
return null; // Return null if the key doesn't exist
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure we return a number or null
|
|
|
|
return typeof value === "number" ? value : null;
|
|
|
|
}
|
|
|
|
|
2024-09-28 05:57:35 +01:00
|
|
|
/**
|
|
|
|
* Sorts the player history based on date,
|
|
|
|
* so the most recent date is first
|
|
|
|
*
|
|
|
|
* @param foundPlayer the player
|
|
|
|
* @param player the scoresaber player
|
|
|
|
* @param rawPlayer the raw scoresaber player
|
|
|
|
*/
|
|
|
|
export async function seedPlayerHistory(
|
|
|
|
foundPlayer: IPlayer,
|
|
|
|
player: ScoreSaberPlayer,
|
2024-09-30 22:16:55 +01:00
|
|
|
rawPlayer: ScoreSaberPlayerToken
|
2024-09-28 05:57:35 +01:00
|
|
|
): Promise<Map<string, PlayerHistory>> {
|
|
|
|
// Loop through rankHistory in reverse, from current day backwards
|
2024-09-30 22:16:55 +01:00
|
|
|
const playerRankHistory = rawPlayer.histories.split(",").map(value => {
|
2024-09-28 05:57:35 +01:00
|
|
|
return parseInt(value);
|
|
|
|
});
|
|
|
|
playerRankHistory.push(player.rank);
|
|
|
|
|
|
|
|
let daysAgo = 0; // Start from current day
|
|
|
|
for (let i = playerRankHistory.length - 1; i >= 0; i--) {
|
|
|
|
const rank = playerRankHistory[i];
|
|
|
|
const date = getMidnightAlignedDate(getDaysAgoDate(daysAgo));
|
|
|
|
foundPlayer.setStatisticHistory(date, {
|
|
|
|
rank: rank,
|
|
|
|
});
|
|
|
|
daysAgo += 1; // Increment daysAgo for each earlier rank
|
|
|
|
}
|
|
|
|
|
|
|
|
foundPlayer.sortStatisticHistory();
|
|
|
|
await foundPlayer.save();
|
|
|
|
return foundPlayer.getStatisticHistory();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tracks a players statistics
|
|
|
|
*
|
|
|
|
* This is ONLY to be used in Trigger.
|
|
|
|
*
|
|
|
|
* @param io the io from Trigger
|
|
|
|
* @param dateToday the date to use
|
|
|
|
* @param foundPlayer the player to track
|
|
|
|
*/
|
2024-09-30 22:16:55 +01:00
|
|
|
export async function trackScoreSaberPlayer(dateToday: Date, foundPlayer: IPlayer, io?: IO) {
|
2024-09-28 05:57:35 +01:00
|
|
|
io && (await io.logger.info(`Updating statistics for ${foundPlayer.id}...`));
|
|
|
|
|
|
|
|
// Check if the player is inactive and if we check their inactive status again
|
|
|
|
if (
|
|
|
|
foundPlayer.rawPlayer &&
|
|
|
|
foundPlayer.rawPlayer.inactive &&
|
2024-09-30 22:16:55 +01:00
|
|
|
Date.now() - foundPlayer.getLastTracked().getTime() > INACTIVE_CHECK_AGAIN_TIME
|
2024-09-28 05:57:35 +01:00
|
|
|
) {
|
2024-09-30 22:16:55 +01:00
|
|
|
io && (await io.logger.warn(`Player ${foundPlayer.id} is inactive, skipping...`));
|
2024-09-28 05:57:35 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lookup player data from the ScoreSaber service
|
|
|
|
const response = await scoresaberService.lookupPlayer(foundPlayer.id, true);
|
|
|
|
if (response == undefined) {
|
2024-09-30 22:16:55 +01:00
|
|
|
io && (await io.logger.warn(`Player ${foundPlayer.id} not found on ScoreSaber`));
|
2024-09-28 05:57:35 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
const { player, rawPlayer } = response;
|
|
|
|
foundPlayer.rawPlayer = player; // Update the raw player data
|
|
|
|
|
|
|
|
if (player.inactive) {
|
|
|
|
io && (await io.logger.warn(`Player ${foundPlayer.id} is inactive`));
|
|
|
|
await foundPlayer.save(); // Save the player
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const statisticHistory = foundPlayer.getStatisticHistory();
|
|
|
|
|
|
|
|
// Seed the history with ScoreSaber data if no history exists
|
|
|
|
if (statisticHistory.size === 0) {
|
|
|
|
io && (await io.logger.info(`Seeding history for ${foundPlayer.id}...`));
|
|
|
|
await seedPlayerHistory(foundPlayer, player, rawPlayer);
|
|
|
|
io && (await io.logger.info(`Seeded history for ${foundPlayer.id}`));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update current day's statistics
|
2024-09-30 07:50:24 +01:00
|
|
|
let history = foundPlayer.getHistoryByDate(dateToday);
|
2024-09-28 05:57:35 +01:00
|
|
|
if (history == undefined) {
|
|
|
|
history = {}; // Initialize if history is not found
|
|
|
|
}
|
2024-10-02 10:22:10 +01:00
|
|
|
|
2024-09-28 05:57:35 +01:00
|
|
|
// Set the history data
|
|
|
|
history.pp = player.pp;
|
|
|
|
history.countryRank = player.countryRank;
|
|
|
|
history.rank = player.rank;
|
2024-10-02 10:22:10 +01:00
|
|
|
history.accuracy = {
|
|
|
|
averageRankedAccuracy: rawPlayer.scoreStats.averageRankedAccuracy,
|
|
|
|
};
|
2024-09-28 05:57:35 +01:00
|
|
|
foundPlayer.setStatisticHistory(dateToday, history);
|
|
|
|
foundPlayer.sortStatisticHistory();
|
2024-10-01 09:52:36 +01:00
|
|
|
foundPlayer.lastTracked = new Date();
|
2024-09-28 05:57:35 +01:00
|
|
|
await foundPlayer.save();
|
|
|
|
|
|
|
|
io && (await io.logger.info(`Updated statistics for ${foundPlayer.id}`));
|
|
|
|
}
|