refactor racing use cases
This commit is contained in:
@@ -1,38 +1,59 @@
|
||||
import { describe, it, expect, beforeEach, vi, Mock } from 'vitest';
|
||||
import { ReviewProtestUseCase } from './ReviewProtestUseCase';
|
||||
import { ReviewProtestUseCase, type ReviewProtestInput, type ReviewProtestResult, type ReviewProtestErrorCode } from './ReviewProtestUseCase';
|
||||
import type { IProtestRepository } from '../../domain/repositories/IProtestRepository';
|
||||
import type { IRaceRepository } from '../../domain/repositories/IRaceRepository';
|
||||
import type { ILeagueMembershipRepository } from '../../domain/repositories/ILeagueMembershipRepository';
|
||||
import type { ApplicationErrorCode } from '@core/shared/errors/ApplicationErrorCode';
|
||||
import type { Logger, UseCaseOutputPort } from '@core/shared/application';
|
||||
import { Result } from '@core/shared/application/Result';
|
||||
|
||||
describe('ReviewProtestUseCase', () => {
|
||||
let useCase: ReviewProtestUseCase;
|
||||
let protestRepository: { findById: Mock; update: Mock };
|
||||
let raceRepository: { findById: Mock };
|
||||
let leagueMembershipRepository: { getLeagueMembers: Mock };
|
||||
let logger: { debug: Mock; info: Mock; warn: Mock; error: Mock };
|
||||
let output: UseCaseOutputPort<ReviewProtestResult> & { present: Mock };
|
||||
|
||||
beforeEach(() => {
|
||||
protestRepository = { findById: vi.fn(), update: vi.fn() };
|
||||
raceRepository = { findById: vi.fn() };
|
||||
leagueMembershipRepository = { getLeagueMembers: vi.fn() };
|
||||
logger = {
|
||||
debug: vi.fn(),
|
||||
info: vi.fn(),
|
||||
warn: vi.fn(),
|
||||
error: vi.fn(),
|
||||
};
|
||||
output = { present: vi.fn() } as unknown as UseCaseOutputPort<ReviewProtestResult> & { present: Mock };
|
||||
useCase = new ReviewProtestUseCase(
|
||||
protestRepository as unknown as IProtestRepository,
|
||||
raceRepository as unknown as IRaceRepository,
|
||||
leagueMembershipRepository as unknown as ILeagueMembershipRepository,
|
||||
logger as unknown as Logger,
|
||||
output,
|
||||
);
|
||||
});
|
||||
|
||||
it('should return protest not found error', async () => {
|
||||
protestRepository.findById.mockResolvedValue(null);
|
||||
|
||||
const result = await useCase.execute({
|
||||
const input: ReviewProtestInput = {
|
||||
protestId: 'protest-1',
|
||||
stewardId: 'steward-1',
|
||||
decision: 'uphold',
|
||||
decisionNotes: 'Notes',
|
||||
});
|
||||
};
|
||||
|
||||
const result = await useCase.execute(input);
|
||||
|
||||
expect(result.isErr()).toBe(true);
|
||||
expect(result.unwrapErr()).toEqual({ code: 'PROTEST_NOT_FOUND' });
|
||||
const error = result.unwrapErr() as ApplicationErrorCode<ReviewProtestErrorCode, { message: string }>;
|
||||
expect(error).toEqual({
|
||||
code: 'PROTEST_NOT_FOUND',
|
||||
details: { message: 'Protest not found' },
|
||||
});
|
||||
expect(output.present).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should return race not found error', async () => {
|
||||
@@ -40,15 +61,22 @@ describe('ReviewProtestUseCase', () => {
|
||||
protestRepository.findById.mockResolvedValue(mockProtest);
|
||||
raceRepository.findById.mockResolvedValue(null);
|
||||
|
||||
const result = await useCase.execute({
|
||||
const input: ReviewProtestInput = {
|
||||
protestId: 'protest-1',
|
||||
stewardId: 'steward-1',
|
||||
decision: 'uphold',
|
||||
decisionNotes: 'Notes',
|
||||
});
|
||||
};
|
||||
|
||||
const result = await useCase.execute(input);
|
||||
|
||||
expect(result.isErr()).toBe(true);
|
||||
expect(result.unwrapErr()).toEqual({ code: 'RACE_NOT_FOUND' });
|
||||
const error = result.unwrapErr() as ApplicationErrorCode<ReviewProtestErrorCode, { message: string }>;
|
||||
expect(error).toEqual({
|
||||
code: 'RACE_NOT_FOUND',
|
||||
details: { message: 'Race not found' },
|
||||
});
|
||||
expect(output.present).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should return not league admin error', async () => {
|
||||
@@ -58,19 +86,26 @@ describe('ReviewProtestUseCase', () => {
|
||||
raceRepository.findById.mockResolvedValue(mockRace);
|
||||
leagueMembershipRepository.getLeagueMembers.mockResolvedValue([]);
|
||||
|
||||
const result = await useCase.execute({
|
||||
const input: ReviewProtestInput = {
|
||||
protestId: 'protest-1',
|
||||
stewardId: 'steward-1',
|
||||
decision: 'uphold',
|
||||
decisionNotes: 'Notes',
|
||||
});
|
||||
};
|
||||
|
||||
const result = await useCase.execute(input);
|
||||
|
||||
expect(result.isErr()).toBe(true);
|
||||
expect(result.unwrapErr()).toEqual({ code: 'NOT_LEAGUE_ADMIN' });
|
||||
const error = result.unwrapErr() as ApplicationErrorCode<ReviewProtestErrorCode, { message: string }>;
|
||||
expect(error).toEqual({
|
||||
code: 'NOT_LEAGUE_ADMIN',
|
||||
details: { message: 'Only league owners and admins can review protests' },
|
||||
});
|
||||
expect(output.present).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should uphold protest successfully', async () => {
|
||||
const mockProtest = { raceId: 'race-1', uphold: vi.fn().mockReturnValue({}), dismiss: vi.fn() };
|
||||
const mockProtest = { id: 'protest-1', raceId: 'race-1', uphold: vi.fn().mockReturnValue({}), dismiss: vi.fn() };
|
||||
const mockRace = { leagueId: 'league-1' };
|
||||
const memberships = [{ driverId: 'steward-1', status: 'active', role: 'admin' }];
|
||||
protestRepository.findById.mockResolvedValue(mockProtest);
|
||||
@@ -78,20 +113,29 @@ describe('ReviewProtestUseCase', () => {
|
||||
leagueMembershipRepository.getLeagueMembers.mockResolvedValue(memberships);
|
||||
protestRepository.update.mockResolvedValue(undefined);
|
||||
|
||||
const result = await useCase.execute({
|
||||
const input: ReviewProtestInput = {
|
||||
protestId: 'protest-1',
|
||||
stewardId: 'steward-1',
|
||||
decision: 'uphold',
|
||||
decisionNotes: 'Notes',
|
||||
});
|
||||
};
|
||||
|
||||
const result = await useCase.execute(input);
|
||||
|
||||
expect(result.isOk()).toBe(true);
|
||||
expect(result.unwrap()).toBeUndefined();
|
||||
expect(protestRepository.update).toHaveBeenCalledWith(mockProtest.uphold());
|
||||
expect(output.present).toHaveBeenCalledTimes(1);
|
||||
const presented = output.present.mock.calls[0]![0] as ReviewProtestResult;
|
||||
expect(presented).toEqual({
|
||||
leagueId: 'league-1',
|
||||
protestId: 'protest-1',
|
||||
status: 'upheld',
|
||||
});
|
||||
});
|
||||
|
||||
it('should dismiss protest successfully', async () => {
|
||||
const mockProtest = { raceId: 'race-1', uphold: vi.fn(), dismiss: vi.fn().mockReturnValue({}) };
|
||||
const mockProtest = { id: 'protest-1', raceId: 'race-1', uphold: vi.fn(), dismiss: vi.fn().mockReturnValue({}) };
|
||||
const mockRace = { leagueId: 'league-1' };
|
||||
const memberships = [{ driverId: 'steward-1', status: 'active', role: 'owner' }];
|
||||
protestRepository.findById.mockResolvedValue(mockProtest);
|
||||
@@ -99,15 +143,49 @@ describe('ReviewProtestUseCase', () => {
|
||||
leagueMembershipRepository.getLeagueMembers.mockResolvedValue(memberships);
|
||||
protestRepository.update.mockResolvedValue(undefined);
|
||||
|
||||
const result = await useCase.execute({
|
||||
const input: ReviewProtestInput = {
|
||||
protestId: 'protest-1',
|
||||
stewardId: 'steward-1',
|
||||
decision: 'dismiss',
|
||||
decisionNotes: 'Notes',
|
||||
});
|
||||
};
|
||||
|
||||
const result = await useCase.execute(input);
|
||||
|
||||
expect(result.isOk()).toBe(true);
|
||||
expect(result.unwrap()).toBeUndefined();
|
||||
expect(protestRepository.update).toHaveBeenCalledWith(mockProtest.dismiss());
|
||||
expect(output.present).toHaveBeenCalledTimes(1);
|
||||
const presented = output.present.mock.calls[0]![0] as ReviewProtestResult;
|
||||
expect(presented).toEqual({
|
||||
leagueId: 'league-1',
|
||||
protestId: 'protest-1',
|
||||
status: 'dismissed',
|
||||
});
|
||||
});
|
||||
|
||||
it('should return repository error when update throws', async () => {
|
||||
const mockProtest = { id: 'protest-1', raceId: 'race-1', uphold: vi.fn().mockReturnValue({}), dismiss: vi.fn() };
|
||||
const mockRace = { leagueId: 'league-1' };
|
||||
const memberships = [{ driverId: 'steward-1', status: 'active', role: 'admin' }];
|
||||
protestRepository.findById.mockResolvedValue(mockProtest);
|
||||
raceRepository.findById.mockResolvedValue(mockRace);
|
||||
leagueMembershipRepository.getLeagueMembers.mockResolvedValue(memberships);
|
||||
protestRepository.update.mockRejectedValue(new Error('DB error'));
|
||||
|
||||
const input: ReviewProtestInput = {
|
||||
protestId: 'protest-1',
|
||||
stewardId: 'steward-1',
|
||||
decision: 'uphold',
|
||||
decisionNotes: 'Notes',
|
||||
};
|
||||
|
||||
const result = await useCase.execute(input);
|
||||
|
||||
expect(result.isErr()).toBe(true);
|
||||
const error = result.unwrapErr() as ApplicationErrorCode<ReviewProtestErrorCode, { message: string }>;
|
||||
expect(error.code).toBe('REPOSITORY_ERROR');
|
||||
expect(error.details?.message).toBe('Failed to review protest');
|
||||
expect(output.present).not.toHaveBeenCalled();
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user