feat: 用 SSE 替代 SWR 轮询,实现房间状态实时推送
SSE 断连时自动降级为 2s 轮询,重连后切回 SSE。
This commit is contained in:
@@ -0,0 +1,76 @@
|
|||||||
|
import { buildRoomStatus } from "@/lib/buildRoomStatus";
|
||||||
|
import { subscribe } from "@/lib/roomEvents";
|
||||||
|
|
||||||
|
export const dynamic = "force-dynamic";
|
||||||
|
|
||||||
|
export async function GET(
|
||||||
|
req: Request,
|
||||||
|
{ params }: { params: Promise<{ id: string }> },
|
||||||
|
) {
|
||||||
|
const { id } = await params;
|
||||||
|
|
||||||
|
const encoder = new TextEncoder();
|
||||||
|
|
||||||
|
const stream = new ReadableStream({
|
||||||
|
start(controller) {
|
||||||
|
const send = (obj: object) => {
|
||||||
|
try {
|
||||||
|
controller.enqueue(
|
||||||
|
encoder.encode(`data: ${JSON.stringify(obj)}\n\n`),
|
||||||
|
);
|
||||||
|
} catch {
|
||||||
|
/* controller already closed */
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let alive = true;
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const status = await buildRoomStatus(id);
|
||||||
|
if (!status) {
|
||||||
|
send({ error: "room_not_found" });
|
||||||
|
controller.close();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (alive) send(status);
|
||||||
|
})();
|
||||||
|
|
||||||
|
const unsubscribe = subscribe(id, async () => {
|
||||||
|
if (!alive) return;
|
||||||
|
try {
|
||||||
|
const status = await buildRoomStatus(id);
|
||||||
|
if (status && alive) send(status);
|
||||||
|
} catch {
|
||||||
|
/* ignore transient read errors */
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const heartbeat = setInterval(() => {
|
||||||
|
try {
|
||||||
|
controller.enqueue(encoder.encode(": heartbeat\n\n"));
|
||||||
|
} catch {
|
||||||
|
clearInterval(heartbeat);
|
||||||
|
}
|
||||||
|
}, 30_000);
|
||||||
|
|
||||||
|
req.signal.addEventListener("abort", () => {
|
||||||
|
alive = false;
|
||||||
|
unsubscribe();
|
||||||
|
clearInterval(heartbeat);
|
||||||
|
try {
|
||||||
|
controller.close();
|
||||||
|
} catch {
|
||||||
|
/* already closed */
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return new Response(stream, {
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "text/event-stream",
|
||||||
|
"Cache-Control": "no-cache, no-transform",
|
||||||
|
Connection: "keep-alive",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
import { NextResponse } from "next/server";
|
import { NextResponse } from "next/server";
|
||||||
import { atomicUpdateRoom } from "@/lib/store";
|
import { atomicUpdateRoom } from "@/lib/store";
|
||||||
|
import { notify } from "@/lib/roomEvents";
|
||||||
|
|
||||||
export async function POST(
|
export async function POST(
|
||||||
req: Request,
|
req: Request,
|
||||||
@@ -27,6 +28,8 @@ export async function POST(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
notify(id);
|
||||||
|
|
||||||
return NextResponse.json({
|
return NextResponse.json({
|
||||||
roomId: id,
|
roomId: id,
|
||||||
userCount: updated.users.length,
|
userCount: updated.users.length,
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { NextResponse } from "next/server";
|
import { NextResponse } from "next/server";
|
||||||
import { atomicUpdateRoom } from "@/lib/store";
|
import { atomicUpdateRoom } from "@/lib/store";
|
||||||
|
import { notify } from "@/lib/roomEvents";
|
||||||
|
|
||||||
export async function POST(
|
export async function POST(
|
||||||
req: Request,
|
req: Request,
|
||||||
@@ -36,6 +37,8 @@ export async function POST(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
notify(id);
|
||||||
|
|
||||||
return NextResponse.json({ ok: true });
|
return NextResponse.json({ ok: true });
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error("Failed to reset room:", e);
|
console.error("Failed to reset room:", e);
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import { NextResponse } from "next/server";
|
import { NextResponse } from "next/server";
|
||||||
import { getRoomData } from "@/lib/store";
|
import { buildRoomStatus } from "@/lib/buildRoomStatus";
|
||||||
import type { MatchType } from "@/types";
|
|
||||||
|
|
||||||
export async function GET(
|
export async function GET(
|
||||||
_req: Request,
|
_req: Request,
|
||||||
@@ -9,60 +8,16 @@ export async function GET(
|
|||||||
const { id } = await params;
|
const { id } = await params;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const data = await getRoomData(id);
|
const status = await buildRoomStatus(id);
|
||||||
|
|
||||||
if (!data) {
|
if (!status) {
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
{ error: "房间不存在或已过期" },
|
{ error: "房间不存在或已过期" },
|
||||||
{ status: 404 },
|
{ status: 404 },
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const total = data.restaurants.length;
|
return NextResponse.json(status);
|
||||||
const allFinished =
|
|
||||||
data.users.length > 0 &&
|
|
||||||
data.users.every((u) => (data.swipeCounts[u] ?? 0) >= total);
|
|
||||||
|
|
||||||
let match = data.match;
|
|
||||||
let matchType: MatchType = null;
|
|
||||||
let matchLikes = 0;
|
|
||||||
let runnerUps: { id: string; likes: number }[] = [];
|
|
||||||
|
|
||||||
if (match) {
|
|
||||||
matchType = "unanimous";
|
|
||||||
matchLikes = data.users.length;
|
|
||||||
} else if (allFinished && data.restaurants.length > 0) {
|
|
||||||
const ranked = rankRestaurants(data.likes, data.restaurants);
|
|
||||||
if (ranked[0].likes > 0) {
|
|
||||||
match = ranked[0].id;
|
|
||||||
matchType = "best";
|
|
||||||
matchLikes = ranked[0].likes;
|
|
||||||
runnerUps = ranked.slice(1, 3).filter((r) => r.likes > 0);
|
|
||||||
} else {
|
|
||||||
match = ranked[0].id;
|
|
||||||
matchType = "no_match";
|
|
||||||
matchLikes = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const likeCounts: Record<string, number> = {};
|
|
||||||
for (const [rid, users] of Object.entries(data.likes)) {
|
|
||||||
if (users.length > 0) {
|
|
||||||
likeCounts[rid] = users.length;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return NextResponse.json({
|
|
||||||
roomId: id,
|
|
||||||
userCount: data.users.length,
|
|
||||||
match,
|
|
||||||
matchType,
|
|
||||||
matchLikes,
|
|
||||||
runnerUps,
|
|
||||||
likeCounts,
|
|
||||||
swipeCounts: data.swipeCounts,
|
|
||||||
restaurants: data.restaurants,
|
|
||||||
});
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error("Failed to get room:", e);
|
console.error("Failed to get room:", e);
|
||||||
return NextResponse.json(
|
return NextResponse.json(
|
||||||
@@ -71,13 +26,3 @@ export async function GET(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function rankRestaurants(
|
|
||||||
likes: Record<string, string[]>,
|
|
||||||
restaurants: { id: string; rating: number }[],
|
|
||||||
): { id: string; likes: number }[] {
|
|
||||||
return restaurants
|
|
||||||
.map((r) => ({ id: r.id, likes: likes[r.id]?.length ?? 0, rating: r.rating }))
|
|
||||||
.sort((a, b) => b.likes - a.likes || b.rating - a.rating)
|
|
||||||
.map(({ id, likes: l }) => ({ id, likes: l }));
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { NextResponse } from "next/server";
|
import { NextResponse } from "next/server";
|
||||||
import { atomicUpdateRoom } from "@/lib/store";
|
import { atomicUpdateRoom } from "@/lib/store";
|
||||||
|
import { notify } from "@/lib/roomEvents";
|
||||||
|
|
||||||
export async function POST(
|
export async function POST(
|
||||||
req: Request,
|
req: Request,
|
||||||
@@ -52,6 +53,8 @@ export async function POST(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
notify(id);
|
||||||
|
|
||||||
return NextResponse.json({
|
return NextResponse.json({
|
||||||
match: updated.match,
|
match: updated.match,
|
||||||
likeCount: updated.likes[rid]?.length ?? 0,
|
likeCount: updated.likes[rid]?.length ?? 0,
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { NextResponse } from "next/server";
|
import { NextResponse } from "next/server";
|
||||||
import { atomicUpdateRoom } from "@/lib/store";
|
import { atomicUpdateRoom } from "@/lib/store";
|
||||||
|
import { notify } from "@/lib/roomEvents";
|
||||||
|
|
||||||
export async function POST(
|
export async function POST(
|
||||||
req: Request,
|
req: Request,
|
||||||
@@ -46,6 +47,8 @@ export async function POST(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
notify(id);
|
||||||
|
|
||||||
return NextResponse.json({ ok: true });
|
return NextResponse.json({ ok: true });
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error("Failed to undo swipe:", e);
|
console.error("Failed to undo swipe:", e);
|
||||||
|
|||||||
+39
-12
@@ -1,28 +1,55 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import useSWR from "swr";
|
import useSWR from "swr";
|
||||||
import { useRef } from "react";
|
import { useEffect, useRef } from "react";
|
||||||
import { RoomStatus } from "@/types";
|
import { RoomStatus } from "@/types";
|
||||||
|
|
||||||
const fetcher = (url: string) => fetch(url).then((r) => r.json());
|
const fetcher = (url: string) => fetch(url).then((r) => r.json());
|
||||||
|
|
||||||
export function useRoomPolling(roomId: string) {
|
export function useRoomPolling(roomId: string) {
|
||||||
const settled = useRef(false);
|
|
||||||
|
|
||||||
const { data, error, isLoading, mutate } = useSWR<RoomStatus>(
|
const { data, error, isLoading, mutate } = useSWR<RoomStatus>(
|
||||||
`/api/room/${roomId}`,
|
`/api/room/${roomId}`,
|
||||||
fetcher,
|
fetcher,
|
||||||
{
|
{ revalidateOnFocus: true },
|
||||||
refreshInterval: settled.current ? 0 : 1500,
|
|
||||||
revalidateOnFocus: true,
|
|
||||||
},
|
|
||||||
);
|
);
|
||||||
|
|
||||||
if (data?.match != null) {
|
const fallbackRef = useRef<ReturnType<typeof setInterval> | null>(null);
|
||||||
settled.current = true;
|
|
||||||
} else {
|
useEffect(() => {
|
||||||
settled.current = false;
|
const es = new EventSource(`/api/room/${roomId}/events`);
|
||||||
}
|
|
||||||
|
es.onmessage = (e) => {
|
||||||
|
try {
|
||||||
|
const parsed = JSON.parse(e.data);
|
||||||
|
if (parsed.roomId) {
|
||||||
|
mutate(parsed, { revalidate: false });
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
/* malformed message */
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
es.onerror = () => {
|
||||||
|
if (!fallbackRef.current) {
|
||||||
|
fallbackRef.current = setInterval(() => mutate(), 2000);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
es.onopen = () => {
|
||||||
|
if (fallbackRef.current) {
|
||||||
|
clearInterval(fallbackRef.current);
|
||||||
|
fallbackRef.current = null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
es.close();
|
||||||
|
if (fallbackRef.current) {
|
||||||
|
clearInterval(fallbackRef.current);
|
||||||
|
fallbackRef.current = null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, [roomId, mutate]);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
userCount: data?.userCount ?? 0,
|
userCount: data?.userCount ?? 0,
|
||||||
|
|||||||
@@ -0,0 +1,69 @@
|
|||||||
|
import { getRoomData } from "./store";
|
||||||
|
import type { RoomStatus, MatchType } from "@/types";
|
||||||
|
|
||||||
|
export async function buildRoomStatus(
|
||||||
|
roomId: string,
|
||||||
|
): Promise<RoomStatus | null> {
|
||||||
|
const data = await getRoomData(roomId);
|
||||||
|
if (!data) return null;
|
||||||
|
|
||||||
|
const total = data.restaurants.length;
|
||||||
|
const allFinished =
|
||||||
|
data.users.length > 0 &&
|
||||||
|
data.users.every((u) => (data.swipeCounts[u] ?? 0) >= total);
|
||||||
|
|
||||||
|
let match = data.match;
|
||||||
|
let matchType: MatchType = null;
|
||||||
|
let matchLikes = 0;
|
||||||
|
let runnerUps: { id: string; likes: number }[] = [];
|
||||||
|
|
||||||
|
if (match) {
|
||||||
|
matchType = "unanimous";
|
||||||
|
matchLikes = data.users.length;
|
||||||
|
} else if (allFinished && data.restaurants.length > 0) {
|
||||||
|
const ranked = rankRestaurants(data.likes, data.restaurants);
|
||||||
|
if (ranked[0].likes > 0) {
|
||||||
|
match = ranked[0].id;
|
||||||
|
matchType = "best";
|
||||||
|
matchLikes = ranked[0].likes;
|
||||||
|
runnerUps = ranked.slice(1, 3).filter((r) => r.likes > 0);
|
||||||
|
} else {
|
||||||
|
match = ranked[0].id;
|
||||||
|
matchType = "no_match";
|
||||||
|
matchLikes = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const likeCounts: Record<string, number> = {};
|
||||||
|
for (const [rid, users] of Object.entries(data.likes)) {
|
||||||
|
if (users.length > 0) {
|
||||||
|
likeCounts[rid] = users.length;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
roomId,
|
||||||
|
userCount: data.users.length,
|
||||||
|
match,
|
||||||
|
matchType,
|
||||||
|
matchLikes,
|
||||||
|
runnerUps,
|
||||||
|
likeCounts,
|
||||||
|
swipeCounts: data.swipeCounts,
|
||||||
|
restaurants: data.restaurants,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function rankRestaurants(
|
||||||
|
likes: Record<string, string[]>,
|
||||||
|
restaurants: { id: string; rating: number }[],
|
||||||
|
): { id: string; likes: number }[] {
|
||||||
|
return restaurants
|
||||||
|
.map((r) => ({
|
||||||
|
id: r.id,
|
||||||
|
likes: likes[r.id]?.length ?? 0,
|
||||||
|
rating: r.rating,
|
||||||
|
}))
|
||||||
|
.sort((a, b) => b.likes - a.likes || b.rating - a.rating)
|
||||||
|
.map(({ id, likes: l }) => ({ id, likes: l }));
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
type Listener = () => void;
|
||||||
|
|
||||||
|
const listeners = new Map<string, Set<Listener>>();
|
||||||
|
|
||||||
|
export function subscribe(roomId: string, listener: Listener): () => void {
|
||||||
|
if (!listeners.has(roomId)) {
|
||||||
|
listeners.set(roomId, new Set());
|
||||||
|
}
|
||||||
|
const set = listeners.get(roomId)!;
|
||||||
|
set.add(listener);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
set.delete(listener);
|
||||||
|
if (set.size === 0) listeners.delete(roomId);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function notify(roomId: string) {
|
||||||
|
const set = listeners.get(roomId);
|
||||||
|
if (set) {
|
||||||
|
for (const fn of set) fn();
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user