65 lines
2.1 KiB
TypeScript
65 lines
2.1 KiB
TypeScript
import { NextRequest, NextResponse } from "next/server";
|
|
import Redis from "ioredis";
|
|
|
|
// Klien Redis khusus untuk Subscriber (Mendengarkan event OmniBrain)
|
|
// Karena ini adalah endpoint streaming, kita membutuhkan instance Redis tersendiri per koneksi (opsional),
|
|
// namun karena SSE bersifat long-lived, kita instansiasi dalam blok handler.
|
|
const redisUrl = process.env.REDIS_URL || "redis://localhost:6379";
|
|
|
|
export async function GET(req: NextRequest) {
|
|
const searchParams = req.nextUrl.searchParams;
|
|
const channel = searchParams.get("channel"); // Misalnya: "room_alpha_events"
|
|
|
|
if (!channel) {
|
|
return NextResponse.json({ error: "Missing channel" }, { status: 400 });
|
|
}
|
|
|
|
const redis = new Redis(redisUrl);
|
|
|
|
const stream = new ReadableStream({
|
|
async start(controller) {
|
|
// 1. Subscribe ke channel spesifik di Redis
|
|
await redis.subscribe(channel, (err, _count) => {
|
|
if (err) {
|
|
console.error("[SSE] Gagal subscribe ke Redis:", err);
|
|
controller.error(err);
|
|
} else {
|
|
console.log(`[SSE] OmniBrain terhubung ke saluran: ${channel}`);
|
|
}
|
|
});
|
|
|
|
// 2. Dengarkan pesan yang dipancarkan oleh API /emit
|
|
redis.on("message", (subChannel, message) => {
|
|
if (subChannel === channel) {
|
|
// Format SSE: "data: {JSON}\n\n"
|
|
controller.enqueue(`data: ${message}\n\n`);
|
|
}
|
|
});
|
|
|
|
// 3. Keep-Alive (Mencegah koneksi HTTP ditutup oleh Proxy/Nginx)
|
|
const keepAlive = setInterval(() => {
|
|
controller.enqueue(":\n\n"); // Komentar SSE untuk keep-alive
|
|
}, 15000);
|
|
|
|
// 4. Deteksi klien putus
|
|
req.signal.addEventListener("abort", () => {
|
|
clearInterval(keepAlive);
|
|
redis.disconnect();
|
|
console.log(`[SSE] Klien terputus dari saluran: ${channel}`);
|
|
});
|
|
},
|
|
cancel() {
|
|
redis.disconnect();
|
|
}
|
|
});
|
|
|
|
return new NextResponse(stream, {
|
|
headers: {
|
|
"Content-Type": "text/event-stream",
|
|
"Cache-Control": "no-cache, no-transform",
|
|
"Connection": "keep-alive",
|
|
"Access-Control-Allow-Origin": "*",
|
|
},
|
|
});
|
|
}
|