mirror of
https://gitee.com/wanwujie/deer-flow
synced 2026-04-08 00:10:21 +08:00
174 lines
4.5 KiB
TypeScript
174 lines
4.5 KiB
TypeScript
// Copyright (c) 2025 Bytedance Ltd. and/or its affiliates
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
import { env } from "~/env";
|
|
|
|
import type { MCPServerMetadata } from "../mcp";
|
|
import { extractReplayIdFromSearchParams } from "../replay/get-replay-id";
|
|
import { fetchStream } from "../sse";
|
|
import { sleep } from "../utils";
|
|
|
|
import { resolveServiceURL } from "./resolve-service-url";
|
|
import type { ChatEvent } from "./types";
|
|
|
|
export async function* chatStream(
|
|
userMessage: string,
|
|
params: {
|
|
thread_id: string;
|
|
auto_accepted_plan: boolean;
|
|
max_plan_iterations: number;
|
|
max_step_num: number;
|
|
interrupt_feedback?: string;
|
|
enable_background_investigation: boolean;
|
|
mcp_settings?: {
|
|
servers: Record<
|
|
string,
|
|
MCPServerMetadata & {
|
|
enabled_tools: string[];
|
|
add_to_agents: string[];
|
|
}
|
|
>;
|
|
};
|
|
},
|
|
options: { abortSignal?: AbortSignal } = {},
|
|
) {
|
|
if (
|
|
env.NEXT_PUBLIC_STATIC_WEBSITE_ONLY ||
|
|
location.search.includes("mock") ||
|
|
location.search.includes("replay=")
|
|
) {
|
|
return yield* chatReplayStream(userMessage, params, options);
|
|
}
|
|
const stream = fetchStream(resolveServiceURL("chat/stream"), {
|
|
body: JSON.stringify({
|
|
messages: [{ role: "user", content: userMessage }],
|
|
...params,
|
|
}),
|
|
signal: options.abortSignal,
|
|
});
|
|
for await (const event of stream) {
|
|
yield {
|
|
type: event.event,
|
|
data: JSON.parse(event.data),
|
|
} as ChatEvent;
|
|
}
|
|
}
|
|
|
|
async function* chatReplayStream(
|
|
userMessage: string,
|
|
params: {
|
|
thread_id: string;
|
|
auto_accepted_plan: boolean;
|
|
max_plan_iterations: number;
|
|
max_step_num: number;
|
|
interrupt_feedback?: string;
|
|
} = {
|
|
thread_id: "__mock__",
|
|
auto_accepted_plan: false,
|
|
max_plan_iterations: 3,
|
|
max_step_num: 1,
|
|
interrupt_feedback: undefined,
|
|
},
|
|
options: { abortSignal?: AbortSignal } = {},
|
|
): AsyncIterable<ChatEvent> {
|
|
const urlParams = new URLSearchParams(window.location.search);
|
|
let replayFilePath = "";
|
|
if (urlParams.has("mock")) {
|
|
replayFilePath =
|
|
params.interrupt_feedback === "accepted"
|
|
? "/mock/before-interrupt.txt"
|
|
: "/mock/after-interrupt.txt";
|
|
} else {
|
|
const replayId = extractReplayIdFromSearchParams(window.location.search);
|
|
if (replayId) {
|
|
replayFilePath = `/replay/${replayId}.txt`;
|
|
} else {
|
|
// Fallback to a default replay
|
|
replayFilePath = `/replay/eiffel-tower-vs-tallest-building.txt`;
|
|
}
|
|
}
|
|
const text = await fetchReplay(replayFilePath, {
|
|
abortSignal: options.abortSignal,
|
|
});
|
|
const chunks = text.split("\n\n");
|
|
for (const chunk of chunks) {
|
|
const [eventRaw, dataRaw] = chunk.split("\n") as [string, string];
|
|
const [, event] = eventRaw.split("event: ", 2) as [string, string];
|
|
const [, data] = dataRaw.split("data: ", 2) as [string, string];
|
|
|
|
try {
|
|
const chatEvent = {
|
|
type: event,
|
|
data: JSON.parse(data),
|
|
} as ChatEvent;
|
|
if (chatEvent.type === "message_chunk") {
|
|
if (!chatEvent.data.finish_reason) {
|
|
await sleepInReplay(50);
|
|
}
|
|
} else if (chatEvent.type === "tool_call_result") {
|
|
await sleepInReplay(500);
|
|
}
|
|
yield chatEvent;
|
|
if (chatEvent.type === "tool_call_result") {
|
|
await sleepInReplay(800);
|
|
} else if (chatEvent.type === "message_chunk") {
|
|
if (chatEvent.data.role === "user") {
|
|
await sleepInReplay(500);
|
|
}
|
|
}
|
|
} catch (e) {
|
|
console.error(e);
|
|
}
|
|
}
|
|
}
|
|
|
|
const replayCache = new Map<string, string>();
|
|
export async function fetchReplay(
|
|
url: string,
|
|
options: { abortSignal?: AbortSignal } = {},
|
|
) {
|
|
if (replayCache.has(url)) {
|
|
return replayCache.get(url)!;
|
|
}
|
|
const res = await fetch(url, {
|
|
signal: options.abortSignal,
|
|
});
|
|
if (!res.ok) {
|
|
throw new Error(`Failed to fetch replay: ${res.statusText}`);
|
|
}
|
|
const text = await res.text();
|
|
replayCache.set(url, text);
|
|
return text;
|
|
}
|
|
|
|
export async function fetchReplayTitle() {
|
|
const res = chatReplayStream(
|
|
"",
|
|
{
|
|
thread_id: "__mock__",
|
|
auto_accepted_plan: false,
|
|
max_plan_iterations: 3,
|
|
max_step_num: 1,
|
|
},
|
|
{},
|
|
);
|
|
for await (const event of res) {
|
|
if (event.type === "message_chunk") {
|
|
return event.data.content;
|
|
}
|
|
}
|
|
}
|
|
|
|
export async function sleepInReplay(ms: number) {
|
|
if (fastForwardReplaying) {
|
|
await sleep(0);
|
|
} else {
|
|
await sleep(ms);
|
|
}
|
|
}
|
|
|
|
let fastForwardReplaying = false;
|
|
export function fastForwardReplay(value: boolean) {
|
|
fastForwardReplaying = value;
|
|
}
|