claude-mpm 5.4.48__py3-none-any.whl → 5.6.34__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 (467) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +66 -241
  3. claude_mpm/agents/CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md +413 -0
  4. claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +109 -1925
  5. claude_mpm/agents/PM_INSTRUCTIONS.md +119 -689
  6. claude_mpm/agents/WORKFLOW.md +2 -0
  7. claude_mpm/agents/templates/circuit-breakers.md +26 -17
  8. claude_mpm/cli/__init__.py +5 -1
  9. claude_mpm/cli/commands/agents.py +2 -4
  10. claude_mpm/cli/commands/agents_reconcile.py +197 -0
  11. claude_mpm/cli/commands/autotodos.py +566 -0
  12. claude_mpm/cli/commands/commander.py +216 -0
  13. claude_mpm/cli/commands/configure.py +620 -21
  14. claude_mpm/cli/commands/configure_agent_display.py +3 -1
  15. claude_mpm/cli/commands/hook_errors.py +60 -60
  16. claude_mpm/cli/commands/monitor.py +2 -2
  17. claude_mpm/cli/commands/mpm_init/core.py +15 -8
  18. claude_mpm/cli/commands/profile.py +9 -10
  19. claude_mpm/cli/commands/run.py +35 -3
  20. claude_mpm/cli/commands/skill_source.py +51 -2
  21. claude_mpm/cli/commands/skills.py +171 -17
  22. claude_mpm/cli/executor.py +120 -16
  23. claude_mpm/cli/interactive/__init__.py +10 -0
  24. claude_mpm/cli/interactive/agent_wizard.py +30 -50
  25. claude_mpm/cli/interactive/questionary_styles.py +65 -0
  26. claude_mpm/cli/interactive/skill_selector.py +481 -0
  27. claude_mpm/cli/parsers/base_parser.py +76 -1
  28. claude_mpm/cli/parsers/commander_parser.py +116 -0
  29. claude_mpm/cli/parsers/profile_parser.py +0 -1
  30. claude_mpm/cli/parsers/run_parser.py +10 -0
  31. claude_mpm/cli/parsers/skill_source_parser.py +4 -0
  32. claude_mpm/cli/parsers/skills_parser.py +5 -0
  33. claude_mpm/cli/startup.py +544 -511
  34. claude_mpm/cli/startup_display.py +74 -6
  35. claude_mpm/cli/startup_logging.py +2 -2
  36. claude_mpm/cli/utils.py +7 -3
  37. claude_mpm/commander/__init__.py +78 -0
  38. claude_mpm/commander/adapters/__init__.py +60 -0
  39. claude_mpm/commander/adapters/auggie.py +260 -0
  40. claude_mpm/commander/adapters/base.py +288 -0
  41. claude_mpm/commander/adapters/claude_code.py +392 -0
  42. claude_mpm/commander/adapters/codex.py +237 -0
  43. claude_mpm/commander/adapters/communication.py +366 -0
  44. claude_mpm/commander/adapters/example_usage.py +310 -0
  45. claude_mpm/commander/adapters/mpm.py +389 -0
  46. claude_mpm/commander/adapters/registry.py +204 -0
  47. claude_mpm/commander/api/__init__.py +16 -0
  48. claude_mpm/commander/api/app.py +121 -0
  49. claude_mpm/commander/api/errors.py +133 -0
  50. claude_mpm/commander/api/routes/__init__.py +8 -0
  51. claude_mpm/commander/api/routes/events.py +184 -0
  52. claude_mpm/commander/api/routes/inbox.py +171 -0
  53. claude_mpm/commander/api/routes/messages.py +148 -0
  54. claude_mpm/commander/api/routes/projects.py +271 -0
  55. claude_mpm/commander/api/routes/sessions.py +226 -0
  56. claude_mpm/commander/api/routes/work.py +296 -0
  57. claude_mpm/commander/api/schemas.py +186 -0
  58. claude_mpm/commander/chat/__init__.py +7 -0
  59. claude_mpm/commander/chat/cli.py +146 -0
  60. claude_mpm/commander/chat/commands.py +96 -0
  61. claude_mpm/commander/chat/repl.py +310 -0
  62. claude_mpm/commander/config.py +51 -0
  63. claude_mpm/commander/config_loader.py +115 -0
  64. claude_mpm/commander/core/__init__.py +10 -0
  65. claude_mpm/commander/core/block_manager.py +325 -0
  66. claude_mpm/commander/core/response_manager.py +323 -0
  67. claude_mpm/commander/daemon.py +603 -0
  68. claude_mpm/commander/env_loader.py +59 -0
  69. claude_mpm/commander/events/__init__.py +26 -0
  70. claude_mpm/commander/events/manager.py +332 -0
  71. claude_mpm/commander/frameworks/__init__.py +12 -0
  72. claude_mpm/commander/frameworks/base.py +146 -0
  73. claude_mpm/commander/frameworks/claude_code.py +58 -0
  74. claude_mpm/commander/frameworks/mpm.py +62 -0
  75. claude_mpm/commander/inbox/__init__.py +16 -0
  76. claude_mpm/commander/inbox/dedup.py +128 -0
  77. claude_mpm/commander/inbox/inbox.py +224 -0
  78. claude_mpm/commander/inbox/models.py +70 -0
  79. claude_mpm/commander/instance_manager.py +450 -0
  80. claude_mpm/commander/llm/__init__.py +6 -0
  81. claude_mpm/commander/llm/openrouter_client.py +167 -0
  82. claude_mpm/commander/llm/summarizer.py +70 -0
  83. claude_mpm/commander/memory/__init__.py +45 -0
  84. claude_mpm/commander/memory/compression.py +347 -0
  85. claude_mpm/commander/memory/embeddings.py +230 -0
  86. claude_mpm/commander/memory/entities.py +310 -0
  87. claude_mpm/commander/memory/example_usage.py +290 -0
  88. claude_mpm/commander/memory/integration.py +325 -0
  89. claude_mpm/commander/memory/search.py +381 -0
  90. claude_mpm/commander/memory/store.py +657 -0
  91. claude_mpm/commander/models/__init__.py +18 -0
  92. claude_mpm/commander/models/events.py +121 -0
  93. claude_mpm/commander/models/project.py +162 -0
  94. claude_mpm/commander/models/work.py +214 -0
  95. claude_mpm/commander/parsing/__init__.py +20 -0
  96. claude_mpm/commander/parsing/extractor.py +132 -0
  97. claude_mpm/commander/parsing/output_parser.py +270 -0
  98. claude_mpm/commander/parsing/patterns.py +100 -0
  99. claude_mpm/commander/persistence/__init__.py +11 -0
  100. claude_mpm/commander/persistence/event_store.py +274 -0
  101. claude_mpm/commander/persistence/state_store.py +309 -0
  102. claude_mpm/commander/persistence/work_store.py +164 -0
  103. claude_mpm/commander/polling/__init__.py +13 -0
  104. claude_mpm/commander/polling/event_detector.py +104 -0
  105. claude_mpm/commander/polling/output_buffer.py +49 -0
  106. claude_mpm/commander/polling/output_poller.py +153 -0
  107. claude_mpm/commander/project_session.py +268 -0
  108. claude_mpm/commander/proxy/__init__.py +12 -0
  109. claude_mpm/commander/proxy/formatter.py +89 -0
  110. claude_mpm/commander/proxy/output_handler.py +191 -0
  111. claude_mpm/commander/proxy/relay.py +155 -0
  112. claude_mpm/commander/registry.py +410 -0
  113. claude_mpm/commander/runtime/__init__.py +10 -0
  114. claude_mpm/commander/runtime/executor.py +191 -0
  115. claude_mpm/commander/runtime/monitor.py +346 -0
  116. claude_mpm/commander/session/__init__.py +6 -0
  117. claude_mpm/commander/session/context.py +81 -0
  118. claude_mpm/commander/session/manager.py +59 -0
  119. claude_mpm/commander/tmux_orchestrator.py +361 -0
  120. claude_mpm/commander/web/__init__.py +1 -0
  121. claude_mpm/commander/work/__init__.py +30 -0
  122. claude_mpm/commander/work/executor.py +207 -0
  123. claude_mpm/commander/work/queue.py +405 -0
  124. claude_mpm/commander/workflow/__init__.py +27 -0
  125. claude_mpm/commander/workflow/event_handler.py +241 -0
  126. claude_mpm/commander/workflow/notifier.py +146 -0
  127. claude_mpm/commands/mpm-config.md +8 -0
  128. claude_mpm/commands/mpm-doctor.md +8 -0
  129. claude_mpm/commands/mpm-help.md +8 -0
  130. claude_mpm/commands/mpm-init.md +8 -0
  131. claude_mpm/commands/mpm-monitor.md +8 -0
  132. claude_mpm/commands/mpm-organize.md +8 -0
  133. claude_mpm/commands/mpm-postmortem.md +8 -0
  134. claude_mpm/commands/mpm-session-resume.md +9 -1
  135. claude_mpm/commands/mpm-status.md +8 -0
  136. claude_mpm/commands/mpm-ticket-view.md +8 -0
  137. claude_mpm/commands/mpm-version.md +8 -0
  138. claude_mpm/commands/mpm.md +8 -0
  139. claude_mpm/config/agent_presets.py +8 -7
  140. claude_mpm/config/skill_sources.py +16 -0
  141. claude_mpm/constants.py +1 -0
  142. claude_mpm/core/claude_runner.py +154 -2
  143. claude_mpm/core/config.py +35 -22
  144. claude_mpm/core/config_constants.py +74 -9
  145. claude_mpm/core/constants.py +56 -12
  146. claude_mpm/core/hook_manager.py +51 -3
  147. claude_mpm/core/interactive_session.py +12 -11
  148. claude_mpm/core/logger.py +26 -9
  149. claude_mpm/core/logging_utils.py +39 -13
  150. claude_mpm/core/network_config.py +148 -0
  151. claude_mpm/core/oneshot_session.py +7 -6
  152. claude_mpm/core/optimized_startup.py +3 -1
  153. claude_mpm/core/output_style_manager.py +66 -18
  154. claude_mpm/core/shared/config_loader.py +3 -1
  155. claude_mpm/core/socketio_pool.py +47 -15
  156. claude_mpm/core/unified_config.py +54 -8
  157. claude_mpm/core/unified_paths.py +95 -90
  158. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.C33zOoyM.css +1 -0
  159. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.CW1J-YuA.css +1 -0
  160. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/1WZnGYqX.js +24 -0
  161. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/67pF3qNn.js +1 -0
  162. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/6RxdMKe4.js +1 -0
  163. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/8cZrfX0h.js +60 -0
  164. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/9a6T2nm-.js +7 -0
  165. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B443AUzu.js +1 -0
  166. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B8AwtY2H.js +1 -0
  167. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BF15LAsF.js +1 -0
  168. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BQaXIfA_.js +331 -0
  169. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BRcwIQNr.js +4 -0
  170. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{uj46x2Wr.js → BSNlmTZj.js} +1 -1
  171. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BV6nKitt.js +43 -0
  172. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BViJ8lZt.js +128 -0
  173. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BcQ-Q0FE.js +1 -0
  174. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Bpyvgze_.js +30 -0
  175. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BzTRqg-z.js +1 -0
  176. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C0Fr8dve.js +1 -0
  177. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C3rbW_a-.js +1 -0
  178. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C8WYN38h.js +1 -0
  179. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C9I8FlXH.js +61 -0
  180. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIQcWgO2.js +36 -0
  181. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIctN7YN.js +7 -0
  182. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CKrS_JZW.js +145 -0
  183. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CR6P9C4A.js +89 -0
  184. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRRR9MD_.js +2 -0
  185. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRcR2DqT.js +334 -0
  186. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CSXtMOf0.js +1 -0
  187. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CT-sbxSk.js +1 -0
  188. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWm6DJsp.js +1 -0
  189. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CmKTTxBW.js +1 -0
  190. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CpqQ1Kzn.js +1 -0
  191. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Cu_Erd72.js +261 -0
  192. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D2nGpDRe.js +1 -0
  193. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9iCMida.js +267 -0
  194. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9ykgMoY.js +10 -0
  195. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DL2Ldur1.js +1 -0
  196. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DPfltzjH.js +165 -0
  197. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{N4qtv3Hx.js → DR8nis88.js} +2 -2
  198. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DUliQN2b.js +1 -0
  199. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DVp1hx9R.js +1 -0
  200. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DXlhR01x.js +122 -0
  201. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D_lyTybS.js +1 -0
  202. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DngoTTgh.js +1 -0
  203. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DqkmHtDC.js +220 -0
  204. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DsDh8EYs.js +1 -0
  205. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DypDmXgd.js +139 -0
  206. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Gi6I4Gst.js +1 -0
  207. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/IPYC-LnN.js +162 -0
  208. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JTLiF7dt.js +24 -0
  209. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JpevfAFt.js +68 -0
  210. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DjhvlsAc.js → NqQ1dWOy.js} +1 -1
  211. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/R8CEIRAd.js +2 -0
  212. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Zxy7qc-l.js +64 -0
  213. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/q9Hm6zAU.js +1 -0
  214. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/qtd3IeO4.js +15 -0
  215. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/ulBFON_C.js +65 -0
  216. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/wQVh1CoA.js +10 -0
  217. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.Dr7t0z2J.js +2 -0
  218. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.BGhZHUS3.js +1 -0
  219. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/{0.CAGBuiOw.js → 0.RgBboRvH.js} +1 -1
  220. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DG-KkbDf.js +1 -0
  221. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.D_jnf-x6.js +1 -0
  222. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -1
  223. claude_mpm/dashboard/static/svelte-build/index.html +11 -11
  224. claude_mpm/dashboard-svelte/node_modules/katex/src/fonts/generate_fonts.py +58 -0
  225. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_tfms.py +114 -0
  226. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_ttfs.py +122 -0
  227. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/format_json.py +28 -0
  228. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/parse_tfm.py +211 -0
  229. claude_mpm/experimental/cli_enhancements.py +2 -1
  230. claude_mpm/hooks/claude_hooks/INTEGRATION_EXAMPLE.md +243 -0
  231. claude_mpm/hooks/claude_hooks/README_AUTO_PAUSE.md +403 -0
  232. claude_mpm/hooks/claude_hooks/__pycache__/auto_pause_handler.cpython-311.pyc +0 -0
  233. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
  234. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
  235. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
  236. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
  237. claude_mpm/hooks/claude_hooks/auto_pause_handler.py +485 -0
  238. claude_mpm/hooks/claude_hooks/event_handlers.py +527 -136
  239. claude_mpm/hooks/claude_hooks/hook_handler.py +170 -104
  240. claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
  241. claude_mpm/hooks/claude_hooks/installer.py +206 -36
  242. claude_mpm/hooks/claude_hooks/memory_integration.py +52 -32
  243. claude_mpm/hooks/claude_hooks/response_tracking.py +43 -60
  244. claude_mpm/hooks/claude_hooks/services/__init__.py +21 -0
  245. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
  246. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
  247. claude_mpm/hooks/claude_hooks/services/__pycache__/container.cpython-311.pyc +0 -0
  248. claude_mpm/hooks/claude_hooks/services/__pycache__/protocols.cpython-311.pyc +0 -0
  249. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
  250. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
  251. claude_mpm/hooks/claude_hooks/services/connection_manager.py +41 -26
  252. claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +38 -105
  253. claude_mpm/hooks/claude_hooks/services/container.py +310 -0
  254. claude_mpm/hooks/claude_hooks/services/protocols.py +328 -0
  255. claude_mpm/hooks/claude_hooks/services/state_manager.py +25 -38
  256. claude_mpm/hooks/claude_hooks/services/subagent_processor.py +75 -77
  257. claude_mpm/hooks/session_resume_hook.py +89 -1
  258. claude_mpm/hooks/templates/pre_tool_use_simple.py +6 -6
  259. claude_mpm/hooks/templates/pre_tool_use_template.py +16 -8
  260. claude_mpm/init.py +215 -2
  261. claude_mpm/scripts/claude-hook-handler.sh +46 -19
  262. claude_mpm/scripts/start_activity_logging.py +0 -0
  263. claude_mpm/services/agents/agent_recommendation_service.py +8 -8
  264. claude_mpm/services/agents/agent_selection_service.py +2 -2
  265. claude_mpm/services/agents/cache_git_manager.py +1 -1
  266. claude_mpm/services/agents/deployment/agent_discovery_service.py +3 -1
  267. claude_mpm/services/agents/deployment/agent_format_converter.py +8 -6
  268. claude_mpm/services/agents/deployment/agent_template_builder.py +14 -4
  269. claude_mpm/services/agents/deployment/deployment_reconciler.py +577 -0
  270. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +4 -4
  271. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +4 -1
  272. claude_mpm/services/agents/deployment/startup_reconciliation.py +138 -0
  273. claude_mpm/services/agents/git_source_manager.py +6 -2
  274. claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
  275. claude_mpm/services/agents/single_tier_deployment_service.py +4 -4
  276. claude_mpm/services/agents/sources/git_source_sync_service.py +10 -5
  277. claude_mpm/services/agents/startup_sync.py +5 -2
  278. claude_mpm/services/cli/__init__.py +3 -0
  279. claude_mpm/services/cli/incremental_pause_manager.py +561 -0
  280. claude_mpm/services/cli/session_resume_helper.py +10 -2
  281. claude_mpm/services/command_deployment_service.py +44 -26
  282. claude_mpm/services/delegation_detector.py +175 -0
  283. claude_mpm/services/diagnostics/checks/agent_sources_check.py +30 -0
  284. claude_mpm/services/diagnostics/checks/configuration_check.py +24 -0
  285. claude_mpm/services/diagnostics/checks/installation_check.py +22 -0
  286. claude_mpm/services/diagnostics/checks/mcp_services_check.py +23 -0
  287. claude_mpm/services/diagnostics/doctor_reporter.py +31 -1
  288. claude_mpm/services/diagnostics/models.py +14 -1
  289. claude_mpm/services/event_log.py +325 -0
  290. claude_mpm/services/hook_installer_service.py +77 -8
  291. claude_mpm/services/infrastructure/__init__.py +4 -0
  292. claude_mpm/services/infrastructure/context_usage_tracker.py +291 -0
  293. claude_mpm/services/infrastructure/resume_log_generator.py +24 -5
  294. claude_mpm/services/monitor/daemon_manager.py +15 -4
  295. claude_mpm/services/monitor/management/lifecycle.py +8 -3
  296. claude_mpm/services/monitor/server.py +106 -16
  297. claude_mpm/services/pm_skills_deployer.py +267 -94
  298. claude_mpm/services/profile_manager.py +10 -4
  299. claude_mpm/services/skills/git_skill_source_manager.py +192 -29
  300. claude_mpm/services/skills/selective_skill_deployer.py +211 -46
  301. claude_mpm/services/skills/skill_discovery_service.py +74 -4
  302. claude_mpm/services/skills_deployer.py +188 -67
  303. claude_mpm/services/socketio/handlers/hook.py +14 -7
  304. claude_mpm/services/socketio/server/main.py +12 -4
  305. claude_mpm/skills/__init__.py +2 -1
  306. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
  307. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
  308. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
  309. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
  310. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
  311. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
  312. claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
  313. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
  314. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
  315. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
  316. claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
  317. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
  318. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
  319. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
  320. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
  321. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
  322. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
  323. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
  324. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
  325. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
  326. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
  327. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
  328. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
  329. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
  330. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
  331. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
  332. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
  333. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
  334. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
  335. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
  336. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
  337. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
  338. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
  339. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
  340. claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
  341. claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
  342. claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
  343. claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
  344. claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
  345. claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
  346. claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
  347. claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
  348. claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
  349. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
  350. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
  351. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
  352. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
  353. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
  354. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
  355. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
  356. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
  357. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
  358. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
  359. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
  360. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
  361. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
  362. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
  363. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
  364. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
  365. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
  366. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
  367. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
  368. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
  369. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
  370. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
  371. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
  372. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
  373. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
  374. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
  375. claude_mpm/skills/bundled/pm/mpm/SKILL.md +38 -0
  376. claude_mpm/skills/bundled/pm/mpm-agent-update-workflow/SKILL.md +75 -0
  377. claude_mpm/skills/bundled/pm/mpm-bug-reporting/SKILL.md +248 -0
  378. claude_mpm/skills/bundled/pm/mpm-circuit-breaker-enforcement/SKILL.md +476 -0
  379. claude_mpm/skills/bundled/pm/mpm-config/SKILL.md +29 -0
  380. claude_mpm/skills/bundled/pm/mpm-delegation-patterns/SKILL.md +167 -0
  381. claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -0
  382. claude_mpm/skills/bundled/pm/mpm-git-file-tracking/SKILL.md +113 -0
  383. claude_mpm/skills/bundled/pm/mpm-help/SKILL.md +35 -0
  384. claude_mpm/skills/bundled/pm/mpm-init/SKILL.md +125 -0
  385. claude_mpm/skills/bundled/pm/mpm-monitor/SKILL.md +32 -0
  386. claude_mpm/skills/bundled/pm/mpm-organize/SKILL.md +121 -0
  387. claude_mpm/skills/bundled/pm/mpm-postmortem/SKILL.md +22 -0
  388. claude_mpm/skills/bundled/pm/mpm-pr-workflow/SKILL.md +124 -0
  389. claude_mpm/skills/bundled/pm/mpm-session-management/SKILL.md +312 -0
  390. claude_mpm/skills/bundled/pm/mpm-session-pause/SKILL.md +170 -0
  391. claude_mpm/skills/bundled/pm/mpm-session-resume/SKILL.md +31 -0
  392. claude_mpm/skills/bundled/pm/mpm-status/SKILL.md +37 -0
  393. claude_mpm/skills/bundled/pm/mpm-teaching-mode/SKILL.md +657 -0
  394. claude_mpm/skills/bundled/pm/mpm-ticket-view/SKILL.md +110 -0
  395. claude_mpm/skills/bundled/pm/mpm-ticketing-integration/SKILL.md +154 -0
  396. claude_mpm/skills/bundled/pm/mpm-tool-usage-guide/SKILL.md +386 -0
  397. claude_mpm/skills/bundled/pm/mpm-verification-protocols/SKILL.md +198 -0
  398. claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
  399. claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
  400. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
  401. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
  402. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
  403. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
  404. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
  405. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
  406. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
  407. claude_mpm/skills/bundled/security-scanning.md +112 -0
  408. claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
  409. claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
  410. claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
  411. claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
  412. claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
  413. claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
  414. claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
  415. claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
  416. claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
  417. claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
  418. claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
  419. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
  420. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
  421. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
  422. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
  423. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
  424. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
  425. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
  426. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
  427. claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
  428. claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
  429. claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
  430. claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
  431. claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
  432. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
  433. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
  434. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
  435. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
  436. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
  437. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
  438. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
  439. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
  440. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
  441. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
  442. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
  443. claude_mpm/skills/registry.py +295 -90
  444. claude_mpm/skills/skill_manager.py +29 -23
  445. claude_mpm/templates/.pre-commit-config.yaml +112 -0
  446. claude_mpm/utils/agent_dependency_loader.py +103 -4
  447. claude_mpm/utils/robust_installer.py +45 -24
  448. claude_mpm-5.6.34.dist-info/METADATA +393 -0
  449. {claude_mpm-5.4.48.dist-info → claude_mpm-5.6.34.dist-info}/RECORD +453 -151
  450. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.B_FtCwCQ.css +0 -1
  451. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.Cl_eSA4x.css +0 -1
  452. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BgChzWQ1.js +0 -1
  453. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIXEwuWe.js +0 -1
  454. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWc5urbQ.js +0 -1
  455. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DMkZpdF2.js +0 -2
  456. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.DTL5mJO-.js +0 -2
  457. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.DzuEhzqh.js +0 -1
  458. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DFLC8jdE.js +0 -1
  459. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.DPvEihJJ.js +0 -10
  460. claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-311.pyc +0 -0
  461. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager.cpython-311.pyc +0 -0
  462. claude_mpm-5.4.48.dist-info/METADATA +0 -999
  463. {claude_mpm-5.4.48.dist-info → claude_mpm-5.6.34.dist-info}/WHEEL +0 -0
  464. {claude_mpm-5.4.48.dist-info → claude_mpm-5.6.34.dist-info}/entry_points.txt +0 -0
  465. {claude_mpm-5.4.48.dist-info → claude_mpm-5.6.34.dist-info}/licenses/LICENSE +0 -0
  466. {claude_mpm-5.4.48.dist-info → claude_mpm-5.6.34.dist-info}/licenses/LICENSE-FAQ.md +0 -0
  467. {claude_mpm-5.4.48.dist-info → claude_mpm-5.6.34.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,741 @@
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