import test from "node:test"; import assert from "node:assert/strict"; import os from "node:os"; import path from "node:path"; import { mkdtemp, rm } from "node:fs/promises"; let runtimeRoot = ""; let readState: (typeof import("../src/lib/boss-data"))["readState"]; let writeState: (typeof import("../src/lib/boss-data"))["writeState"]; let saveAiAccount: (typeof import("../src/lib/boss-data"))["saveAiAccount"]; let updateMasterAgentPromptPolicy: (typeof import("../src/lib/boss-data"))["updateMasterAgentPromptPolicy"]; let updateUserMasterPrompt: (typeof import("../src/lib/boss-data"))["updateUserMasterPrompt"]; let createUserMasterMemory: (typeof import("../src/lib/boss-data"))["createUserMasterMemory"]; let updateProjectAgentControls: (typeof import("../src/lib/boss-data"))["updateProjectAgentControls"]; let appendProjectMessage: (typeof import("../src/lib/boss-data"))["appendProjectMessage"]; let resolveMasterAgentExecutionConfig: (typeof import("../src/lib/boss-master-agent"))["resolveMasterAgentExecutionConfig"]; let replyToMasterAgentUserMessage: (typeof import("../src/lib/boss-master-agent"))["replyToMasterAgentUserMessage"]; let completeMasterAgentTask: (typeof import("../src/lib/boss-data"))["completeMasterAgentTask"]; async function setup() { if (runtimeRoot) return; runtimeRoot = await mkdtemp(path.join(os.tmpdir(), "boss-master-agent-thread-status-")); process.env.BOSS_RUNTIME_ROOT = runtimeRoot; process.env.BOSS_STATE_FILE = path.join(runtimeRoot, "boss-state.json"); const [data, masterAgent] = await Promise.all([ import("../src/lib/boss-data.ts"), import("../src/lib/boss-master-agent.ts"), ]); readState = data.readState; writeState = data.writeState; saveAiAccount = data.saveAiAccount; updateMasterAgentPromptPolicy = data.updateMasterAgentPromptPolicy; updateUserMasterPrompt = data.updateUserMasterPrompt; createUserMasterMemory = data.createUserMasterMemory; updateProjectAgentControls = data.updateProjectAgentControls; appendProjectMessage = data.appendProjectMessage; resolveMasterAgentExecutionConfig = masterAgent.resolveMasterAgentExecutionConfig; replyToMasterAgentUserMessage = masterAgent.replyToMasterAgentUserMessage; completeMasterAgentTask = data.completeMasterAgentTask; } test.after(async () => { if (runtimeRoot) { await rm(runtimeRoot, { recursive: true, force: true }); } }); test("主 Agent 执行 prompt 命中线程时只读取相关状态文档和最近进展事件,不再常态注入深拉兜底", async () => { await setup(); await saveAiAccount({ accountId: "master-codex-primary", label: "主 GPT", role: "primary", provider: "master_codex_node", displayName: "krisolo · Master Codex Node", nodeId: "mac-studio", nodeLabel: "Mac Studio", enabled: true, setActive: true, status: "ready", loginStatusNote: "主节点可用。", }); await updateMasterAgentPromptPolicy({ globalPrompt: "管理员全局主提示词", updatedBy: "krisolo", }); await updateUserMasterPrompt("krisolo", "用户私有主提示词"); await updateProjectAgentControls("master-agent", { promptOverride: "当前对话提示词", }); await createUserMasterMemory({ account: "krisolo", scope: "project", projectId: "master-agent", title: "项目记忆", content: "项目记忆正文", memoryType: "project_progress", tags: ["线程状态"], }); await createUserMasterMemory({ account: "master-codex-primary", scope: "project", projectId: "master-agent", title: "项目记忆", content: "项目记忆正文", memoryType: "project_progress", tags: ["线程状态"], }); const state = await readState(); const auditProject = state.projects.find((project) => project.id === "audit-collab"); assert.ok(auditProject, "expected seeded audit-collab project"); const masterProject = state.projects.find((project) => project.id === "master-agent"); assert.ok(masterProject, "expected seeded master-agent project"); auditProject!.projectUnderstanding = { projectGoal: "深拉兜底目标", currentProgress: "深拉兜底进度", technicalArchitecture: "深拉兜底架构", currentBlockers: "深拉兜底阻塞", recommendedNextStep: "深拉兜底下一步", sourceTaskId: "task-deep-pull", updatedAt: "2026-04-04T18:00:00+08:00", sourceKind: "thread_sync", }; masterProject!.projectUnderstanding = { projectGoal: "主 Agent 旧目标", currentProgress: "主 Agent 旧进度", technicalArchitecture: "主 Agent 旧架构", currentBlockers: "主 Agent 旧阻塞", recommendedNextStep: "主 Agent 旧下一步", sourceTaskId: "task-master-legacy", updatedAt: "2026-04-04T17:50:00+08:00", sourceKind: "thread_sync", }; state.threadStatusDocuments = [ { documentId: "thread-status-doc-master", projectId: "master-agent", threadId: "thread-master-main", threadDisplayName: "主 Agent 汇总", folderName: "主控线程", deviceId: "mac-studio", projectGoal: "主 Agent 额外状态目标", currentPhase: "主 Agent 额外阶段", currentProgress: "主 Agent 额外进度", technicalArchitecture: "主 Agent 额外架构", currentBlockers: "主 Agent 额外阻塞", recommendedNextStep: "主 Agent 额外下一步", keyFiles: ["src/lib/boss-master-agent.ts"], keyCommands: ["npm run lint"], updatedAt: "2026-04-04T18:03:00+08:00", sourceTaskId: "task-master-status", sourceKind: "incremental_sync", }, { documentId: "thread-status-doc-1", projectId: "audit-collab", threadId: "thread-audit-chief", threadDisplayName: "审计对话", folderName: "审计群聊", deviceId: "mac-studio", projectGoal: "线程状态目标", currentPhase: "线程状态阶段", currentProgress: "线程状态进度", technicalArchitecture: "线程状态架构", currentBlockers: "线程状态阻塞", recommendedNextStep: "线程状态下一步", keyFiles: ["src/lib/boss-master-agent.ts"], keyCommands: ["npm run build"], updatedAt: "2026-04-04T18:01:00+08:00", sourceTaskId: "task-thread-status", sourceKind: "incremental_sync", }, ]; state.threadProgressEvents = [ { eventId: "thread-progress-event-master", projectId: "master-agent", threadId: "thread-master-main", threadDisplayName: "主 Agent 汇总", deviceId: "mac-studio", eventType: "progress_updated", summary: "主 Agent 额外进展摘要", phase: "主 Agent 额外阶段", blockerDelta: "主 Agent 额外阻塞", nextStepDelta: "主 Agent 额外下一步", createdAt: "2026-04-04T18:03:30+08:00", sourceTaskId: "task-master-progress", }, { eventId: "thread-progress-event-1", projectId: "audit-collab", threadId: "thread-audit-chief", threadDisplayName: "审计对话", deviceId: "mac-studio", eventType: "progress_updated", summary: "最近进展事件摘要", phase: "线程状态阶段", blockerDelta: "线程状态阻塞", nextStepDelta: "线程状态下一步", createdAt: "2026-04-04T18:02:00+08:00", sourceTaskId: "task-thread-progress", }, ]; await writeState(state); const resolved = await resolveMasterAgentExecutionConfig( "master-agent", "krisolo", "审计对话,请继续推进线程状态同步", ); assert.ok(resolved.projectMemories.length > 0); assert.equal(resolved.projectMemories[0]?.content, "项目记忆正文"); assert.ok(resolved.executionPrompt.includes("当前对话提示词")); assert.ok( resolved.executionPrompt.indexOf("管理员全局主提示词:") < resolved.executionPrompt.indexOf("用户私有主提示词:") && resolved.executionPrompt.indexOf("用户私有主提示词:") < resolved.executionPrompt.indexOf("当前对话附加提示词:") && resolved.executionPrompt.indexOf("当前对话附加提示词:") < resolved.executionPrompt.indexOf("当前消息:"), ); const reply = await replyToMasterAgentUserMessage({ requestText: "审计对话,请继续推进线程状态同步", requestedBy: "Boss 超级管理员", requestedByAccount: "krisolo", mode: "enqueue", }); assert.equal(reply.ok, true); assert.equal(reply.masterReplyState, "queued"); const queuedTask = (await readState()).masterAgentTasks.find( (task) => task.projectId === "master-agent" && task.requestText === "审计对话,请继续推进线程状态同步", ); assert.ok(queuedTask, "expected master-agent task to be queued"); assert.ok(queuedTask?.executionPrompt.includes("线程状态文档:")); assert.ok(queuedTask?.executionPrompt.includes("线程状态目标")); assert.ok(!queuedTask?.executionPrompt.includes("主 Agent 额外状态目标")); assert.ok(queuedTask?.executionPrompt.includes("最近进展事件:")); assert.ok(queuedTask?.executionPrompt.includes("最近进展事件摘要")); assert.ok(!queuedTask?.executionPrompt.includes("主 Agent 额外进展摘要")); assert.ok(!queuedTask?.executionPrompt.includes("深拉兜底目标")); assert.ok(!queuedTask?.executionPrompt.includes("关键时刻深拉线程兜底:")); }); test("主 Agent 执行 prompt 在未命中时退回最近活跃项目,且不会常态注入深拉兜底", async () => { await setup(); const state = await readState(); const auditProject = state.projects.find((project) => project.id === "audit-collab"); const masterProject = state.projects.find((project) => project.id === "master-agent"); assert.ok(auditProject, "expected seeded audit-collab project"); assert.ok(masterProject, "expected seeded master-agent project"); auditProject!.projectUnderstanding = { projectGoal: "审计兜底目标", currentProgress: "审计兜底进度", technicalArchitecture: "审计兜底架构", currentBlockers: "审计兜底阻塞", recommendedNextStep: "审计兜底下一步", sourceTaskId: "task-audit-fallback", updatedAt: "2026-04-04T17:55:00+08:00", sourceKind: "thread_sync", }; masterProject!.projectUnderstanding = { projectGoal: "最近活跃目标", currentProgress: "最近活跃进度", technicalArchitecture: "最近活跃架构", currentBlockers: "最近活跃阻塞", recommendedNextStep: "最近活跃下一步", sourceTaskId: "task-master-active", updatedAt: "2026-04-04T18:05:00+08:00", sourceKind: "thread_sync", }; state.threadStatusDocuments = [ { documentId: "thread-status-doc-audit-fallback", projectId: "audit-collab", threadId: "thread-audit-chief", threadDisplayName: "审计对话", folderName: "审计群聊", deviceId: "mac-studio", projectGoal: "审计兜底状态", currentPhase: "审计兜底阶段", currentProgress: "审计兜底进度", technicalArchitecture: "审计兜底架构", currentBlockers: "审计兜底阻塞", recommendedNextStep: "审计兜底下一步", keyFiles: ["src/lib/boss-master-agent.ts"], keyCommands: ["npm run build"], updatedAt: "2026-04-04T17:56:00+08:00", sourceTaskId: "task-audit-status", sourceKind: "incremental_sync", }, { documentId: "thread-status-doc-master-fallback", projectId: "master-agent", threadId: "thread-master-main", threadDisplayName: "主 Agent 汇总", folderName: "主控线程", deviceId: "mac-studio", projectGoal: "最近活跃状态", currentPhase: "最近活跃阶段", currentProgress: "最近活跃进度", technicalArchitecture: "最近活跃架构", currentBlockers: "最近活跃阻塞", recommendedNextStep: "最近活跃下一步", keyFiles: ["src/lib/boss-data.ts"], keyCommands: ["npm run lint"], updatedAt: "2026-04-04T18:06:00+08:00", sourceTaskId: "task-master-status-fallback", sourceKind: "incremental_sync", }, ]; state.threadProgressEvents = [ { eventId: "thread-progress-event-audit-fallback", projectId: "audit-collab", threadId: "thread-audit-chief", threadDisplayName: "审计对话", deviceId: "mac-studio", eventType: "progress_updated", summary: "审计兜底进展摘要", phase: "审计兜底阶段", blockerDelta: "审计兜底阻塞", nextStepDelta: "审计兜底下一步", createdAt: "2026-04-04T17:56:30+08:00", sourceTaskId: "task-audit-progress", }, { eventId: "thread-progress-event-master-fallback", projectId: "master-agent", threadId: "thread-master-main", threadDisplayName: "主 Agent 汇总", deviceId: "mac-studio", eventType: "progress_updated", summary: "最近活跃进展摘要", phase: "最近活跃阶段", blockerDelta: "最近活跃阻塞", nextStepDelta: "最近活跃下一步", createdAt: "2026-04-04T18:06:30+08:00", sourceTaskId: "task-master-progress-fallback", }, ]; await writeState(state); const reply = await replyToMasterAgentUserMessage({ requestText: "请继续推进线程状态同步(仅深拉兜底)", requestedBy: "Boss 超级管理员", requestedByAccount: "krisolo", mode: "enqueue", }); assert.equal(reply.ok, true); const queuedTask = (await readState()).masterAgentTasks.find( (task) => task.projectId === "master-agent" && task.requestText === "请继续推进线程状态同步(仅深拉兜底)", ); assert.ok(queuedTask, "expected master-agent task to be queued"); assert.ok(queuedTask?.executionPrompt.includes("最近活跃状态")); assert.ok(queuedTask?.executionPrompt.includes("最近活跃进展摘要")); assert.ok(!queuedTask?.executionPrompt.includes("关键时刻深拉线程兜底:")); }); test("主 Agent 执行 prompt 在没有线程状态文档和进展事件时才会注入深拉兜底", async () => { await setup(); const state = await readState(); const auditProject = state.projects.find((project) => project.id === "audit-collab"); assert.ok(auditProject, "expected seeded audit-collab project"); auditProject!.projectUnderstanding = { projectGoal: "深拉兜底目标", currentProgress: "深拉兜底进度", technicalArchitecture: "深拉兜底架构", currentBlockers: "深拉兜底阻塞", recommendedNextStep: "深拉兜底下一步", sourceTaskId: "task-deep-pull", updatedAt: "2026-04-04T18:00:00+08:00", sourceKind: "thread_sync", }; state.threadStatusDocuments = []; state.threadProgressEvents = []; await writeState(state); const reply = await replyToMasterAgentUserMessage({ requestText: "请继续推进线程状态同步", requestedBy: "Boss 超级管理员", requestedByAccount: "krisolo", mode: "enqueue", }); assert.equal(reply.ok, true); const queuedTask = (await readState()).masterAgentTasks.find( (task) => task.projectId === "master-agent" && task.requestText === "请继续推进线程状态同步", ); assert.ok(queuedTask, "expected master-agent task to be queued"); assert.ok(queuedTask?.executionPrompt.includes("关键时刻深拉线程兜底:")); assert.ok(queuedTask?.executionPrompt.includes("深拉兜底目标")); }); test("非主会话里 @主Agent 时,运行时摘要只允许读取当前项目上下文,不允许串到最近活跃项目", async () => { await setup(); await saveAiAccount({ accountId: "master-codex-primary-thread-scope", label: "主 GPT", role: "primary", provider: "master_codex_node", displayName: "krisolo · Master Codex Node", nodeId: "mac-studio", nodeLabel: "Mac Studio", enabled: true, setActive: true, status: "ready", loginStatusNote: "主节点可用。", }); const state = await readState(); const auditProject = state.projects.find((project) => project.id === "audit-collab"); assert.ok(auditProject, "expected seeded audit-collab project"); state.projects.push({ id: "aitoukui-thread", name: "AItoukui", pinned: false, deviceIds: ["mac-studio"], preview: "等待同步", updatedAt: "2026-04-05T11:59:00+08:00", lastMessageAt: "2026-04-05T11:59:00+08:00", isGroup: false, threadMeta: { projectId: "aitoukui-thread", threadId: "thread-aitoukui-main", threadDisplayName: "AItoukui 主线程", folderName: "AItoukui", activityIconCount: 1, updatedAt: "2026-04-05T11:59:00+08:00", lastObservedCodexActivityAt: "2026-04-05T11:59:00+08:00", codexThreadRef: "thread-aitoukui-main", codexFolderRef: "aitoukui", }, groupMembers: [], createdByAgent: true, collaborationMode: "development", approvalState: "not_required", unreadCount: 0, riskLevel: "low", messages: [], goals: [], versions: [], }); state.threadStatusDocuments = [ { documentId: "thread-status-doc-current-project", projectId: "audit-collab", threadId: "thread-audit-chief", threadDisplayName: "审计对话", folderName: "审计群聊", deviceId: "mac-studio", projectGoal: "当前项目目标", currentPhase: "当前项目阶段", currentProgress: "当前项目进度", technicalArchitecture: "当前项目架构", currentBlockers: "当前项目阻塞", recommendedNextStep: "当前项目下一步", keyFiles: ["src/lib/current-thread.ts"], keyCommands: ["npm run lint"], updatedAt: "2026-04-05T10:00:00+08:00", sourceTaskId: "task-current-project", sourceKind: "incremental_sync", }, { documentId: "thread-status-doc-other-project", projectId: "aitoukui-thread", threadId: "thread-aitoukui-main", threadDisplayName: "AItoukui 主线程", folderName: "AItoukui", deviceId: "mac-studio", projectGoal: "别的项目目标", currentPhase: "别的项目阶段", currentProgress: "别的项目进度", technicalArchitecture: "别的项目架构", currentBlockers: "别的项目阻塞", recommendedNextStep: "别的项目下一步", keyFiles: ["src/lib/aitoukui-thread.ts"], keyCommands: ["npm run build"], updatedAt: "2026-04-05T12:00:00+08:00", sourceTaskId: "task-other-project", sourceKind: "incremental_sync", }, ]; state.threadProgressEvents = [ { eventId: "thread-progress-current-project", projectId: "audit-collab", threadId: "thread-audit-chief", threadDisplayName: "审计对话", deviceId: "mac-studio", eventType: "progress_updated", summary: "当前项目进展摘要", phase: "当前项目阶段", blockerDelta: "当前项目阻塞变化", nextStepDelta: "当前项目下一步变化", createdAt: "2026-04-05T10:02:00+08:00", sourceTaskId: "task-current-progress", }, { eventId: "thread-progress-other-project", projectId: "aitoukui-thread", threadId: "thread-aitoukui-main", threadDisplayName: "AItoukui 主线程", deviceId: "mac-studio", eventType: "progress_updated", summary: "别的项目进展摘要", phase: "别的项目阶段", blockerDelta: "别的项目阻塞变化", nextStepDelta: "别的项目下一步变化", createdAt: "2026-04-05T12:01:00+08:00", sourceTaskId: "task-other-progress", }, ]; await writeState(state); const reply = await replyToMasterAgentUserMessage({ requestText: "请你看一下当前项目下一步怎么推进", requestedBy: "Boss 超级管理员", requestedByAccount: "krisolo", projectId: "audit-collab", mode: "enqueue", }); assert.equal(reply.ok, true); const queuedTask = (await readState()).masterAgentTasks.find( (task) => task.projectId === "audit-collab" && task.requestText === "请你看一下当前项目下一步怎么推进", ); assert.ok(queuedTask, "expected a current-thread master-agent task to be queued"); assert.ok(queuedTask?.executionPrompt.includes("当前项目目标")); assert.ok(queuedTask?.executionPrompt.includes("当前项目进展摘要")); assert.ok(!queuedTask?.executionPrompt.includes("别的项目目标")); assert.ok(!queuedTask?.executionPrompt.includes("别的项目进展摘要")); }); test("项目理解同步 prompt 强制线程先基于本地文档和代码汇总,并允许回写版本记录摘要", async () => { await setup(); const state = await readState(); state.projects.push({ id: "understanding-sync-thread", name: "项目理解同步线程", pinned: false, deviceIds: ["mac-studio"], preview: "等待同步", updatedAt: "2026-04-04T18:00:00+08:00", lastMessageAt: "2026-04-04T18:00:00+08:00", isGroup: false, threadMeta: { projectId: "understanding-sync-thread", threadId: "thread-understanding-sync", threadDisplayName: "项目理解同步线程", folderName: "理解同步", activityIconCount: 1, updatedAt: "2026-04-04T18:00:00+08:00", lastObservedCodexActivityAt: "2026-04-04T18:00:00+08:00", codexThreadRef: "thread-understanding-sync", codexFolderRef: "understanding-sync-folder", }, groupMembers: [], createdByAgent: true, collaborationMode: "development", approvalState: "not_required", unreadCount: 0, riskLevel: "low", messages: [], goals: [], versions: [], }); const project = state.projects.find((item) => item.id === "understanding-sync-thread"); assert.ok(project, "expected seeded understanding-sync-thread project"); project!.versions = []; project!.threadMeta.lastObservedCodexActivityAt = "2026-04-04T18:00:00+08:00"; await writeState(state); await updateProjectAgentControls("understanding-sync-thread", { takeoverEnabled: true, }); await appendProjectMessage({ projectId: "understanding-sync-thread", sender: "device", senderLabel: "项目理解同步线程", body: "已根据本地开发文档补齐线程状态同步。", kind: "text", }); const queuedTask = (await readState()).masterAgentTasks.find( (task) => task.projectId === "master-agent" && task.projectUnderstandingTargetProjectId === "understanding-sync-thread", ); assert.ok(queuedTask, "expected project understanding sync task"); assert.match(queuedTask!.executionPrompt, /先基于当前项目本地可见的开发文档和实际代码进行汇总/); assert.match(queuedTask!.executionPrompt, /优先检查 README、docs、架构文档、版本记录和最近改动的关键代码文件/); assert.match(queuedTask!.executionPrompt, /"versionRecord": "一句中文版本记录摘要"/); await completeMasterAgentTask({ taskId: queuedTask!.taskId, deviceId: "mac-studio", status: "completed", replyBody: JSON.stringify({ projectGoal: "完成审计对话线程状态同步", currentProgress: "已切到线程状态文档优先", technicalArchitecture: "Next.js 控制面配合 Android 原生客户端", currentBlockers: "", recommendedNextStep: "继续补强版本记录同步", versionRecord: "项目理解同步已改成先读本地文档和代码,再回写结构化摘要。", }), }); const refreshed = await readState(); const refreshedProject = refreshed.projects.find((item) => item.id === "understanding-sync-thread"); assert.ok(refreshedProject, "expected refreshed project"); assert.equal( refreshedProject!.projectUnderstanding?.projectGoal, "完成审计对话线程状态同步", ); assert.equal(refreshedProject!.versions[0]?.summary, "项目理解同步已改成先读本地文档和代码,再回写结构化摘要。"); }); test("主 Agent 总结项目目标和版本记录时不注入 OTA 与设备运行态噪音", async () => { await setup(); await saveAiAccount({ accountId: "master-codex-primary", label: "主 GPT", role: "primary", provider: "master_codex_node", displayName: "krisolo · Master Codex Node", nodeId: "mac-studio", nodeLabel: "Mac Studio", enabled: true, setActive: true, status: "ready", loginStatusNote: "主节点可用。", }); const state = await readState(); const auditProject = state.projects.find((project) => project.id === "audit-collab"); assert.ok(auditProject, "expected seeded audit-collab project"); auditProject!.projectUnderstanding = { projectGoal: "审计线程目标", currentProgress: "正在补齐版本记录摘要", technicalArchitecture: "Next.js 控制面 + Android 原生客户端 + local-agent", currentBlockers: "", recommendedNextStep: "继续同步版本记录入口", sourceTaskId: "task-audit-summary", updatedAt: "2026-04-05T10:00:00+08:00", sourceKind: "thread_sync", }; state.threadStatusDocuments = [ { documentId: "thread-status-doc-audit-summary", projectId: "audit-collab", threadId: "thread-audit-chief", threadDisplayName: "审计对话", folderName: "审计群聊", deviceId: "mac-studio", projectGoal: "审计线程目标", currentPhase: "整理项目目标与版本记录", currentProgress: "版本记录入口正在统一", technicalArchitecture: "Next.js 控制面 + Android 原生客户端 + local-agent", currentBlockers: "", recommendedNextStep: "继续回填版本记录摘要", keyFiles: ["src/lib/boss-master-agent.ts"], keyCommands: ["npm run test"], updatedAt: "2026-04-05T10:01:00+08:00", sourceTaskId: "task-audit-summary-doc", sourceKind: "incremental_sync", }, ]; state.threadProgressEvents = [ { eventId: "thread-progress-event-audit-summary", projectId: "audit-collab", threadId: "thread-audit-chief", threadDisplayName: "审计对话", deviceId: "mac-studio", eventType: "progress_updated", summary: "项目目标和版本记录已开始重新汇总", phase: "整理项目目标与版本记录", blockerDelta: "", nextStepDelta: "同步到会话顶部入口", createdAt: "2026-04-05T10:02:00+08:00", sourceTaskId: "task-audit-summary-event", }, ]; state.appLogs = [ { logId: "app-log-summary-noise", deviceId: "mac-studio", category: "sync", message: "这条日志不应该出现在项目目标总结 prompt 里", createdAt: "2026-04-05T10:03:00+08:00", }, ]; state.otaUpdates = [ { releaseId: "ota-summary-noise", version: "v2.1.0", currentVersion: "v2.0.0", channel: "stable", packageType: "android_shell", status: "available", summary: ["这条 OTA 不应该污染项目总结"], targetScope: "Boss Android 原生客户端", requiredRole: "highest_admin", publishedAt: "2026-04-05T10:04:00+08:00", }, ]; await writeState(state); const requestText = "请总结审计对话当前的项目目标和版本记录"; const reply = await replyToMasterAgentUserMessage({ requestText, requestedBy: "Boss 超级管理员", requestedByAccount: "krisolo", mode: "enqueue", }); assert.equal(reply.ok, true); const queuedTask = (await readState()).masterAgentTasks.find( (task) => task.projectId === "master-agent" && task.requestText === requestText, ); assert.ok(queuedTask, "expected summary task to be queued"); assert.ok(queuedTask?.executionPrompt.includes("线程状态文档:")); assert.ok(queuedTask?.executionPrompt.includes("审计线程目标")); assert.ok(!queuedTask?.executionPrompt.includes("最新 APP 日志:")); assert.ok(!queuedTask?.executionPrompt.includes("高风险线程:")); assert.ok(!queuedTask?.executionPrompt.includes("在线设备:")); assert.ok(!queuedTask?.executionPrompt.includes("认证状态:")); assert.ok(!queuedTask?.executionPrompt.includes("可用 OTA:")); assert.ok(queuedTask?.executionPrompt.includes("回复风格:像专业职业经理人")); assert.ok(queuedTask?.executionPrompt.includes("先给结论,再给推进动作")); assert.ok(queuedTask?.executionPrompt.includes("不要堆背景,不要重复系统状态")); }); test("主 Agent 处理 OTA 和设备运行态问题时保留相关运行时摘要", async () => { await setup(); await saveAiAccount({ accountId: "master-codex-primary", label: "主 GPT", role: "primary", provider: "master_codex_node", displayName: "krisolo · Master Codex Node", nodeId: "mac-studio", nodeLabel: "Mac Studio", enabled: true, setActive: true, status: "ready", loginStatusNote: "主节点可用。", }); const state = await readState(); state.appLogs = [ { logId: "app-log-runtime-detail", deviceId: "mac-studio", category: "ota", message: "检测到 Android OTA 可更新", createdAt: "2026-04-05T11:00:00+08:00", }, ]; state.otaUpdates = [ { releaseId: "ota-runtime-detail", version: "v2.2.0", currentVersion: "v2.1.0", channel: "stable", packageType: "android_shell", status: "available", summary: ["加入新的设备运行态提示"], targetScope: "Boss Android 原生客户端", requiredRole: "highest_admin", publishedAt: "2026-04-05T11:01:00+08:00", }, ]; await writeState(state); const requestText = "帮我看一下当前可用 OTA、在线设备状态和最近 APP 日志"; const reply = await replyToMasterAgentUserMessage({ requestText, requestedBy: "Boss 超级管理员", requestedByAccount: "krisolo", mode: "enqueue", }); assert.equal(reply.ok, true); const queuedTask = (await readState()).masterAgentTasks.find( (task) => task.projectId === "master-agent" && task.requestText === requestText, ); assert.ok(queuedTask, "expected runtime-detail task to be queued"); assert.ok(queuedTask?.executionPrompt.includes("最新 APP 日志:")); assert.ok(queuedTask?.executionPrompt.includes("在线设备:")); assert.ok(queuedTask?.executionPrompt.includes("认证状态:")); assert.ok(queuedTask?.executionPrompt.includes("可用 OTA:")); assert.ok(queuedTask?.executionPrompt.includes("v2.2.0 -> Boss Android 原生客户端")); });