mirror of
https://github.com/Tony0410/readlater.git
synced 2026-05-24 22:01:41 +08:00
Add TTS proxy to fix CORS issues
- /api/tts proxies requests to Edge TTS and Kokoro - Uses Docker container names for internal networking - Removes URL config from settings (handled server-side) - Fixes localStorage merge for new settings fields Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
74
src/app/api/tts/route.ts
Normal file
74
src/app/api/tts/route.ts
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
import { NextRequest, NextResponse } from "next/server";
|
||||||
|
|
||||||
|
// POST /api/tts - Proxy TTS requests to avoid CORS issues
|
||||||
|
export async function POST(request: NextRequest) {
|
||||||
|
try {
|
||||||
|
const body = await request.json();
|
||||||
|
const { engine, url, text, voice, speed } = body;
|
||||||
|
|
||||||
|
if (!text) {
|
||||||
|
return NextResponse.json({ error: "Text is required" }, { status: 400 });
|
||||||
|
}
|
||||||
|
|
||||||
|
let ttsUrl: string;
|
||||||
|
let ttsBody: Record<string, unknown>;
|
||||||
|
|
||||||
|
if (engine === "edge") {
|
||||||
|
// Use Docker container name for internal networking, fallback to provided URL
|
||||||
|
const edgeHost = process.env.EDGE_TTS_URL || url || "http://edge-tts:5050";
|
||||||
|
ttsUrl = `${edgeHost}/v1/audio/speech`;
|
||||||
|
ttsBody = {
|
||||||
|
model: "tts-1",
|
||||||
|
input: text,
|
||||||
|
voice: voice || "en-US-AvaNeural",
|
||||||
|
response_format: "mp3",
|
||||||
|
speed: speed || 1.0,
|
||||||
|
};
|
||||||
|
} else if (engine === "kokoro") {
|
||||||
|
// Use Docker container name for internal networking, fallback to provided URL
|
||||||
|
const kokoroHost = process.env.KOKORO_TTS_URL || url || "http://kokoro-tts:8880";
|
||||||
|
ttsUrl = `${kokoroHost}/v1/audio/speech`;
|
||||||
|
ttsBody = {
|
||||||
|
model: "kokoro",
|
||||||
|
input: text,
|
||||||
|
voice: voice || "af_bella",
|
||||||
|
response_format: "mp3",
|
||||||
|
speed: speed || 1.0,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return NextResponse.json({ error: "Invalid engine" }, { status: 400 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch(ttsUrl, {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
body: JSON.stringify(ttsBody),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
const errorText = await response.text();
|
||||||
|
console.error("TTS error:", response.status, errorText);
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: `TTS service error: ${response.status}` },
|
||||||
|
{ status: 502 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const audioBuffer = await response.arrayBuffer();
|
||||||
|
|
||||||
|
return new NextResponse(audioBuffer, {
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "audio/mpeg",
|
||||||
|
"Content-Length": audioBuffer.byteLength.toString(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error("TTS proxy error:", error);
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: error instanceof Error ? error.message : "TTS proxy failed" },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -269,50 +269,22 @@ export function SettingsPanel({
|
|||||||
</section>
|
</section>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Edge TTS URL */}
|
{/* Edge TTS info */}
|
||||||
{ttsSettings.engine === "edge" && (
|
{ttsSettings.engine === "edge" && (
|
||||||
<section className="mb-8">
|
<section className="mb-8">
|
||||||
<h3 className="text-sm font-medium text-[var(--muted)] uppercase tracking-wide mb-4">
|
<p className="text-sm text-[var(--muted)]">
|
||||||
Edge TTS URL
|
Uses Microsoft neural voices via Edge TTS. Fast and natural sounding.
|
||||||
</h3>
|
Requires the edge-tts Docker container running on the server.
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
value={ttsSettings.edgeUrl}
|
|
||||||
onChange={(e) =>
|
|
||||||
onTTSSettingsChange({
|
|
||||||
...ttsSettings,
|
|
||||||
edgeUrl: e.target.value,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
placeholder="http://localhost:5050"
|
|
||||||
className="w-full p-3 rounded-lg border border-[var(--border)] bg-[var(--background)] text-[var(--foreground)]"
|
|
||||||
/>
|
|
||||||
<p className="text-xs text-[var(--muted)] mt-2">
|
|
||||||
Uses Microsoft neural voices. Fast and natural sounding.
|
|
||||||
</p>
|
</p>
|
||||||
</section>
|
</section>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Kokoro URL */}
|
{/* Kokoro info */}
|
||||||
{ttsSettings.engine === "kokoro" && (
|
{ttsSettings.engine === "kokoro" && (
|
||||||
<section className="mb-8">
|
<section className="mb-8">
|
||||||
<h3 className="text-sm font-medium text-[var(--muted)] uppercase tracking-wide mb-4">
|
<p className="text-sm text-[var(--muted)]">
|
||||||
Kokoro API URL
|
High quality local TTS. Slower as it generates full audio before playing.
|
||||||
</h3>
|
Requires the kokoro Docker container running on the server.
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
value={ttsSettings.kokoroUrl}
|
|
||||||
onChange={(e) =>
|
|
||||||
onTTSSettingsChange({
|
|
||||||
...ttsSettings,
|
|
||||||
kokoroUrl: e.target.value,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
placeholder="http://localhost:8880"
|
|
||||||
className="w-full p-3 rounded-lg border border-[var(--border)] bg-[var(--background)] text-[var(--foreground)]"
|
|
||||||
/>
|
|
||||||
<p className="text-xs text-[var(--muted)] mt-2">
|
|
||||||
High quality but slower. Generates full audio before playing.
|
|
||||||
</p>
|
</p>
|
||||||
</section>
|
</section>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -118,22 +118,24 @@ export function useTTS({ settings, text }: UseTTSOptions): UseTTSReturn {
|
|||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const response = await fetch(`${settings.edgeUrl}/v1/audio/speech`, {
|
// Use proxy to avoid CORS issues
|
||||||
|
const response = await fetch("/api/tts", {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
},
|
},
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
model: "tts-1",
|
engine: "edge",
|
||||||
input: textRef.current,
|
url: settings.edgeUrl,
|
||||||
|
text: textRef.current,
|
||||||
voice: settings.voice || "en-US-AvaNeural",
|
voice: settings.voice || "en-US-AvaNeural",
|
||||||
response_format: "mp3",
|
|
||||||
speed: settings.speed,
|
speed: settings.speed,
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error(`Edge TTS API error: ${response.status}`);
|
const error = await response.json().catch(() => ({}));
|
||||||
|
throw new Error(error.error || `TTS error: ${response.status}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const blob = await response.blob();
|
const blob = await response.blob();
|
||||||
@@ -179,7 +181,7 @@ export function useTTS({ settings, text }: UseTTSOptions): UseTTSReturn {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Edge TTS error:", error);
|
console.error("Edge TTS error:", error);
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
alert("Failed to connect to Edge TTS. Make sure it's running at " + settings.edgeUrl);
|
alert(error instanceof Error ? error.message : "Edge TTS failed");
|
||||||
}
|
}
|
||||||
}, [settings.edgeUrl, settings.speed, settings.voice]);
|
}, [settings.edgeUrl, settings.speed, settings.voice]);
|
||||||
|
|
||||||
@@ -187,22 +189,24 @@ export function useTTS({ settings, text }: UseTTSOptions): UseTTSReturn {
|
|||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const response = await fetch(`${settings.kokoroUrl}/v1/audio/speech`, {
|
// Use proxy to avoid CORS issues
|
||||||
|
const response = await fetch("/api/tts", {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
},
|
},
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
model: "kokoro",
|
engine: "kokoro",
|
||||||
input: textRef.current,
|
url: settings.kokoroUrl,
|
||||||
|
text: textRef.current,
|
||||||
voice: "af_bella",
|
voice: "af_bella",
|
||||||
response_format: "mp3",
|
|
||||||
speed: settings.speed,
|
speed: settings.speed,
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error(`Kokoro API error: ${response.status}`);
|
const error = await response.json().catch(() => ({}));
|
||||||
|
throw new Error(error.error || `TTS error: ${response.status}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const blob = await response.blob();
|
const blob = await response.blob();
|
||||||
@@ -249,7 +253,7 @@ export function useTTS({ settings, text }: UseTTSOptions): UseTTSReturn {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Kokoro TTS error:", error);
|
console.error("Kokoro TTS error:", error);
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
alert("Failed to connect to Kokoro. Make sure it's running at " + settings.kokoroUrl);
|
alert(error instanceof Error ? error.message : "Kokoro TTS failed");
|
||||||
}
|
}
|
||||||
}, [settings.kokoroUrl, settings.speed]);
|
}, [settings.kokoroUrl, settings.speed]);
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user