integration tests
This commit is contained in:
175
adapters/events/InMemoryHealthEventPublisher.ts
Normal file
175
adapters/events/InMemoryHealthEventPublisher.ts
Normal 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;
|
||||
}
|
||||
}
|
||||
@@ -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,
|
||||
};
|
||||
}
|
||||
}
|
||||
@@ -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;
|
||||
}
|
||||
}
|
||||
@@ -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();
|
||||
}
|
||||
}
|
||||
69
adapters/leagues/events/InMemoryLeagueEventPublisher.ts
Normal file
69
adapters/leagues/events/InMemoryLeagueEventPublisher.ts
Normal 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];
|
||||
}
|
||||
}
|
||||
@@ -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,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
93
adapters/media/events/InMemoryMediaEventPublisher.ts
Normal file
93
adapters/media/events/InMemoryMediaEventPublisher.ts
Normal 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
|
||||
);
|
||||
}
|
||||
}
|
||||
121
adapters/media/persistence/inmemory/InMemoryAvatarRepository.ts
Normal file
121
adapters/media/persistence/inmemory/InMemoryAvatarRepository.ts
Normal 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;
|
||||
}
|
||||
}
|
||||
106
adapters/media/persistence/inmemory/InMemoryMediaRepository.ts
Normal file
106
adapters/media/persistence/inmemory/InMemoryMediaRepository.ts
Normal 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;
|
||||
}
|
||||
}
|
||||
109
adapters/media/ports/InMemoryMediaStorageAdapter.ts
Normal file
109
adapters/media/ports/InMemoryMediaStorageAdapter.ts
Normal 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);
|
||||
}
|
||||
}
|
||||
@@ -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 {
|
||||
|
||||
@@ -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();
|
||||
}
|
||||
}
|
||||
@@ -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 {
|
||||
|
||||
@@ -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
|
||||
}
|
||||
}
|
||||
@@ -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
|
||||
}
|
||||
}
|
||||
|
||||
@@ -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();
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user