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,857 +0,0 @@
1
- // Ultra-simple directory browser - maximum compatibility and debugging
2
- console.log('[code-simple.js] Script loaded at', new Date().toISOString());
3
-
4
- // Global function for onclick handlers
5
- function loadDir(path) {
6
- console.log('[loadDir] Called with path:', path);
7
- if (window.simpleCodeView) {
8
- window.simpleCodeView.loadDirectory(path);
9
- } else {
10
- console.error('[loadDir] simpleCodeView not initialized');
11
- }
12
- }
13
-
14
- function goUp() {
15
- console.log('[goUp] Called');
16
- if (window.simpleCodeView) {
17
- window.simpleCodeView.goUp();
18
- } else {
19
- console.error('[goUp] simpleCodeView not initialized');
20
- }
21
- }
22
-
23
- function analyzeFileFromPath(filePath) {
24
- console.log('[analyzeFileFromPath] Called with path:', filePath);
25
- if (window.simpleCodeView) {
26
- window.simpleCodeView.analyzeFileFromPath(filePath);
27
- } else {
28
- console.error('[analyzeFileFromPath] simpleCodeView not initialized');
29
- }
30
- }
31
-
32
- class SimpleCodeView {
33
- constructor() {
34
- console.log('[SimpleCodeView] Constructor called');
35
- // Try to get the current working directory from various sources
36
- this.currentPath = this.getInitialPath();
37
- this.container = null;
38
- this.apiBase = window.location.origin;
39
- console.log('[SimpleCodeView] API base:', this.apiBase);
40
-
41
- // Tree view properties
42
- this.currentView = 'directory';
43
- this.socket = null;
44
- this.svg = null;
45
- this.treeGroup = null;
46
- this.treeLayout = null;
47
- this.treeData = null;
48
- this.width = 800;
49
- this.height = 600;
50
- this.margin = {top: 20, right: 20, bottom: 20, left: 120};
51
- }
52
-
53
- init(container) {
54
- console.log('[SimpleCodeView.init] Starting with container:', container);
55
-
56
- if (!container) {
57
- console.error('[SimpleCodeView.init] No container provided!');
58
- document.body.innerHTML += '<div style="color:red;font-size:20px;">ERROR: No container for SimpleCodeView</div>';
59
- return;
60
- }
61
-
62
- this.container = container;
63
- this.render();
64
-
65
- // Load initial directory after a short delay to ensure DOM is ready
66
- setTimeout(() => {
67
- console.log('[SimpleCodeView.init] Loading initial directory after delay');
68
- this.loadDirectory(this.currentPath);
69
- }, 100);
70
- }
71
-
72
- render() {
73
- console.log('[SimpleCodeView.render] Rendering UI');
74
-
75
- const html = `
76
- <div class="simple-code-view" style="padding: 20px;">
77
- <h2>Simple Code Browser</h2>
78
-
79
- <div class="view-toggle" style="margin: 10px 0; padding: 10px; background: #f0f0f0; border-radius: 4px;">
80
- <button id="dir-view-btn" onclick="window.simpleCodeView.setView('directory')" class="active"
81
- style="margin-right: 10px; padding: 8px 16px; border: 1px solid #ccc; background: #007cba; color: white; border-radius: 4px; cursor: pointer;">
82
- 📁 Directory View
83
- </button>
84
- <button id="tree-view-btn" onclick="window.simpleCodeView.setView('tree')"
85
- style="padding: 8px 16px; border: 1px solid #ccc; background: #f9f9f9; color: #333; border-radius: 4px; cursor: pointer;">
86
- 🌳 Tree View
87
- </button>
88
- </div>
89
-
90
- <div id="status-bar" style="padding: 10px; background: #e0e0e0; border-radius: 4px; margin-bottom: 10px;">
91
- Status: Initializing...
92
- </div>
93
-
94
- <div class="path-bar" style="margin: 10px 0; padding: 10px; background: #f0f0f0; border-radius: 4px;">
95
- <strong>Current Path:</strong>
96
- <input type="text" id="path-input" value="${this.currentPath}" style="width: 50%; margin: 0 10px;">
97
- <button id="load-btn" onclick="loadDir(document.getElementById('path-input').value)">Load</button>
98
- <button id="up-btn" onclick="goUp()">Go Up</button>
99
- </div>
100
-
101
- <div id="error-display" style="display:none; padding: 10px; background: #fee; color: red; border: 1px solid #fcc; border-radius: 4px; margin: 10px 0;">
102
- </div>
103
-
104
- <div id="directory-contents" style="border: 1px solid #ccc; padding: 10px; min-height: 400px; background: white;">
105
- <div style="color: #666;">Waiting to load directory...</div>
106
- </div>
107
-
108
- <div id="tree-view-container" style="display: none;">
109
- <div class="file-selector" style="margin: 10px 0; padding: 10px; background: #f9f9f9; border-radius: 4px;">
110
- <input type="text" id="file-path-input" placeholder="Enter file path to analyze (e.g., ./src/claude_mpm/core/framework_loader.py)"
111
- style="width: 70%; padding: 8px; margin-right: 10px; border: 1px solid #ccc; border-radius: 4px;">
112
- <button onclick="window.simpleCodeView.analyzeFile()"
113
- style="padding: 8px 16px; border: 1px solid #ccc; background: #28a745; color: white; border-radius: 4px; cursor: pointer;">
114
- Analyze File
115
- </button>
116
- </div>
117
- <div id="tree-visualization" style="border: 1px solid #ccc; min-height: 500px; background: white; overflow: auto; position: relative;">
118
- <div style="padding: 20px; text-align: center; color: #666;">Enter a file path above and click "Analyze File" to view AST tree</div>
119
- </div>
120
- </div>
121
-
122
- <div id="debug-info" style="margin-top: 10px; padding: 10px; background: #f9f9f9; font-family: monospace; font-size: 12px;">
123
- <strong>Debug Info:</strong><br>
124
- API Base: ${this.apiBase}<br>
125
- Current Path: ${this.currentPath}<br>
126
- Status: Waiting for first load...
127
- </div>
128
- </div>
129
- `;
130
-
131
- this.container.innerHTML = html;
132
- console.log('[SimpleCodeView.render] UI rendered');
133
-
134
- this.updateStatus('UI Rendered - Ready to load directory', 'blue');
135
- }
136
-
137
- updateStatus(message, color = 'black') {
138
- console.log('[SimpleCodeView.updateStatus]', message);
139
- const statusBar = document.getElementById('status-bar');
140
- if (statusBar) {
141
- statusBar.innerHTML = `Status: ${message}`;
142
- statusBar.style.color = color;
143
- }
144
- }
145
-
146
- showError(message) {
147
- console.error('[SimpleCodeView.showError]', message);
148
- const errorDiv = document.getElementById('error-display');
149
- if (errorDiv) {
150
- errorDiv.style.display = 'block';
151
- errorDiv.innerHTML = `Error: ${message}`;
152
- }
153
- this.updateStatus('Error occurred', 'red');
154
- }
155
-
156
- hideError() {
157
- const errorDiv = document.getElementById('error-display');
158
- if (errorDiv) {
159
- errorDiv.style.display = 'none';
160
- }
161
- }
162
-
163
- async loadDirectory(path) {
164
- console.log('[SimpleCodeView.loadDirectory] Loading path:', path);
165
-
166
- this.currentPath = path;
167
- this.hideError();
168
- this.updateStatus(`Loading ${path}...`, 'blue');
169
-
170
- // Update path input
171
- const pathInput = document.getElementById('path-input');
172
- if (pathInput) {
173
- pathInput.value = path;
174
- }
175
-
176
- // Update debug info
177
- const debugDiv = document.getElementById('debug-info');
178
- const contentsDiv = document.getElementById('directory-contents');
179
-
180
- const apiUrl = `${this.apiBase}/api/directory/list?path=${encodeURIComponent(path)}`;
181
-
182
- if (debugDiv) {
183
- debugDiv.innerHTML = `
184
- <strong>Debug Info:</strong><br>
185
- API URL: ${apiUrl}<br>
186
- Timestamp: ${new Date().toISOString()}<br>
187
- Status: Fetching...
188
- `;
189
- }
190
-
191
- try {
192
- console.log('[SimpleCodeView.loadDirectory] Fetching:', apiUrl);
193
-
194
- const response = await fetch(apiUrl);
195
- console.log('[SimpleCodeView.loadDirectory] Response status:', response.status);
196
-
197
- if (!response.ok) {
198
- throw new Error(`HTTP ${response.status}: ${response.statusText}`);
199
- }
200
-
201
- const data = await response.json();
202
- console.log('[SimpleCodeView.loadDirectory] Data received:', data);
203
-
204
- // Update debug info with response and filtering status
205
- if (debugDiv) {
206
- let debugContent = `
207
- <strong>Debug Info:</strong><br>
208
- API URL: ${apiUrl}<br>
209
- Response Status: ${response.status}<br>
210
- Path Exists: ${data.exists}<br>
211
- Is Directory: ${data.is_directory}<br>
212
- Item Count: ${data.contents ? data.contents.length : 0}<br>
213
- `;
214
-
215
- // Add filtering information
216
- if (data.filtered) {
217
- debugContent += `<strong>Filtering:</strong> ${data.filter_info || 'Filtered view'}<br>`;
218
- if (data.summary) {
219
- debugContent += `<strong>Items:</strong> ${data.summary.directories} directories, ${data.summary.code_files} code files<br>`;
220
- }
221
- }
222
-
223
- debugContent += `
224
- <details>
225
- <summary>Raw Response (click to expand)</summary>
226
- <pre style="overflow-x: auto;">${JSON.stringify(data, null, 2)}</pre>
227
- </details>
228
- `;
229
-
230
- debugDiv.innerHTML = debugContent;
231
- }
232
-
233
- // Display contents
234
- if (!data.exists) {
235
- contentsDiv.innerHTML = '<p style="color: red;">❌ Path does not exist</p>';
236
- this.updateStatus('Path does not exist', 'red');
237
- } else if (!data.is_directory) {
238
- contentsDiv.innerHTML = '<p style="color: orange;">⚠️ Path is not a directory</p>';
239
- this.updateStatus('Not a directory', 'orange');
240
- } else if (data.error) {
241
- contentsDiv.innerHTML = `<p style="color: red;">❌ Error: ${data.error}</p>`;
242
- this.showError(data.error);
243
- } else if (!data.contents || data.contents.length === 0) {
244
- contentsDiv.innerHTML = '<p style="color: gray;">📭 No code files or subdirectories found (hidden files/folders not shown)</p>';
245
- this.updateStatus('No code content found', 'gray');
246
- } else {
247
- // Build the list with filtering indicator
248
- let headerText = `Found ${data.contents.length} items`;
249
- if (data.filtered && data.summary) {
250
- headerText += ` (${data.summary.directories} directories, ${data.summary.code_files} code files)`;
251
- }
252
- headerText += ':';
253
-
254
- let html = `<div style="margin-bottom: 10px; color: #666;">${headerText}</div>`;
255
-
256
- // Add filtering notice if applicable
257
- if (data.filtered) {
258
- html += `<div style="margin-bottom: 10px; padding: 8px; background: #e8f4fd; border-left: 3px solid #2196f3; color: #1565c0; font-size: 13px;">
259
- 🔍 Filtered view: ${data.filter_info || 'Showing only code-related files and directories'}
260
- </div>`;
261
- }
262
-
263
- html += '<ul style="list-style: none; padding: 0; margin: 0;">';
264
-
265
- // Sort: directories first, then files
266
- const sorted = data.contents.sort((a, b) => {
267
- if (a.is_directory !== b.is_directory) {
268
- return a.is_directory ? -1 : 1;
269
- }
270
- return a.name.localeCompare(b.name);
271
- });
272
-
273
- for (const item of sorted) {
274
- if (item.is_directory) {
275
- // Make directories clickable
276
- html += `<li style="padding: 5px 0;">
277
- 📁 <a href="#" onclick="loadDir('${item.path.replace(/'/g, "\\'")}'); return false;" style="color: blue; text-decoration: none; cursor: pointer;">
278
- ${item.name}/
279
- </a>
280
- </li>`;
281
- } else {
282
- // Check if it's a code file and make it clickable
283
- const isCodeFile = this.isCodeFile(item.name);
284
- const fileIcon = this.getFileIcon(item.name);
285
-
286
- if (isCodeFile) {
287
- // Code files - clickable to show AST
288
- html += `<li style="padding: 5px 0;">
289
- ${fileIcon} <a href="#" onclick="analyzeFileFromPath('${item.path.replace(/'/g, "\\'")}'); return false;" style="color: #0066cc; text-decoration: none; cursor: pointer; font-weight: 500;" title="Click to view AST">
290
- ${item.name}
291
- </a>
292
- </li>`;
293
- } else {
294
- // Non-code files - not clickable
295
- html += `<li style="padding: 5px 0;">
296
- 📄 <span style="color: #666;">${item.name}</span>
297
- </li>`;
298
- }
299
- }
300
- }
301
-
302
- html += '</ul>';
303
- contentsDiv.innerHTML = html;
304
- this.updateStatus(`Loaded ${data.contents.length} items`, 'green');
305
- }
306
-
307
- } catch (error) {
308
- console.error('[SimpleCodeView.loadDirectory] Error:', error);
309
-
310
- const errorMsg = `Failed to load directory: ${error.message}`;
311
- this.showError(errorMsg);
312
-
313
- if (contentsDiv) {
314
- contentsDiv.innerHTML = `
315
- <div style="color: red;">
316
- <p>❌ Failed to load directory</p>
317
- <p>Error: ${error.message}</p>
318
- <p style="font-size: 12px;">Check browser console for details</p>
319
- </div>
320
- `;
321
- }
322
-
323
- if (debugDiv) {
324
- debugDiv.innerHTML += `<br><span style="color:red;">ERROR: ${error.stack || error.message}</span>`;
325
- }
326
- }
327
- }
328
-
329
- goUp() {
330
- console.log('[SimpleCodeView.goUp] Current path:', this.currentPath);
331
- if (this.currentPath === '/' || this.currentPath === '') {
332
- console.log('[SimpleCodeView.goUp] Already at root');
333
- this.updateStatus('Already at root directory', 'orange');
334
- return;
335
- }
336
-
337
- const lastSlash = this.currentPath.lastIndexOf('/');
338
- const parent = lastSlash > 0 ? this.currentPath.substring(0, lastSlash) : '/';
339
- console.log('[SimpleCodeView.goUp] Going up to:', parent);
340
- this.loadDirectory(parent);
341
- }
342
-
343
- // Tree view methods
344
- setView(view) {
345
- console.log('[SimpleCodeView.setView] Switching to view:', view);
346
- this.currentView = view;
347
-
348
- const dirContents = document.getElementById('directory-contents');
349
- const treeContainer = document.getElementById('tree-view-container');
350
- const dirBtn = document.getElementById('dir-view-btn');
351
- const treeBtn = document.getElementById('tree-view-btn');
352
-
353
- if (view === 'tree') {
354
- dirContents.style.display = 'none';
355
- treeContainer.style.display = 'block';
356
-
357
- // Update button styles
358
- dirBtn.style.background = '#f9f9f9';
359
- dirBtn.style.color = '#333';
360
- dirBtn.classList.remove('active');
361
-
362
- treeBtn.style.background = '#007cba';
363
- treeBtn.style.color = 'white';
364
- treeBtn.classList.add('active');
365
-
366
- this.initializeTreeView();
367
- } else {
368
- dirContents.style.display = 'block';
369
- treeContainer.style.display = 'none';
370
-
371
- // Update button styles
372
- treeBtn.style.background = '#f9f9f9';
373
- treeBtn.style.color = '#333';
374
- treeBtn.classList.remove('active');
375
-
376
- dirBtn.style.background = '#007cba';
377
- dirBtn.style.color = 'white';
378
- dirBtn.classList.add('active');
379
- }
380
- }
381
-
382
- initializeTreeView() {
383
- console.log('[SimpleCodeView.initializeTreeView] Initializing tree view');
384
-
385
- if (!window.d3) {
386
- this.updateStatus('D3.js not loaded - cannot initialize tree view', 'red');
387
- return;
388
- }
389
-
390
- this.initializeSocket();
391
- }
392
-
393
- initializeSocket() {
394
- console.log('[SimpleCodeView.initializeSocket] Initializing Socket.IO connection');
395
-
396
- if (!window.io) {
397
- this.updateStatus('Socket.IO not loaded - using fallback mode', 'orange');
398
- return;
399
- }
400
-
401
- try {
402
- this.socket = io('/');
403
-
404
- this.socket.on('connect', () => {
405
- console.log('[SimpleCodeView] Socket connected');
406
- this.updateStatus('Connected to analysis server', 'green');
407
- });
408
-
409
- this.socket.on('disconnect', () => {
410
- console.log('[SimpleCodeView] Socket disconnected');
411
- this.updateStatus('Disconnected from analysis server', 'orange');
412
- });
413
-
414
- this.socket.on('analysis_progress', (data) => {
415
- console.log('[SimpleCodeView] Analysis progress:', data);
416
- this.updateStatus(`Analysis: ${data.message}`, 'blue');
417
- });
418
-
419
- this.socket.on('analysis_complete', (data) => {
420
- console.log('[SimpleCodeView] Analysis complete:', data);
421
- this.updateStatus('Analysis complete - rendering tree', 'green');
422
- this.renderTree(data);
423
- });
424
-
425
- this.socket.on('analysis_error', (error) => {
426
- console.error('[SimpleCodeView] Analysis error:', error);
427
- this.showError(`Analysis failed: ${error.message || error}`);
428
- });
429
-
430
- } catch (error) {
431
- console.error('[SimpleCodeView] Failed to initialize socket:', error);
432
- this.updateStatus('Socket connection failed - using fallback mode', 'orange');
433
- }
434
- }
435
-
436
- async analyzeFile() {
437
- console.log('[SimpleCodeView.analyzeFile] Starting file analysis');
438
-
439
- const fileInput = document.getElementById('file-path-input');
440
- const filePath = fileInput.value.trim();
441
-
442
- if (!filePath) {
443
- this.showError('Please enter a file path');
444
- return;
445
- }
446
-
447
- this.hideError();
448
- this.updateStatus(`Analyzing file: ${filePath}`, 'blue');
449
-
450
- // Clear previous tree
451
- const treeViz = document.getElementById('tree-visualization');
452
- if (treeViz) {
453
- treeViz.innerHTML = '<div style="padding: 20px; text-align: center; color: #666;">Analyzing file...</div>';
454
- }
455
-
456
- try {
457
- if (this.socket && this.socket.connected) {
458
- // Use Socket.IO for real-time analysis
459
- this.socket.emit('analyze_file', {
460
- path: filePath,
461
- working_directory: this.currentPath
462
- });
463
- } else {
464
- // Fallback to HTTP API (we'll create a simple endpoint)
465
- await this.analyzeFileHTTP(filePath);
466
- }
467
- } catch (error) {
468
- console.error('[SimpleCodeView.analyzeFile] Error:', error);
469
- this.showError(`Failed to analyze file: ${error.message}`);
470
- }
471
- }
472
-
473
- async analyzeFileHTTP(filePath) {
474
- console.log('[SimpleCodeView.analyzeFileHTTP] Using HTTP fallback for:', filePath);
475
-
476
- try {
477
- // Create a simple mock analysis for demonstration
478
- // In a real implementation, this would call a proper analysis endpoint
479
- setTimeout(() => {
480
- const mockData = this.createMockTreeData(filePath);
481
- this.renderTree(mockData);
482
- }, 1000);
483
-
484
- } catch (error) {
485
- throw new Error(`HTTP analysis failed: ${error.message}`);
486
- }
487
- }
488
-
489
- createMockTreeData(filePath) {
490
- // Create mock AST data for demonstration
491
- const fileName = filePath.split('/').pop() || 'file';
492
- const ext = fileName.split('.').pop()?.toLowerCase();
493
-
494
- let mockData = {
495
- name: fileName,
496
- type: 'module',
497
- children: []
498
- };
499
-
500
- if (ext === 'py') {
501
- mockData.children = [
502
- {
503
- name: 'imports',
504
- type: 'imports',
505
- children: [
506
- { name: 'import os', type: 'import' },
507
- { name: 'from pathlib import Path', type: 'import' }
508
- ]
509
- },
510
- {
511
- name: 'MyClass',
512
- type: 'class',
513
- children: [
514
- { name: '__init__', type: 'method' },
515
- { name: 'process_data', type: 'method' },
516
- { name: 'save_results', type: 'method' }
517
- ]
518
- },
519
- {
520
- name: 'helper_function',
521
- type: 'function',
522
- children: []
523
- }
524
- ];
525
- } else if (ext === 'js' || ext === 'ts') {
526
- mockData.children = [
527
- {
528
- name: 'imports',
529
- type: 'imports',
530
- children: [
531
- { name: "import React from 'react'", type: 'import' },
532
- { name: "import { useState } from 'react'", type: 'import' }
533
- ]
534
- },
535
- {
536
- name: 'MyComponent',
537
- type: 'function',
538
- children: [
539
- { name: 'useState', type: 'hook' },
540
- { name: 'useEffect', type: 'hook' },
541
- { name: 'handleClick', type: 'function' }
542
- ]
543
- }
544
- ];
545
- } else {
546
- mockData.children = [
547
- { name: 'Content Section 1', type: 'section' },
548
- { name: 'Content Section 2', type: 'section' },
549
- { name: 'Content Section 3', type: 'section' }
550
- ];
551
- }
552
-
553
- return mockData;
554
- }
555
-
556
- renderTree(data) {
557
- console.log('[SimpleCodeView.renderTree] Rendering tree with data:', data);
558
-
559
- if (!data || !window.d3) {
560
- this.showError('Cannot render tree: missing data or D3.js');
561
- return;
562
- }
563
-
564
- this.treeData = data;
565
-
566
- // Clear previous visualization
567
- const container = document.getElementById('tree-visualization');
568
- if (!container) {
569
- this.showError('Tree visualization container not found');
570
- return;
571
- }
572
-
573
- container.innerHTML = '';
574
-
575
- // Create SVG
576
- const margin = this.margin;
577
- const width = this.width - margin.left - margin.right;
578
- const height = this.height - margin.top - margin.bottom;
579
-
580
- this.svg = d3.select('#tree-visualization')
581
- .append('svg')
582
- .attr('width', this.width)
583
- .attr('height', this.height);
584
-
585
- // Add zoom behavior
586
- const zoom = d3.zoom()
587
- .scaleExtent([0.1, 4])
588
- .on('zoom', (event) => {
589
- this.treeGroup.attr('transform', event.transform);
590
- });
591
-
592
- this.svg.call(zoom);
593
-
594
- // Create main group
595
- this.treeGroup = this.svg.append('g')
596
- .attr('transform', `translate(${margin.left},${margin.top})`);
597
-
598
- // Create tree layout
599
- this.treeLayout = d3.tree()
600
- .size([height, width]);
601
-
602
- // Convert data to hierarchy
603
- const hierarchy = d3.hierarchy(data);
604
-
605
- // Generate tree layout
606
- const treeData = this.treeLayout(hierarchy);
607
-
608
- // Add links
609
- const links = this.treeGroup.selectAll('.link')
610
- .data(treeData.links())
611
- .enter()
612
- .append('path')
613
- .attr('class', 'link')
614
- .attr('d', d3.linkHorizontal()
615
- .x(d => d.y)
616
- .y(d => d.x)
617
- )
618
- .style('fill', 'none')
619
- .style('stroke', '#ccc')
620
- .style('stroke-width', '2px');
621
-
622
- // Add nodes
623
- const nodes = this.treeGroup.selectAll('.node')
624
- .data(treeData.descendants())
625
- .enter()
626
- .append('g')
627
- .attr('class', 'tree-node')
628
- .attr('transform', d => `translate(${d.y},${d.x})`);
629
-
630
- // Add circles for nodes
631
- nodes.append('circle')
632
- .attr('r', 6)
633
- .style('fill', d => this.getNodeColor(d.data.type))
634
- .style('stroke', '#333')
635
- .style('stroke-width', '2px');
636
-
637
- // Add labels
638
- nodes.append('text')
639
- .attr('dy', '.35em')
640
- .attr('x', d => d.children ? -13 : 13)
641
- .style('text-anchor', d => d.children ? 'end' : 'start')
642
- .style('font-size', '12px')
643
- .style('font-family', 'Arial, sans-serif')
644
- .text(d => d.data.name);
645
-
646
- // Add tooltips
647
- nodes.append('title')
648
- .text(d => `${d.data.type}: ${d.data.name}`);
649
-
650
- // Add legend
651
- this.addLegend(container);
652
-
653
- this.updateStatus('Tree visualization rendered successfully', 'green');
654
- }
655
-
656
- getNodeColor(type) {
657
- const colors = {
658
- 'module': '#1f77b4',
659
- 'class': '#ff7f0e',
660
- 'function': '#2ca02c',
661
- 'method': '#d62728',
662
- 'import': '#9467bd',
663
- 'imports': '#8c564b',
664
- 'section': '#e377c2',
665
- 'hook': '#7f7f7f'
666
- };
667
- return colors[type] || '#bcbd22';
668
- }
669
-
670
- // Check if file is a code file
671
- isCodeFile(filename) {
672
- const codeExtensions = [
673
- '.py', '.js', '.jsx', '.ts', '.tsx', '.java', '.cpp', '.c', '.h',
674
- '.cs', '.go', '.rs', '.rb', '.php', '.swift', '.kt', '.scala',
675
- '.r', '.m', '.mm', '.sh', '.bash', '.zsh', '.sql', '.html',
676
- '.css', '.scss', '.sass', '.less', '.xml', '.json', '.yaml', '.yml',
677
- '.md', '.rst', '.txt', '.log', '.conf', '.ini', '.toml'
678
- ];
679
- const ext = filename.toLowerCase().substring(filename.lastIndexOf('.'));
680
- return codeExtensions.includes(ext);
681
- }
682
-
683
- // Get appropriate icon for file type
684
- getFileIcon(filename) {
685
- const ext = filename.toLowerCase().substring(filename.lastIndexOf('.'));
686
- const iconMap = {
687
- '.py': '🐍',
688
- '.js': '📜',
689
- '.jsx': '⚛️',
690
- '.ts': '📘',
691
- '.tsx': '⚛️',
692
- '.json': '📋',
693
- '.html': '🌐',
694
- '.css': '🎨',
695
- '.md': '📝',
696
- '.yml': '⚙️',
697
- '.yaml': '⚙️',
698
- '.sh': '🔧',
699
- '.go': '🐹',
700
- '.rs': '🦀',
701
- '.java': '☕',
702
- '.rb': '💎',
703
- '.php': '🐘',
704
- '.cpp': '⚙️',
705
- '.c': '⚙️',
706
- '.h': '📄',
707
- '.cs': '💜',
708
- '.swift': '🦉',
709
- '.kt': '🚀',
710
- '.scala': '📈',
711
- '.r': '📊',
712
- '.sql': '🗃️',
713
- '.scss': '🎨',
714
- '.sass': '🎨',
715
- '.less': '🎨',
716
- '.xml': '📑',
717
- '.bash': '🔧',
718
- '.zsh': '🔧',
719
- '.md': '📝',
720
- '.rst': '📝',
721
- '.txt': '📄',
722
- '.log': '📋',
723
- '.conf': '⚙️',
724
- '.ini': '⚙️',
725
- '.toml': '⚙️'
726
- };
727
- return iconMap[ext] || '💻';
728
- }
729
-
730
- // New method to analyze file from directory click
731
- analyzeFileFromPath(filePath) {
732
- console.log('[SimpleCodeView.analyzeFileFromPath] Analyzing file:', filePath);
733
-
734
- // Switch to tree view
735
- this.setView('tree');
736
-
737
- // Set the file path in the input
738
- const fileInput = document.getElementById('file-path-input');
739
- if (fileInput) {
740
- fileInput.value = filePath;
741
- }
742
-
743
- // Trigger analysis
744
- this.analyzeFile();
745
- }
746
-
747
- addLegend(container) {
748
- // Remove existing legend
749
- const existingLegend = container.querySelector('.tree-legend');
750
- if (existingLegend) {
751
- existingLegend.remove();
752
- }
753
-
754
- const legend = document.createElement('div');
755
- legend.className = 'tree-legend';
756
-
757
- const legendTypes = [
758
- { type: 'module', label: 'Module/File' },
759
- { type: 'class', label: 'Class' },
760
- { type: 'function', label: 'Function' },
761
- { type: 'method', label: 'Method' },
762
- { type: 'imports', label: 'Imports' },
763
- { type: 'import', label: 'Import' },
764
- { type: 'section', label: 'Section' },
765
- { type: 'hook', label: 'Hook/Other' }
766
- ];
767
-
768
- legend.innerHTML = `
769
- <strong>Legend</strong><br>
770
- ${legendTypes.map(item => `
771
- <div class="tree-legend-item">
772
- <div class="tree-legend-color" style="background-color: ${this.getNodeColor(item.type)};"></div>
773
- ${item.label}
774
- </div>
775
- `).join('')}
776
- <hr style="margin: 8px 0;">
777
- <div style="font-size: 11px; color: #666;">
778
- Zoom: Mouse wheel<br>
779
- Pan: Click and drag
780
- </div>
781
- `;
782
-
783
- container.appendChild(legend);
784
- }
785
-
786
- /**
787
- * Get initial path from various sources
788
- * @returns {string} Initial path to use
789
- */
790
- getInitialPath() {
791
- // Try to get from working directory manager
792
- if (window.dashboard && window.dashboard.workingDirectoryManager) {
793
- const dir = window.dashboard.workingDirectoryManager.getCurrentWorkingDir();
794
- if (dir) return dir;
795
- }
796
-
797
- // Try to get from working directory element
798
- const workingDirPath = document.getElementById('working-dir-path');
799
- if (workingDirPath && workingDirPath.textContent && workingDirPath.textContent !== 'Loading...') {
800
- return workingDirPath.textContent.trim();
801
- }
802
-
803
- // Try to get from footer
804
- const footerDir = document.getElementById('footer-working-dir');
805
- if (footerDir && footerDir.textContent && footerDir.textContent !== 'Unknown') {
806
- return footerDir.textContent.trim();
807
- }
808
-
809
- // Try to get from recent events
810
- if (window.socketClient && window.socketClient.events) {
811
- const eventsWithDir = window.socketClient.events
812
- .filter(e => e.data && (e.data.working_directory || e.data.cwd || e.data.working_dir))
813
- .reverse();
814
-
815
- if (eventsWithDir.length > 0) {
816
- const recentEvent = eventsWithDir[0];
817
- const dir = recentEvent.data.working_directory ||
818
- recentEvent.data.cwd ||
819
- recentEvent.data.working_dir;
820
- if (dir) return dir;
821
- }
822
- }
823
-
824
- // Default fallback
825
- return '/';
826
- }
827
- }
828
-
829
- // Create global instance
830
- console.log('[code-simple.js] Creating global simpleCodeView instance');
831
- window.simpleCodeView = new SimpleCodeView();
832
-
833
- // Auto-initialize when DOM is ready
834
- if (document.readyState === 'loading') {
835
- console.log('[code-simple.js] DOM still loading, waiting for DOMContentLoaded');
836
- document.addEventListener('DOMContentLoaded', () => {
837
- console.log('[code-simple.js] DOMContentLoaded fired');
838
- const container = document.getElementById('code-container');
839
- if (container) {
840
- window.simpleCodeView.init(container);
841
- } else {
842
- console.error('[code-simple.js] No code-container element found!');
843
- }
844
- });
845
- } else {
846
- console.log('[code-simple.js] DOM already loaded, initializing immediately');
847
- setTimeout(() => {
848
- const container = document.getElementById('code-container');
849
- if (container) {
850
- window.simpleCodeView.init(container);
851
- } else {
852
- console.error('[code-simple.js] No code-container element found!');
853
- }
854
- }, 0);
855
- }
856
-
857
- console.log('[code-simple.js] Script setup complete');