- 集成MCP连接管理器,支持MCP服务器连接 - 添加多种内置工具:ClarifyTool、CronTool、DelegateTool、ExecuteCodeTool、 PatchFileTool、ProcessTool、SendMessageTool、SpawnTool、TerminalTool、 TodoTool、WebFetchTool、WebSearchTool、WriteFileTool等 - 实现工具注册和装配功能 - 添加技能选择上下文参数 - 支持思考模式控制参数thinking_enabled feat(coordinator): 重构任务执行计划器参数命名 - 将learning_candidate_enabled重命名为allow_candidate_generation - 更新TeamGraphScheduler中的参数传递 - 修改LocalAgentRunner中的相关参数处理 - 更新README文档中的相应描述 refactor(context): 标准化工具调用参数格式 - 添加_json导入用于参数序列化 - 实现_provider_tool_calls方法标准化OpenAI兼容的工具调用载荷 - 修复工具调用中参数非字符串类型的序列化问题 refactor(session): 优化消息历史记录过滤逻辑 - 修改get_messages_as_conversation为基于运行状态过滤消息 - 排除未完成、失败或错误结束的运行记录 - 改进对话历史的可见性控制机制 fix(store): 修复FTS索引重建逻辑 - 添加异常处理防止FTS索引创建失败 - 实现_rebuild_fts_index方法重新构建全文搜索索引 - 优化索引触发器和表的维护流程
122 lines
3.6 KiB
TypeScript
122 lines
3.6 KiB
TypeScript
'use client';
|
|
|
|
import React from 'react';
|
|
|
|
import { getStatus, listSessions, wsManager } from '@/lib/api';
|
|
import { useChatStore } from '@/lib/store';
|
|
import type { ProcessWsEvent, SessionUpdatedEvent, WsEvent } from '@/types';
|
|
|
|
function scheduleWhenIdle(task: () => void, timeout = 1200): () => void {
|
|
if (typeof window === 'undefined') {
|
|
task();
|
|
return () => {};
|
|
}
|
|
|
|
const idleWindow = window as Window &
|
|
typeof globalThis & {
|
|
requestIdleCallback?: (callback: IdleRequestCallback, options?: IdleRequestOptions) => number;
|
|
cancelIdleCallback?: (handle: number) => void;
|
|
};
|
|
|
|
if (typeof idleWindow.requestIdleCallback === 'function') {
|
|
const id = idleWindow.requestIdleCallback(() => task(), { timeout });
|
|
return () => idleWindow.cancelIdleCallback?.(id);
|
|
}
|
|
|
|
const id = globalThis.setTimeout(task, 250);
|
|
return () => globalThis.clearTimeout(id);
|
|
}
|
|
|
|
function isProcessEvent(data: WsEvent | Record<string, unknown>): data is ProcessWsEvent {
|
|
const type = typeof data.type === 'string' ? data.type : '';
|
|
return type.startsWith('process_') || type === 'process_cancel_ack';
|
|
}
|
|
|
|
function isSessionUpdatedEvent(data: WsEvent | Record<string, unknown>): data is SessionUpdatedEvent {
|
|
return data.type === 'session_updated' && typeof data.session_id === 'string';
|
|
}
|
|
|
|
export function AppRuntimeBridge() {
|
|
const sessionId = useChatStore((state) => state.sessionId);
|
|
const setSessions = useChatStore((state) => state.setSessions);
|
|
const setWsStatus = useChatStore((state) => state.setWsStatus);
|
|
const setNanobotReady = useChatStore((state) => state.setNanobotReady);
|
|
const resetProcessState = useChatStore((state) => state.resetProcessState);
|
|
const ingestProcessEvent = useChatStore((state) => state.ingestProcessEvent);
|
|
const statusCheckCleanupRef = React.useRef<(() => void) | null>(null);
|
|
const statusCheckInFlightRef = React.useRef(false);
|
|
|
|
const loadSessions = React.useCallback(async () => {
|
|
try {
|
|
const sessions = await listSessions();
|
|
setSessions(sessions);
|
|
} catch {
|
|
// backend may still be offline during first render
|
|
}
|
|
}, [setSessions]);
|
|
|
|
const scheduleStatusCheck = React.useCallback(() => {
|
|
if (statusCheckInFlightRef.current) return;
|
|
|
|
statusCheckCleanupRef.current?.();
|
|
statusCheckCleanupRef.current = scheduleWhenIdle(async () => {
|
|
statusCheckInFlightRef.current = true;
|
|
try {
|
|
await getStatus();
|
|
setNanobotReady(true);
|
|
} catch {
|
|
setNanobotReady(false);
|
|
} finally {
|
|
statusCheckInFlightRef.current = false;
|
|
}
|
|
});
|
|
}, [setNanobotReady]);
|
|
|
|
React.useEffect(() => {
|
|
void loadSessions();
|
|
}, [loadSessions]);
|
|
|
|
React.useEffect(() => {
|
|
resetProcessState();
|
|
wsManager.connect(sessionId);
|
|
}, [resetProcessState, sessionId]);
|
|
|
|
React.useEffect(() => {
|
|
const unsubStatus = wsManager.onStatusChange((status) => {
|
|
setWsStatus(status);
|
|
if (status === 'connected') {
|
|
scheduleStatusCheck();
|
|
} else {
|
|
statusCheckCleanupRef.current?.();
|
|
statusCheckCleanupRef.current = null;
|
|
setNanobotReady(null);
|
|
}
|
|
});
|
|
|
|
return () => {
|
|
statusCheckCleanupRef.current?.();
|
|
statusCheckCleanupRef.current = null;
|
|
unsubStatus();
|
|
};
|
|
}, [scheduleStatusCheck, setNanobotReady, setWsStatus]);
|
|
|
|
React.useEffect(() => {
|
|
const unsubMessage = wsManager.onMessage((data) => {
|
|
if (isSessionUpdatedEvent(data)) {
|
|
void loadSessions();
|
|
return;
|
|
}
|
|
|
|
if (isProcessEvent(data)) {
|
|
ingestProcessEvent(data);
|
|
}
|
|
});
|
|
|
|
return () => {
|
|
unsubMessage();
|
|
};
|
|
}, [ingestProcessEvent, loadSessions]);
|
|
|
|
return null;
|
|
}
|