2025-12-06 12:46:40 +09:00
|
|
|
import { randomUUID } from "crypto"
|
|
|
|
|
import { z } from "zod"
|
|
|
|
|
import { getLangfuseClient } from "@/lib/langfuse"
|
2025-12-05 21:15:02 +09:00
|
|
|
|
|
|
|
|
const saveSchema = z.object({
|
2025-12-06 12:46:40 +09:00
|
|
|
filename: z.string().min(1).max(255),
|
|
|
|
|
format: z.enum(["drawio", "png", "svg"]),
|
|
|
|
|
sessionId: z.string().min(1).max(200).optional(),
|
|
|
|
|
})
|
2025-12-05 21:15:02 +09:00
|
|
|
|
|
|
|
|
export async function POST(req: Request) {
|
2025-12-06 12:46:40 +09:00
|
|
|
const langfuse = getLangfuseClient()
|
|
|
|
|
if (!langfuse) {
|
|
|
|
|
return Response.json({ success: true, logged: false })
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Validate input
|
|
|
|
|
let data
|
|
|
|
|
try {
|
|
|
|
|
data = saveSchema.parse(await req.json())
|
|
|
|
|
} catch {
|
|
|
|
|
return Response.json(
|
|
|
|
|
{ success: false, error: "Invalid input" },
|
|
|
|
|
{ status: 400 },
|
|
|
|
|
)
|
|
|
|
|
}
|
2025-12-05 21:15:02 +09:00
|
|
|
|
2025-12-06 12:46:40 +09:00
|
|
|
const { filename, format, sessionId } = data
|
2025-12-05 21:15:02 +09:00
|
|
|
|
2025-12-23 16:26:45 +09:00
|
|
|
// Skip logging if no sessionId - prevents attaching to wrong user's trace
|
|
|
|
|
if (!sessionId) {
|
|
|
|
|
return Response.json({ success: true, logged: false })
|
|
|
|
|
}
|
|
|
|
|
|
2025-12-06 12:46:40 +09:00
|
|
|
try {
|
|
|
|
|
const timestamp = new Date().toISOString()
|
2025-12-05 21:15:02 +09:00
|
|
|
|
2025-12-06 12:46:40 +09:00
|
|
|
// Find the most recent chat trace for this session to attach the save flag
|
|
|
|
|
const tracesResponse = await langfuse.api.trace.list({
|
|
|
|
|
sessionId,
|
|
|
|
|
limit: 1,
|
|
|
|
|
})
|
2025-12-05 21:15:02 +09:00
|
|
|
|
2025-12-06 12:46:40 +09:00
|
|
|
const traces = tracesResponse.data || []
|
|
|
|
|
const latestTrace = traces[0]
|
2025-12-05 21:15:02 +09:00
|
|
|
|
2025-12-06 12:46:40 +09:00
|
|
|
if (latestTrace) {
|
|
|
|
|
// Add a score to the existing trace to flag that user saved
|
|
|
|
|
await langfuse.api.ingestion.batch({
|
|
|
|
|
batch: [
|
|
|
|
|
{
|
|
|
|
|
type: "score-create",
|
|
|
|
|
id: randomUUID(),
|
|
|
|
|
timestamp,
|
|
|
|
|
body: {
|
|
|
|
|
id: randomUUID(),
|
|
|
|
|
traceId: latestTrace.id,
|
|
|
|
|
name: "diagram-saved",
|
|
|
|
|
value: 1,
|
|
|
|
|
comment: `User saved diagram as ${filename}.${format}`,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
// If no trace found, skip logging (user hasn't chatted yet)
|
2025-12-05 21:15:02 +09:00
|
|
|
|
2025-12-06 12:46:40 +09:00
|
|
|
return Response.json({ success: true, logged: !!latestTrace })
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error("Langfuse save error:", error)
|
|
|
|
|
return Response.json(
|
|
|
|
|
{ success: false, error: "Failed to log save" },
|
|
|
|
|
{ status: 500 },
|
|
|
|
|
)
|
2025-12-05 21:15:02 +09:00
|
|
|
}
|
|
|
|
|
}
|