refactor racing use cases

This commit is contained in:
2025-12-21 00:43:42 +01:00
parent e9d6f90bb2
commit c12656d671
308 changed files with 14401 additions and 7419 deletions

View File

@@ -1,5 +1,7 @@
import { describe, it, expect, vi } from 'vitest';
import { UpdateTeamUseCase } from './UpdateTeamUseCase';
import { UpdateTeamUseCase, type UpdateTeamInput, type UpdateTeamResult, type UpdateTeamErrorCode } from './UpdateTeamUseCase';
import type { UseCaseOutputPort } from '@core/shared/application/UseCaseOutputPort';
import type { ApplicationErrorCode } from '@core/shared/errors/ApplicationErrorCode';
import type { ITeamRepository } from '../../domain/repositories/ITeamRepository';
import type { ITeamMembershipRepository } from '../../domain/repositories/ITeamMembershipRepository';
@@ -9,9 +11,11 @@ describe('UpdateTeamUseCase', () => {
role: 'owner',
};
const mockUpdatedTeam = { id: 'team-1' };
const mockTeam = {
id: 'team-1',
update: vi.fn().mockReturnValue({}),
update: vi.fn().mockReturnValue(mockUpdatedTeam),
};
const mockTeamRepository = {
@@ -23,24 +27,36 @@ describe('UpdateTeamUseCase', () => {
getMembership: vi.fn().mockResolvedValue(mockMembership),
} as unknown as ITeamMembershipRepository;
const useCase = new UpdateTeamUseCase(mockTeamRepository, mockMembershipRepository);
const present = vi.fn<(data: UpdateTeamResult) => void>();
const output: UseCaseOutputPort<UpdateTeamResult> & { present: typeof present } = {
present,
};
const command = {
const useCase = new UpdateTeamUseCase(mockTeamRepository, mockMembershipRepository, output);
const command: UpdateTeamInput = {
teamId: 'team-1',
updates: { name: 'New Name', tag: 'NEW' },
updatedBy: 'user-1',
updates: { name: 'New Name', tag: 'NEW' },
};
const result = await useCase.execute(command);
expect(result.isOk()).toBe(true);
expect(result.unwrap()).toBeUndefined();
expect(mockMembershipRepository.getMembership).toHaveBeenCalledWith('team-1', 'user-1');
expect(mockTeamRepository.findById).toHaveBeenCalledWith('team-1');
expect(mockTeam.update).toHaveBeenCalledWith({ name: 'New Name', tag: 'NEW' });
expect(mockTeamRepository.update).toHaveBeenCalledWith({});
expect(mockTeamRepository.update).toHaveBeenCalledWith(mockUpdatedTeam);
expect(present).toHaveBeenCalledTimes(1);
const firstCall = present.mock.calls[0];
expect(firstCall).toBeDefined();
const presented = firstCall![0] as UpdateTeamResult;
expect(presented.team).toBe(mockUpdatedTeam);
});
it('returns error if insufficient permissions', async () => {
it('returns permission denied error if insufficient permissions', async () => {
const mockMembership = {
role: 'member',
};
@@ -49,18 +65,26 @@ describe('UpdateTeamUseCase', () => {
getMembership: vi.fn().mockResolvedValue(mockMembership),
} as unknown as ITeamMembershipRepository;
const useCase = new UpdateTeamUseCase({} as unknown as ITeamRepository, mockMembershipRepository);
const present = vi.fn<(data: UpdateTeamResult) => void>();
const output: UseCaseOutputPort<UpdateTeamResult> & { present: typeof present } = {
present,
};
const command = {
const useCase = new UpdateTeamUseCase({} as unknown as ITeamRepository, mockMembershipRepository, output);
const command: UpdateTeamInput = {
teamId: 'team-1',
updates: { name: 'New Name' },
updatedBy: 'user-1',
updates: { name: 'New Name' },
};
const result = await useCase.execute(command);
expect(result.isErr()).toBe(true);
expect(result.unwrapErr()).toEqual({ code: 'INSUFFICIENT_PERMISSIONS' });
const error = result.unwrapErr() as ApplicationErrorCode<UpdateTeamErrorCode, { message: string }>;
expect(error.code).toBe('PERMISSION_DENIED');
expect(error.details?.message).toBe('User does not have permission to update this team');
expect(present).not.toHaveBeenCalled();
});
it('returns error if team not found', async () => {
@@ -76,17 +100,60 @@ describe('UpdateTeamUseCase', () => {
getMembership: vi.fn().mockResolvedValue(mockMembership),
} as unknown as ITeamMembershipRepository;
const useCase = new UpdateTeamUseCase(mockTeamRepository, mockMembershipRepository);
const present = vi.fn<(data: UpdateTeamResult) => void>();
const output: UseCaseOutputPort<UpdateTeamResult> & { present: typeof present } = {
present,
};
const command = {
const useCase = new UpdateTeamUseCase(mockTeamRepository, mockMembershipRepository, output);
const command: UpdateTeamInput = {
teamId: 'team-1',
updates: { name: 'New Name' },
updatedBy: 'user-1',
updates: { name: 'New Name' },
};
const result = await useCase.execute(command);
expect(result.isErr()).toBe(true);
expect(result.unwrapErr()).toEqual({ code: 'TEAM_NOT_FOUND' });
const error = result.unwrapErr() as ApplicationErrorCode<UpdateTeamErrorCode, { message: string }>;
expect(error.code).toBe('TEAM_NOT_FOUND');
expect(error.details?.message).toBe('Team not found');
expect(present).not.toHaveBeenCalled();
});
});
it('returns repository error on unexpected failure', async () => {
const mockMembership = {
role: 'owner',
};
const mockTeamRepository = {
findById: vi.fn().mockRejectedValue(new Error('db error')),
} as unknown as ITeamRepository;
const mockMembershipRepository = {
getMembership: vi.fn().mockResolvedValue(mockMembership),
} as unknown as ITeamMembershipRepository;
const present = vi.fn<(data: UpdateTeamResult) => void>();
const output: UseCaseOutputPort<UpdateTeamResult> & { present: typeof present } = {
present,
};
const useCase = new UpdateTeamUseCase(mockTeamRepository, mockMembershipRepository, output);
const command: UpdateTeamInput = {
teamId: 'team-1',
updatedBy: 'user-1',
updates: { name: 'New Name' },
};
const result = await useCase.execute(command);
expect(result.isErr()).toBe(true);
const error = result.unwrapErr() as ApplicationErrorCode<UpdateTeamErrorCode, { message: string }>;
expect(error.code).toBe('REPOSITORY_ERROR');
expect(error.details?.message).toBe('db error');
expect(present).not.toHaveBeenCalled();
});
});