107 lines
3.7 KiB
TypeScript
107 lines
3.7 KiB
TypeScript
import type { IRaceRepository } from '../../domain/repositories/IRaceRepository';
|
|
import type { ILeagueRepository } from '../../domain/repositories/ILeagueRepository';
|
|
import type { Logger } from '@core/shared/application';
|
|
import type { UseCaseOutputPort } from '@core/shared/application/UseCaseOutputPort';
|
|
import { Result } from '@core/shared/application/Result';
|
|
import type { ApplicationErrorCode } from '@core/shared/errors/ApplicationErrorCode';
|
|
import type { RaceStatus } from '../../domain/entities/Race';
|
|
|
|
export type GetAllRacesPageDataInput = {};
|
|
|
|
export interface GetAllRacesPageRaceItem {
|
|
id: string;
|
|
track: string;
|
|
car: string;
|
|
scheduledAt: string;
|
|
status: RaceStatus;
|
|
leagueId: string;
|
|
leagueName: string;
|
|
strengthOfField: number | null;
|
|
}
|
|
|
|
export interface GetAllRacesPageDataFilters {
|
|
statuses: { value: 'all' | RaceStatus; label: string }[];
|
|
leagues: { id: string; name: string }[];
|
|
}
|
|
|
|
export interface GetAllRacesPageDataResult {
|
|
races: GetAllRacesPageRaceItem[];
|
|
filters: GetAllRacesPageDataFilters;
|
|
}
|
|
|
|
export type GetAllRacesPageDataErrorCode = 'REPOSITORY_ERROR';
|
|
|
|
export class GetAllRacesPageDataUseCase {
|
|
constructor(
|
|
private readonly raceRepository: IRaceRepository,
|
|
private readonly leagueRepository: ILeagueRepository,
|
|
private readonly logger: Logger,
|
|
private readonly output: UseCaseOutputPort<GetAllRacesPageDataResult>,
|
|
) {}
|
|
|
|
async execute(
|
|
_input: GetAllRacesPageDataInput,
|
|
): Promise<Result<void, ApplicationErrorCode<GetAllRacesPageDataErrorCode, { message: string }>>> {
|
|
void _input;
|
|
this.logger.debug('Executing GetAllRacesPageDataUseCase');
|
|
try {
|
|
const [allRaces, allLeagues] = await Promise.all([
|
|
this.raceRepository.findAll(),
|
|
this.leagueRepository.findAll(),
|
|
]);
|
|
this.logger.info(`Found ${allRaces.length} races and ${allLeagues.length} leagues.`);
|
|
|
|
const leagueMap = new Map(allLeagues.map(league => [league.id.toString(), league.name.toString()]));
|
|
|
|
const races: GetAllRacesPageRaceItem[] = allRaces
|
|
.slice()
|
|
.sort((a, b) => b.scheduledAt.getTime() - a.scheduledAt.getTime())
|
|
.map(race => ({
|
|
id: race.id,
|
|
track: race.track,
|
|
car: race.car,
|
|
scheduledAt: race.scheduledAt.toISOString(),
|
|
status: race.status,
|
|
leagueId: race.leagueId,
|
|
leagueName: leagueMap.get(race.leagueId) ?? 'Unknown League',
|
|
strengthOfField: race.strengthOfField ?? null,
|
|
}));
|
|
|
|
const uniqueLeagues = new Map<string, { id: string; name: string }>();
|
|
for (const league of allLeagues) {
|
|
uniqueLeagues.set(league.id.toString(), { id: league.id.toString(), name: league.name.toString() });
|
|
}
|
|
|
|
const filters: GetAllRacesPageDataFilters = {
|
|
statuses: [
|
|
{ value: 'all', label: 'All Statuses' },
|
|
{ value: 'scheduled', label: 'Scheduled' },
|
|
{ value: 'running', label: 'Live' },
|
|
{ value: 'completed', label: 'Completed' },
|
|
{ value: 'cancelled', label: 'Cancelled' },
|
|
],
|
|
leagues: Array.from(uniqueLeagues.values()),
|
|
};
|
|
|
|
const result: GetAllRacesPageDataResult = {
|
|
races,
|
|
filters,
|
|
};
|
|
|
|
this.logger.debug('Successfully retrieved all races page data.');
|
|
this.output.present(result);
|
|
|
|
return Result.ok(undefined);
|
|
} catch (error) {
|
|
this.logger.error(
|
|
'Error executing GetAllRacesPageDataUseCase',
|
|
error instanceof Error ? error : new Error(String(error)),
|
|
);
|
|
return Result.err({
|
|
code: 'REPOSITORY_ERROR',
|
|
details: { message: error instanceof Error ? error.message : 'Unknown error occurred' },
|
|
});
|
|
}
|
|
}
|
|
}
|