claude-mpm 5.0.2__py3-none-any.whl → 5.4.3__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 (184) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +2002 -0
  3. claude_mpm/agents/PM_INSTRUCTIONS.md +1218 -905
  4. claude_mpm/agents/agent_loader.py +10 -17
  5. claude_mpm/agents/base_agent_loader.py +10 -35
  6. claude_mpm/agents/frontmatter_validator.py +68 -0
  7. claude_mpm/agents/templates/circuit-breakers.md +431 -45
  8. claude_mpm/cli/__init__.py +0 -1
  9. claude_mpm/cli/commands/__init__.py +2 -0
  10. claude_mpm/cli/commands/agent_state_manager.py +67 -23
  11. claude_mpm/cli/commands/agents.py +446 -25
  12. claude_mpm/cli/commands/auto_configure.py +535 -233
  13. claude_mpm/cli/commands/configure.py +1500 -147
  14. claude_mpm/cli/commands/configure_agent_display.py +13 -6
  15. claude_mpm/cli/commands/mpm_init/core.py +158 -1
  16. claude_mpm/cli/commands/mpm_init/knowledge_extractor.py +481 -0
  17. claude_mpm/cli/commands/mpm_init/prompts.py +280 -0
  18. claude_mpm/cli/commands/postmortem.py +401 -0
  19. claude_mpm/cli/commands/run.py +1 -39
  20. claude_mpm/cli/commands/skills.py +322 -19
  21. claude_mpm/cli/commands/summarize.py +413 -0
  22. claude_mpm/cli/executor.py +8 -0
  23. claude_mpm/cli/interactive/agent_wizard.py +302 -195
  24. claude_mpm/cli/parsers/agents_parser.py +137 -0
  25. claude_mpm/cli/parsers/auto_configure_parser.py +13 -0
  26. claude_mpm/cli/parsers/base_parser.py +9 -0
  27. claude_mpm/cli/parsers/skills_parser.py +7 -0
  28. claude_mpm/cli/startup.py +133 -85
  29. claude_mpm/commands/mpm-agents-auto-configure.md +2 -2
  30. claude_mpm/commands/mpm-agents-list.md +2 -2
  31. claude_mpm/commands/mpm-config-view.md +2 -2
  32. claude_mpm/commands/mpm-help.md +3 -0
  33. claude_mpm/commands/{mpm-ticket-organize.md → mpm-organize.md} +4 -5
  34. claude_mpm/commands/mpm-postmortem.md +123 -0
  35. claude_mpm/commands/mpm-session-resume.md +2 -2
  36. claude_mpm/commands/mpm-ticket-view.md +2 -2
  37. claude_mpm/config/agent_presets.py +312 -82
  38. claude_mpm/config/agent_sources.py +27 -0
  39. claude_mpm/config/skill_presets.py +392 -0
  40. claude_mpm/constants.py +1 -0
  41. claude_mpm/core/claude_runner.py +2 -25
  42. claude_mpm/core/framework/loaders/agent_loader.py +8 -5
  43. claude_mpm/core/framework/loaders/file_loader.py +54 -101
  44. claude_mpm/core/interactive_session.py +19 -5
  45. claude_mpm/core/oneshot_session.py +16 -4
  46. claude_mpm/core/output_style_manager.py +173 -43
  47. claude_mpm/core/protocols/__init__.py +23 -0
  48. claude_mpm/core/protocols/runner_protocol.py +103 -0
  49. claude_mpm/core/protocols/session_protocol.py +131 -0
  50. claude_mpm/core/shared/singleton_manager.py +11 -4
  51. claude_mpm/core/socketio_pool.py +3 -3
  52. claude_mpm/core/system_context.py +38 -0
  53. claude_mpm/core/unified_agent_registry.py +134 -16
  54. claude_mpm/core/unified_config.py +22 -0
  55. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-313.pyc +0 -0
  56. claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-313.pyc +0 -0
  57. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-313.pyc +0 -0
  58. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-313.pyc +0 -0
  59. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-313.pyc +0 -0
  60. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-313.pyc +0 -0
  61. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-313.pyc +0 -0
  62. claude_mpm/hooks/claude_hooks/correlation_manager.py +60 -0
  63. claude_mpm/hooks/claude_hooks/event_handlers.py +35 -2
  64. claude_mpm/hooks/claude_hooks/hook_handler.py +4 -0
  65. claude_mpm/hooks/claude_hooks/memory_integration.py +12 -1
  66. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-313.pyc +0 -0
  67. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-313.pyc +0 -0
  68. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-313.pyc +0 -0
  69. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-313.pyc +0 -0
  70. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-313.pyc +0 -0
  71. claude_mpm/hooks/claude_hooks/services/connection_manager.py +4 -0
  72. claude_mpm/models/agent_definition.py +7 -0
  73. claude_mpm/scripts/launch_monitor.py +93 -13
  74. claude_mpm/services/agents/agent_recommendation_service.py +279 -0
  75. claude_mpm/services/agents/cache_git_manager.py +621 -0
  76. claude_mpm/services/agents/deployment/agent_template_builder.py +3 -2
  77. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +110 -3
  78. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +518 -55
  79. claude_mpm/services/agents/git_source_manager.py +20 -0
  80. claude_mpm/services/agents/sources/git_source_sync_service.py +45 -6
  81. claude_mpm/services/agents/toolchain_detector.py +6 -5
  82. claude_mpm/services/analysis/__init__.py +35 -0
  83. claude_mpm/services/analysis/clone_detector.py +1030 -0
  84. claude_mpm/services/analysis/postmortem_reporter.py +474 -0
  85. claude_mpm/services/analysis/postmortem_service.py +765 -0
  86. claude_mpm/services/command_deployment_service.py +106 -5
  87. claude_mpm/services/core/base.py +7 -2
  88. claude_mpm/services/diagnostics/checks/mcp_services_check.py +7 -15
  89. claude_mpm/services/event_bus/config.py +3 -1
  90. claude_mpm/services/git/git_operations_service.py +8 -8
  91. claude_mpm/services/mcp_config_manager.py +75 -145
  92. claude_mpm/services/mcp_service_verifier.py +6 -3
  93. claude_mpm/services/monitor/daemon.py +37 -10
  94. claude_mpm/services/monitor/daemon_manager.py +134 -21
  95. claude_mpm/services/monitor/server.py +225 -19
  96. claude_mpm/services/project/project_organizer.py +4 -0
  97. claude_mpm/services/runner_configuration_service.py +16 -3
  98. claude_mpm/services/session_management_service.py +16 -4
  99. claude_mpm/services/socketio/event_normalizer.py +15 -1
  100. claude_mpm/services/socketio/server/core.py +160 -21
  101. claude_mpm/services/version_control/git_operations.py +103 -0
  102. claude_mpm/utils/agent_filters.py +261 -0
  103. claude_mpm/utils/gitignore.py +3 -0
  104. claude_mpm/utils/migration.py +372 -0
  105. claude_mpm/utils/progress.py +5 -1
  106. {claude_mpm-5.0.2.dist-info → claude_mpm-5.4.3.dist-info}/METADATA +69 -84
  107. {claude_mpm-5.0.2.dist-info → claude_mpm-5.4.3.dist-info}/RECORD +112 -153
  108. {claude_mpm-5.0.2.dist-info → claude_mpm-5.4.3.dist-info}/entry_points.txt +0 -2
  109. claude_mpm/dashboard/analysis_runner.py +0 -455
  110. claude_mpm/dashboard/index.html +0 -13
  111. claude_mpm/dashboard/open_dashboard.py +0 -66
  112. claude_mpm/dashboard/static/css/activity.css +0 -1958
  113. claude_mpm/dashboard/static/css/connection-status.css +0 -370
  114. claude_mpm/dashboard/static/css/dashboard.css +0 -4701
  115. claude_mpm/dashboard/static/js/components/activity-tree.js +0 -1871
  116. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +0 -777
  117. claude_mpm/dashboard/static/js/components/agent-inference.js +0 -956
  118. claude_mpm/dashboard/static/js/components/build-tracker.js +0 -333
  119. claude_mpm/dashboard/static/js/components/code-simple.js +0 -857
  120. claude_mpm/dashboard/static/js/components/connection-debug.js +0 -654
  121. claude_mpm/dashboard/static/js/components/diff-viewer.js +0 -891
  122. claude_mpm/dashboard/static/js/components/event-processor.js +0 -542
  123. claude_mpm/dashboard/static/js/components/event-viewer.js +0 -1155
  124. claude_mpm/dashboard/static/js/components/export-manager.js +0 -368
  125. claude_mpm/dashboard/static/js/components/file-change-tracker.js +0 -443
  126. claude_mpm/dashboard/static/js/components/file-change-viewer.js +0 -690
  127. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +0 -724
  128. claude_mpm/dashboard/static/js/components/file-viewer.js +0 -580
  129. claude_mpm/dashboard/static/js/components/hud-library-loader.js +0 -211
  130. claude_mpm/dashboard/static/js/components/hud-manager.js +0 -671
  131. claude_mpm/dashboard/static/js/components/hud-visualizer.js +0 -1718
  132. claude_mpm/dashboard/static/js/components/module-viewer.js +0 -2764
  133. claude_mpm/dashboard/static/js/components/session-manager.js +0 -579
  134. claude_mpm/dashboard/static/js/components/socket-manager.js +0 -368
  135. claude_mpm/dashboard/static/js/components/ui-state-manager.js +0 -749
  136. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +0 -1824
  137. claude_mpm/dashboard/static/js/components/working-directory.js +0 -920
  138. claude_mpm/dashboard/static/js/connection-manager.js +0 -536
  139. claude_mpm/dashboard/static/js/dashboard.js +0 -1914
  140. claude_mpm/dashboard/static/js/extension-error-handler.js +0 -164
  141. claude_mpm/dashboard/static/js/socket-client.js +0 -1474
  142. claude_mpm/dashboard/static/js/tab-isolation-fix.js +0 -185
  143. claude_mpm/dashboard/static/socket.io.min.js +0 -7
  144. claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +0 -7
  145. claude_mpm/dashboard/templates/code_simple.html +0 -153
  146. claude_mpm/dashboard/templates/index.html +0 -606
  147. claude_mpm/dashboard/test_dashboard.html +0 -372
  148. claude_mpm/scripts/mcp_server.py +0 -75
  149. claude_mpm/scripts/mcp_wrapper.py +0 -39
  150. claude_mpm/services/mcp_gateway/__init__.py +0 -159
  151. claude_mpm/services/mcp_gateway/auto_configure.py +0 -369
  152. claude_mpm/services/mcp_gateway/config/__init__.py +0 -17
  153. claude_mpm/services/mcp_gateway/config/config_loader.py +0 -296
  154. claude_mpm/services/mcp_gateway/config/config_schema.py +0 -243
  155. claude_mpm/services/mcp_gateway/config/configuration.py +0 -429
  156. claude_mpm/services/mcp_gateway/core/__init__.py +0 -43
  157. claude_mpm/services/mcp_gateway/core/base.py +0 -312
  158. claude_mpm/services/mcp_gateway/core/exceptions.py +0 -253
  159. claude_mpm/services/mcp_gateway/core/interfaces.py +0 -443
  160. claude_mpm/services/mcp_gateway/core/process_pool.py +0 -971
  161. claude_mpm/services/mcp_gateway/core/singleton_manager.py +0 -315
  162. claude_mpm/services/mcp_gateway/core/startup_verification.py +0 -316
  163. claude_mpm/services/mcp_gateway/main.py +0 -589
  164. claude_mpm/services/mcp_gateway/registry/__init__.py +0 -12
  165. claude_mpm/services/mcp_gateway/registry/service_registry.py +0 -412
  166. claude_mpm/services/mcp_gateway/registry/tool_registry.py +0 -489
  167. claude_mpm/services/mcp_gateway/server/__init__.py +0 -15
  168. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +0 -414
  169. claude_mpm/services/mcp_gateway/server/stdio_handler.py +0 -372
  170. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -712
  171. claude_mpm/services/mcp_gateway/tools/__init__.py +0 -36
  172. claude_mpm/services/mcp_gateway/tools/base_adapter.py +0 -485
  173. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +0 -789
  174. claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +0 -654
  175. claude_mpm/services/mcp_gateway/tools/health_check_tool.py +0 -456
  176. claude_mpm/services/mcp_gateway/tools/hello_world.py +0 -551
  177. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +0 -555
  178. claude_mpm/services/mcp_gateway/utils/__init__.py +0 -14
  179. claude_mpm/services/mcp_gateway/utils/package_version_checker.py +0 -160
  180. claude_mpm/services/mcp_gateway/utils/update_preferences.py +0 -170
  181. /claude_mpm/agents/{OUTPUT_STYLE.md → CLAUDE_MPM_OUTPUT_STYLE.md} +0 -0
  182. {claude_mpm-5.0.2.dist-info → claude_mpm-5.4.3.dist-info}/WHEEL +0 -0
  183. {claude_mpm-5.0.2.dist-info → claude_mpm-5.4.3.dist-info}/licenses/LICENSE +0 -0
  184. {claude_mpm-5.0.2.dist-info → claude_mpm-5.4.3.dist-info}/top_level.txt +0 -0
@@ -1,1155 +0,0 @@
1
- /**
2
- * Event Viewer Component
3
- * Handles event display, filtering, and selection
4
- */
5
-
6
- class EventViewer {
7
- constructor(containerId, socketClient) {
8
- this.container = document.getElementById(containerId);
9
- this.socketClient = socketClient;
10
-
11
- // State
12
- this.events = [];
13
- this.filteredEvents = [];
14
- this.selectedEventIndex = -1;
15
- this.filteredEventElements = [];
16
- this.autoScroll = true;
17
-
18
- // Filters
19
- this.searchFilter = '';
20
- this.typeFilter = '';
21
- this.sessionFilter = '';
22
-
23
- // Event type tracking
24
- this.eventTypeCount = {};
25
- this.availableEventTypes = new Set();
26
- this.errorCount = 0;
27
- this.eventsThisMinute = 0;
28
- this.lastMinute = new Date().getMinutes();
29
-
30
- this.init();
31
- }
32
-
33
- /**
34
- * Initialize the event viewer
35
- */
36
- init() {
37
- this.setupEventHandlers();
38
- this.setupKeyboardNavigation();
39
-
40
- // Subscribe to socket events
41
- this.socketClient.onEventUpdate((events, sessions) => {
42
- // Ensure we always have a valid events array
43
- this.events = Array.isArray(events) ? events : [];
44
- console.log('[EventViewer] Events updated - received:', this.events.length);
45
-
46
- // Update debug metrics
47
- const debugReceivedEl = document.getElementById('debug-events-received');
48
- if (debugReceivedEl) {
49
- debugReceivedEl.textContent = this.events.length;
50
- }
51
-
52
- this.updateDisplay();
53
- });
54
- }
55
-
56
- /**
57
- * Setup event handlers for UI controls
58
- */
59
- setupEventHandlers() {
60
- // Search input
61
- const searchInput = document.getElementById('events-search-input');
62
- if (searchInput) {
63
- searchInput.addEventListener('input', (e) => {
64
- this.searchFilter = e.target.value.toLowerCase();
65
- this.applyFilters();
66
- });
67
- }
68
-
69
- // Type filter
70
- const typeFilter = document.getElementById('events-type-filter');
71
- if (typeFilter) {
72
- typeFilter.addEventListener('change', (e) => {
73
- this.typeFilter = e.target.value;
74
- this.applyFilters();
75
- });
76
- }
77
- }
78
-
79
- /**
80
- * Setup keyboard navigation for events
81
- * Note: This is now handled by the unified Dashboard navigation system
82
- */
83
- setupKeyboardNavigation() {
84
- // Keyboard navigation is now handled by Dashboard.setupUnifiedKeyboardNavigation()
85
- // This method is kept for backward compatibility but does nothing
86
- }
87
-
88
- /**
89
- * Handle arrow key navigation
90
- * @param {number} direction - Direction: 1 for down, -1 for up
91
- */
92
- handleArrowNavigation(direction) {
93
- if (this.filteredEventElements.length === 0) return;
94
-
95
- // Calculate new index
96
- let newIndex = this.selectedEventIndex + direction;
97
-
98
- // Wrap around
99
- if (newIndex >= this.filteredEventElements.length) {
100
- newIndex = 0;
101
- } else if (newIndex < 0) {
102
- newIndex = this.filteredEventElements.length - 1;
103
- }
104
-
105
- this.showEventDetails(newIndex);
106
- }
107
-
108
- /**
109
- * Apply filters to events
110
- */
111
- applyFilters() {
112
- // Defensive check to ensure events array exists
113
- if (!this.events || !Array.isArray(this.events)) {
114
- console.warn('EventViewer: events array is not initialized, using empty array');
115
- this.events = [];
116
- }
117
-
118
- this.filteredEvents = this.events.filter(event => {
119
- // NO AUTOMATIC FILTERING - All events are shown by default for complete visibility
120
- // Users can apply their own filters using the search and type filter controls
121
-
122
- // User-controlled search filter
123
- if (this.searchFilter) {
124
- const searchableText = [
125
- event.type || '',
126
- event.subtype || '',
127
- JSON.stringify(event.data || {})
128
- ].join(' ').toLowerCase();
129
-
130
- if (!searchableText.includes(this.searchFilter)) {
131
- return false;
132
- }
133
- }
134
-
135
- // User-controlled type filter - handles full hook types (like "hook.user_prompt") and main types
136
- if (this.typeFilter) {
137
- // Use the same logic as formatEventType to get the full event type
138
- const eventType = event.type && event.type.trim() !== '' ? event.type : '';
139
- const fullEventType = event.subtype && eventType ? `${eventType}.${event.subtype}` : eventType;
140
- if (fullEventType !== this.typeFilter) {
141
- return false;
142
- }
143
- }
144
-
145
- // User-controlled session filter
146
- if (this.sessionFilter && this.sessionFilter !== '') {
147
- if (!event.data || event.data.session_id !== this.sessionFilter) {
148
- return false;
149
- }
150
- }
151
-
152
- // Allow all events through unless filtered by user controls
153
- return true;
154
- });
155
-
156
- this.renderEvents();
157
- this.updateMetrics();
158
- }
159
-
160
- /**
161
- * Update available event types and populate dropdown
162
- */
163
- updateEventTypeDropdown() {
164
- const dropdown = document.getElementById('events-type-filter');
165
- if (!dropdown) return;
166
-
167
- // Extract unique event types from current events
168
- // Use the same logic as formatEventType to get full event type names
169
- const eventTypes = new Set();
170
- // Defensive check to ensure events array exists
171
- if (!this.events || !Array.isArray(this.events)) {
172
- console.warn('EventViewer: events array is not initialized in updateEventTypeDropdown');
173
- this.events = [];
174
- }
175
-
176
- this.events.forEach(event => {
177
- if (event.type && event.type.trim() !== '') {
178
- // Combine type and subtype if subtype exists, otherwise just use type
179
- const fullType = event.subtype ? `${event.type}.${event.subtype}` : event.type;
180
- eventTypes.add(fullType);
181
- }
182
- });
183
-
184
- // Check if event types have changed
185
- const currentTypes = Array.from(eventTypes).sort();
186
- const previousTypes = Array.from(this.availableEventTypes).sort();
187
-
188
- if (JSON.stringify(currentTypes) === JSON.stringify(previousTypes)) {
189
- return; // No change needed
190
- }
191
-
192
- // Update our tracking
193
- this.availableEventTypes = eventTypes;
194
-
195
- // Store the current selection
196
- const currentSelection = dropdown.value;
197
-
198
- // Clear existing options except "All Events"
199
- dropdown.innerHTML = '<option value="">All Events</option>';
200
-
201
- // Add new options sorted alphabetically
202
- const sortedTypes = Array.from(eventTypes).sort();
203
- sortedTypes.forEach(type => {
204
- const option = document.createElement('option');
205
- option.value = type;
206
- option.textContent = type;
207
- dropdown.appendChild(option);
208
- });
209
-
210
- // Restore selection if it still exists
211
- if (currentSelection && eventTypes.has(currentSelection)) {
212
- dropdown.value = currentSelection;
213
- } else if (currentSelection && !eventTypes.has(currentSelection)) {
214
- // If the previously selected type no longer exists, clear the filter
215
- dropdown.value = '';
216
- this.typeFilter = '';
217
- }
218
- }
219
-
220
- /**
221
- * Update the display with current events
222
- */
223
- updateDisplay() {
224
- this.updateEventTypeDropdown();
225
- this.applyFilters();
226
- }
227
-
228
- /**
229
- * Render events in the UI
230
- */
231
- renderEvents() {
232
- // CRITICAL FIX: Use the container passed to constructor, not hardcoded events-list
233
- // This prevents events from being rendered in the wrong tab
234
- const eventsList = this.container;
235
- if (!eventsList) {
236
- console.warn('[EventViewer] Container not found, skipping render');
237
- return;
238
- }
239
-
240
- // SAFETY: Basic check to ensure we're rendering to the correct container
241
- if (eventsList.id !== 'events-list') {
242
- console.error('[EventViewer] CRITICAL: Attempting to render to wrong container:', eventsList.id);
243
- return;
244
- }
245
-
246
- // Check if events tab exists and render regardless of active state
247
- // This allows events to be pre-rendered when tab becomes active
248
- const eventsTab = document.getElementById('events-tab');
249
- if (!eventsTab) {
250
- console.warn('[EventViewer] Events tab not found in DOM');
251
- return;
252
- }
253
-
254
- console.log('[EventViewer] Rendering events - count:', this.filteredEvents.length);
255
-
256
- // Check if user is at bottom BEFORE rendering (for autoscroll decision)
257
- const wasAtBottom = (eventsList.scrollTop + eventsList.clientHeight >= eventsList.scrollHeight - 10);
258
-
259
- if (this.filteredEvents.length === 0) {
260
- eventsList.innerHTML = `
261
- <div class="no-events">
262
- ${this.events.length === 0 ?
263
- 'Connect to Socket.IO server to see events...' :
264
- 'No events match current filters...'}
265
- </div>
266
- `;
267
- this.filteredEventElements = [];
268
- return;
269
- }
270
-
271
- const html = this.filteredEvents.map((event, index) => {
272
- const timestamp = new Date(event.timestamp).toLocaleTimeString();
273
- const eventClass = event.type ? `event-${event.type}` : 'event-default';
274
- const isSelected = index === this.selectedEventIndex;
275
-
276
- // Get main content and timestamp separately
277
- const mainContent = this.formatSingleRowEventContent(event);
278
-
279
- // Check if this is an Edit/MultiEdit tool event and add diff viewer
280
- const diffViewer = this.createInlineEditDiffViewer(event, index);
281
-
282
- return `
283
- <div class="event-item single-row ${eventClass} ${isSelected ? 'selected' : ''}"
284
- onclick="eventViewer.showEventDetails(${index})"
285
- data-index="${index}">
286
- <span class="event-single-row-content">
287
- <span class="event-content-main">${mainContent}</span>
288
- <span class="event-timestamp">${timestamp}</span>
289
- </span>
290
- ${diffViewer}
291
- </div>
292
- `;
293
- }).join('');
294
-
295
- eventsList.innerHTML = html;
296
-
297
- // Update filtered elements reference
298
- this.filteredEventElements = Array.from(eventsList.querySelectorAll('.event-item'));
299
-
300
- console.log('[EventViewer] Events rendered - filtered:', this.filteredEvents.length, 'elements:', this.filteredEventElements.length);
301
-
302
- // Update debug metrics
303
- const debugRenderedEl = document.getElementById('debug-events-rendered');
304
- if (debugRenderedEl) {
305
- debugRenderedEl.textContent = this.filteredEvents.length;
306
- }
307
-
308
- // Update Dashboard navigation items if we're in the events tab
309
- if (window.dashboard && window.dashboard.currentTab === 'events' &&
310
- window.dashboard.tabNavigation && window.dashboard.tabNavigation.events) {
311
- window.dashboard.tabNavigation.events.items = this.filteredEventElements;
312
- }
313
-
314
- // Auto-scroll only if user was already at bottom before rendering
315
- if (this.filteredEvents.length > 0 && wasAtBottom && this.autoScroll) {
316
- // Use requestAnimationFrame to ensure DOM has updated
317
- requestAnimationFrame(() => {
318
- eventsList.scrollTop = eventsList.scrollHeight;
319
- });
320
- }
321
- }
322
-
323
- /**
324
- * Format event type for display
325
- * @param {Object} event - Event object
326
- * @returns {string} Formatted event type
327
- */
328
- formatEventType(event) {
329
- // If we have type and subtype, use them
330
- if (event.type && event.subtype) {
331
- // Check if type and subtype are identical or subtype is 'generic' to prevent redundant display
332
- if (event.type === event.subtype || event.subtype === 'generic') {
333
- return event.type;
334
- }
335
- return `${event.type}.${event.subtype}`;
336
- }
337
- // If we have just type, use it
338
- if (event.type) {
339
- return event.type;
340
- }
341
- // If we have originalEventName (from transformation), use it as fallback
342
- if (event.originalEventName) {
343
- return event.originalEventName;
344
- }
345
- // Last resort fallback
346
- return 'unknown';
347
- }
348
-
349
- /**
350
- * Format event data for display
351
- * @param {Object} event - Event object
352
- * @returns {string} Formatted event data
353
- */
354
- formatEventData(event) {
355
- if (!event.data) return 'No data';
356
-
357
- // Special formatting for different event types
358
- switch (event.type) {
359
- case 'session':
360
- return this.formatSessionEvent(event);
361
- case 'claude':
362
- return this.formatClaudeEvent(event);
363
- case 'agent':
364
- return this.formatAgentEvent(event);
365
- case 'hook':
366
- return this.formatHookEvent(event);
367
- case 'todo':
368
- return this.formatTodoEvent(event);
369
- case 'memory':
370
- return this.formatMemoryEvent(event);
371
- case 'log':
372
- return this.formatLogEvent(event);
373
- case 'code':
374
- return this.formatCodeEvent(event);
375
- default:
376
- return this.formatGenericEvent(event);
377
- }
378
- }
379
-
380
- /**
381
- * Format session event data
382
- */
383
- formatSessionEvent(event) {
384
- const data = event.data;
385
- if (event.subtype === 'started') {
386
- return `<strong>Session started:</strong> ${data.session_id || 'Unknown'}`;
387
- } else if (event.subtype === 'ended') {
388
- return `<strong>Session ended:</strong> ${data.session_id || 'Unknown'}`;
389
- }
390
- return `<strong>Session:</strong> ${JSON.stringify(data)}`;
391
- }
392
-
393
- /**
394
- * Format Claude event data
395
- */
396
- formatClaudeEvent(event) {
397
- const data = event.data;
398
- if (event.subtype === 'request') {
399
- const prompt = data.prompt || data.message || '';
400
- const truncated = prompt.length > 100 ? prompt.substring(0, 100) + '...' : prompt;
401
- return `<strong>Request:</strong> ${truncated}`;
402
- } else if (event.subtype === 'response') {
403
- const response = data.response || data.content || '';
404
- const truncated = response.length > 100 ? response.substring(0, 100) + '...' : response;
405
- return `<strong>Response:</strong> ${truncated}`;
406
- }
407
- return `<strong>Claude:</strong> ${JSON.stringify(data)}`;
408
- }
409
-
410
- /**
411
- * Format agent event data
412
- */
413
- formatAgentEvent(event) {
414
- const data = event.data;
415
- if (event.subtype === 'loaded') {
416
- return `<strong>Agent loaded:</strong> ${data.agent_type || data.name || 'Unknown'}`;
417
- } else if (event.subtype === 'executed') {
418
- return `<strong>Agent executed:</strong> ${data.agent_type || data.name || 'Unknown'}`;
419
- }
420
- return `<strong>Agent:</strong> ${JSON.stringify(data)}`;
421
- }
422
-
423
- /**
424
- * Format hook event data
425
- */
426
- formatHookEvent(event) {
427
- const data = event.data;
428
- const eventType = data.event_type || event.subtype || 'unknown';
429
-
430
- // Format based on specific hook event type
431
- switch (eventType) {
432
- case 'user_prompt':
433
- const prompt = data.prompt_text || data.prompt_preview || '';
434
- const truncated = prompt.length > 80 ? prompt.substring(0, 80) + '...' : prompt;
435
- return `<strong>User Prompt:</strong> ${truncated || 'No prompt text'}`;
436
-
437
- case 'pre_tool':
438
- const toolName = data.tool_name || 'Unknown tool';
439
- const operation = data.operation_type || 'operation';
440
- return `<strong>Pre-Tool (${operation}):</strong> ${toolName}`;
441
-
442
- case 'post_tool':
443
- const postToolName = data.tool_name || 'Unknown tool';
444
- const status = data.success ? 'success' : data.status || 'failed';
445
- const duration = data.duration_ms ? ` (${data.duration_ms}ms)` : '';
446
- return `<strong>Post-Tool (${status}):</strong> ${postToolName}${duration}`;
447
-
448
- case 'notification':
449
- const notifType = data.notification_type || 'notification';
450
- const message = data.message_preview || data.message || 'No message';
451
- return `<strong>Notification (${notifType}):</strong> ${message}`;
452
-
453
- case 'stop':
454
- const reason = data.reason || 'unknown';
455
- const stopType = data.stop_type || 'normal';
456
- return `<strong>Stop (${stopType}):</strong> ${reason}`;
457
-
458
- case 'subagent_start':
459
- // Try multiple locations for agent type
460
- const startAgentType = data.agent_type || data.agent || data.subagent_type || 'Unknown';
461
- const startPrompt = data.prompt || data.description || data.task || 'No description';
462
- const startTruncated = startPrompt.length > 60 ? startPrompt.substring(0, 60) + '...' : startPrompt;
463
- // Format with proper agent type display
464
- const startAgentDisplay = this.formatAgentType(startAgentType);
465
- return `<strong>Subagent Start (${startAgentDisplay}):</strong> ${startTruncated}`;
466
-
467
- case 'subagent_stop':
468
- // Try multiple locations for agent type
469
- const agentType = data.agent_type || data.agent || data.subagent_type || 'Unknown';
470
- const stopReason = data.reason || data.stop_reason || 'completed';
471
- // Format with proper agent type display
472
- const stopAgentDisplay = this.formatAgentType(agentType);
473
- // Include task completion status if available
474
- const isCompleted = data.structured_response?.task_completed;
475
- const completionStatus = isCompleted !== undefined ? (isCompleted ? ' ✓' : ' ✗') : '';
476
- return `<strong>Subagent Stop (${stopAgentDisplay})${completionStatus}:</strong> ${stopReason}`;
477
-
478
- default:
479
- // Fallback to original logic for unknown hook types
480
- const hookName = data.hook_name || data.name || data.event_type || 'Unknown';
481
- const phase = event.subtype || eventType;
482
- return `<strong>Hook ${phase}:</strong> ${hookName}`;
483
- }
484
- }
485
-
486
- /**
487
- * Format todo event data
488
- */
489
- formatTodoEvent(event) {
490
- const data = event.data;
491
- if (data.todos && Array.isArray(data.todos)) {
492
- const count = data.todos.length;
493
- return `<strong>Todo updated:</strong> ${count} item${count !== 1 ? 's' : ''}`;
494
- }
495
- return `<strong>Todo:</strong> ${JSON.stringify(data)}`;
496
- }
497
-
498
- /**
499
- * Format memory event data
500
- */
501
- formatMemoryEvent(event) {
502
- const data = event.data;
503
- const operation = data.operation || 'unknown';
504
- return `<strong>Memory ${operation}:</strong> ${data.key || 'Unknown key'}`;
505
- }
506
-
507
- /**
508
- * Format log event data
509
- */
510
- formatLogEvent(event) {
511
- const data = event.data;
512
- const level = data.level || 'info';
513
- const message = data.message || '';
514
- const truncated = message.length > 80 ? message.substring(0, 80) + '...' : message;
515
- return `<strong>[${level.toUpperCase()}]</strong> ${truncated}`;
516
- }
517
-
518
- /**
519
- * Format code analysis event data
520
- */
521
- formatCodeEvent(event) {
522
- const data = event.data || {};
523
-
524
- // Handle different code event subtypes
525
- if (event.subtype === 'progress') {
526
- const message = data.message || 'Processing...';
527
- const percentage = data.percentage;
528
- if (percentage !== undefined) {
529
- return `<strong>Progress:</strong> ${message} (${Math.round(percentage)}%)`;
530
- }
531
- return `<strong>Progress:</strong> ${message}`;
532
- } else if (event.subtype === 'analysis:queued') {
533
- return `<strong>Queued:</strong> Analysis for ${data.path || 'Unknown path'}`;
534
- } else if (event.subtype === 'analysis:start') {
535
- return `<strong>Started:</strong> Analyzing ${data.path || 'Unknown path'}`;
536
- } else if (event.subtype === 'analysis:complete') {
537
- const duration = data.duration ? ` (${data.duration.toFixed(2)}s)` : '';
538
- return `<strong>Complete:</strong> Analysis finished${duration}`;
539
- } else if (event.subtype === 'analysis:error') {
540
- return `<strong>Error:</strong> ${data.message || 'Analysis failed'}`;
541
- } else if (event.subtype === 'analysis:cancelled') {
542
- return `<strong>Cancelled:</strong> Analysis stopped for ${data.path || 'Unknown path'}`;
543
- } else if (event.subtype === 'file:start') {
544
- return `<strong>File:</strong> Processing ${data.file || 'Unknown file'}`;
545
- } else if (event.subtype === 'file:complete') {
546
- const nodes = data.nodes_count !== undefined ? ` (${data.nodes_count} nodes)` : '';
547
- return `<strong>File done:</strong> ${data.file || 'Unknown file'}${nodes}`;
548
- } else if (event.subtype === 'node:found') {
549
- return `<strong>Node:</strong> Found ${data.node_type || 'element'} "${data.name || 'unnamed'}"`;
550
- } else if (event.subtype === 'error') {
551
- return `<strong>Error:</strong> ${data.error || 'Unknown error'} in ${data.file || 'file'}`;
552
- }
553
-
554
- // Generic fallback for code events
555
- const json = JSON.stringify(data);
556
- return `<strong>Code:</strong> ${json.length > 100 ? json.substring(0, 100) + '...' : json}`;
557
- }
558
-
559
- /**
560
- * Format generic event data
561
- */
562
- formatGenericEvent(event) {
563
- const data = event.data;
564
- if (typeof data === 'string') {
565
- return data.length > 100 ? data.substring(0, 100) + '...' : data;
566
- }
567
- return JSON.stringify(data);
568
- }
569
-
570
- /**
571
- * Format agent type for display with proper capitalization
572
- * @param {string} agentType - The raw agent type string
573
- * @returns {string} Formatted agent type for display
574
- */
575
- formatAgentType(agentType) {
576
- // Handle common agent type patterns
577
- const agentTypeMap = {
578
- 'research': 'Research',
579
- 'architect': 'Architect',
580
- 'engineer': 'Engineer',
581
- 'qa': 'QA',
582
- 'pm': 'PM',
583
- 'project_manager': 'PM',
584
- 'research_agent': 'Research',
585
- 'architect_agent': 'Architect',
586
- 'engineer_agent': 'Engineer',
587
- 'qa_agent': 'QA',
588
- 'unknown': 'Unknown'
589
- };
590
-
591
- // Try to find a match in the map (case-insensitive)
592
- const lowerType = (agentType || 'unknown').toLowerCase();
593
- if (agentTypeMap[lowerType]) {
594
- return agentTypeMap[lowerType];
595
- }
596
-
597
- // If not in map, try to extract the agent name from patterns like "Research Agent" or "research_agent"
598
- const match = agentType.match(/^(\w+)(?:_agent|Agent)?$/i);
599
- if (match && match[1]) {
600
- // Capitalize first letter
601
- return match[1].charAt(0).toUpperCase() + match[1].slice(1).toLowerCase();
602
- }
603
-
604
- // Fallback: just capitalize first letter of whatever we have
605
- return agentType.charAt(0).toUpperCase() + agentType.slice(1);
606
- }
607
-
608
- /**
609
- * Format event content for single-row display (without timestamp)
610
- * Format: "{type}.{subtype}" followed by data details
611
- * @param {Object} event - Event object
612
- * @returns {string} Formatted single-row event content string
613
- */
614
- formatSingleRowEventContent(event) {
615
- const eventType = this.formatEventType(event);
616
- const data = event.data || {};
617
-
618
- // Include source if it's not the default 'system' source
619
- const sourcePrefix = event.source && event.source !== 'system' ? `[${event.source}] ` : '';
620
-
621
- // Extract meaningful details from the data package for different event types
622
- let dataDetails = '';
623
-
624
- switch (event.type) {
625
- case 'hook':
626
- // Hook events: show tool name and operation details
627
- const toolName = event.tool_name || data.tool_name || 'Unknown';
628
- const hookType = event.subtype || 'Unknown';
629
-
630
- // Format specific hook types
631
- if (hookType === 'pre_tool' || hookType === 'post_tool') {
632
- const operation = data.operation_type || '';
633
- const status = hookType === 'post_tool' && data.success !== undefined
634
- ? (data.success ? '✓' : '✗')
635
- : '';
636
- dataDetails = `${toolName}${operation ? ` (${operation})` : ''}${status ? ` ${status}` : ''}`;
637
- } else if (hookType === 'user_prompt') {
638
- const prompt = data.prompt_text || data.prompt_preview || '';
639
- const truncated = prompt.length > 60 ? prompt.substring(0, 60) + '...' : prompt;
640
- dataDetails = truncated || 'No prompt text';
641
- } else if (hookType === 'subagent_start') {
642
- // Enhanced agent type detection
643
- const agentType = data.agent_type || data.agent || data.subagent_type || 'Unknown';
644
- const agentDisplay = this.formatAgentType(agentType);
645
- const prompt = data.prompt || data.description || data.task || '';
646
- const truncated = prompt.length > 40 ? prompt.substring(0, 40) + '...' : prompt;
647
- dataDetails = truncated ? `${agentDisplay} - ${truncated}` : agentDisplay;
648
- } else if (hookType === 'subagent_stop') {
649
- // Enhanced agent type detection for subagent_stop
650
- const agentType = data.agent_type || data.agent || data.subagent_type || 'Unknown';
651
- const agentDisplay = this.formatAgentType(agentType);
652
- const reason = data.reason || data.stop_reason || 'completed';
653
- const isCompleted = data.structured_response?.task_completed;
654
- const status = isCompleted !== undefined ? (isCompleted ? '✓' : '✗') : '';
655
- dataDetails = `${agentDisplay}${status ? ' ' + status : ''} - ${reason}`;
656
- } else if (hookType === 'stop') {
657
- const reason = data.reason || 'completed';
658
- const stopType = data.stop_type || 'normal';
659
- dataDetails = `${stopType} - ${reason}`;
660
- } else {
661
- dataDetails = toolName;
662
- }
663
- break;
664
-
665
- case 'agent':
666
- // Agent events: show agent name and status
667
- const agentName = event.subagent_type || data.subagent_type || 'PM';
668
- const status = data.status || '';
669
- dataDetails = `${agentName}${status ? ` - ${status}` : ''}`;
670
- break;
671
-
672
- case 'todo':
673
- // Todo events: show item count and status changes
674
- if (data.todos && Array.isArray(data.todos)) {
675
- const count = data.todos.length;
676
- const completed = data.todos.filter(t => t.status === 'completed').length;
677
- const inProgress = data.todos.filter(t => t.status === 'in_progress').length;
678
- dataDetails = `${count} items (${completed} completed, ${inProgress} in progress)`;
679
- } else {
680
- dataDetails = 'Todo update';
681
- }
682
- break;
683
-
684
- case 'memory':
685
- // Memory events: show operation and key
686
- const operation = data.operation || 'unknown';
687
- const key = data.key || 'unknown';
688
- const value = data.value ? ` = ${JSON.stringify(data.value).substring(0, 30)}...` : '';
689
- dataDetails = `${operation}: ${key}${value}`;
690
- break;
691
-
692
- case 'session':
693
- // Session events: show session ID
694
- const sessionId = data.session_id || 'unknown';
695
- dataDetails = `ID: ${sessionId}`;
696
- break;
697
-
698
- case 'claude':
699
- // Claude events: show request/response preview
700
- if (event.subtype === 'request') {
701
- const prompt = data.prompt || data.message || '';
702
- const truncated = prompt.length > 60 ? prompt.substring(0, 60) + '...' : prompt;
703
- dataDetails = truncated || 'Empty request';
704
- } else if (event.subtype === 'response') {
705
- const response = data.response || data.content || '';
706
- const truncated = response.length > 60 ? response.substring(0, 60) + '...' : response;
707
- dataDetails = truncated || 'Empty response';
708
- } else {
709
- dataDetails = data.message || 'Claude interaction';
710
- }
711
- break;
712
-
713
- case 'log':
714
- // Log events: show log level and message
715
- const level = data.level || 'info';
716
- const message = data.message || '';
717
- const truncated = message.length > 60 ? message.substring(0, 60) + '...' : message;
718
- dataDetails = `[${level.toUpperCase()}] ${truncated}`;
719
- break;
720
-
721
- case 'test':
722
- // Test events: show test name or details
723
- const testName = data.test_name || data.name || 'Test';
724
- dataDetails = testName;
725
- break;
726
-
727
- default:
728
- // Generic events: show any available data
729
- if (typeof data === 'string') {
730
- dataDetails = data.length > 60 ? data.substring(0, 60) + '...' : data;
731
- } else if (data.message) {
732
- dataDetails = data.message.length > 60 ? data.message.substring(0, 60) + '...' : data.message;
733
- } else if (data.name) {
734
- dataDetails = data.name;
735
- } else if (Object.keys(data).length > 0) {
736
- // Show first meaningful field from data
737
- const firstKey = Object.keys(data).find(k => !['timestamp', 'id'].includes(k));
738
- if (firstKey) {
739
- const value = data[firstKey];
740
- dataDetails = `${firstKey}: ${typeof value === 'object' ? JSON.stringify(value).substring(0, 40) + '...' : value}`;
741
- }
742
- }
743
- break;
744
- }
745
-
746
- // Return formatted string: "[source] {type}.{subtype} - {data details}"
747
- // The eventType already contains the type.subtype format from formatEventType()
748
- const fullType = `${sourcePrefix}${eventType}`;
749
- return dataDetails ? `${fullType} - ${dataDetails}` : fullType;
750
- }
751
-
752
- /**
753
- * Get display name for hook types
754
- * @param {string} hookType - Hook subtype
755
- * @param {Object} data - Event data
756
- * @returns {string} Display name
757
- */
758
- getHookDisplayName(hookType, data) {
759
- const hookNames = {
760
- 'pre_tool': 'Pre-Tool',
761
- 'post_tool': 'Post-Tool',
762
- 'user_prompt': 'User-Prompt',
763
- 'stop': 'Stop',
764
- 'subagent_start': 'Subagent-Start',
765
- 'subagent_stop': 'Subagent-Stop',
766
- 'notification': 'Notification'
767
- };
768
-
769
- // Handle non-string hookType safely
770
- if (hookNames[hookType]) {
771
- return hookNames[hookType];
772
- }
773
-
774
- // Convert to string and handle null/undefined
775
- const typeStr = String(hookType || 'unknown');
776
- return typeStr.replace(/_/g, ' ');
777
- }
778
-
779
- /**
780
- * Get event category for display
781
- * @param {Object} event - Event object
782
- * @returns {string} Category
783
- */
784
- getEventCategory(event) {
785
- const data = event.data || {};
786
- const toolName = event.tool_name || data.tool_name || '';
787
-
788
- // Categorize based on tool type
789
- if (['Read', 'Write', 'Edit', 'MultiEdit'].includes(toolName)) {
790
- return 'file_operations';
791
- } else if (['Bash', 'grep', 'Glob'].includes(toolName)) {
792
- return 'system_operations';
793
- } else if (toolName === 'TodoWrite') {
794
- return 'task_management';
795
- } else if (toolName === 'Task') {
796
- return 'agent_delegation';
797
- } else if (event.subtype === 'subagent_start' || event.subtype === 'subagent_stop') {
798
- return 'agent_delegation';
799
- } else if (event.subtype === 'stop') {
800
- return 'session_control';
801
- }
802
-
803
- return 'general';
804
- }
805
-
806
- /**
807
- * Show event details and update selection
808
- * @param {number} index - Index of event to show
809
- */
810
- showEventDetails(index) {
811
- // Defensive checks
812
- if (!this.filteredEvents || !Array.isArray(this.filteredEvents)) {
813
- console.warn('EventViewer: filteredEvents array is not initialized');
814
- return;
815
- }
816
- if (index < 0 || index >= this.filteredEvents.length) return;
817
-
818
- // Update selection
819
- this.selectedEventIndex = index;
820
-
821
- // Get the selected event
822
- const event = this.filteredEvents[index];
823
-
824
- // Coordinate with Dashboard unified navigation system
825
- if (window.dashboard) {
826
- // Update the dashboard's navigation state for events tab
827
- if (window.dashboard.tabNavigation && window.dashboard.tabNavigation.events) {
828
- window.dashboard.tabNavigation.events.selectedIndex = index;
829
- }
830
- if (window.dashboard.selectCard) {
831
- window.dashboard.selectCard('events', index, 'event', event);
832
- }
833
- }
834
-
835
- // Update visual selection (this will be handled by Dashboard.updateCardSelectionUI())
836
- this.filteredEventElements.forEach((el, i) => {
837
- el.classList.toggle('selected', i === index);
838
- });
839
-
840
- // Notify other components about selection
841
- document.dispatchEvent(new CustomEvent('eventSelected', {
842
- detail: { event, index }
843
- }));
844
-
845
- // Scroll to selected event if not visible
846
- const selectedElement = this.filteredEventElements[index];
847
- if (selectedElement) {
848
- selectedElement.scrollIntoView({
849
- behavior: 'smooth',
850
- block: 'nearest'
851
- });
852
- }
853
- }
854
-
855
- /**
856
- * Clear event selection
857
- */
858
- clearSelection() {
859
- this.selectedEventIndex = -1;
860
- this.filteredEventElements.forEach(el => {
861
- el.classList.remove('selected');
862
- });
863
-
864
- // Coordinate with Dashboard unified navigation system
865
- if (window.dashboard) {
866
- if (window.dashboard.tabNavigation && window.dashboard.tabNavigation.events) {
867
- window.dashboard.tabNavigation.events.selectedIndex = -1;
868
- }
869
- if (window.dashboard.clearCardSelection) {
870
- window.dashboard.clearCardSelection();
871
- }
872
- }
873
-
874
- // Notify other components
875
- document.dispatchEvent(new CustomEvent('eventSelectionCleared'));
876
- }
877
-
878
- /**
879
- * Update metrics display
880
- */
881
- updateMetrics() {
882
- // Update event type counts
883
- this.eventTypeCount = {};
884
- this.errorCount = 0;
885
-
886
- // Defensive check to ensure events array exists
887
- if (!this.events || !Array.isArray(this.events)) {
888
- console.warn('EventViewer: events array is not initialized in updateMetrics');
889
- this.events = [];
890
- }
891
-
892
- this.events.forEach(event => {
893
- const type = event.type || 'unknown';
894
- this.eventTypeCount[type] = (this.eventTypeCount[type] || 0) + 1;
895
-
896
- if (event.type === 'log' &&
897
- event.data &&
898
- ['error', 'critical'].includes(event.data.level)) {
899
- this.errorCount++;
900
- }
901
- });
902
-
903
- // Update events per minute
904
- const currentMinute = new Date().getMinutes();
905
- if (currentMinute !== this.lastMinute) {
906
- this.lastMinute = currentMinute;
907
- this.eventsThisMinute = 0;
908
- }
909
-
910
- // Count events in the last minute
911
- const oneMinuteAgo = new Date(Date.now() - 60000);
912
- this.eventsThisMinute = this.events.filter(event =>
913
- new Date(event.timestamp) > oneMinuteAgo
914
- ).length;
915
-
916
- // Update UI
917
- this.updateMetricsUI();
918
- }
919
-
920
- /**
921
- * Update metrics in the UI
922
- */
923
- updateMetricsUI() {
924
- const totalEventsEl = document.getElementById('total-events');
925
- const eventsPerMinuteEl = document.getElementById('events-per-minute');
926
- const uniqueTypesEl = document.getElementById('unique-types');
927
- const errorCountEl = document.getElementById('error-count');
928
-
929
- if (totalEventsEl) totalEventsEl.textContent = this.events.length;
930
- if (eventsPerMinuteEl) eventsPerMinuteEl.textContent = this.eventsThisMinute;
931
- if (uniqueTypesEl) uniqueTypesEl.textContent = Object.keys(this.eventTypeCount).length;
932
- if (errorCountEl) errorCountEl.textContent = this.errorCount;
933
- }
934
-
935
- /**
936
- * Export events to JSON
937
- */
938
- exportEvents() {
939
- const dataStr = JSON.stringify(this.filteredEvents, null, 2);
940
- const dataBlob = new Blob([dataStr], { type: 'application/json' });
941
- const url = URL.createObjectURL(dataBlob);
942
-
943
- const link = document.createElement('a');
944
- link.href = url;
945
- link.download = `claude-mpm-events-${new Date().toISOString().split('T')[0]}.json`;
946
- link.click();
947
-
948
- URL.revokeObjectURL(url);
949
- }
950
-
951
- /**
952
- * Clear all events
953
- */
954
- clearEvents() {
955
- this.socketClient.clearEvents();
956
- this.selectedEventIndex = -1;
957
- this.updateDisplay();
958
- }
959
-
960
- /**
961
- * Set session filter
962
- * @param {string} sessionId - Session ID to filter by
963
- */
964
- setSessionFilter(sessionId) {
965
- this.sessionFilter = sessionId;
966
- this.applyFilters();
967
- }
968
-
969
- /**
970
- * Get current filter state
971
- * @returns {Object} Current filters
972
- */
973
- getFilters() {
974
- return {
975
- search: this.searchFilter,
976
- type: this.typeFilter,
977
- session: this.sessionFilter
978
- };
979
- }
980
-
981
- /**
982
- * Get filtered events (used by HUD and other components)
983
- * @returns {Array} Array of filtered events
984
- */
985
- getFilteredEvents() {
986
- return this.filteredEvents;
987
- }
988
-
989
- /**
990
- * Get all events (unfiltered, used by HUD for complete visualization)
991
- * @returns {Array} Array of all events
992
- */
993
- getAllEvents() {
994
- return this.events;
995
- }
996
-
997
- /**
998
- * Create inline diff viewer for Edit/MultiEdit tool events
999
- * WHY: Provides immediate visibility of file changes without needing to open modals
1000
- * DESIGN DECISION: Shows inline diffs only for Edit/MultiEdit events to avoid clutter
1001
- * @param {Object} event - Event object
1002
- * @param {number} index - Event index for unique IDs
1003
- * @returns {string} HTML for inline diff viewer
1004
- */
1005
- createInlineEditDiffViewer(event, index) {
1006
- const data = event.data || {};
1007
- const toolName = event.tool_name || data.tool_name || '';
1008
-
1009
- // Only show for Edit and MultiEdit tools
1010
- if (!['Edit', 'MultiEdit'].includes(toolName)) {
1011
- return '';
1012
- }
1013
-
1014
- // Extract edit parameters based on tool type
1015
- let edits = [];
1016
- if (toolName === 'Edit') {
1017
- // Single edit
1018
- const parameters = event.tool_parameters || data.tool_parameters || {};
1019
- if (parameters.old_string && parameters.new_string) {
1020
- edits.push({
1021
- old_string: parameters.old_string,
1022
- new_string: parameters.new_string,
1023
- file_path: parameters.file_path || 'unknown'
1024
- });
1025
- }
1026
- } else if (toolName === 'MultiEdit') {
1027
- // Multiple edits
1028
- const parameters = event.tool_parameters || data.tool_parameters || {};
1029
- if (parameters.edits && Array.isArray(parameters.edits)) {
1030
- edits = parameters.edits.map(edit => ({
1031
- ...edit,
1032
- file_path: parameters.file_path || 'unknown'
1033
- }));
1034
- }
1035
- }
1036
-
1037
- if (edits.length === 0) {
1038
- return '';
1039
- }
1040
-
1041
- // Create collapsible diff section
1042
- const diffId = `edit-diff-${index}`;
1043
- const isMultiEdit = edits.length > 1;
1044
-
1045
- let diffContent = '';
1046
- edits.forEach((edit, editIndex) => {
1047
- const editId = `${diffId}-${editIndex}`;
1048
- const diffHtml = this.createDiffHtml(edit.old_string, edit.new_string);
1049
-
1050
- diffContent += `
1051
- <div class="edit-diff-section">
1052
- ${isMultiEdit ? `<div class="edit-diff-header">Edit ${editIndex + 1}</div>` : ''}
1053
- <div class="diff-content">${diffHtml}</div>
1054
- </div>
1055
- `;
1056
- });
1057
-
1058
- return `
1059
- <div class="inline-edit-diff-viewer">
1060
- <div class="diff-toggle-header" onclick="eventViewer.toggleEditDiff('${diffId}', event)">
1061
- <span class="diff-toggle-icon">📋</span>
1062
- <span class="diff-toggle-text">Show ${isMultiEdit ? edits.length + ' edits' : 'edit'}</span>
1063
- <span class="diff-toggle-arrow">▼</span>
1064
- </div>
1065
- <div id="${diffId}" class="diff-content-container" style="display: none;">
1066
- ${diffContent}
1067
- </div>
1068
- </div>
1069
- `;
1070
- }
1071
-
1072
- /**
1073
- * Create HTML diff visualization
1074
- * WHY: Provides clear visual representation of text changes similar to git diff
1075
- * @param {string} oldText - Original text
1076
- * @param {string} newText - Modified text
1077
- * @returns {string} HTML diff content
1078
- */
1079
- createDiffHtml(oldText, newText) {
1080
- // Simple line-by-line diff implementation
1081
- const oldLines = oldText.split('\n');
1082
- const newLines = newText.split('\n');
1083
-
1084
- let diffHtml = '';
1085
- let i = 0, j = 0;
1086
-
1087
- // Simple diff algorithm - can be enhanced with proper diff library if needed
1088
- while (i < oldLines.length || j < newLines.length) {
1089
- const oldLine = i < oldLines.length ? oldLines[i] : null;
1090
- const newLine = j < newLines.length ? newLines[j] : null;
1091
-
1092
- if (oldLine === null) {
1093
- // New line added
1094
- diffHtml += `<div class="diff-line diff-added">+ ${this.escapeHtml(newLine)}</div>`;
1095
- j++;
1096
- } else if (newLine === null) {
1097
- // Old line removed
1098
- diffHtml += `<div class="diff-line diff-removed">- ${this.escapeHtml(oldLine)}</div>`;
1099
- i++;
1100
- } else if (oldLine === newLine) {
1101
- // Lines are the same
1102
- diffHtml += `<div class="diff-line diff-unchanged"> ${this.escapeHtml(oldLine)}</div>`;
1103
- i++;
1104
- j++;
1105
- } else {
1106
- // Lines are different - show both
1107
- diffHtml += `<div class="diff-line diff-removed">- ${this.escapeHtml(oldLine)}</div>`;
1108
- diffHtml += `<div class="diff-line diff-added">+ ${this.escapeHtml(newLine)}</div>`;
1109
- i++;
1110
- j++;
1111
- }
1112
- }
1113
-
1114
- return `<div class="diff-container">${diffHtml}</div>`;
1115
- }
1116
-
1117
- /**
1118
- * Toggle edit diff visibility
1119
- * @param {string} diffId - Diff container ID
1120
- * @param {Event} event - Click event
1121
- */
1122
- toggleEditDiff(diffId, event) {
1123
- // Prevent event bubbling to parent event item
1124
- event.stopPropagation();
1125
-
1126
- const diffContainer = document.getElementById(diffId);
1127
- const arrow = event.currentTarget.querySelector('.diff-toggle-arrow');
1128
-
1129
- if (diffContainer) {
1130
- const isVisible = diffContainer.style.display !== 'none';
1131
- diffContainer.style.display = isVisible ? 'none' : 'block';
1132
- if (arrow) {
1133
- arrow.textContent = isVisible ? '▼' : '▲';
1134
- }
1135
- }
1136
- }
1137
-
1138
- /**
1139
- * Escape HTML characters for safe display
1140
- * @param {string} text - Text to escape
1141
- * @returns {string} Escaped text
1142
- */
1143
- escapeHtml(text) {
1144
- const div = document.createElement('div');
1145
- div.textContent = text;
1146
- return div.innerHTML;
1147
- }
1148
- }
1149
-
1150
- // ES6 Module export
1151
- export { EventViewer };
1152
- export default EventViewer;
1153
-
1154
- // Backward compatibility - keep window export for non-module usage
1155
- window.EventViewer = EventViewer;