generated from fahricansecer/boilerplate-be
This commit is contained in:
0
gemini_diagnostics.log
Normal file
0
gemini_diagnostics.log
Normal file
@@ -15,7 +15,9 @@ import type { ContentGenerationRequest } from './content-generation.service';
|
||||
import type { Platform } from './services/platform-generator.service';
|
||||
import type { VariationType } from './services/variation.service';
|
||||
import type { BrandVoice } from './services/brand-voice.service';
|
||||
import { Public } from '../../common/decorators';
|
||||
import { Public, CurrentUser } from '../../common/decorators';
|
||||
import type { User } from '@prisma/client';
|
||||
|
||||
|
||||
@Controller('content-generation')
|
||||
export class ContentGenerationController {
|
||||
@@ -23,12 +25,35 @@ export class ContentGenerationController {
|
||||
|
||||
// ========== FULL GENERATION ==========
|
||||
|
||||
@Public()
|
||||
@Post('generate')
|
||||
generateContent(@Body() body: ContentGenerationRequest) {
|
||||
return this.service.generateContent(body);
|
||||
async generateContent(
|
||||
@Body() body: ContentGenerationRequest,
|
||||
@CurrentUser() user: any,
|
||||
) {
|
||||
console.log('[ContentGenerationController] Received generation request:', JSON.stringify(body));
|
||||
console.log('[ContentGenerationController] User context:', user?.id || 'NO_USER');
|
||||
|
||||
if (!user) {
|
||||
console.warn('[ContentGenerationController] No user found in context, proceeding without save or with mock user');
|
||||
}
|
||||
|
||||
const bundle = await this.service.generateContent(body);
|
||||
console.log('[ContentGenerationController] Bundle generated, platforms count:', bundle.platforms.length);
|
||||
|
||||
let masterContentId = 'not-saved';
|
||||
if (user?.id) {
|
||||
const saved = await this.service.saveGeneratedBundle(user.id, bundle);
|
||||
masterContentId = saved.masterContentId;
|
||||
console.log('[ContentGenerationController] Bundle saved, masterContentId:', masterContentId);
|
||||
}
|
||||
|
||||
return {
|
||||
...bundle,
|
||||
masterContentId,
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
// ========== NICHES ==========
|
||||
|
||||
@Get('niches')
|
||||
|
||||
@@ -14,9 +14,12 @@ import { VariationService } from './services/variation.service';
|
||||
import { SeoModule } from '../seo/seo.module';
|
||||
import { NeuroMarketingModule } from '../neuro-marketing/neuro-marketing.module';
|
||||
import { GeminiModule } from '../gemini/gemini.module';
|
||||
import { VisualGenerationModule } from '../visual-generation/visual-generation.module';
|
||||
|
||||
|
||||
@Module({
|
||||
imports: [PrismaModule, SeoModule, NeuroMarketingModule, GeminiModule],
|
||||
imports: [PrismaModule, SeoModule, NeuroMarketingModule, GeminiModule, VisualGenerationModule],
|
||||
|
||||
providers: [
|
||||
ContentGenerationService,
|
||||
NicheService,
|
||||
|
||||
@@ -2,6 +2,7 @@
|
||||
// Path: src/modules/content-generation/content-generation.service.ts
|
||||
|
||||
import { Injectable, Logger } from '@nestjs/common';
|
||||
import { randomUUID } from 'crypto';
|
||||
import { PrismaService } from '../../database/prisma.service';
|
||||
import { NicheService, Niche, NicheAnalysis } from './services/niche.service';
|
||||
import { DeepResearchService, ResearchResult, ResearchQuery } from './services/deep-research.service';
|
||||
@@ -9,8 +10,11 @@ import { PlatformGeneratorService, Platform, GeneratedContent, MultiPlatformCont
|
||||
import { HashtagService, HashtagSet } from './services/hashtag.service';
|
||||
import { BrandVoiceService, BrandVoice, VoiceApplication } from './services/brand-voice.service';
|
||||
import { VariationService, VariationSet, VariationOptions } from './services/variation.service';
|
||||
import { SeoService, FullSeoAnalysis } from '../seo/seo.service';
|
||||
import { SeoService, FullSeoAnalysis as SeoDTO } from '../seo/seo.service';
|
||||
import { NeuroMarketingService } from '../neuro-marketing/neuro-marketing.service';
|
||||
import { StorageService } from '../visual-generation/services/storage.service';
|
||||
import { ContentType as PrismaContentType, ContentStatus as PrismaContentStatus, MasterContentType as PrismaMasterContentType } from '@prisma/client';
|
||||
|
||||
|
||||
export interface ContentGenerationRequest {
|
||||
topic: string;
|
||||
@@ -62,8 +66,10 @@ export class ContentGenerationService {
|
||||
private readonly variationService: VariationService,
|
||||
private readonly seoService: SeoService,
|
||||
private readonly neuroService: NeuroMarketingService,
|
||||
private readonly storageService: StorageService,
|
||||
) { }
|
||||
|
||||
|
||||
// ========== FULL GENERATION WORKFLOW ==========
|
||||
|
||||
/**
|
||||
@@ -79,6 +85,7 @@ export class ContentGenerationService {
|
||||
brandVoiceId,
|
||||
variationCount = 3,
|
||||
} = request;
|
||||
console.log(`[ContentGenerationService] Starting generation for topic: ${topic}, platforms: ${platforms.join(', ')}`);
|
||||
|
||||
// Analyze niche if provided
|
||||
let nicheAnalysis: NicheAnalysis | undefined;
|
||||
@@ -101,6 +108,7 @@ export class ContentGenerationService {
|
||||
const platformContent: GeneratedContent[] = [];
|
||||
for (const platform of platforms) {
|
||||
try {
|
||||
console.log(`[ContentGenerationService] Generating for platform: ${platform}`);
|
||||
// Use AI generation when available
|
||||
const aiContent = await this.platformService.generateAIContent(
|
||||
topic,
|
||||
@@ -109,6 +117,11 @@ export class ContentGenerationService {
|
||||
'standard',
|
||||
'tr',
|
||||
);
|
||||
console.log(`[ContentGenerationService] AI content length for ${platform}: ${aiContent?.length || 0}`);
|
||||
|
||||
if (!aiContent || aiContent.trim().length === 0) {
|
||||
console.warn(`[ContentGenerationService] AI Content is empty for ${platform}`);
|
||||
}
|
||||
|
||||
const config = this.platformService.getPlatformConfig(platform);
|
||||
let content: GeneratedContent = {
|
||||
@@ -135,12 +148,15 @@ export class ContentGenerationService {
|
||||
}
|
||||
|
||||
platformContent.push(content);
|
||||
console.log(`[ContentGenerationService] Successfully pushed content for ${platform}`);
|
||||
} catch (error) {
|
||||
this.logger.error(`Failed to generate content for platform ${platform}: ${error.message}`);
|
||||
// Continue to next platform
|
||||
}
|
||||
}
|
||||
|
||||
console.log(`[ContentGenerationService] Generated content for ${platformContent.length} platforms`);
|
||||
|
||||
// Generate variations for primary platform
|
||||
const variations: VariationSet[] = [];
|
||||
if (variationCount > 0 && platformContent.length > 0) {
|
||||
@@ -189,7 +205,7 @@ export class ContentGenerationService {
|
||||
}
|
||||
|
||||
return {
|
||||
id: `gen-${Date.now()}`,
|
||||
id: randomUUID(),
|
||||
topic,
|
||||
niche: nicheAnalysis,
|
||||
research,
|
||||
@@ -201,6 +217,101 @@ export class ContentGenerationService {
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* Persist generated content bundle to database
|
||||
*/
|
||||
async saveGeneratedBundle(userId: string, bundle: GeneratedContentBundle): Promise<{ masterContentId: string }> {
|
||||
console.log(`[ContentGenerationService] Saving bundle for user ${userId}, topic: ${bundle.topic}`);
|
||||
try {
|
||||
return await this.prisma.$transaction(async (tx) => {
|
||||
// 1. Create DeepResearch if it exists in bundle
|
||||
let researchId: string | undefined;
|
||||
if (bundle.research) {
|
||||
const research = await tx.deepResearch.create({
|
||||
data: {
|
||||
userId,
|
||||
topic: bundle.topic,
|
||||
query: bundle.topic, // Simplified for now
|
||||
summary: bundle.research.summary,
|
||||
sources: JSON.parse(JSON.stringify(bundle.research.sources)),
|
||||
keyFindings: JSON.parse(JSON.stringify(bundle.research.keyFindings)),
|
||||
status: 'completed',
|
||||
completedAt: new Date(),
|
||||
}
|
||||
});
|
||||
researchId = research.id;
|
||||
}
|
||||
|
||||
// 2. Create MasterContent
|
||||
const masterContent = await tx.masterContent.create({
|
||||
data: {
|
||||
userId,
|
||||
title: bundle.topic,
|
||||
body: bundle.platforms[0]?.content || '', // Use first platform as master body for now
|
||||
type: PrismaMasterContentType.BLOG, // Default
|
||||
status: PrismaContentStatus.DRAFT,
|
||||
researchId,
|
||||
summary: bundle.research?.summary,
|
||||
}
|
||||
});
|
||||
|
||||
// 3. Create platform-specific content
|
||||
for (const platformContent of bundle.platforms) {
|
||||
// Map SocialPlatform/Platform to ContentType enum
|
||||
let contentType: PrismaContentType = PrismaContentType.BLOG;
|
||||
if (platformContent.platform === 'twitter') contentType = PrismaContentType.TWITTER;
|
||||
else if (platformContent.platform === 'instagram') contentType = PrismaContentType.INSTAGRAM;
|
||||
else if (platformContent.platform === 'linkedin') contentType = PrismaContentType.LINKEDIN;
|
||||
else if (platformContent.platform === 'facebook') contentType = PrismaContentType.FACEBOOK;
|
||||
else if (platformContent.platform === 'tiktok') contentType = PrismaContentType.TIKTOK;
|
||||
|
||||
const content = await tx.content.create({
|
||||
data: {
|
||||
userId,
|
||||
masterContentId: masterContent.id,
|
||||
type: contentType,
|
||||
title: `${bundle.topic} - ${platformContent.platform}`,
|
||||
body: platformContent.content,
|
||||
hashtags: platformContent.hashtags,
|
||||
status: PrismaContentStatus.DRAFT,
|
||||
researchId,
|
||||
}
|
||||
});
|
||||
|
||||
// Save SEO data if available
|
||||
if (bundle.seo) {
|
||||
await tx.contentSeo.create({
|
||||
data: {
|
||||
contentId: content.id,
|
||||
metaTitle: bundle.seo.meta.title,
|
||||
metaDescription: bundle.seo.meta.description,
|
||||
seoScore: bundle.seo.score,
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Save Psychology data if available
|
||||
if (bundle.neuro) {
|
||||
await tx.contentPsychology.create({
|
||||
data: {
|
||||
contentId: content.id,
|
||||
triggersUsed: bundle.neuro.triggersUsed,
|
||||
engagementScore: bundle.neuro.score,
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
console.log(`[ContentGenerationService] Bundle saved successfully. MasterContentId: ${masterContent.id}`);
|
||||
return { masterContentId: masterContent.id };
|
||||
});
|
||||
} catch (error) {
|
||||
console.error(`[ContentGenerationService] Failed to save bundle:`, error);
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// ========== NICHE OPERATIONS ==========
|
||||
|
||||
getNiches(): Niche[] {
|
||||
|
||||
@@ -1,6 +1,14 @@
|
||||
import { Injectable, OnModuleInit, Logger } from '@nestjs/common';
|
||||
import { ConfigService } from '@nestjs/config';
|
||||
import { GoogleGenAI } from '@google/genai';
|
||||
import * as fs from 'fs';
|
||||
import * as path from 'path';
|
||||
|
||||
const LOG_FILE = path.join(process.cwd(), 'gemini_diagnostics.log');
|
||||
function logToFile(message: string) {
|
||||
const timestamp = new Date().toISOString();
|
||||
fs.appendFileSync(LOG_FILE, `[${timestamp}] ${message}\n`);
|
||||
}
|
||||
|
||||
export interface GeminiGenerateOptions {
|
||||
model?: string;
|
||||
@@ -51,7 +59,7 @@ export class GeminiService implements OnModuleInit {
|
||||
this.isEnabled = this.configService.get<boolean>('gemini.enabled', false);
|
||||
this.defaultModel = this.configService.get<string>(
|
||||
'gemini.defaultModel',
|
||||
'gemini-2.5-flash',
|
||||
'gemini-1.5-flash',
|
||||
);
|
||||
}
|
||||
|
||||
@@ -87,7 +95,9 @@ export class GeminiService implements OnModuleInit {
|
||||
* Check if Gemini is available and properly configured
|
||||
*/
|
||||
isAvailable(): boolean {
|
||||
return this.isEnabled && this.client !== null;
|
||||
const available = this.isEnabled && !!this.client;
|
||||
logToFile(`[GeminiService] isAvailable: ${available} (isEnabled: ${this.isEnabled}, hasClient: ${!!this.client})`);
|
||||
return available;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -127,6 +137,7 @@ export class GeminiService implements OnModuleInit {
|
||||
parts: [{ text: prompt }],
|
||||
});
|
||||
|
||||
logToFile(`[GeminiService] Calling generateContent with model: ${model}`);
|
||||
const response = await this.client!.models.generateContent({
|
||||
model,
|
||||
contents,
|
||||
@@ -135,6 +146,7 @@ export class GeminiService implements OnModuleInit {
|
||||
maxOutputTokens: options.maxTokens,
|
||||
},
|
||||
});
|
||||
logToFile(`[GeminiService] Response: ${JSON.stringify(response).substring(0, 1000)}`);
|
||||
|
||||
return {
|
||||
text: (response.text || '').trim(),
|
||||
|
||||
63
src/modules/visual-generation/services/storage.service.ts
Normal file
63
src/modules/visual-generation/services/storage.service.ts
Normal file
@@ -0,0 +1,63 @@
|
||||
import { Injectable, Logger } from '@nestjs/common';
|
||||
import * as fs from 'fs';
|
||||
import * as path from 'path';
|
||||
import { randomUUID } from 'crypto';
|
||||
|
||||
@Injectable()
|
||||
export class StorageService {
|
||||
private readonly logger = new Logger(StorageService.name);
|
||||
private readonly uploadDir = path.join(process.cwd(), 'uploads');
|
||||
|
||||
constructor() {
|
||||
this.ensureUploadDirExists();
|
||||
}
|
||||
|
||||
private ensureUploadDirExists() {
|
||||
if (!fs.existsSync(this.uploadDir)) {
|
||||
fs.mkdirSync(this.uploadDir, { recursive: true });
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Saves a file from a URL or Buffer to the local filesystem
|
||||
* @param source The source URL or Buffer
|
||||
* @param extension The file extension (e.g., '.png')
|
||||
* @returns The public URL and local path
|
||||
*/
|
||||
async saveFile(source: string | Buffer, extension: string): Promise<{ url: string; localPath: string }> {
|
||||
const filename = `${randomUUID()}${extension}`;
|
||||
const localPath = path.join(this.uploadDir, filename);
|
||||
|
||||
if (typeof source === 'string' && source.startsWith('http')) {
|
||||
// In a real scenario, we would download the file here.
|
||||
// For this task, since we are using mock URLs from AI services,
|
||||
// we'll simulate the download by creating an empty file or
|
||||
// if it's a base64 string, we would decode it.
|
||||
// Since GeminiImageService returns mock storage.example.com URLs,
|
||||
// we'll just log and create a placeholder file for now to satisfy "save to disk".
|
||||
this.logger.log(`Simulating download from ${source} to ${localPath}`);
|
||||
fs.writeFileSync(localPath, Buffer.from('placeholder content'));
|
||||
} else if (Buffer.isBuffer(source)) {
|
||||
fs.writeFileSync(localPath, source);
|
||||
} else {
|
||||
throw new Error('Invalid file source');
|
||||
}
|
||||
|
||||
// Return the relative URL that can be served by the static file middleware
|
||||
// Assuming the backend serves the 'uploads' directory at '/uploads'
|
||||
const publicUrl = `/uploads/${filename}`;
|
||||
|
||||
return { url: publicUrl, localPath };
|
||||
}
|
||||
|
||||
/**
|
||||
* Deletes a file from the filesystem
|
||||
* @param filename The name of the file to delete
|
||||
*/
|
||||
async deleteFile(filename: string): Promise<void> {
|
||||
const localPath = path.join(this.uploadDir, filename);
|
||||
if (fs.existsSync(localPath)) {
|
||||
fs.unlinkSync(localPath);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -10,6 +10,8 @@ import { VeoVideoService } from './services/veo-video.service';
|
||||
import { NeuroVisualService } from './services/neuro-visual.service';
|
||||
import { TemplateEditorService } from './services/template-editor.service';
|
||||
import { AssetLibraryService } from './services/asset-library.service';
|
||||
import { StorageService } from './services/storage.service';
|
||||
|
||||
|
||||
@Module({
|
||||
imports: [PrismaModule],
|
||||
@@ -20,8 +22,10 @@ import { AssetLibraryService } from './services/asset-library.service';
|
||||
NeuroVisualService,
|
||||
TemplateEditorService,
|
||||
AssetLibraryService,
|
||||
StorageService,
|
||||
],
|
||||
controllers: [VisualGenerationController],
|
||||
exports: [VisualGenerationService],
|
||||
exports: [VisualGenerationService, StorageService],
|
||||
|
||||
})
|
||||
export class VisualGenerationModule { }
|
||||
|
||||
Reference in New Issue
Block a user