190 lines
7.0 KiB
TypeScript
190 lines
7.0 KiB
TypeScript
import { Router, Request, Response } from "express";
|
|
import { pipelineOrchestrator } from "./PipelineOrchestrator";
|
|
import { z } from "zod";
|
|
|
|
const router = Router();
|
|
|
|
const createPipelineSchema = z.object({
|
|
prompt: z.string().min(5, "O prompt deve ter pelo menos 5 caracteres"),
|
|
metadata: z.record(z.any()).optional(),
|
|
budget: z.object({
|
|
maxTokens: z.number().optional(),
|
|
maxTimeMs: z.number().optional(),
|
|
maxCalls: z.number().optional(),
|
|
}).optional(),
|
|
});
|
|
|
|
router.post("/", async (req: Request, res: Response) => {
|
|
try {
|
|
const parsed = createPipelineSchema.safeParse(req.body);
|
|
if (!parsed.success) {
|
|
return res.status(400).json({ success: false, error: parsed.error.errors[0].message });
|
|
}
|
|
|
|
const userId = (req.user as any)?.id || "anonymous";
|
|
const metadata = { ...(parsed.data.metadata || {}), budget: parsed.data.budget };
|
|
const pipeline = await pipelineOrchestrator.createPipeline(parsed.data.prompt, userId, metadata);
|
|
const started = await pipelineOrchestrator.startPipeline(pipeline.id);
|
|
|
|
res.json({ success: true, pipeline: started });
|
|
} catch (error: any) {
|
|
console.error("[Pipeline] Erro ao criar:", error);
|
|
res.status(500).json({ success: false, error: error.message });
|
|
}
|
|
});
|
|
|
|
router.get("/", async (req: Request, res: Response) => {
|
|
try {
|
|
const limit = Math.min(parseInt(req.query.limit as string) || 20, 50);
|
|
const pipelines = await pipelineOrchestrator.getRecentPipelines(limit);
|
|
const enriched = await Promise.all(pipelines.map(async (p: any) => {
|
|
const pendingCount = await pipelineOrchestrator.getPendingStagingCount(p.id);
|
|
return { ...p, hasPendingChanges: pendingCount > 0, pendingStagingCount: pendingCount };
|
|
}));
|
|
res.json({ success: true, pipelines: enriched });
|
|
} catch (error: any) {
|
|
res.status(500).json({ success: false, error: error.message });
|
|
}
|
|
});
|
|
|
|
router.get("/:id", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
const details = await pipelineOrchestrator.getPipelineWithDetails(id);
|
|
if (!details) {
|
|
return res.status(404).json({ success: false, error: "Pipeline não encontrado" });
|
|
}
|
|
res.json({ success: true, ...details });
|
|
} catch (error: any) {
|
|
res.status(500).json({ success: false, error: error.message });
|
|
}
|
|
});
|
|
|
|
router.get("/:id/staging", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
const changes = await pipelineOrchestrator.getStagingChanges(id);
|
|
res.json({ success: true, changes });
|
|
} catch (error: any) {
|
|
res.status(500).json({ success: false, error: error.message });
|
|
}
|
|
});
|
|
|
|
router.get("/:id/runbook", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
const runbook = await pipelineOrchestrator.getPipelineRunbook(id);
|
|
if (!runbook) {
|
|
return res.status(404).json({ success: false, error: "Runbook não encontrado" });
|
|
}
|
|
res.json({ success: true, runbook });
|
|
} catch (error: any) {
|
|
res.status(500).json({ success: false, error: error.message });
|
|
}
|
|
});
|
|
|
|
const approveSchema = z.object({
|
|
selectedFiles: z.array(z.string()).optional(),
|
|
});
|
|
|
|
router.post("/:id/approve", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
const parsed = approveSchema.safeParse(req.body);
|
|
const selectedFiles = parsed.success ? parsed.data.selectedFiles : undefined;
|
|
const reviewedBy = (req.user as any)?.id || "user";
|
|
const result = await pipelineOrchestrator.approveStagingChanges(id, reviewedBy, selectedFiles);
|
|
res.json({ success: true, ...result });
|
|
} catch (error: any) {
|
|
res.status(400).json({ success: false, error: error.message });
|
|
}
|
|
});
|
|
|
|
router.post("/:id/reject", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
const reviewedBy = (req.user as any)?.id || "user";
|
|
await pipelineOrchestrator.rejectStagingChanges(id, reviewedBy);
|
|
res.json({ success: true, message: "Alterações rejeitadas" });
|
|
} catch (error: any) {
|
|
res.status(400).json({ success: false, error: error.message });
|
|
}
|
|
});
|
|
|
|
router.post("/:id/rollback", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
const rolledBackBy = (req.user as any)?.id || "user";
|
|
const result = await pipelineOrchestrator.rollbackPipeline(id, rolledBackBy);
|
|
res.json({ success: true, ...result });
|
|
} catch (error: any) {
|
|
res.status(400).json({ success: false, error: error.message });
|
|
}
|
|
});
|
|
|
|
router.get("/:id/stream", async (req: Request, res: Response) => {
|
|
const id = parseInt(req.params.id);
|
|
|
|
res.writeHead(200, {
|
|
"Content-Type": "text/event-stream",
|
|
"Cache-Control": "no-cache",
|
|
"Connection": "keep-alive",
|
|
});
|
|
|
|
const sendEvent = (event: string, data: any) => {
|
|
res.write(`event: ${event}\ndata: ${JSON.stringify(data)}\n\n`);
|
|
};
|
|
|
|
const onPhaseStarted = (d: any) => { if (d.pipelineId === id) sendEvent("phase_started", d); };
|
|
const onPhaseCompleted = (d: any) => { if (d.pipelineId === id) sendEvent("phase_completed", d); };
|
|
const onStagingReady = (d: any) => { if (d.pipelineId === id) sendEvent("staging_ready", d); };
|
|
const onCompleted = (d: any) => { if (d.pipelineId === id) sendEvent("completed", d); };
|
|
const onFailed = (d: any) => { if (d.pipelineId === id) sendEvent("failed", d); };
|
|
const onRolledBack = (d: any) => { if (d.pipelineId === id) sendEvent("rolled_back", d); };
|
|
|
|
pipelineOrchestrator.on("pipeline:phase_started", onPhaseStarted);
|
|
pipelineOrchestrator.on("pipeline:phase_completed", onPhaseCompleted);
|
|
pipelineOrchestrator.on("pipeline:staging_ready", onStagingReady);
|
|
pipelineOrchestrator.on("pipeline:completed", onCompleted);
|
|
pipelineOrchestrator.on("pipeline:failed", onFailed);
|
|
pipelineOrchestrator.on("pipeline:rolled_back", onRolledBack);
|
|
|
|
const pollInterval = setInterval(async () => {
|
|
try {
|
|
const pipeline = await pipelineOrchestrator.getPipeline(id);
|
|
if (pipeline) {
|
|
sendEvent("status", {
|
|
status: pipeline.status,
|
|
phase: pipeline.currentPhase,
|
|
phases: pipeline.phases,
|
|
budget: pipeline.budget,
|
|
correlationId: pipeline.correlationId,
|
|
});
|
|
}
|
|
} catch {}
|
|
}, 5000);
|
|
|
|
const initialPipeline = await pipelineOrchestrator.getPipeline(id);
|
|
if (initialPipeline) {
|
|
sendEvent("status", {
|
|
status: initialPipeline.status,
|
|
phase: initialPipeline.currentPhase,
|
|
phases: initialPipeline.phases,
|
|
budget: initialPipeline.budget,
|
|
correlationId: initialPipeline.correlationId,
|
|
});
|
|
}
|
|
|
|
req.on("close", () => {
|
|
clearInterval(pollInterval);
|
|
pipelineOrchestrator.off("pipeline:phase_started", onPhaseStarted);
|
|
pipelineOrchestrator.off("pipeline:phase_completed", onPhaseCompleted);
|
|
pipelineOrchestrator.off("pipeline:staging_ready", onStagingReady);
|
|
pipelineOrchestrator.off("pipeline:completed", onCompleted);
|
|
pipelineOrchestrator.off("pipeline:failed", onFailed);
|
|
pipelineOrchestrator.off("pipeline:rolled_back", onRolledBack);
|
|
});
|
|
});
|
|
|
|
export default router;
|