module cleanup
This commit is contained in:
@@ -0,0 +1,53 @@
|
||||
import type { Logger } from '@core/shared/application';
|
||||
import type { IPageViewRepository } from '../../domain/repositories/IPageViewRepository';
|
||||
|
||||
export interface GetAnalyticsMetricsInput {
|
||||
startDate?: Date;
|
||||
endDate?: Date;
|
||||
}
|
||||
|
||||
export interface GetAnalyticsMetricsOutput {
|
||||
pageViews: number;
|
||||
uniqueVisitors: number;
|
||||
averageSessionDuration: number;
|
||||
bounceRate: number;
|
||||
}
|
||||
|
||||
export class GetAnalyticsMetricsUseCase {
|
||||
constructor(
|
||||
private readonly pageViewRepository: IPageViewRepository,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(input: GetAnalyticsMetricsInput = {}): Promise<GetAnalyticsMetricsOutput> {
|
||||
try {
|
||||
const startDate = input.startDate ?? new Date(Date.now() - 30 * 24 * 60 * 60 * 1000); // 30 days ago
|
||||
const endDate = input.endDate ?? new Date();
|
||||
|
||||
// For now, return placeholder values as actual implementation would require
|
||||
// aggregating data across all entities or specifying which entity
|
||||
// This is a simplified version
|
||||
const pageViews = 0;
|
||||
const uniqueVisitors = 0;
|
||||
const averageSessionDuration = 0;
|
||||
const bounceRate = 0;
|
||||
|
||||
this.logger.info('Analytics metrics retrieved', {
|
||||
startDate,
|
||||
endDate,
|
||||
pageViews,
|
||||
uniqueVisitors,
|
||||
});
|
||||
|
||||
return {
|
||||
pageViews,
|
||||
uniqueVisitors,
|
||||
averageSessionDuration,
|
||||
bounceRate,
|
||||
};
|
||||
} catch (error) {
|
||||
this.logger.error('Failed to get analytics metrics', { error, input });
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,43 @@
|
||||
import type { Logger } from '@core/shared/application';
|
||||
|
||||
export interface GetDashboardDataInput {}
|
||||
|
||||
export interface GetDashboardDataOutput {
|
||||
totalUsers: number;
|
||||
activeUsers: number;
|
||||
totalRaces: number;
|
||||
totalLeagues: number;
|
||||
}
|
||||
|
||||
export class GetDashboardDataUseCase {
|
||||
constructor(
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(_input: GetDashboardDataInput = {}): Promise<GetDashboardDataOutput> {
|
||||
try {
|
||||
// Placeholder implementation - would need repositories from identity and racing domains
|
||||
const totalUsers = 0;
|
||||
const activeUsers = 0;
|
||||
const totalRaces = 0;
|
||||
const totalLeagues = 0;
|
||||
|
||||
this.logger.info('Dashboard data retrieved', {
|
||||
totalUsers,
|
||||
activeUsers,
|
||||
totalRaces,
|
||||
totalLeagues,
|
||||
});
|
||||
|
||||
return {
|
||||
totalUsers,
|
||||
activeUsers,
|
||||
totalRaces,
|
||||
totalLeagues,
|
||||
};
|
||||
} catch (error) {
|
||||
this.logger.error('Failed to get dashboard data', { error });
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,13 +1,7 @@
|
||||
/**
|
||||
* Use Case: RecordEngagementUseCase
|
||||
*
|
||||
* Records an engagement event when a visitor interacts with an entity.
|
||||
*/
|
||||
|
||||
import type { AsyncUseCase } from '@core/shared/application';
|
||||
import type { Logger } from '@core/shared/application';
|
||||
import { EngagementEvent, type EngagementAction, type EngagementEntityType } from '../../domain/entities/EngagementEvent';
|
||||
import type { IEngagementRepository } from '../../domain/repositories/IEngagementRepository';
|
||||
import { EngagementEvent } from '../../domain/entities/EngagementEvent';
|
||||
import type { EngagementAction, EngagementEntityType } from '../../domain/types/EngagementEvent';
|
||||
|
||||
export interface RecordEngagementInput {
|
||||
action: EngagementAction;
|
||||
@@ -24,43 +18,41 @@ export interface RecordEngagementOutput {
|
||||
engagementWeight: number;
|
||||
}
|
||||
|
||||
export class RecordEngagementUseCase
|
||||
implements AsyncUseCase<RecordEngagementInput, RecordEngagementOutput> {
|
||||
export class RecordEngagementUseCase {
|
||||
constructor(
|
||||
private readonly engagementRepository: IEngagementRepository,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(input: RecordEngagementInput): Promise<RecordEngagementOutput> {
|
||||
this.logger.debug('Executing RecordEngagementUseCase', { input });
|
||||
try {
|
||||
const eventId = `eng-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;
|
||||
|
||||
const baseProps: Omit<Parameters<typeof EngagementEvent.create>[0], 'timestamp'> = {
|
||||
id: eventId,
|
||||
const engagementEvent = EngagementEvent.create({
|
||||
id: crypto.randomUUID(),
|
||||
action: input.action,
|
||||
entityType: input.entityType,
|
||||
entityId: input.entityId,
|
||||
actorId: input.actorId,
|
||||
actorType: input.actorType,
|
||||
sessionId: input.sessionId,
|
||||
};
|
||||
|
||||
const event = EngagementEvent.create({
|
||||
...baseProps,
|
||||
...(input.actorId !== undefined ? { actorId: input.actorId } : {}),
|
||||
...(input.metadata !== undefined ? { metadata: input.metadata } : {}),
|
||||
metadata: input.metadata,
|
||||
});
|
||||
|
||||
await this.engagementRepository.save(event);
|
||||
this.logger.info('Engagement recorded successfully', { eventId, input });
|
||||
await this.engagementRepository.save(engagementEvent);
|
||||
|
||||
this.logger.info('Engagement event recorded', {
|
||||
engagementId: engagementEvent.id,
|
||||
action: input.action,
|
||||
entityId: input.entityId,
|
||||
entityType: input.entityType,
|
||||
});
|
||||
|
||||
return {
|
||||
eventId,
|
||||
engagementWeight: event.getEngagementWeight(),
|
||||
eventId: engagementEvent.id,
|
||||
engagementWeight: engagementEvent.getEngagementWeight(),
|
||||
};
|
||||
} catch (error) {
|
||||
this.logger.error('Error recording engagement', error instanceof Error ? error : new Error(String(error)), { input });
|
||||
this.logger.error('Failed to record engagement event', { error: error as Error, input });
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,14 +1,7 @@
|
||||
/**
|
||||
* Use Case: RecordPageViewUseCase
|
||||
*
|
||||
* Records a page view event when a visitor accesses an entity page.
|
||||
*/
|
||||
|
||||
import type { AsyncUseCase } from '@core/shared/application';
|
||||
import type { Logger } from '@core/shared/application';
|
||||
import type { IPageViewRepository } from '../../domain/repositories/IPageViewRepository';
|
||||
import { PageView } from '../../domain/entities/PageView';
|
||||
import type { EntityType, VisitorType } from '../../domain/types/PageView';
|
||||
import type { IPageViewRepository } from '../repositories/IPageViewRepository';
|
||||
|
||||
export interface RecordPageViewInput {
|
||||
entityType: EntityType;
|
||||
@@ -25,41 +18,40 @@ export interface RecordPageViewOutput {
|
||||
pageViewId: string;
|
||||
}
|
||||
|
||||
export class RecordPageViewUseCase
|
||||
implements AsyncUseCase<RecordPageViewInput, RecordPageViewOutput> {
|
||||
export class RecordPageViewUseCase {
|
||||
constructor(
|
||||
private readonly pageViewRepository: IPageViewRepository,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(input: RecordPageViewInput): Promise<RecordPageViewOutput> {
|
||||
this.logger.debug('Executing RecordPageViewUseCase', { input });
|
||||
try {
|
||||
const pageViewId = `pv-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;
|
||||
|
||||
const baseProps: Omit<Parameters<typeof PageView.create>[0], 'timestamp'> = {
|
||||
id: pageViewId,
|
||||
const pageView = PageView.create({
|
||||
id: crypto.randomUUID(),
|
||||
entityType: input.entityType,
|
||||
entityId: input.entityId,
|
||||
visitorId: input.visitorId,
|
||||
visitorType: input.visitorType,
|
||||
sessionId: input.sessionId,
|
||||
};
|
||||
|
||||
const pageView = PageView.create({
|
||||
...baseProps,
|
||||
...(input.visitorId !== undefined ? { visitorId: input.visitorId } : {}),
|
||||
...(input.referrer !== undefined ? { referrer: input.referrer } : {}),
|
||||
...(input.userAgent !== undefined ? { userAgent: input.userAgent } : {}),
|
||||
...(input.country !== undefined ? { country: input.country } : {}),
|
||||
referrer: input.referrer,
|
||||
userAgent: input.userAgent,
|
||||
country: input.country,
|
||||
});
|
||||
|
||||
await this.pageViewRepository.save(pageView);
|
||||
this.logger.info('Page view recorded successfully', { pageViewId, input });
|
||||
return { pageViewId };
|
||||
|
||||
this.logger.info('Page view recorded', {
|
||||
pageViewId: pageView.id,
|
||||
entityId: input.entityId,
|
||||
entityType: input.entityType,
|
||||
});
|
||||
|
||||
return {
|
||||
pageViewId: pageView.id,
|
||||
};
|
||||
} catch (error) {
|
||||
const err = error instanceof Error ? error : new Error(String(error));
|
||||
this.logger.error('Error recording page view', err, { input });
|
||||
this.logger.error('Failed to record page view', { error, input });
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
12
core/analytics/domain/repositories/IPageViewRepository.ts
Normal file
12
core/analytics/domain/repositories/IPageViewRepository.ts
Normal file
@@ -0,0 +1,12 @@
|
||||
import type { PageView } from '../entities/PageView';
|
||||
|
||||
export interface IPageViewRepository {
|
||||
save(pageView: PageView): Promise<void>;
|
||||
findById(id: string): Promise<PageView | null>;
|
||||
findByEntityId(entityId: string): Promise<PageView[]>;
|
||||
findBySessionId(sessionId: string): Promise<PageView[]>;
|
||||
countByEntityId(entityId: string): Promise<number>;
|
||||
getUniqueVisitorsCount(entityId: string, startDate: Date, endDate: Date): Promise<number>;
|
||||
getAverageSessionDuration(entityId: string, startDate: Date, endDate: Date): Promise<number>;
|
||||
getBounceRate(entityId: string, startDate: Date, endDate: Date): Promise<number>;
|
||||
}
|
||||
@@ -5,24 +5,30 @@
|
||||
* Kept in domain/types so domain/entities contains only entity classes.
|
||||
*/
|
||||
|
||||
export type EngagementAction =
|
||||
| 'click_sponsor_logo'
|
||||
| 'click_sponsor_url'
|
||||
| 'download_livery_pack'
|
||||
| 'join_league'
|
||||
| 'register_race'
|
||||
| 'view_standings'
|
||||
| 'view_schedule'
|
||||
| 'share_social'
|
||||
| 'contact_sponsor';
|
||||
export const EngagementAction = {
|
||||
CLICK_SPONSOR_LOGO: 'click_sponsor_logo',
|
||||
CLICK_SPONSOR_URL: 'click_sponsor_url',
|
||||
DOWNLOAD_LIVERY_PACK: 'download_livery_pack',
|
||||
JOIN_LEAGUE: 'join_league',
|
||||
REGISTER_RACE: 'register_race',
|
||||
VIEW_STANDINGS: 'view_standings',
|
||||
VIEW_SCHEDULE: 'view_schedule',
|
||||
SHARE_SOCIAL: 'share_social',
|
||||
CONTACT_SPONSOR: 'contact_sponsor',
|
||||
} as const;
|
||||
|
||||
export type EngagementEntityType =
|
||||
| 'league'
|
||||
| 'driver'
|
||||
| 'team'
|
||||
| 'race'
|
||||
| 'sponsor'
|
||||
| 'sponsorship';
|
||||
export type EngagementAction = typeof EngagementAction[keyof typeof EngagementAction];
|
||||
|
||||
export const EngagementEntityType = {
|
||||
LEAGUE: 'league',
|
||||
DRIVER: 'driver',
|
||||
TEAM: 'team',
|
||||
RACE: 'race',
|
||||
SPONSOR: 'sponsor',
|
||||
SPONSORSHIP: 'sponsorship',
|
||||
} as const;
|
||||
|
||||
export type EngagementEntityType = typeof EngagementEntityType[keyof typeof EngagementEntityType];
|
||||
|
||||
export interface EngagementEventProps {
|
||||
id: string;
|
||||
|
||||
@@ -5,19 +5,23 @@
|
||||
* Kept in domain/types so domain/entities contains only entity classes.
|
||||
*/
|
||||
|
||||
export enum EntityType {
|
||||
LEAGUE = 'league',
|
||||
DRIVER = 'driver',
|
||||
TEAM = 'team',
|
||||
RACE = 'race',
|
||||
SPONSOR = 'sponsor',
|
||||
}
|
||||
export const EntityType = {
|
||||
LEAGUE: 'league',
|
||||
DRIVER: 'driver',
|
||||
TEAM: 'team',
|
||||
RACE: 'race',
|
||||
SPONSOR: 'sponsor',
|
||||
} as const;
|
||||
|
||||
export enum VisitorType {
|
||||
ANONYMOUS = 'anonymous',
|
||||
DRIVER = 'driver',
|
||||
SPONSOR = 'sponsor',
|
||||
}
|
||||
export type EntityType = typeof EntityType[keyof typeof EntityType];
|
||||
|
||||
export const VisitorType = {
|
||||
ANONYMOUS: 'anonymous',
|
||||
DRIVER: 'driver',
|
||||
SPONSOR: 'sponsor',
|
||||
} as const;
|
||||
|
||||
export type VisitorType = typeof VisitorType[keyof typeof VisitorType];
|
||||
|
||||
export interface PageViewProps {
|
||||
id: string;
|
||||
|
||||
34
core/media/application/ports/MediaStoragePort.ts
Normal file
34
core/media/application/ports/MediaStoragePort.ts
Normal file
@@ -0,0 +1,34 @@
|
||||
/**
|
||||
* Port: MediaStoragePort
|
||||
*
|
||||
* Defines the contract for media file storage operations.
|
||||
*/
|
||||
|
||||
export interface UploadOptions {
|
||||
filename: string;
|
||||
mimeType: string;
|
||||
metadata?: Record<string, any>;
|
||||
}
|
||||
|
||||
export interface UploadResult {
|
||||
success: boolean;
|
||||
url?: string;
|
||||
filename?: string;
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface MediaStoragePort {
|
||||
/**
|
||||
* Upload a media file
|
||||
* @param buffer File buffer
|
||||
* @param options Upload options
|
||||
* @returns Upload result with URL
|
||||
*/
|
||||
uploadMedia(buffer: Buffer, options: UploadOptions): Promise<UploadResult>;
|
||||
|
||||
/**
|
||||
* Delete a media file by URL
|
||||
* @param url Media URL to delete
|
||||
*/
|
||||
deleteMedia(url: string): Promise<void>;
|
||||
}
|
||||
@@ -0,0 +1,8 @@
|
||||
export interface DeleteMediaResult {
|
||||
success: boolean;
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface IDeleteMediaPresenter {
|
||||
present(result: DeleteMediaResult): void;
|
||||
}
|
||||
14
core/media/application/presenters/IGetAvatarPresenter.ts
Normal file
14
core/media/application/presenters/IGetAvatarPresenter.ts
Normal file
@@ -0,0 +1,14 @@
|
||||
export interface GetAvatarResult {
|
||||
success: boolean;
|
||||
avatar?: {
|
||||
id: string;
|
||||
driverId: string;
|
||||
mediaUrl: string;
|
||||
selectedAt: Date;
|
||||
};
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface IGetAvatarPresenter {
|
||||
present(result: GetAvatarResult): void;
|
||||
}
|
||||
20
core/media/application/presenters/IGetMediaPresenter.ts
Normal file
20
core/media/application/presenters/IGetMediaPresenter.ts
Normal file
@@ -0,0 +1,20 @@
|
||||
export interface GetMediaResult {
|
||||
success: boolean;
|
||||
media?: {
|
||||
id: string;
|
||||
filename: string;
|
||||
originalName: string;
|
||||
mimeType: string;
|
||||
size: number;
|
||||
url: string;
|
||||
type: string;
|
||||
uploadedBy: string;
|
||||
uploadedAt: Date;
|
||||
metadata?: Record<string, any>;
|
||||
};
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface IGetMediaPresenter {
|
||||
present(result: GetMediaResult): void;
|
||||
}
|
||||
@@ -8,6 +8,6 @@ export interface RequestAvatarGenerationResultDTO {
|
||||
export interface IRequestAvatarGenerationPresenter {
|
||||
reset(): void;
|
||||
present(dto: RequestAvatarGenerationResultDTO): void;
|
||||
get viewModel(): any;
|
||||
getViewModel(): any;
|
||||
get viewModel(): RequestAvatarGenerationResultDTO;
|
||||
getViewModel(): RequestAvatarGenerationResultDTO;
|
||||
}
|
||||
@@ -0,0 +1,9 @@
|
||||
export interface SelectAvatarResult {
|
||||
success: boolean;
|
||||
selectedAvatarUrl?: string;
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface ISelectAvatarPresenter {
|
||||
present(result: SelectAvatarResult): void;
|
||||
}
|
||||
@@ -0,0 +1,8 @@
|
||||
export interface UpdateAvatarResult {
|
||||
success: boolean;
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface IUpdateAvatarPresenter {
|
||||
present(result: UpdateAvatarResult): void;
|
||||
}
|
||||
10
core/media/application/presenters/IUploadMediaPresenter.ts
Normal file
10
core/media/application/presenters/IUploadMediaPresenter.ts
Normal file
@@ -0,0 +1,10 @@
|
||||
export interface UploadMediaResult {
|
||||
success: boolean;
|
||||
mediaId?: string;
|
||||
url?: string;
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface IUploadMediaPresenter {
|
||||
present(result: UploadMediaResult): void;
|
||||
}
|
||||
77
core/media/application/use-cases/DeleteMediaUseCase.ts
Normal file
77
core/media/application/use-cases/DeleteMediaUseCase.ts
Normal file
@@ -0,0 +1,77 @@
|
||||
/**
|
||||
* Use Case: DeleteMediaUseCase
|
||||
*
|
||||
* Handles the business logic for deleting media files.
|
||||
*/
|
||||
|
||||
import type { IMediaRepository } from '../../domain/repositories/IMediaRepository';
|
||||
import type { MediaStoragePort } from '../ports/MediaStoragePort';
|
||||
import type { Logger } from '@core/shared/application';
|
||||
import type { IDeleteMediaPresenter } from '../presenters/IDeleteMediaPresenter';
|
||||
|
||||
export interface DeleteMediaInput {
|
||||
mediaId: string;
|
||||
}
|
||||
|
||||
export interface DeleteMediaResult {
|
||||
success: boolean;
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface IDeleteMediaPresenter {
|
||||
present(result: DeleteMediaResult): void;
|
||||
}
|
||||
|
||||
export class DeleteMediaUseCase {
|
||||
constructor(
|
||||
private readonly mediaRepo: IMediaRepository,
|
||||
private readonly mediaStorage: MediaStoragePort,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(
|
||||
input: DeleteMediaInput,
|
||||
presenter: IDeleteMediaPresenter,
|
||||
): Promise<void> {
|
||||
try {
|
||||
this.logger.info('[DeleteMediaUseCase] Deleting media', {
|
||||
mediaId: input.mediaId,
|
||||
});
|
||||
|
||||
const media = await this.mediaRepo.findById(input.mediaId);
|
||||
|
||||
if (!media) {
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: 'Media not found',
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
// Delete from storage
|
||||
await this.mediaStorage.deleteMedia(media.url.value);
|
||||
|
||||
// Delete from repository
|
||||
await this.mediaRepo.delete(input.mediaId);
|
||||
|
||||
presenter.present({
|
||||
success: true,
|
||||
});
|
||||
|
||||
this.logger.info('[DeleteMediaUseCase] Media deleted successfully', {
|
||||
mediaId: input.mediaId,
|
||||
});
|
||||
|
||||
} catch (error) {
|
||||
this.logger.error('[DeleteMediaUseCase] Error deleting media', {
|
||||
error: error instanceof Error ? error.message : 'Unknown error',
|
||||
mediaId: input.mediaId,
|
||||
});
|
||||
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: 'Internal error occurred while deleting media',
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
77
core/media/application/use-cases/GetAvatarUseCase.ts
Normal file
77
core/media/application/use-cases/GetAvatarUseCase.ts
Normal file
@@ -0,0 +1,77 @@
|
||||
/**
|
||||
* Use Case: GetAvatarUseCase
|
||||
*
|
||||
* Handles the business logic for retrieving a driver's avatar.
|
||||
*/
|
||||
|
||||
import type { IAvatarRepository } from '../../domain/repositories/IAvatarRepository';
|
||||
import type { Logger } from '@core/shared/application';
|
||||
import type { IGetAvatarPresenter } from '../presenters/IGetAvatarPresenter';
|
||||
|
||||
export interface GetAvatarInput {
|
||||
driverId: string;
|
||||
}
|
||||
|
||||
export interface GetAvatarResult {
|
||||
success: boolean;
|
||||
avatar?: {
|
||||
id: string;
|
||||
driverId: string;
|
||||
mediaUrl: string;
|
||||
selectedAt: Date;
|
||||
};
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface IGetAvatarPresenter {
|
||||
present(result: GetAvatarResult): void;
|
||||
}
|
||||
|
||||
export class GetAvatarUseCase {
|
||||
constructor(
|
||||
private readonly avatarRepo: IAvatarRepository,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(
|
||||
input: GetAvatarInput,
|
||||
presenter: IGetAvatarPresenter,
|
||||
): Promise<void> {
|
||||
try {
|
||||
this.logger.info('[GetAvatarUseCase] Getting avatar', {
|
||||
driverId: input.driverId,
|
||||
});
|
||||
|
||||
const avatar = await this.avatarRepo.findActiveByDriverId(input.driverId);
|
||||
|
||||
if (!avatar) {
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: 'Avatar not found',
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
presenter.present({
|
||||
success: true,
|
||||
avatar: {
|
||||
id: avatar.id,
|
||||
driverId: avatar.driverId,
|
||||
mediaUrl: avatar.mediaUrl.value,
|
||||
selectedAt: avatar.selectedAt,
|
||||
},
|
||||
});
|
||||
|
||||
} catch (error) {
|
||||
this.logger.error('[GetAvatarUseCase] Error getting avatar', {
|
||||
error: error instanceof Error ? error.message : 'Unknown error',
|
||||
driverId: input.driverId,
|
||||
});
|
||||
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: 'Internal error occurred while retrieving avatar',
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
89
core/media/application/use-cases/GetMediaUseCase.ts
Normal file
89
core/media/application/use-cases/GetMediaUseCase.ts
Normal file
@@ -0,0 +1,89 @@
|
||||
/**
|
||||
* Use Case: GetMediaUseCase
|
||||
*
|
||||
* Handles the business logic for retrieving media information.
|
||||
*/
|
||||
|
||||
import type { IMediaRepository } from '../../domain/repositories/IMediaRepository';
|
||||
import type { Logger } from '@core/shared/application';
|
||||
import type { IGetMediaPresenter } from '../presenters/IGetMediaPresenter';
|
||||
|
||||
export interface GetMediaInput {
|
||||
mediaId: string;
|
||||
}
|
||||
|
||||
export interface GetMediaResult {
|
||||
success: boolean;
|
||||
media?: {
|
||||
id: string;
|
||||
filename: string;
|
||||
originalName: string;
|
||||
mimeType: string;
|
||||
size: number;
|
||||
url: string;
|
||||
type: string;
|
||||
uploadedBy: string;
|
||||
uploadedAt: Date;
|
||||
metadata?: Record<string, any>;
|
||||
};
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface IGetMediaPresenter {
|
||||
present(result: GetMediaResult): void;
|
||||
}
|
||||
|
||||
export class GetMediaUseCase {
|
||||
constructor(
|
||||
private readonly mediaRepo: IMediaRepository,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(
|
||||
input: GetMediaInput,
|
||||
presenter: IGetMediaPresenter,
|
||||
): Promise<void> {
|
||||
try {
|
||||
this.logger.info('[GetMediaUseCase] Getting media', {
|
||||
mediaId: input.mediaId,
|
||||
});
|
||||
|
||||
const media = await this.mediaRepo.findById(input.mediaId);
|
||||
|
||||
if (!media) {
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: 'Media not found',
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
presenter.present({
|
||||
success: true,
|
||||
media: {
|
||||
id: media.id,
|
||||
filename: media.filename,
|
||||
originalName: media.originalName,
|
||||
mimeType: media.mimeType,
|
||||
size: media.size,
|
||||
url: media.url.value,
|
||||
type: media.type,
|
||||
uploadedBy: media.uploadedBy,
|
||||
uploadedAt: media.uploadedAt,
|
||||
metadata: media.metadata,
|
||||
},
|
||||
});
|
||||
|
||||
} catch (error) {
|
||||
this.logger.error('[GetMediaUseCase] Error getting media', {
|
||||
error: error instanceof Error ? error.message : 'Unknown error',
|
||||
mediaId: input.mediaId,
|
||||
});
|
||||
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: 'Internal error occurred while retrieving media',
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,155 +1,136 @@
|
||||
import type { UseCase, Logger } from '@core/shared/application';
|
||||
/**
|
||||
* Use Case: RequestAvatarGenerationUseCase
|
||||
*
|
||||
* Handles the business logic for requesting avatar generation from a face photo.
|
||||
*/
|
||||
|
||||
import { v4 as uuidv4 } from 'uuid';
|
||||
import type { IAvatarGenerationRepository } from '../../domain/repositories/IAvatarGenerationRepository';
|
||||
import type { FaceValidationPort } from '../ports/FaceValidationPort';
|
||||
import type { AvatarGenerationPort } from '../ports/AvatarGenerationPort';
|
||||
import type { IRequestAvatarGenerationPresenter, RequestAvatarGenerationResultDTO } from '../presenters/IRequestAvatarGenerationPresenter';
|
||||
import type { Logger } from '@core/shared/application';
|
||||
import { AvatarGenerationRequest } from '../../domain/entities/AvatarGenerationRequest';
|
||||
import type { RacingSuitColor, AvatarStyle } from '../../domain/types/AvatarGenerationRequest';
|
||||
import type { IRequestAvatarGenerationPresenter } from '../presenters/IRequestAvatarGenerationPresenter';
|
||||
import type { RacingSuitColor } from '../../domain/types/AvatarGenerationRequest';
|
||||
|
||||
export interface RequestAvatarGenerationCommand {
|
||||
export interface RequestAvatarGenerationInput {
|
||||
userId: string;
|
||||
facePhotoData: string; // Base64 encoded image data
|
||||
facePhotoData: string;
|
||||
suitColor: RacingSuitColor;
|
||||
style?: AvatarStyle;
|
||||
style?: 'realistic' | 'cartoon' | 'pixel-art';
|
||||
}
|
||||
|
||||
export class RequestAvatarGenerationUseCase
|
||||
implements UseCase<RequestAvatarGenerationCommand, RequestAvatarGenerationResultDTO, any, IRequestAvatarGenerationPresenter> {
|
||||
export class RequestAvatarGenerationUseCase {
|
||||
constructor(
|
||||
private readonly avatarRepository: IAvatarGenerationRepository,
|
||||
private readonly avatarRepo: IAvatarGenerationRepository,
|
||||
private readonly faceValidation: FaceValidationPort,
|
||||
private readonly avatarGeneration: AvatarGenerationPort,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(command: RequestAvatarGenerationCommand, presenter: IRequestAvatarGenerationPresenter): Promise<void> {
|
||||
presenter.reset();
|
||||
this.logger.debug(
|
||||
`Executing RequestAvatarGenerationUseCase for userId: ${command.userId}`,
|
||||
command,
|
||||
);
|
||||
|
||||
async execute(
|
||||
input: RequestAvatarGenerationInput,
|
||||
presenter: IRequestAvatarGenerationPresenter,
|
||||
): Promise<void> {
|
||||
try {
|
||||
// Create the generation request
|
||||
const requestId = this.generateId();
|
||||
this.logger.info('[RequestAvatarGenerationUseCase] Starting avatar generation request', {
|
||||
userId: input.userId,
|
||||
suitColor: input.suitColor,
|
||||
});
|
||||
|
||||
// Create the avatar generation request entity
|
||||
const requestId = uuidv4();
|
||||
const request = AvatarGenerationRequest.create({
|
||||
id: requestId,
|
||||
userId: command.userId,
|
||||
facePhotoUrl: `data:image/jpeg;base64,${command.facePhotoData}`,
|
||||
suitColor: command.suitColor,
|
||||
...(command.style ? { style: command.style } : {}),
|
||||
userId: input.userId,
|
||||
facePhotoUrl: input.facePhotoData, // Assuming facePhotoData is a URL or base64
|
||||
suitColor: input.suitColor,
|
||||
style: input.style,
|
||||
});
|
||||
|
||||
this.logger.info(`Avatar generation request created with ID: ${requestId}`);
|
||||
// Save initial request
|
||||
await this.avatarRepo.save(request);
|
||||
|
||||
// Mark as validating
|
||||
// Present initial status
|
||||
presenter.present({
|
||||
requestId,
|
||||
status: 'validating',
|
||||
});
|
||||
|
||||
// Validate face photo
|
||||
request.markAsValidating();
|
||||
await this.avatarRepository.save(request);
|
||||
this.logger.debug(`Request ${requestId} marked as validating.`);
|
||||
await this.avatarRepo.save(request);
|
||||
|
||||
// Validate the face photo
|
||||
const validationResult = await this.faceValidation.validateFacePhoto(command.facePhotoData);
|
||||
this.logger.debug(
|
||||
`Face validation result for request ${requestId}:`,
|
||||
validationResult,
|
||||
);
|
||||
|
||||
if (!validationResult.isValid) {
|
||||
const errorMessage = validationResult.errorMessage || 'Face validation failed';
|
||||
const validationResult = await this.faceValidation.validateFacePhoto(input.facePhotoData);
|
||||
|
||||
if (!validationResult.isValid || !validationResult.hasFace || validationResult.faceCount !== 1) {
|
||||
const errorMessage = validationResult.errorMessage || 'Invalid face photo: must contain exactly one face';
|
||||
request.fail(errorMessage);
|
||||
await this.avatarRepository.save(request);
|
||||
this.logger.error(`Face validation failed for request ${requestId}: ${errorMessage}`);
|
||||
await this.avatarRepo.save(request);
|
||||
|
||||
presenter.present({
|
||||
requestId,
|
||||
status: 'failed',
|
||||
errorMessage: validationResult.errorMessage || 'Please upload a clear photo of your face',
|
||||
errorMessage,
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
if (!validationResult.hasFace) {
|
||||
const errorMessage = 'No face detected in the image';
|
||||
request.fail(errorMessage);
|
||||
await this.avatarRepository.save(request);
|
||||
this.logger.error(`No face detected for request ${requestId}: ${errorMessage}`);
|
||||
presenter.present({
|
||||
requestId,
|
||||
status: 'failed',
|
||||
errorMessage: 'No face detected. Please upload a photo that clearly shows your face.',
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
if (validationResult.faceCount > 1) {
|
||||
const errorMessage = 'Multiple faces detected';
|
||||
request.fail(errorMessage);
|
||||
await this.avatarRepository.save(request);
|
||||
this.logger.error(`Multiple faces detected for request ${requestId}: ${errorMessage}`);
|
||||
presenter.present({
|
||||
requestId,
|
||||
status: 'failed',
|
||||
errorMessage: 'Multiple faces detected. Please upload a photo with only your face.',
|
||||
});
|
||||
return;
|
||||
}
|
||||
this.logger.info(`Face validation successful for request ${requestId}.`);
|
||||
|
||||
// Mark as generating
|
||||
request.markAsGenerating();
|
||||
await this.avatarRepository.save(request);
|
||||
this.logger.debug(`Request ${requestId} marked as generating.`);
|
||||
|
||||
// Generate avatars
|
||||
const generationResult = await this.avatarGeneration.generateAvatars({
|
||||
facePhotoUrl: request.facePhotoUrl.value,
|
||||
request.markAsGenerating();
|
||||
await this.avatarRepo.save(request);
|
||||
|
||||
const generationOptions = {
|
||||
facePhotoUrl: input.facePhotoData,
|
||||
prompt: request.buildPrompt(),
|
||||
suitColor: request.suitColor,
|
||||
style: request.style,
|
||||
count: 3, // Generate 3 options
|
||||
});
|
||||
this.logger.debug(
|
||||
`Avatar generation service result for request ${requestId}:`,
|
||||
generationResult,
|
||||
);
|
||||
suitColor: input.suitColor,
|
||||
style: input.style || 'realistic',
|
||||
count: 3, // Generate 3 avatar options
|
||||
};
|
||||
|
||||
const generationResult = await this.avatarGeneration.generateAvatars(generationOptions);
|
||||
|
||||
if (!generationResult.success) {
|
||||
const errorMessage = generationResult.errorMessage || 'Avatar generation failed';
|
||||
const errorMessage = generationResult.errorMessage || 'Failed to generate avatars';
|
||||
request.fail(errorMessage);
|
||||
await this.avatarRepository.save(request);
|
||||
this.logger.error(`Avatar generation failed for request ${requestId}: ${errorMessage}`);
|
||||
await this.avatarRepo.save(request);
|
||||
|
||||
presenter.present({
|
||||
requestId,
|
||||
status: 'failed',
|
||||
errorMessage: generationResult.errorMessage || 'Failed to generate avatars. Please try again.',
|
||||
errorMessage,
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
// Complete with generated avatars
|
||||
const avatarUrls = generationResult.avatars.map(a => a.url);
|
||||
// Complete the request
|
||||
const avatarUrls = generationResult.avatars.map(avatar => avatar.url);
|
||||
request.completeWithAvatars(avatarUrls);
|
||||
await this.avatarRepository.save(request);
|
||||
this.logger.info(`Avatar generation completed successfully for request ${requestId}.`);
|
||||
await this.avatarRepo.save(request);
|
||||
|
||||
presenter.present({
|
||||
requestId,
|
||||
status: 'completed',
|
||||
avatarUrls,
|
||||
});
|
||||
} catch (error) {
|
||||
this.logger.error(
|
||||
`An unexpected error occurred during avatar generation for userId: ${command.userId}`,
|
||||
error as Error,
|
||||
);
|
||||
// Re-throw or return a generic error, depending on desired error handling strategy
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
|
||||
private generateId(): string {
|
||||
if (typeof crypto !== 'undefined' && crypto.randomUUID) {
|
||||
return crypto.randomUUID();
|
||||
this.logger.info('[RequestAvatarGenerationUseCase] Avatar generation completed', {
|
||||
requestId,
|
||||
userId: input.userId,
|
||||
avatarCount: avatarUrls.length,
|
||||
});
|
||||
|
||||
} catch (error) {
|
||||
this.logger.error('[RequestAvatarGenerationUseCase] Error during avatar generation', {
|
||||
error: error instanceof Error ? error.message : 'Unknown error',
|
||||
userId: input.userId,
|
||||
});
|
||||
|
||||
presenter.present({
|
||||
requestId: uuidv4(), // Fallback ID
|
||||
status: 'failed',
|
||||
errorMessage: 'Internal error occurred during avatar generation',
|
||||
});
|
||||
}
|
||||
return 'avatar-' + Date.now().toString(36) + Math.random().toString(36).substr(2, 9);
|
||||
}
|
||||
}
|
||||
@@ -1,16 +1,16 @@
|
||||
/**
|
||||
* Use Case: SelectAvatarUseCase
|
||||
*
|
||||
* Allows a user to select one of the generated avatars as their profile avatar.
|
||||
*
|
||||
* Handles the business logic for selecting a generated avatar from the options.
|
||||
*/
|
||||
|
||||
import type { AsyncUseCase, Logger } from '@core/shared/application';
|
||||
import type { IAvatarGenerationRepository } from '../../domain/repositories/IAvatarGenerationRepository';
|
||||
import type { Logger } from '@core/shared/application';
|
||||
import type { ISelectAvatarPresenter } from '../presenters/ISelectAvatarPresenter';
|
||||
|
||||
export interface SelectAvatarCommand {
|
||||
export interface SelectAvatarInput {
|
||||
requestId: string;
|
||||
userId: string;
|
||||
avatarIndex: number;
|
||||
selectedIndex: number;
|
||||
}
|
||||
|
||||
export interface SelectAvatarResult {
|
||||
@@ -19,60 +19,69 @@ export interface SelectAvatarResult {
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export class SelectAvatarUseCase
|
||||
implements AsyncUseCase<SelectAvatarCommand, SelectAvatarResult> {
|
||||
export interface ISelectAvatarPresenter {
|
||||
present(result: SelectAvatarResult): void;
|
||||
}
|
||||
|
||||
export class SelectAvatarUseCase {
|
||||
constructor(
|
||||
private readonly avatarRepository: IAvatarGenerationRepository,
|
||||
private readonly avatarRepo: IAvatarGenerationRepository,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(command: SelectAvatarCommand): Promise<SelectAvatarResult> {
|
||||
this.logger.debug(`Executing SelectAvatarUseCase for userId: ${command.userId}, requestId: ${command.requestId}, avatarIndex: ${command.avatarIndex}`);
|
||||
|
||||
const request = await this.avatarRepository.findById(command.requestId);
|
||||
|
||||
if (!request) {
|
||||
this.logger.info(`Avatar generation request not found for requestId: ${command.requestId}`);
|
||||
return {
|
||||
success: false,
|
||||
errorMessage: 'Avatar generation request not found',
|
||||
};
|
||||
}
|
||||
|
||||
if (request.userId !== command.userId) {
|
||||
this.logger.info(`Permission denied for userId: ${command.userId} to select avatar for requestId: ${command.requestId}`);
|
||||
return {
|
||||
success: false,
|
||||
errorMessage: 'You do not have permission to select this avatar',
|
||||
};
|
||||
}
|
||||
|
||||
if (request.status !== 'completed') {
|
||||
this.logger.info(`Avatar generation not completed for requestId: ${command.requestId}, current status: ${request.status}`);
|
||||
return {
|
||||
success: false,
|
||||
errorMessage: 'Avatar generation is not yet complete',
|
||||
};
|
||||
}
|
||||
|
||||
async execute(
|
||||
input: SelectAvatarInput,
|
||||
presenter: ISelectAvatarPresenter,
|
||||
): Promise<void> {
|
||||
try {
|
||||
request.selectAvatar(command.avatarIndex);
|
||||
await this.avatarRepository.save(request);
|
||||
this.logger.info('[SelectAvatarUseCase] Selecting avatar', {
|
||||
requestId: input.requestId,
|
||||
selectedIndex: input.selectedIndex,
|
||||
});
|
||||
|
||||
const request = await this.avatarRepo.findById(input.requestId);
|
||||
|
||||
if (!request) {
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: 'Avatar generation request not found',
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
if (request.status !== 'completed') {
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: 'Avatar generation is not completed yet',
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
request.selectAvatar(input.selectedIndex);
|
||||
await this.avatarRepo.save(request);
|
||||
|
||||
const selectedAvatarUrl = request.selectedAvatarUrl;
|
||||
const result: SelectAvatarResult =
|
||||
selectedAvatarUrl !== undefined
|
||||
? { success: true, selectedAvatarUrl }
|
||||
: { success: true };
|
||||
|
||||
this.logger.info(`Avatar selected successfully for userId: ${command.userId}, requestId: ${command.requestId}, selectedAvatarUrl: ${selectedAvatarUrl}`);
|
||||
return result;
|
||||
presenter.present({
|
||||
success: true,
|
||||
selectedAvatarUrl,
|
||||
});
|
||||
|
||||
this.logger.info('[SelectAvatarUseCase] Avatar selected successfully', {
|
||||
requestId: input.requestId,
|
||||
selectedAvatarUrl,
|
||||
});
|
||||
|
||||
} catch (error) {
|
||||
this.logger.error(`Failed to select avatar for userId: ${command.userId}, requestId: ${command.requestId}: ${error instanceof Error ? error.message : 'Unknown error'}`, error);
|
||||
return {
|
||||
this.logger.error('[SelectAvatarUseCase] Error selecting avatar', {
|
||||
error: error instanceof Error ? error.message : 'Unknown error',
|
||||
requestId: input.requestId,
|
||||
});
|
||||
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: error instanceof Error ? error.message : 'Failed to select avatar',
|
||||
};
|
||||
errorMessage: 'Internal error occurred while selecting avatar',
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
81
core/media/application/use-cases/UpdateAvatarUseCase.ts
Normal file
81
core/media/application/use-cases/UpdateAvatarUseCase.ts
Normal file
@@ -0,0 +1,81 @@
|
||||
/**
|
||||
* Use Case: UpdateAvatarUseCase
|
||||
*
|
||||
* Handles the business logic for updating a driver's avatar.
|
||||
*/
|
||||
|
||||
import type { IAvatarRepository } from '../../domain/repositories/IAvatarRepository';
|
||||
import type { Logger } from '@core/shared/application';
|
||||
import { Avatar } from '../../domain/entities/Avatar';
|
||||
import type { IUpdateAvatarPresenter } from '../presenters/IUpdateAvatarPresenter';
|
||||
import { v4 as uuidv4 } from 'uuid';
|
||||
|
||||
export interface UpdateAvatarInput {
|
||||
driverId: string;
|
||||
mediaUrl: string;
|
||||
}
|
||||
|
||||
export interface UpdateAvatarResult {
|
||||
success: boolean;
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface IUpdateAvatarPresenter {
|
||||
present(result: UpdateAvatarResult): void;
|
||||
}
|
||||
|
||||
export class UpdateAvatarUseCase {
|
||||
constructor(
|
||||
private readonly avatarRepo: IAvatarRepository,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(
|
||||
input: UpdateAvatarInput,
|
||||
presenter: IUpdateAvatarPresenter,
|
||||
): Promise<void> {
|
||||
try {
|
||||
this.logger.info('[UpdateAvatarUseCase] Updating avatar', {
|
||||
driverId: input.driverId,
|
||||
mediaUrl: input.mediaUrl,
|
||||
});
|
||||
|
||||
// Deactivate current active avatar
|
||||
const currentAvatar = await this.avatarRepo.findActiveByDriverId(input.driverId);
|
||||
if (currentAvatar) {
|
||||
currentAvatar.deactivate();
|
||||
await this.avatarRepo.save(currentAvatar);
|
||||
}
|
||||
|
||||
// Create new avatar
|
||||
const avatarId = uuidv4();
|
||||
const newAvatar = Avatar.create({
|
||||
id: avatarId,
|
||||
driverId: input.driverId,
|
||||
mediaUrl: input.mediaUrl,
|
||||
});
|
||||
|
||||
await this.avatarRepo.save(newAvatar);
|
||||
|
||||
presenter.present({
|
||||
success: true,
|
||||
});
|
||||
|
||||
this.logger.info('[UpdateAvatarUseCase] Avatar updated successfully', {
|
||||
driverId: input.driverId,
|
||||
avatarId,
|
||||
});
|
||||
|
||||
} catch (error) {
|
||||
this.logger.error('[UpdateAvatarUseCase] Error updating avatar', {
|
||||
error: error instanceof Error ? error.message : 'Unknown error',
|
||||
driverId: input.driverId,
|
||||
});
|
||||
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: 'Internal error occurred while updating avatar',
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
111
core/media/application/use-cases/UploadMediaUseCase.ts
Normal file
111
core/media/application/use-cases/UploadMediaUseCase.ts
Normal file
@@ -0,0 +1,111 @@
|
||||
/**
|
||||
* Use Case: UploadMediaUseCase
|
||||
*
|
||||
* Handles the business logic for uploading media files.
|
||||
*/
|
||||
|
||||
import type { IMediaRepository } from '../../domain/repositories/IMediaRepository';
|
||||
import type { MediaStoragePort } from '../ports/MediaStoragePort';
|
||||
import type { Logger } from '@core/shared/application';
|
||||
import { Media } from '../../domain/entities/Media';
|
||||
import type { IUploadMediaPresenter } from '../presenters/IUploadMediaPresenter';
|
||||
import { v4 as uuidv4 } from 'uuid';
|
||||
|
||||
export interface UploadMediaInput {
|
||||
file: Express.Multer.File;
|
||||
uploadedBy: string;
|
||||
metadata?: Record<string, any>;
|
||||
}
|
||||
|
||||
export interface UploadMediaResult {
|
||||
success: boolean;
|
||||
mediaId?: string;
|
||||
url?: string;
|
||||
errorMessage?: string;
|
||||
}
|
||||
|
||||
export interface IUploadMediaPresenter {
|
||||
present(result: UploadMediaResult): void;
|
||||
}
|
||||
|
||||
export class UploadMediaUseCase {
|
||||
constructor(
|
||||
private readonly mediaRepo: IMediaRepository,
|
||||
private readonly mediaStorage: MediaStoragePort,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(
|
||||
input: UploadMediaInput,
|
||||
presenter: IUploadMediaPresenter,
|
||||
): Promise<void> {
|
||||
try {
|
||||
this.logger.info('[UploadMediaUseCase] Starting media upload', {
|
||||
filename: input.file.originalname,
|
||||
size: input.file.size,
|
||||
uploadedBy: input.uploadedBy,
|
||||
});
|
||||
|
||||
// Upload file to storage service
|
||||
const uploadResult = await this.mediaStorage.uploadMedia(input.file.buffer, {
|
||||
filename: input.file.originalname,
|
||||
mimeType: input.file.mimetype,
|
||||
metadata: input.metadata,
|
||||
});
|
||||
|
||||
if (!uploadResult.success) {
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: uploadResult.errorMessage || 'Failed to upload media',
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
// Determine media type
|
||||
const mediaType: 'image' | 'video' | 'document' = input.file.mimetype.startsWith('image/')
|
||||
? 'image'
|
||||
: input.file.mimetype.startsWith('video/')
|
||||
? 'video'
|
||||
: 'document';
|
||||
|
||||
// Create media entity
|
||||
const mediaId = uuidv4();
|
||||
const media = Media.create({
|
||||
id: mediaId,
|
||||
filename: uploadResult.filename || input.file.originalname,
|
||||
originalName: input.file.originalname,
|
||||
mimeType: input.file.mimetype,
|
||||
size: input.file.size,
|
||||
url: uploadResult.url,
|
||||
type: mediaType,
|
||||
uploadedBy: input.uploadedBy,
|
||||
metadata: input.metadata,
|
||||
});
|
||||
|
||||
// Save to repository
|
||||
await this.mediaRepo.save(media);
|
||||
|
||||
presenter.present({
|
||||
success: true,
|
||||
mediaId,
|
||||
url: uploadResult.url,
|
||||
});
|
||||
|
||||
this.logger.info('[UploadMediaUseCase] Media uploaded successfully', {
|
||||
mediaId,
|
||||
url: uploadResult.url,
|
||||
});
|
||||
|
||||
} catch (error) {
|
||||
this.logger.error('[UploadMediaUseCase] Error uploading media', {
|
||||
error: error instanceof Error ? error.message : 'Unknown error',
|
||||
filename: input.file.originalname,
|
||||
});
|
||||
|
||||
presenter.present({
|
||||
success: false,
|
||||
errorMessage: 'Internal error occurred during media upload',
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
73
core/media/domain/entities/Avatar.ts
Normal file
73
core/media/domain/entities/Avatar.ts
Normal file
@@ -0,0 +1,73 @@
|
||||
/**
|
||||
* Domain Entity: Avatar
|
||||
*
|
||||
* Represents a user's selected avatar.
|
||||
*/
|
||||
|
||||
import type { IEntity } from '@core/shared/domain';
|
||||
import { MediaUrl } from '../value-objects/MediaUrl';
|
||||
|
||||
export interface AvatarProps {
|
||||
id: string;
|
||||
driverId: string;
|
||||
mediaUrl: string;
|
||||
selectedAt: Date;
|
||||
isActive: boolean;
|
||||
}
|
||||
|
||||
export class Avatar implements IEntity<string> {
|
||||
readonly id: string;
|
||||
readonly driverId: string;
|
||||
readonly mediaUrl: MediaUrl;
|
||||
readonly selectedAt: Date;
|
||||
private _isActive: boolean;
|
||||
|
||||
private constructor(props: AvatarProps) {
|
||||
this.id = props.id;
|
||||
this.driverId = props.driverId;
|
||||
this.mediaUrl = MediaUrl.create(props.mediaUrl);
|
||||
this.selectedAt = props.selectedAt;
|
||||
this._isActive = props.isActive;
|
||||
}
|
||||
|
||||
static create(props: {
|
||||
id: string;
|
||||
driverId: string;
|
||||
mediaUrl: string;
|
||||
}): Avatar {
|
||||
if (!props.driverId) {
|
||||
throw new Error('Driver ID is required');
|
||||
}
|
||||
if (!props.mediaUrl) {
|
||||
throw new Error('Media URL is required');
|
||||
}
|
||||
|
||||
return new Avatar({
|
||||
...props,
|
||||
selectedAt: new Date(),
|
||||
isActive: true,
|
||||
});
|
||||
}
|
||||
|
||||
static reconstitute(props: AvatarProps): Avatar {
|
||||
return new Avatar(props);
|
||||
}
|
||||
|
||||
get isActive(): boolean {
|
||||
return this._isActive;
|
||||
}
|
||||
|
||||
deactivate(): void {
|
||||
this._isActive = false;
|
||||
}
|
||||
|
||||
toProps(): AvatarProps {
|
||||
return {
|
||||
id: this.id,
|
||||
driverId: this.driverId,
|
||||
mediaUrl: this.mediaUrl.value,
|
||||
selectedAt: this.selectedAt,
|
||||
isActive: this._isActive,
|
||||
};
|
||||
}
|
||||
}
|
||||
95
core/media/domain/entities/Media.ts
Normal file
95
core/media/domain/entities/Media.ts
Normal file
@@ -0,0 +1,95 @@
|
||||
/**
|
||||
* Domain Entity: Media
|
||||
*
|
||||
* Represents a media file (image, video, etc.) stored in the system.
|
||||
*/
|
||||
|
||||
import type { IEntity } from '@core/shared/domain';
|
||||
import { MediaUrl } from '../value-objects/MediaUrl';
|
||||
|
||||
export type MediaType = 'image' | 'video' | 'document';
|
||||
|
||||
export interface MediaProps {
|
||||
id: string;
|
||||
filename: string;
|
||||
originalName: string;
|
||||
mimeType: string;
|
||||
size: number;
|
||||
url: string;
|
||||
type: MediaType;
|
||||
uploadedBy: string;
|
||||
uploadedAt: Date;
|
||||
metadata?: Record<string, any>;
|
||||
}
|
||||
|
||||
export class Media implements IEntity<string> {
|
||||
readonly id: string;
|
||||
readonly filename: string;
|
||||
readonly originalName: string;
|
||||
readonly mimeType: string;
|
||||
readonly size: number;
|
||||
readonly url: MediaUrl;
|
||||
readonly type: MediaType;
|
||||
readonly uploadedBy: string;
|
||||
readonly uploadedAt: Date;
|
||||
readonly metadata?: Record<string, any>;
|
||||
|
||||
private constructor(props: MediaProps) {
|
||||
this.id = props.id;
|
||||
this.filename = props.filename;
|
||||
this.originalName = props.originalName;
|
||||
this.mimeType = props.mimeType;
|
||||
this.size = props.size;
|
||||
this.url = MediaUrl.create(props.url);
|
||||
this.type = props.type;
|
||||
this.uploadedBy = props.uploadedBy;
|
||||
this.uploadedAt = props.uploadedAt;
|
||||
this.metadata = props.metadata;
|
||||
}
|
||||
|
||||
static create(props: {
|
||||
id: string;
|
||||
filename: string;
|
||||
originalName: string;
|
||||
mimeType: string;
|
||||
size: number;
|
||||
url: string;
|
||||
type: MediaType;
|
||||
uploadedBy: string;
|
||||
metadata?: Record<string, any>;
|
||||
}): Media {
|
||||
if (!props.filename) {
|
||||
throw new Error('Filename is required');
|
||||
}
|
||||
if (!props.url) {
|
||||
throw new Error('URL is required');
|
||||
}
|
||||
if (!props.uploadedBy) {
|
||||
throw new Error('Uploaded by is required');
|
||||
}
|
||||
|
||||
return new Media({
|
||||
...props,
|
||||
uploadedAt: new Date(),
|
||||
});
|
||||
}
|
||||
|
||||
static reconstitute(props: MediaProps): Media {
|
||||
return new Media(props);
|
||||
}
|
||||
|
||||
toProps(): MediaProps {
|
||||
return {
|
||||
id: this.id,
|
||||
filename: this.filename,
|
||||
originalName: this.originalName,
|
||||
mimeType: this.mimeType,
|
||||
size: this.size,
|
||||
url: this.url.value,
|
||||
type: this.type,
|
||||
uploadedBy: this.uploadedBy,
|
||||
uploadedAt: this.uploadedAt,
|
||||
metadata: this.metadata,
|
||||
};
|
||||
}
|
||||
}
|
||||
34
core/media/domain/repositories/IAvatarRepository.ts
Normal file
34
core/media/domain/repositories/IAvatarRepository.ts
Normal file
@@ -0,0 +1,34 @@
|
||||
/**
|
||||
* Repository Interface: IAvatarRepository
|
||||
*
|
||||
* Defines the contract for avatar persistence.
|
||||
*/
|
||||
|
||||
import type { Avatar } from '../entities/Avatar';
|
||||
|
||||
export interface IAvatarRepository {
|
||||
/**
|
||||
* Save an avatar
|
||||
*/
|
||||
save(avatar: Avatar): Promise<void>;
|
||||
|
||||
/**
|
||||
* Find avatar by ID
|
||||
*/
|
||||
findById(id: string): Promise<Avatar | null>;
|
||||
|
||||
/**
|
||||
* Find active avatar for a driver
|
||||
*/
|
||||
findActiveByDriverId(driverId: string): Promise<Avatar | null>;
|
||||
|
||||
/**
|
||||
* Find all avatars for a driver
|
||||
*/
|
||||
findByDriverId(driverId: string): Promise<Avatar[]>;
|
||||
|
||||
/**
|
||||
* Delete an avatar
|
||||
*/
|
||||
delete(id: string): Promise<void>;
|
||||
}
|
||||
29
core/media/domain/repositories/IMediaRepository.ts
Normal file
29
core/media/domain/repositories/IMediaRepository.ts
Normal file
@@ -0,0 +1,29 @@
|
||||
/**
|
||||
* Repository Interface: IMediaRepository
|
||||
*
|
||||
* Defines the contract for media file persistence.
|
||||
*/
|
||||
|
||||
import type { Media } from '../entities/Media';
|
||||
|
||||
export interface IMediaRepository {
|
||||
/**
|
||||
* Save a media file
|
||||
*/
|
||||
save(media: Media): Promise<void>;
|
||||
|
||||
/**
|
||||
* Find a media file by ID
|
||||
*/
|
||||
findById(id: string): Promise<Media | null>;
|
||||
|
||||
/**
|
||||
* Find media files by uploader
|
||||
*/
|
||||
findByUploadedBy(uploadedBy: string): Promise<Media[]>;
|
||||
|
||||
/**
|
||||
* Delete a media file
|
||||
*/
|
||||
delete(id: string): Promise<void>;
|
||||
}
|
||||
@@ -3,11 +3,36 @@ export * from './application/ports/ImageServicePort';
|
||||
export * from './application/ports/FaceValidationPort';
|
||||
export * from './application/ports/AvatarGenerationPort';
|
||||
|
||||
// Ports
|
||||
export * from './application/ports/ImageServicePort';
|
||||
export * from './application/ports/FaceValidationPort';
|
||||
export * from './application/ports/AvatarGenerationPort';
|
||||
export * from './application/ports/MediaStoragePort';
|
||||
|
||||
// Presenters
|
||||
export * from './application/presenters/IRequestAvatarGenerationPresenter';
|
||||
export * from './application/presenters/ISelectAvatarPresenter';
|
||||
export * from './application/presenters/IUploadMediaPresenter';
|
||||
export * from './application/presenters/IGetMediaPresenter';
|
||||
export * from './application/presenters/IDeleteMediaPresenter';
|
||||
export * from './application/presenters/IGetAvatarPresenter';
|
||||
export * from './application/presenters/IUpdateAvatarPresenter';
|
||||
|
||||
// Use Cases
|
||||
export * from './application/use-cases/RequestAvatarGenerationUseCase';
|
||||
export * from './application/use-cases/SelectAvatarUseCase';
|
||||
export * from './application/use-cases/UploadMediaUseCase';
|
||||
export * from './application/use-cases/GetMediaUseCase';
|
||||
export * from './application/use-cases/DeleteMediaUseCase';
|
||||
export * from './application/use-cases/GetAvatarUseCase';
|
||||
export * from './application/use-cases/UpdateAvatarUseCase';
|
||||
|
||||
// Domain
|
||||
export * from './domain/entities/AvatarGenerationRequest';
|
||||
export * from './domain/entities/Media';
|
||||
export * from './domain/entities/Avatar';
|
||||
export * from './domain/repositories/IAvatarGenerationRepository';
|
||||
export type { AvatarGenerationRequestProps } from './domain/types/AvatarGenerationRequest';
|
||||
export * from './domain/repositories/IMediaRepository';
|
||||
export * from './domain/repositories/IAvatarRepository';
|
||||
export type { AvatarGenerationRequestProps } from './domain/types/AvatarGenerationRequest';
|
||||
export type { MediaType } from './domain/entities/Media';
|
||||
@@ -5,16 +5,15 @@
|
||||
*/
|
||||
|
||||
import type { IPaymentRepository } from '../../domain/repositories/IPaymentRepository';
|
||||
import type { PaymentType, PayerType, PaymentStatus, Payment } from '../../domain/entities/Payment';
|
||||
import type { Payment, PaymentType, PayerType, PaymentStatus } from '../../domain/entities/Payment';
|
||||
import type {
|
||||
ICreatePaymentPresenter,
|
||||
CreatePaymentResultDTO,
|
||||
CreatePaymentViewModel,
|
||||
PaymentDto,
|
||||
} from '../presenters/ICreatePaymentPresenter';
|
||||
import type { UseCase } from '@core/shared/application/UseCase';
|
||||
|
||||
const PLATFORM_FEE_RATE = 0.10;
|
||||
|
||||
export interface CreatePaymentInput {
|
||||
type: PaymentType;
|
||||
amount: number;
|
||||
@@ -37,7 +36,8 @@ export class CreatePaymentUseCase
|
||||
|
||||
const { type, amount, payerId, payerType, leagueId, seasonId } = input;
|
||||
|
||||
const platformFee = amount * PLATFORM_FEE_RATE;
|
||||
// Calculate platform fee (assume 5% for now)
|
||||
const platformFee = Math.round(amount * 0.05 * 100) / 100;
|
||||
const netAmount = amount - platformFee;
|
||||
|
||||
const id = `payment-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;
|
||||
@@ -51,29 +51,31 @@ export class CreatePaymentUseCase
|
||||
payerType,
|
||||
leagueId,
|
||||
seasonId,
|
||||
status: 'pending' as PaymentStatus,
|
||||
status: PaymentStatus.PENDING,
|
||||
createdAt: new Date(),
|
||||
};
|
||||
|
||||
const createdPayment = await this.paymentRepository.create(payment);
|
||||
|
||||
const dto: CreatePaymentResultDTO = {
|
||||
payment: {
|
||||
id: createdPayment.id,
|
||||
type: createdPayment.type,
|
||||
amount: createdPayment.amount,
|
||||
platformFee: createdPayment.platformFee,
|
||||
netAmount: createdPayment.netAmount,
|
||||
payerId: createdPayment.payerId,
|
||||
payerType: createdPayment.payerType,
|
||||
leagueId: createdPayment.leagueId,
|
||||
seasonId: createdPayment.seasonId,
|
||||
status: createdPayment.status,
|
||||
createdAt: createdPayment.createdAt,
|
||||
completedAt: createdPayment.completedAt,
|
||||
},
|
||||
const dto: PaymentDto = {
|
||||
id: createdPayment.id,
|
||||
type: createdPayment.type,
|
||||
amount: createdPayment.amount,
|
||||
platformFee: createdPayment.platformFee,
|
||||
netAmount: createdPayment.netAmount,
|
||||
payerId: createdPayment.payerId,
|
||||
payerType: createdPayment.payerType,
|
||||
leagueId: createdPayment.leagueId,
|
||||
seasonId: createdPayment.seasonId,
|
||||
status: createdPayment.status,
|
||||
createdAt: createdPayment.createdAt,
|
||||
completedAt: createdPayment.completedAt,
|
||||
};
|
||||
|
||||
presenter.present(dto);
|
||||
const result: CreatePaymentResultDTO = {
|
||||
payment: dto,
|
||||
};
|
||||
|
||||
presenter.present(result);
|
||||
}
|
||||
}
|
||||
@@ -10,6 +10,7 @@ import type {
|
||||
IGetPaymentsPresenter,
|
||||
GetPaymentsResultDTO,
|
||||
GetPaymentsViewModel,
|
||||
PaymentDto,
|
||||
} from '../presenters/IGetPaymentsPresenter';
|
||||
import type { UseCase } from '@core/shared/application/UseCase';
|
||||
|
||||
@@ -30,27 +31,27 @@ export class GetPaymentsUseCase
|
||||
): Promise<void> {
|
||||
presenter.reset();
|
||||
|
||||
const payments = await this.paymentRepository.findByFilters({
|
||||
leagueId: input.leagueId,
|
||||
payerId: input.payerId,
|
||||
type: input.type,
|
||||
});
|
||||
const { leagueId, payerId, type } = input;
|
||||
|
||||
const payments = await this.paymentRepository.findByFilters({ leagueId, payerId, type });
|
||||
|
||||
const dtos: PaymentDto[] = payments.map(payment => ({
|
||||
id: payment.id,
|
||||
type: payment.type,
|
||||
amount: payment.amount,
|
||||
platformFee: payment.platformFee,
|
||||
netAmount: payment.netAmount,
|
||||
payerId: payment.payerId,
|
||||
payerType: payment.payerType,
|
||||
leagueId: payment.leagueId,
|
||||
seasonId: payment.seasonId,
|
||||
status: payment.status,
|
||||
createdAt: payment.createdAt,
|
||||
completedAt: payment.completedAt,
|
||||
}));
|
||||
|
||||
const dto: GetPaymentsResultDTO = {
|
||||
payments: payments.map(payment => ({
|
||||
id: payment.id,
|
||||
type: payment.type,
|
||||
amount: payment.amount,
|
||||
platformFee: payment.platformFee,
|
||||
netAmount: payment.netAmount,
|
||||
payerId: payment.payerId,
|
||||
payerType: payment.payerType,
|
||||
leagueId: payment.leagueId,
|
||||
seasonId: payment.seasonId,
|
||||
status: payment.status,
|
||||
createdAt: payment.createdAt,
|
||||
completedAt: payment.completedAt,
|
||||
})),
|
||||
payments: dtos,
|
||||
};
|
||||
|
||||
presenter.present(dto);
|
||||
|
||||
@@ -10,6 +10,7 @@ import type {
|
||||
IUpdatePaymentStatusPresenter,
|
||||
UpdatePaymentStatusResultDTO,
|
||||
UpdatePaymentStatusViewModel,
|
||||
PaymentDto,
|
||||
} from '../presenters/IUpdatePaymentStatusPresenter';
|
||||
import type { UseCase } from '@core/shared/application/UseCase';
|
||||
|
||||
@@ -31,35 +32,38 @@ export class UpdatePaymentStatusUseCase
|
||||
|
||||
const { paymentId, status } = input;
|
||||
|
||||
const payment = await this.paymentRepository.findById(paymentId);
|
||||
if (!payment) {
|
||||
throw new Error('Payment not found');
|
||||
const existingPayment = await this.paymentRepository.findById(paymentId);
|
||||
if (!existingPayment) {
|
||||
throw new Error(`Payment with id ${paymentId} not found`);
|
||||
}
|
||||
|
||||
payment.status = status;
|
||||
if (status === ('completed' as PaymentStatus)) {
|
||||
payment.completedAt = new Date();
|
||||
}
|
||||
|
||||
const updatedPayment = await this.paymentRepository.update(payment);
|
||||
|
||||
const dto: UpdatePaymentStatusResultDTO = {
|
||||
payment: {
|
||||
id: updatedPayment.id,
|
||||
type: updatedPayment.type,
|
||||
amount: updatedPayment.amount,
|
||||
platformFee: updatedPayment.platformFee,
|
||||
netAmount: updatedPayment.netAmount,
|
||||
payerId: updatedPayment.payerId,
|
||||
payerType: updatedPayment.payerType,
|
||||
leagueId: updatedPayment.leagueId,
|
||||
seasonId: updatedPayment.seasonId,
|
||||
status: updatedPayment.status,
|
||||
createdAt: updatedPayment.createdAt,
|
||||
completedAt: updatedPayment.completedAt,
|
||||
},
|
||||
const updatedPayment = {
|
||||
...existingPayment,
|
||||
status,
|
||||
completedAt: status === PaymentStatus.COMPLETED ? new Date() : existingPayment.completedAt,
|
||||
};
|
||||
|
||||
presenter.present(dto);
|
||||
const savedPayment = await this.paymentRepository.update(updatedPayment);
|
||||
|
||||
const dto: PaymentDto = {
|
||||
id: savedPayment.id,
|
||||
type: savedPayment.type,
|
||||
amount: savedPayment.amount,
|
||||
platformFee: savedPayment.platformFee,
|
||||
netAmount: savedPayment.netAmount,
|
||||
payerId: savedPayment.payerId,
|
||||
payerType: savedPayment.payerType,
|
||||
leagueId: savedPayment.leagueId,
|
||||
seasonId: savedPayment.seasonId,
|
||||
status: savedPayment.status,
|
||||
createdAt: savedPayment.createdAt,
|
||||
completedAt: savedPayment.completedAt,
|
||||
};
|
||||
|
||||
const result: UpdatePaymentStatusResultDTO = {
|
||||
payment: dto,
|
||||
};
|
||||
|
||||
presenter.present(result);
|
||||
}
|
||||
}
|
||||
15
core/racing/application/presenters/ICreateLeaguePresenter.ts
Normal file
15
core/racing/application/presenters/ICreateLeaguePresenter.ts
Normal file
@@ -0,0 +1,15 @@
|
||||
import { Presenter } from './Presenter';
|
||||
|
||||
export interface CreateLeagueResultDTO {
|
||||
leagueId: string;
|
||||
seasonId: string;
|
||||
scoringPresetId?: string;
|
||||
scoringPresetName?: string;
|
||||
}
|
||||
|
||||
export interface CreateLeagueViewModel {
|
||||
leagueId: string;
|
||||
success: boolean;
|
||||
}
|
||||
|
||||
export interface ICreateLeaguePresenter extends Presenter<CreateLeagueResultDTO, CreateLeagueViewModel> {}
|
||||
13
core/racing/application/presenters/IJoinLeaguePresenter.ts
Normal file
13
core/racing/application/presenters/IJoinLeaguePresenter.ts
Normal file
@@ -0,0 +1,13 @@
|
||||
import { Presenter } from './Presenter';
|
||||
|
||||
export interface JoinLeagueResultDTO {
|
||||
id: string;
|
||||
}
|
||||
|
||||
export interface JoinLeagueViewModel {
|
||||
success: boolean;
|
||||
membershipId?: string;
|
||||
error?: string;
|
||||
}
|
||||
|
||||
export interface IJoinLeaguePresenter extends Presenter<JoinLeagueResultDTO, JoinLeagueViewModel> {}
|
||||
@@ -1,11 +1,11 @@
|
||||
import type { Presenter } from '@core/shared/presentation/Presenter';
|
||||
|
||||
export interface RemoveLeagueMemberViewModel {
|
||||
success: boolean;
|
||||
}
|
||||
import { Presenter } from './Presenter';
|
||||
|
||||
export interface RemoveLeagueMemberResultDTO {
|
||||
success: boolean;
|
||||
}
|
||||
|
||||
export interface RemoveLeagueMemberViewModel {
|
||||
success: boolean;
|
||||
}
|
||||
|
||||
export interface IRemoveLeagueMemberPresenter extends Presenter<RemoveLeagueMemberResultDTO, RemoveLeagueMemberViewModel> {}
|
||||
@@ -0,0 +1,12 @@
|
||||
import { Presenter } from './Presenter';
|
||||
|
||||
export interface TransferLeagueOwnershipResultDTO {
|
||||
success: boolean;
|
||||
}
|
||||
|
||||
export interface TransferLeagueOwnershipViewModel {
|
||||
success: boolean;
|
||||
error?: string;
|
||||
}
|
||||
|
||||
export interface ITransferLeagueOwnershipPresenter extends Presenter<TransferLeagueOwnershipResultDTO, TransferLeagueOwnershipViewModel> {}
|
||||
@@ -1,11 +1,11 @@
|
||||
import type { Presenter } from '@core/shared/presentation/Presenter';
|
||||
|
||||
export interface UpdateLeagueMemberRoleViewModel {
|
||||
success: boolean;
|
||||
}
|
||||
import { Presenter } from './Presenter';
|
||||
|
||||
export interface UpdateLeagueMemberRoleResultDTO {
|
||||
success: boolean;
|
||||
}
|
||||
|
||||
export interface UpdateLeagueMemberRoleViewModel {
|
||||
success: boolean;
|
||||
}
|
||||
|
||||
export interface IUpdateLeagueMemberRolePresenter extends Presenter<UpdateLeagueMemberRoleResultDTO, UpdateLeagueMemberRoleViewModel> {}
|
||||
@@ -0,0 +1,118 @@
|
||||
import { describe, it, expect, beforeEach, vi } from 'vitest';
|
||||
import { GetTeamMembershipUseCase } from './GetTeamMembershipUseCase';
|
||||
import type { ITeamMembershipRepository } from '../../domain/repositories/ITeamMembershipRepository';
|
||||
import type { Logger } from '@core/shared/application';
|
||||
|
||||
describe('GetTeamMembershipUseCase', () => {
|
||||
const mockGetMembership = vi.fn();
|
||||
const mockMembershipRepo: ITeamMembershipRepository = {
|
||||
getMembership: mockGetMembership,
|
||||
getActiveMembershipForDriver: vi.fn(),
|
||||
getTeamMembers: vi.fn(),
|
||||
saveMembership: vi.fn(),
|
||||
removeMembership: vi.fn(),
|
||||
getJoinRequests: vi.fn(),
|
||||
countByTeamId: vi.fn(),
|
||||
saveJoinRequest: vi.fn(),
|
||||
removeJoinRequest: vi.fn(),
|
||||
};
|
||||
|
||||
const mockLogger: Logger = {
|
||||
debug: vi.fn(),
|
||||
info: vi.fn(),
|
||||
warn: vi.fn(),
|
||||
error: vi.fn(),
|
||||
};
|
||||
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
});
|
||||
|
||||
it('should return membership data when membership exists', async () => {
|
||||
const useCase = new GetTeamMembershipUseCase(
|
||||
mockMembershipRepo,
|
||||
mockLogger,
|
||||
);
|
||||
|
||||
const teamId = 'team1';
|
||||
const driverId = 'driver1';
|
||||
const membership = {
|
||||
teamId,
|
||||
driverId,
|
||||
role: 'manager' as const,
|
||||
status: 'active' as const,
|
||||
joinedAt: new Date('2023-01-01'),
|
||||
};
|
||||
|
||||
mockGetMembership.mockResolvedValue(membership);
|
||||
|
||||
const result = await useCase.execute({ teamId, driverId });
|
||||
|
||||
expect(result.isOk()).toBe(true);
|
||||
expect(result.value).toEqual({
|
||||
role: 'manager',
|
||||
joinedAt: '2023-01-01T00:00:00.000Z',
|
||||
isActive: true,
|
||||
});
|
||||
});
|
||||
|
||||
it('should return null when no membership exists', async () => {
|
||||
const useCase = new GetTeamMembershipUseCase(
|
||||
mockMembershipRepo,
|
||||
mockLogger,
|
||||
);
|
||||
|
||||
const teamId = 'team1';
|
||||
const driverId = 'driver1';
|
||||
|
||||
mockGetMembership.mockResolvedValue(null);
|
||||
|
||||
const result = await useCase.execute({ teamId, driverId });
|
||||
|
||||
expect(result.isOk()).toBe(true);
|
||||
expect(result.value).toBe(null);
|
||||
});
|
||||
|
||||
it('should map driver role to member', async () => {
|
||||
const useCase = new GetTeamMembershipUseCase(
|
||||
mockMembershipRepo,
|
||||
mockLogger,
|
||||
);
|
||||
|
||||
const teamId = 'team1';
|
||||
const driverId = 'driver1';
|
||||
const membership = {
|
||||
teamId,
|
||||
driverId,
|
||||
role: 'driver' as const,
|
||||
status: 'active' as const,
|
||||
joinedAt: new Date('2023-01-01'),
|
||||
};
|
||||
|
||||
mockGetMembership.mockResolvedValue(membership);
|
||||
|
||||
const result = await useCase.execute({ teamId, driverId });
|
||||
|
||||
expect(result.isOk()).toBe(true);
|
||||
expect(result.value?.role).toBe('member');
|
||||
});
|
||||
|
||||
it('should return error when repository throws', async () => {
|
||||
const useCase = new GetTeamMembershipUseCase(
|
||||
mockMembershipRepo,
|
||||
mockLogger,
|
||||
);
|
||||
|
||||
const teamId = 'team1';
|
||||
const driverId = 'driver1';
|
||||
const error = new Error('Repository error');
|
||||
|
||||
mockGetMembership.mockRejectedValue(error);
|
||||
|
||||
const result = await useCase.execute({ teamId, driverId });
|
||||
|
||||
expect(result.isErr()).toBe(true);
|
||||
expect(result.unwrapErr().code).toBe('REPOSITORY_ERROR');
|
||||
expect(result.unwrapErr().details.message).toBe('Failed to retrieve team membership');
|
||||
});
|
||||
});
|
||||
@@ -0,0 +1,40 @@
|
||||
import type { ITeamMembershipRepository } from '../../domain/repositories/ITeamMembershipRepository';
|
||||
import { Result } from '@core/shared/application/Result';
|
||||
import type { ApplicationErrorCode } from '@core/shared/errors/ApplicationErrorCode';
|
||||
import type { AsyncUseCase, Logger } from '@core/shared/application';
|
||||
|
||||
/**
|
||||
* Use Case for retrieving a driver's membership in a team.
|
||||
*/
|
||||
export class GetTeamMembershipUseCase
|
||||
implements AsyncUseCase<{ teamId: string; driverId: string }, { role: 'owner' | 'manager' | 'member'; joinedAt: string; isActive: boolean } | null, 'REPOSITORY_ERROR'>
|
||||
{
|
||||
constructor(
|
||||
private readonly membershipRepository: ITeamMembershipRepository,
|
||||
private readonly logger: Logger,
|
||||
) {}
|
||||
|
||||
async execute(input: { teamId: string; driverId: string }): Promise<Result<{ role: 'owner' | 'manager' | 'member'; joinedAt: string; isActive: boolean } | null, ApplicationErrorCode<'REPOSITORY_ERROR'>>> {
|
||||
this.logger.debug(`Executing GetTeamMembershipUseCase for teamId: ${input.teamId}, driverId: ${input.driverId}`);
|
||||
|
||||
try {
|
||||
const membership = await this.membershipRepository.getMembership(input.teamId, input.driverId);
|
||||
if (!membership) {
|
||||
this.logger.debug(`No membership found for teamId: ${input.teamId}, driverId: ${input.driverId}`);
|
||||
return Result.ok(null);
|
||||
}
|
||||
|
||||
const result = {
|
||||
role: membership.role === 'driver' ? 'member' : membership.role as 'owner' | 'manager' | 'member',
|
||||
joinedAt: membership.joinedAt.toISOString(),
|
||||
isActive: membership.status === 'active',
|
||||
};
|
||||
|
||||
this.logger.info(`Successfully retrieved membership for teamId: ${input.teamId}, driverId: ${input.driverId}`);
|
||||
return Result.ok(result);
|
||||
} catch (error) {
|
||||
this.logger.error(`Error in GetTeamMembershipUseCase for teamId: ${input.teamId}, driverId: ${input.driverId}`, error as Error);
|
||||
return Result.err({ code: 'REPOSITORY_ERROR', details: { message: 'Failed to retrieve team membership' } });
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,6 +1,7 @@
|
||||
import type { ILeagueMembershipRepository } from '../../domain/repositories/ILeagueMembershipRepository';
|
||||
import type { IRejectLeagueJoinRequestPresenter, RejectLeagueJoinRequestResultDTO, RejectLeagueJoinRequestViewModel } from '../presenters/IRejectLeagueJoinRequestPresenter';
|
||||
import type { UseCase } from '@core/shared/application/UseCase';
|
||||
import { Result } from '@core/shared/application/Result';
|
||||
import type { ApplicationErrorCode } from '@core/shared/errors/ApplicationErrorCode';
|
||||
import type { AsyncUseCase } from '@core/shared/application';
|
||||
|
||||
export interface RejectLeagueJoinRequestUseCaseParams {
|
||||
requestId: string;
|
||||
@@ -11,13 +12,12 @@ export interface RejectLeagueJoinRequestResultDTO {
|
||||
message: string;
|
||||
}
|
||||
|
||||
export class RejectLeagueJoinRequestUseCase implements UseCase<RejectLeagueJoinRequestUseCaseParams, RejectLeagueJoinRequestResultDTO, RejectLeagueJoinRequestViewModel, IRejectLeagueJoinRequestPresenter> {
|
||||
export class RejectLeagueJoinRequestUseCase implements AsyncUseCase<RejectLeagueJoinRequestUseCaseParams, RejectLeagueJoinRequestResultDTO, 'NO_ERROR'> {
|
||||
constructor(private readonly leagueMembershipRepository: ILeagueMembershipRepository) {}
|
||||
|
||||
async execute(params: RejectLeagueJoinRequestUseCaseParams, presenter: IRejectLeagueJoinRequestPresenter): Promise<void> {
|
||||
async execute(params: RejectLeagueJoinRequestUseCaseParams): Promise<Result<RejectLeagueJoinRequestResultDTO, ApplicationErrorCode<'NO_ERROR'>>> {
|
||||
await this.leagueMembershipRepository.removeJoinRequest(params.requestId);
|
||||
const dto: RejectLeagueJoinRequestResultDTO = { success: true, message: 'Join request rejected.' };
|
||||
presenter.reset();
|
||||
presenter.present(dto);
|
||||
return Result.ok(dto);
|
||||
}
|
||||
}
|
||||
@@ -7,6 +7,7 @@ import type { ILeagueRepository } from '@core/racing/domain/repositories/ILeague
|
||||
import type {
|
||||
MembershipRole,
|
||||
} from '@core/racing/domain/entities/LeagueMembership';
|
||||
import type { TransferLeagueOwnershipResultDTO } from '../presenters/ITransferLeagueOwnershipPresenter';
|
||||
|
||||
export interface TransferLeagueOwnershipCommandDTO {
|
||||
leagueId: string;
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
import { Result } from '@core/shared/application/Result';
|
||||
import type { ApplicationErrorCode } from '@core/shared/errors/ApplicationErrorCode';
|
||||
import type { ILeagueMembershipRepository } from '../../domain/repositories/ILeagueMembershipRepository';
|
||||
import type { UpdateLeagueMemberRoleResultDTO } from '../presenters/IUpdateLeagueMemberRolePresenter';
|
||||
|
||||
export interface UpdateLeagueMemberRoleUseCaseParams {
|
||||
leagueId: string;
|
||||
@@ -11,7 +12,7 @@ export interface UpdateLeagueMemberRoleUseCaseParams {
|
||||
export class UpdateLeagueMemberRoleUseCase {
|
||||
constructor(private readonly leagueMembershipRepository: ILeagueMembershipRepository) {}
|
||||
|
||||
async execute(params: UpdateLeagueMemberRoleUseCaseParams): Promise<Result<void, ApplicationErrorCode<'MEMBERSHIP_NOT_FOUND'>>> {
|
||||
async execute(params: UpdateLeagueMemberRoleUseCaseParams): Promise<Result<UpdateLeagueMemberRoleResultDTO, ApplicationErrorCode<'MEMBERSHIP_NOT_FOUND'>>> {
|
||||
const memberships = await this.leagueMembershipRepository.getLeagueMembers(params.leagueId);
|
||||
const membership = memberships.find(m => m.driverId === params.targetDriverId);
|
||||
if (!membership) {
|
||||
@@ -21,6 +22,6 @@ export class UpdateLeagueMemberRoleUseCase {
|
||||
...membership,
|
||||
role: params.newRole,
|
||||
});
|
||||
return Result.ok(undefined);
|
||||
return Result.ok({ success: true });
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user