claude-mpm 4.21.3__py3-none-any.whl → 5.0.2__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (484) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_PM.md +12 -0
  3. claude_mpm/agents/OUTPUT_STYLE.md +3 -48
  4. claude_mpm/agents/PM_INSTRUCTIONS.md +632 -334
  5. claude_mpm/agents/WORKFLOW.md +75 -2
  6. claude_mpm/agents/__init__.py +6 -0
  7. claude_mpm/agents/agent_loader.py +1 -4
  8. claude_mpm/agents/base_agent.json +6 -3
  9. claude_mpm/agents/frontmatter_validator.py +1 -1
  10. claude_mpm/agents/templates/{circuit_breakers.md → circuit-breakers.md} +370 -3
  11. claude_mpm/agents/templates/context-management-examples.md +544 -0
  12. claude_mpm/agents/templates/{pm_red_flags.md → pm-red-flags.md} +89 -19
  13. claude_mpm/agents/templates/pr-workflow-examples.md +427 -0
  14. claude_mpm/agents/templates/research-gate-examples.md +669 -0
  15. claude_mpm/agents/templates/structured-questions-examples.md +615 -0
  16. claude_mpm/agents/templates/ticket-completeness-examples.md +139 -0
  17. claude_mpm/agents/templates/ticketing-examples.md +277 -0
  18. claude_mpm/cli/__init__.py +38 -2
  19. claude_mpm/cli/commands/agent_source.py +774 -0
  20. claude_mpm/cli/commands/agent_state_manager.py +125 -20
  21. claude_mpm/cli/commands/agents.py +684 -13
  22. claude_mpm/cli/commands/agents_cleanup.py +210 -0
  23. claude_mpm/cli/commands/agents_discover.py +338 -0
  24. claude_mpm/cli/commands/aggregate.py +1 -1
  25. claude_mpm/cli/commands/analyze.py +3 -3
  26. claude_mpm/cli/commands/auto_configure.py +2 -6
  27. claude_mpm/cli/commands/cleanup.py +1 -1
  28. claude_mpm/cli/commands/config.py +7 -4
  29. claude_mpm/cli/commands/configure.py +478 -44
  30. claude_mpm/cli/commands/configure_agent_display.py +4 -4
  31. claude_mpm/cli/commands/configure_navigation.py +63 -46
  32. claude_mpm/cli/commands/debug.py +12 -12
  33. claude_mpm/cli/commands/doctor.py +10 -2
  34. claude_mpm/cli/commands/hook_errors.py +277 -0
  35. claude_mpm/cli/commands/local_deploy.py +1 -4
  36. claude_mpm/cli/commands/mcp_install_commands.py +1 -1
  37. claude_mpm/cli/commands/mpm_init/core.py +50 -2
  38. claude_mpm/cli/commands/mpm_init/git_activity.py +10 -10
  39. claude_mpm/cli/commands/mpm_init/prompts.py +6 -6
  40. claude_mpm/cli/commands/run.py +124 -128
  41. claude_mpm/cli/commands/skill_source.py +694 -0
  42. claude_mpm/cli/commands/skills.py +435 -1
  43. claude_mpm/cli/executor.py +78 -3
  44. claude_mpm/cli/interactive/agent_wizard.py +919 -41
  45. claude_mpm/cli/parsers/agent_source_parser.py +171 -0
  46. claude_mpm/cli/parsers/agents_parser.py +173 -4
  47. claude_mpm/cli/parsers/base_parser.py +49 -0
  48. claude_mpm/cli/parsers/config_parser.py +96 -43
  49. claude_mpm/cli/parsers/skill_source_parser.py +169 -0
  50. claude_mpm/cli/parsers/skills_parser.py +138 -0
  51. claude_mpm/cli/parsers/source_parser.py +138 -0
  52. claude_mpm/cli/startup.py +499 -84
  53. claude_mpm/cli/startup_display.py +480 -0
  54. claude_mpm/cli/utils.py +1 -1
  55. claude_mpm/cli_module/commands.py +1 -1
  56. claude_mpm/commands/{mpm-auto-configure.md → mpm-agents-auto-configure.md} +9 -0
  57. claude_mpm/commands/mpm-agents-detect.md +9 -0
  58. claude_mpm/commands/{mpm-agents.md → mpm-agents-list.md} +9 -0
  59. claude_mpm/commands/mpm-agents-recommend.md +9 -0
  60. claude_mpm/commands/{mpm-config.md → mpm-config-view.md} +9 -0
  61. claude_mpm/commands/mpm-doctor.md +9 -0
  62. claude_mpm/commands/mpm-help.md +11 -2
  63. claude_mpm/commands/mpm-init.md +27 -2
  64. claude_mpm/commands/mpm-monitor.md +9 -0
  65. claude_mpm/commands/{mpm-resume.md → mpm-session-resume.md} +9 -0
  66. claude_mpm/commands/mpm-status.md +9 -0
  67. claude_mpm/commands/{mpm-organize.md → mpm-ticket-organize.md} +9 -0
  68. claude_mpm/commands/mpm-ticket-view.md +552 -0
  69. claude_mpm/commands/mpm-version.md +9 -0
  70. claude_mpm/commands/mpm.md +10 -0
  71. claude_mpm/config/agent_presets.py +258 -0
  72. claude_mpm/config/agent_sources.py +325 -0
  73. claude_mpm/config/skill_sources.py +590 -0
  74. claude_mpm/constants.py +12 -0
  75. claude_mpm/core/api_validator.py +1 -1
  76. claude_mpm/core/claude_runner.py +17 -10
  77. claude_mpm/core/config.py +24 -0
  78. claude_mpm/core/constants.py +1 -1
  79. claude_mpm/core/framework/__init__.py +3 -16
  80. claude_mpm/core/framework/loaders/instruction_loader.py +25 -5
  81. claude_mpm/core/framework/processors/metadata_processor.py +1 -1
  82. claude_mpm/core/hook_error_memory.py +381 -0
  83. claude_mpm/core/hook_manager.py +41 -2
  84. claude_mpm/core/interactive_session.py +112 -5
  85. claude_mpm/core/logger.py +3 -1
  86. claude_mpm/core/oneshot_session.py +94 -4
  87. claude_mpm/dashboard/static/css/activity.css +69 -69
  88. claude_mpm/dashboard/static/css/connection-status.css +10 -10
  89. claude_mpm/dashboard/static/css/dashboard.css +15 -15
  90. claude_mpm/dashboard/static/js/components/activity-tree.js +178 -178
  91. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +101 -101
  92. claude_mpm/dashboard/static/js/components/agent-inference.js +31 -31
  93. claude_mpm/dashboard/static/js/components/build-tracker.js +59 -59
  94. claude_mpm/dashboard/static/js/components/code-simple.js +107 -107
  95. claude_mpm/dashboard/static/js/components/connection-debug.js +101 -101
  96. claude_mpm/dashboard/static/js/components/diff-viewer.js +113 -113
  97. claude_mpm/dashboard/static/js/components/event-viewer.js +12 -12
  98. claude_mpm/dashboard/static/js/components/file-change-tracker.js +57 -57
  99. claude_mpm/dashboard/static/js/components/file-change-viewer.js +74 -74
  100. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +6 -6
  101. claude_mpm/dashboard/static/js/components/file-viewer.js +42 -42
  102. claude_mpm/dashboard/static/js/components/module-viewer.js +27 -27
  103. claude_mpm/dashboard/static/js/components/session-manager.js +14 -14
  104. claude_mpm/dashboard/static/js/components/socket-manager.js +1 -1
  105. claude_mpm/dashboard/static/js/components/ui-state-manager.js +14 -14
  106. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +110 -110
  107. claude_mpm/dashboard/static/js/components/working-directory.js +8 -8
  108. claude_mpm/dashboard/static/js/connection-manager.js +76 -76
  109. claude_mpm/dashboard/static/js/dashboard.js +76 -58
  110. claude_mpm/dashboard/static/js/extension-error-handler.js +22 -22
  111. claude_mpm/dashboard/static/js/socket-client.js +138 -121
  112. claude_mpm/dashboard/templates/code_simple.html +23 -23
  113. claude_mpm/dashboard/templates/index.html +18 -18
  114. claude_mpm/experimental/cli_enhancements.py +1 -5
  115. claude_mpm/hooks/claude_hooks/event_handlers.py +3 -1
  116. claude_mpm/hooks/claude_hooks/hook_handler.py +24 -7
  117. claude_mpm/hooks/claude_hooks/installer.py +45 -0
  118. claude_mpm/hooks/failure_learning/__init__.py +2 -8
  119. claude_mpm/hooks/failure_learning/failure_detection_hook.py +1 -6
  120. claude_mpm/hooks/failure_learning/fix_detection_hook.py +1 -6
  121. claude_mpm/hooks/failure_learning/learning_extraction_hook.py +1 -6
  122. claude_mpm/hooks/kuzu_response_hook.py +1 -5
  123. claude_mpm/hooks/templates/pre_tool_use_simple.py +78 -0
  124. claude_mpm/hooks/templates/pre_tool_use_template.py +323 -0
  125. claude_mpm/models/git_repository.py +198 -0
  126. claude_mpm/scripts/claude-hook-handler.sh +3 -3
  127. claude_mpm/scripts/start_activity_logging.py +3 -1
  128. claude_mpm/services/agents/agent_builder.py +45 -9
  129. claude_mpm/services/agents/agent_preset_service.py +238 -0
  130. claude_mpm/services/agents/agent_selection_service.py +484 -0
  131. claude_mpm/services/agents/auto_deploy_index_parser.py +569 -0
  132. claude_mpm/services/agents/deployment/agent_deployment.py +126 -2
  133. claude_mpm/services/agents/deployment/agent_discovery_service.py +105 -73
  134. claude_mpm/services/agents/deployment/agent_format_converter.py +1 -1
  135. claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +1 -5
  136. claude_mpm/services/agents/deployment/agent_metrics_collector.py +3 -3
  137. claude_mpm/services/agents/deployment/agent_restore_handler.py +1 -4
  138. claude_mpm/services/agents/deployment/agent_template_builder.py +236 -15
  139. claude_mpm/services/agents/deployment/agents_directory_resolver.py +101 -15
  140. claude_mpm/services/agents/deployment/async_agent_deployment.py +2 -1
  141. claude_mpm/services/agents/deployment/facade/deployment_facade.py +3 -3
  142. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +115 -15
  143. claude_mpm/services/agents/deployment/pipeline/pipeline_executor.py +2 -2
  144. claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +1 -4
  145. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +363 -0
  146. claude_mpm/services/agents/deployment/single_agent_deployer.py +2 -2
  147. claude_mpm/services/agents/deployment/system_instructions_deployer.py +168 -46
  148. claude_mpm/services/agents/deployment/validation/deployment_validator.py +2 -2
  149. claude_mpm/services/agents/git_source_manager.py +629 -0
  150. claude_mpm/services/agents/loading/framework_agent_loader.py +9 -12
  151. claude_mpm/services/agents/local_template_manager.py +50 -10
  152. claude_mpm/services/agents/single_tier_deployment_service.py +696 -0
  153. claude_mpm/services/agents/sources/__init__.py +13 -0
  154. claude_mpm/services/agents/sources/agent_sync_state.py +516 -0
  155. claude_mpm/services/agents/sources/git_source_sync_service.py +1055 -0
  156. claude_mpm/services/agents/startup_sync.py +239 -0
  157. claude_mpm/services/agents/toolchain_detector.py +474 -0
  158. claude_mpm/services/cli/session_pause_manager.py +1 -1
  159. claude_mpm/services/cli/unified_dashboard_manager.py +1 -1
  160. claude_mpm/services/command_deployment_service.py +92 -1
  161. claude_mpm/services/core/interfaces/__init__.py +1 -3
  162. claude_mpm/services/core/interfaces/health.py +1 -4
  163. claude_mpm/services/core/models/__init__.py +2 -11
  164. claude_mpm/services/diagnostics/checks/__init__.py +4 -0
  165. claude_mpm/services/diagnostics/checks/agent_check.py +0 -2
  166. claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
  167. claude_mpm/services/diagnostics/checks/instructions_check.py +1 -2
  168. claude_mpm/services/diagnostics/checks/mcp_check.py +0 -1
  169. claude_mpm/services/diagnostics/checks/monitor_check.py +0 -1
  170. claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
  171. claude_mpm/services/diagnostics/diagnostic_runner.py +9 -0
  172. claude_mpm/services/diagnostics/doctor_reporter.py +40 -10
  173. claude_mpm/services/event_bus/direct_relay.py +3 -3
  174. claude_mpm/services/event_bus/event_bus.py +36 -3
  175. claude_mpm/services/events/consumers/logging.py +1 -2
  176. claude_mpm/services/git/__init__.py +21 -0
  177. claude_mpm/services/git/git_operations_service.py +494 -0
  178. claude_mpm/services/github/__init__.py +21 -0
  179. claude_mpm/services/github/github_cli_service.py +397 -0
  180. claude_mpm/services/infrastructure/monitoring/__init__.py +1 -5
  181. claude_mpm/services/infrastructure/monitoring/aggregator.py +1 -6
  182. claude_mpm/services/infrastructure/monitoring/resources.py +1 -1
  183. claude_mpm/services/instructions/__init__.py +9 -0
  184. claude_mpm/services/instructions/instruction_cache_service.py +374 -0
  185. claude_mpm/services/local_ops/__init__.py +3 -13
  186. claude_mpm/services/local_ops/health_checks/__init__.py +1 -3
  187. claude_mpm/services/local_ops/health_manager.py +1 -4
  188. claude_mpm/services/local_ops/process_manager.py +1 -1
  189. claude_mpm/services/local_ops/resource_monitor.py +2 -2
  190. claude_mpm/services/mcp_gateway/config/configuration.py +1 -1
  191. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +1 -6
  192. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -2
  193. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +1 -1
  194. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +6 -2
  195. claude_mpm/services/memory/optimizer.py +1 -1
  196. claude_mpm/services/model/model_router.py +8 -9
  197. claude_mpm/services/monitor/daemon.py +1 -1
  198. claude_mpm/services/monitor/server.py +2 -2
  199. claude_mpm/services/native_agent_converter.py +356 -0
  200. claude_mpm/services/port_manager.py +1 -1
  201. claude_mpm/services/pr/__init__.py +14 -0
  202. claude_mpm/services/pr/pr_template_service.py +329 -0
  203. claude_mpm/services/project/documentation_manager.py +2 -1
  204. claude_mpm/services/project/toolchain_analyzer.py +3 -1
  205. claude_mpm/services/runner_configuration_service.py +1 -0
  206. claude_mpm/services/self_upgrade_service.py +165 -7
  207. claude_mpm/services/skills/__init__.py +18 -0
  208. claude_mpm/services/skills/git_skill_source_manager.py +1169 -0
  209. claude_mpm/services/skills/skill_discovery_service.py +568 -0
  210. claude_mpm/services/skills_config.py +547 -0
  211. claude_mpm/services/skills_deployer.py +955 -0
  212. claude_mpm/services/socketio/handlers/connection.py +1 -1
  213. claude_mpm/services/socketio/handlers/git.py +2 -2
  214. claude_mpm/services/socketio/server/core.py +1 -4
  215. claude_mpm/services/socketio/server/main.py +1 -3
  216. claude_mpm/services/system_instructions_service.py +1 -3
  217. claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +0 -3
  218. claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +0 -1
  219. claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +1 -1
  220. claude_mpm/services/unified/deployment_strategies/vercel.py +1 -5
  221. claude_mpm/services/unified/unified_deployment.py +1 -5
  222. claude_mpm/services/version_control/conflict_resolution.py +6 -4
  223. claude_mpm/services/visualization/__init__.py +1 -5
  224. claude_mpm/services/visualization/mermaid_generator.py +2 -3
  225. claude_mpm/skills/bundled/infrastructure/env-manager/scripts/validate_env.py +576 -0
  226. claude_mpm/skills/bundled/performance-profiling.md +6 -0
  227. claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +2 -2
  228. claude_mpm/skills/skills_registry.py +0 -1
  229. claude_mpm/templates/questions/__init__.py +38 -0
  230. claude_mpm/templates/questions/base.py +193 -0
  231. claude_mpm/templates/questions/pr_strategy.py +311 -0
  232. claude_mpm/templates/questions/project_init.py +385 -0
  233. claude_mpm/templates/questions/ticket_mgmt.py +394 -0
  234. claude_mpm/tools/__main__.py +8 -8
  235. claude_mpm/tools/code_tree_analyzer/analysis.py +1 -1
  236. claude_mpm/utils/agent_dependency_loader.py +80 -13
  237. claude_mpm/utils/dependency_cache.py +3 -1
  238. claude_mpm/utils/gitignore.py +241 -0
  239. claude_mpm/utils/log_cleanup.py +3 -3
  240. claude_mpm/utils/progress.py +383 -0
  241. claude_mpm/utils/robust_installer.py +3 -5
  242. claude_mpm/utils/structured_questions.py +619 -0
  243. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/METADATA +429 -59
  244. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/RECORD +252 -425
  245. claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -17
  246. claude_mpm/agents/templates/.claude-mpm/memories/engineer_memories.md +0 -3
  247. claude_mpm/agents/templates/agent-manager.json +0 -273
  248. claude_mpm/agents/templates/agentic-coder-optimizer.json +0 -248
  249. claude_mpm/agents/templates/api_qa.json +0 -180
  250. claude_mpm/agents/templates/clerk-ops.json +0 -235
  251. claude_mpm/agents/templates/code_analyzer.json +0 -101
  252. claude_mpm/agents/templates/content-agent.json +0 -358
  253. claude_mpm/agents/templates/dart_engineer.json +0 -307
  254. claude_mpm/agents/templates/data_engineer.json +0 -225
  255. claude_mpm/agents/templates/documentation.json +0 -211
  256. claude_mpm/agents/templates/engineer.json +0 -210
  257. claude_mpm/agents/templates/gcp_ops_agent.json +0 -253
  258. claude_mpm/agents/templates/golang_engineer.json +0 -270
  259. claude_mpm/agents/templates/imagemagick.json +0 -264
  260. claude_mpm/agents/templates/java_engineer.json +0 -346
  261. claude_mpm/agents/templates/local_ops_agent.json +0 -1840
  262. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250826_014258_728.md +0 -39
  263. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250901_010124_142.md +0 -400
  264. claude_mpm/agents/templates/memory_manager.json +0 -158
  265. claude_mpm/agents/templates/nextjs_engineer.json +0 -285
  266. claude_mpm/agents/templates/ops.json +0 -185
  267. claude_mpm/agents/templates/php-engineer.json +0 -287
  268. claude_mpm/agents/templates/product_owner.json +0 -338
  269. claude_mpm/agents/templates/project_organizer.json +0 -140
  270. claude_mpm/agents/templates/prompt-engineer.json +0 -737
  271. claude_mpm/agents/templates/python_engineer.json +0 -387
  272. claude_mpm/agents/templates/qa.json +0 -242
  273. claude_mpm/agents/templates/react_engineer.json +0 -238
  274. claude_mpm/agents/templates/refactoring_engineer.json +0 -276
  275. claude_mpm/agents/templates/research.json +0 -188
  276. claude_mpm/agents/templates/ruby-engineer.json +0 -280
  277. claude_mpm/agents/templates/rust_engineer.json +0 -275
  278. claude_mpm/agents/templates/security.json +0 -202
  279. claude_mpm/agents/templates/svelte-engineer.json +0 -225
  280. claude_mpm/agents/templates/ticketing.json +0 -177
  281. claude_mpm/agents/templates/typescript_engineer.json +0 -285
  282. claude_mpm/agents/templates/vercel_ops_agent.json +0 -412
  283. claude_mpm/agents/templates/version_control.json +0 -157
  284. claude_mpm/agents/templates/web_qa.json +0 -399
  285. claude_mpm/agents/templates/web_ui.json +0 -189
  286. claude_mpm/commands/mpm-tickets.md +0 -102
  287. claude_mpm/dashboard/.claude-mpm/socketio-instances.json +0 -1
  288. claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +0 -188
  289. claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +0 -156
  290. claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +0 -38
  291. claude_mpm/dashboard/react/components/shared/FilterBar.module.css +0 -92
  292. claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +0 -248
  293. claude_mpm/dashboard/static/archive/activity_dashboard_test.html +0 -61
  294. claude_mpm/dashboard/static/archive/test_activity_connection.html +0 -179
  295. claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +0 -68
  296. claude_mpm/dashboard/static/archive/test_dashboard.html +0 -409
  297. claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +0 -519
  298. claude_mpm/dashboard/static/archive/test_dashboard_verification.html +0 -181
  299. claude_mpm/dashboard/static/archive/test_file_data.html +0 -315
  300. claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +0 -243
  301. claude_mpm/dashboard/static/archive/test_file_tree_fix.html +0 -234
  302. claude_mpm/dashboard/static/archive/test_file_tree_rename.html +0 -117
  303. claude_mpm/dashboard/static/archive/test_file_tree_tab.html +0 -115
  304. claude_mpm/dashboard/static/archive/test_file_viewer.html +0 -224
  305. claude_mpm/dashboard/static/archive/test_final_activity.html +0 -220
  306. claude_mpm/dashboard/static/archive/test_tab_fix.html +0 -139
  307. claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +0 -1
  308. claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
  309. claude_mpm/dashboard/static/built/components/agent-hierarchy.js +0 -777
  310. claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
  311. claude_mpm/dashboard/static/built/components/build-tracker.js +0 -333
  312. claude_mpm/dashboard/static/built/components/code-simple.js +0 -857
  313. claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +0 -353
  314. claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +0 -235
  315. claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +0 -409
  316. claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +0 -435
  317. claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
  318. claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
  319. claude_mpm/dashboard/static/built/components/connection-debug.js +0 -654
  320. claude_mpm/dashboard/static/built/components/diff-viewer.js +0 -891
  321. claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
  322. claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
  323. claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
  324. claude_mpm/dashboard/static/built/components/file-change-tracker.js +0 -443
  325. claude_mpm/dashboard/static/built/components/file-change-viewer.js +0 -690
  326. claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
  327. claude_mpm/dashboard/static/built/components/file-viewer.js +0 -2
  328. claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
  329. claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
  330. claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
  331. claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
  332. claude_mpm/dashboard/static/built/components/nav-bar.js +0 -145
  333. claude_mpm/dashboard/static/built/components/page-structure.js +0 -429
  334. claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
  335. claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
  336. claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
  337. claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
  338. claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
  339. claude_mpm/dashboard/static/built/connection-manager.js +0 -536
  340. claude_mpm/dashboard/static/built/dashboard.js +0 -2
  341. claude_mpm/dashboard/static/built/extension-error-handler.js +0 -164
  342. claude_mpm/dashboard/static/built/react/events.js +0 -30
  343. claude_mpm/dashboard/static/built/shared/dom-helpers.js +0 -396
  344. claude_mpm/dashboard/static/built/shared/event-bus.js +0 -330
  345. claude_mpm/dashboard/static/built/shared/event-filter-service.js +0 -540
  346. claude_mpm/dashboard/static/built/shared/logger.js +0 -385
  347. claude_mpm/dashboard/static/built/shared/page-structure.js +0 -249
  348. claude_mpm/dashboard/static/built/shared/tooltip-service.js +0 -253
  349. claude_mpm/dashboard/static/built/socket-client.js +0 -2
  350. claude_mpm/dashboard/static/built/tab-isolation-fix.js +0 -185
  351. claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +0 -1
  352. claude_mpm/dashboard/static/dist/components/activity-tree.js +0 -2
  353. claude_mpm/dashboard/static/dist/components/agent-inference.js +0 -2
  354. claude_mpm/dashboard/static/dist/components/code-tree.js +0 -2
  355. claude_mpm/dashboard/static/dist/components/code-viewer.js +0 -2
  356. claude_mpm/dashboard/static/dist/components/event-processor.js +0 -2
  357. claude_mpm/dashboard/static/dist/components/event-viewer.js +0 -2
  358. claude_mpm/dashboard/static/dist/components/export-manager.js +0 -2
  359. claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +0 -2
  360. claude_mpm/dashboard/static/dist/components/file-viewer.js +0 -2
  361. claude_mpm/dashboard/static/dist/components/hud-library-loader.js +0 -2
  362. claude_mpm/dashboard/static/dist/components/hud-manager.js +0 -2
  363. claude_mpm/dashboard/static/dist/components/hud-visualizer.js +0 -2
  364. claude_mpm/dashboard/static/dist/components/module-viewer.js +0 -2
  365. claude_mpm/dashboard/static/dist/components/session-manager.js +0 -2
  366. claude_mpm/dashboard/static/dist/components/socket-manager.js +0 -2
  367. claude_mpm/dashboard/static/dist/components/ui-state-manager.js +0 -2
  368. claude_mpm/dashboard/static/dist/components/unified-data-viewer.js +0 -2
  369. claude_mpm/dashboard/static/dist/components/working-directory.js +0 -2
  370. claude_mpm/dashboard/static/dist/dashboard.js +0 -2
  371. claude_mpm/dashboard/static/dist/react/events.js +0 -30
  372. claude_mpm/dashboard/static/dist/socket-client.js +0 -2
  373. claude_mpm/dashboard/static/events.html +0 -607
  374. claude_mpm/dashboard/static/index.html +0 -635
  375. claude_mpm/dashboard/static/js/shared/dom-helpers.js +0 -396
  376. claude_mpm/dashboard/static/js/shared/event-bus.js +0 -330
  377. claude_mpm/dashboard/static/js/shared/logger.js +0 -385
  378. claude_mpm/dashboard/static/js/shared/tooltip-service.js +0 -253
  379. claude_mpm/dashboard/static/js/stores/dashboard-store.js +0 -562
  380. claude_mpm/dashboard/static/legacy/activity.html +0 -736
  381. claude_mpm/dashboard/static/legacy/agents.html +0 -786
  382. claude_mpm/dashboard/static/legacy/files.html +0 -747
  383. claude_mpm/dashboard/static/legacy/tools.html +0 -831
  384. claude_mpm/dashboard/static/monitors.html +0 -431
  385. claude_mpm/dashboard/static/production/events.html +0 -659
  386. claude_mpm/dashboard/static/production/main.html +0 -698
  387. claude_mpm/dashboard/static/production/monitors.html +0 -483
  388. claude_mpm/dashboard/static/test-archive/dashboard.html +0 -635
  389. claude_mpm/dashboard/static/test-archive/debug-events.html +0 -147
  390. claude_mpm/dashboard/static/test-archive/test-navigation.html +0 -256
  391. claude_mpm/dashboard/static/test-archive/test-react-exports.html +0 -180
  392. claude_mpm/dashboard/static/test-archive/test_debug.html +0 -25
  393. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +0 -79
  394. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +0 -178
  395. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +0 -577
  396. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +0 -467
  397. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +0 -537
  398. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +0 -730
  399. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +0 -112
  400. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +0 -146
  401. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +0 -412
  402. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +0 -81
  403. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +0 -362
  404. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +0 -312
  405. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +0 -152
  406. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +0 -668
  407. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +0 -587
  408. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +0 -438
  409. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +0 -391
  410. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +0 -119
  411. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +0 -148
  412. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +0 -483
  413. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +0 -452
  414. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +0 -449
  415. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +0 -411
  416. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +0 -14
  417. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +0 -58
  418. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +0 -68
  419. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +0 -69
  420. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +0 -131
  421. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +0 -325
  422. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +0 -490
  423. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +0 -425
  424. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +0 -499
  425. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +0 -86
  426. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +0 -43
  427. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +0 -47
  428. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +0 -65
  429. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +0 -30
  430. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +0 -16
  431. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +0 -160
  432. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +0 -412
  433. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +0 -602
  434. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +0 -915
  435. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +0 -916
  436. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +0 -752
  437. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +0 -1237
  438. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +0 -189
  439. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +0 -500
  440. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +0 -464
  441. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +0 -619
  442. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +0 -437
  443. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +0 -231
  444. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +0 -170
  445. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +0 -602
  446. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +0 -821
  447. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +0 -742
  448. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +0 -726
  449. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +0 -764
  450. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +0 -831
  451. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +0 -226
  452. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +0 -901
  453. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +0 -901
  454. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +0 -775
  455. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +0 -937
  456. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +0 -770
  457. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +0 -961
  458. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +0 -119
  459. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +0 -253
  460. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +0 -145
  461. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +0 -543
  462. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +0 -741
  463. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +0 -470
  464. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +0 -458
  465. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +0 -639
  466. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +0 -140
  467. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +0 -572
  468. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +0 -411
  469. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +0 -569
  470. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +0 -695
  471. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +0 -184
  472. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +0 -459
  473. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +0 -479
  474. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +0 -687
  475. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +0 -758
  476. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +0 -868
  477. /claude_mpm/agents/templates/{git_file_tracking.md → git-file-tracking.md} +0 -0
  478. /claude_mpm/agents/templates/{pm_examples.md → pm-examples.md} +0 -0
  479. /claude_mpm/agents/templates/{response_format.md → response-format.md} +0 -0
  480. /claude_mpm/agents/templates/{validation_templates.md → validation-templates.md} +0 -0
  481. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/WHEEL +0 -0
  482. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/entry_points.txt +0 -0
  483. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/licenses/LICENSE +0 -0
  484. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/top_level.txt +0 -0
@@ -1,741 +0,0 @@
1
- # Real-World TDD Examples
2
-
3
- > **Part of**: [Test-Driven Development](../SKILL.md)
4
- > **Category**: testing
5
- > **Reading Level**: Intermediate
6
-
7
- ## Purpose
8
-
9
- Real-world scenarios demonstrating test-driven development in action, with complete RED/GREEN/REFACTOR cycles and step-by-step walkthroughs.
10
-
11
- ## Example 1: Form Validation
12
-
13
- ### Scenario
14
- Build email validation for user registration form.
15
-
16
- ### Iteration 1: Basic Email Validation
17
-
18
- **RED - Write Failing Test:**
19
- ```typescript
20
- describe('Email Validation', () => {
21
- test('accepts valid email format', () => {
22
- const result = validateEmail('user@example.com');
23
- expect(result.valid).toBe(true);
24
- });
25
- });
26
- ```
27
-
28
- **VERIFY RED - Run Test:**
29
- ```bash
30
- $ npm test
31
- FAIL: accepts valid email format
32
- ReferenceError: validateEmail is not defined
33
- ```
34
- ✓ Fails correctly - function doesn't exist
35
-
36
- **GREEN - Minimal Implementation:**
37
- ```typescript
38
- interface ValidationResult {
39
- valid: boolean;
40
- error?: string;
41
- }
42
-
43
- function validateEmail(email: string): ValidationResult {
44
- return { valid: true }; // Simplest thing that passes
45
- }
46
- ```
47
-
48
- **VERIFY GREEN - Run Test:**
49
- ```bash
50
- $ npm test
51
- PASS: accepts valid email format
52
- ```
53
- ✓ Test passes
54
-
55
- ### Iteration 2: Reject Invalid Format
56
-
57
- **RED:**
58
- ```typescript
59
- test('rejects email without @ symbol', () => {
60
- const result = validateEmail('userexample.com');
61
- expect(result.valid).toBe(false);
62
- expect(result.error).toBe('Invalid email format');
63
- });
64
- ```
65
-
66
- **VERIFY RED:**
67
- ```bash
68
- $ npm test
69
- FAIL: rejects email without @ symbol
70
- Expected valid: false, Received: true
71
- ```
72
- ✓ Fails correctly
73
-
74
- **GREEN:**
75
- ```typescript
76
- function validateEmail(email: string): ValidationResult {
77
- if (!email.includes('@')) {
78
- return { valid: false, error: 'Invalid email format' };
79
- }
80
- return { valid: true };
81
- }
82
- ```
83
-
84
- **VERIFY GREEN:**
85
- ```bash
86
- $ npm test
87
- PASS: accepts valid email format
88
- PASS: rejects email without @ symbol
89
- ```
90
- ✓ Both tests pass
91
-
92
- ### Iteration 3: Reject Multiple @ Symbols
93
-
94
- **RED:**
95
- ```typescript
96
- test('rejects email with multiple @ symbols', () => {
97
- const result = validateEmail('user@@example.com');
98
- expect(result.valid).toBe(false);
99
- });
100
- ```
101
-
102
- **GREEN:**
103
- ```typescript
104
- function validateEmail(email: string): ValidationResult {
105
- if (!email.includes('@')) {
106
- return { valid: false, error: 'Invalid email format' };
107
- }
108
- if (email.indexOf('@') !== email.lastIndexOf('@')) {
109
- return { valid: false, error: 'Invalid email format' };
110
- }
111
- return { valid: true };
112
- }
113
- ```
114
-
115
- **REFACTOR - Clean Up:**
116
- ```typescript
117
- function validateEmail(email: string): ValidationResult {
118
- const hasAtSymbol = email.includes('@');
119
- const hasSingleAtSymbol = email.indexOf('@') === email.lastIndexOf('@');
120
-
121
- if (!hasAtSymbol || !hasSingleAtSymbol) {
122
- return { valid: false, error: 'Invalid email format' };
123
- }
124
-
125
- return { valid: true };
126
- }
127
- ```
128
-
129
- **Time:** 15 minutes, 3 tests, working validation
130
-
131
- ## Example 2: Bug Fix with TDD
132
-
133
- ### Scenario
134
- Bug reported: Empty shopping cart shows "$0" instead of "Cart is empty" message.
135
-
136
- ### Step 1: Reproduce with Test
137
-
138
- **RED - Failing Test:**
139
- ```typescript
140
- describe('Shopping Cart Display', () => {
141
- test('shows empty message when cart has no items', () => {
142
- const cart = new ShoppingCart();
143
- const display = cart.getDisplayMessage();
144
- expect(display).toBe('Cart is empty');
145
- });
146
- });
147
- ```
148
-
149
- **VERIFY RED:**
150
- ```bash
151
- $ npm test
152
- FAIL: shows empty message when cart has no items
153
- Expected: 'Cart is empty'
154
- Received: 'Total: $0'
155
- ```
156
- ✓ Test reproduces the bug
157
-
158
- ### Step 2: Fix with Minimal Change
159
-
160
- **GREEN:**
161
- ```typescript
162
- class ShoppingCart {
163
- private items: Item[] = [];
164
-
165
- getDisplayMessage(): string {
166
- if (this.items.length === 0) {
167
- return 'Cart is empty';
168
- }
169
- return `Total: $${this.getTotal()}`;
170
- }
171
-
172
- getTotal(): number {
173
- return this.items.reduce((sum, item) => sum + item.price, 0);
174
- }
175
- }
176
- ```
177
-
178
- **VERIFY GREEN:**
179
- ```bash
180
- $ npm test
181
- PASS: shows empty message when cart has no items
182
- PASS: calculates total correctly (existing test)
183
- ```
184
- ✓ Bug fixed, no regressions
185
-
186
- ### Step 3: Prevent Regression
187
-
188
- Test stays in suite permanently - bug can never return without test failing.
189
-
190
- **Time:** 10 minutes, bug fixed with regression protection
191
-
192
- ## Example 3: API Client Development
193
-
194
- ### Scenario
195
- Build HTTP client with retry logic for failed requests.
196
-
197
- ### Iteration 1: Basic Request
198
-
199
- **RED:**
200
- ```typescript
201
- describe('HTTP Client', () => {
202
- test('makes GET request successfully', async () => {
203
- const client = new HttpClient('https://api.example.com');
204
- const response = await client.get('/users/1');
205
- expect(response.status).toBe(200);
206
- expect(response.data).toBeDefined();
207
- });
208
- });
209
- ```
210
-
211
- **GREEN:**
212
- ```typescript
213
- class HttpClient {
214
- constructor(private baseUrl: string) {}
215
-
216
- async get(path: string): Promise<Response> {
217
- const res = await fetch(`${this.baseUrl}${path}`);
218
- return {
219
- status: res.status,
220
- data: await res.json()
221
- };
222
- }
223
- }
224
- ```
225
-
226
- ### Iteration 2: Handle Network Errors
227
-
228
- **RED:**
229
- ```typescript
230
- test('retries on network error', async () => {
231
- const client = new HttpClient('https://api.example.com');
232
-
233
- // Mock fetch to fail once then succeed
234
- let attempts = 0;
235
- global.fetch = jest.fn().mockImplementation(() => {
236
- attempts++;
237
- if (attempts === 1) {
238
- throw new Error('Network error');
239
- }
240
- return Promise.resolve({
241
- status: 200,
242
- json: () => Promise.resolve({ id: 1 })
243
- });
244
- });
245
-
246
- const response = await client.get('/users/1');
247
-
248
- expect(response.status).toBe(200);
249
- expect(attempts).toBe(2);
250
- });
251
- ```
252
-
253
- **GREEN:**
254
- ```typescript
255
- class HttpClient {
256
- constructor(private baseUrl: string) {}
257
-
258
- async get(path: string): Promise<Response> {
259
- try {
260
- return await this.makeRequest(path);
261
- } catch (error) {
262
- // Retry once
263
- return await this.makeRequest(path);
264
- }
265
- }
266
-
267
- private async makeRequest(path: string): Promise<Response> {
268
- const res = await fetch(`${this.baseUrl}${path}`);
269
- return {
270
- status: res.status,
271
- data: await res.json()
272
- };
273
- }
274
- }
275
- ```
276
-
277
- ### Iteration 3: Configurable Retries
278
-
279
- **RED:**
280
- ```typescript
281
- test('retries up to max attempts', async () => {
282
- const client = new HttpClient('https://api.example.com', {
283
- maxRetries: 3
284
- });
285
-
286
- let attempts = 0;
287
- global.fetch = jest.fn().mockImplementation(() => {
288
- attempts++;
289
- if (attempts < 3) {
290
- throw new Error('Network error');
291
- }
292
- return Promise.resolve({
293
- status: 200,
294
- json: () => Promise.resolve({ id: 1 })
295
- });
296
- });
297
-
298
- const response = await client.get('/users/1');
299
-
300
- expect(response.status).toBe(200);
301
- expect(attempts).toBe(3);
302
- });
303
- ```
304
-
305
- **GREEN:**
306
- ```typescript
307
- interface HttpClientOptions {
308
- maxRetries?: number;
309
- }
310
-
311
- class HttpClient {
312
- private maxRetries: number;
313
-
314
- constructor(
315
- private baseUrl: string,
316
- options: HttpClientOptions = {}
317
- ) {
318
- this.maxRetries = options.maxRetries ?? 1;
319
- }
320
-
321
- async get(path: string): Promise<Response> {
322
- let lastError: Error;
323
-
324
- for (let attempt = 0; attempt <= this.maxRetries; attempt++) {
325
- try {
326
- return await this.makeRequest(path);
327
- } catch (error) {
328
- lastError = error as Error;
329
- }
330
- }
331
-
332
- throw lastError!;
333
- }
334
-
335
- private async makeRequest(path: string): Promise<Response> {
336
- const res = await fetch(`${this.baseUrl}${path}`);
337
- return {
338
- status: res.status,
339
- data: await res.json()
340
- };
341
- }
342
- }
343
- ```
344
-
345
- **REFACTOR:**
346
- ```typescript
347
- // Extract retry logic to utility
348
- async function withRetry<T>(
349
- fn: () => Promise<T>,
350
- maxRetries: number
351
- ): Promise<T> {
352
- let lastError: Error;
353
-
354
- for (let attempt = 0; attempt <= maxRetries; attempt++) {
355
- try {
356
- return await fn();
357
- } catch (error) {
358
- lastError = error as Error;
359
- }
360
- }
361
-
362
- throw lastError!;
363
- }
364
-
365
- class HttpClient {
366
- constructor(
367
- private baseUrl: string,
368
- private options: HttpClientOptions = {}
369
- ) {}
370
-
371
- async get(path: string): Promise<Response> {
372
- return withRetry(
373
- () => this.makeRequest(path),
374
- this.options.maxRetries ?? 1
375
- );
376
- }
377
-
378
- private async makeRequest(path: string): Promise<Response> {
379
- const res = await fetch(`${this.baseUrl}${path}`);
380
- return {
381
- status: res.status,
382
- data: await res.json()
383
- };
384
- }
385
- }
386
- ```
387
-
388
- **Time:** 30 minutes, robust HTTP client with retry logic
389
-
390
- ## Example 4: Refactoring with Test Protection
391
-
392
- ### Scenario
393
- Legacy code needs refactoring - extract business logic from controller.
394
-
395
- ### Step 1: Write Tests for Current Behavior
396
-
397
- **RED (characterization tests):**
398
- ```typescript
399
- describe('User Registration', () => {
400
- test('registers new user with valid data', async () => {
401
- const controller = new UserController();
402
- const result = await controller.register({
403
- email: 'user@example.com',
404
- password: 'SecurePass123',
405
- name: 'Test User'
406
- });
407
-
408
- expect(result.success).toBe(true);
409
- expect(result.userId).toBeDefined();
410
- });
411
-
412
- test('rejects duplicate email', async () => {
413
- const controller = new UserController();
414
- await controller.register({
415
- email: 'existing@example.com',
416
- password: 'Pass123',
417
- name: 'First User'
418
- });
419
-
420
- const result = await controller.register({
421
- email: 'existing@example.com',
422
- password: 'Pass456',
423
- name: 'Second User'
424
- });
425
-
426
- expect(result.success).toBe(false);
427
- expect(result.error).toBe('Email already exists');
428
- });
429
- });
430
- ```
431
-
432
- **GREEN - Tests pass on legacy code:**
433
- ```bash
434
- $ npm test
435
- PASS: registers new user with valid data
436
- PASS: rejects duplicate email
437
- ```
438
-
439
- ### Step 2: Refactor with Test Protection
440
-
441
- **Before (everything in controller):**
442
- ```typescript
443
- class UserController {
444
- async register(data: UserData) {
445
- // Validation
446
- if (!data.email.includes('@')) {
447
- return { success: false, error: 'Invalid email' };
448
- }
449
- if (data.password.length < 8) {
450
- return { success: false, error: 'Password too short' };
451
- }
452
-
453
- // Check duplicate
454
- const existing = await db.users.findByEmail(data.email);
455
- if (existing) {
456
- return { success: false, error: 'Email already exists' };
457
- }
458
-
459
- // Create user
460
- const user = await db.users.create({
461
- email: data.email,
462
- password: hashPassword(data.password),
463
- name: data.name
464
- });
465
-
466
- return { success: true, userId: user.id };
467
- }
468
- }
469
- ```
470
-
471
- **After (extracted service):**
472
- ```typescript
473
- // Extract service
474
- class UserService {
475
- async registerUser(data: UserData): Promise<User> {
476
- this.validateUserData(data);
477
- await this.checkDuplicateEmail(data.email);
478
- return this.createUser(data);
479
- }
480
-
481
- private validateUserData(data: UserData): void {
482
- if (!data.email.includes('@')) {
483
- throw new Error('Invalid email');
484
- }
485
- if (data.password.length < 8) {
486
- throw new Error('Password too short');
487
- }
488
- }
489
-
490
- private async checkDuplicateEmail(email: string): Promise<void> {
491
- const existing = await db.users.findByEmail(email);
492
- if (existing) {
493
- throw new Error('Email already exists');
494
- }
495
- }
496
-
497
- private async createUser(data: UserData): Promise<User> {
498
- return db.users.create({
499
- email: data.email,
500
- password: hashPassword(data.password),
501
- name: data.name
502
- });
503
- }
504
- }
505
-
506
- // Simplified controller
507
- class UserController {
508
- private userService = new UserService();
509
-
510
- async register(data: UserData) {
511
- try {
512
- const user = await this.userService.registerUser(data);
513
- return { success: true, userId: user.id };
514
- } catch (error) {
515
- return { success: false, error: error.message };
516
- }
517
- }
518
- }
519
- ```
520
-
521
- **Verify refactoring:**
522
- ```bash
523
- $ npm test
524
- PASS: registers new user with valid data
525
- PASS: rejects duplicate email
526
- ```
527
- ✓ Behavior unchanged, structure improved
528
-
529
- ### Step 3: Add Tests for New Service
530
-
531
- **Now test service directly:**
532
- ```typescript
533
- describe('UserService', () => {
534
- test('validates email format', async () => {
535
- const service = new UserService();
536
- await expect(
537
- service.registerUser({
538
- email: 'invalid',
539
- password: 'Pass123456',
540
- name: 'Test'
541
- })
542
- ).rejects.toThrow('Invalid email');
543
- });
544
-
545
- test('validates password length', async () => {
546
- const service = new UserService();
547
- await expect(
548
- service.registerUser({
549
- email: 'test@example.com',
550
- password: 'short',
551
- name: 'Test'
552
- })
553
- ).rejects.toThrow('Password too short');
554
- });
555
- });
556
- ```
557
-
558
- **Time:** 45 minutes, safe refactoring with test protection
559
-
560
- ## Example 5: Building a Feature from Scratch
561
-
562
- ### Scenario
563
- Implement shopping cart with add/remove/total functionality.
564
-
565
- ### Complete TDD Session
566
-
567
- **Iteration 1 - Add Item:**
568
- ```typescript
569
- // RED
570
- test('adds item to cart', () => {
571
- const cart = new ShoppingCart();
572
- cart.addItem({ id: 1, name: 'Book', price: 10 });
573
- expect(cart.getItemCount()).toBe(1);
574
- });
575
-
576
- // GREEN
577
- class ShoppingCart {
578
- private items: Item[] = [];
579
-
580
- addItem(item: Item) {
581
- this.items.push(item);
582
- }
583
-
584
- getItemCount(): number {
585
- return this.items.length;
586
- }
587
- }
588
- ```
589
-
590
- **Iteration 2 - Calculate Total:**
591
- ```typescript
592
- // RED
593
- test('calculates total price', () => {
594
- const cart = new ShoppingCart();
595
- cart.addItem({ id: 1, name: 'Book', price: 10 });
596
- cart.addItem({ id: 2, name: 'Pen', price: 5 });
597
- expect(cart.getTotal()).toBe(15);
598
- });
599
-
600
- // GREEN
601
- class ShoppingCart {
602
- private items: Item[] = [];
603
-
604
- addItem(item: Item) {
605
- this.items.push(item);
606
- }
607
-
608
- getItemCount(): number {
609
- return this.items.length;
610
- }
611
-
612
- getTotal(): number {
613
- return this.items.reduce((sum, item) => sum + item.price, 0);
614
- }
615
- }
616
- ```
617
-
618
- **Iteration 3 - Remove Item:**
619
- ```typescript
620
- // RED
621
- test('removes item from cart', () => {
622
- const cart = new ShoppingCart();
623
- cart.addItem({ id: 1, name: 'Book', price: 10 });
624
- cart.addItem({ id: 2, name: 'Pen', price: 5 });
625
- cart.removeItem(1);
626
- expect(cart.getItemCount()).toBe(1);
627
- expect(cart.getTotal()).toBe(5);
628
- });
629
-
630
- // GREEN
631
- class ShoppingCart {
632
- private items: Item[] = [];
633
-
634
- addItem(item: Item) {
635
- this.items.push(item);
636
- }
637
-
638
- removeItem(itemId: number) {
639
- this.items = this.items.filter(item => item.id !== itemId);
640
- }
641
-
642
- getItemCount(): number {
643
- return this.items.length;
644
- }
645
-
646
- getTotal(): number {
647
- return this.items.reduce((sum, item) => sum + item.price, 0);
648
- }
649
- }
650
- ```
651
-
652
- **Iteration 4 - Handle Empty Cart:**
653
- ```typescript
654
- // RED
655
- test('returns zero for empty cart', () => {
656
- const cart = new ShoppingCart();
657
- expect(cart.getTotal()).toBe(0);
658
- });
659
-
660
- // GREEN - Already passes!
661
- ```
662
-
663
- **Iteration 5 - Quantity Support:**
664
- ```typescript
665
- // RED
666
- test('handles item quantity', () => {
667
- const cart = new ShoppingCart();
668
- cart.addItem({ id: 1, name: 'Book', price: 10 }, 3);
669
- expect(cart.getTotal()).toBe(30);
670
- });
671
-
672
- // GREEN
673
- interface CartItem extends Item {
674
- quantity: number;
675
- }
676
-
677
- class ShoppingCart {
678
- private items: CartItem[] = [];
679
-
680
- addItem(item: Item, quantity: number = 1) {
681
- this.items.push({ ...item, quantity });
682
- }
683
-
684
- removeItem(itemId: number) {
685
- this.items = this.items.filter(item => item.id !== itemId);
686
- }
687
-
688
- getItemCount(): number {
689
- return this.items.reduce((sum, item) => sum + item.quantity, 0);
690
- }
691
-
692
- getTotal(): number {
693
- return this.items.reduce(
694
- (sum, item) => sum + (item.price * item.quantity),
695
- 0
696
- );
697
- }
698
- }
699
- ```
700
-
701
- **Time:** 25 minutes, complete shopping cart with 5 tests
702
-
703
- ## Key Patterns Demonstrated
704
-
705
- ### Incremental Development
706
- - Each test adds one small behavior
707
- - Build complexity gradually
708
- - Each step verified before next
709
-
710
- ### Test Protection
711
- - Refactoring safe with tests
712
- - Regressions caught immediately
713
- - Behavior preserved across changes
714
-
715
- ### Test-First Benefits
716
- - Clear requirements from tests
717
- - No over-engineering
718
- - 100% relevant test coverage
719
-
720
- ### Time Investment
721
- - Small features: 10-25 minutes
722
- - Medium features: 25-45 minutes
723
- - Includes tests, implementation, refactoring
724
- - Compare to: 15 min coding + 60-120 min debugging
725
-
726
- ## Summary
727
-
728
- TDD in practice:
729
- - Start with failing test
730
- - Implement minimally
731
- - Refactor safely
732
- - Build incrementally
733
- - Fast feedback loop
734
- - High confidence
735
-
736
- ## Related References
737
-
738
- - [Workflow](workflow.md): Complete RED/GREEN/REFACTOR process
739
- - [Philosophy](philosophy.md): Why TDD works
740
- - [Anti-patterns](anti-patterns.md): Common mistakes
741
- - [Integration](integration.md): TDD with other skills