Files
gridpilot.gg/core/racing/application/use-cases/GetDriversLeaderboardUseCase.ts
2025-12-31 15:39:28 +01:00

128 lines
4.1 KiB
TypeScript

import type { Logger, UseCase, UseCaseOutputPort } from '@core/shared/application';
import { Result } from '@core/shared/application/Result';
import type { ApplicationErrorCode } from '@core/shared/errors/ApplicationErrorCode';
import type { Driver } from '../../domain/entities/Driver';
import type { Team } from '../../domain/entities/Team';
import type { IDriverRepository } from '../../domain/repositories/IDriverRepository';
import type { IDriverStatsUseCase } from './IDriverStatsUseCase';
import type { IRankingUseCase } from './IRankingUseCase';
import { SkillLevelService, type SkillLevel } from '../../domain/services/SkillLevelService';
import { MediaReference } from '@core/domain/media/MediaReference';
export type GetDriversLeaderboardInput = {
leagueId?: string;
seasonId?: string;
};
export interface DriverLeaderboardItem {
driver: Driver;
team?: Team;
rating: number;
skillLevel: SkillLevel;
racesCompleted: number;
wins: number;
podiums: number;
isActive: boolean;
rank: number;
avatarRef?: MediaReference;
}
export interface GetDriversLeaderboardResult {
items: DriverLeaderboardItem[];
totalRaces: number;
totalWins: number;
activeCount: number;
}
export type GetDriversLeaderboardErrorCode =
| 'LEAGUE_NOT_FOUND'
| 'SEASON_NOT_FOUND'
| 'REPOSITORY_ERROR';
/**
* Use Case for retrieving driver leaderboard data.
* Returns a Result containing the domain leaderboard model.
*/
export class GetDriversLeaderboardUseCase implements UseCase<GetDriversLeaderboardInput, void, GetDriversLeaderboardErrorCode> {
constructor(
private readonly driverRepository: IDriverRepository,
private readonly rankingUseCase: IRankingUseCase,
private readonly driverStatsUseCase: IDriverStatsUseCase,
private readonly logger: Logger,
private readonly output: UseCaseOutputPort<GetDriversLeaderboardResult>,
) {}
async execute(
input: GetDriversLeaderboardInput,
): Promise<
Result<
void,
ApplicationErrorCode<GetDriversLeaderboardErrorCode, { message: string }>
>
> {
this.logger.debug('Executing GetDriversLeaderboardUseCase', { input });
try {
const drivers = await this.driverRepository.findAll();
const rankings = await this.rankingUseCase.getAllDriverRankings();
// Get stats for all drivers
const statsPromises = drivers.map(driver =>
this.driverStatsUseCase.getDriverStats(driver.id)
);
const statsResults = await Promise.all(statsPromises);
const statsMap = new Map<string, any>();
drivers.forEach((driver, idx) => {
if (statsResults[idx]) {
statsMap.set(driver.id, statsResults[idx]);
}
});
const items: DriverLeaderboardItem[] = drivers.map((driver) => {
const ranking = rankings.find((r) => r.driverId === driver.id);
const stats = statsMap.get(driver.id);
const rating = ranking?.rating ?? 0;
const racesCompleted = stats?.totalRaces ?? 0;
const skillLevel: SkillLevel = SkillLevelService.getSkillLevel(rating);
return {
driver,
rating,
skillLevel,
racesCompleted,
wins: stats?.wins ?? 0,
podiums: stats?.podiums ?? 0,
isActive: racesCompleted > 0,
rank: ranking?.overallRank ?? 0,
avatarRef: driver.avatarRef,
};
});
const totalRaces = items.reduce((sum, d) => sum + d.racesCompleted, 0);
const totalWins = items.reduce((sum, d) => sum + d.wins, 0);
const activeCount = items.filter(d => d.isActive).length;
const result: GetDriversLeaderboardResult = {
items: items.sort((a, b) => b.rating - a.rating),
totalRaces,
totalWins,
activeCount,
};
this.logger.debug('Successfully computed drivers leaderboard');
this.output.present(result);
return Result.ok(void 0);
} catch (error) {
const err = error instanceof Error ? error : new Error(String(error));
this.logger.error('Error executing GetDriversLeaderboardUseCase', err);
return Result.err({
code: 'REPOSITORY_ERROR',
details: { message: err.message ?? 'Unknown error occurred' },
});
}
}
}