codepp 0.0.437__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (288) hide show
  1. code_puppy/__init__.py +10 -0
  2. code_puppy/__main__.py +10 -0
  3. code_puppy/agents/__init__.py +31 -0
  4. code_puppy/agents/agent_c_reviewer.py +155 -0
  5. code_puppy/agents/agent_code_puppy.py +117 -0
  6. code_puppy/agents/agent_code_reviewer.py +90 -0
  7. code_puppy/agents/agent_cpp_reviewer.py +132 -0
  8. code_puppy/agents/agent_creator_agent.py +638 -0
  9. code_puppy/agents/agent_golang_reviewer.py +151 -0
  10. code_puppy/agents/agent_helios.py +124 -0
  11. code_puppy/agents/agent_javascript_reviewer.py +160 -0
  12. code_puppy/agents/agent_manager.py +742 -0
  13. code_puppy/agents/agent_pack_leader.py +385 -0
  14. code_puppy/agents/agent_planning.py +165 -0
  15. code_puppy/agents/agent_python_programmer.py +169 -0
  16. code_puppy/agents/agent_python_reviewer.py +90 -0
  17. code_puppy/agents/agent_qa_expert.py +163 -0
  18. code_puppy/agents/agent_qa_kitten.py +208 -0
  19. code_puppy/agents/agent_scheduler.py +121 -0
  20. code_puppy/agents/agent_security_auditor.py +181 -0
  21. code_puppy/agents/agent_terminal_qa.py +323 -0
  22. code_puppy/agents/agent_typescript_reviewer.py +166 -0
  23. code_puppy/agents/base_agent.py +2156 -0
  24. code_puppy/agents/event_stream_handler.py +348 -0
  25. code_puppy/agents/json_agent.py +202 -0
  26. code_puppy/agents/pack/__init__.py +34 -0
  27. code_puppy/agents/pack/bloodhound.py +304 -0
  28. code_puppy/agents/pack/husky.py +327 -0
  29. code_puppy/agents/pack/retriever.py +393 -0
  30. code_puppy/agents/pack/shepherd.py +348 -0
  31. code_puppy/agents/pack/terrier.py +287 -0
  32. code_puppy/agents/pack/watchdog.py +367 -0
  33. code_puppy/agents/prompt_reviewer.py +145 -0
  34. code_puppy/agents/subagent_stream_handler.py +276 -0
  35. code_puppy/api/__init__.py +13 -0
  36. code_puppy/api/app.py +169 -0
  37. code_puppy/api/main.py +21 -0
  38. code_puppy/api/pty_manager.py +453 -0
  39. code_puppy/api/routers/__init__.py +12 -0
  40. code_puppy/api/routers/agents.py +36 -0
  41. code_puppy/api/routers/commands.py +217 -0
  42. code_puppy/api/routers/config.py +75 -0
  43. code_puppy/api/routers/sessions.py +234 -0
  44. code_puppy/api/templates/terminal.html +361 -0
  45. code_puppy/api/websocket.py +154 -0
  46. code_puppy/callbacks.py +692 -0
  47. code_puppy/chatgpt_codex_client.py +338 -0
  48. code_puppy/claude_cache_client.py +672 -0
  49. code_puppy/cli_runner.py +1073 -0
  50. code_puppy/command_line/__init__.py +1 -0
  51. code_puppy/command_line/add_model_menu.py +1092 -0
  52. code_puppy/command_line/agent_menu.py +662 -0
  53. code_puppy/command_line/attachments.py +395 -0
  54. code_puppy/command_line/autosave_menu.py +704 -0
  55. code_puppy/command_line/clipboard.py +527 -0
  56. code_puppy/command_line/colors_menu.py +532 -0
  57. code_puppy/command_line/command_handler.py +293 -0
  58. code_puppy/command_line/command_registry.py +150 -0
  59. code_puppy/command_line/config_commands.py +719 -0
  60. code_puppy/command_line/core_commands.py +867 -0
  61. code_puppy/command_line/diff_menu.py +865 -0
  62. code_puppy/command_line/file_path_completion.py +73 -0
  63. code_puppy/command_line/load_context_completion.py +52 -0
  64. code_puppy/command_line/mcp/__init__.py +10 -0
  65. code_puppy/command_line/mcp/base.py +32 -0
  66. code_puppy/command_line/mcp/catalog_server_installer.py +175 -0
  67. code_puppy/command_line/mcp/custom_server_form.py +688 -0
  68. code_puppy/command_line/mcp/custom_server_installer.py +195 -0
  69. code_puppy/command_line/mcp/edit_command.py +148 -0
  70. code_puppy/command_line/mcp/handler.py +138 -0
  71. code_puppy/command_line/mcp/help_command.py +147 -0
  72. code_puppy/command_line/mcp/install_command.py +214 -0
  73. code_puppy/command_line/mcp/install_menu.py +705 -0
  74. code_puppy/command_line/mcp/list_command.py +94 -0
  75. code_puppy/command_line/mcp/logs_command.py +235 -0
  76. code_puppy/command_line/mcp/remove_command.py +82 -0
  77. code_puppy/command_line/mcp/restart_command.py +100 -0
  78. code_puppy/command_line/mcp/search_command.py +123 -0
  79. code_puppy/command_line/mcp/start_all_command.py +135 -0
  80. code_puppy/command_line/mcp/start_command.py +117 -0
  81. code_puppy/command_line/mcp/status_command.py +184 -0
  82. code_puppy/command_line/mcp/stop_all_command.py +112 -0
  83. code_puppy/command_line/mcp/stop_command.py +80 -0
  84. code_puppy/command_line/mcp/test_command.py +107 -0
  85. code_puppy/command_line/mcp/utils.py +129 -0
  86. code_puppy/command_line/mcp/wizard_utils.py +334 -0
  87. code_puppy/command_line/mcp_completion.py +174 -0
  88. code_puppy/command_line/model_picker_completion.py +197 -0
  89. code_puppy/command_line/model_settings_menu.py +932 -0
  90. code_puppy/command_line/motd.py +96 -0
  91. code_puppy/command_line/onboarding_slides.py +179 -0
  92. code_puppy/command_line/onboarding_wizard.py +342 -0
  93. code_puppy/command_line/pin_command_completion.py +329 -0
  94. code_puppy/command_line/prompt_toolkit_completion.py +846 -0
  95. code_puppy/command_line/session_commands.py +302 -0
  96. code_puppy/command_line/shell_passthrough.py +145 -0
  97. code_puppy/command_line/skills_completion.py +160 -0
  98. code_puppy/command_line/uc_menu.py +893 -0
  99. code_puppy/command_line/utils.py +93 -0
  100. code_puppy/command_line/wiggum_state.py +78 -0
  101. code_puppy/config.py +1770 -0
  102. code_puppy/error_logging.py +134 -0
  103. code_puppy/gemini_code_assist.py +385 -0
  104. code_puppy/gemini_model.py +754 -0
  105. code_puppy/hook_engine/README.md +105 -0
  106. code_puppy/hook_engine/__init__.py +21 -0
  107. code_puppy/hook_engine/aliases.py +155 -0
  108. code_puppy/hook_engine/engine.py +221 -0
  109. code_puppy/hook_engine/executor.py +296 -0
  110. code_puppy/hook_engine/matcher.py +156 -0
  111. code_puppy/hook_engine/models.py +240 -0
  112. code_puppy/hook_engine/registry.py +106 -0
  113. code_puppy/hook_engine/validator.py +144 -0
  114. code_puppy/http_utils.py +361 -0
  115. code_puppy/keymap.py +128 -0
  116. code_puppy/main.py +10 -0
  117. code_puppy/mcp_/__init__.py +66 -0
  118. code_puppy/mcp_/async_lifecycle.py +286 -0
  119. code_puppy/mcp_/blocking_startup.py +469 -0
  120. code_puppy/mcp_/captured_stdio_server.py +275 -0
  121. code_puppy/mcp_/circuit_breaker.py +290 -0
  122. code_puppy/mcp_/config_wizard.py +507 -0
  123. code_puppy/mcp_/dashboard.py +308 -0
  124. code_puppy/mcp_/error_isolation.py +407 -0
  125. code_puppy/mcp_/examples/retry_example.py +226 -0
  126. code_puppy/mcp_/health_monitor.py +589 -0
  127. code_puppy/mcp_/managed_server.py +428 -0
  128. code_puppy/mcp_/manager.py +807 -0
  129. code_puppy/mcp_/mcp_logs.py +224 -0
  130. code_puppy/mcp_/registry.py +451 -0
  131. code_puppy/mcp_/retry_manager.py +337 -0
  132. code_puppy/mcp_/server_registry_catalog.py +1126 -0
  133. code_puppy/mcp_/status_tracker.py +355 -0
  134. code_puppy/mcp_/system_tools.py +209 -0
  135. code_puppy/mcp_prompts/__init__.py +1 -0
  136. code_puppy/mcp_prompts/hook_creator.py +103 -0
  137. code_puppy/messaging/__init__.py +255 -0
  138. code_puppy/messaging/bus.py +613 -0
  139. code_puppy/messaging/commands.py +167 -0
  140. code_puppy/messaging/markdown_patches.py +57 -0
  141. code_puppy/messaging/message_queue.py +361 -0
  142. code_puppy/messaging/messages.py +569 -0
  143. code_puppy/messaging/queue_console.py +271 -0
  144. code_puppy/messaging/renderers.py +311 -0
  145. code_puppy/messaging/rich_renderer.py +1158 -0
  146. code_puppy/messaging/spinner/__init__.py +83 -0
  147. code_puppy/messaging/spinner/console_spinner.py +240 -0
  148. code_puppy/messaging/spinner/spinner_base.py +95 -0
  149. code_puppy/messaging/subagent_console.py +460 -0
  150. code_puppy/model_factory.py +848 -0
  151. code_puppy/model_switching.py +63 -0
  152. code_puppy/model_utils.py +168 -0
  153. code_puppy/models.json +174 -0
  154. code_puppy/models_dev_api.json +1 -0
  155. code_puppy/models_dev_parser.py +592 -0
  156. code_puppy/plugins/__init__.py +186 -0
  157. code_puppy/plugins/agent_skills/__init__.py +22 -0
  158. code_puppy/plugins/agent_skills/config.py +175 -0
  159. code_puppy/plugins/agent_skills/discovery.py +136 -0
  160. code_puppy/plugins/agent_skills/downloader.py +392 -0
  161. code_puppy/plugins/agent_skills/installer.py +22 -0
  162. code_puppy/plugins/agent_skills/metadata.py +219 -0
  163. code_puppy/plugins/agent_skills/prompt_builder.py +60 -0
  164. code_puppy/plugins/agent_skills/register_callbacks.py +241 -0
  165. code_puppy/plugins/agent_skills/remote_catalog.py +322 -0
  166. code_puppy/plugins/agent_skills/skill_catalog.py +257 -0
  167. code_puppy/plugins/agent_skills/skills_install_menu.py +664 -0
  168. code_puppy/plugins/agent_skills/skills_menu.py +781 -0
  169. code_puppy/plugins/antigravity_oauth/__init__.py +10 -0
  170. code_puppy/plugins/antigravity_oauth/accounts.py +406 -0
  171. code_puppy/plugins/antigravity_oauth/antigravity_model.py +706 -0
  172. code_puppy/plugins/antigravity_oauth/config.py +42 -0
  173. code_puppy/plugins/antigravity_oauth/constants.py +133 -0
  174. code_puppy/plugins/antigravity_oauth/oauth.py +478 -0
  175. code_puppy/plugins/antigravity_oauth/register_callbacks.py +518 -0
  176. code_puppy/plugins/antigravity_oauth/storage.py +288 -0
  177. code_puppy/plugins/antigravity_oauth/test_plugin.py +319 -0
  178. code_puppy/plugins/antigravity_oauth/token.py +167 -0
  179. code_puppy/plugins/antigravity_oauth/transport.py +863 -0
  180. code_puppy/plugins/antigravity_oauth/utils.py +168 -0
  181. code_puppy/plugins/chatgpt_oauth/__init__.py +8 -0
  182. code_puppy/plugins/chatgpt_oauth/config.py +52 -0
  183. code_puppy/plugins/chatgpt_oauth/oauth_flow.py +329 -0
  184. code_puppy/plugins/chatgpt_oauth/register_callbacks.py +176 -0
  185. code_puppy/plugins/chatgpt_oauth/test_plugin.py +301 -0
  186. code_puppy/plugins/chatgpt_oauth/utils.py +523 -0
  187. code_puppy/plugins/claude_code_hooks/__init__.py +1 -0
  188. code_puppy/plugins/claude_code_hooks/config.py +137 -0
  189. code_puppy/plugins/claude_code_hooks/register_callbacks.py +175 -0
  190. code_puppy/plugins/claude_code_oauth/README.md +167 -0
  191. code_puppy/plugins/claude_code_oauth/SETUP.md +93 -0
  192. code_puppy/plugins/claude_code_oauth/__init__.py +25 -0
  193. code_puppy/plugins/claude_code_oauth/config.py +52 -0
  194. code_puppy/plugins/claude_code_oauth/register_callbacks.py +453 -0
  195. code_puppy/plugins/claude_code_oauth/test_plugin.py +283 -0
  196. code_puppy/plugins/claude_code_oauth/token_refresh_heartbeat.py +241 -0
  197. code_puppy/plugins/claude_code_oauth/utils.py +640 -0
  198. code_puppy/plugins/customizable_commands/__init__.py +0 -0
  199. code_puppy/plugins/customizable_commands/register_callbacks.py +152 -0
  200. code_puppy/plugins/example_custom_command/README.md +280 -0
  201. code_puppy/plugins/example_custom_command/register_callbacks.py +51 -0
  202. code_puppy/plugins/file_permission_handler/__init__.py +4 -0
  203. code_puppy/plugins/file_permission_handler/register_callbacks.py +470 -0
  204. code_puppy/plugins/frontend_emitter/__init__.py +25 -0
  205. code_puppy/plugins/frontend_emitter/emitter.py +121 -0
  206. code_puppy/plugins/frontend_emitter/register_callbacks.py +261 -0
  207. code_puppy/plugins/hook_creator/__init__.py +1 -0
  208. code_puppy/plugins/hook_creator/register_callbacks.py +33 -0
  209. code_puppy/plugins/hook_manager/__init__.py +1 -0
  210. code_puppy/plugins/hook_manager/config.py +290 -0
  211. code_puppy/plugins/hook_manager/hooks_menu.py +564 -0
  212. code_puppy/plugins/hook_manager/register_callbacks.py +227 -0
  213. code_puppy/plugins/oauth_puppy_html.py +228 -0
  214. code_puppy/plugins/scheduler/__init__.py +1 -0
  215. code_puppy/plugins/scheduler/register_callbacks.py +88 -0
  216. code_puppy/plugins/scheduler/scheduler_menu.py +522 -0
  217. code_puppy/plugins/scheduler/scheduler_wizard.py +341 -0
  218. code_puppy/plugins/shell_safety/__init__.py +6 -0
  219. code_puppy/plugins/shell_safety/agent_shell_safety.py +69 -0
  220. code_puppy/plugins/shell_safety/command_cache.py +156 -0
  221. code_puppy/plugins/shell_safety/register_callbacks.py +202 -0
  222. code_puppy/plugins/synthetic_status/__init__.py +1 -0
  223. code_puppy/plugins/synthetic_status/register_callbacks.py +132 -0
  224. code_puppy/plugins/synthetic_status/status_api.py +147 -0
  225. code_puppy/plugins/universal_constructor/__init__.py +13 -0
  226. code_puppy/plugins/universal_constructor/models.py +138 -0
  227. code_puppy/plugins/universal_constructor/register_callbacks.py +47 -0
  228. code_puppy/plugins/universal_constructor/registry.py +302 -0
  229. code_puppy/plugins/universal_constructor/sandbox.py +584 -0
  230. code_puppy/prompts/antigravity_system_prompt.md +1 -0
  231. code_puppy/pydantic_patches.py +356 -0
  232. code_puppy/reopenable_async_client.py +232 -0
  233. code_puppy/round_robin_model.py +150 -0
  234. code_puppy/scheduler/__init__.py +41 -0
  235. code_puppy/scheduler/__main__.py +9 -0
  236. code_puppy/scheduler/cli.py +118 -0
  237. code_puppy/scheduler/config.py +126 -0
  238. code_puppy/scheduler/daemon.py +280 -0
  239. code_puppy/scheduler/executor.py +155 -0
  240. code_puppy/scheduler/platform.py +19 -0
  241. code_puppy/scheduler/platform_unix.py +22 -0
  242. code_puppy/scheduler/platform_win.py +32 -0
  243. code_puppy/session_storage.py +338 -0
  244. code_puppy/status_display.py +257 -0
  245. code_puppy/summarization_agent.py +176 -0
  246. code_puppy/terminal_utils.py +418 -0
  247. code_puppy/tools/__init__.py +501 -0
  248. code_puppy/tools/agent_tools.py +603 -0
  249. code_puppy/tools/ask_user_question/__init__.py +26 -0
  250. code_puppy/tools/ask_user_question/constants.py +73 -0
  251. code_puppy/tools/ask_user_question/demo_tui.py +55 -0
  252. code_puppy/tools/ask_user_question/handler.py +232 -0
  253. code_puppy/tools/ask_user_question/models.py +304 -0
  254. code_puppy/tools/ask_user_question/registration.py +26 -0
  255. code_puppy/tools/ask_user_question/renderers.py +309 -0
  256. code_puppy/tools/ask_user_question/terminal_ui.py +329 -0
  257. code_puppy/tools/ask_user_question/theme.py +155 -0
  258. code_puppy/tools/ask_user_question/tui_loop.py +423 -0
  259. code_puppy/tools/browser/__init__.py +37 -0
  260. code_puppy/tools/browser/browser_control.py +289 -0
  261. code_puppy/tools/browser/browser_interactions.py +545 -0
  262. code_puppy/tools/browser/browser_locators.py +640 -0
  263. code_puppy/tools/browser/browser_manager.py +378 -0
  264. code_puppy/tools/browser/browser_navigation.py +251 -0
  265. code_puppy/tools/browser/browser_screenshot.py +179 -0
  266. code_puppy/tools/browser/browser_scripts.py +462 -0
  267. code_puppy/tools/browser/browser_workflows.py +221 -0
  268. code_puppy/tools/browser/chromium_terminal_manager.py +259 -0
  269. code_puppy/tools/browser/terminal_command_tools.py +534 -0
  270. code_puppy/tools/browser/terminal_screenshot_tools.py +552 -0
  271. code_puppy/tools/browser/terminal_tools.py +525 -0
  272. code_puppy/tools/command_runner.py +1346 -0
  273. code_puppy/tools/common.py +1409 -0
  274. code_puppy/tools/display.py +84 -0
  275. code_puppy/tools/file_modifications.py +886 -0
  276. code_puppy/tools/file_operations.py +802 -0
  277. code_puppy/tools/scheduler_tools.py +412 -0
  278. code_puppy/tools/skills_tools.py +244 -0
  279. code_puppy/tools/subagent_context.py +158 -0
  280. code_puppy/tools/tools_content.py +51 -0
  281. code_puppy/tools/universal_constructor.py +889 -0
  282. code_puppy/uvx_detection.py +242 -0
  283. code_puppy/version_checker.py +82 -0
  284. codepp-0.0.437.dist-info/METADATA +766 -0
  285. codepp-0.0.437.dist-info/RECORD +288 -0
  286. codepp-0.0.437.dist-info/WHEEL +4 -0
  287. codepp-0.0.437.dist-info/entry_points.txt +3 -0
  288. codepp-0.0.437.dist-info/licenses/LICENSE +21 -0
@@ -0,0 +1,460 @@
1
+ """SubAgentConsoleManager - Aggregated display for parallel sub-agents.
2
+
3
+ Provides a Rich Live dashboard that shows real-time status of multiple
4
+ running sub-agents, each in its own panel with spinner animations,
5
+ status badges, and performance metrics.
6
+
7
+ Usage:
8
+ >>> manager = SubAgentConsoleManager.get_instance()
9
+ >>> manager.register_agent("session-123", "code-puppy", "gpt-4o")
10
+ >>> manager.update_agent("session-123", status="running", tool_call_count=5)
11
+ >>> manager.unregister_agent("session-123")
12
+ """
13
+
14
+ import threading
15
+ import time
16
+ from dataclasses import dataclass, field
17
+ from typing import Dict, List, Optional
18
+
19
+ from rich.console import Console, Group
20
+ from rich.live import Live
21
+ from rich.panel import Panel
22
+ from rich.table import Table
23
+ from rich.text import Text
24
+
25
+ from code_puppy.messaging.messages import SubAgentStatusMessage
26
+
27
+ # =============================================================================
28
+ # Status Configuration
29
+ # =============================================================================
30
+
31
+ STATUS_STYLES = {
32
+ "starting": {"color": "cyan", "spinner": "dots", "emoji": "🚀"},
33
+ "running": {"color": "green", "spinner": "dots", "emoji": "🐕"},
34
+ "thinking": {"color": "magenta", "spinner": "dots", "emoji": "🤔"},
35
+ "tool_calling": {"color": "yellow", "spinner": "dots12", "emoji": "🔧"},
36
+ "completed": {"color": "green", "spinner": None, "emoji": "✅"},
37
+ "error": {"color": "red", "spinner": None, "emoji": "❌"},
38
+ }
39
+
40
+ DEFAULT_STYLE = {"color": "white", "spinner": "dots", "emoji": "⏳"}
41
+
42
+
43
+ # =============================================================================
44
+ # Agent State Tracking
45
+ # =============================================================================
46
+
47
+
48
+ @dataclass
49
+ class AgentState:
50
+ """Internal state tracking for a single sub-agent.
51
+
52
+ Tracks all metrics needed for rendering the agent's status panel,
53
+ including timing, tool usage, and error information.
54
+ """
55
+
56
+ session_id: str
57
+ agent_name: str
58
+ model_name: str
59
+ status: str = "starting"
60
+ tool_call_count: int = 0
61
+ token_count: int = 0
62
+ current_tool: Optional[str] = None
63
+ start_time: float = field(default_factory=time.time)
64
+ error_message: Optional[str] = None
65
+
66
+ def elapsed_seconds(self) -> float:
67
+ """Calculate elapsed time since agent started."""
68
+ return time.time() - self.start_time
69
+
70
+ def elapsed_formatted(self) -> str:
71
+ """Format elapsed time as human-readable string."""
72
+ elapsed = self.elapsed_seconds()
73
+ if elapsed < 60:
74
+ return f"{elapsed:.1f}s"
75
+ minutes = int(elapsed // 60)
76
+ seconds = elapsed % 60
77
+ return f"{minutes}m {seconds:.1f}s"
78
+
79
+ def to_status_message(self) -> SubAgentStatusMessage:
80
+ """Convert to a SubAgentStatusMessage for bus emission."""
81
+ return SubAgentStatusMessage(
82
+ session_id=self.session_id,
83
+ agent_name=self.agent_name,
84
+ model_name=self.model_name,
85
+ status=self.status, # type: ignore[arg-type]
86
+ tool_call_count=self.tool_call_count,
87
+ token_count=self.token_count,
88
+ current_tool=self.current_tool,
89
+ elapsed_seconds=self.elapsed_seconds(),
90
+ error_message=self.error_message,
91
+ )
92
+
93
+
94
+ # =============================================================================
95
+ # SubAgent Console Manager
96
+ # =============================================================================
97
+
98
+
99
+ class SubAgentConsoleManager:
100
+ """Manager for displaying multiple parallel sub-agents in Rich Live panels.
101
+
102
+ This is a singleton that tracks all running sub-agents and renders them
103
+ in a unified Rich Live display. Each agent gets its own panel with:
104
+ - Agent name and session ID
105
+ - Model being used
106
+ - Status with spinner animation (for active states)
107
+ - Tool call count and current tool
108
+ - Token count
109
+ - Elapsed time
110
+
111
+ The display auto-starts when the first agent registers and auto-stops
112
+ when the last agent unregisters.
113
+
114
+ Thread-safe: All operations are protected by locks.
115
+ """
116
+
117
+ _instance: Optional["SubAgentConsoleManager"] = None
118
+ _lock = threading.Lock()
119
+
120
+ def __init__(self, console: Optional[Console] = None):
121
+ """Initialize the manager.
122
+
123
+ Args:
124
+ console: Optional Rich Console instance. If not provided,
125
+ a new one will be created.
126
+ """
127
+ self.console = console or Console()
128
+ self._agents: Dict[str, AgentState] = {}
129
+ self._agents_lock = threading.RLock() # Reentrant lock for agent operations
130
+ self._live: Optional[Live] = None
131
+ self._update_thread: Optional[threading.Thread] = None
132
+ self._stop_event = threading.Event()
133
+
134
+ @classmethod
135
+ def get_instance(
136
+ cls, console: Optional[Console] = None
137
+ ) -> "SubAgentConsoleManager":
138
+ """Get or create the singleton instance.
139
+
140
+ Thread-safe singleton pattern using double-checked locking.
141
+
142
+ Args:
143
+ console: Optional Rich Console to use. Only used when creating
144
+ the initial instance.
145
+
146
+ Returns:
147
+ The singleton SubAgentConsoleManager instance.
148
+ """
149
+ if cls._instance is None:
150
+ with cls._lock:
151
+ # Double-check inside lock
152
+ if cls._instance is None:
153
+ cls._instance = cls(console)
154
+ return cls._instance
155
+
156
+ @classmethod
157
+ def reset_instance(cls) -> None:
158
+ """Reset the singleton instance (primarily for testing).
159
+
160
+ Stops any running display and clears the singleton.
161
+ """
162
+ with cls._lock:
163
+ if cls._instance is not None:
164
+ cls._instance._stop_display()
165
+ cls._instance = None
166
+
167
+ # =========================================================================
168
+ # Agent Registration
169
+ # =========================================================================
170
+
171
+ def register_agent(self, session_id: str, agent_name: str, model_name: str) -> None:
172
+ """Register a new sub-agent and start display if needed.
173
+
174
+ Args:
175
+ session_id: Unique identifier for this agent session.
176
+ agent_name: Name of the agent (e.g., 'code-puppy', 'qa-kitten').
177
+ model_name: Name of the model being used (e.g., 'gpt-4o').
178
+ """
179
+ with self._agents_lock:
180
+ # Create new agent state
181
+ self._agents[session_id] = AgentState(
182
+ session_id=session_id,
183
+ agent_name=agent_name,
184
+ model_name=model_name,
185
+ )
186
+
187
+ # Start display if this is the first agent
188
+ if len(self._agents) == 1:
189
+ self._start_display()
190
+
191
+ def update_agent(self, session_id: str, **kwargs) -> None:
192
+ """Update status of an existing agent.
193
+
194
+ Args:
195
+ session_id: The session ID of the agent to update.
196
+ **kwargs: Fields to update. Valid fields:
197
+ - status: Current status string
198
+ - tool_call_count: Number of tools called
199
+ - token_count: Tokens in context
200
+ - current_tool: Name of tool being called (or None)
201
+ - error_message: Error message if status is 'error'
202
+ """
203
+ with self._agents_lock:
204
+ if session_id not in self._agents:
205
+ return # Silently ignore updates for unknown agents
206
+
207
+ agent = self._agents[session_id]
208
+
209
+ # Update only provided fields
210
+ if "status" in kwargs:
211
+ agent.status = kwargs["status"]
212
+ if "tool_call_count" in kwargs:
213
+ agent.tool_call_count = kwargs["tool_call_count"]
214
+ if "token_count" in kwargs:
215
+ agent.token_count = kwargs["token_count"]
216
+ if "current_tool" in kwargs:
217
+ agent.current_tool = kwargs["current_tool"]
218
+ if "error_message" in kwargs:
219
+ agent.error_message = kwargs["error_message"]
220
+
221
+ def unregister_agent(
222
+ self, session_id: str, final_status: str = "completed"
223
+ ) -> None:
224
+ """Remove an agent from tracking.
225
+
226
+ Args:
227
+ session_id: The session ID of the agent to remove.
228
+ final_status: Final status to set before removal (for display).
229
+ Defaults to 'completed'.
230
+ """
231
+ with self._agents_lock:
232
+ if session_id in self._agents:
233
+ # Set final status
234
+ self._agents[session_id].status = final_status
235
+ # Remove from tracking
236
+ del self._agents[session_id]
237
+
238
+ # Stop display if no agents remain
239
+ if not self._agents:
240
+ self._stop_display()
241
+
242
+ def get_agent_state(self, session_id: str) -> Optional[AgentState]:
243
+ """Get the current state of an agent.
244
+
245
+ Args:
246
+ session_id: The session ID to look up.
247
+
248
+ Returns:
249
+ The AgentState if found, None otherwise.
250
+ """
251
+ with self._agents_lock:
252
+ return self._agents.get(session_id)
253
+
254
+ def get_all_agents(self) -> List[AgentState]:
255
+ """Get a list of all currently tracked agents.
256
+
257
+ Returns:
258
+ List of AgentState objects (copies to prevent mutation).
259
+ """
260
+ with self._agents_lock:
261
+ return list(self._agents.values())
262
+
263
+ # =========================================================================
264
+ # Display Management
265
+ # =========================================================================
266
+
267
+ def _start_display(self) -> None:
268
+ """Start the Rich Live display.
269
+
270
+ Creates the Live context and starts a background thread to
271
+ continuously refresh the display.
272
+ """
273
+ if self._live is not None:
274
+ return # Already running
275
+
276
+ self._stop_event.clear()
277
+
278
+ # Create Live display
279
+ self._live = Live(
280
+ self._render_display(),
281
+ console=self.console,
282
+ refresh_per_second=10,
283
+ transient=True, # Clear when stopped
284
+ )
285
+ self._live.start()
286
+
287
+ # Start background update thread
288
+ self._update_thread = threading.Thread(
289
+ target=self._update_loop, daemon=True, name="SubAgentDisplayUpdater"
290
+ )
291
+ self._update_thread.start()
292
+
293
+ def _stop_display(self) -> None:
294
+ """Stop the Rich Live display when no agents remain."""
295
+ # Signal stop
296
+ self._stop_event.set()
297
+
298
+ # Stop update thread
299
+ if self._update_thread is not None:
300
+ self._update_thread.join(timeout=1.0)
301
+ self._update_thread = None
302
+
303
+ # Stop Live display
304
+ if self._live is not None:
305
+ try:
306
+ self._live.stop()
307
+ except Exception:
308
+ pass # Ignore errors during cleanup
309
+ self._live = None
310
+
311
+ def _update_loop(self) -> None:
312
+ """Background thread that refreshes the display."""
313
+ while not self._stop_event.is_set():
314
+ try:
315
+ if self._live is not None:
316
+ self._live.update(self._render_display())
317
+ except Exception:
318
+ pass # Ignore rendering errors, keep trying
319
+
320
+ # Sleep between updates (10 FPS)
321
+ time.sleep(0.1)
322
+
323
+ # =========================================================================
324
+ # Rendering
325
+ # =========================================================================
326
+
327
+ def _render_display(self) -> Group:
328
+ """Render all agent panels as a Rich Group.
329
+
330
+ Returns:
331
+ A Group containing all agent panels stacked vertically.
332
+ """
333
+ with self._agents_lock:
334
+ if not self._agents:
335
+ return Group(Text("No active sub-agents", style="dim"))
336
+
337
+ panels = [
338
+ self._render_agent_panel(agent) for agent in self._agents.values()
339
+ ]
340
+ return Group(*panels)
341
+
342
+ def _render_agent_panel(self, agent: AgentState) -> Panel:
343
+ """Render a single agent's status panel.
344
+
345
+ Args:
346
+ agent: The AgentState to render.
347
+
348
+ Returns:
349
+ A Rich Panel containing the agent's status information.
350
+ """
351
+ style_config = STATUS_STYLES.get(agent.status, DEFAULT_STYLE)
352
+ color = style_config["color"]
353
+ spinner_name = style_config["spinner"]
354
+ emoji = style_config["emoji"]
355
+
356
+ # Build the content table
357
+ table = Table.grid(padding=(0, 2))
358
+ table.add_column("label", style="dim")
359
+ table.add_column("value")
360
+
361
+ # Status row with spinner (if active)
362
+ status_text = Text()
363
+ status_text.append(f"{emoji} ", style=color)
364
+ if spinner_name:
365
+ # For active statuses, we add the status text
366
+ # The spinner is visual only in Rich Live
367
+ status_text.append(agent.status.upper(), style=f"bold {color}")
368
+ else:
369
+ status_text.append(agent.status.upper(), style=f"bold {color}")
370
+
371
+ table.add_row("Status:", status_text)
372
+
373
+ # Model
374
+ table.add_row("Model:", Text(agent.model_name, style="cyan"))
375
+
376
+ # Session ID (truncated for display)
377
+ session_display = agent.session_id
378
+ if len(session_display) > 24:
379
+ session_display = session_display[:21] + "..."
380
+ table.add_row("Session:", Text(session_display, style="dim"))
381
+
382
+ # Tool calls
383
+ tool_text = Text()
384
+ tool_text.append(str(agent.tool_call_count), style="bold yellow")
385
+ if agent.current_tool:
386
+ tool_text.append(" (calling: ", style="dim")
387
+ tool_text.append(agent.current_tool, style="yellow")
388
+ tool_text.append(")", style="dim")
389
+ table.add_row("Tools:", tool_text)
390
+
391
+ # Token count
392
+ token_display = f"{agent.token_count:,}" if agent.token_count else "0"
393
+ table.add_row("Tokens:", Text(token_display, style="blue"))
394
+
395
+ # Elapsed time
396
+ table.add_row("Elapsed:", Text(agent.elapsed_formatted(), style="magenta"))
397
+
398
+ # Error message (if any)
399
+ if agent.error_message:
400
+ error_text = Text(agent.error_message, style="red")
401
+ table.add_row("Error:", error_text)
402
+
403
+ # Build panel title with spinner for active states
404
+ title = Text()
405
+ title.append("🐕 ", style="bold")
406
+ title.append(agent.agent_name, style=f"bold {color}")
407
+
408
+ # Create panel
409
+ return Panel(
410
+ table,
411
+ title=title,
412
+ border_style=color,
413
+ padding=(0, 1),
414
+ )
415
+
416
+ # =========================================================================
417
+ # Context Manager Support
418
+ # =========================================================================
419
+
420
+ def __enter__(self) -> "SubAgentConsoleManager":
421
+ """Support use as context manager."""
422
+ return self
423
+
424
+ def __exit__(self, exc_type, exc_val, exc_tb) -> None:
425
+ """Clean up on context exit."""
426
+ self._stop_display()
427
+
428
+
429
+ # =============================================================================
430
+ # Convenience Functions
431
+ # =============================================================================
432
+
433
+
434
+ def get_subagent_console_manager(
435
+ console: Optional[Console] = None,
436
+ ) -> SubAgentConsoleManager:
437
+ """Get the singleton SubAgentConsoleManager instance.
438
+
439
+ Convenience function for accessing the manager.
440
+
441
+ Args:
442
+ console: Optional Rich Console (only used on first call).
443
+
444
+ Returns:
445
+ The singleton SubAgentConsoleManager.
446
+ """
447
+ return SubAgentConsoleManager.get_instance(console)
448
+
449
+
450
+ # =============================================================================
451
+ # Exports
452
+ # =============================================================================
453
+
454
+ __all__ = [
455
+ "AgentState",
456
+ "SubAgentConsoleManager",
457
+ "get_subagent_console_manager",
458
+ "STATUS_STYLES",
459
+ "DEFAULT_STYLE",
460
+ ]