Files
dragonsstash/src/app/api/ingestion/trigger/route.ts
2026-03-05 20:34:53 +00:00

65 lines
1.8 KiB
TypeScript

import { NextResponse } from "next/server";
import { authenticateApiRequest } from "@/lib/telegram/api-auth";
import { triggerIngestionSchema } from "@/schemas/telegram";
import { prisma } from "@/lib/prisma";
export const dynamic = "force-dynamic";
export async function POST(request: Request) {
const authResult = await authenticateApiRequest(request, true);
if ("error" in authResult) return authResult.error;
let body: unknown = {};
try {
body = await request.json();
} catch {
// Empty body is fine — triggers all accounts
}
const parsed = triggerIngestionSchema.safeParse(body);
if (!parsed.success) {
return NextResponse.json(
{ error: "Invalid parameters", details: parsed.error.flatten() },
{ status: 400 }
);
}
// Find accounts to trigger
const where: { isActive: boolean; authState: "AUTHENTICATED"; id?: string } = {
isActive: true,
authState: "AUTHENTICATED",
};
if (parsed.data.accountId) {
where.id = parsed.data.accountId;
}
const accounts = await prisma.telegramAccount.findMany({
where,
select: { id: true },
});
if (accounts.length === 0) {
return NextResponse.json(
{ triggered: false, message: "No eligible accounts found" },
{ status: 404 }
);
}
// Send pg_notify for immediate worker pickup.
// The worker creates its own IngestionRun records with proper activity tracking.
try {
await prisma.$queryRawUnsafe(
`SELECT pg_notify('ingestion_trigger', $1)`,
accounts.map((a: { id: string }) => a.id).join(",")
);
} catch {
// pg_notify is best-effort — worker will pick up on next scheduled cycle anyway
}
return NextResponse.json({
triggered: true,
accountIds: accounts.map((a: { id: string }) => a.id),
message: `Ingestion triggered for ${accounts.length} account(s)`,
});
}