mirror of
https://github.com/kjanat/livedash-node.git
synced 2026-02-13 23:15:43 +01:00
- Fix syntax errors in skills markdown files (.github/skills, .opencode/skills) - Change typescript to tsx for code blocks with JSX - Replace ellipsis (...) in array examples with valid syntax - Separate CSS from TypeScript into distinct code blocks - Convert JavaScript object examples to valid JSON in docs - Fix enum definitions with proper comma separation
117 lines
3.5 KiB
TypeScript
117 lines
3.5 KiB
TypeScript
import { ProcessingStage } from "@prisma/client";
|
|
import { type NextRequest, NextResponse } from "next/server";
|
|
import { neonAuth } from "@/lib/auth/server";
|
|
import { prisma } from "@/lib/prisma";
|
|
import { processUnprocessedSessions } from "@/lib/processingScheduler";
|
|
import { getSessionsNeedingProcessing } from "@/lib/processingStatusManager";
|
|
|
|
// MIGRATED: Removed "export const dynamic = 'force-dynamic'" - dynamic by default with Cache Components
|
|
|
|
export async function POST(request: NextRequest) {
|
|
const { session: authSession, user: authUser } = await neonAuth();
|
|
if (!authSession || !authUser?.email) {
|
|
return NextResponse.json({ error: "Not logged in" }, { status: 401 });
|
|
}
|
|
|
|
const user = await prisma.user.findUnique({
|
|
where: { email: authUser.email },
|
|
select: {
|
|
id: true,
|
|
email: true,
|
|
role: true,
|
|
companyId: true,
|
|
company: {
|
|
select: {
|
|
id: true,
|
|
name: true,
|
|
status: true,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
if (!user) {
|
|
return NextResponse.json({ error: "No user found" }, { status: 401 });
|
|
}
|
|
|
|
// Check if user has ADMIN role
|
|
if (user.role !== "ADMIN") {
|
|
return NextResponse.json(
|
|
{ error: "Admin access required" },
|
|
{ status: 403 }
|
|
);
|
|
}
|
|
|
|
try {
|
|
// Get optional parameters from request body
|
|
const body = await request.json();
|
|
const { batchSize, maxConcurrency } = body;
|
|
|
|
// Validate parameters
|
|
const validatedBatchSize =
|
|
batchSize && batchSize > 0 ? Number.parseInt(batchSize) : null;
|
|
const validatedMaxConcurrency =
|
|
maxConcurrency && maxConcurrency > 0
|
|
? Number.parseInt(maxConcurrency)
|
|
: 5;
|
|
|
|
// Check how many sessions need AI processing using the new status system
|
|
const sessionsNeedingAI = await getSessionsNeedingProcessing(
|
|
ProcessingStage.AI_ANALYSIS,
|
|
1000 // Get count only
|
|
);
|
|
|
|
// Filter to sessions for this company
|
|
const companySessionsNeedingAI = sessionsNeedingAI.filter(
|
|
(statusRecord) => statusRecord.session.companyId === user.companyId
|
|
);
|
|
|
|
const unprocessedCount = companySessionsNeedingAI.length;
|
|
|
|
if (unprocessedCount === 0) {
|
|
return NextResponse.json({
|
|
success: true,
|
|
message: "No sessions requiring AI processing found",
|
|
unprocessedCount: 0,
|
|
processedCount: 0,
|
|
});
|
|
}
|
|
|
|
// Start processing (this will run asynchronously)
|
|
const _startTime = Date.now();
|
|
|
|
// Note: We're calling the function but not awaiting it to avoid timeout
|
|
// The processing will continue in the background
|
|
processUnprocessedSessions(validatedBatchSize, validatedMaxConcurrency)
|
|
.then(() => {
|
|
console.log(
|
|
`[Manual Trigger] Processing completed for company ${user.companyId}`
|
|
);
|
|
})
|
|
.catch((error) => {
|
|
console.error(
|
|
`[Manual Trigger] Processing failed for company ${user.companyId}:`,
|
|
error
|
|
);
|
|
});
|
|
|
|
return NextResponse.json({
|
|
success: true,
|
|
message: `Started processing ${unprocessedCount} unprocessed sessions`,
|
|
unprocessedCount,
|
|
batchSize: validatedBatchSize || unprocessedCount,
|
|
maxConcurrency: validatedMaxConcurrency,
|
|
startedAt: new Date().toISOString(),
|
|
});
|
|
} catch (error) {
|
|
console.error("[Manual Trigger] Error:", error);
|
|
return NextResponse.json(
|
|
{
|
|
error: "Failed to trigger processing",
|
|
details: error instanceof Error ? error.message : String(error),
|
|
},
|
|
{ status: 500 }
|
|
);
|
|
}
|
|
}
|