claude-mpm 4.20.3__py3-none-any.whl → 4.25.10__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 (454) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_PM.md +23 -6
  3. claude_mpm/agents/OUTPUT_STYLE.md +3 -48
  4. claude_mpm/agents/PM_INSTRUCTIONS.md +1783 -34
  5. claude_mpm/agents/WORKFLOW.md +75 -2
  6. claude_mpm/agents/base_agent.json +6 -3
  7. claude_mpm/agents/frontmatter_validator.py +1 -1
  8. claude_mpm/agents/templates/api_qa.json +5 -2
  9. claude_mpm/agents/templates/circuit_breakers.md +108 -2
  10. claude_mpm/agents/templates/documentation.json +33 -6
  11. claude_mpm/agents/templates/javascript_engineer_agent.json +380 -0
  12. claude_mpm/agents/templates/php-engineer.json +10 -4
  13. claude_mpm/agents/templates/pm_red_flags.md +89 -19
  14. claude_mpm/agents/templates/project_organizer.json +7 -3
  15. claude_mpm/agents/templates/qa.json +2 -1
  16. claude_mpm/agents/templates/react_engineer.json +1 -0
  17. claude_mpm/agents/templates/research.json +82 -12
  18. claude_mpm/agents/templates/security.json +4 -4
  19. claude_mpm/agents/templates/tauri_engineer.json +274 -0
  20. claude_mpm/agents/templates/ticketing.json +10 -6
  21. claude_mpm/agents/templates/version_control.json +4 -2
  22. claude_mpm/agents/templates/web_qa.json +2 -1
  23. claude_mpm/cli/README.md +253 -0
  24. claude_mpm/cli/__init__.py +11 -1
  25. claude_mpm/cli/commands/aggregate.py +1 -1
  26. claude_mpm/cli/commands/analyze.py +3 -3
  27. claude_mpm/cli/commands/cleanup.py +1 -1
  28. claude_mpm/cli/commands/configure_agent_display.py +4 -4
  29. claude_mpm/cli/commands/debug.py +12 -12
  30. claude_mpm/cli/commands/hook_errors.py +277 -0
  31. claude_mpm/cli/commands/mcp_install_commands.py +1 -1
  32. claude_mpm/cli/commands/mcp_install_commands.py.backup +284 -0
  33. claude_mpm/cli/commands/mpm_init/README.md +365 -0
  34. claude_mpm/cli/commands/mpm_init/__init__.py +73 -0
  35. claude_mpm/cli/commands/mpm_init/core.py +573 -0
  36. claude_mpm/cli/commands/mpm_init/display.py +341 -0
  37. claude_mpm/cli/commands/mpm_init/git_activity.py +427 -0
  38. claude_mpm/cli/commands/mpm_init/modes.py +397 -0
  39. claude_mpm/cli/commands/mpm_init/prompts.py +442 -0
  40. claude_mpm/cli/commands/mpm_init_cli.py +396 -0
  41. claude_mpm/cli/commands/mpm_init_handler.py +67 -1
  42. claude_mpm/cli/commands/run.py +124 -128
  43. claude_mpm/cli/commands/skills.py +522 -34
  44. claude_mpm/cli/executor.py +56 -0
  45. claude_mpm/cli/interactive/agent_wizard.py +5 -5
  46. claude_mpm/cli/parsers/base_parser.py +28 -0
  47. claude_mpm/cli/parsers/mpm_init_parser.py +42 -0
  48. claude_mpm/cli/parsers/skills_parser.py +138 -0
  49. claude_mpm/cli/startup.py +111 -8
  50. claude_mpm/cli/startup_display.py +480 -0
  51. claude_mpm/cli/utils.py +1 -1
  52. claude_mpm/cli_module/commands.py +1 -1
  53. claude_mpm/cli_module/refactoring_guide.md +253 -0
  54. claude_mpm/commands/mpm-help.md +3 -0
  55. claude_mpm/commands/mpm-init.md +19 -3
  56. claude_mpm/commands/mpm-resume.md +372 -0
  57. claude_mpm/commands/mpm-tickets.md +56 -7
  58. claude_mpm/commands/mpm.md +1 -0
  59. claude_mpm/config/agent_capabilities.yaml +658 -0
  60. claude_mpm/config/async_logging_config.yaml +145 -0
  61. claude_mpm/constants.py +12 -0
  62. claude_mpm/core/.claude-mpm/logs/hooks_20250730.log +34 -0
  63. claude_mpm/core/api_validator.py +1 -1
  64. claude_mpm/core/claude_runner.py +14 -1
  65. claude_mpm/core/config.py +8 -0
  66. claude_mpm/core/constants.py +1 -1
  67. claude_mpm/core/framework/processors/metadata_processor.py +1 -1
  68. claude_mpm/core/hook_error_memory.py +381 -0
  69. claude_mpm/core/hook_manager.py +41 -2
  70. claude_mpm/core/interactive_session.py +48 -3
  71. claude_mpm/core/interfaces.py +56 -1
  72. claude_mpm/core/logger.py +3 -1
  73. claude_mpm/core/oneshot_session.py +39 -0
  74. claude_mpm/d2/.gitignore +22 -0
  75. claude_mpm/d2/ARCHITECTURE_COMPARISON.md +273 -0
  76. claude_mpm/d2/FLASK_INTEGRATION.md +156 -0
  77. claude_mpm/d2/IMPLEMENTATION_SUMMARY.md +452 -0
  78. claude_mpm/d2/QUICKSTART.md +186 -0
  79. claude_mpm/d2/README.md +232 -0
  80. claude_mpm/d2/STORE_FIX_SUMMARY.md +167 -0
  81. claude_mpm/d2/SVELTE5_STORES_GUIDE.md +180 -0
  82. claude_mpm/d2/TESTING.md +288 -0
  83. claude_mpm/d2/index.html +118 -0
  84. claude_mpm/d2/package.json +19 -0
  85. claude_mpm/d2/src/App.svelte +110 -0
  86. claude_mpm/d2/src/components/Header.svelte +153 -0
  87. claude_mpm/d2/src/components/MainContent.svelte +74 -0
  88. claude_mpm/d2/src/components/Sidebar.svelte +85 -0
  89. claude_mpm/d2/src/components/tabs/EventsTab.svelte +326 -0
  90. claude_mpm/d2/src/lib/socketio.js +144 -0
  91. claude_mpm/d2/src/main.js +7 -0
  92. claude_mpm/d2/src/stores/events.js +114 -0
  93. claude_mpm/d2/src/stores/socket.js +108 -0
  94. claude_mpm/d2/src/stores/theme.js +65 -0
  95. claude_mpm/d2/svelte.config.js +12 -0
  96. claude_mpm/d2/vite.config.js +15 -0
  97. claude_mpm/dashboard/.claude-mpm/memories/README.md +36 -0
  98. claude_mpm/dashboard/BUILD_NUMBER +1 -0
  99. claude_mpm/dashboard/README.md +121 -0
  100. claude_mpm/dashboard/VERSION +1 -0
  101. claude_mpm/dashboard/react/components/DataInspector/DataInspector.tsx +273 -0
  102. claude_mpm/dashboard/react/components/ErrorBoundary.tsx +75 -0
  103. claude_mpm/dashboard/react/components/EventViewer/EventViewer.tsx +141 -0
  104. claude_mpm/dashboard/react/components/shared/ConnectionStatus.tsx +36 -0
  105. claude_mpm/dashboard/react/components/shared/FilterBar.tsx +89 -0
  106. claude_mpm/dashboard/react/contexts/DashboardContext.tsx +215 -0
  107. claude_mpm/dashboard/react/entries/events.tsx +165 -0
  108. claude_mpm/dashboard/react/hooks/useEvents.ts +191 -0
  109. claude_mpm/dashboard/react/hooks/useSocket.ts +225 -0
  110. claude_mpm/dashboard/static/built/REFACTORING_SUMMARY.md +170 -0
  111. claude_mpm/dashboard/static/built/components/activity-tree.js.map +1 -0
  112. claude_mpm/dashboard/static/built/components/agent-hierarchy.js +101 -101
  113. claude_mpm/dashboard/static/built/components/agent-inference.js.map +1 -0
  114. claude_mpm/dashboard/static/built/components/build-tracker.js +59 -59
  115. claude_mpm/dashboard/static/built/components/code-simple.js +107 -107
  116. claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +29 -29
  117. claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +24 -24
  118. claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +27 -27
  119. claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +25 -25
  120. claude_mpm/dashboard/static/built/components/code-tree.js.map +1 -0
  121. claude_mpm/dashboard/static/built/components/code-viewer.js.map +1 -0
  122. claude_mpm/dashboard/static/built/components/connection-debug.js +101 -101
  123. claude_mpm/dashboard/static/built/components/diff-viewer.js +113 -113
  124. claude_mpm/dashboard/static/built/components/event-processor.js.map +1 -0
  125. claude_mpm/dashboard/static/built/components/event-viewer.js.map +1 -0
  126. claude_mpm/dashboard/static/built/components/export-manager.js.map +1 -0
  127. claude_mpm/dashboard/static/built/components/file-change-tracker.js +57 -57
  128. claude_mpm/dashboard/static/built/components/file-change-viewer.js +74 -74
  129. claude_mpm/dashboard/static/built/components/file-tool-tracker.js.map +1 -0
  130. claude_mpm/dashboard/static/built/components/file-viewer.js.map +1 -0
  131. claude_mpm/dashboard/static/built/components/hud-library-loader.js.map +1 -0
  132. claude_mpm/dashboard/static/built/components/hud-manager.js.map +1 -0
  133. claude_mpm/dashboard/static/built/components/hud-visualizer.js.map +1 -0
  134. claude_mpm/dashboard/static/built/components/module-viewer.js.map +1 -0
  135. claude_mpm/dashboard/static/built/components/session-manager.js.map +1 -0
  136. claude_mpm/dashboard/static/built/components/socket-manager.js.map +1 -0
  137. claude_mpm/dashboard/static/built/components/ui-state-manager.js.map +1 -0
  138. claude_mpm/dashboard/static/built/components/unified-data-viewer.js.map +1 -0
  139. claude_mpm/dashboard/static/built/components/working-directory.js.map +1 -0
  140. claude_mpm/dashboard/static/built/connection-manager.js +76 -76
  141. claude_mpm/dashboard/static/built/dashboard.js.map +1 -0
  142. claude_mpm/dashboard/static/built/extension-error-handler.js +22 -22
  143. claude_mpm/dashboard/static/built/react/events.js.map +1 -0
  144. claude_mpm/dashboard/static/built/shared/dom-helpers.js +9 -9
  145. claude_mpm/dashboard/static/built/shared/event-bus.js +5 -5
  146. claude_mpm/dashboard/static/built/shared/logger.js +16 -16
  147. claude_mpm/dashboard/static/built/shared/tooltip-service.js +6 -6
  148. claude_mpm/dashboard/static/built/socket-client.js.map +1 -0
  149. claude_mpm/dashboard/static/css/activity.css +69 -69
  150. claude_mpm/dashboard/static/css/connection-status.css +10 -10
  151. claude_mpm/dashboard/static/css/dashboard.css +15 -15
  152. claude_mpm/dashboard/static/index.html +22 -22
  153. claude_mpm/dashboard/static/js/REFACTORING_SUMMARY.md +170 -0
  154. claude_mpm/dashboard/static/js/components/activity-tree.js +178 -178
  155. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +101 -101
  156. claude_mpm/dashboard/static/js/components/agent-inference.js +31 -31
  157. claude_mpm/dashboard/static/js/components/build-tracker.js +59 -59
  158. claude_mpm/dashboard/static/js/components/code-simple.js +107 -107
  159. claude_mpm/dashboard/static/js/components/connection-debug.js +101 -101
  160. claude_mpm/dashboard/static/js/components/diff-viewer.js +113 -113
  161. claude_mpm/dashboard/static/js/components/event-viewer.js +12 -12
  162. claude_mpm/dashboard/static/js/components/file-change-tracker.js +57 -57
  163. claude_mpm/dashboard/static/js/components/file-change-viewer.js +74 -74
  164. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +6 -6
  165. claude_mpm/dashboard/static/js/components/file-viewer.js +42 -42
  166. claude_mpm/dashboard/static/js/components/module-viewer.js +27 -27
  167. claude_mpm/dashboard/static/js/components/session-manager.js +14 -14
  168. claude_mpm/dashboard/static/js/components/socket-manager.js +1 -1
  169. claude_mpm/dashboard/static/js/components/ui-state-manager.js +14 -14
  170. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +110 -110
  171. claude_mpm/dashboard/static/js/components/working-directory.js +8 -8
  172. claude_mpm/dashboard/static/js/connection-manager.js +76 -76
  173. claude_mpm/dashboard/static/js/dashboard.js +76 -58
  174. claude_mpm/dashboard/static/js/extension-error-handler.js +22 -22
  175. claude_mpm/dashboard/static/js/shared/dom-helpers.js +9 -9
  176. claude_mpm/dashboard/static/js/shared/event-bus.js +5 -5
  177. claude_mpm/dashboard/static/js/shared/logger.js +16 -16
  178. claude_mpm/dashboard/static/js/shared/tooltip-service.js +6 -6
  179. claude_mpm/dashboard/static/js/socket-client.js +138 -121
  180. claude_mpm/dashboard/static/navigation-test-results.md +118 -0
  181. claude_mpm/dashboard/static/production/main.html +21 -21
  182. claude_mpm/dashboard/static/test-archive/dashboard.html +22 -22
  183. claude_mpm/dashboard/templates/.claude-mpm/memories/README.md +36 -0
  184. claude_mpm/dashboard/templates/.claude-mpm/memories/engineer_agent.md +39 -0
  185. claude_mpm/dashboard/templates/.claude-mpm/memories/version_control_agent.md +38 -0
  186. claude_mpm/dashboard/templates/code_simple.html +23 -23
  187. claude_mpm/dashboard/templates/index.html +18 -18
  188. claude_mpm/hooks/README.md +143 -0
  189. claude_mpm/hooks/claude_hooks/event_handlers.py +3 -1
  190. claude_mpm/hooks/claude_hooks/hook_handler.py +24 -7
  191. claude_mpm/hooks/claude_hooks/installer.py +45 -0
  192. claude_mpm/hooks/templates/README.md +180 -0
  193. claude_mpm/hooks/templates/pre_tool_use_simple.py +78 -0
  194. claude_mpm/hooks/templates/pre_tool_use_template.py +323 -0
  195. claude_mpm/hooks/templates/settings.json.example +147 -0
  196. claude_mpm/schemas/agent_schema.json +596 -0
  197. claude_mpm/schemas/frontmatter_schema.json +165 -0
  198. claude_mpm/scripts/claude-hook-handler.sh +3 -3
  199. claude_mpm/scripts/start_activity_logging.py +3 -1
  200. claude_mpm/services/agents/deployment/agent_format_converter.py +1 -1
  201. claude_mpm/services/agents/deployment/agent_metrics_collector.py +3 -3
  202. claude_mpm/services/agents/deployment/facade/deployment_facade.py +3 -3
  203. claude_mpm/services/agents/deployment/pipeline/pipeline_executor.py +2 -2
  204. claude_mpm/services/agents/loading/framework_agent_loader.py +8 -8
  205. claude_mpm/services/agents/local_template_manager.py +3 -1
  206. claude_mpm/services/cli/session_pause_manager.py +504 -0
  207. claude_mpm/services/cli/session_resume_helper.py +36 -16
  208. claude_mpm/services/cli/unified_dashboard_manager.py +1 -1
  209. claude_mpm/services/core/base.py +26 -11
  210. claude_mpm/services/core/interfaces.py +56 -1
  211. claude_mpm/services/core/models/agent_config.py +3 -0
  212. claude_mpm/services/core/models/process.py +4 -0
  213. claude_mpm/services/diagnostics/checks/agent_check.py +0 -2
  214. claude_mpm/services/diagnostics/checks/instructions_check.py +1 -2
  215. claude_mpm/services/diagnostics/checks/mcp_check.py +0 -1
  216. claude_mpm/services/diagnostics/checks/monitor_check.py +0 -1
  217. claude_mpm/services/diagnostics/doctor_reporter.py +6 -4
  218. claude_mpm/services/diagnostics/models.py +21 -0
  219. claude_mpm/services/event_bus/README.md +244 -0
  220. claude_mpm/services/event_bus/direct_relay.py +3 -3
  221. claude_mpm/services/event_bus/event_bus.py +36 -3
  222. claude_mpm/services/event_bus/relay.py +23 -7
  223. claude_mpm/services/events/README.md +303 -0
  224. claude_mpm/services/events/consumers/logging.py +1 -2
  225. claude_mpm/services/framework_claude_md_generator/README.md +119 -0
  226. claude_mpm/services/infrastructure/monitoring/resources.py +1 -1
  227. claude_mpm/services/local_ops/__init__.py +2 -0
  228. claude_mpm/services/local_ops/process_manager.py +1 -1
  229. claude_mpm/services/local_ops/resource_monitor.py +2 -2
  230. claude_mpm/services/mcp_gateway/README.md +185 -0
  231. claude_mpm/services/mcp_gateway/auto_configure.py +31 -25
  232. claude_mpm/services/mcp_gateway/config/configuration.py +1 -1
  233. claude_mpm/services/mcp_gateway/core/process_pool.py +19 -10
  234. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -2
  235. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +1 -1
  236. claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +26 -21
  237. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +6 -2
  238. claude_mpm/services/memory/failure_tracker.py +19 -4
  239. claude_mpm/services/memory/optimizer.py +1 -1
  240. claude_mpm/services/model/model_router.py +8 -9
  241. claude_mpm/services/monitor/daemon.py +1 -1
  242. claude_mpm/services/monitor/server.py +2 -2
  243. claude_mpm/services/native_agent_converter.py +356 -0
  244. claude_mpm/services/port_manager.py +1 -1
  245. claude_mpm/services/project/documentation_manager.py +2 -1
  246. claude_mpm/services/project/toolchain_analyzer.py +3 -1
  247. claude_mpm/services/runner_configuration_service.py +1 -0
  248. claude_mpm/services/self_upgrade_service.py +165 -7
  249. claude_mpm/services/skills_config.py +547 -0
  250. claude_mpm/services/skills_deployer.py +955 -0
  251. claude_mpm/services/socketio/handlers/connection.py +1 -1
  252. claude_mpm/services/socketio/handlers/connection.py.backup +217 -0
  253. claude_mpm/services/socketio/handlers/git.py +2 -2
  254. claude_mpm/services/socketio/handlers/hook.py.backup +154 -0
  255. claude_mpm/services/static/.gitkeep +2 -0
  256. claude_mpm/services/system_instructions_service.py +1 -3
  257. claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +0 -3
  258. claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +0 -1
  259. claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +1 -1
  260. claude_mpm/services/version_control/VERSION +1 -0
  261. claude_mpm/services/version_control/conflict_resolution.py +6 -4
  262. claude_mpm/services/visualization/mermaid_generator.py +2 -3
  263. claude_mpm/skills/__init__.py +3 -3
  264. claude_mpm/skills/agent_skills_injector.py +42 -49
  265. claude_mpm/skills/bundled/.gitkeep +2 -0
  266. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +4 -0
  267. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +108 -114
  268. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
  269. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
  270. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
  271. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
  272. claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
  273. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +46 -41
  274. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
  275. claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
  276. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +36 -73
  277. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
  278. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
  279. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +100 -125
  280. claude_mpm/skills/bundled/debugging/root-cause-tracing/find-polluter.sh +63 -0
  281. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
  282. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
  283. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
  284. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
  285. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +28 -72
  286. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +11 -0
  287. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
  288. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
  289. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +272 -0
  290. claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
  291. claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
  292. claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
  293. claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
  294. claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
  295. claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
  296. claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
  297. claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
  298. claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
  299. claude_mpm/skills/bundled/infrastructure/env-manager/scripts/validate_env.py +576 -0
  300. claude_mpm/skills/bundled/main/artifacts-builder/LICENSE.txt +202 -0
  301. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +13 -1
  302. claude_mpm/skills/bundled/main/artifacts-builder/scripts/bundle-artifact.sh +54 -0
  303. claude_mpm/skills/bundled/main/artifacts-builder/scripts/init-artifact.sh +322 -0
  304. claude_mpm/skills/bundled/main/artifacts-builder/scripts/shadcn-components.tar.gz +0 -0
  305. claude_mpm/skills/bundled/main/internal-comms/LICENSE.txt +202 -0
  306. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +11 -0
  307. claude_mpm/skills/bundled/main/mcp-builder/LICENSE.txt +202 -0
  308. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +109 -277
  309. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
  310. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
  311. claude_mpm/skills/bundled/main/mcp-builder/scripts/connections.py +17 -10
  312. claude_mpm/skills/bundled/main/mcp-builder/scripts/evaluation.py +92 -39
  313. claude_mpm/skills/bundled/main/mcp-builder/scripts/example_evaluation.xml +22 -0
  314. claude_mpm/skills/bundled/main/mcp-builder/scripts/requirements.txt +2 -0
  315. claude_mpm/skills/bundled/main/skill-creator/LICENSE.txt +202 -0
  316. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +135 -155
  317. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
  318. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
  319. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
  320. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
  321. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
  322. claude_mpm/skills/bundled/main/skill-creator/scripts/init_skill.py +13 -12
  323. claude_mpm/skills/bundled/main/skill-creator/scripts/package_skill.py +5 -3
  324. claude_mpm/skills/bundled/main/skill-creator/scripts/quick_validate.py +19 -12
  325. claude_mpm/skills/bundled/performance-profiling.md +6 -0
  326. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
  327. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
  328. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
  329. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
  330. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
  331. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
  332. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
  333. claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
  334. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
  335. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
  336. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
  337. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
  338. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
  339. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
  340. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
  341. claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
  342. claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
  343. claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
  344. claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
  345. claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
  346. claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
  347. claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
  348. claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
  349. claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
  350. claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
  351. claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
  352. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +21 -25
  353. claude_mpm/skills/bundled/testing/condition-based-waiting/example.ts +158 -0
  354. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
  355. claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
  356. claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
  357. claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
  358. claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
  359. claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
  360. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +86 -250
  361. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
  362. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
  363. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
  364. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
  365. claude_mpm/skills/bundled/testing/webapp-testing/LICENSE.txt +202 -0
  366. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +145 -57
  367. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
  368. claude_mpm/skills/bundled/testing/webapp-testing/examples/console_logging.py +6 -6
  369. claude_mpm/skills/bundled/testing/webapp-testing/examples/element_discovery.py +13 -9
  370. claude_mpm/skills/bundled/testing/webapp-testing/examples/static_html_automation.py +8 -8
  371. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
  372. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
  373. claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +37 -15
  374. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
  375. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
  376. claude_mpm/skills/skills_registry.py +44 -48
  377. claude_mpm/skills/skills_service.py +117 -108
  378. claude_mpm/templates/questions/EXAMPLES.md +501 -0
  379. claude_mpm/templates/questions/__init__.py +43 -0
  380. claude_mpm/templates/questions/base.py +193 -0
  381. claude_mpm/templates/questions/pr_strategy.py +314 -0
  382. claude_mpm/templates/questions/project_init.py +388 -0
  383. claude_mpm/templates/questions/ticket_mgmt.py +397 -0
  384. claude_mpm/tools/README_SOCKETIO_DEBUG.md +224 -0
  385. claude_mpm/tools/__main__.py +8 -8
  386. claude_mpm/tools/code_tree_analyzer/README.md +64 -0
  387. claude_mpm/tools/code_tree_analyzer/__init__.py +45 -0
  388. claude_mpm/tools/code_tree_analyzer/analysis.py +299 -0
  389. claude_mpm/tools/code_tree_analyzer/cache.py +131 -0
  390. claude_mpm/tools/code_tree_analyzer/core.py +380 -0
  391. claude_mpm/tools/code_tree_analyzer/discovery.py +403 -0
  392. claude_mpm/tools/code_tree_analyzer/events.py +168 -0
  393. claude_mpm/tools/code_tree_analyzer/gitignore.py +308 -0
  394. claude_mpm/tools/code_tree_analyzer/models.py +39 -0
  395. claude_mpm/tools/code_tree_analyzer/multilang_analyzer.py +224 -0
  396. claude_mpm/tools/code_tree_analyzer/python_analyzer.py +284 -0
  397. claude_mpm/utils/agent_dependency_loader.py +3 -3
  398. claude_mpm/utils/dependency_cache.py +3 -1
  399. claude_mpm/utils/gitignore.py +241 -0
  400. claude_mpm/utils/log_cleanup.py +3 -3
  401. claude_mpm/utils/robust_installer.py +3 -5
  402. claude_mpm/utils/structured_questions.py +619 -0
  403. {claude_mpm-4.20.3.dist-info → claude_mpm-4.25.10.dist-info}/METADATA +218 -31
  404. {claude_mpm-4.20.3.dist-info → claude_mpm-4.25.10.dist-info}/RECORD +409 -246
  405. claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -17
  406. claude_mpm/agents/templates/.claude-mpm/memories/engineer_memories.md +0 -3
  407. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250826_014258_728.md +0 -39
  408. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250901_010124_142.md +0 -400
  409. claude_mpm/cli/commands/mpm_init.py +0 -2093
  410. claude_mpm/dashboard/.claude-mpm/socketio-instances.json +0 -1
  411. claude_mpm/dashboard/static/archive/activity_dashboard_test.html +0 -61
  412. claude_mpm/dashboard/static/archive/test_activity_connection.html +0 -179
  413. claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +0 -68
  414. claude_mpm/dashboard/static/archive/test_dashboard.html +0 -409
  415. claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +0 -519
  416. claude_mpm/dashboard/static/archive/test_dashboard_verification.html +0 -181
  417. claude_mpm/dashboard/static/archive/test_file_data.html +0 -315
  418. claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +0 -243
  419. claude_mpm/dashboard/static/archive/test_file_tree_fix.html +0 -234
  420. claude_mpm/dashboard/static/archive/test_file_tree_rename.html +0 -117
  421. claude_mpm/dashboard/static/archive/test_file_tree_tab.html +0 -115
  422. claude_mpm/dashboard/static/archive/test_file_viewer.html +0 -224
  423. claude_mpm/dashboard/static/archive/test_final_activity.html +0 -220
  424. claude_mpm/dashboard/static/archive/test_tab_fix.html +0 -139
  425. claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +0 -1
  426. claude_mpm/dashboard/static/dist/components/activity-tree.js +0 -2
  427. claude_mpm/dashboard/static/dist/components/agent-inference.js +0 -2
  428. claude_mpm/dashboard/static/dist/components/code-tree.js +0 -2
  429. claude_mpm/dashboard/static/dist/components/code-viewer.js +0 -2
  430. claude_mpm/dashboard/static/dist/components/event-processor.js +0 -2
  431. claude_mpm/dashboard/static/dist/components/event-viewer.js +0 -2
  432. claude_mpm/dashboard/static/dist/components/export-manager.js +0 -2
  433. claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +0 -2
  434. claude_mpm/dashboard/static/dist/components/file-viewer.js +0 -2
  435. claude_mpm/dashboard/static/dist/components/hud-library-loader.js +0 -2
  436. claude_mpm/dashboard/static/dist/components/hud-manager.js +0 -2
  437. claude_mpm/dashboard/static/dist/components/hud-visualizer.js +0 -2
  438. claude_mpm/dashboard/static/dist/components/module-viewer.js +0 -2
  439. claude_mpm/dashboard/static/dist/components/session-manager.js +0 -2
  440. claude_mpm/dashboard/static/dist/components/socket-manager.js +0 -2
  441. claude_mpm/dashboard/static/dist/components/ui-state-manager.js +0 -2
  442. claude_mpm/dashboard/static/dist/components/unified-data-viewer.js +0 -2
  443. claude_mpm/dashboard/static/dist/components/working-directory.js +0 -2
  444. claude_mpm/dashboard/static/dist/dashboard.js +0 -2
  445. claude_mpm/dashboard/static/dist/react/events.js +0 -30
  446. claude_mpm/dashboard/static/dist/socket-client.js +0 -2
  447. claude_mpm/dashboard/static/test-archive/test_debug.html +0 -25
  448. claude_mpm/skills/bundled/debugging/verification-before-completion/references/common-failures.md +0 -213
  449. claude_mpm/tools/code_tree_analyzer.py +0 -1825
  450. /claude_mpm/skills/bundled/collaboration/requesting-code-review/{code-reviewer.md → references/code-reviewer-template.md} +0 -0
  451. {claude_mpm-4.20.3.dist-info → claude_mpm-4.25.10.dist-info}/WHEEL +0 -0
  452. {claude_mpm-4.20.3.dist-info → claude_mpm-4.25.10.dist-info}/entry_points.txt +0 -0
  453. {claude_mpm-4.20.3.dist-info → claude_mpm-4.25.10.dist-info}/licenses/LICENSE +0 -0
  454. {claude_mpm-4.20.3.dist-info → claude_mpm-4.25.10.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,576 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Environment variable validation script.
4
+
5
+ This script validates .env files for:
6
+ - Structure (valid key-value format, no duplicates, proper quoting)
7
+ - Completeness (compare with .env.example)
8
+ - Naming conventions (UPPERCASE_WITH_UNDERSCORES)
9
+ - Framework-specific rules (Next.js, Express, Flask, etc.)
10
+
11
+ Security: NEVER logs actual secret values.
12
+ """
13
+
14
+ import argparse
15
+ import json
16
+ import re
17
+ import sys
18
+ from pathlib import Path
19
+ from typing import Dict, List, Optional
20
+
21
+
22
+ class ValidationError:
23
+ """Represents a validation error."""
24
+
25
+ def __init__(
26
+ self, line: Optional[int], key: str, message: str, severity: str = "error"
27
+ ):
28
+ self.line = line
29
+ self.key = key
30
+ self.message = message
31
+ self.severity = severity # "error" or "warning"
32
+
33
+ def to_dict(self) -> Dict:
34
+ return {
35
+ "line": self.line,
36
+ "key": self.key,
37
+ "message": self.message,
38
+ "severity": self.severity,
39
+ }
40
+
41
+ def __str__(self) -> str:
42
+ prefix = "❌" if self.severity == "error" else "⚠️ "
43
+ line_info = f"Line {self.line}: " if self.line else ""
44
+ return f"{prefix} {line_info}{self.key}: {self.message}"
45
+
46
+
47
+ class EnvValidator:
48
+ """Validates environment variable files."""
49
+
50
+ # Valid variable name pattern: UPPERCASE_WITH_UNDERSCORES
51
+ VALID_NAME_PATTERN = re.compile(r"^[A-Z][A-Z0-9_]*$")
52
+
53
+ # Framework-specific prefixes
54
+ FRAMEWORK_PREFIXES = {
55
+ "nextjs": ["NEXT_PUBLIC_", "NEXT_"],
56
+ "vite": ["VITE_"],
57
+ "react": ["REACT_APP_"],
58
+ }
59
+
60
+ # Secret indicators (variables that should never be in NEXT_PUBLIC_, etc.)
61
+ SECRET_INDICATORS = ["secret", "key", "password", "token", "private", "api_key"]
62
+
63
+ def __init__(self, framework: Optional[str] = None, strict: bool = False):
64
+ self.framework = framework
65
+ self.strict = strict
66
+ self.errors: List[ValidationError] = []
67
+ self.warnings: List[ValidationError] = []
68
+
69
+ def parse_env_file(self, env_file: Path) -> Dict[str, str]:
70
+ """Parse .env file safely. Returns dict of key-value pairs."""
71
+ vars_dict = {}
72
+
73
+ with open(env_file) as f:
74
+ for line_num, line in enumerate(f, 1):
75
+ line = line.strip()
76
+
77
+ # Skip empty lines and comments
78
+ if not line or line.startswith("#"):
79
+ continue
80
+
81
+ # Check for valid format
82
+ if "=" not in line:
83
+ self.errors.append(
84
+ ValidationError(
85
+ line_num,
86
+ "",
87
+ f"Invalid format (missing =): {line[:50]}",
88
+ "error",
89
+ )
90
+ )
91
+ continue
92
+
93
+ # Split on first = only
94
+ key, value = line.split("=", 1)
95
+ key = key.strip()
96
+
97
+ # Remove quotes from value
98
+ value = value.strip()
99
+ if value and value[0] == value[-1] and value[0] in ('"', "'"):
100
+ value = value[1:-1]
101
+
102
+ vars_dict[key] = value
103
+
104
+ return vars_dict
105
+
106
+ def validate_structure(self, env_file: Path) -> List[ValidationError]:
107
+ """Validate basic file structure."""
108
+ errors = []
109
+
110
+ with open(env_file) as f:
111
+ for line_num, line in enumerate(f, 1):
112
+ line = line.strip()
113
+
114
+ # Skip empty lines and comments
115
+ if not line or line.startswith("#"):
116
+ continue
117
+
118
+ # Check for inline comments (warning)
119
+ if "#" in line and not line.startswith("#"):
120
+ key_value = line.split("#", 1)[0]
121
+ if "=" in key_value:
122
+ key, value = key_value.split("=", 1)
123
+ # Check if # is inside quotes
124
+ value = value.strip()
125
+ if not (value.startswith('"') or value.startswith("'")):
126
+ errors.append(
127
+ ValidationError(
128
+ line_num,
129
+ key.strip(),
130
+ "Possible inline comment (not all parsers support this)",
131
+ "warning",
132
+ )
133
+ )
134
+
135
+ # Check format
136
+ if "=" not in line:
137
+ errors.append(
138
+ ValidationError(
139
+ line_num,
140
+ "",
141
+ f"Invalid format (missing =): {line[:50]}",
142
+ "error",
143
+ )
144
+ )
145
+ continue
146
+
147
+ key, value = line.split("=", 1)
148
+ key = key.strip()
149
+
150
+ # Validate key name
151
+ if not self.VALID_NAME_PATTERN.match(key):
152
+ errors.append(
153
+ ValidationError(
154
+ line_num,
155
+ key,
156
+ "Invalid naming (use UPPERCASE_WITH_UNDERSCORES)",
157
+ "error",
158
+ )
159
+ )
160
+
161
+ # Check for spaces without quotes
162
+ value = value.strip()
163
+ if value and " " in value:
164
+ if not (value.startswith('"') or value.startswith("'")):
165
+ errors.append(
166
+ ValidationError(
167
+ line_num,
168
+ key,
169
+ "Value with spaces should be quoted",
170
+ "warning",
171
+ )
172
+ )
173
+
174
+ return errors
175
+
176
+ def check_duplicates(self, env_file: Path) -> Dict[str, List[int]]:
177
+ """Find duplicate keys and their line numbers."""
178
+ keys: Dict[str, List[int]] = {}
179
+
180
+ with open(env_file) as f:
181
+ for line_num, line in enumerate(f, 1):
182
+ line = line.strip()
183
+ if not line or line.startswith("#"):
184
+ continue
185
+
186
+ if "=" in line:
187
+ key = line.split("=", 1)[0].strip()
188
+ if key in keys:
189
+ keys[key].append(line_num)
190
+ else:
191
+ keys[key] = [line_num]
192
+
193
+ # Return only duplicates
194
+ duplicates = {k: v for k, v in keys.items() if len(v) > 1}
195
+
196
+ # Add errors for duplicates
197
+ for key, lines in duplicates.items():
198
+ self.errors.append(
199
+ ValidationError(
200
+ None,
201
+ key,
202
+ f"Duplicate key found on lines: {', '.join(map(str, lines))}",
203
+ "error",
204
+ )
205
+ )
206
+
207
+ return duplicates
208
+
209
+ def compare_env_files(self, env_file: Path, example_file: Path) -> Dict:
210
+ """Compare .env against .env.example."""
211
+ if not example_file.exists():
212
+ self.warnings.append(
213
+ ValidationError(
214
+ None, "", f".env.example not found: {example_file}", "warning"
215
+ )
216
+ )
217
+ return {"missing": set(), "extra": set(), "common": set()}
218
+
219
+ env_vars = set(self.parse_env_file(env_file).keys())
220
+ example_vars = set(self.parse_env_file(example_file).keys())
221
+
222
+ missing = example_vars - env_vars
223
+ extra = env_vars - example_vars
224
+ common = env_vars & example_vars
225
+
226
+ # Add errors for missing required variables
227
+ for var in missing:
228
+ self.errors.append(
229
+ ValidationError(
230
+ None,
231
+ var,
232
+ "Required variable missing (defined in .env.example)",
233
+ "error",
234
+ )
235
+ )
236
+
237
+ # Add warnings for undocumented variables
238
+ for var in extra:
239
+ self.warnings.append(
240
+ ValidationError(
241
+ None, var, "Variable not documented in .env.example", "warning"
242
+ )
243
+ )
244
+
245
+ return {"missing": missing, "extra": extra, "common": common}
246
+
247
+ def validate_framework_specific(self, env_file: Path) -> List[ValidationError]:
248
+ """Validate framework-specific rules."""
249
+ errors = []
250
+ vars_dict = self.parse_env_file(env_file)
251
+
252
+ if self.framework == "nextjs":
253
+ errors.extend(self._validate_nextjs(vars_dict))
254
+ elif self.framework == "vite":
255
+ errors.extend(self._validate_vite(vars_dict))
256
+ elif self.framework == "react":
257
+ errors.extend(self._validate_react(vars_dict))
258
+ elif self.framework == "nodejs":
259
+ errors.extend(self._validate_nodejs(vars_dict))
260
+ elif self.framework == "flask":
261
+ errors.extend(self._validate_flask(vars_dict))
262
+
263
+ return errors
264
+
265
+ def _validate_nextjs(self, vars_dict: Dict[str, str]) -> List[ValidationError]:
266
+ """Validate Next.js environment variables."""
267
+ errors = []
268
+
269
+ for key, value in vars_dict.items():
270
+ # Check for secrets in NEXT_PUBLIC_ vars
271
+ if key.startswith("NEXT_PUBLIC_"):
272
+ if any(
273
+ indicator in key.lower() for indicator in self.SECRET_INDICATORS
274
+ ):
275
+ errors.append(
276
+ ValidationError(
277
+ None,
278
+ key,
279
+ "SECURITY: Secret in NEXT_PUBLIC_ variable (exposed to browser)",
280
+ "error",
281
+ )
282
+ )
283
+
284
+ # Check for API URLs without NEXT_PUBLIC_ prefix
285
+ if "api" in key.lower() and "url" in key.lower():
286
+ if not key.startswith("NEXT_PUBLIC_") and not key.endswith("_SECRET"):
287
+ errors.append(
288
+ ValidationError(
289
+ None,
290
+ key,
291
+ "API URL without NEXT_PUBLIC_ prefix (not accessible client-side)",
292
+ "warning",
293
+ )
294
+ )
295
+
296
+ return errors
297
+
298
+ def _validate_vite(self, vars_dict: Dict[str, str]) -> List[ValidationError]:
299
+ """Validate Vite environment variables."""
300
+ errors = []
301
+
302
+ for key in vars_dict:
303
+ # Check for secrets in VITE_ vars
304
+ if key.startswith("VITE_"):
305
+ if any(
306
+ indicator in key.lower() for indicator in self.SECRET_INDICATORS
307
+ ):
308
+ errors.append(
309
+ ValidationError(
310
+ None,
311
+ key,
312
+ "SECURITY: Secret in VITE_ variable (exposed to browser)",
313
+ "error",
314
+ )
315
+ )
316
+
317
+ # Warn about non-VITE_ vars
318
+ elif key not in ["NODE_ENV", "PORT"]:
319
+ errors.append(
320
+ ValidationError(
321
+ None,
322
+ key,
323
+ "Variable not prefixed with VITE_ (not accessible in client code)",
324
+ "warning",
325
+ )
326
+ )
327
+
328
+ return errors
329
+
330
+ def _validate_react(self, vars_dict: Dict[str, str]) -> List[ValidationError]:
331
+ """Validate Create React App environment variables."""
332
+ errors = []
333
+
334
+ for key in vars_dict:
335
+ # Check for secrets in REACT_APP_ vars
336
+ if key.startswith("REACT_APP_"):
337
+ if any(
338
+ indicator in key.lower() for indicator in self.SECRET_INDICATORS
339
+ ):
340
+ errors.append(
341
+ ValidationError(
342
+ None,
343
+ key,
344
+ "SECURITY: Secret in REACT_APP_ variable (exposed to browser)",
345
+ "error",
346
+ )
347
+ )
348
+
349
+ return errors
350
+
351
+ def _validate_nodejs(self, vars_dict: Dict[str, str]) -> List[ValidationError]:
352
+ """Validate Node.js environment variables."""
353
+ errors = []
354
+
355
+ # Check NODE_ENV value
356
+ if "NODE_ENV" in vars_dict:
357
+ valid_values = ["development", "production", "test"]
358
+ if vars_dict["NODE_ENV"] not in valid_values:
359
+ # SECURITY: Never expose actual variable values in error messages
360
+ # to prevent accidental secret leakage in logs/CI output
361
+ errors.append(
362
+ ValidationError(
363
+ None,
364
+ "NODE_ENV",
365
+ f"Invalid value for NODE_ENV, expected one of {valid_values}",
366
+ "error",
367
+ )
368
+ )
369
+
370
+ # Check PORT is numeric
371
+ if "PORT" in vars_dict:
372
+ try:
373
+ port = int(vars_dict["PORT"])
374
+ if not (1 <= port <= 65535):
375
+ errors.append(
376
+ ValidationError(
377
+ None, "PORT", "PORT must be between 1 and 65535", "error"
378
+ )
379
+ )
380
+ except ValueError:
381
+ errors.append(
382
+ ValidationError(None, "PORT", "PORT must be numeric", "error")
383
+ )
384
+
385
+ return errors
386
+
387
+ def _validate_flask(self, vars_dict: Dict[str, str]) -> List[ValidationError]:
388
+ """Validate Flask environment variables."""
389
+ errors = []
390
+
391
+ # Check required vars
392
+ required = ["FLASK_APP", "SECRET_KEY"]
393
+ for var in required:
394
+ if var not in vars_dict:
395
+ errors.append(
396
+ ValidationError(
397
+ None, var, "Required Flask variable missing", "error"
398
+ )
399
+ )
400
+
401
+ # Check FLASK_APP ends with .py
402
+ if "FLASK_APP" in vars_dict:
403
+ if not vars_dict["FLASK_APP"].endswith(".py"):
404
+ errors.append(
405
+ ValidationError(
406
+ None,
407
+ "FLASK_APP",
408
+ "FLASK_APP should point to a .py file",
409
+ "warning",
410
+ )
411
+ )
412
+
413
+ # Check FLASK_ENV value
414
+ if "FLASK_ENV" in vars_dict:
415
+ valid_values = ["development", "production"]
416
+ if vars_dict["FLASK_ENV"] not in valid_values:
417
+ errors.append(
418
+ ValidationError(
419
+ None,
420
+ "FLASK_ENV",
421
+ f"Invalid value, expected one of {valid_values}",
422
+ "error",
423
+ )
424
+ )
425
+
426
+ return errors
427
+
428
+ def validate(self, env_file: Path, example_file: Optional[Path] = None) -> Dict:
429
+ """Run all validations. Returns summary dict."""
430
+ self.errors = []
431
+ self.warnings = []
432
+
433
+ # 1. Structure validation
434
+ structure_errors = self.validate_structure(env_file)
435
+ self.errors.extend([e for e in structure_errors if e.severity == "error"])
436
+ self.warnings.extend([e for e in structure_errors if e.severity == "warning"])
437
+
438
+ # 2. Check duplicates
439
+ self.check_duplicates(env_file)
440
+
441
+ # 3. Compare with .env.example
442
+ if example_file:
443
+ self.compare_env_files(env_file, example_file)
444
+
445
+ # 4. Framework-specific validation
446
+ if self.framework:
447
+ framework_errors = self.validate_framework_specific(env_file)
448
+ self.errors.extend([e for e in framework_errors if e.severity == "error"])
449
+ self.warnings.extend(
450
+ [e for e in framework_errors if e.severity == "warning"]
451
+ )
452
+
453
+ return {
454
+ "valid": len(self.errors) == 0,
455
+ "errors": self.errors,
456
+ "warnings": self.warnings,
457
+ "error_count": len(self.errors),
458
+ "warning_count": len(self.warnings),
459
+ }
460
+
461
+
462
+ def generate_example(env_file: Path, output_file: Path) -> None:
463
+ """Generate .env.example from .env file (with values removed)."""
464
+ with open(env_file) as f:
465
+ lines = f.readlines()
466
+
467
+ with open(output_file, "w") as f:
468
+ for line in lines:
469
+ line = line.strip()
470
+
471
+ # Keep comments and empty lines
472
+ if not line or line.startswith("#"):
473
+ f.write(line + "\n")
474
+ continue
475
+
476
+ # Replace values with placeholders
477
+ if "=" in line:
478
+ key, _ = line.split("=", 1)
479
+ f.write(f"{key}=your-{key.lower().replace('_', '-')}-here\n")
480
+
481
+ print(f"✅ Generated {output_file}")
482
+
483
+
484
+ def main():
485
+ parser = argparse.ArgumentParser(description="Validate environment variable files")
486
+ parser.add_argument("file", type=Path, help="Path to .env file to validate")
487
+ parser.add_argument(
488
+ "--compare-with",
489
+ type=Path,
490
+ help="Compare with .env.example file",
491
+ metavar="FILE",
492
+ )
493
+ parser.add_argument(
494
+ "--framework",
495
+ choices=["nextjs", "vite", "react", "nodejs", "flask", "generic"],
496
+ help="Framework-specific validation",
497
+ )
498
+ parser.add_argument(
499
+ "--strict", action="store_true", help="Treat warnings as errors"
500
+ )
501
+ parser.add_argument("--json", action="store_true", help="Output results as JSON")
502
+ parser.add_argument(
503
+ "--quiet", action="store_true", help="Only show errors, not warnings"
504
+ )
505
+ parser.add_argument(
506
+ "--generate-example",
507
+ type=Path,
508
+ help="Generate .env.example file",
509
+ metavar="OUTPUT",
510
+ )
511
+
512
+ args = parser.parse_args()
513
+
514
+ # Check input file exists
515
+ if not args.file.exists():
516
+ print(f"❌ Error: File not found: {args.file}")
517
+ sys.exit(2)
518
+
519
+ # Generate example if requested
520
+ if args.generate_example:
521
+ generate_example(args.file, args.generate_example)
522
+ return
523
+
524
+ # Run validation
525
+ validator = EnvValidator(framework=args.framework, strict=args.strict)
526
+ result = validator.validate(args.file, example_file=args.compare_with)
527
+
528
+ # Output results
529
+ if args.json:
530
+ output = {
531
+ "valid": result["valid"],
532
+ "file": str(args.file),
533
+ "errors": [e.to_dict() for e in result["errors"]],
534
+ "warnings": [w.to_dict() for w in result["warnings"]],
535
+ }
536
+ print(json.dumps(output, indent=2))
537
+ else:
538
+ print(f"\n🔍 Validating: {args.file}")
539
+ if args.framework:
540
+ print(f"📦 Framework: {args.framework}")
541
+ print()
542
+
543
+ # Show errors
544
+ if result["errors"]:
545
+ print("❌ Errors:")
546
+ for error in result["errors"]:
547
+ print(f" {error}")
548
+ print()
549
+
550
+ # Show warnings (unless quiet)
551
+ if result["warnings"] and not args.quiet:
552
+ print("⚠️ Warnings:")
553
+ for warning in result["warnings"]:
554
+ print(f" {warning}")
555
+ print()
556
+
557
+ # Summary
558
+ if result["valid"]:
559
+ print("✅ Validation passed!")
560
+ else:
561
+ print(f"❌ Validation failed: {result['error_count']} error(s)")
562
+
563
+ if not args.quiet:
564
+ print(
565
+ f"📊 Summary: {result['error_count']} errors, {result['warning_count']} warnings"
566
+ )
567
+
568
+ # Exit code
569
+ if result["errors"] or (args.strict and result["warnings"]):
570
+ sys.exit(1)
571
+ else:
572
+ sys.exit(0)
573
+
574
+
575
+ if __name__ == "__main__":
576
+ main()