feat: 前端跑通 React Flow 画布→DSL→提交→SSE + 偏好记忆面板 (④)

桌面端前端从骨架变为可用:画布编排 Agent → 导出 DSL → 提交 Gateway →
逐 token 流式展示;偏好记忆面板让用户登记画像(→ memory_upsert)。

- lib/api.ts: submitTask(POST) / streamTokens(EventSource SSE token/done) / setMemory(PUT)
- canvas/AgentCanvas: 加节点(提示词)/连线/运行(导出DSL交上层), React Flow 工具栏
- panels/MemoryPanel: 登记偏好(key/value)→PUT /api/v1/memory
- App: 身份(user/session)+记忆面板+SSE 输出面板,串起提交→流式
- postcss.config + vite-env.d.ts(import.meta.env) 补齐构建;删 WikiPanel(stale Qdrant)
- Gateway: 加 CORS 中间件(放开跨源 + X-User-ID/X-Session-ID 头 + OPTIONS)
- Makefile: web 目标(Vite dev); .claude/launch.json(preview 配置), 忽略 settings.local

验证: npm run build(tsc+vite)✓; 真实浏览器跑通——加节点→运行→SSE 流出含
注入画像(称呼:老王)的回答, 第2轮 UI 显示'已有1轮历史'(短期历史经 Eino 图回灌)

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
Blizzard
2026-06-10 14:32:15 +08:00
parent 4928ffc0f7
commit 7d2891d88a
12 changed files with 3183 additions and 35 deletions
File diff suppressed because it is too large Load Diff
@@ -0,0 +1,6 @@
export default {
plugins: {
tailwindcss: {},
autoprefixer: {},
},
};
+78 -7
View File
@@ -1,15 +1,86 @@
import { AgentCanvas } from "./canvas/AgentCanvas";
import { WikiPanel } from "./wiki/WikiPanel";
import { useCallback, useRef, useState } from "react";
// UI Representation Layer —— 顶层布局:左侧编排画布 + 右侧 Wiki 面板。
import { AgentCanvas } from "./canvas/AgentCanvas";
import { MemoryPanel } from "./panels/MemoryPanel";
import { submitTask, streamTokens, type Identity } from "./lib/api";
import type { TaskDsl } from "./lib/dsl";
// 顶层布局:左侧 React Flow 编排画布 + 右侧 身份 / 偏好记忆 / 运行输出。
export default function App() {
const [identity, setIdentity] = useState<Identity>({ userId: "wt", sessionId: "sess-ui" });
const [output, setOutput] = useState("");
const [status, setStatus] = useState("就绪");
const [running, setRunning] = useState(false);
const closeRef = useRef<(() => void) | null>(null);
const onRun = useCallback(
async (dsl: TaskDsl) => {
closeRef.current?.();
setOutput("");
setRunning(true);
setStatus("提交任务…");
try {
const taskId = await submitTask(dsl, identity);
setStatus(`流式中 · ${taskId}`);
closeRef.current = streamTokens(
taskId,
(t) => setOutput((o) => o + t),
() => {
setStatus("完成 ✓");
setRunning(false);
},
() => {
setStatus("连接中断");
setRunning(false);
},
);
} catch (e) {
setStatus(`${(e as Error).message}`);
setRunning(false);
}
},
[identity],
);
return (
<div className="flex h-screen w-screen">
<div className="flex h-screen w-screen text-gray-900">
<main className="flex-1 border-r">
<AgentCanvas />
<AgentCanvas onRun={onRun} running={running} />
</main>
<aside className="w-96 overflow-auto">
<WikiPanel />
<aside className="flex w-[26rem] flex-col overflow-auto">
<section className="border-b p-4">
<h2 className="mb-2 text-sm font-semibold text-gray-700"> / </h2>
<div className="flex gap-2">
<label className="flex-1 text-xs text-gray-500">
<input
className="mt-1 w-full rounded border px-2 py-1 text-sm text-gray-900"
value={identity.userId}
onChange={(e) => setIdentity((id) => ({ ...id, userId: e.target.value }))}
/>
</label>
<label className="flex-1 text-xs text-gray-500">
<input
className="mt-1 w-full rounded border px-2 py-1 text-sm text-gray-900"
value={identity.sessionId}
onChange={(e) => setIdentity((id) => ({ ...id, sessionId: e.target.value }))}
/>
</label>
</div>
</section>
<MemoryPanel identity={identity} />
<section className="flex flex-1 flex-col p-4">
<div className="mb-2 flex items-center justify-between">
<h2 className="text-sm font-semibold text-gray-700">SSE Token </h2>
<span className="text-xs text-gray-400">{status}</span>
</div>
<pre className="flex-1 whitespace-pre-wrap rounded bg-gray-900 p-3 text-xs leading-relaxed text-emerald-300">
{output || "在左侧加节点 → 运行,模型会注入你的偏好与历史后流式作答。"}
</pre>
</section>
</aside>
</div>
);
@@ -1,4 +1,4 @@
import { useCallback } from "react";
import { useCallback, useState } from "react";
import {
ReactFlow,
Background,
@@ -7,32 +7,66 @@ import {
useNodesState,
useEdgesState,
type Connection,
type Node,
} from "@xyflow/react";
import "@xyflow/react/dist/style.css";
import { exportDsl } from "../lib/dsl";
import { exportDsl, type TaskDsl } from "../lib/dsl";
// React Flow Canvas —— Agent 编排,可导出 JSON DSL 提交到 Gateway。
export function AgentCanvas() {
const [nodes, , onNodesChange] = useNodesState([]);
let seq = 0;
// React Flow Canvas —— Agent 编排:加节点、连线、导出 JSON DSL 并交给上层运行。
export function AgentCanvas({
onRun,
running,
}: {
onRun: (dsl: TaskDsl) => void;
running: boolean;
}) {
const [nodes, setNodes, onNodesChange] = useNodesState<Node>([]);
const [edges, setEdges, onEdgesChange] = useEdgesState([]);
const [prompt, setPrompt] = useState("总结这段文本");
const onConnect = useCallback(
(c: Connection) => setEdges((eds) => addEdge(c, eds)),
[setEdges],
);
const onExport = useCallback(() => {
const dsl = exportDsl(nodes, edges); // → JSON DSL export
// TODO: 经 Wails 强绑定调用 App.SubmitDSL(dsl)
console.log(dsl);
}, [nodes, edges]);
const addNode = useCallback(() => {
const id = `n${++seq}`;
setNodes((ns) => [
...ns,
{
id,
type: "default",
position: { x: 80 + ((seq * 40) % 320), y: 60 + seq * 70 },
data: { label: `${id}: ${prompt}`, prompt },
},
]);
}, [prompt, setNodes]);
const run = useCallback(() => onRun(exportDsl(nodes, edges)), [nodes, edges, onRun]);
return (
<div className="h-full w-full">
<button onClick={onExport} className="absolute z-10 m-2 rounded border px-3 py-1">
DSL
</button>
<div className="relative h-full w-full">
<div className="absolute left-0 right-0 top-0 z-10 flex items-center gap-2 border-b bg-white/90 p-2">
<input
className="flex-1 rounded border px-2 py-1 text-sm"
value={prompt}
onChange={(e) => setPrompt(e.target.value)}
placeholder="节点提示词"
/>
<button onClick={addNode} className="rounded border px-3 py-1 text-sm hover:bg-gray-50">
Agent
</button>
<button
onClick={run}
disabled={running || nodes.length === 0}
className="rounded bg-violet-600 px-3 py-1 text-sm font-medium text-white disabled:opacity-40"
>
{running ? "运行中…" : "▶ 运行"}
</button>
</div>
<ReactFlow
nodes={nodes}
edges={edges}
+68
View File
@@ -0,0 +1,68 @@
// Gateway HTTP/SSE 客户端:提交 DSL 任务、订阅 Token 流、登记偏好记忆。
import type { TaskDsl } from "./dsl";
// 开发期直连 Gateway;Wails 打包后可改为本地后端地址或经 Go 绑定。
export const GATEWAY: string =
(import.meta.env.VITE_GATEWAY as string | undefined) ?? "http://localhost:8080";
export interface Identity {
userId: string;
sessionId: string;
}
// submitTask: POST /api/v1/tasks,返回 task_id。
export async function submitTask(dsl: TaskDsl, id: Identity): Promise<string> {
const res = await fetch(`${GATEWAY}/api/v1/tasks`, {
method: "POST",
headers: {
"Content-Type": "application/json",
"X-User-ID": id.userId,
"X-Session-ID": id.sessionId,
},
body: JSON.stringify(dsl),
});
if (!res.ok) throw new Error(`submit failed: ${res.status} ${await res.text()}`);
const data = (await res.json()) as { task_id: string };
return data.task_id;
}
// streamTokens: 订阅 SSE /api/v1/tasks/:id/stream,逐 token 回调,done 收尾。
// 返回关闭函数。注意 EventSource 无法带请求头,但流按 task_id 寻址,无需身份头。
export function streamTokens(
taskId: string,
onToken: (t: string) => void,
onDone: () => void,
onError?: (e: unknown) => void,
): () => void {
const es = new EventSource(`${GATEWAY}/api/v1/tasks/${taskId}/stream`);
es.addEventListener("token", (e) => onToken((e as MessageEvent).data));
es.addEventListener("done", () => {
es.close();
onDone();
});
es.onerror = (e) => {
es.close();
onError?.(e);
};
return () => es.close();
}
// setMemory: PUT /api/v1/memory,登记一条用户偏好(→ mcp-go memory_upsert)。
export async function setMemory(
id: Identity,
key: string,
value: string,
): Promise<string> {
const res = await fetch(`${GATEWAY}/api/v1/memory`, {
method: "PUT",
headers: {
"Content-Type": "application/json",
"X-User-ID": id.userId,
"X-Session-ID": id.sessionId,
},
body: JSON.stringify({ key, value }),
});
const data = (await res.json()) as { message?: string; error?: string };
if (!res.ok) throw new Error(data.error ?? `memory failed: ${res.status}`);
return data.message ?? "ok";
}
@@ -0,0 +1,54 @@
import { useState } from "react";
import { setMemory, type Identity } from "../lib/api";
// 偏好记忆面板 —— 让用户显式登记/纠正模型对自己的记忆(→ PUT /api/v1/memory)。
export function MemoryPanel({ identity }: { identity: Identity }) {
const [key, setKey] = useState("回答偏好");
const [value, setValue] = useState("简洁、中文、多给要点");
const [saved, setSaved] = useState<Array<{ key: string; value: string }>>([]);
const [msg, setMsg] = useState("");
const save = async () => {
if (!key.trim()) return;
try {
const m = await setMemory(identity, key.trim(), value.trim());
setSaved((s) => [{ key: key.trim(), value: value.trim() }, ...s.filter((x) => x.key !== key.trim())]);
setMsg(m);
} catch (e) {
setMsg(`${(e as Error).message}`);
}
};
return (
<section className="border-b p-4">
<h2 className="mb-2 text-sm font-semibold text-gray-700"></h2>
<div className="flex flex-col gap-2">
<input
className="rounded border px-2 py-1 text-sm"
value={key}
onChange={(e) => setKey(e.target.value)}
placeholder="键,如 称呼 / 回答偏好"
/>
<textarea
className="h-16 resize-none rounded border px-2 py-1 text-sm"
value={value}
onChange={(e) => setValue(e.target.value)}
placeholder="值"
/>
<button onClick={save} className="self-end rounded bg-emerald-600 px-3 py-1 text-sm text-white">
</button>
</div>
{msg && <p className="mt-2 text-xs text-emerald-700">{msg}</p>}
{saved.length > 0 && (
<ul className="mt-3 space-y-1">
{saved.map((s) => (
<li key={s.key} className="rounded bg-gray-50 px-2 py-1 text-xs text-gray-600">
<span className="font-medium">{s.key}</span>{s.value}
</li>
))}
</ul>
)}
</section>
);
}
+1
View File
@@ -0,0 +1 @@
/// <reference types="vite/client" />
@@ -1,13 +0,0 @@
// LLM Wiki Management Panel —— 管理知识库条目,触发第 5 层混合检索。
export function WikiPanel() {
return (
<div className="p-4">
<h2 className="mb-2 text-lg font-semibold">LLM Wiki</h2>
<input
className="mb-3 w-full rounded border px-2 py-1"
placeholder="搜索 WikiHybrid: Bleve + Qdrant + Neo4j"
/>
{/* TODO: 检索结果列表 / 条目编辑 */}
</div>
);
}