This commit is contained in:
2025-12-14 18:11:59 +01:00
parent acc15e8d8d
commit 217337862c
91 changed files with 5919 additions and 1999 deletions

View File

@@ -1,11 +1,4 @@
/**
* Use Case: RequestAvatarGenerationUseCase
*
* Initiates the avatar generation process by validating the face photo
* and creating a generation request.
*/
import type { AsyncUseCase } from '@gridpilot/shared/application';
import type { AsyncUseCase, ILogger } from '@gridpilot/shared/application';
import type { IAvatarGenerationRepository } from '../../domain/repositories/IAvatarGenerationRepository';
import type { FaceValidationPort } from '../ports/FaceValidationPort';
import type { AvatarGenerationPort } from '../ports/AvatarGenerationPort';
@@ -32,89 +25,126 @@ export class RequestAvatarGenerationUseCase
private readonly avatarRepository: IAvatarGenerationRepository,
private readonly faceValidation: FaceValidationPort,
private readonly avatarGeneration: AvatarGenerationPort,
private readonly logger: ILogger,
) {}
async execute(command: RequestAvatarGenerationCommand): Promise<RequestAvatarGenerationResult> {
// Create the generation request
const requestId = this.generateId();
const request = AvatarGenerationRequest.create({
id: requestId,
userId: command.userId,
facePhotoUrl: `data:image/jpeg;base64,${command.facePhotoData}`,
suitColor: command.suitColor,
...(command.style ? { style: command.style } : {}),
});
this.logger.debug(
`Executing RequestAvatarGenerationUseCase for userId: ${command.userId}`,
command,
);
// Mark as validating
request.markAsValidating();
await this.avatarRepository.save(request);
try {
// Create the generation request
const requestId = this.generateId();
const request = AvatarGenerationRequest.create({
id: requestId,
userId: command.userId,
facePhotoUrl: `data:image/jpeg;base64,${command.facePhotoData}`,
suitColor: command.suitColor,
...(command.style ? { style: command.style } : {}),
});
// Validate the face photo
const validationResult = await this.faceValidation.validateFacePhoto(command.facePhotoData);
if (!validationResult.isValid) {
request.fail(validationResult.errorMessage || 'Face validation failed');
this.logger.info(`Avatar generation request created with ID: ${requestId}`);
// Mark as validating
request.markAsValidating();
await this.avatarRepository.save(request);
this.logger.debug(`Request ${requestId} marked as validating.`);
// 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';
request.fail(errorMessage);
await this.avatarRepository.save(request);
this.logger.error(`Face validation failed for request ${requestId}: ${errorMessage}`);
return {
requestId,
status: 'failed',
errorMessage: validationResult.errorMessage || 'Please upload a clear photo of your face',
};
}
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}`);
return {
requestId,
status: 'failed',
errorMessage: 'No face detected. Please upload a photo that clearly shows your face.',
};
}
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}`);
return {
requestId,
status: 'failed',
errorMessage: 'Multiple faces detected. Please upload a photo with only your face.',
};
}
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,
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,
);
if (!generationResult.success) {
const errorMessage = generationResult.errorMessage || 'Avatar generation failed';
request.fail(errorMessage);
await this.avatarRepository.save(request);
this.logger.error(`Avatar generation failed for request ${requestId}: ${errorMessage}`);
return {
requestId,
status: 'failed',
errorMessage: generationResult.errorMessage || 'Failed to generate avatars. Please try again.',
};
}
// Complete with generated avatars
const avatarUrls = generationResult.avatars.map(a => a.url);
request.completeWithAvatars(avatarUrls);
await this.avatarRepository.save(request);
this.logger.info(`Avatar generation completed successfully for request ${requestId}.`);
return {
requestId,
status: 'failed',
errorMessage: validationResult.errorMessage || 'Please upload a clear photo of your face',
status: 'completed',
avatarUrls,
};
} catch (error) {
this.logger.error(
`An unexpected error occurred during avatar generation for userId: ${command.userId}`,
error,
);
// Re-throw or return a generic error, depending on desired error handling strategy
throw error;
}
if (!validationResult.hasFace) {
request.fail('No face detected in the image');
await this.avatarRepository.save(request);
return {
requestId,
status: 'failed',
errorMessage: 'No face detected. Please upload a photo that clearly shows your face.',
};
}
if (validationResult.faceCount > 1) {
request.fail('Multiple faces detected');
await this.avatarRepository.save(request);
return {
requestId,
status: 'failed',
errorMessage: 'Multiple faces detected. Please upload a photo with only your face.',
};
}
// Mark as generating
request.markAsGenerating();
await this.avatarRepository.save(request);
// Generate avatars
const generationResult = await this.avatarGeneration.generateAvatars({
facePhotoUrl: request.facePhotoUrl.value,
prompt: request.buildPrompt(),
suitColor: request.suitColor,
style: request.style,
count: 3, // Generate 3 options
});
if (!generationResult.success) {
request.fail(generationResult.errorMessage || 'Avatar generation failed');
await this.avatarRepository.save(request);
return {
requestId,
status: 'failed',
errorMessage: generationResult.errorMessage || 'Failed to generate avatars. Please try again.',
};
}
// Complete with generated avatars
const avatarUrls = generationResult.avatars.map(a => a.url);
request.completeWithAvatars(avatarUrls);
await this.avatarRepository.save(request);
return {
requestId,
status: 'completed',
avatarUrls,
};
}
private generateId(): string {