integration tests
Some checks failed
Contract Testing / contract-tests (pull_request) Failing after 4m51s
Contract Testing / contract-snapshot (pull_request) Has been skipped

This commit is contained in:
2026-01-22 17:29:06 +01:00
parent f61ebda9b7
commit 597bb48248
68 changed files with 11832 additions and 3498 deletions

View File

@@ -0,0 +1,175 @@
/**
* In-Memory Health Event Publisher
*
* Tracks health-related events for testing purposes.
* This publisher allows verification of event emission patterns
* without requiring external event bus infrastructure.
*/
import {
HealthEventPublisher,
HealthCheckCompletedEvent,
HealthCheckFailedEvent,
HealthCheckTimeoutEvent,
ConnectedEvent,
DisconnectedEvent,
DegradedEvent,
CheckingEvent,
} from '../../../core/health/ports/HealthEventPublisher';
export interface HealthCheckCompletedEventWithType {
type: 'HealthCheckCompleted';
healthy: boolean;
responseTime: number;
timestamp: Date;
endpoint?: string;
}
export interface HealthCheckFailedEventWithType {
type: 'HealthCheckFailed';
error: string;
timestamp: Date;
endpoint?: string;
}
export interface HealthCheckTimeoutEventWithType {
type: 'HealthCheckTimeout';
timestamp: Date;
endpoint?: string;
}
export interface ConnectedEventWithType {
type: 'Connected';
timestamp: Date;
responseTime: number;
}
export interface DisconnectedEventWithType {
type: 'Disconnected';
timestamp: Date;
consecutiveFailures: number;
}
export interface DegradedEventWithType {
type: 'Degraded';
timestamp: Date;
reliability: number;
}
export interface CheckingEventWithType {
type: 'Checking';
timestamp: Date;
}
export type HealthEvent =
| HealthCheckCompletedEventWithType
| HealthCheckFailedEventWithType
| HealthCheckTimeoutEventWithType
| ConnectedEventWithType
| DisconnectedEventWithType
| DegradedEventWithType
| CheckingEventWithType;
export class InMemoryHealthEventPublisher implements HealthEventPublisher {
private events: HealthEvent[] = [];
private shouldFail: boolean = false;
/**
* Publish a health check completed event
*/
async publishHealthCheckCompleted(event: HealthCheckCompletedEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.events.push({ type: 'HealthCheckCompleted', ...event });
}
/**
* Publish a health check failed event
*/
async publishHealthCheckFailed(event: HealthCheckFailedEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.events.push({ type: 'HealthCheckFailed', ...event });
}
/**
* Publish a health check timeout event
*/
async publishHealthCheckTimeout(event: HealthCheckTimeoutEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.events.push({ type: 'HealthCheckTimeout', ...event });
}
/**
* Publish a connected event
*/
async publishConnected(event: ConnectedEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.events.push({ type: 'Connected', ...event });
}
/**
* Publish a disconnected event
*/
async publishDisconnected(event: DisconnectedEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.events.push({ type: 'Disconnected', ...event });
}
/**
* Publish a degraded event
*/
async publishDegraded(event: DegradedEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.events.push({ type: 'Degraded', ...event });
}
/**
* Publish a checking event
*/
async publishChecking(event: CheckingEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.events.push({ type: 'Checking', ...event });
}
/**
* Get all published events
*/
getEvents(): HealthEvent[] {
return [...this.events];
}
/**
* Get events by type
*/
getEventsByType<T extends HealthEvent['type']>(type: T): Extract<HealthEvent, { type: T }>[] {
return this.events.filter((event): event is Extract<HealthEvent, { type: T }> => event.type === type);
}
/**
* Get the count of events
*/
getEventCount(): number {
return this.events.length;
}
/**
* Get the count of events by type
*/
getEventCountByType(type: HealthEvent['type']): number {
return this.events.filter(event => event.type === type).length;
}
/**
* Clear all published events
*/
clear(): void {
this.events = [];
this.shouldFail = false;
}
/**
* Configure the publisher to fail on publish
*/
setShouldFail(shouldFail: boolean): void {
this.shouldFail = shouldFail;
}
}

View File

@@ -0,0 +1,197 @@
/**
* In-Memory Health Check Adapter
*
* Simulates API health check responses for testing purposes.
* This adapter allows controlled testing of health check scenarios
* without making actual HTTP requests.
*/
import {
HealthCheckQuery,
ConnectionStatus,
ConnectionHealth,
HealthCheckResult,
} from '../../../../core/health/ports/HealthCheckQuery';
export interface HealthCheckResponse {
healthy: boolean;
responseTime: number;
error?: string;
timestamp: Date;
}
export class InMemoryHealthCheckAdapter implements HealthCheckQuery {
private responses: Map<string, HealthCheckResponse> = new Map();
public shouldFail: boolean = false;
public failError: string = 'Network error';
private responseTime: number = 50;
private health: ConnectionHealth = {
status: 'disconnected',
lastCheck: null,
lastSuccess: null,
lastFailure: null,
consecutiveFailures: 0,
totalRequests: 0,
successfulRequests: 0,
failedRequests: 0,
averageResponseTime: 0,
};
/**
* Configure the adapter to return a specific response
*/
configureResponse(endpoint: string, response: HealthCheckResponse): void {
this.responses.set(endpoint, response);
}
/**
* Configure the adapter to fail all requests
*/
setShouldFail(shouldFail: boolean, error?: string): void {
this.shouldFail = shouldFail;
if (error) {
this.failError = error;
}
}
/**
* Set the response time for health checks
*/
setResponseTime(time: number): void {
this.responseTime = time;
}
/**
* Perform a health check against an endpoint
*/
async performHealthCheck(): Promise<HealthCheckResult> {
// Simulate network delay
await new Promise(resolve => setTimeout(resolve, this.responseTime));
if (this.shouldFail) {
this.recordFailure(this.failError);
return {
healthy: false,
responseTime: this.responseTime,
error: this.failError,
timestamp: new Date(),
};
}
// Default successful response
this.recordSuccess(this.responseTime);
return {
healthy: true,
responseTime: this.responseTime,
timestamp: new Date(),
};
}
/**
* Get current connection status
*/
getStatus(): ConnectionStatus {
return this.health.status;
}
/**
* Get detailed health information
*/
getHealth(): ConnectionHealth {
return { ...this.health };
}
/**
* Get reliability percentage
*/
getReliability(): number {
if (this.health.totalRequests === 0) return 0;
return (this.health.successfulRequests / this.health.totalRequests) * 100;
}
/**
* Check if API is currently available
*/
isAvailable(): boolean {
return this.health.status === 'connected' || this.health.status === 'degraded';
}
/**
* Record a successful health check
*/
private recordSuccess(responseTime: number): void {
this.health.totalRequests++;
this.health.successfulRequests++;
this.health.consecutiveFailures = 0;
this.health.lastSuccess = new Date();
this.health.lastCheck = new Date();
// Update average response time
const total = this.health.successfulRequests;
this.health.averageResponseTime =
((this.health.averageResponseTime * (total - 1)) + responseTime) / total;
this.updateStatus();
}
/**
* Record a failed health check
*/
private recordFailure(error: string): void {
this.health.totalRequests++;
this.health.failedRequests++;
this.health.consecutiveFailures++;
this.health.lastFailure = new Date();
this.health.lastCheck = new Date();
this.updateStatus();
}
/**
* Update connection status based on current metrics
*/
private updateStatus(): void {
const reliability = this.health.totalRequests > 0
? this.health.successfulRequests / this.health.totalRequests
: 0;
// More nuanced status determination
if (this.health.totalRequests === 0) {
// No requests yet - don't assume disconnected
this.health.status = 'checking';
} else if (this.health.consecutiveFailures >= 3) {
// Multiple consecutive failures indicates real connectivity issue
this.health.status = 'disconnected';
} else if (reliability < 0.7 && this.health.totalRequests >= 5) {
// Only degrade if we have enough samples and reliability is low
this.health.status = 'degraded';
} else if (reliability >= 0.7 || this.health.successfulRequests > 0) {
// If we have any successes, we're connected
this.health.status = 'connected';
} else {
// Default to checking if uncertain
this.health.status = 'checking';
}
}
/**
* Clear all configured responses and settings
*/
clear(): void {
this.responses.clear();
this.shouldFail = false;
this.failError = 'Network error';
this.responseTime = 50;
this.health = {
status: 'disconnected',
lastCheck: null,
lastSuccess: null,
lastFailure: null,
consecutiveFailures: 0,
totalRequests: 0,
successfulRequests: 0,
failedRequests: 0,
averageResponseTime: 0,
};
}
}

View File

@@ -0,0 +1,70 @@
/**
* Infrastructure Adapter: InMemoryLeaderboardsEventPublisher
*
* In-memory implementation of LeaderboardsEventPublisher.
* Stores events in arrays for testing purposes.
*/
import {
LeaderboardsEventPublisher,
GlobalLeaderboardsAccessedEvent,
DriverRankingsAccessedEvent,
TeamRankingsAccessedEvent,
LeaderboardsErrorEvent,
} from '../../../core/leaderboards/application/ports/LeaderboardsEventPublisher';
export class InMemoryLeaderboardsEventPublisher implements LeaderboardsEventPublisher {
private globalLeaderboardsAccessedEvents: GlobalLeaderboardsAccessedEvent[] = [];
private driverRankingsAccessedEvents: DriverRankingsAccessedEvent[] = [];
private teamRankingsAccessedEvents: TeamRankingsAccessedEvent[] = [];
private leaderboardsErrorEvents: LeaderboardsErrorEvent[] = [];
private shouldFail: boolean = false;
async publishGlobalLeaderboardsAccessed(event: GlobalLeaderboardsAccessedEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.globalLeaderboardsAccessedEvents.push(event);
}
async publishDriverRankingsAccessed(event: DriverRankingsAccessedEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.driverRankingsAccessedEvents.push(event);
}
async publishTeamRankingsAccessed(event: TeamRankingsAccessedEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.teamRankingsAccessedEvents.push(event);
}
async publishLeaderboardsError(event: LeaderboardsErrorEvent): Promise<void> {
if (this.shouldFail) throw new Error('Event publisher failed');
this.leaderboardsErrorEvents.push(event);
}
getGlobalLeaderboardsAccessedEventCount(): number {
return this.globalLeaderboardsAccessedEvents.length;
}
getDriverRankingsAccessedEventCount(): number {
return this.driverRankingsAccessedEvents.length;
}
getTeamRankingsAccessedEventCount(): number {
return this.teamRankingsAccessedEvents.length;
}
getLeaderboardsErrorEventCount(): number {
return this.leaderboardsErrorEvents.length;
}
clear(): void {
this.globalLeaderboardsAccessedEvents = [];
this.driverRankingsAccessedEvents = [];
this.teamRankingsAccessedEvents = [];
this.leaderboardsErrorEvents = [];
this.shouldFail = false;
}
setShouldFail(shouldFail: boolean): void {
this.shouldFail = shouldFail;
}
}

View File

@@ -0,0 +1,44 @@
/**
* Infrastructure Adapter: InMemoryLeaderboardsRepository
*
* In-memory implementation of LeaderboardsRepository.
* Stores data in a Map structure.
*/
import {
LeaderboardsRepository,
LeaderboardDriverData,
LeaderboardTeamData,
} from '../../../../core/leaderboards/application/ports/LeaderboardsRepository';
export class InMemoryLeaderboardsRepository implements LeaderboardsRepository {
private drivers: Map<string, LeaderboardDriverData> = new Map();
private teams: Map<string, LeaderboardTeamData> = new Map();
async findAllDrivers(): Promise<LeaderboardDriverData[]> {
return Array.from(this.drivers.values());
}
async findAllTeams(): Promise<LeaderboardTeamData[]> {
return Array.from(this.teams.values());
}
async findDriversByTeamId(teamId: string): Promise<LeaderboardDriverData[]> {
return Array.from(this.drivers.values()).filter(
(driver) => driver.teamId === teamId,
);
}
addDriver(driver: LeaderboardDriverData): void {
this.drivers.set(driver.id, driver);
}
addTeam(team: LeaderboardTeamData): void {
this.teams.set(team.id, team);
}
clear(): void {
this.drivers.clear();
this.teams.clear();
}
}

View File

@@ -0,0 +1,69 @@
import {
LeagueEventPublisher,
LeagueCreatedEvent,
LeagueUpdatedEvent,
LeagueDeletedEvent,
LeagueAccessedEvent,
} from '../../../core/leagues/application/ports/LeagueEventPublisher';
export class InMemoryLeagueEventPublisher implements LeagueEventPublisher {
private leagueCreatedEvents: LeagueCreatedEvent[] = [];
private leagueUpdatedEvents: LeagueUpdatedEvent[] = [];
private leagueDeletedEvents: LeagueDeletedEvent[] = [];
private leagueAccessedEvents: LeagueAccessedEvent[] = [];
async emitLeagueCreated(event: LeagueCreatedEvent): Promise<void> {
this.leagueCreatedEvents.push(event);
}
async emitLeagueUpdated(event: LeagueUpdatedEvent): Promise<void> {
this.leagueUpdatedEvents.push(event);
}
async emitLeagueDeleted(event: LeagueDeletedEvent): Promise<void> {
this.leagueDeletedEvents.push(event);
}
async emitLeagueAccessed(event: LeagueAccessedEvent): Promise<void> {
this.leagueAccessedEvents.push(event);
}
getLeagueCreatedEventCount(): number {
return this.leagueCreatedEvents.length;
}
getLeagueUpdatedEventCount(): number {
return this.leagueUpdatedEvents.length;
}
getLeagueDeletedEventCount(): number {
return this.leagueDeletedEvents.length;
}
getLeagueAccessedEventCount(): number {
return this.leagueAccessedEvents.length;
}
clear(): void {
this.leagueCreatedEvents = [];
this.leagueUpdatedEvents = [];
this.leagueDeletedEvents = [];
this.leagueAccessedEvents = [];
}
getLeagueCreatedEvents(): LeagueCreatedEvent[] {
return [...this.leagueCreatedEvents];
}
getLeagueUpdatedEvents(): LeagueUpdatedEvent[] {
return [...this.leagueUpdatedEvents];
}
getLeagueDeletedEvents(): LeagueDeletedEvent[] {
return [...this.leagueDeletedEvents];
}
getLeagueAccessedEvents(): LeagueAccessedEvent[] {
return [...this.leagueAccessedEvents];
}
}

View File

@@ -1,64 +1,310 @@
import {
DashboardRepository,
DriverData,
RaceData,
LeagueStandingData,
ActivityData,
FriendData,
} from '../../../../core/dashboard/application/ports/DashboardRepository';
LeagueRepository,
LeagueData,
LeagueStats,
LeagueFinancials,
LeagueStewardingMetrics,
LeaguePerformanceMetrics,
LeagueRatingMetrics,
LeagueTrendMetrics,
LeagueSuccessRateMetrics,
LeagueResolutionTimeMetrics,
LeagueComplexSuccessRateMetrics,
LeagueComplexResolutionTimeMetrics,
} from '../../../../core/leagues/application/ports/LeagueRepository';
export class InMemoryLeagueRepository implements DashboardRepository {
private drivers: Map<string, DriverData> = new Map();
private upcomingRaces: Map<string, RaceData[]> = new Map();
private leagueStandings: Map<string, LeagueStandingData[]> = new Map();
private recentActivity: Map<string, ActivityData[]> = new Map();
private friends: Map<string, FriendData[]> = new Map();
export class InMemoryLeagueRepository implements LeagueRepository {
private leagues: Map<string, LeagueData> = new Map();
private leagueStats: Map<string, LeagueStats> = new Map();
private leagueFinancials: Map<string, LeagueFinancials> = new Map();
private leagueStewardingMetrics: Map<string, LeagueStewardingMetrics> = new Map();
private leaguePerformanceMetrics: Map<string, LeaguePerformanceMetrics> = new Map();
private leagueRatingMetrics: Map<string, LeagueRatingMetrics> = new Map();
private leagueTrendMetrics: Map<string, LeagueTrendMetrics> = new Map();
private leagueSuccessRateMetrics: Map<string, LeagueSuccessRateMetrics> = new Map();
private leagueResolutionTimeMetrics: Map<string, LeagueResolutionTimeMetrics> = new Map();
private leagueComplexSuccessRateMetrics: Map<string, LeagueComplexSuccessRateMetrics> = new Map();
private leagueComplexResolutionTimeMetrics: Map<string, LeagueComplexResolutionTimeMetrics> = new Map();
async findDriverById(driverId: string): Promise<DriverData | null> {
return this.drivers.get(driverId) || null;
async create(league: LeagueData): Promise<LeagueData> {
this.leagues.set(league.id, league);
return league;
}
async getUpcomingRaces(driverId: string): Promise<RaceData[]> {
return this.upcomingRaces.get(driverId) || [];
async findById(id: string): Promise<LeagueData | null> {
return this.leagues.get(id) || null;
}
async getLeagueStandings(driverId: string): Promise<LeagueStandingData[]> {
return this.leagueStandings.get(driverId) || [];
async findByName(name: string): Promise<LeagueData | null> {
for (const league of Array.from(this.leagues.values())) {
if (league.name === name) {
return league;
}
}
return null;
}
async getRecentActivity(driverId: string): Promise<ActivityData[]> {
return this.recentActivity.get(driverId) || [];
async findByOwner(ownerId: string): Promise<LeagueData[]> {
const leagues: LeagueData[] = [];
for (const league of Array.from(this.leagues.values())) {
if (league.ownerId === ownerId) {
leagues.push(league);
}
}
return leagues;
}
async getFriends(driverId: string): Promise<FriendData[]> {
return this.friends.get(driverId) || [];
async search(query: string): Promise<LeagueData[]> {
const results: LeagueData[] = [];
const lowerQuery = query.toLowerCase();
for (const league of Array.from(this.leagues.values())) {
if (
league.name.toLowerCase().includes(lowerQuery) ||
league.description?.toLowerCase().includes(lowerQuery)
) {
results.push(league);
}
}
return results;
}
addDriver(driver: DriverData): void {
this.drivers.set(driver.id, driver);
async update(id: string, updates: Partial<LeagueData>): Promise<LeagueData> {
const league = this.leagues.get(id);
if (!league) {
throw new Error(`League with id ${id} not found`);
}
const updated = { ...league, ...updates };
this.leagues.set(id, updated);
return updated;
}
addUpcomingRaces(driverId: string, races: RaceData[]): void {
this.upcomingRaces.set(driverId, races);
async delete(id: string): Promise<void> {
this.leagues.delete(id);
this.leagueStats.delete(id);
this.leagueFinancials.delete(id);
this.leagueStewardingMetrics.delete(id);
this.leaguePerformanceMetrics.delete(id);
this.leagueRatingMetrics.delete(id);
this.leagueTrendMetrics.delete(id);
this.leagueSuccessRateMetrics.delete(id);
this.leagueResolutionTimeMetrics.delete(id);
this.leagueComplexSuccessRateMetrics.delete(id);
this.leagueComplexResolutionTimeMetrics.delete(id);
}
addLeagueStandings(driverId: string, standings: LeagueStandingData[]): void {
this.leagueStandings.set(driverId, standings);
async getStats(leagueId: string): Promise<LeagueStats> {
return this.leagueStats.get(leagueId) || this.createDefaultStats(leagueId);
}
addRecentActivity(driverId: string, activities: ActivityData[]): void {
this.recentActivity.set(driverId, activities);
async updateStats(leagueId: string, stats: LeagueStats): Promise<LeagueStats> {
this.leagueStats.set(leagueId, stats);
return stats;
}
addFriends(driverId: string, friends: FriendData[]): void {
this.friends.set(driverId, friends);
async getFinancials(leagueId: string): Promise<LeagueFinancials> {
return this.leagueFinancials.get(leagueId) || this.createDefaultFinancials(leagueId);
}
async updateFinancials(leagueId: string, financials: LeagueFinancials): Promise<LeagueFinancials> {
this.leagueFinancials.set(leagueId, financials);
return financials;
}
async getStewardingMetrics(leagueId: string): Promise<LeagueStewardingMetrics> {
return this.leagueStewardingMetrics.get(leagueId) || this.createDefaultStewardingMetrics(leagueId);
}
async updateStewardingMetrics(leagueId: string, metrics: LeagueStewardingMetrics): Promise<LeagueStewardingMetrics> {
this.leagueStewardingMetrics.set(leagueId, metrics);
return metrics;
}
async getPerformanceMetrics(leagueId: string): Promise<LeaguePerformanceMetrics> {
return this.leaguePerformanceMetrics.get(leagueId) || this.createDefaultPerformanceMetrics(leagueId);
}
async updatePerformanceMetrics(leagueId: string, metrics: LeaguePerformanceMetrics): Promise<LeaguePerformanceMetrics> {
this.leaguePerformanceMetrics.set(leagueId, metrics);
return metrics;
}
async getRatingMetrics(leagueId: string): Promise<LeagueRatingMetrics> {
return this.leagueRatingMetrics.get(leagueId) || this.createDefaultRatingMetrics(leagueId);
}
async updateRatingMetrics(leagueId: string, metrics: LeagueRatingMetrics): Promise<LeagueRatingMetrics> {
this.leagueRatingMetrics.set(leagueId, metrics);
return metrics;
}
async getTrendMetrics(leagueId: string): Promise<LeagueTrendMetrics> {
return this.leagueTrendMetrics.get(leagueId) || this.createDefaultTrendMetrics(leagueId);
}
async updateTrendMetrics(leagueId: string, metrics: LeagueTrendMetrics): Promise<LeagueTrendMetrics> {
this.leagueTrendMetrics.set(leagueId, metrics);
return metrics;
}
async getSuccessRateMetrics(leagueId: string): Promise<LeagueSuccessRateMetrics> {
return this.leagueSuccessRateMetrics.get(leagueId) || this.createDefaultSuccessRateMetrics(leagueId);
}
async updateSuccessRateMetrics(leagueId: string, metrics: LeagueSuccessRateMetrics): Promise<LeagueSuccessRateMetrics> {
this.leagueSuccessRateMetrics.set(leagueId, metrics);
return metrics;
}
async getResolutionTimeMetrics(leagueId: string): Promise<LeagueResolutionTimeMetrics> {
return this.leagueResolutionTimeMetrics.get(leagueId) || this.createDefaultResolutionTimeMetrics(leagueId);
}
async updateResolutionTimeMetrics(leagueId: string, metrics: LeagueResolutionTimeMetrics): Promise<LeagueResolutionTimeMetrics> {
this.leagueResolutionTimeMetrics.set(leagueId, metrics);
return metrics;
}
async getComplexSuccessRateMetrics(leagueId: string): Promise<LeagueComplexSuccessRateMetrics> {
return this.leagueComplexSuccessRateMetrics.get(leagueId) || this.createDefaultComplexSuccessRateMetrics(leagueId);
}
async updateComplexSuccessRateMetrics(leagueId: string, metrics: LeagueComplexSuccessRateMetrics): Promise<LeagueComplexSuccessRateMetrics> {
this.leagueComplexSuccessRateMetrics.set(leagueId, metrics);
return metrics;
}
async getComplexResolutionTimeMetrics(leagueId: string): Promise<LeagueComplexResolutionTimeMetrics> {
return this.leagueComplexResolutionTimeMetrics.get(leagueId) || this.createDefaultComplexResolutionTimeMetrics(leagueId);
}
async updateComplexResolutionTimeMetrics(leagueId: string, metrics: LeagueComplexResolutionTimeMetrics): Promise<LeagueComplexResolutionTimeMetrics> {
this.leagueComplexResolutionTimeMetrics.set(leagueId, metrics);
return metrics;
}
clear(): void {
this.drivers.clear();
this.upcomingRaces.clear();
this.leagueStandings.clear();
this.recentActivity.clear();
this.friends.clear();
this.leagues.clear();
this.leagueStats.clear();
this.leagueFinancials.clear();
this.leagueStewardingMetrics.clear();
this.leaguePerformanceMetrics.clear();
this.leagueRatingMetrics.clear();
this.leagueTrendMetrics.clear();
this.leagueSuccessRateMetrics.clear();
this.leagueResolutionTimeMetrics.clear();
this.leagueComplexSuccessRateMetrics.clear();
this.leagueComplexResolutionTimeMetrics.clear();
}
private createDefaultStats(leagueId: string): LeagueStats {
return {
leagueId,
memberCount: 1,
raceCount: 0,
sponsorCount: 0,
prizePool: 0,
rating: 0,
reviewCount: 0,
};
}
private createDefaultFinancials(leagueId: string): LeagueFinancials {
return {
leagueId,
walletBalance: 0,
totalRevenue: 0,
totalFees: 0,
pendingPayouts: 0,
netBalance: 0,
};
}
private createDefaultStewardingMetrics(leagueId: string): LeagueStewardingMetrics {
return {
leagueId,
averageResolutionTime: 0,
averageProtestResolutionTime: 0,
averagePenaltyAppealSuccessRate: 0,
averageProtestSuccessRate: 0,
averageStewardingActionSuccessRate: 0,
};
}
private createDefaultPerformanceMetrics(leagueId: string): LeaguePerformanceMetrics {
return {
leagueId,
averageLapTime: 0,
averageFieldSize: 0,
averageIncidentCount: 0,
averagePenaltyCount: 0,
averageProtestCount: 0,
averageStewardingActionCount: 0,
};
}
private createDefaultRatingMetrics(leagueId: string): LeagueRatingMetrics {
return {
leagueId,
overallRating: 0,
ratingTrend: 0,
rankTrend: 0,
pointsTrend: 0,
winRateTrend: 0,
podiumRateTrend: 0,
dnfRateTrend: 0,
};
}
private createDefaultTrendMetrics(leagueId: string): LeagueTrendMetrics {
return {
leagueId,
incidentRateTrend: 0,
penaltyRateTrend: 0,
protestRateTrend: 0,
stewardingActionRateTrend: 0,
stewardingTimeTrend: 0,
protestResolutionTimeTrend: 0,
};
}
private createDefaultSuccessRateMetrics(leagueId: string): LeagueSuccessRateMetrics {
return {
leagueId,
penaltyAppealSuccessRate: 0,
protestSuccessRate: 0,
stewardingActionSuccessRate: 0,
stewardingActionAppealSuccessRate: 0,
stewardingActionPenaltySuccessRate: 0,
stewardingActionProtestSuccessRate: 0,
};
}
private createDefaultResolutionTimeMetrics(leagueId: string): LeagueResolutionTimeMetrics {
return {
leagueId,
averageStewardingTime: 0,
averageProtestResolutionTime: 0,
averageStewardingActionAppealPenaltyProtestResolutionTime: 0,
};
}
private createDefaultComplexSuccessRateMetrics(leagueId: string): LeagueComplexSuccessRateMetrics {
return {
leagueId,
stewardingActionAppealPenaltyProtestSuccessRate: 0,
stewardingActionAppealProtestSuccessRate: 0,
stewardingActionPenaltyProtestSuccessRate: 0,
stewardingActionAppealPenaltyProtestSuccessRate2: 0,
};
}
private createDefaultComplexResolutionTimeMetrics(leagueId: string): LeagueComplexResolutionTimeMetrics {
return {
leagueId,
stewardingActionAppealPenaltyProtestResolutionTime: 0,
stewardingActionAppealProtestResolutionTime: 0,
stewardingActionPenaltyProtestResolutionTime: 0,
stewardingActionAppealPenaltyProtestResolutionTime2: 0,
};
}
}

View File

@@ -0,0 +1,93 @@
/**
* Infrastructure Adapter: InMemoryMediaEventPublisher
*
* In-memory implementation of MediaEventPublisher for testing purposes.
* Stores events in memory for verification in integration tests.
*/
import type { Logger } from '@core/shared/domain/Logger';
import type { DomainEvent } from '@core/shared/domain/DomainEvent';
export interface MediaEvent {
eventType: string;
aggregateId: string;
eventData: unknown;
occurredAt: Date;
}
export class InMemoryMediaEventPublisher {
private events: MediaEvent[] = [];
constructor(private readonly logger: Logger) {
this.logger.info('[InMemoryMediaEventPublisher] Initialized.');
}
/**
* Publish a domain event
*/
async publish(event: DomainEvent): Promise<void> {
this.logger.debug(`[InMemoryMediaEventPublisher] Publishing event: ${event.eventType} for aggregate: ${event.aggregateId}`);
const mediaEvent: MediaEvent = {
eventType: event.eventType,
aggregateId: event.aggregateId,
eventData: event.eventData,
occurredAt: event.occurredAt,
};
this.events.push(mediaEvent);
this.logger.info(`Event ${event.eventType} published successfully.`);
}
/**
* Get all published events
*/
getEvents(): MediaEvent[] {
return [...this.events];
}
/**
* Get events by event type
*/
getEventsByType(eventType: string): MediaEvent[] {
return this.events.filter(event => event.eventType === eventType);
}
/**
* Get events by aggregate ID
*/
getEventsByAggregateId(aggregateId: string): MediaEvent[] {
return this.events.filter(event => event.aggregateId === aggregateId);
}
/**
* Get the total number of events
*/
getEventCount(): number {
return this.events.length;
}
/**
* Clear all events
*/
clear(): void {
this.events = [];
this.logger.info('[InMemoryMediaEventPublisher] All events cleared.');
}
/**
* Check if an event of a specific type was published
*/
hasEvent(eventType: string): boolean {
return this.events.some(event => event.eventType === eventType);
}
/**
* Check if an event was published for a specific aggregate
*/
hasEventForAggregate(eventType: string, aggregateId: string): boolean {
return this.events.some(
event => event.eventType === eventType && event.aggregateId === aggregateId
);
}
}

View File

@@ -0,0 +1,121 @@
/**
* Infrastructure Adapter: InMemoryAvatarRepository
*
* In-memory implementation of AvatarRepository for testing purposes.
* Stores avatar entities in memory for fast, deterministic testing.
*/
import type { Avatar } from '@core/media/domain/entities/Avatar';
import type { AvatarRepository } from '@core/media/domain/repositories/AvatarRepository';
import type { Logger } from '@core/shared/domain/Logger';
export class InMemoryAvatarRepository implements AvatarRepository {
private avatars: Map<string, Avatar> = new Map();
private driverAvatars: Map<string, Avatar[]> = new Map();
constructor(private readonly logger: Logger) {
this.logger.info('[InMemoryAvatarRepository] Initialized.');
}
async save(avatar: Avatar): Promise<void> {
this.logger.debug(`[InMemoryAvatarRepository] Saving avatar: ${avatar.id} for driver: ${avatar.driverId}`);
// Store by ID
this.avatars.set(avatar.id, avatar);
// Store by driver ID
if (!this.driverAvatars.has(avatar.driverId)) {
this.driverAvatars.set(avatar.driverId, []);
}
const driverAvatars = this.driverAvatars.get(avatar.driverId)!;
const existingIndex = driverAvatars.findIndex(a => a.id === avatar.id);
if (existingIndex > -1) {
driverAvatars[existingIndex] = avatar;
} else {
driverAvatars.push(avatar);
}
this.logger.info(`Avatar ${avatar.id} for driver ${avatar.driverId} saved successfully.`);
}
async findById(id: string): Promise<Avatar | null> {
this.logger.debug(`[InMemoryAvatarRepository] Finding avatar by ID: ${id}`);
const avatar = this.avatars.get(id) ?? null;
if (avatar) {
this.logger.info(`Found avatar by ID: ${id}`);
} else {
this.logger.warn(`Avatar with ID ${id} not found.`);
}
return avatar;
}
async findActiveByDriverId(driverId: string): Promise<Avatar | null> {
this.logger.debug(`[InMemoryAvatarRepository] Finding active avatar for driver: ${driverId}`);
const driverAvatars = this.driverAvatars.get(driverId) ?? [];
const activeAvatar = driverAvatars.find(avatar => avatar.isActive) ?? null;
if (activeAvatar) {
this.logger.info(`Found active avatar for driver ${driverId}: ${activeAvatar.id}`);
} else {
this.logger.warn(`No active avatar found for driver: ${driverId}`);
}
return activeAvatar;
}
async findByDriverId(driverId: string): Promise<Avatar[]> {
this.logger.debug(`[InMemoryAvatarRepository] Finding all avatars for driver: ${driverId}`);
const driverAvatars = this.driverAvatars.get(driverId) ?? [];
this.logger.info(`Found ${driverAvatars.length} avatars for driver ${driverId}.`);
return driverAvatars;
}
async delete(id: string): Promise<void> {
this.logger.debug(`[InMemoryAvatarRepository] Deleting avatar with ID: ${id}`);
const avatarToDelete = this.avatars.get(id);
if (!avatarToDelete) {
this.logger.warn(`Avatar with ID ${id} not found for deletion.`);
return;
}
// Remove from avatars map
this.avatars.delete(id);
// Remove from driver avatars
const driverAvatars = this.driverAvatars.get(avatarToDelete.driverId);
if (driverAvatars) {
const filtered = driverAvatars.filter(avatar => avatar.id !== id);
if (filtered.length > 0) {
this.driverAvatars.set(avatarToDelete.driverId, filtered);
} else {
this.driverAvatars.delete(avatarToDelete.driverId);
}
}
this.logger.info(`Avatar ${id} deleted successfully.`);
}
/**
* Clear all avatars from the repository
*/
clear(): void {
this.avatars.clear();
this.driverAvatars.clear();
this.logger.info('[InMemoryAvatarRepository] All avatars cleared.');
}
/**
* Get the total number of avatars stored
*/
get size(): number {
return this.avatars.size;
}
}

View File

@@ -0,0 +1,106 @@
/**
* Infrastructure Adapter: InMemoryMediaRepository
*
* In-memory implementation of MediaRepository for testing purposes.
* Stores media entities in memory for fast, deterministic testing.
*/
import type { Media } from '@core/media/domain/entities/Media';
import type { MediaRepository } from '@core/media/domain/repositories/MediaRepository';
import type { Logger } from '@core/shared/domain/Logger';
export class InMemoryMediaRepository implements MediaRepository {
private media: Map<string, Media> = new Map();
private uploadedByMedia: Map<string, Media[]> = new Map();
constructor(private readonly logger: Logger) {
this.logger.info('[InMemoryMediaRepository] Initialized.');
}
async save(media: Media): Promise<void> {
this.logger.debug(`[InMemoryMediaRepository] Saving media: ${media.id} for uploader: ${media.uploadedBy}`);
// Store by ID
this.media.set(media.id, media);
// Store by uploader
if (!this.uploadedByMedia.has(media.uploadedBy)) {
this.uploadedByMedia.set(media.uploadedBy, []);
}
const uploaderMedia = this.uploadedByMedia.get(media.uploadedBy)!;
const existingIndex = uploaderMedia.findIndex(m => m.id === media.id);
if (existingIndex > -1) {
uploaderMedia[existingIndex] = media;
} else {
uploaderMedia.push(media);
}
this.logger.info(`Media ${media.id} for uploader ${media.uploadedBy} saved successfully.`);
}
async findById(id: string): Promise<Media | null> {
this.logger.debug(`[InMemoryMediaRepository] Finding media by ID: ${id}`);
const media = this.media.get(id) ?? null;
if (media) {
this.logger.info(`Found media by ID: ${id}`);
} else {
this.logger.warn(`Media with ID ${id} not found.`);
}
return media;
}
async findByUploadedBy(uploadedBy: string): Promise<Media[]> {
this.logger.debug(`[InMemoryMediaRepository] Finding all media for uploader: ${uploadedBy}`);
const uploaderMedia = this.uploadedByMedia.get(uploadedBy) ?? [];
this.logger.info(`Found ${uploaderMedia.length} media files for uploader ${uploadedBy}.`);
return uploaderMedia;
}
async delete(id: string): Promise<void> {
this.logger.debug(`[InMemoryMediaRepository] Deleting media with ID: ${id}`);
const mediaToDelete = this.media.get(id);
if (!mediaToDelete) {
this.logger.warn(`Media with ID ${id} not found for deletion.`);
return;
}
// Remove from media map
this.media.delete(id);
// Remove from uploader media
const uploaderMedia = this.uploadedByMedia.get(mediaToDelete.uploadedBy);
if (uploaderMedia) {
const filtered = uploaderMedia.filter(media => media.id !== id);
if (filtered.length > 0) {
this.uploadedByMedia.set(mediaToDelete.uploadedBy, filtered);
} else {
this.uploadedByMedia.delete(mediaToDelete.uploadedBy);
}
}
this.logger.info(`Media ${id} deleted successfully.`);
}
/**
* Clear all media from the repository
*/
clear(): void {
this.media.clear();
this.uploadedByMedia.clear();
this.logger.info('[InMemoryMediaRepository] All media cleared.');
}
/**
* Get the total number of media files stored
*/
get size(): number {
return this.media.size;
}
}

View File

@@ -0,0 +1,109 @@
/**
* Infrastructure Adapter: InMemoryMediaStorageAdapter
*
* In-memory implementation of MediaStoragePort for testing purposes.
* Simulates file storage without actual filesystem operations.
*/
import type { MediaStoragePort, UploadOptions, UploadResult } from '@core/media/application/ports/MediaStoragePort';
import type { Logger } from '@core/shared/domain/Logger';
export class InMemoryMediaStorageAdapter implements MediaStoragePort {
private storage: Map<string, Buffer> = new Map();
private metadata: Map<string, { size: number; contentType: string }> = new Map();
constructor(private readonly logger: Logger) {
this.logger.info('[InMemoryMediaStorageAdapter] Initialized.');
}
async uploadMedia(buffer: Buffer, options: UploadOptions): Promise<UploadResult> {
this.logger.debug(`[InMemoryMediaStorageAdapter] Uploading media: ${options.filename}`);
// Validate content type
const allowedTypes = ['image/png', 'image/jpeg', 'image/svg+xml', 'image/gif'];
if (!allowedTypes.includes(options.mimeType)) {
return {
success: false,
errorMessage: `Content type ${options.mimeType} is not allowed`,
};
}
// Generate storage key
const storageKey = `uploaded/${Date.now()}-${options.filename.replace(/[^a-zA-Z0-9.-]/g, '_')}`;
// Store buffer and metadata
this.storage.set(storageKey, buffer);
this.metadata.set(storageKey, {
size: buffer.length,
contentType: options.mimeType,
});
this.logger.info(`Media uploaded successfully: ${storageKey}`);
return {
success: true,
filename: options.filename,
url: storageKey,
};
}
async deleteMedia(storageKey: string): Promise<void> {
this.logger.debug(`[InMemoryMediaStorageAdapter] Deleting media: ${storageKey}`);
this.storage.delete(storageKey);
this.metadata.delete(storageKey);
this.logger.info(`Media deleted successfully: ${storageKey}`);
}
async getBytes(storageKey: string): Promise<Buffer | null> {
this.logger.debug(`[InMemoryMediaStorageAdapter] Getting bytes for: ${storageKey}`);
const buffer = this.storage.get(storageKey) ?? null;
if (buffer) {
this.logger.info(`Retrieved bytes for: ${storageKey}`);
} else {
this.logger.warn(`No bytes found for: ${storageKey}`);
}
return buffer;
}
async getMetadata(storageKey: string): Promise<{ size: number; contentType: string } | null> {
this.logger.debug(`[InMemoryMediaStorageAdapter] Getting metadata for: ${storageKey}`);
const meta = this.metadata.get(storageKey) ?? null;
if (meta) {
this.logger.info(`Retrieved metadata for: ${storageKey}`);
} else {
this.logger.warn(`No metadata found for: ${storageKey}`);
}
return meta;
}
/**
* Clear all stored media
*/
clear(): void {
this.storage.clear();
this.metadata.clear();
this.logger.info('[InMemoryMediaStorageAdapter] All media cleared.');
}
/**
* Get the total number of stored media files
*/
get size(): number {
return this.storage.size;
}
/**
* Check if a storage key exists
*/
has(storageKey: string): boolean {
return this.storage.has(storageKey);
}
}

View File

@@ -93,6 +93,12 @@ export class InMemoryDriverRepository implements DriverRepository {
return Promise.resolve(this.iracingIdIndex.has(iracingId));
}
async clear(): Promise<void> {
this.logger.info('[InMemoryDriverRepository] Clearing all drivers');
this.drivers.clear();
this.iracingIdIndex.clear();
}
// Serialization methods for persistence
serialize(driver: Driver): Record<string, unknown> {
return {

View File

@@ -212,4 +212,10 @@ async getMembership(teamId: string, driverId: string): Promise<TeamMembership |
throw error;
}
}
async clear(): Promise<void> {
this.logger.info('[InMemoryTeamMembershipRepository] Clearing all memberships and join requests');
this.membershipsByTeam.clear();
this.joinRequestsByTeam.clear();
}
}

View File

@@ -124,6 +124,11 @@ export class InMemoryTeamRepository implements TeamRepository {
}
}
async clear(): Promise<void> {
this.logger.info('[InMemoryTeamRepository] Clearing all teams');
this.teams.clear();
}
// Serialization methods for persistence
serialize(team: Team): Record<string, unknown> {
return {

View File

@@ -104,4 +104,9 @@ export class InMemoryDriverExtendedProfileProvider implements DriverExtendedProf
openToRequests: hash % 2 === 0,
};
}
clear(): void {
this.logger.info('[InMemoryDriverExtendedProfileProvider] Clearing all data');
// No data to clear as this provider generates data on-the-fly
}
}

View File

@@ -32,4 +32,9 @@ export class InMemoryDriverRatingProvider implements DriverRatingProvider {
}
return ratingsMap;
}
clear(): void {
this.logger.info('[InMemoryDriverRatingProvider] Clearing all data');
// No data to clear as this provider generates data on-the-fly
}
}

View File

@@ -153,4 +153,10 @@ export class InMemorySocialGraphRepository implements SocialGraphRepository {
throw error;
}
}
async clear(): Promise<void> {
this.logger.info('[InMemorySocialGraphRepository] Clearing all friendships and drivers');
this.friendships = [];
this.driversById.clear();
}
}