claude-mpm 5.1.9__py3-none-any.whl → 5.4.48__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.

Potentially problematic release.


This version of claude-mpm might be problematic. Click here for more details.

Files changed (248) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/__init__.py +4 -0
  3. claude_mpm/agents/BASE_AGENT.md +164 -0
  4. claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +1 -1
  5. claude_mpm/agents/MEMORY.md +1 -1
  6. claude_mpm/agents/PM_INSTRUCTIONS.md +843 -900
  7. claude_mpm/agents/WORKFLOW.md +5 -254
  8. claude_mpm/agents/agent_loader.py +13 -44
  9. claude_mpm/agents/base_agent.json +1 -1
  10. claude_mpm/agents/frontmatter_validator.py +2 -2
  11. claude_mpm/agents/templates/circuit-breakers.md +138 -1
  12. claude_mpm/cli/__main__.py +4 -0
  13. claude_mpm/cli/chrome_devtools_installer.py +175 -0
  14. claude_mpm/cli/commands/agent_state_manager.py +18 -27
  15. claude_mpm/cli/commands/agents.py +9 -40
  16. claude_mpm/cli/commands/auto_configure.py +210 -25
  17. claude_mpm/cli/commands/config.py +88 -2
  18. claude_mpm/cli/commands/configure.py +1098 -159
  19. claude_mpm/cli/commands/configure_agent_display.py +25 -6
  20. claude_mpm/cli/commands/mpm_init/core.py +225 -46
  21. claude_mpm/cli/commands/mpm_init/knowledge_extractor.py +481 -0
  22. claude_mpm/cli/commands/mpm_init/prompts.py +280 -0
  23. claude_mpm/cli/commands/postmortem.py +1 -1
  24. claude_mpm/cli/commands/profile.py +277 -0
  25. claude_mpm/cli/commands/skills.py +218 -197
  26. claude_mpm/cli/commands/summarize.py +413 -0
  27. claude_mpm/cli/executor.py +21 -3
  28. claude_mpm/cli/interactive/agent_wizard.py +2 -2
  29. claude_mpm/cli/parsers/agents_parser.py +0 -9
  30. claude_mpm/cli/parsers/auto_configure_parser.py +0 -138
  31. claude_mpm/cli/parsers/base_parser.py +12 -0
  32. claude_mpm/cli/parsers/config_parser.py +153 -83
  33. claude_mpm/cli/parsers/profile_parser.py +148 -0
  34. claude_mpm/cli/parsers/skills_parser.py +0 -5
  35. claude_mpm/cli/startup.py +876 -149
  36. claude_mpm/commands/mpm-config.md +28 -0
  37. claude_mpm/commands/mpm-doctor.md +9 -22
  38. claude_mpm/commands/mpm-help.md +5 -287
  39. claude_mpm/commands/mpm-init.md +81 -507
  40. claude_mpm/commands/mpm-monitor.md +15 -402
  41. claude_mpm/commands/mpm-organize.md +120 -0
  42. claude_mpm/commands/mpm-postmortem.md +6 -108
  43. claude_mpm/commands/mpm-session-resume.md +12 -363
  44. claude_mpm/commands/mpm-status.md +5 -69
  45. claude_mpm/commands/mpm-ticket-view.md +52 -495
  46. claude_mpm/commands/mpm-version.md +5 -107
  47. claude_mpm/config/agent_sources.py +27 -0
  48. claude_mpm/core/config.py +2 -4
  49. claude_mpm/core/framework/formatters/content_formatter.py +3 -13
  50. claude_mpm/core/framework/loaders/agent_loader.py +8 -5
  51. claude_mpm/core/framework/loaders/instruction_loader.py +52 -11
  52. claude_mpm/core/framework_loader.py +4 -2
  53. claude_mpm/core/logger.py +13 -0
  54. claude_mpm/core/optimized_startup.py +59 -0
  55. claude_mpm/core/shared/config_loader.py +1 -1
  56. claude_mpm/core/socketio_pool.py +3 -3
  57. claude_mpm/core/unified_agent_registry.py +5 -15
  58. claude_mpm/dashboard/static/svelte-build/_app/env.js +1 -0
  59. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.B_FtCwCQ.css +1 -0
  60. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.Cl_eSA4x.css +1 -0
  61. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BgChzWQ1.js +1 -0
  62. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIXEwuWe.js +1 -0
  63. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWc5urbQ.js +1 -0
  64. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DMkZpdF2.js +2 -0
  65. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DjhvlsAc.js +1 -0
  66. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/N4qtv3Hx.js +2 -0
  67. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/uj46x2Wr.js +1 -0
  68. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.DTL5mJO-.js +2 -0
  69. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.DzuEhzqh.js +1 -0
  70. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/0.CAGBuiOw.js +1 -0
  71. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DFLC8jdE.js +1 -0
  72. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.DPvEihJJ.js +10 -0
  73. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -0
  74. claude_mpm/dashboard/static/svelte-build/favicon.svg +7 -0
  75. claude_mpm/dashboard/static/svelte-build/index.html +36 -0
  76. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-311.pyc +0 -0
  77. claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-311.pyc +0 -0
  78. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
  79. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
  80. claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-311.pyc +0 -0
  81. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
  82. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
  83. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-311.pyc +0 -0
  84. claude_mpm/hooks/claude_hooks/correlation_manager.py +60 -0
  85. claude_mpm/hooks/claude_hooks/event_handlers.py +211 -78
  86. claude_mpm/hooks/claude_hooks/hook_handler.py +155 -1
  87. claude_mpm/hooks/claude_hooks/installer.py +33 -10
  88. claude_mpm/hooks/claude_hooks/memory_integration.py +26 -9
  89. claude_mpm/hooks/claude_hooks/response_tracking.py +2 -3
  90. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
  91. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager.cpython-311.pyc +0 -0
  92. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
  93. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-311.pyc +0 -0
  94. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
  95. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
  96. claude_mpm/hooks/claude_hooks/services/connection_manager.py +30 -6
  97. claude_mpm/hooks/kuzu_memory_hook.py +5 -5
  98. claude_mpm/hooks/memory_integration_hook.py +46 -1
  99. claude_mpm/init.py +63 -19
  100. claude_mpm/models/git_repository.py +3 -3
  101. claude_mpm/scripts/claude-hook-handler.sh +58 -18
  102. claude_mpm/scripts/launch_monitor.py +93 -13
  103. claude_mpm/services/agents/agent_builder.py +3 -3
  104. claude_mpm/services/agents/agent_recommendation_service.py +278 -0
  105. claude_mpm/services/agents/agent_review_service.py +280 -0
  106. claude_mpm/services/agents/cache_git_manager.py +6 -6
  107. claude_mpm/services/agents/deployment/agent_deployment.py +29 -7
  108. claude_mpm/services/agents/deployment/agent_discovery_service.py +4 -5
  109. claude_mpm/services/agents/deployment/agent_format_converter.py +23 -13
  110. claude_mpm/services/agents/deployment/agent_template_builder.py +32 -20
  111. claude_mpm/services/agents/deployment/agents_directory_resolver.py +2 -2
  112. claude_mpm/services/agents/deployment/async_agent_deployment.py +31 -27
  113. claude_mpm/services/agents/deployment/local_template_deployment.py +3 -1
  114. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +247 -35
  115. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +392 -87
  116. claude_mpm/services/agents/git_source_manager.py +53 -4
  117. claude_mpm/services/agents/loading/base_agent_manager.py +1 -13
  118. claude_mpm/services/agents/recommender.py +5 -3
  119. claude_mpm/services/agents/single_tier_deployment_service.py +2 -2
  120. claude_mpm/services/agents/sources/git_source_sync_service.py +120 -7
  121. claude_mpm/services/agents/startup_sync.py +22 -2
  122. claude_mpm/services/agents/toolchain_detector.py +10 -6
  123. claude_mpm/services/analysis/__init__.py +11 -1
  124. claude_mpm/services/analysis/clone_detector.py +1030 -0
  125. claude_mpm/services/command_deployment_service.py +81 -10
  126. claude_mpm/services/diagnostics/checks/agent_check.py +2 -2
  127. claude_mpm/services/diagnostics/checks/agent_sources_check.py +1 -1
  128. claude_mpm/services/event_bus/config.py +3 -1
  129. claude_mpm/services/git/git_operations_service.py +101 -16
  130. claude_mpm/services/monitor/daemon.py +9 -2
  131. claude_mpm/services/monitor/daemon_manager.py +39 -3
  132. claude_mpm/services/monitor/management/lifecycle.py +8 -1
  133. claude_mpm/services/monitor/server.py +698 -22
  134. claude_mpm/services/pm_skills_deployer.py +711 -0
  135. claude_mpm/services/profile_manager.py +331 -0
  136. claude_mpm/services/self_upgrade_service.py +120 -12
  137. claude_mpm/services/skills/__init__.py +3 -0
  138. claude_mpm/services/skills/git_skill_source_manager.py +130 -2
  139. claude_mpm/services/skills/selective_skill_deployer.py +704 -0
  140. claude_mpm/services/skills/skill_to_agent_mapper.py +406 -0
  141. claude_mpm/services/skills_deployer.py +127 -9
  142. claude_mpm/services/socketio/dashboard_server.py +1 -0
  143. claude_mpm/services/socketio/event_normalizer.py +51 -6
  144. claude_mpm/services/socketio/server/core.py +386 -108
  145. claude_mpm/services/version_control/git_operations.py +103 -0
  146. claude_mpm/skills/skill_manager.py +92 -3
  147. claude_mpm/utils/agent_dependency_loader.py +14 -2
  148. claude_mpm/utils/agent_filters.py +17 -44
  149. claude_mpm/utils/migration.py +4 -4
  150. claude_mpm/utils/robust_installer.py +47 -3
  151. {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.48.dist-info}/METADATA +53 -87
  152. {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.48.dist-info}/RECORD +157 -197
  153. claude_mpm-5.4.48.dist-info/entry_points.txt +5 -0
  154. claude_mpm-5.4.48.dist-info/licenses/LICENSE +94 -0
  155. claude_mpm-5.4.48.dist-info/licenses/LICENSE-FAQ.md +153 -0
  156. claude_mpm/agents/BASE_AGENT_TEMPLATE.md +0 -292
  157. claude_mpm/agents/BASE_DOCUMENTATION.md +0 -53
  158. claude_mpm/agents/BASE_OPS.md +0 -219
  159. claude_mpm/agents/BASE_PM.md +0 -480
  160. claude_mpm/agents/BASE_PROMPT_ENGINEER.md +0 -787
  161. claude_mpm/agents/BASE_QA.md +0 -167
  162. claude_mpm/agents/BASE_RESEARCH.md +0 -53
  163. claude_mpm/agents/base_agent_loader.py +0 -601
  164. claude_mpm/cli/commands/agents_detect.py +0 -380
  165. claude_mpm/cli/commands/agents_recommend.py +0 -309
  166. claude_mpm/cli/ticket_cli.py +0 -35
  167. claude_mpm/commands/mpm-agents-auto-configure.md +0 -278
  168. claude_mpm/commands/mpm-agents-detect.md +0 -177
  169. claude_mpm/commands/mpm-agents-list.md +0 -131
  170. claude_mpm/commands/mpm-agents-recommend.md +0 -223
  171. claude_mpm/commands/mpm-config-view.md +0 -150
  172. claude_mpm/commands/mpm-ticket-organize.md +0 -304
  173. claude_mpm/dashboard/analysis_runner.py +0 -455
  174. claude_mpm/dashboard/index.html +0 -13
  175. claude_mpm/dashboard/open_dashboard.py +0 -66
  176. claude_mpm/dashboard/static/css/activity.css +0 -1958
  177. claude_mpm/dashboard/static/css/connection-status.css +0 -370
  178. claude_mpm/dashboard/static/css/dashboard.css +0 -4701
  179. claude_mpm/dashboard/static/js/components/activity-tree.js +0 -1871
  180. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +0 -777
  181. claude_mpm/dashboard/static/js/components/agent-inference.js +0 -956
  182. claude_mpm/dashboard/static/js/components/build-tracker.js +0 -333
  183. claude_mpm/dashboard/static/js/components/code-simple.js +0 -857
  184. claude_mpm/dashboard/static/js/components/connection-debug.js +0 -654
  185. claude_mpm/dashboard/static/js/components/diff-viewer.js +0 -891
  186. claude_mpm/dashboard/static/js/components/event-processor.js +0 -542
  187. claude_mpm/dashboard/static/js/components/event-viewer.js +0 -1155
  188. claude_mpm/dashboard/static/js/components/export-manager.js +0 -368
  189. claude_mpm/dashboard/static/js/components/file-change-tracker.js +0 -443
  190. claude_mpm/dashboard/static/js/components/file-change-viewer.js +0 -690
  191. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +0 -724
  192. claude_mpm/dashboard/static/js/components/file-viewer.js +0 -580
  193. claude_mpm/dashboard/static/js/components/hud-library-loader.js +0 -211
  194. claude_mpm/dashboard/static/js/components/hud-manager.js +0 -671
  195. claude_mpm/dashboard/static/js/components/hud-visualizer.js +0 -1718
  196. claude_mpm/dashboard/static/js/components/module-viewer.js +0 -2764
  197. claude_mpm/dashboard/static/js/components/session-manager.js +0 -579
  198. claude_mpm/dashboard/static/js/components/socket-manager.js +0 -368
  199. claude_mpm/dashboard/static/js/components/ui-state-manager.js +0 -749
  200. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +0 -1824
  201. claude_mpm/dashboard/static/js/components/working-directory.js +0 -920
  202. claude_mpm/dashboard/static/js/connection-manager.js +0 -536
  203. claude_mpm/dashboard/static/js/dashboard.js +0 -1914
  204. claude_mpm/dashboard/static/js/extension-error-handler.js +0 -164
  205. claude_mpm/dashboard/static/js/socket-client.js +0 -1474
  206. claude_mpm/dashboard/static/js/tab-isolation-fix.js +0 -185
  207. claude_mpm/dashboard/static/socket.io.min.js +0 -7
  208. claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +0 -7
  209. claude_mpm/dashboard/templates/code_simple.html +0 -153
  210. claude_mpm/dashboard/templates/index.html +0 -606
  211. claude_mpm/dashboard/test_dashboard.html +0 -372
  212. claude_mpm/scripts/mcp_server.py +0 -75
  213. claude_mpm/scripts/mcp_wrapper.py +0 -39
  214. claude_mpm/services/mcp_gateway/__init__.py +0 -159
  215. claude_mpm/services/mcp_gateway/auto_configure.py +0 -369
  216. claude_mpm/services/mcp_gateway/config/__init__.py +0 -17
  217. claude_mpm/services/mcp_gateway/config/config_loader.py +0 -296
  218. claude_mpm/services/mcp_gateway/config/config_schema.py +0 -243
  219. claude_mpm/services/mcp_gateway/config/configuration.py +0 -429
  220. claude_mpm/services/mcp_gateway/core/__init__.py +0 -43
  221. claude_mpm/services/mcp_gateway/core/base.py +0 -312
  222. claude_mpm/services/mcp_gateway/core/exceptions.py +0 -253
  223. claude_mpm/services/mcp_gateway/core/interfaces.py +0 -443
  224. claude_mpm/services/mcp_gateway/core/process_pool.py +0 -977
  225. claude_mpm/services/mcp_gateway/core/singleton_manager.py +0 -315
  226. claude_mpm/services/mcp_gateway/core/startup_verification.py +0 -316
  227. claude_mpm/services/mcp_gateway/main.py +0 -589
  228. claude_mpm/services/mcp_gateway/registry/__init__.py +0 -12
  229. claude_mpm/services/mcp_gateway/registry/service_registry.py +0 -412
  230. claude_mpm/services/mcp_gateway/registry/tool_registry.py +0 -489
  231. claude_mpm/services/mcp_gateway/server/__init__.py +0 -15
  232. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +0 -414
  233. claude_mpm/services/mcp_gateway/server/stdio_handler.py +0 -372
  234. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -712
  235. claude_mpm/services/mcp_gateway/tools/__init__.py +0 -36
  236. claude_mpm/services/mcp_gateway/tools/base_adapter.py +0 -485
  237. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +0 -789
  238. claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +0 -654
  239. claude_mpm/services/mcp_gateway/tools/health_check_tool.py +0 -456
  240. claude_mpm/services/mcp_gateway/tools/hello_world.py +0 -551
  241. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +0 -555
  242. claude_mpm/services/mcp_gateway/utils/__init__.py +0 -14
  243. claude_mpm/services/mcp_gateway/utils/package_version_checker.py +0 -160
  244. claude_mpm/services/mcp_gateway/utils/update_preferences.py +0 -170
  245. claude_mpm-5.1.9.dist-info/entry_points.txt +0 -10
  246. claude_mpm-5.1.9.dist-info/licenses/LICENSE +0 -21
  247. {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.48.dist-info}/WHEEL +0 -0
  248. {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.48.dist-info}/top_level.txt +0 -0
@@ -1,777 +0,0 @@
1
- /**
2
- * Agent Hierarchy Component
3
- *
4
- * Displays agents in a hierarchical tree structure with PM at the top level.
5
- * Shows subagents as children of the PM that spawned them, tracking delegation
6
- * relationships from Task tool calls.
7
- *
8
- * WHY: Provides clear visualization of agent delegation relationships, making it
9
- * easier to understand the flow of work through PM and subagent delegations.
10
- *
11
- * DESIGN DECISION: Uses tree-based visualization with expand/collapse functionality
12
- * to handle complex delegation chains while maintaining performance with large
13
- * event streams. Separates hierarchy building from rendering for flexibility.
14
- */
15
-
16
- class AgentHierarchy {
17
- constructor(agentInference, eventViewer) {
18
- this.agentInference = agentInference;
19
- this.eventViewer = eventViewer;
20
-
21
- // Hierarchy state
22
- this.state = {
23
- // Tree structure with PM nodes at root
24
- hierarchyTree: null,
25
- // Map of agent ID to node for quick lookups
26
- nodeMap: new Map(),
27
- // Expanded state for tree nodes
28
- expandedNodes: new Set(),
29
- // Currently selected node
30
- selectedNode: null
31
- };
32
-
33
- // Default expand all nodes initially
34
- this.expandAll = true;
35
-
36
- // Set up event listeners
37
- this.setupEventListeners();
38
-
39
- console.log('Agent hierarchy component initialized');
40
- }
41
-
42
- /**
43
- * Set up event listeners for safe interaction
44
- */
45
- setupEventListeners() {
46
- // Use event delegation for toggle node clicks to avoid undefined dashboard errors
47
- document.addEventListener('click', (event) => {
48
- const toggleTarget = event.target.closest('[data-toggle-node]');
49
- if (toggleTarget && window.dashboard && window.dashboard.agentHierarchy) {
50
- const nodeId = toggleTarget.dataset.toggleNode;
51
- window.dashboard.agentHierarchy.toggleNode(nodeId);
52
- }
53
- });
54
- }
55
-
56
- /**
57
- * Build hierarchical structure from events
58
- * @returns {Object} Tree structure with PM at root
59
- */
60
- buildHierarchy() {
61
- // Process agent inference first
62
- this.agentInference.processAgentInference();
63
-
64
- // Get PM delegations and events
65
- const pmDelegations = this.agentInference.getPMDelegations();
66
- const events = this.eventViewer.events;
67
- const eventAgentMap = this.agentInference.getEventAgentMap();
68
-
69
- // Create root PM nodes
70
- const mainPM = {
71
- id: 'pm_main',
72
- type: 'pm',
73
- name: 'PM (Main Session)',
74
- children: [],
75
- events: [],
76
- eventCount: 0,
77
- status: 'active',
78
- startTime: null,
79
- endTime: null,
80
- expanded: true
81
- };
82
-
83
- // Map to store multiple implied PM groups
84
- const impliedPMGroups = new Map();
85
-
86
- // Clear node map
87
- this.state.nodeMap.clear();
88
- this.state.nodeMap.set(mainPM.id, mainPM);
89
-
90
- // Track which agents have been added
91
- const processedAgents = new Set();
92
-
93
- // Process explicit PM delegations
94
- for (const [delegationId, delegation] of pmDelegations) {
95
- const agentNode = {
96
- id: delegationId,
97
- type: 'subagent',
98
- name: delegation.agentName,
99
- delegationContext: this.extractDelegationContext(delegation.pmCall),
100
- children: [], // Subagents could theoretically delegate to others
101
- events: delegation.agentEvents,
102
- eventCount: delegation.agentEvents.length,
103
- status: delegation.endIndex ? 'completed' : 'active',
104
- startTime: delegation.timestamp,
105
- endTime: delegation.endIndex ? events[delegation.endIndex]?.timestamp : null,
106
- startIndex: delegation.startIndex,
107
- endIndex: delegation.endIndex,
108
- expanded: this.expandAll || this.state.expandedNodes.has(delegationId)
109
- };
110
-
111
- mainPM.children.push(agentNode);
112
- this.state.nodeMap.set(delegationId, agentNode);
113
- processedAgents.add(delegation.agentName);
114
-
115
- // Update main PM stats
116
- mainPM.eventCount++;
117
- if (!mainPM.startTime || new Date(delegation.timestamp) < new Date(mainPM.startTime)) {
118
- mainPM.startTime = delegation.timestamp;
119
- }
120
- if (delegation.endIndex && events[delegation.endIndex]) {
121
- const endTime = events[delegation.endIndex].timestamp;
122
- if (!mainPM.endTime || new Date(endTime) > new Date(mainPM.endTime)) {
123
- mainPM.endTime = endTime;
124
- }
125
- }
126
- }
127
-
128
- // Get orphan subagent groups from agent inference
129
- const orphanGroups = this.agentInference.getOrphanGroups();
130
-
131
- // Create implied PM nodes for each orphan group
132
- let impliedPMCounter = 1;
133
- for (const [groupingKey, orphans] of orphanGroups) {
134
- // Create an implied PM node for this group
135
- const impliedPM = {
136
- id: `pm_implied_${groupingKey}`,
137
- type: 'pm',
138
- name: `PM (Implied #${impliedPMCounter})`,
139
- children: [],
140
- events: [],
141
- eventCount: 0,
142
- status: 'inferred',
143
- startTime: null,
144
- endTime: null,
145
- expanded: true,
146
- isImplied: true,
147
- tooltip: 'Inferred PM - Subagents started without explicit PM delegation'
148
- };
149
-
150
- impliedPMGroups.set(groupingKey, impliedPM);
151
- this.state.nodeMap.set(impliedPM.id, impliedPM);
152
- impliedPMCounter++;
153
-
154
- // Group orphan events by agent name within this implied PM
155
- const agentEventGroups = new Map();
156
-
157
- for (const orphan of orphans) {
158
- // Find all events for this orphan agent
159
- const agentEvents = [];
160
- events.forEach((event, index) => {
161
- const inference = eventAgentMap.get(index);
162
- if (inference && inference.agentName === orphan.agentName) {
163
- // Check if this event is orphaned (not in any PM delegation)
164
- let isOrphan = true;
165
- for (const [_, delegation] of pmDelegations) {
166
- if (delegation.agentEvents.some(e => e.eventIndex === index)) {
167
- isOrphan = false;
168
- break;
169
- }
170
- }
171
-
172
- if (isOrphan) {
173
- agentEvents.push({
174
- eventIndex: index,
175
- event: event,
176
- inference: inference
177
- });
178
- }
179
- }
180
- });
181
-
182
- if (agentEvents.length > 0) {
183
- if (!agentEventGroups.has(orphan.agentName)) {
184
- agentEventGroups.set(orphan.agentName, []);
185
- }
186
- agentEventGroups.get(orphan.agentName).push(...agentEvents);
187
- }
188
- }
189
-
190
- // Create subagent nodes for each agent in this implied PM group
191
- for (const [agentName, agentEvents] of agentEventGroups) {
192
- if (agentEvents.length === 0) continue;
193
-
194
- const firstEvent = agentEvents[0].event;
195
- const lastEvent = agentEvents[agentEvents.length - 1].event;
196
-
197
- const agentNode = {
198
- id: `implied_agent_${groupingKey}_${agentName}`,
199
- type: 'subagent',
200
- name: agentName,
201
- delegationContext: 'Orphan agent - no explicit PM delegation found',
202
- children: [],
203
- events: agentEvents,
204
- eventCount: agentEvents.length,
205
- status: 'inferred',
206
- startTime: firstEvent.timestamp,
207
- endTime: lastEvent.timestamp,
208
- startIndex: agentEvents[0].eventIndex,
209
- endIndex: agentEvents[agentEvents.length - 1].eventIndex,
210
- expanded: this.expandAll,
211
- isImplied: true,
212
- tooltip: 'This agent was spawned without an explicit PM Task delegation'
213
- };
214
-
215
- impliedPM.children.push(agentNode);
216
- this.state.nodeMap.set(agentNode.id, agentNode);
217
-
218
- // Update implied PM stats
219
- impliedPM.eventCount += agentEvents.length;
220
- if (!impliedPM.startTime || new Date(firstEvent.timestamp) < new Date(impliedPM.startTime)) {
221
- impliedPM.startTime = firstEvent.timestamp;
222
- }
223
- if (!impliedPM.endTime || new Date(lastEvent.timestamp) > new Date(impliedPM.endTime)) {
224
- impliedPM.endTime = lastEvent.timestamp;
225
- }
226
- }
227
- }
228
-
229
- // Also find completely orphaned subagent events (not caught by SubagentStart)
230
- const uncategorizedOrphans = [];
231
- events.forEach((event, index) => {
232
- const inference = eventAgentMap.get(index);
233
- if (inference && inference.type === 'subagent') {
234
- // Check if this agent is already in a PM delegation or implied PM
235
- let isOrphan = true;
236
-
237
- // Check explicit delegations
238
- for (const [_, delegation] of pmDelegations) {
239
- if (delegation.agentEvents.some(e => e.eventIndex === index)) {
240
- isOrphan = false;
241
- break;
242
- }
243
- }
244
-
245
- // Check implied PMs
246
- if (isOrphan) {
247
- for (const [_, impliedPM] of impliedPMGroups) {
248
- for (const child of impliedPM.children) {
249
- if (child.events.some(e => e.eventIndex === index)) {
250
- isOrphan = false;
251
- break;
252
- }
253
- }
254
- if (!isOrphan) break;
255
- }
256
- }
257
-
258
- if (isOrphan) {
259
- uncategorizedOrphans.push({
260
- eventIndex: index,
261
- event: event,
262
- inference: inference
263
- });
264
- }
265
- }
266
- });
267
-
268
- // If there are uncategorized orphans, create a generic implied PM for them
269
- if (uncategorizedOrphans.length > 0) {
270
- const genericImpliedPM = {
271
- id: 'pm_implied_generic',
272
- type: 'pm',
273
- name: 'PM (Implied - Uncategorized)',
274
- children: [],
275
- events: [],
276
- eventCount: 0,
277
- status: 'inferred',
278
- startTime: null,
279
- endTime: null,
280
- expanded: true,
281
- isImplied: true,
282
- tooltip: 'Orphan agents without clear grouping'
283
- };
284
-
285
- // Group by agent name
286
- const agentGroups = new Map();
287
- for (const orphan of uncategorizedOrphans) {
288
- const agentName = orphan.inference.agentName;
289
- if (!agentGroups.has(agentName)) {
290
- agentGroups.set(agentName, []);
291
- }
292
- agentGroups.get(agentName).push(orphan);
293
- }
294
-
295
- // Create nodes for each agent
296
- for (const [agentName, agentEvents] of agentGroups) {
297
- const firstEvent = agentEvents[0].event;
298
- const lastEvent = agentEvents[agentEvents.length - 1].event;
299
-
300
- const agentNode = {
301
- id: `implied_generic_${agentName}`,
302
- type: 'subagent',
303
- name: agentName,
304
- delegationContext: 'Uncategorized orphan agent',
305
- children: [],
306
- events: agentEvents,
307
- eventCount: agentEvents.length,
308
- status: 'inferred',
309
- startTime: firstEvent.timestamp,
310
- endTime: lastEvent.timestamp,
311
- startIndex: agentEvents[0].eventIndex,
312
- endIndex: agentEvents[agentEvents.length - 1].eventIndex,
313
- expanded: this.expandAll,
314
- isImplied: true
315
- };
316
-
317
- genericImpliedPM.children.push(agentNode);
318
- this.state.nodeMap.set(agentNode.id, agentNode);
319
- genericImpliedPM.eventCount += agentEvents.length;
320
-
321
- if (!genericImpliedPM.startTime || new Date(firstEvent.timestamp) < new Date(genericImpliedPM.startTime)) {
322
- genericImpliedPM.startTime = firstEvent.timestamp;
323
- }
324
- if (!genericImpliedPM.endTime || new Date(lastEvent.timestamp) > new Date(genericImpliedPM.endTime)) {
325
- genericImpliedPM.endTime = lastEvent.timestamp;
326
- }
327
- }
328
-
329
- if (genericImpliedPM.children.length > 0) {
330
- impliedPMGroups.set('generic', genericImpliedPM);
331
- this.state.nodeMap.set(genericImpliedPM.id, genericImpliedPM);
332
- }
333
- }
334
-
335
- // Count PM's own events (not delegated)
336
- let pmOwnEvents = 0;
337
- events.forEach((event, index) => {
338
- const inference = eventAgentMap.get(index);
339
- if (inference && inference.type === 'main_agent') {
340
- pmOwnEvents++;
341
- mainPM.events.push({
342
- eventIndex: index,
343
- event: event,
344
- inference: inference
345
- });
346
- }
347
- });
348
- mainPM.eventCount += pmOwnEvents;
349
-
350
- // Update PM status based on children
351
- if (mainPM.children.length > 0) {
352
- const hasActive = mainPM.children.some(child => child.status === 'active');
353
- mainPM.status = hasActive ? 'active' : 'completed';
354
- }
355
-
356
- // Build final tree structure
357
- const tree = {
358
- roots: []
359
- };
360
-
361
- // Only add PMs that have content
362
- if (mainPM.eventCount > 0 || mainPM.children.length > 0) {
363
- tree.roots.push(mainPM);
364
- }
365
-
366
- // Add all implied PM groups that have content
367
- for (const [_, impliedPM] of impliedPMGroups) {
368
- if (impliedPM.children.length > 0) {
369
- tree.roots.push(impliedPM);
370
- }
371
- }
372
-
373
- this.state.hierarchyTree = tree;
374
-
375
- console.log('Hierarchy built:', {
376
- mainPM: {
377
- children: mainPM.children.length,
378
- events: mainPM.eventCount,
379
- ownEvents: pmOwnEvents
380
- },
381
- impliedPMGroups: impliedPMGroups.size,
382
- totalImpliedAgents: Array.from(impliedPMGroups.values())
383
- .reduce((sum, pm) => sum + pm.children.length, 0)
384
- });
385
-
386
- return tree;
387
- }
388
-
389
- /**
390
- * Extract delegation context from PM Task call
391
- * @param {Object} pmCall - The PM's Task tool call event
392
- * @returns {string} Description of what was delegated
393
- */
394
- extractDelegationContext(pmCall) {
395
- if (!pmCall) return 'Unknown delegation';
396
-
397
- // Try to extract task description from tool parameters
398
- const params = pmCall.tool_parameters || pmCall.data?.tool_parameters || {};
399
- const task = params.task || params.request || params.description;
400
-
401
- if (task) {
402
- // Truncate long tasks
403
- const maxLength = 100;
404
- if (task.length > maxLength) {
405
- return task.substring(0, maxLength) + '...';
406
- }
407
- return task;
408
- }
409
-
410
- // Fallback to tool input
411
- const toolInput = pmCall.tool_input || pmCall.data?.tool_input;
412
- if (toolInput && typeof toolInput === 'string') {
413
- const maxLength = 100;
414
- if (toolInput.length > maxLength) {
415
- return toolInput.substring(0, maxLength) + '...';
416
- }
417
- return toolInput;
418
- }
419
-
420
- return 'Task delegation';
421
- }
422
-
423
- /**
424
- * Render the hierarchy tree to HTML
425
- * @param {Object} filters - Optional filters for display
426
- * @returns {string} HTML string for the hierarchy
427
- */
428
- render(filters = {}) {
429
- const tree = this.state.hierarchyTree || this.buildHierarchy();
430
-
431
- if (!tree.roots || tree.roots.length === 0) {
432
- return '<div class="agent-hierarchy-empty">No agent activity detected</div>';
433
- }
434
-
435
- // Apply filters if provided
436
- const filteredTree = this.applyFilters(tree, filters);
437
-
438
- // Generate HTML
439
- const html = filteredTree.roots.map(root => this.renderNode(root, 0)).join('');
440
-
441
- return `<div class="agent-hierarchy">${html}</div>`;
442
- }
443
-
444
- /**
445
- * Render a single node and its children
446
- * @param {Object} node - Node to render
447
- * @param {number} level - Indentation level
448
- * @returns {string} HTML string for the node
449
- */
450
- renderNode(node, level) {
451
- const isExpanded = node.expanded || this.state.expandedNodes.has(node.id);
452
- const hasChildren = node.children && node.children.length > 0;
453
- const isSelected = this.state.selectedNode === node.id;
454
-
455
- // Icon based on node type and status
456
- const icon = this.getNodeIcon(node);
457
- const expandIcon = hasChildren ? (isExpanded ? '▼' : '▶') : '&nbsp;&nbsp;';
458
-
459
- // Status color
460
- const statusClass = this.getStatusClass(node.status);
461
-
462
- // Add special styling for implied nodes
463
- const impliedClass = node.isImplied ? 'agent-node-implied' : '';
464
- const tooltipAttr = node.tooltip ? `title="${this.escapeHtml(node.tooltip)}"` : '';
465
-
466
- // Build node HTML
467
- let html = `
468
- <div class="agent-node agent-node-level-${level} ${isSelected ? 'agent-node-selected' : ''} ${impliedClass}"
469
- data-node-id="${node.id}" ${tooltipAttr}>
470
- <div class="agent-node-header ${statusClass}"
471
- data-toggle-node="${node.id}" style="cursor: pointer">
472
- <span class="agent-node-expand">${expandIcon}</span>
473
- <span class="agent-node-icon">${icon}</span>
474
- <span class="agent-node-name">${this.escapeHtml(node.name)}</span>
475
- <span class="agent-node-stats">
476
- <span class="agent-event-count">${node.eventCount} events</span>
477
- ${node.status ? `<span class="agent-status">${node.status}</span>` : ''}
478
- </span>
479
- </div>
480
- `;
481
-
482
- // Add details if expanded
483
- if (isExpanded && (node.delegationContext || node.startTime)) {
484
- html += '<div class="agent-node-details">';
485
-
486
- if (node.delegationContext && node.delegationContext !== 'Unknown delegation') {
487
- html += `
488
- <div class="agent-delegation-context">
489
- <strong>Task:</strong> ${this.escapeHtml(node.delegationContext)}
490
- </div>
491
- `;
492
- }
493
-
494
- if (node.startTime) {
495
- const duration = this.calculateDuration(node.startTime, node.endTime);
496
- html += `
497
- <div class="agent-timing">
498
- <span class="agent-time-start">${this.formatTime(node.startTime)}</span>
499
- ${duration ? `<span class="agent-duration">(${duration})</span>` : ''}
500
- </div>
501
- `;
502
- }
503
-
504
- html += '</div>';
505
- }
506
-
507
- // Render children if expanded
508
- if (isExpanded && hasChildren) {
509
- html += '<div class="agent-node-children">';
510
- html += node.children.map(child => this.renderNode(child, level + 1)).join('');
511
- html += '</div>';
512
- }
513
-
514
- html += '</div>';
515
-
516
- return html;
517
- }
518
-
519
- /**
520
- * Get icon for node based on type and status
521
- * @param {Object} node - Node to get icon for
522
- * @returns {string} Icon HTML/emoji
523
- */
524
- getNodeIcon(node) {
525
- if (node.type === 'pm') {
526
- return node.isImplied ? '🔍' : '👔';
527
- }
528
-
529
- // Map agent names to icons
530
- const agentIcons = {
531
- 'Engineer Agent': '🔧',
532
- 'Research Agent': '🔍',
533
- 'QA Agent': '✅',
534
- 'Documentation Agent': '📝',
535
- 'Security Agent': '🔒',
536
- 'Ops Agent': '⚙️',
537
- 'Version Control Agent': '📦',
538
- 'Data Engineer Agent': '💾',
539
- 'Test Integration Agent': '🧪'
540
- };
541
-
542
- return agentIcons[node.name] || '🤖';
543
- }
544
-
545
- /**
546
- * Get status class for styling
547
- * @param {string} status - Node status
548
- * @returns {string} CSS class name
549
- */
550
- getStatusClass(status) {
551
- switch (status) {
552
- case 'active':
553
- return 'agent-status-active';
554
- case 'completed':
555
- return 'agent-status-completed';
556
- case 'pending':
557
- return 'agent-status-pending';
558
- case 'inferred':
559
- return 'agent-status-inferred';
560
- default:
561
- return 'agent-status-unknown';
562
- }
563
- }
564
-
565
- /**
566
- * Toggle node expansion
567
- * @param {string} nodeId - ID of node to toggle
568
- */
569
- toggleNode(nodeId) {
570
- const node = this.state.nodeMap.get(nodeId);
571
- if (!node) return;
572
-
573
- if (this.state.expandedNodes.has(nodeId)) {
574
- this.state.expandedNodes.delete(nodeId);
575
- node.expanded = false;
576
- } else {
577
- this.state.expandedNodes.add(nodeId);
578
- node.expanded = true;
579
- }
580
-
581
- // Trigger re-render
582
- if (window.dashboard) {
583
- window.dashboard.renderCurrentTab();
584
- }
585
- }
586
-
587
- /**
588
- * Select a node
589
- * @param {string} nodeId - ID of node to select
590
- */
591
- selectNode(nodeId) {
592
- this.state.selectedNode = nodeId;
593
- const node = this.state.nodeMap.get(nodeId);
594
-
595
- if (node) {
596
- // Dispatch event for other components to react
597
- const event = new CustomEvent('agentNodeSelected', {
598
- detail: { node: node }
599
- });
600
- document.dispatchEvent(event);
601
- }
602
- }
603
-
604
- /**
605
- * Apply filters to the tree
606
- * @param {Object} tree - Tree to filter
607
- * @param {Object} filters - Filter criteria
608
- * @returns {Object} Filtered tree
609
- */
610
- applyFilters(tree, filters) {
611
- if (!filters || Object.keys(filters).length === 0) {
612
- return tree;
613
- }
614
-
615
- // Clone tree structure for filtering
616
- const filteredTree = {
617
- roots: []
618
- };
619
-
620
- for (const root of tree.roots) {
621
- const filteredRoot = this.filterNode(root, filters);
622
- if (filteredRoot) {
623
- filteredTree.roots.push(filteredRoot);
624
- }
625
- }
626
-
627
- return filteredTree;
628
- }
629
-
630
- /**
631
- * Filter a single node and its children
632
- * @param {Object} node - Node to filter
633
- * @param {Object} filters - Filter criteria
634
- * @returns {Object|null} Filtered node or null if filtered out
635
- */
636
- filterNode(node, filters) {
637
- // Check if node matches filters
638
- let matches = true;
639
-
640
- if (filters.searchText) {
641
- const searchLower = filters.searchText.toLowerCase();
642
- matches = matches && (
643
- node.name.toLowerCase().includes(searchLower) ||
644
- (node.delegationContext && node.delegationContext.toLowerCase().includes(searchLower))
645
- );
646
- }
647
-
648
- if (filters.agentType) {
649
- matches = matches && node.name.includes(filters.agentType);
650
- }
651
-
652
- if (filters.status) {
653
- matches = matches && node.status === filters.status;
654
- }
655
-
656
- // Filter children recursively
657
- let filteredChildren = [];
658
- if (node.children) {
659
- for (const child of node.children) {
660
- const filteredChild = this.filterNode(child, filters);
661
- if (filteredChild) {
662
- filteredChildren.push(filteredChild);
663
- }
664
- }
665
- }
666
-
667
- // Include node if it matches or has matching children
668
- if (matches || filteredChildren.length > 0) {
669
- return {
670
- ...node,
671
- children: filteredChildren
672
- };
673
- }
674
-
675
- return null;
676
- }
677
-
678
- /**
679
- * Format timestamp for display
680
- * @param {string} timestamp - ISO timestamp
681
- * @returns {string} Formatted time
682
- */
683
- formatTime(timestamp) {
684
- if (!timestamp) return '';
685
- const date = new Date(timestamp);
686
- return date.toLocaleTimeString('en-US', {
687
- hour: '2-digit',
688
- minute: '2-digit',
689
- second: '2-digit',
690
- hour12: false
691
- });
692
- }
693
-
694
- /**
695
- * Calculate duration between timestamps
696
- * @param {string} start - Start timestamp
697
- * @param {string} end - End timestamp
698
- * @returns {string} Formatted duration
699
- */
700
- calculateDuration(start, end) {
701
- if (!start || !end) return '';
702
-
703
- const startTime = new Date(start).getTime();
704
- const endTime = new Date(end).getTime();
705
- const duration = endTime - startTime;
706
-
707
- if (duration < 1000) {
708
- return `${duration}ms`;
709
- } else if (duration < 60000) {
710
- return `${(duration / 1000).toFixed(1)}s`;
711
- } else {
712
- const minutes = Math.floor(duration / 60000);
713
- const seconds = Math.floor((duration % 60000) / 1000);
714
- return `${minutes}m ${seconds}s`;
715
- }
716
- }
717
-
718
- /**
719
- * Escape HTML for safe rendering
720
- * @param {string} text - Text to escape
721
- * @returns {string} Escaped text
722
- */
723
- escapeHtml(text) {
724
- if (!text) return '';
725
- const div = document.createElement('div');
726
- div.textContent = text;
727
- return div.innerHTML;
728
- }
729
-
730
- /**
731
- * Update hierarchy when new events arrive
732
- * @param {Array} events - New events
733
- */
734
- updateWithNewEvents(events) {
735
- // Rebuild hierarchy with new events
736
- this.buildHierarchy();
737
- }
738
-
739
- /**
740
- * Clear the hierarchy
741
- */
742
- clear() {
743
- this.state.hierarchyTree = null;
744
- this.state.nodeMap.clear();
745
- this.state.expandedNodes.clear();
746
- this.state.selectedNode = null;
747
- }
748
-
749
- /**
750
- * Expand all nodes
751
- */
752
- expandAllNodes() {
753
- for (const [nodeId, node] of this.state.nodeMap) {
754
- this.state.expandedNodes.add(nodeId);
755
- node.expanded = true;
756
- }
757
- this.expandAll = true;
758
- }
759
-
760
- /**
761
- * Collapse all nodes
762
- */
763
- collapseAllNodes() {
764
- this.state.expandedNodes.clear();
765
- for (const [nodeId, node] of this.state.nodeMap) {
766
- node.expanded = false;
767
- }
768
- this.expandAll = false;
769
- }
770
- }
771
-
772
- // ES6 Module export
773
- export { AgentHierarchy };
774
- export default AgentHierarchy;
775
-
776
- // Make AgentHierarchy globally available for dist/dashboard.js
777
- window.AgentHierarchy = AgentHierarchy;