mirror of
https://github.com/JezzWTF/vibepod.git
synced 2026-06-01 15:22:14 +00:00
e64048e500
- Add a top-level doc comment to useStreamingGeneration.ts and document the streaming lifecycle. - Add docstrings to helper functions in useStreamingGeneration.ts. - Add section comments to web/app/page.tsx around reducer state, server health polling, and generation handling. - Add file-level comments to API proxy routes explaining the security architecture. - Add a file map / maintainer guide comment to server/vibevoice_server.py. - Add docstrings for key internal helpers in server/vibevoice_server.py. - Document environment variables used by the server in server/vibevoice_server.py. - Add comments identifying VibePod-specific patches around VibeVoice internals. - Format server/vibevoice_server.py with black. Co-authored-by: LyAhn <27559362+LyAhn@users.noreply.github.com>
65 lines
2.1 KiB
TypeScript
65 lines
2.1 KiB
TypeScript
/**
|
|
* API Proxy Route: POST /api/generate
|
|
*
|
|
* This route proxies requests from the frontend to the FastAPI backend's /generate endpoint.
|
|
*
|
|
* Security Architecture:
|
|
* The FastAPI backend is configured to bind only to localhost (127.0.0.1). This prevents
|
|
* unauthenticated public access to the model inference engine. Next.js acts as a secure
|
|
* proxy, allowing the frontend to interact with the backend while maintaining a
|
|
* single public-facing origin.
|
|
*/
|
|
import { NextRequest, NextResponse } from "next/server";
|
|
|
|
export const dynamic = "force-dynamic";
|
|
export const runtime = "nodejs";
|
|
|
|
export async function POST(request: NextRequest) {
|
|
const pythonServerUrl = process.env.VIBEVOICE_SERVER_URL ?? "http://localhost:8000";
|
|
|
|
try {
|
|
const body = (await request.json()) as {
|
|
text: string;
|
|
speaker?: string;
|
|
cfg_scale?: number;
|
|
inference_steps?: number;
|
|
};
|
|
|
|
if (!body.text?.trim()) {
|
|
return NextResponse.json({ error: "Missing or empty text field" }, { status: 400 });
|
|
}
|
|
|
|
const upstream = await fetch(`${pythonServerUrl}/generate`, {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({
|
|
text: body.text.trim(),
|
|
speaker: body.speaker ?? "carter",
|
|
cfg_scale: body.cfg_scale ?? 1.5,
|
|
inference_steps: body.inference_steps ?? 10,
|
|
}),
|
|
signal: request.signal,
|
|
});
|
|
|
|
if (!upstream.ok) {
|
|
const text = await upstream.text().catch(() => "Unknown error");
|
|
return NextResponse.json({ error: text }, { status: upstream.status });
|
|
}
|
|
|
|
// Proxy the SSE stream through to the browser
|
|
return new NextResponse(upstream.body, {
|
|
status: 200,
|
|
headers: {
|
|
"Content-Type": "text/event-stream",
|
|
"Cache-Control": "no-cache, no-transform",
|
|
Connection: "keep-alive",
|
|
"X-Content-Type-Options": "nosniff",
|
|
"X-Accel-Buffering": "no",
|
|
},
|
|
});
|
|
} catch (err) {
|
|
const message = err instanceof Error ? err.message : "Failed to connect to VibeVoice server";
|
|
return NextResponse.json({ error: message }, { status: 502 });
|
|
}
|
|
}
|