# core/main_terminal.py - 主终端(集成对话持久化) import asyncio import json from datetime import datetime from pathlib import Path from typing import Any, Dict, List, Optional, Set, TYPE_CHECKING from datetime import datetime try: from config import ( OUTPUT_FORMATS, DATA_DIR, PROMPTS_DIR, NEED_CONFIRMATION, MAX_TERMINALS, TERMINAL_BUFFER_SIZE, TERMINAL_DISPLAY_SIZE, MAX_READ_FILE_CHARS, READ_TOOL_DEFAULT_MAX_CHARS, READ_TOOL_DEFAULT_CONTEXT_BEFORE, READ_TOOL_DEFAULT_CONTEXT_AFTER, READ_TOOL_MAX_CONTEXT_BEFORE, READ_TOOL_MAX_CONTEXT_AFTER, READ_TOOL_DEFAULT_MAX_MATCHES, READ_TOOL_MAX_MATCHES, READ_TOOL_MAX_FILE_SIZE, MAX_FOCUS_FILE_CHARS, TERMINAL_SANDBOX_MOUNT_PATH, TERMINAL_SANDBOX_CPUS, TERMINAL_SANDBOX_MEMORY, PROJECT_MAX_STORAGE_MB, ) except ImportError: import sys from pathlib import Path project_root = Path(__file__).resolve().parents[1] if str(project_root) not in sys.path: sys.path.insert(0, str(project_root)) from config import ( OUTPUT_FORMATS, DATA_DIR, PROMPTS_DIR, NEED_CONFIRMATION, MAX_TERMINALS, TERMINAL_BUFFER_SIZE, TERMINAL_DISPLAY_SIZE, MAX_READ_FILE_CHARS, READ_TOOL_DEFAULT_MAX_CHARS, READ_TOOL_DEFAULT_CONTEXT_BEFORE, READ_TOOL_DEFAULT_CONTEXT_AFTER, READ_TOOL_MAX_CONTEXT_BEFORE, READ_TOOL_MAX_CONTEXT_AFTER, READ_TOOL_DEFAULT_MAX_MATCHES, READ_TOOL_MAX_MATCHES, READ_TOOL_MAX_FILE_SIZE, MAX_FOCUS_FILE_CHARS, TERMINAL_SANDBOX_MOUNT_PATH, TERMINAL_SANDBOX_CPUS, TERMINAL_SANDBOX_MEMORY, PROJECT_MAX_STORAGE_MB, ) from modules.file_manager import FileManager from modules.search_engine import SearchEngine from modules.terminal_ops import TerminalOperator from modules.memory_manager import MemoryManager from modules.terminal_manager import TerminalManager from modules.todo_manager import TodoManager from modules.sub_agent_manager import SubAgentManager from modules.webpage_extractor import extract_webpage_content, tavily_extract from modules.ocr_client import OCRClient from modules.easter_egg_manager import EasterEggManager from modules.personalization_manager import ( load_personalization_config, build_personalization_prompt, ) try: from config.limits import THINKING_FAST_INTERVAL except ImportError: THINKING_FAST_INTERVAL = 10 from modules.container_monitor import collect_stats, inspect_state from core.tool_config import TOOL_CATEGORIES from utils.api_client import DeepSeekClient from utils.context_manager import ContextManager from utils.tool_result_formatter import format_tool_result_for_context from utils.logger import setup_logger if TYPE_CHECKING: from modules.user_container_manager import ContainerHandle logger = setup_logger(__name__) # 临时禁用长度检查 DISABLE_LENGTH_CHECK = True class MainTerminal: def __init__( self, project_path: str, thinking_mode: bool = False, run_mode: Optional[str] = None, data_dir: Optional[str] = None, container_session: Optional["ContainerHandle"] = None, usage_tracker: Optional[object] = None, ): self.project_path = project_path allowed_modes = {"fast", "thinking", "deep"} initial_mode = run_mode if run_mode in allowed_modes else ("thinking" if thinking_mode else "fast") self.run_mode = initial_mode self.thinking_mode = initial_mode != "fast" # False=快速模式, True=思考模式 self.deep_thinking_mode = initial_mode == "deep" self.data_dir = Path(data_dir).expanduser().resolve() if data_dir else Path(DATA_DIR).resolve() self.usage_tracker = usage_tracker # 初始化组件 self.api_client = DeepSeekClient(thinking_mode=self.thinking_mode) self.api_client.set_deep_thinking_mode(self.deep_thinking_mode) self.context_manager = ContextManager(project_path, data_dir=str(self.data_dir)) self.context_manager.main_terminal = self self.container_mount_path = TERMINAL_SANDBOX_MOUNT_PATH or "/workspace" self.container_cpu_limit = TERMINAL_SANDBOX_CPUS or "未限制" self.container_memory_limit = TERMINAL_SANDBOX_MEMORY or "未限制" self.project_storage_limit = f"{PROJECT_MAX_STORAGE_MB}MB" if PROJECT_MAX_STORAGE_MB else "未限制" self.project_storage_limit_bytes = ( PROJECT_MAX_STORAGE_MB * 1024 * 1024 if PROJECT_MAX_STORAGE_MB else None ) self.container_session: Optional["ContainerHandle"] = None self.memory_manager = MemoryManager(data_dir=str(self.data_dir)) self.file_manager = FileManager(project_path, container_session=container_session) self.search_engine = SearchEngine() self.terminal_ops = TerminalOperator(project_path, container_session=container_session) self.ocr_client = OCRClient(project_path, self.file_manager) # 新增:终端管理器 self.terminal_manager = TerminalManager( project_path=project_path, max_terminals=MAX_TERMINALS, terminal_buffer_size=TERMINAL_BUFFER_SIZE, terminal_display_size=TERMINAL_DISPLAY_SIZE, broadcast_callback=None, # CLI模式不需要广播 container_session=container_session, ) self._apply_container_session(container_session) self.todo_manager = TodoManager(self.context_manager) self.sub_agent_manager = SubAgentManager( project_path=self.project_path, data_dir=str(self.data_dir) ) self.easter_egg_manager = EasterEggManager() self._announced_sub_agent_tasks = set() # 聚焦文件管理 self.focused_files = {} # {path: content} 存储聚焦的文件内容 self.current_session_id = 0 # 用于标识不同的任务会话 # 工具启用状态 self.tool_category_states = { key: category.default_enabled for key, category in TOOL_CATEGORIES.items() } self.disabled_tools = set() self.disabled_notice_tools = set() self._refresh_disabled_tools() self.thinking_fast_interval = THINKING_FAST_INTERVAL self.default_disabled_tool_categories: List[str] = [] self.apply_personalization_preferences() # 新增:自动开始新对话 self._ensure_conversation() # 命令映射 self.commands = { "help": self.show_help, "exit": self.exit_system, "status": self.show_status, "memory": self.manage_memory, "clear": self.clear_conversation, "history": self.show_history, "files": self.show_files, "mode": self.toggle_mode, "focused": self.show_focused_files, "terminals": self.show_terminals, # 新增:对话管理命令 "conversations": self.show_conversations, "load": self.load_conversation_command, "new": self.new_conversation_command, "save": self.save_conversation_command } #self.context_manager._web_terminal_callback = message_callback #self.context_manager._focused_files = self.focused_files # 引用传递 def _apply_container_session(self, session: Optional["ContainerHandle"]): self.container_session = session if session and session.mode == "docker": self.container_mount_path = session.mount_path or (TERMINAL_SANDBOX_MOUNT_PATH or "/workspace") else: self.container_mount_path = TERMINAL_SANDBOX_MOUNT_PATH or "/workspace" def record_model_call(self, is_thinking: bool): tracker = getattr(self, "usage_tracker", None) if not tracker: return True, {} mode = "thinking" if is_thinking else "fast" try: allowed, info = tracker.check_and_increment(mode) if allowed: self._notify_quota_update(mode) return allowed, info except Exception: return True, {} def record_search_call(self): tracker = getattr(self, "usage_tracker", None) if not tracker: return True, {} try: allowed, info = tracker.check_and_increment("search") if allowed: self._notify_quota_update("search") return allowed, info except Exception: return True, {} def _notify_quota_update(self, metric: str): callback = getattr(self, "quota_update_callback", None) if callable(callback): try: callback(metric) except Exception: pass def update_container_session(self, session: Optional["ContainerHandle"]): self._apply_container_session(session) if getattr(self, "terminal_manager", None): self.terminal_manager.update_container_session(session) if getattr(self, "terminal_ops", None): self.terminal_ops.set_container_session(session) if getattr(self, "file_manager", None): self.file_manager.set_container_session(session) def _ensure_conversation(self): """确保CLI模式下存在可用的对话ID""" if self.context_manager.current_conversation_id: return latest_list = self.context_manager.get_conversation_list(limit=1, offset=0) conversations = latest_list.get("conversations", []) if latest_list else [] if conversations: latest = conversations[0] conv_id = latest.get("id") if conv_id and self.context_manager.load_conversation_by_id(conv_id): print(f"{OUTPUT_FORMATS['info']} 已加载最近对话: {conv_id}") return conversation_id = self.context_manager.start_new_conversation( project_path=self.project_path, thinking_mode=self.thinking_mode, run_mode=self.run_mode ) print(f"{OUTPUT_FORMATS['info']} 新建对话: {conversation_id}") @staticmethod def _clamp_int(value, default, min_value=None, max_value=None): """将输入转换为整数并限制范围。""" if value is None: return default try: num = int(value) except (TypeError, ValueError): return default if min_value is not None: num = max(min_value, num) if max_value is not None: num = min(max_value, num) return num @staticmethod def _parse_optional_line(value, field_name: str): """解析可选的行号参数。""" if value is None: return None, None try: number = int(value) except (TypeError, ValueError): return None, f"{field_name} 必须是整数" if number < 1: return None, f"{field_name} 必须大于等于1" return number, None @staticmethod def _truncate_text_block(text: str, max_chars: int): """对单段文本应用字符限制。""" if max_chars and len(text) > max_chars: return text[:max_chars], True, max_chars return text, False, len(text) @staticmethod def _limit_text_chunks(chunks: List[Dict], text_key: str, max_chars: int): """对多个文本片段应用全局字符限制。""" if max_chars is None or max_chars <= 0: return chunks, False, sum(len(chunk.get(text_key, "") or "") for chunk in chunks) remaining = max_chars limited_chunks: List[Dict] = [] truncated = False consumed = 0 for chunk in chunks: snippet = chunk.get(text_key, "") or "" snippet_len = len(snippet) chunk_copy = dict(chunk) if remaining <= 0: truncated = True break if snippet_len > remaining: chunk_copy[text_key] = snippet[:remaining] chunk_copy["truncated"] = True consumed += remaining limited_chunks.append(chunk_copy) truncated = True remaining = 0 break limited_chunks.append(chunk_copy) consumed += snippet_len remaining -= snippet_len return limited_chunks, truncated, consumed def _record_sub_agent_message(self, message: Optional[str], task_id: Optional[str] = None, inline: bool = False): """以 system 消息记录子智能体状态。""" if not message: return if task_id and task_id in self._announced_sub_agent_tasks: return if task_id: self._announced_sub_agent_tasks.add(task_id) logger.info( "[SubAgent] record message | task=%s | inline=%s | content=%s", task_id, inline, message.replace("\n", "\\n")[:200], ) metadata = {"sub_agent_notice": True, "inline": inline} if task_id: metadata["task_id"] = task_id self.context_manager.add_conversation("system", message, metadata=metadata) print(f"{OUTPUT_FORMATS['info']} {message}") def apply_personalization_preferences(self, config: Optional[Dict[str, Any]] = None): """Apply persisted personalization settings that affect runtime behavior.""" try: effective_config = config or load_personalization_config(self.data_dir) except Exception: effective_config = {} interval = effective_config.get("thinking_interval") if isinstance(interval, int) and interval > 0: self.thinking_fast_interval = interval else: self.thinking_fast_interval = THINKING_FAST_INTERVAL disabled_categories = [] raw_disabled = effective_config.get("disabled_tool_categories") if isinstance(raw_disabled, list): disabled_categories = [ key for key in raw_disabled if isinstance(key, str) and key in TOOL_CATEGORIES ] self.default_disabled_tool_categories = disabled_categories # Reset category states to defaults before applying overrides for key, category in TOOL_CATEGORIES.items(): self.tool_category_states[key] = False if key in disabled_categories else category.default_enabled self._refresh_disabled_tools() preferred_mode = effective_config.get("default_run_mode") if isinstance(preferred_mode, str): normalized_mode = preferred_mode.strip().lower() if normalized_mode in {"fast", "thinking", "deep"} and normalized_mode != self.run_mode: try: self.set_run_mode(normalized_mode) except ValueError: logger.warning("忽略无效默认运行模式: %s", preferred_mode) def _handle_read_tool(self, arguments: Dict) -> Dict: """集中处理 read_file 工具的三种模式。""" file_path = arguments.get("path") if not file_path: return {"success": False, "error": "缺少文件路径参数"} read_type = (arguments.get("type") or "read").lower() if read_type not in {"read", "search", "extract"}: return {"success": False, "error": f"未知的读取类型: {read_type}"} max_chars = self._clamp_int( arguments.get("max_chars"), READ_TOOL_DEFAULT_MAX_CHARS, 1, MAX_READ_FILE_CHARS ) base_result = { "success": True, "type": read_type, "path": None, "encoding": "utf-8", "max_chars": max_chars, "truncated": False } if read_type == "read": start_line, error = self._parse_optional_line(arguments.get("start_line"), "start_line") if error: return {"success": False, "error": error} end_line_val = arguments.get("end_line") end_line = None if end_line_val is not None: end_line, error = self._parse_optional_line(end_line_val, "end_line") if error: return {"success": False, "error": error} if start_line and end_line < start_line: return {"success": False, "error": "end_line 必须大于等于 start_line"} read_result = self.file_manager.read_text_segment( file_path, start_line=start_line, end_line=end_line, size_limit=READ_TOOL_MAX_FILE_SIZE ) if not read_result.get("success"): return read_result content, truncated, char_count = self._truncate_text_block(read_result["content"], max_chars) base_result.update({ "path": read_result["path"], "content": content, "line_start": read_result["line_start"], "line_end": read_result["line_end"], "total_lines": read_result["total_lines"], "file_size": read_result["size"], "char_count": char_count, "message": f"已读取 {read_result['path']} 的内容(行 {read_result['line_start']}~{read_result['line_end']})" }) base_result["truncated"] = truncated self.context_manager.load_file(read_result["path"]) return base_result if read_type == "search": query = arguments.get("query") if not query: return {"success": False, "error": "搜索模式需要提供 query 参数"} max_matches = self._clamp_int( arguments.get("max_matches"), READ_TOOL_DEFAULT_MAX_MATCHES, 1, READ_TOOL_MAX_MATCHES ) context_before = self._clamp_int( arguments.get("context_before"), READ_TOOL_DEFAULT_CONTEXT_BEFORE, 0, READ_TOOL_MAX_CONTEXT_BEFORE ) context_after = self._clamp_int( arguments.get("context_after"), READ_TOOL_DEFAULT_CONTEXT_AFTER, 0, READ_TOOL_MAX_CONTEXT_AFTER ) case_sensitive = bool(arguments.get("case_sensitive")) search_result = self.file_manager.search_text( file_path, query=query, max_matches=max_matches, context_before=context_before, context_after=context_after, case_sensitive=case_sensitive, size_limit=READ_TOOL_MAX_FILE_SIZE ) if not search_result.get("success"): return search_result matches = search_result["matches"] limited_matches, truncated, char_count = self._limit_text_chunks(matches, "snippet", max_chars) base_result.update({ "path": search_result["path"], "file_size": search_result["size"], "query": query, "max_matches": max_matches, "actual_matches": len(matches), "returned_matches": len(limited_matches), "context_before": context_before, "context_after": context_after, "case_sensitive": case_sensitive, "matches": limited_matches, "char_count": char_count, "message": f"在 {search_result['path']} 中搜索 \"{query}\",返回 {len(limited_matches)} 条结果" }) base_result["truncated"] = truncated return base_result # extract segments = arguments.get("segments") if not isinstance(segments, list) or not segments: return {"success": False, "error": "extract 模式需要提供 segments 数组"} extract_result = self.file_manager.extract_segments( file_path, segments=segments, size_limit=READ_TOOL_MAX_FILE_SIZE ) if not extract_result.get("success"): return extract_result limited_segments, truncated, char_count = self._limit_text_chunks( extract_result["segments"], "content", max_chars ) base_result.update({ "path": extract_result["path"], "segments": limited_segments, "file_size": extract_result["size"], "total_lines": extract_result["total_lines"], "segment_count": len(limited_segments), "char_count": char_count, "message": f"已从 {extract_result['path']} 抽取 {len(limited_segments)} 个片段" }) base_result["truncated"] = truncated self.context_manager.load_file(extract_result["path"]) return base_result def set_tool_category_enabled(self, category: str, enabled: bool) -> None: """设置工具类别的启用状态 / Toggle tool category enablement.""" if category not in TOOL_CATEGORIES: raise ValueError(f"未知的工具类别: {category}") self.tool_category_states[category] = bool(enabled) self._refresh_disabled_tools() def get_tool_settings_snapshot(self) -> List[Dict[str, object]]: """获取工具类别状态快照 / Return tool category states snapshot.""" snapshot: List[Dict[str, object]] = [] for key, category in TOOL_CATEGORIES.items(): snapshot.append({ "id": key, "label": category.label, "enabled": self.tool_category_states.get(key, category.default_enabled), "tools": list(category.tools), }) return snapshot def _refresh_disabled_tools(self) -> None: """刷新禁用工具列表 / Refresh disabled tool set.""" disabled = set() notice = set() for key, enabled in self.tool_category_states.items(): if not enabled: category = TOOL_CATEGORIES[key] disabled.update(category.tools) if not getattr(category, "silent_when_disabled", False): notice.update(category.tools) self.disabled_tools = disabled self.disabled_notice_tools = notice def _format_disabled_tool_notice(self) -> Optional[str]: """生成禁用工具提示信息 / Format disabled tool notice.""" if not self.disabled_notice_tools: return None lines = ["=== 工具可用性提醒 ==="] for tool_name in sorted(self.disabled_notice_tools): lines.append(f"{tool_name}:已被用户禁用") lines.append("=== 提示结束 ===") return "\n".join(lines) async def run(self): """运行主终端循环""" print(f"\n{OUTPUT_FORMATS['info']} 主终端已启动") print(f"{OUTPUT_FORMATS['info']} 当前对话: {self.context_manager.current_conversation_id}") while True: try: # 获取用户输入(使用人的表情) user_input = input("\n👤 > ").strip() if not user_input: continue # 处理命令(命令不记录到对话历史) if user_input.startswith('/'): await self.handle_command(user_input[1:]) elif user_input.lower() in ['exit', 'quit', 'q']: # 用户可能忘记加斜杠 print(f"{OUTPUT_FORMATS['info']} 提示: 使用 /exit 退出系统") continue elif user_input.lower() == 'help': print(f"{OUTPUT_FORMATS['info']} 提示: 使用 /help 查看帮助") continue else: # 确保有活动对话 self._ensure_conversation() # 只有非命令的输入才记录到对话历史 self.context_manager.add_conversation("user", user_input) # 新增:开始新的任务会话 self.current_session_id += 1 # AI回复前空一行,并显示机器人图标 print("\n🤖 >", end=" ") await self.handle_task(user_input) # 回复后自动空一行(在handle_task完成后) except KeyboardInterrupt: print(f"\n{OUTPUT_FORMATS['warning']} 使用 /exit 退出系统") continue except Exception as e: logger.error(f"主终端错误: {e}", exc_info=True) print(f"{OUTPUT_FORMATS['error']} 发生错误: {e}") # 错误后仍然尝试自动保存 try: self.context_manager.auto_save_conversation() except: pass async def handle_command(self, command: str): """处理系统命令""" parts = command.split(maxsplit=1) cmd = parts[0].lower() args = parts[1] if len(parts) > 1 else "" if cmd in self.commands: await self.commands[cmd](args) else: print(f"{OUTPUT_FORMATS['error']} 未知命令: {cmd}") await self.show_help() async def handle_task(self, user_input: str): """处理用户任务(完全修复版:彻底解决对话记录重复问题)""" try: # 如果是思考模式,每个新任务重置状态 # 注意:这里重置的是当前任务的第一次调用标志,确保新用户请求重新思考 if self.thinking_mode: self.api_client.start_new_task(force_deep=self.deep_thinking_mode) # 新增:开始新的任务会话 self.current_session_id += 1 # 构建上下文 context = self.build_context() # 构建消息 messages = self.build_messages(context, user_input) # 定义可用工具 tools = self.define_tools() # 用于收集本次任务的所有信息(关键:不立即保存到对话历史) collected_tool_calls = [] collected_tool_results = [] final_response = "" final_thinking = "" # 工具处理器:只执行工具,收集信息,绝不保存到对话历史 async def tool_handler(tool_name: str, arguments: Dict) -> str: # 执行工具调用 result = await self.handle_tool_call(tool_name, arguments) # 生成工具调用ID tool_call_id = f"call_{datetime.now().timestamp()}_{tool_name}" # 收集工具调用信息(不保存) tool_call_info = { "id": tool_call_id, "type": "function", "function": { "name": tool_name, "arguments": json.dumps(arguments, ensure_ascii=False) } } collected_tool_calls.append(tool_call_info) # 处理工具结果用于保存 try: parsed = json.loads(result) result_data = parsed if isinstance(parsed, dict) else {} except Exception: result_data = {} tool_result_content = format_tool_result_for_context(tool_name, result_data, result) # 收集工具结果(不保存) collected_tool_results.append({ "tool_call_id": tool_call_id, "name": tool_name, "content": tool_result_content, "system_message": result_data.get("system_message") if isinstance(result_data, dict) else None, "task_id": result_data.get("task_id") if isinstance(result_data, dict) else None, "raw_result_data": result_data if result_data else None, }) return result # 调用带工具的API(模型自己决定是否使用工具) response = await self.api_client.chat_with_tools( messages=messages, tools=tools, tool_handler=tool_handler ) # 保存响应内容 final_response = response # 获取思考内容(如果有的话) if self.api_client.current_task_thinking: final_thinking = self.api_client.current_task_thinking # ===== 统一保存到对话历史(关键修复) ===== # 1. 构建助手回复内容(思考内容通过 reasoning_content 存储) assistant_content = final_response or "已完成操作。" # 2. 保存assistant消息(包含tool_calls但不包含结果) self.context_manager.add_conversation( "assistant", assistant_content, collected_tool_calls if collected_tool_calls else None, reasoning_content=final_thinking or None ) # 3. 保存独立的tool消息 for tool_result in collected_tool_results: self.context_manager.add_conversation( "tool", tool_result["content"], tool_call_id=tool_result["tool_call_id"], name=tool_result["name"] ) system_message = tool_result.get("system_message") if system_message: self._record_sub_agent_message(system_message, tool_result.get("task_id"), inline=False) # 补充TODO完成提示,放在tool消息之后保证格式正确 raw_payload = tool_result.get("raw_result_data") or {} todo_note = raw_payload.get("system_note") if todo_note: self.context_manager.add_conversation("system", todo_note) # 4. 在终端显示执行信息(不保存到历史) if collected_tool_calls: tool_names = [tc['function']['name'] for tc in collected_tool_calls] for tool_name in tool_names: if tool_name == "create_file": print(f"{OUTPUT_FORMATS['file']} 创建文件") elif tool_name == "read_file": print(f"{OUTPUT_FORMATS['file']} 读取文件") elif tool_name == "ocr_image": print(f"{OUTPUT_FORMATS['file']} 图片OCR") elif tool_name == "write_file_diff": print(f"{OUTPUT_FORMATS['file']} 应用补丁") elif tool_name == "delete_file": print(f"{OUTPUT_FORMATS['file']} 删除文件") elif tool_name == "terminal_session": print(f"{OUTPUT_FORMATS['session']} 终端会话操作") elif tool_name == "terminal_input": print(f"{OUTPUT_FORMATS['terminal']} 执行终端命令") elif tool_name == "web_search": print(f"{OUTPUT_FORMATS['search']} 网络搜索") elif tool_name == "run_python": print(f"{OUTPUT_FORMATS['code']} 执行Python代码") elif tool_name == "run_command": print(f"{OUTPUT_FORMATS['terminal']} 执行系统命令") elif tool_name == "update_memory": print(f"{OUTPUT_FORMATS['memory']} 更新记忆") elif tool_name == "focus_file": print(f"🔍 聚焦文件") elif tool_name == "unfocus_file": print(f"❌ 取消聚焦") elif tool_name == "sleep": print(f"{OUTPUT_FORMATS['info']} 等待操作") else: print(f"{OUTPUT_FORMATS['action']} 执行: {tool_name}") if len(tool_names) > 1: print(f"{OUTPUT_FORMATS['info']} 共执行 {len(tool_names)} 个操作") except Exception as e: logger.error(f"任务处理错误: {e}", exc_info=True) print(f"{OUTPUT_FORMATS['error']} 任务处理失败: {e}") # 错误时也尝试自动保存 try: self.context_manager.auto_save_conversation() except: pass async def show_conversations(self, args: str = ""): """显示对话列表""" try: limit = 10 # 默认显示最近10个对话 if args: try: limit = int(args) limit = max(1, min(limit, 50)) # 限制在1-50之间 except ValueError: print(f"{OUTPUT_FORMATS['warning']} 无效数量,使用默认值10") limit = 10 conversations = self.context_manager.get_conversation_list(limit=limit) if not conversations["conversations"]: print(f"{OUTPUT_FORMATS['info']} 暂无对话记录") return print(f"\n📚 最近 {len(conversations['conversations'])} 个对话:") print("="*70) for i, conv in enumerate(conversations["conversations"], 1): # 状态图标 status_icon = "🟢" if conv["status"] == "active" else "📦" if conv["status"] == "archived" else "❌" # 当前对话标记 current_mark = " [当前]" if conv["id"] == self.context_manager.current_conversation_id else "" # 思考模式标记 mode_mark = "💭" if conv["thinking_mode"] else "⚡" print(f"{i:2d}. {status_icon} {conv['id'][:16]}...{current_mark}") print(f" {mode_mark} {conv['title'][:50]}{'...' if len(conv['title']) > 50 else ''}") print(f" 📅 {conv['updated_at'][:19]} | 💬 {conv['total_messages']} 条消息 | 🔧 {conv['total_tools']} 个工具") print(f" 📁 {conv['project_path']}") print() print(f"总计: {conversations['total']} 个对话") if conversations["has_more"]: print(f"使用 /conversations {limit + 10} 查看更多") except Exception as e: print(f"{OUTPUT_FORMATS['error']} 获取对话列表失败: {e}") async def load_conversation_command(self, args: str): """加载指定对话""" if not args: print(f"{OUTPUT_FORMATS['error']} 请指定对话ID") print("使用方法: /load <对话ID>") await self.show_conversations("5") # 显示最近5个对话作为提示 return conversation_id = args.strip() try: success = self.context_manager.load_conversation_by_id(conversation_id) if success: print(f"{OUTPUT_FORMATS['success']} 对话已加载: {conversation_id}") print(f"{OUTPUT_FORMATS['info']} 消息数量: {len(self.context_manager.conversation_history)}") # 如果是思考模式,重置状态(下次任务会重新思考) if self.thinking_mode: self.api_client.start_new_task(force_deep=self.deep_thinking_mode) self.current_session_id += 1 else: print(f"{OUTPUT_FORMATS['error']} 对话加载失败") except Exception as e: print(f"{OUTPUT_FORMATS['error']} 加载对话异常: {e}") async def new_conversation_command(self, args: str = ""): """创建新对话""" try: conversation_id = self.context_manager.start_new_conversation( project_path=self.project_path, thinking_mode=self.thinking_mode ) print(f"{OUTPUT_FORMATS['success']} 已创建新对话: {conversation_id}") # 重置相关状态 if self.thinking_mode: self.api_client.start_new_task(force_deep=self.deep_thinking_mode) self.current_session_id += 1 except Exception as e: print(f"{OUTPUT_FORMATS['error']} 创建新对话失败: {e}") async def save_conversation_command(self, args: str = ""): """手动保存当前对话""" try: success = self.context_manager.save_current_conversation() if success: print(f"{OUTPUT_FORMATS['success']} 对话已保存") else: print(f"{OUTPUT_FORMATS['error']} 对话保存失败") except Exception as e: print(f"{OUTPUT_FORMATS['error']} 保存对话异常: {e}") # ===== 修改现有命令,集成对话管理 ===== async def clear_conversation(self, args: str = ""): """清除对话记录(修改版:创建新对话而不是清空)""" if input("确认创建新对话? 当前对话将被保存 (y/n): ").lower() == 'y': try: # 保存当前对话 if self.context_manager.current_conversation_id: self.context_manager.save_current_conversation() # 创建新对话 await self.new_conversation_command() print(f"{OUTPUT_FORMATS['success']} 已开始新对话") except Exception as e: print(f"{OUTPUT_FORMATS['error']} 创建新对话失败: {e}") async def show_status(self, args: str = ""): """显示系统状态""" # 上下文状态 context_status = self.context_manager.check_context_size() # 记忆状态 memory_stats = self.memory_manager.get_memory_stats() # 文件结构 structure = self.context_manager.get_project_structure() # 聚焦文件状态 focused_size = sum(len(content) for content in self.focused_files.values()) # 终端会话状态 terminal_status = self.terminal_manager.list_terminals() # 思考模式状态 thinking_status = self.get_run_mode_label() if self.thinking_mode: thinking_status += f" ({'等待新任务' if self.api_client.current_task_first_call else '任务进行中'})" # 新增:对话统计 conversation_stats = self.context_manager.get_conversation_statistics() status_text = f""" 📊 系统状态: 项目路径: {self.project_path} 运行模式: {thinking_status} 当前对话: {self.context_manager.current_conversation_id or '无'} 上下文使用: {context_status['usage_percent']:.1f}% 当前消息: {len(self.context_manager.conversation_history)} 条 聚焦文件: {len(self.focused_files)}/3 个 ({focused_size/1024:.1f}KB) 终端会话: {terminal_status['total']}/{terminal_status['max_allowed']} 个 当前会话ID: {self.current_session_id} 项目文件: {structure['total_files']} 个 项目大小: {structure['total_size'] / 1024 / 1024:.2f} MB 对话总数: {conversation_stats.get('total_conversations', 0)} 个 历史消息: {conversation_stats.get('total_messages', 0)} 条 工具调用: {conversation_stats.get('total_tools', 0)} 次 主记忆: {memory_stats['main_memory']['lines']} 行 任务记忆: {memory_stats['task_memory']['lines']} 行 """ container_report = self._container_status_report() if container_report: status_text += container_report print(status_text) def _container_status_report(self) -> str: session = getattr(self, "container_session", None) if not session or session.mode != "docker": return "" stats = collect_stats(session.container_name, session.sandbox_bin) state = inspect_state(session.container_name, session.sandbox_bin) lines = [f" 容器: {session.container_name or '未知'}"] if stats: cpu = stats.get("cpu_percent") mem = stats.get("memory", {}) net = stats.get("net_io", {}) block = stats.get("block_io", {}) lines.append(f" CPU: {cpu:.2f}%" if cpu is not None else " CPU: 未知") if mem: used = mem.get("used_bytes") limit = mem.get("limit_bytes") percent = mem.get("percent") mem_line = " 内存: " if used is not None: mem_line += f"{used / (1024 * 1024):.2f}MB" if limit: mem_line += f" / {limit / (1024 * 1024):.2f}MB" if percent is not None: mem_line += f" ({percent:.2f}%)" lines.append(mem_line) if net: rx = net.get("rx_bytes") or 0 tx = net.get("tx_bytes") or 0 lines.append(f" 网络: ↓{rx/1024:.1f}KB ↑{tx/1024:.1f}KB") if block: read = block.get("read_bytes") or 0 write = block.get("write_bytes") or 0 lines.append(f" 磁盘: 读 {read/1024:.1f}KB / 写 {write/1024:.1f}KB") else: lines.append(" 指标: 暂无") if state: lines.append(f" 状态: {state.get('status')}") return "\n".join(lines) + "\n" async def save_state(self): """保存状态""" try: # 保存对话历史(使用新的持久化系统) self.context_manager.save_current_conversation() # 保存文件备注 self.context_manager.save_annotations() print(f"{OUTPUT_FORMATS['success']} 状态已保存") except Exception as e: print(f"{OUTPUT_FORMATS['error']} 状态保存失败: {e}") async def show_help(self, args: str = ""): """显示帮助信息""" # 根据当前模式显示不同的帮助信息 mode_info = "" if self.thinking_mode: mode_info = "\n💡 思考模式:\n - 每个新任务首次调用深度思考\n - 同一任务后续调用快速响应\n - 每个新任务都会重新思考" else: mode_info = "\n⚡ 快速模式:\n - 不进行思考,直接响应\n - 适合简单任务和快速交互" help_text = f""" 📚 可用命令: /help - 显示此帮助信息 /exit - 退出系统 /status - 显示系统状态 /memory - 管理记忆 /clear - 创建新对话 /history - 显示对话历史 /files - 显示项目文件 /focused - 显示聚焦文件 /terminals - 显示终端会话 /mode - 切换运行模式 🗂️ 对话管理: /conversations [数量] - 显示对话列表 /load <对话ID> - 加载指定对话 /new - 创建新对话 /save - 手动保存当前对话 💡 使用提示: - 直接输入任务描述,系统会自动判断是否需要执行 - 使用 Ctrl+C 可以中断当前操作 - 重要操作会要求确认 - 所有对话都会自动保存,不用担心丢失 🔍 文件聚焦功能: - 系统可以聚焦最多3个文件,实现"边看边改" - 聚焦的文件内容会持续显示在上下文中 - 适合需要频繁查看和修改的文件 📺 持久化终端: - 可以打开最多3个终端会话 - 终端保持运行状态,支持交互式程序 - 使用 terminal_session 和 terminal_input 工具控制{mode_info} """ print(help_text) # ===== 保持原有的其他方法不变,只需要小修改 ===== def define_tools(self) -> List[Dict]: """定义可用工具(添加确认工具)""" current_time = datetime.now().strftime("%Y-%m-%d %H:%M:%S") tools = [ { "type": "function", "function": { "name": "sleep", "description": "等待指定的秒数。用于等待长时间操作完成,如安装包、编译、服务启动等。当终端或进程需要时间完成操作时使用。", "parameters": { "type": "object", "properties": { "seconds": { "type": "number", "description": "等待的秒数,可以是小数(如2.5秒)。建议范围:0.5-30秒" }, "reason": { "type": "string", "description": "等待的原因说明(可选)" } }, "required": ["seconds"] } } }, { "type": "function", "function": { "name": "create_file", "description": "创建新文件(仅创建空文件,正文请使用 write_file_diff 提交补丁)", "parameters": { "type": "object", "properties": { "path": {"type": "string", "description": "文件路径"}, "file_type": {"type": "string", "enum": ["txt", "py", "md"], "description": "文件类型"}, "annotation": {"type": "string", "description": "文件备注"} }, "required": ["path", "file_type", "annotation"] } } }, { "type": "function", "function": { "name": "read_file", "description": "读取/搜索/抽取 UTF-8 文本文件内容。通过 type 参数选择 read(阅读)、search(搜索)、extract(具体行段),支持限制返回字符数。若文件非 UTF-8 或过大,请改用 run_python。", "parameters": { "type": "object", "properties": { "path": {"type": "string", "description": "文件路径"}, "type": { "type": "string", "enum": ["read", "search", "extract"], "description": "读取模式:read=阅读、search=搜索、extract=按行抽取" }, "max_chars": { "type": "integer", "description": "返回内容的最大字符数,默认与 config 一致" }, "start_line": { "type": "integer", "description": "[read] 可选的起始行号(1开始)" }, "end_line": { "type": "integer", "description": "[read] 可选的结束行号(>=start_line)" }, "query": { "type": "string", "description": "[search] 搜索关键词" }, "max_matches": { "type": "integer", "description": "[search] 最多返回多少条命中(默认5,最大50)" }, "context_before": { "type": "integer", "description": "[search] 命中行向上追加的行数(默认1,最大3)" }, "context_after": { "type": "integer", "description": "[search] 命中行向下追加的行数(默认1,最大5)" }, "case_sensitive": { "type": "boolean", "description": "[search] 是否区分大小写,默认 false" }, "segments": { "type": "array", "description": "[extract] 需要抽取的行区间", "items": { "type": "object", "properties": { "label": { "type": "string", "description": "该片段的标签(可选)" }, "start_line": { "type": "integer", "description": "起始行号(>=1)" }, "end_line": { "type": "integer", "description": "结束行号(>=start_line)" } }, "required": ["start_line", "end_line"] }, "minItems": 1 } }, "required": ["path", "type"] } } }, { "type": "function", "function": { "name": "ocr_image", "description": "使用 Qwen3-VL模型 读取图片中的文字或根据提示生成描述,仅支持本地图片路径。", "parameters": { "type": "object", "properties": { "path": {"type": "string", "description": "项目内的图片路径"}, "prompt": {"type": "string", "description": "传递给 OCR 模型的提示词,如“请识别图片中的文字”,“图中的手机是什么颜色的”必须使用中文提示词。"} }, "required": ["path", "prompt"] } } }, { "type": "function", "function": { "name": "delete_file", "description": "删除文件", "parameters": { "type": "object", "properties": { "path": {"type": "string", "description": "文件路径"} }, "required": ["path"] } } }, { "type": "function", "function": { "name": "rename_file", "description": "重命名文件", "parameters": { "type": "object", "properties": { "old_path": {"type": "string", "description": "原文件路径"}, "new_path": {"type": "string", "description": "新文件路径"} }, "required": ["old_path", "new_path"] } } }, { "type": "function", "function": { "name": "write_file_diff", "description": "使用统一 diff(`@@` 块、`-`/`+`/空格行)对单个文件做精确编辑:追加、插入、替换、删除都可以在一次调用里完成。\\n硬性规则:\\n\\n1) 补丁必须被 `*** Begin Patch` 与 `*** End Patch` 包裹。\\n2) 每个修改块必须以 `@@ [id:数字]` 开头。\\n3) 块内每一行只能是三类之一:\\n - 上下文行:以空格开头(` ␠`),表示“文件里必须原样存在”的锚点;\\n - 删除行:以 `-` 开头,表示要从文件中移除的原文;\\n - 新增行:以 `+` 开头,表示要写入的新内容。\\n4) 任何“想新增/想删除/想替换”的内容都必须逐行写 `+` 或 `-`;空行也必须写成单独一行的 `+`(这一行只有 `+` 和换行);如果你把多行新内容直接贴上去却不加 `+`,它会被当成上下文锚点去匹配原文件,极易导致“未找到匹配的原文”。\\n5) 重要语义:一个块里如果完全没有上下文行(空格开头)也没有删除行(`-`),那么它会被视为“仅追加(append-only)”,也就是把所有 `+` 行追加到文件末尾——这对“给空文件写正文”很合适,但对“插入到中间”是错误的。\\n\\n正面案例(至少 5 个,且都包含多行原文/多处修改)\\n\\n1) 给空文件写完整正文(追加到末尾;空文件=正确)\\n目标:新建 README.md 后一次性写入标题、安装、用法、FAQ(多段落、多行)。\\n要点:没有上下文/删除行 → 追加模式;空文件时最常用。\\n\\n*** Begin Patch\\n@@ [id:1]\\n+# 项目名称\\n+\\n+一个简短说明:这个项目用于……\\n+\\n+## 安装\\n+\\n+```bash\\n+pip install -r requirements.txt\\n+```\\n+\\n+## 快速开始\\n+\\n+```bash\\n+python main.py\\n+```\\n+\\n+## 常见问题\\n+\\n+- Q: 为什么会报 xxx?\\n+ A: 先检查 yyy,再确认 zzz。\\n+\\n*** End Patch\\n\\n2) “不删除,直接插入内容”到函数内部(必须用上下文锚定插入位置)\\n目标:在 def build_prompt(...): 里插入日志与参数归一化,但不改动其它行。\\n要点:插入发生在“两个上下文行之间”,上下文必须精确(包含缩进)。\\n\\n*** Begin Patch\\n@@ [id:1]\\n def build_prompt(user_text: str, system_text: str, tags: list):\\n prompt_parts = []\\n+ # 参数归一化:去掉首尾空白,避免模型误判\\n+ user_text = (user_text or \\\"\\\").strip()\\n+ system_text = (system_text or \\\"\\\").strip()\\n+\\n+ logger.debug(\\\"build_prompt: tags=%s, user_len=%d\\\", tags, len(user_text))\\n prompt_parts.append(system_text)\\n prompt_parts.append(user_text)\\n if tags:\\n prompt_parts.append(\\\"TAGS: \\\" + \\\",\\\".join(tags))\\n*** End Patch\\n\\n3) 复杂替换:整段函数重构(多行 old/new + 保留稳定上下文)\\n目标:把旧的 apply_patch()(弱校验)替换成新实现(多分支、异常信息更清晰)。\\n要点:替换不是“改一行”,而是“删一段、加一段”,并用函数签名/相邻代码作锚点。\\n\\n*** Begin Patch\\n@@ [id:1]\\n class FilePatcher:\\n def __init__(self, root: Path):\\n self.root = root\\n\\n def apply_patch(self, path: str, patch_text: str) -> dict:\\n- # old: naive replace\\n- content = (self.root / path).read_text(encoding=\\\"utf-8\\\")\\n- content = content.replace(\\\"foo\\\", \\\"bar\\\")\\n- (self.root / path).write_text(content, encoding=\\\"utf-8\\\")\\n- return {\\\"success\\\": True}\\n+ full_path = (self.root / path).resolve()\\n+ if self.root not in full_path.parents and full_path != self.root:\\n+ return {\\\"success\\\": False, \\\"error\\\": \\\"非法路径:越界访问\\\"}\\n+\\n+ if \\\"*** Begin Patch\\\" not in patch_text or \\\"*** End Patch\\\" not in patch_text:\\n+ return {\\\"success\\\": False, \\\"error\\\": \\\"补丁格式错误:缺少 Begin/End 标记\\\"}\\n+\\n+ try:\\n+ original = full_path.read_text(encoding=\\\"utf-8\\\")\\n+ except Exception as e:\\n+ return {\\\"success\\\": False, \\\"error\\\": f\\\"读取失败: {e}\\\"}\\n+\\n+ # 这里省略:解析 blocks、逐块应用、失败回滚等\\n+ updated = original\\n+ try:\\n+ full_path.write_text(updated, encoding=\\\"utf-8\\\")\\n+ except Exception as e:\\n+ return {\\\"success\\\": False, \\\"error\\\": f\\\"写入失败: {e}\\\"}\\n+\\n+ return {\\\"success\\\": True, \\\"message\\\": \\\"已应用补丁\\\"}\\n*** End Patch\\n\\n4) 复杂多块:同一文件里同时“加 import + 替换逻辑 + 插入新 helper + 删除旧函数”\\n目标:一次调用完成 4 种操作,且每块都有足够上下文,避免误匹配。\\n要点:不同区域用不同 @@ [id:n] 分块,互不干扰。\\n\\n*** Begin Patch\\n@@ [id:1]\\n-import json\\n+import json\\n+import re\\n from pathlib import Path\\n\\n@@ [id:2]\\n def normalize_user_input(text: str) -> str:\\n- return text\\n+ text = (text or \\\"\\\").strip()\\n+ # 压缩多余空白,减少提示词抖动\\n+ text = re.sub(r\\\"\\\\\\\\s+\\\", \\\" \\\", text)\\n+ return text\\n\\n@@ [id:3]\\n def load_config(path: str) -> dict:\\n cfg_path = Path(path)\\n if not cfg_path.exists():\\n return {}\\n data = cfg_path.read_text(encoding=\\\"utf-8\\\")\\n return json.loads(data)\\n+\\n+def safe_get(cfg: dict, key: str, default=None):\\n+ if not isinstance(cfg, dict):\\n+ return default\\n+ return cfg.get(key, default)\\n\\n@@ [id:4]\\n-def legacy_parse_flags(argv):\\n- # deprecated, kept for compatibility\\n- flags = {}\\n- for item in argv:\\n- if item.startswith(\\\"--\\\"):\\n- k, _, v = item[2:].partition(\\\"=\\\")\\n- flags[k] = v or True\\n- return flags\\n-\\n def main():\\n cfg = load_config(\\\"config.json\\\")\\n # ...\\n*** End Patch\\n\\n5) 删除示例:删除一整段“废弃配置块”,并顺手修正周围空行(多行删除 + 上下文)\\n目标:删掉 DEPRECATED_* 配置和旧注释,确保删除位置精确。\\n要点:删除行必须逐行 `-`;保留上下文行确保定位。\\n\\n*** Begin Patch\\n@@ [id:1]\\n # ==============================\\n # Runtime Config\\n # ==============================\\n-DEPRECATED_TIMEOUT = 5\\n-DEPRECATED_RETRIES = 1\\n-# 注意:这些字段将在下个版本移除\\n-# 请迁移到 NEW_TIMEOUT / NEW_RETRIES\\n NEW_TIMEOUT = 30\\n NEW_RETRIES = 3\\n*** End Patch\\n\\n如何写“带上下文”的正确姿势(要点)\\n\\n- 上下文要选“稳定锚点”:函数签名、类名、关键注释、紧邻的两三行缩进代码。\\n- 不要用“容易变的行”当唯一锚点:时间戳、日志序号、随机 id、生成内容片段。\\n- 上下文必须字节级一致(空格/Tab/大小写/标点都算),否则会匹配失败。\\n\\n反面案例(至少 3 个,且都是“真实会踩坑”的类型)\\n\\n反例 A(来自一次常见错误):空文件时只有第一行加了 `+`,后面直接贴正文\\n这会让后面的正文变成“上下文锚点”,工具会去空文件里找这些原文,必然失败(常见报错:未找到匹配的原文)。\\n\\n*** Begin Patch\\n@@ [id:1]\\n+\\n仰望U9X·电驭苍穹\\n银箭破空电光闪\\n三千马力云中藏\\n*** End Patch\\n\\n正确做法:正文每一行都要写 `+`(包括空行也写 `+`),空行写法是一行单独的 `+`。\\n\\n(反例:空行没加 `+`,会被当成上下文,空文件/定位修改时容易失败)\\n\\n*** Begin Patch\\n@@ [id:1]\\n+title = \\\"示例\\\"\\n\\n+[db]\\n+enabled = true\\n*** End Patch\\n\\n(正确:空行也要用 `+` 表示)\\n\\n*** Begin Patch\\n@@ [id:1]\\n+title = \\\"示例\\\"\\n+\\n+[db]\\n+enabled = true\\n*** End Patch\\n\\n(对应的正确 patch 示例:向空文件追加多行)\\n\\n*** Begin Patch\\n@@ [id:1]\\n+\\n+仰望U9X·电驭苍穹\\n+银箭破空电光闪\\n+三千马力云中藏\\n*** End Patch\\n\\n反例 B:想“插入到中间”,却只写 `+`(没有任何上下文/删除行)\\n这种块会被当成“追加到文件末尾”,结果内容跑到文件最后,不会插入到你以为的位置。\\n\\n*** Begin Patch\\n@@ [id:1]\\n+# 我以为会插到某个函数上面\\n+print(\\\"hello\\\")\\n*** End Patch\\n\\n正确做法:用上下文锚定插入点(见正面案例 2)。\\n\\n(对应的正确 patch 示例:用上下文把内容插入到函数内部,而不是追加到文件末尾)\\n\\n*** Begin Patch\\n@@ [id:1]\\n def main():\\n config = load_config(\\\"config.json\\\")\\n+ # 这里插入:启动提示(不会移动到文件末尾)\\n+ print(\\\"hello\\\")\\n run(config)\\n*** End Patch\\n\\n反例 C:补丁在第一个 `@@` 之前出现内容 / 或漏掉 Begin/End 标记\\n解析会直接报格式错误(例如:“在检测到第一个 @@ 块之前出现内容”、“缺少 Begin/End 标记”)。\\n\\n(错误形态示意)\\n这里先写了一段说明文字(没有 @@)\\n@@ [id:1]\\n+...\\n\\n正确做法:确保第一段非空内容必须从 `@@ [id:n]` 开始,并且整体有 Begin/End。\\n\\n(对应的正确 patch 示例:完整结构、第一段内容从 @@ 块开始)\\n\\n*** Begin Patch\\n@@ [id:1]\\n # ==============================\\n # Runtime Config\\n # ==============================\\n+# 说明:此处新增一行注释作为示例\\n NEW_TIMEOUT = 30\\n*** End Patch\\n", "parameters": { "type": "object", "properties": { "path": {"type": "string", "description": "目标文件路径(相对项目根目录)。"}, "patch": { "type": "string", "description": "完整补丁文本,格式类似 unified diff,必须用 *** Begin Patch / *** End Patch 包裹,并用 @@ [id:数字] 划分多个修改块。示例:*** Begin Patch\\n@@ [id:1]\\n def main():\\n- def greet(self):\\n- return \"hi\"\\n+ def greet(self, name: str) -> str:\\n+ message = f\"Hello, {name}!\"\\n+ return message\\n@@ [id:2]\\n+\\n+if __name__ == \"__main__\":\\n+ print(\"hello world\")\\n*** End Patch" } }, "required": ["path", "patch"] } } }, { "type": "function", "function": { "name": "create_folder", "description": "创建文件夹", "parameters": { "type": "object", "properties": { "path": {"type": "string", "description": "文件夹路径"} }, "required": ["path"] } } }, { "type": "function", "function": { "name": "focus_file", "description": "聚焦 UTF-8 文本文件,将完整内容持续注入上下文。适合频繁查看/修改的核心文件;超过字符限制或非 UTF-8 时会拒绝。", "parameters": { "type": "object", "properties": { "path": {"type": "string", "description": "文件路径"} }, "required": ["path"] } } }, { "type": "function", "function": { "name": "unfocus_file", "description": "取消聚焦文件,从上下文中移除", "parameters": { "type": "object", "properties": { "path": {"type": "string", "description": "文件路径"} }, "required": ["path"] } } }, { "type": "function", "function": { "name": "terminal_session", "description": "管理持久化终端会话,可打开、关闭、列出或切换终端。请在授权工作区内执行命令,禁止启动需要完整 TTY 的程序(python REPL、vim、top 等)。", "parameters": { "type": "object", "properties": { "action": { "type": "string", "enum": ["open", "close", "list", "switch"], "description": "操作类型:open-打开新终端,close-关闭终端,list-列出所有终端,switch-切换活动终端" }, "session_name": { "type": "string", "description": "终端会话名称(open、close、switch时需要)" }, "working_dir": { "type": "string", "description": "工作目录,相对于项目路径(open时可选)" } }, "required": ["action"] } } }, { "type": "function", "function": { "name": "terminal_input", "description": "向活动终端发送命令或输入。禁止启动会占用终端界面的程序(python/node/nano/vim 等);如遇卡死请结合 terminal_snapshot 并使用 terminal_reset 恢复。", "parameters": { "type": "object", "properties": { "command": { "type": "string", "description": "要执行的命令或发送的输入" }, "session_name": { "type": "string", "description": "目标终端会话名称(可选,默认使用活动终端)" }, "wait_for_output": { "type": "boolean", "description": "是否等待输出(默认true)" }, "timeout": { "type": "number", "description": "等待输出的最长秒数,默认使用配置项 TERMINAL_OUTPUT_WAIT" } }, "required": ["command"] } } }, { "type": "function", "function": { "name": "terminal_snapshot", "description": "获取指定终端最近的输出快照,用于判断当前状态。默认返回末尾的50行,可通过参数调整。", "parameters": { "type": "object", "properties": { "session_name": { "type": "string", "description": "目标终端会话名称(可选,默认活动终端)" }, "lines": { "type": "integer", "description": "返回的最大行数(可选)" }, "max_chars": { "type": "integer", "description": "返回的最大字符数(可选)" } } } } }, { "type": "function", "function": { "name": "terminal_reset", "description": "重置指定终端:关闭当前进程并重新创建同名会话,用于从卡死或非法状态中恢复。请在总结中说明重置原因。", "parameters": { "type": "object", "properties": { "session_name": { "type": "string", "description": "目标终端会话名称(可选,默认活动终端)" } } } } }, { "type": "function", "function": { "name": "web_search", "description": f"当现有资料不足时搜索外部信息(当前时间 {current_time})。调用前说明目的,精准撰写 query,并合理设置时间/主题参数;避免重复或无意义的搜索。", "parameters": { "type": "object", "properties": { "query": { "type": "string", "description": "搜索查询内容(不要包含日期或时间范围)" }, "max_results": { "type": "integer", "description": "最大结果数,可选" }, "topic": { "type": "string", "description": "搜索主题,可选值:general(默认)/news/finance" }, "time_range": { "type": "string", "description": "相对时间范围,可选 day/week/month/year,支持缩写 d/w/m/y;与 days 和 start_date/end_date 互斥" }, "days": { "type": "integer", "description": "最近 N 天,仅当 topic=news 时可用;与 time_range、start_date/end_date 互斥" }, "start_date": { "type": "string", "description": "开始日期,YYYY-MM-DD;必须与 end_date 同时提供,与 time_range、days 互斥" }, "end_date": { "type": "string", "description": "结束日期,YYYY-MM-DD;必须与 start_date 同时提供,与 time_range、days 互斥" }, "country": { "type": "string", "description": "国家过滤,仅 topic=general 可用,使用英文小写国名" } }, "required": ["query"] } } }, { "type": "function", "function": { "name": "extract_webpage", "description": "在 web_search 结果不够详细时提取网页正文。调用前说明用途,注意提取内容会消耗大量 token,超过80000字符将被拒绝。", "parameters": { "type": "object", "properties": { "url": {"type": "string", "description": "要提取内容的网页URL"} }, "required": ["url"] } } }, { "type": "function", "function": { "name": "save_webpage", "description": "提取网页内容并保存为纯文本文件,适合需要长期留存的长文档。请提供网址与目标路径(含 .txt 后缀),落地后请通过终端命令查看。", "parameters": { "type": "object", "properties": { "url": {"type": "string", "description": "要保存的网页URL"}, "target_path": {"type": "string", "description": "保存位置,包含文件名,相对于项目根目录"} }, "required": ["url", "target_path"] } } }, { "type": "function", "function": { "name": "run_python", "description": "执行一次性 Python 脚本,可用于处理二进制或非 UTF-8 文件(如 Excel、Word、PDF、图片),或进行数据分析与验证。请在脚本内显式读取文件并输出结果,避免长时间阻塞。", "parameters": { "type": "object", "properties": { "code": {"type": "string", "description": "Python代码"} }, "required": ["code"] } } }, { "type": "function", "function": { "name": "run_command", "description": "执行一次性终端命令,适合查看文件信息(file/ls/stat/iconv 等)、转换编码或调用 CLI 工具。禁止启动交互式程序;对已聚焦文件仅允许使用 grep -n 等定位命令。输出超过10000字符将被拒绝,可先限制返回体量。", "parameters": { "type": "object", "properties": { "command": {"type": "string", "description": "终端命令"} }, "required": ["command"] } } }, { "type": "function", "function": { "name": "update_memory", "description": "按条目更新记忆列表(自动编号)。append 追加新条目;replace 用序号替换;delete 用序号删除。", "parameters": { "type": "object", "properties": { "memory_type": {"type": "string", "enum": ["main", "task"], "description": "记忆类型"}, "content": {"type": "string", "description": "条目内容。append/replace 时必填"}, "operation": {"type": "string", "enum": ["append", "replace", "delete"], "description": "操作类型"}, "index": {"type": "integer", "description": "要替换/删除的序号(从1开始)"} }, "required": ["memory_type", "operation"] } } }, { "type": "function", "function": { "name": "todo_create", "description": "创建待办列表,将多步骤任务拆解为最多 8 条可执行项。概述请控制在 50 字以内,直接说明清单目标;任务列表只写 2~4 条明确步骤。", "parameters": { "type": "object", "properties": { "overview": {"type": "string", "description": "一句话概述待办清单要完成的目标,50 字以内。"}, "tasks": { "type": "array", "description": "任务列表,建议 2~4 条,每条写清“动词+对象+目标”。", "items": { "type": "object", "properties": { "title": {"type": "string", "description": "单个任务描述,写成可执行的步骤"} }, "required": ["title"] }, "minItems": 1, "maxItems": 8 } }, "required": ["overview", "tasks"] } } }, { "type": "function", "function": { "name": "todo_update_task", "description": "勾选或取消指定任务。在调整任务顺序或内容前,请先向用户说明最新理解或变更。", "parameters": { "type": "object", "properties": { "task_index": {"type": "integer", "description": "任务序号(1-8)"}, "completed": {"type": "boolean", "description": "true=打勾,false=取消"} }, "required": ["task_index", "completed"] } } }, { "type": "function", "function": { "name": "todo_finish", "description": "尝试结束待办列表,需同步汇报每项任务结果。若仍有未完事项,请注明原因与后续建议。", "parameters": { "type": "object", "properties": { "reason": {"type": "string", "description": "可选说明"} } } } }, { "type": "function", "function": { "name": "todo_finish_confirm", "description": "在任务未完成时确认是否提前结束。若确认结束,请说明后续交付建议或遗留风险。", "parameters": { "type": "object", "properties": { "confirm": {"type": "boolean", "description": "true=确认结束,false=继续执行"}, "reason": {"type": "string", "description": "确认结束时的说明"} }, "required": ["confirm"] } } }, { "type": "function", "function": { "name": "close_sub_agent", "description": "强制关闭指定子智能体,适用于长时间无响应、超时或卡死的任务。使用前请确认必要的日志/文件已保留,操作会立即终止该任务。", "parameters": { "type": "object", "properties": { "task_id": {"type": "string", "description": "子智能体任务ID"}, "agent_id": {"type": "integer", "description": "子智能体编号(1~5),若缺少 task_id 可用"} } } } }, { "type": "function", "function": { "name": "create_sub_agent", "description": "创建新的子智能体任务。适合大规模信息搜集、网页提取与多文档总结等会占用大量上下文的工作,需要提供任务摘要、详细要求、交付目录以及参考文件。注意:同一时间最多运行5个子智能体。", "parameters": { "type": "object", "properties": { "agent_id": {"type": "integer", "description": "子智能体代号(1~5)"}, "summary": {"type": "string", "description": "任务摘要,简要说明目标"}, "task": {"type": "string", "description": "任务详细要求"}, "target_dir": {"type": "string", "description": "项目下用于接收交付的相对目录"}, "reference_files": { "type": "array", "description": "提供给子智能体的参考文件列表(相对路径),禁止在summary和task中直接告知子智能体引用图片的路径,必须使用本参数提供", "items": {"type": "string"}, "maxItems": 10 }, "timeout_seconds": {"type": "integer", "description": "子智能体最大运行秒数:单/双次搜索建议180秒,多轮搜索整理建议300秒,深度调研或长篇分析可设600秒"} }, "required": ["agent_id", "summary", "task", "target_dir"] } } }, { "type": "function", "function": { "name": "wait_sub_agent", "description": "等待指定子智能体任务结束(或超时)。任务完成后会返回交付目录,并将结果复制到指定的项目文件夹。调用时 `timeout_seconds` 应不少于对应子智能体的 `timeout_seconds`,否则可能提前终止等待。", "parameters": { "type": "object", "properties": { "task_id": {"type": "string", "description": "子智能体任务ID"}, "agent_id": {"type": "integer", "description": "子智能体代号(可选,用于缺省 task_id 的情况)"}, "timeout_seconds": {"type": "integer", "description": "本次等待的超时时长(秒)"} }, "required": [] } } }, { "type": "function", "function": { "name": "trigger_easter_egg", "description": "触发隐藏彩蛋,用于展示非功能性特效。需指定 effect 参数,例如 flood(灌水)或 snake(贪吃蛇)。", "parameters": { "type": "object", "properties": { "effect": { "type": "string", "description": "彩蛋标识,目前支持 flood(灌水)与 snake(贪吃蛇)。" } }, "required": ["effect"] } } } ] if self.disabled_tools: tools = [ tool for tool in tools if tool.get("function", {}).get("name") not in self.disabled_tools ] return tools async def handle_tool_call(self, tool_name: str, arguments: Dict) -> str: """处理工具调用(添加参数预检查和改进错误处理)""" # 导入字符限制配置 from config import ( MAX_READ_FILE_CHARS, MAX_FOCUS_FILE_CHARS, MAX_RUN_COMMAND_CHARS, MAX_EXTRACT_WEBPAGE_CHARS ) # 检查是否需要确认 if tool_name in NEED_CONFIRMATION: if not await self.confirm_action(tool_name, arguments): return json.dumps({"success": False, "error": "用户取消操作"}) # === 新增:预检查参数大小和格式 === try: # 检查参数总大小 arguments_str = json.dumps(arguments, ensure_ascii=False) if len(arguments_str) > 50000: # 50KB限制 return json.dumps({ "success": False, "error": f"参数过大({len(arguments_str)}字符),超过50KB限制", "suggestion": "请分块处理或减少参数内容" }, ensure_ascii=False) # 针对特定工具的内容检查 if tool_name == "create_file" and "content" in arguments: content = arguments.get("content", "") if not DISABLE_LENGTH_CHECK and len(content) > 9999999999: # 30KB内容限制 return json.dumps({ "success": False, "error": f"文件内容过长({len(content)}字符),建议分块处理", "suggestion": "请拆分内容或使用 write_file_diff 工具输出结构化补丁" }, ensure_ascii=False) # 检查内容中的特殊字符 if '\\' in content and content.count('\\') > len(content) / 10: print(f"{OUTPUT_FORMATS['warning']} 检测到大量转义字符,可能存在格式问题") except Exception as e: return json.dumps({ "success": False, "error": f"参数预检查失败: {str(e)}" }, ensure_ascii=False) try: if tool_name == "read_file": result = self._handle_read_tool(arguments) elif tool_name == "ocr_image": path = arguments.get("path") prompt = arguments.get("prompt") if not path: return json.dumps({"success": False, "error": "缺少 path 参数", "warnings": []}, ensure_ascii=False) result = self.ocr_client.ocr_image(path=path, prompt=prompt or "") # 终端会话管理工具 elif tool_name == "terminal_session": action = arguments["action"] if action == "open": result = self.terminal_manager.open_terminal( session_name=arguments.get("session_name", "default"), working_dir=arguments.get("working_dir"), make_active=True ) if result["success"]: print(f"{OUTPUT_FORMATS['session']} 终端会话已打开: {arguments.get('session_name', 'default')}") elif action == "close": result = self.terminal_manager.close_terminal( session_name=arguments.get("session_name", "default") ) if result["success"]: print(f"{OUTPUT_FORMATS['session']} 终端会话已关闭: {arguments.get('session_name', 'default')}") elif action == "list": result = self.terminal_manager.list_terminals() elif action == "switch": result = self.terminal_manager.switch_terminal( session_name=arguments.get("session_name", "default") ) if result["success"]: print(f"{OUTPUT_FORMATS['session']} 切换到终端: {arguments.get('session_name', 'default')}") else: result = {"success": False, "error": f"未知操作: {action}"} result["action"] = action # 终端输入工具 elif tool_name == "terminal_input": result = self.terminal_manager.send_to_terminal( command=arguments["command"], session_name=arguments.get("session_name"), wait_for_output=arguments.get("wait_for_output", True), timeout=arguments.get("timeout") ) if result["success"]: print(f"{OUTPUT_FORMATS['terminal']} 执行命令: {arguments['command']}") elif tool_name == "terminal_snapshot": result = self.terminal_manager.get_terminal_snapshot( session_name=arguments.get("session_name"), lines=arguments.get("lines"), max_chars=arguments.get("max_chars") ) elif tool_name == "terminal_reset": result = self.terminal_manager.reset_terminal( session_name=arguments.get("session_name") ) if result["success"]: print(f"{OUTPUT_FORMATS['session']} 终端会话已重置: {result['session']}") # sleep工具 elif tool_name == "sleep": seconds = arguments.get("seconds", 1) reason = arguments.get("reason", "等待操作完成") # 限制最大等待时间 max_sleep = 600 # 最多等待60秒 if seconds > max_sleep: result = { "success": False, "error": f"等待时间过长,最多允许 {max_sleep} 秒", "suggestion": f"建议分多次等待或减少等待时间" } else: # 确保秒数为正数 if seconds <= 0: result = { "success": False, "error": "等待时间必须大于0" } else: print(f"{OUTPUT_FORMATS['info']} 等待 {seconds} 秒: {reason}") # 执行等待 import asyncio await asyncio.sleep(seconds) result = { "success": True, "message": f"已等待 {seconds} 秒", "reason": reason, "timestamp": datetime.now().isoformat() } print(f"{OUTPUT_FORMATS['success']} 等待完成") elif tool_name == "create_file": result = self.file_manager.create_file( path=arguments["path"], file_type=arguments["file_type"] ) # 添加备注 if result["success"] and arguments.get("annotation"): self.context_manager.update_annotation( result["path"], arguments["annotation"] ) if result.get("success"): result["message"] = ( f"已创建空文件: {result['path']}。请使用 write_file_diff 提交补丁写入正文。" ) elif tool_name == "delete_file": result = self.file_manager.delete_file(arguments["path"]) # 如果删除成功,同时删除备注和聚焦 if result.get("success") and result.get("action") == "deleted": deleted_path = result.get("path") # 删除备注 if deleted_path in self.context_manager.file_annotations: del self.context_manager.file_annotations[deleted_path] self.context_manager.save_annotations() print(f"🧹 已删除文件备注: {deleted_path}") # 删除聚焦 if deleted_path in self.focused_files: del self.focused_files[deleted_path] print(f"🔍 已取消文件聚焦: {deleted_path}") elif tool_name == "rename_file": result = self.file_manager.rename_file( arguments["old_path"], arguments["new_path"] ) # 如果重命名成功,更新备注和聚焦的key if result.get("success") and result.get("action") == "renamed": old_path = result.get("old_path") new_path = result.get("new_path") # 更新备注 if old_path in self.context_manager.file_annotations: annotation = self.context_manager.file_annotations[old_path] del self.context_manager.file_annotations[old_path] self.context_manager.file_annotations[new_path] = annotation self.context_manager.save_annotations() print(f"📝 已更新文件备注: {old_path} -> {new_path}") # 更新聚焦 if old_path in self.focused_files: content = self.focused_files[old_path] del self.focused_files[old_path] self.focused_files[new_path] = content print(f"🔍 已更新文件聚焦: {old_path} -> {new_path}") elif tool_name == "write_file_diff": path = arguments.get("path") patch_text = arguments.get("patch") if not path or not patch_text: result = {"success": False, "error": "缺少必要参数: path/patch"} else: length_limit = 30000 if not DISABLE_LENGTH_CHECK and len(patch_text) > length_limit: result = { "success": False, "error": f"补丁内容过长({len(patch_text)}字符),超过{length_limit}字符上限", "suggestion": "请拆分补丁后多次调用 write_file_diff。" } else: diff_result = self.file_manager.apply_diff_patch(path, patch_text) result = diff_result elif tool_name == "create_folder": result = self.file_manager.create_folder(arguments["path"]) elif tool_name == "focus_file": path = arguments["path"] # 检查是否已经聚焦 if path in self.focused_files: result = {"success": False, "error": f"文件已经处于聚焦状态: {path}"} else: # 检查聚焦文件数量限制 if len(self.focused_files) >= 3: result = { "success": False, "error": f"已达到最大聚焦文件数量(3个),当前聚焦: {list(self.focused_files.keys())}", "suggestion": "请先使用 unfocus_file 取消部分文件的聚焦" } else: # 读取文件内容 read_result = self.file_manager.read_file(path) if read_result["success"]: # 字符数检查 char_count = len(read_result["content"]) if char_count > MAX_FOCUS_FILE_CHARS: result = { "success": False, "error": f"文件过大,有{char_count}字符,请使用run_command限制字符数返回", "char_count": char_count, "limit": MAX_FOCUS_FILE_CHARS } else: self.focused_files[path] = read_result["content"] result = { "success": True, "message": f"文件已聚焦: {path}", "focused_files": list(self.focused_files.keys()), "file_size": len(read_result["content"]) } print(f"🔍 文件已聚焦: {path} ({len(read_result['content'])} 字节)") else: result = read_result elif tool_name == "unfocus_file": path = arguments["path"] if path in self.focused_files: del self.focused_files[path] result = { "success": True, "message": f"已取消文件聚焦: {path}", "remaining_focused": list(self.focused_files.keys()) } print(f"✖️ 已取消文件聚焦: {path}") else: result = {"success": False, "error": f"文件未处于聚焦状态: {path}"} elif tool_name == "web_search": allowed, quota_info = self.record_search_call() if not allowed: return json.dumps({ "success": False, "error": f"搜索配额已用尽,将在 {quota_info.get('reset_at')} 重置。请向用户说明情况并提供替代方案。", "quota": quota_info }, ensure_ascii=False) search_response = await self.search_engine.search_with_summary( query=arguments["query"], max_results=arguments.get("max_results"), topic=arguments.get("topic"), time_range=arguments.get("time_range"), days=arguments.get("days"), start_date=arguments.get("start_date"), end_date=arguments.get("end_date"), country=arguments.get("country") ) if search_response["success"]: result = { "success": True, "summary": search_response["summary"], "filters": search_response.get("filters", {}), "query": search_response.get("query"), "results": search_response.get("results", []), "total_results": search_response.get("total_results", 0) } else: result = { "success": False, "error": search_response.get("error", "搜索失败"), "filters": search_response.get("filters", {}), "query": search_response.get("query"), "results": search_response.get("results", []), "total_results": search_response.get("total_results", 0) } elif tool_name == "extract_webpage": url = arguments["url"] try: # 从config获取API密钥 from config import TAVILY_API_KEY full_content, _ = await extract_webpage_content( urls=url, api_key=TAVILY_API_KEY, extract_depth="basic", max_urls=1 ) # 字符数检查 char_count = len(full_content) if char_count > MAX_EXTRACT_WEBPAGE_CHARS: result = { "success": False, "error": f"网页提取返回了过长的{char_count}字符,请不要提取这个网页,可以使用网页保存功能,然后使用read工具查找或查看网页", "char_count": char_count, "limit": MAX_EXTRACT_WEBPAGE_CHARS, "url": url } else: result = { "success": True, "url": url, "content": full_content } except Exception as e: result = { "success": False, "error": f"网页提取失败: {str(e)}", "url": url } elif tool_name == "save_webpage": url = arguments["url"] target_path = arguments["target_path"] try: from config import TAVILY_API_KEY except ImportError: TAVILY_API_KEY = None if not TAVILY_API_KEY or TAVILY_API_KEY == "your-tavily-api-key": result = { "success": False, "error": "Tavily API密钥未配置,无法保存网页", "url": url, "path": target_path } else: try: extract_result = await tavily_extract( urls=url, api_key=TAVILY_API_KEY, extract_depth="basic", max_urls=1 ) if not extract_result or "error" in extract_result: error_message = extract_result.get("error", "提取失败,未返回任何内容") if isinstance(extract_result, dict) else "提取失败" result = { "success": False, "error": error_message, "url": url, "path": target_path } else: results_list = extract_result.get("results", []) if isinstance(extract_result, dict) else [] primary_result = None for item in results_list: if item.get("raw_content"): primary_result = item break if primary_result is None and results_list: primary_result = results_list[0] if not primary_result: failed_list = extract_result.get("failed_results", []) if isinstance(extract_result, dict) else [] result = { "success": False, "error": "提取成功结果为空,无法保存", "url": url, "path": target_path, "failed": failed_list } else: content_to_save = primary_result.get("raw_content") or primary_result.get("content") or "" if not content_to_save: result = { "success": False, "error": "网页内容为空,未写入文件", "url": url, "path": target_path } else: write_result = self.file_manager.write_file(target_path, content_to_save, mode="w") if not write_result.get("success"): result = { "success": False, "error": write_result.get("error", "写入文件失败"), "url": url, "path": target_path } else: char_count = len(content_to_save) byte_size = len(content_to_save.encode("utf-8")) result = { "success": True, "url": url, "path": write_result.get("path", target_path), "char_count": char_count, "byte_size": byte_size, "message": f"网页内容已以纯文本保存到 {write_result.get('path', target_path)},请使用终端命令查看(文件建议为 .txt)。" } if isinstance(extract_result, dict) and extract_result.get("failed_results"): result["warnings"] = extract_result["failed_results"] except Exception as e: result = { "success": False, "error": f"网页保存失败: {str(e)}", "url": url, "path": target_path } elif tool_name == "run_python": result = await self.terminal_ops.run_python_code(arguments["code"]) elif tool_name == "run_command": result = await self.terminal_ops.run_command(arguments["command"]) # 字符数检查 if result.get("success") and "output" in result: char_count = len(result["output"]) if char_count > MAX_RUN_COMMAND_CHARS: result = { "success": False, "error": f"结果内容过大,有{char_count}字符,请使用限制字符数的获取内容方式,根据程度选择10k以内的数", "char_count": char_count, "limit": MAX_RUN_COMMAND_CHARS, "command": arguments["command"] } elif tool_name == "update_memory": memory_type = arguments["memory_type"] operation = arguments["operation"] content = arguments.get("content") index = arguments.get("index") # 参数校验 if operation == "append" and (not content or not str(content).strip()): result = {"success": False, "error": "append 操作需要 content"} elif operation == "replace" and (index is None or index <= 0 or not content or not str(content).strip()): result = {"success": False, "error": "replace 操作需要有效的 index 和 content"} elif operation == "delete" and (index is None or index <= 0): result = {"success": False, "error": "delete 操作需要有效的 index"} else: result = self.memory_manager.update_entries( memory_type=memory_type, operation=operation, content=content, index=index ) elif tool_name == "todo_create": result = self.todo_manager.create_todo_list( overview=arguments.get("overview", ""), tasks=arguments.get("tasks", []) ) elif tool_name == "todo_update_task": result = self.todo_manager.update_task_status( task_index=arguments.get("task_index"), completed=arguments.get("completed", True) ) elif tool_name == "todo_finish": result = self.todo_manager.finish_todo( reason=arguments.get("reason") ) elif tool_name == "todo_finish_confirm": result = self.todo_manager.confirm_finish( confirm=arguments.get("confirm", False), reason=arguments.get("reason") ) elif tool_name == "create_sub_agent": result = self.sub_agent_manager.create_sub_agent( agent_id=arguments.get("agent_id"), summary=arguments.get("summary", ""), task=arguments.get("task", ""), target_dir=arguments.get("target_dir", ""), reference_files=arguments.get("reference_files", []), timeout_seconds=arguments.get("timeout_seconds"), conversation_id=self.context_manager.current_conversation_id ) elif tool_name == "wait_sub_agent": wait_timeout = arguments.get("timeout_seconds") if not wait_timeout: task_ref = self.sub_agent_manager.lookup_task( task_id=arguments.get("task_id"), agent_id=arguments.get("agent_id") ) if task_ref: wait_timeout = task_ref.get("timeout_seconds") result = self.sub_agent_manager.wait_for_completion( task_id=arguments.get("task_id"), agent_id=arguments.get("agent_id"), timeout_seconds=wait_timeout ) elif tool_name == "close_sub_agent": result = self.sub_agent_manager.terminate_sub_agent( task_id=arguments.get("task_id"), agent_id=arguments.get("agent_id") ) elif tool_name == "trigger_easter_egg": result = self.easter_egg_manager.trigger_effect(arguments.get("effect")) else: result = {"success": False, "error": f"未知工具: {tool_name}"} except Exception as e: logger.error(f"工具执行失败: {tool_name} - {e}") result = {"success": False, "error": f"工具执行异常: {str(e)}"} return json.dumps(result, ensure_ascii=False) async def confirm_action(self, action: str, arguments: Dict) -> bool: """确认危险操作""" print(f"\n{OUTPUT_FORMATS['confirm']} 需要确认的操作:") print(f" 操作: {action}") print(f" 参数: {json.dumps(arguments, ensure_ascii=False, indent=2)}") response = input("\n是否继续? (y/n): ").strip().lower() return response == 'y' def build_context(self) -> Dict: """构建主终端上下文""" # 读取记忆 memory = self.memory_manager.read_main_memory() # 构建上下文 return self.context_manager.build_main_context(memory) def _tool_calls_followed_by_tools(self, conversation: List[Dict], start_idx: int, tool_calls: List[Dict]) -> bool: """判断指定助手消息的工具调用是否拥有后续的工具响应。""" if not tool_calls: return False expected_ids = [tc.get("id") for tc in tool_calls if tc.get("id")] if not expected_ids: return False matched: Set[str] = set() idx = start_idx + 1 total = len(conversation) while idx < total and len(matched) < len(expected_ids): next_conv = conversation[idx] role = next_conv.get("role") if role == "tool": call_id = next_conv.get("tool_call_id") if call_id in expected_ids: matched.add(call_id) else: break elif role in ("assistant", "user"): break idx += 1 return len(matched) == len(expected_ids) def build_messages(self, context: Dict, user_input: str) -> List[Dict]: """构建消息列表(添加终端内容注入)""" # 加载系统提示 system_prompt = self.load_prompt("main_system") # 格式化系统提示 container_path = self.container_mount_path or "/workspace" container_cpus = self.container_cpu_limit container_memory = self.container_memory_limit project_storage = self.project_storage_limit system_prompt = system_prompt.format( project_path=container_path, container_path=container_path, container_cpus=container_cpus, container_memory=container_memory, project_storage=project_storage, file_tree=context["project_info"]["file_tree"], memory=context["memory"], current_time=datetime.now().strftime("%Y-%m-%d %H:%M:%S") ) messages = [ {"role": "system", "content": system_prompt} ] if self.tool_category_states.get("todo", True): todo_prompt = self.load_prompt("todo_guidelines").strip() if todo_prompt: messages.append({"role": "system", "content": todo_prompt}) if self.tool_category_states.get("sub_agent", True): sub_agent_prompt = self.load_prompt("sub_agent_guidelines").strip() if sub_agent_prompt: messages.append({"role": "system", "content": sub_agent_prompt}) if self.deep_thinking_mode: deep_prompt = self.load_prompt("deep_thinking_mode_guidelines").strip() if deep_prompt: messages.append({"role": "system", "content": deep_prompt}) elif self.thinking_mode: thinking_prompt = self.load_prompt("thinking_mode_guidelines").strip() if thinking_prompt: messages.append({"role": "system", "content": thinking_prompt}) personalization_config = load_personalization_config(self.data_dir) personalization_block = build_personalization_prompt(personalization_config, include_header=False) if personalization_block: personalization_template = self.load_prompt("personalization").strip() if personalization_template and "{personalization_block}" in personalization_template: personalization_text = personalization_template.format(personalization_block=personalization_block) elif personalization_template: personalization_text = f"{personalization_template}\n{personalization_block}" else: personalization_text = personalization_block messages.append({"role": "system", "content": personalization_text}) # 添加对话历史(保留完整结构,包括tool_calls和tool消息) conversation = context["conversation"] for idx, conv in enumerate(conversation): metadata = conv.get("metadata") or {} if conv["role"] == "assistant": # Assistant消息可能包含工具调用 message = { "role": conv["role"], "content": conv["content"] } reasoning = conv.get("reasoning_content") if reasoning: message["reasoning_content"] = reasoning # 如果有工具调用信息,添加到消息中 tool_calls = conv.get("tool_calls") or [] if tool_calls and self._tool_calls_followed_by_tools(conversation, idx, tool_calls): message["tool_calls"] = tool_calls messages.append(message) elif conv["role"] == "tool": # Tool消息需要保留完整结构 message = { "role": "tool", "content": conv["content"], "tool_call_id": conv.get("tool_call_id", ""), "name": conv.get("name", "") } messages.append(message) elif conv["role"] == "system" and metadata.get("sub_agent_notice"): # 转换为用户消息,让模型能及时响应 messages.append({ "role": "user", "content": conv["content"] }) else: # User 或普通 System 消息 messages.append({ "role": conv["role"], "content": conv["content"] }) # 当前用户输入已经在conversation中了,不需要重复添加 todo_message = self.context_manager.render_todo_system_message() if todo_message: messages.append({ "role": "system", "content": todo_message }) # 在最后注入聚焦文件内容作为系统消息 if self.focused_files: focused_content = "\n\n=== 🔍 正在聚焦的文件 ===\n" focused_content += f"(共 {len(self.focused_files)} 个文件处于聚焦状态)\n" for path, content in self.focused_files.items(): size_kb = len(content) / 1024 focused_content += f"\n--- 文件: {path} ({size_kb:.1f}KB) ---\n" focused_content += f"```\n{content}\n```\n" focused_content += "\n=== 聚焦文件结束 ===\n" focused_content += "提示:以上文件正在被聚焦,你可以直接看到完整内容并进行修改,禁止再次读取。" messages.append({ "role": "system", "content": focused_content }) disabled_notice = self._format_disabled_tool_notice() if disabled_notice: messages.append({ "role": "system", "content": disabled_notice }) return messages def load_prompt(self, name: str) -> str: """加载提示模板""" prompt_file = Path(PROMPTS_DIR) / f"{name}.txt" if prompt_file.exists(): with open(prompt_file, 'r', encoding='utf-8') as f: return f.read() return "你是一个AI助手。" async def show_focused_files(self, args: str = ""): """显示当前聚焦的文件""" if not self.focused_files: print(f"{OUTPUT_FORMATS['info']} 当前没有聚焦的文件") else: print(f"\n🔍 聚焦文件列表 ({len(self.focused_files)}/3):") print("="*50) for path, content in self.focused_files.items(): size_kb = len(content) / 1024 lines = content.count('\n') + 1 print(f" 📄 {path}") print(f" 大小: {size_kb:.1f}KB | 行数: {lines}") print("="*50) async def show_terminals(self, args: str = ""): """显示终端会话列表""" result = self.terminal_manager.list_terminals() if result["total"] == 0: print(f"{OUTPUT_FORMATS['info']} 当前没有活动的终端会话") else: print(f"\n📺 终端会话列表 ({result['total']}/{result['max_allowed']}):") print("="*50) for session in result["sessions"]: status_icon = "🟢" if session["is_running"] else "🔴" active_mark = " [活动]" if session["is_active"] else "" print(f" {status_icon} {session['session_name']}{active_mark}") print(f" 工作目录: {session['working_dir']}") print(f" Shell: {session['shell']}") print(f" 运行时间: {session['uptime_seconds']:.1f}秒") if session["is_interactive"]: print(f" ⚠️ 等待输入") print("="*50) async def exit_system(self, args: str = ""): """退出系统""" print(f"{OUTPUT_FORMATS['info']} 正在退出...") # 关闭所有终端会话 self.terminal_manager.close_all() # 保存状态 await self.save_state() exit(0) async def manage_memory(self, args: str = ""): """管理记忆""" if not args: print(""" 🧠 记忆管理: /memory show [main|task] - 显示记忆内容 /memory edit [main|task] - 编辑记忆 /memory clear task - 清空任务记忆 /memory merge - 合并任务记忆到主记忆 /memory backup [main|task]- 备份记忆 """) return parts = args.split() action = parts[0] if parts else "" target = parts[1] if len(parts) > 1 else "main" if action == "show": if target == "main": content = self.memory_manager.read_main_memory() else: content = self.memory_manager.read_task_memory() print(f"\n{'='*50}") print(content) print('='*50) elif action == "clear" and target == "task": if input("确认清空任务记忆? (y/n): ").lower() == 'y': self.memory_manager.clear_task_memory() elif action == "merge": self.memory_manager.merge_memories() elif action == "backup": path = self.memory_manager.backup_memory(target) if path: print(f"备份保存到: {path}") async def show_history(self, args: str = ""): """显示对话历史""" history = self.context_manager.conversation_history[-2000:] # 最近2000条 print("\n📜 对话历史:") print("="*50) for conv in history: timestamp = conv.get("timestamp", "") if conv["role"] == "user": role = "👤 用户" elif conv["role"] == "assistant": role = "🤖 助手" elif conv["role"] == "tool": role = f"🔧 工具[{conv.get('name', 'unknown')}]" else: role = conv["role"] content = conv["content"][:100] + "..." if len(conv["content"]) > 100 else conv["content"] print(f"\n[{timestamp[:19]}] {role}:") print(content) # 如果是助手消息且有工具调用,显示工具信息 if conv["role"] == "assistant" and "tool_calls" in conv and conv["tool_calls"]: tools = [tc["function"]["name"] for tc in conv["tool_calls"]] print(f" 🔗 调用工具: {', '.join(tools)}") print("="*50) async def show_files(self, args: str = ""): """显示项目文件""" structure = self.context_manager.get_project_structure() print(f"\n📁 项目文件结构:") print(self.context_manager._build_file_tree(structure)) print(f"\n总计: {structure['total_files']} 个文件, {structure['total_size'] / 1024 / 1024:.2f} MB") def set_run_mode(self, mode: str) -> str: """统一设置运行模式""" allowed = ["fast", "thinking", "deep"] normalized = mode.lower() if normalized not in allowed: raise ValueError(f"不支持的模式: {mode}") previous_mode = getattr(self, "run_mode", "fast") self.run_mode = normalized self.thinking_mode = normalized != "fast" self.deep_thinking_mode = normalized == "deep" self.api_client.thinking_mode = self.thinking_mode self.api_client.set_deep_thinking_mode(self.deep_thinking_mode) if self.deep_thinking_mode: self.api_client.force_thinking_next_call = False self.api_client.skip_thinking_next_call = False if not self.thinking_mode: self.api_client.start_new_task() elif previous_mode == "deep" and normalized != "deep": self.api_client.start_new_task() return self.run_mode def get_run_mode_label(self) -> str: labels = { "fast": "快速模式(无思考)", "thinking": "思考模式(首次调用使用思考模型)", "deep": "深度思考模式(整轮使用思考模型)" } return labels.get(self.run_mode, "快速模式(无思考)") async def toggle_mode(self, args: str = ""): """切换运行模式""" modes = ["fast", "thinking", "deep"] target_mode = "" if args: candidate = args.strip().lower() if candidate not in modes: print(f"{OUTPUT_FORMATS['error']} 无效模式: {args}。可选: fast / thinking / deep") return target_mode = candidate else: current_index = modes.index(self.run_mode) if self.run_mode in modes else 0 target_mode = modes[(current_index + 1) % len(modes)] if target_mode == self.run_mode: print(f"{OUTPUT_FORMATS['info']} 当前已是 {self.get_run_mode_label()}") return try: self.set_run_mode(target_mode) print(f"{OUTPUT_FORMATS['info']} 已切换到: {self.get_run_mode_label()}") except ValueError as exc: print(f"{OUTPUT_FORMATS['error']} {exc}")