mirror of
https://github.com/DayuanJiang/next-ai-draw-io.git
synced 2026-01-03 06:42:27 +08:00
- Add Zod schema validation for log-feedback and log-save endpoints - Create singleton LangfuseClient to avoid per-request instantiation - Simplify log-save to only flag trace (no XML content sent) - Use generic error messages to prevent info leakage
84 lines
2.0 KiB
TypeScript
84 lines
2.0 KiB
TypeScript
import { observe, updateActiveTrace } from '@langfuse/tracing';
|
|
import { LangfuseClient } from '@langfuse/client';
|
|
import * as api from '@opentelemetry/api';
|
|
|
|
// Singleton LangfuseClient instance for direct API calls
|
|
let langfuseClient: LangfuseClient | null = null;
|
|
|
|
export function getLangfuseClient(): LangfuseClient | null {
|
|
if (!process.env.LANGFUSE_PUBLIC_KEY || !process.env.LANGFUSE_SECRET_KEY) {
|
|
return null;
|
|
}
|
|
|
|
if (!langfuseClient) {
|
|
langfuseClient = new LangfuseClient({
|
|
publicKey: process.env.LANGFUSE_PUBLIC_KEY,
|
|
secretKey: process.env.LANGFUSE_SECRET_KEY,
|
|
baseUrl: process.env.LANGFUSE_BASEURL,
|
|
});
|
|
}
|
|
|
|
return langfuseClient;
|
|
}
|
|
|
|
// Check if Langfuse is configured
|
|
export function isLangfuseEnabled(): boolean {
|
|
return !!process.env.LANGFUSE_PUBLIC_KEY;
|
|
}
|
|
|
|
// Update trace with input data at the start of request
|
|
export function setTraceInput(params: {
|
|
input: string;
|
|
sessionId?: string;
|
|
userId?: string;
|
|
}) {
|
|
if (!isLangfuseEnabled()) return;
|
|
|
|
updateActiveTrace({
|
|
name: 'chat',
|
|
input: params.input,
|
|
sessionId: params.sessionId,
|
|
userId: params.userId,
|
|
});
|
|
}
|
|
|
|
// Update trace with output and end the span
|
|
export function setTraceOutput(output: string) {
|
|
if (!isLangfuseEnabled()) return;
|
|
|
|
updateActiveTrace({ output });
|
|
const activeSpan = api.trace.getActiveSpan();
|
|
if (activeSpan) {
|
|
activeSpan.end();
|
|
}
|
|
}
|
|
|
|
// Get telemetry config for streamText
|
|
export function getTelemetryConfig(params: {
|
|
sessionId?: string;
|
|
userId?: string;
|
|
}) {
|
|
if (!isLangfuseEnabled()) return undefined;
|
|
|
|
return {
|
|
isEnabled: true,
|
|
recordInputs: true,
|
|
recordOutputs: true,
|
|
metadata: {
|
|
sessionId: params.sessionId,
|
|
userId: params.userId,
|
|
},
|
|
};
|
|
}
|
|
|
|
// Wrap a handler with Langfuse observe
|
|
export function wrapWithObserve<T>(
|
|
handler: (req: Request) => Promise<T>
|
|
): (req: Request) => Promise<T> {
|
|
if (!isLangfuseEnabled()) {
|
|
return handler;
|
|
}
|
|
|
|
return observe(handler, { name: 'chat', endOnExit: false });
|
|
}
|