mirror of
https://github.com/kjanat/livedash-node.git
synced 2026-01-16 06:32:10 +01:00
Major code quality overhaul addressing 58% of all linting issues: • Type Safety Improvements: - Replace all any types with proper TypeScript interfaces - Fix Map component shadowing (renamed to CountryMap) - Add comprehensive custom error classes system - Enhance API route type safety • Accessibility Enhancements: - Add explicit button types to all interactive elements - Implement useId() hooks for form element accessibility - Add SVG title attributes for screen readers - Fix static element interactions with keyboard handlers • React Best Practices: - Resolve exhaustive dependencies warnings with useCallback - Extract nested component definitions to top level - Fix array index keys with proper unique identifiers - Improve component organization and prop typing • Code Organization: - Automatic import organization and type import optimization - Fix unused function parameters and variables - Enhanced error handling with structured error responses - Improve component reusability and maintainability Results: 248 → 104 total issues (58% reduction) - Fixed all critical type safety and security issues - Enhanced accessibility compliance significantly - Improved code maintainability and performance
125 lines
3.6 KiB
TypeScript
125 lines
3.6 KiB
TypeScript
import { ProcessingStage } from "@prisma/client";
|
|
import { type NextRequest, NextResponse } from "next/server";
|
|
import { getServerSession } from "next-auth";
|
|
import { authOptions } from "../../../../lib/auth";
|
|
import { prisma } from "../../../../lib/prisma";
|
|
import { processUnprocessedSessions } from "../../../../lib/processingScheduler";
|
|
import { ProcessingStatusManager } from "../../../../lib/processingStatusManager";
|
|
|
|
interface SessionUser {
|
|
email: string;
|
|
name?: string;
|
|
}
|
|
|
|
interface SessionData {
|
|
user: SessionUser;
|
|
}
|
|
|
|
export async function POST(request: NextRequest) {
|
|
const session = (await getServerSession(authOptions)) as SessionData | null;
|
|
|
|
if (!session?.user) {
|
|
return NextResponse.json({ error: "Not logged in" }, { status: 401 });
|
|
}
|
|
|
|
const user = await prisma.user.findUnique({
|
|
where: { email: session.user.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 ? parseInt(batchSize) : null;
|
|
const validatedMaxConcurrency =
|
|
maxConcurrency && maxConcurrency > 0 ? parseInt(maxConcurrency) : 5;
|
|
|
|
// Check how many sessions need AI processing using the new status system
|
|
const sessionsNeedingAI =
|
|
await ProcessingStatusManager.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 }
|
|
);
|
|
}
|
|
}
|