import { beforeEach, describe, expect, it, vi, type Mock } from 'vitest'; import { ReopenRaceUseCase, type ReopenRaceInput, type ReopenRaceResult, type ReopenRaceErrorCode, } from './ReopenRaceUseCase'; import type { IRaceRepository } from '../../domain/repositories/IRaceRepository'; import type { Logger } from '@core/shared/application'; import { Race } from '../../domain/entities/Race'; import { SessionType } from '../../domain/value-objects/SessionType'; import type { ApplicationErrorCode } from '@core/shared/errors/ApplicationErrorCode'; describe('ReopenRaceUseCase', () => { let raceRepository: { findById: Mock; update: Mock; }; let logger: { debug: Mock; warn: Mock; info: Mock; error: Mock; }; let useCase: ReopenRaceUseCase; beforeEach(() => { raceRepository = { findById: vi.fn(), update: vi.fn(), }; logger = { debug: vi.fn(), warn: vi.fn(), info: vi.fn(), error: vi.fn(), }; useCase = new ReopenRaceUseCase(raceRepository as unknown as IRaceRepository, logger as unknown as Logger); }); it('returns RACE_NOT_FOUND when race does not exist', async () => { const input: ReopenRaceInput = { raceId: 'race-404', reopenedById: 'admin-1' }; raceRepository.findById.mockResolvedValue(null); const result = await useCase.execute(input); expect(result.isErr()).toBe(true); const err = result.unwrapErr() as ApplicationErrorCode< ReopenRaceErrorCode, { message: string } >; expect(err.code).toBe('RACE_NOT_FOUND'); expect(err.details.message).toContain('race-404'); }); it('reopens a completed race, persists, and presents the result', async () => { const race = Race.create({ id: 'race-1', leagueId: 'league-1', scheduledAt: new Date('2025-01-01T00:00:00.000Z'), track: 'Track 1', car: 'Car 1', sessionType: SessionType.main(), status: 'completed', }); raceRepository.findById.mockResolvedValue(race); raceRepository.update.mockResolvedValue(race.reopen()); const input: ReopenRaceInput = { raceId: 'race-1', reopenedById: 'admin-1' }; const result = await useCase.execute(input); expect(result.isOk()).toBe(true); expect(raceRepository.update).toHaveBeenCalledTimes(1); const updatedRace = (raceRepository.update as Mock).mock.calls[0]?.[0] as Race; expect(updatedRace.id).toBe('race-1'); expect(updatedRace.status.toString()).toBe('scheduled'); const presented = (expect(presented.race.id).toBe('race-1'); expect(presented.race.status.toString()).toBe('scheduled'); expect(logger.info).toHaveBeenCalled(); }); it('returns INVALID_RACE_STATE when race is already scheduled', async () => { const race = Race.create({ id: 'race-1', leagueId: 'league-1', scheduledAt: new Date('2025-01-01T00:00:00.000Z'), track: 'Track 1', car: 'Car 1', sessionType: SessionType.main(), status: 'scheduled', }); raceRepository.findById.mockResolvedValue(race); const input: ReopenRaceInput = { raceId: 'race-1', reopenedById: 'admin-1' }; const result = await useCase.execute(input); expect(result.isErr()).toBe(true); const err = result.unwrapErr() as ApplicationErrorCode< ReopenRaceErrorCode, { message: string } >; expect(err.code).toBe('INVALID_RACE_STATE'); expect(err.details.message).toContain('already scheduled'); }); it('returns INVALID_RACE_STATE when race is running', async () => { const race = Race.create({ id: 'race-1', leagueId: 'league-1', scheduledAt: new Date('2025-01-01T00:00:00.000Z'), track: 'Track 1', car: 'Car 1', sessionType: SessionType.main(), status: 'running', }); raceRepository.findById.mockResolvedValue(race); const input: ReopenRaceInput = { raceId: 'race-1', reopenedById: 'admin-1' }; const result = await useCase.execute(input); expect(result.isErr()).toBe(true); const err = result.unwrapErr() as ApplicationErrorCode< ReopenRaceErrorCode, { message: string } >; expect(err.code).toBe('INVALID_RACE_STATE'); expect(err.details.message).toContain('running race'); }); it('returns REPOSITORY_ERROR when repository throws unexpected error', async () => { raceRepository.findById.mockRejectedValue(new Error('DB error')); const input: ReopenRaceInput = { raceId: 'race-1', reopenedById: 'admin-1' }; const result = await useCase.execute(input); expect(result.isErr()).toBe(true); const err = result.unwrapErr() as ApplicationErrorCode< ReopenRaceErrorCode, { message: string } >; expect(err.code).toBe('REPOSITORY_ERROR'); expect(err.details.message).toBe('DB error'); }); });