claude-mpm 5.4.85__py3-none-any.whl → 5.6.76__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 (322) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +8 -5
  3. claude_mpm/agents/{CLAUDE_MPM_FOUNDERS_OUTPUT_STYLE.md → CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md} +14 -6
  4. claude_mpm/agents/PM_INSTRUCTIONS.md +109 -706
  5. claude_mpm/agents/WORKFLOW.md +2 -0
  6. claude_mpm/agents/templates/circuit-breakers.md +26 -17
  7. claude_mpm/auth/__init__.py +35 -0
  8. claude_mpm/auth/callback_server.py +328 -0
  9. claude_mpm/auth/models.py +104 -0
  10. claude_mpm/auth/oauth_manager.py +266 -0
  11. claude_mpm/auth/providers/__init__.py +12 -0
  12. claude_mpm/auth/providers/base.py +165 -0
  13. claude_mpm/auth/providers/google.py +261 -0
  14. claude_mpm/auth/token_storage.py +252 -0
  15. claude_mpm/cli/commands/autotodos.py +566 -0
  16. claude_mpm/cli/commands/commander.py +216 -0
  17. claude_mpm/cli/commands/hook_errors.py +60 -60
  18. claude_mpm/cli/commands/mcp.py +29 -17
  19. claude_mpm/cli/commands/mcp_command_router.py +39 -0
  20. claude_mpm/cli/commands/mcp_service_commands.py +304 -0
  21. claude_mpm/cli/commands/monitor.py +2 -2
  22. claude_mpm/cli/commands/mpm_init/core.py +2 -2
  23. claude_mpm/cli/commands/oauth.py +481 -0
  24. claude_mpm/cli/commands/run.py +35 -3
  25. claude_mpm/cli/commands/skill_source.py +51 -2
  26. claude_mpm/cli/commands/skills.py +5 -3
  27. claude_mpm/cli/executor.py +128 -16
  28. claude_mpm/cli/helpers.py +1 -1
  29. claude_mpm/cli/parsers/base_parser.py +84 -1
  30. claude_mpm/cli/parsers/commander_parser.py +116 -0
  31. claude_mpm/cli/parsers/mcp_parser.py +79 -0
  32. claude_mpm/cli/parsers/oauth_parser.py +165 -0
  33. claude_mpm/cli/parsers/run_parser.py +10 -0
  34. claude_mpm/cli/parsers/skill_source_parser.py +4 -0
  35. claude_mpm/cli/parsers/skills_parser.py +5 -0
  36. claude_mpm/cli/startup.py +345 -40
  37. claude_mpm/cli/startup_display.py +76 -7
  38. claude_mpm/cli/startup_logging.py +2 -2
  39. claude_mpm/cli/startup_migrations.py +236 -0
  40. claude_mpm/cli/utils.py +7 -3
  41. claude_mpm/commander/__init__.py +78 -0
  42. claude_mpm/commander/adapters/__init__.py +60 -0
  43. claude_mpm/commander/adapters/auggie.py +260 -0
  44. claude_mpm/commander/adapters/base.py +288 -0
  45. claude_mpm/commander/adapters/claude_code.py +392 -0
  46. claude_mpm/commander/adapters/codex.py +237 -0
  47. claude_mpm/commander/adapters/communication.py +366 -0
  48. claude_mpm/commander/adapters/example_usage.py +310 -0
  49. claude_mpm/commander/adapters/mpm.py +389 -0
  50. claude_mpm/commander/adapters/registry.py +204 -0
  51. claude_mpm/commander/api/__init__.py +16 -0
  52. claude_mpm/commander/api/app.py +121 -0
  53. claude_mpm/commander/api/errors.py +133 -0
  54. claude_mpm/commander/api/routes/__init__.py +8 -0
  55. claude_mpm/commander/api/routes/events.py +184 -0
  56. claude_mpm/commander/api/routes/inbox.py +171 -0
  57. claude_mpm/commander/api/routes/messages.py +148 -0
  58. claude_mpm/commander/api/routes/projects.py +271 -0
  59. claude_mpm/commander/api/routes/sessions.py +226 -0
  60. claude_mpm/commander/api/routes/work.py +296 -0
  61. claude_mpm/commander/api/schemas.py +186 -0
  62. claude_mpm/commander/chat/__init__.py +7 -0
  63. claude_mpm/commander/chat/cli.py +149 -0
  64. claude_mpm/commander/chat/commands.py +124 -0
  65. claude_mpm/commander/chat/repl.py +1957 -0
  66. claude_mpm/commander/config.py +51 -0
  67. claude_mpm/commander/config_loader.py +115 -0
  68. claude_mpm/commander/core/__init__.py +10 -0
  69. claude_mpm/commander/core/block_manager.py +325 -0
  70. claude_mpm/commander/core/response_manager.py +323 -0
  71. claude_mpm/commander/daemon.py +603 -0
  72. claude_mpm/commander/env_loader.py +59 -0
  73. claude_mpm/commander/events/__init__.py +26 -0
  74. claude_mpm/commander/events/manager.py +392 -0
  75. claude_mpm/commander/frameworks/__init__.py +12 -0
  76. claude_mpm/commander/frameworks/base.py +233 -0
  77. claude_mpm/commander/frameworks/claude_code.py +58 -0
  78. claude_mpm/commander/frameworks/mpm.py +57 -0
  79. claude_mpm/commander/git/__init__.py +5 -0
  80. claude_mpm/commander/git/worktree_manager.py +212 -0
  81. claude_mpm/commander/inbox/__init__.py +16 -0
  82. claude_mpm/commander/inbox/dedup.py +128 -0
  83. claude_mpm/commander/inbox/inbox.py +224 -0
  84. claude_mpm/commander/inbox/models.py +70 -0
  85. claude_mpm/commander/instance_manager.py +868 -0
  86. claude_mpm/commander/llm/__init__.py +6 -0
  87. claude_mpm/commander/llm/openrouter_client.py +167 -0
  88. claude_mpm/commander/llm/summarizer.py +70 -0
  89. claude_mpm/commander/memory/__init__.py +45 -0
  90. claude_mpm/commander/memory/compression.py +347 -0
  91. claude_mpm/commander/memory/embeddings.py +230 -0
  92. claude_mpm/commander/memory/entities.py +310 -0
  93. claude_mpm/commander/memory/example_usage.py +290 -0
  94. claude_mpm/commander/memory/integration.py +325 -0
  95. claude_mpm/commander/memory/search.py +381 -0
  96. claude_mpm/commander/memory/store.py +657 -0
  97. claude_mpm/commander/models/__init__.py +18 -0
  98. claude_mpm/commander/models/events.py +127 -0
  99. claude_mpm/commander/models/project.py +162 -0
  100. claude_mpm/commander/models/work.py +214 -0
  101. claude_mpm/commander/parsing/__init__.py +20 -0
  102. claude_mpm/commander/parsing/extractor.py +132 -0
  103. claude_mpm/commander/parsing/output_parser.py +270 -0
  104. claude_mpm/commander/parsing/patterns.py +100 -0
  105. claude_mpm/commander/persistence/__init__.py +11 -0
  106. claude_mpm/commander/persistence/event_store.py +274 -0
  107. claude_mpm/commander/persistence/state_store.py +403 -0
  108. claude_mpm/commander/persistence/work_store.py +164 -0
  109. claude_mpm/commander/polling/__init__.py +13 -0
  110. claude_mpm/commander/polling/event_detector.py +104 -0
  111. claude_mpm/commander/polling/output_buffer.py +49 -0
  112. claude_mpm/commander/polling/output_poller.py +153 -0
  113. claude_mpm/commander/project_session.py +268 -0
  114. claude_mpm/commander/proxy/__init__.py +12 -0
  115. claude_mpm/commander/proxy/formatter.py +89 -0
  116. claude_mpm/commander/proxy/output_handler.py +191 -0
  117. claude_mpm/commander/proxy/relay.py +155 -0
  118. claude_mpm/commander/registry.py +410 -0
  119. claude_mpm/commander/runtime/__init__.py +10 -0
  120. claude_mpm/commander/runtime/executor.py +191 -0
  121. claude_mpm/commander/runtime/monitor.py +346 -0
  122. claude_mpm/commander/session/__init__.py +6 -0
  123. claude_mpm/commander/session/context.py +81 -0
  124. claude_mpm/commander/session/manager.py +59 -0
  125. claude_mpm/commander/tmux_orchestrator.py +362 -0
  126. claude_mpm/commander/web/__init__.py +1 -0
  127. claude_mpm/commander/work/__init__.py +30 -0
  128. claude_mpm/commander/work/executor.py +207 -0
  129. claude_mpm/commander/work/queue.py +405 -0
  130. claude_mpm/commander/workflow/__init__.py +27 -0
  131. claude_mpm/commander/workflow/event_handler.py +241 -0
  132. claude_mpm/commander/workflow/notifier.py +146 -0
  133. claude_mpm/commands/mpm-config.md +8 -0
  134. claude_mpm/commands/mpm-doctor.md +8 -0
  135. claude_mpm/commands/mpm-help.md +8 -0
  136. claude_mpm/commands/mpm-init.md +8 -0
  137. claude_mpm/commands/mpm-monitor.md +8 -0
  138. claude_mpm/commands/mpm-organize.md +8 -0
  139. claude_mpm/commands/mpm-postmortem.md +8 -0
  140. claude_mpm/commands/mpm-session-resume.md +9 -1
  141. claude_mpm/commands/mpm-status.md +8 -0
  142. claude_mpm/commands/mpm-ticket-view.md +8 -0
  143. claude_mpm/commands/mpm-version.md +8 -0
  144. claude_mpm/commands/mpm.md +8 -0
  145. claude_mpm/config/agent_presets.py +8 -7
  146. claude_mpm/config/skill_sources.py +16 -0
  147. claude_mpm/constants.py +5 -0
  148. claude_mpm/core/claude_runner.py +152 -0
  149. claude_mpm/core/config.py +35 -22
  150. claude_mpm/core/config_constants.py +74 -9
  151. claude_mpm/core/constants.py +56 -12
  152. claude_mpm/core/hook_manager.py +53 -4
  153. claude_mpm/core/interactive_session.py +5 -4
  154. claude_mpm/core/logger.py +26 -9
  155. claude_mpm/core/logging_utils.py +39 -13
  156. claude_mpm/core/network_config.py +148 -0
  157. claude_mpm/core/oneshot_session.py +7 -6
  158. claude_mpm/core/output_style_manager.py +52 -12
  159. claude_mpm/core/socketio_pool.py +47 -15
  160. claude_mpm/core/unified_config.py +10 -6
  161. claude_mpm/core/unified_paths.py +68 -80
  162. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.C33zOoyM.css +1 -0
  163. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.CW1J-YuA.css +1 -0
  164. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Cs_tUR18.js → 1WZnGYqX.js} +1 -1
  165. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CDuw-vjf.js → 67pF3qNn.js} +1 -1
  166. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{bTOqqlTd.js → 6RxdMKe4.js} +1 -1
  167. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DwBR2MJi.js → 8cZrfX0h.js} +1 -1
  168. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{ZGh7QtNv.js → 9a6T2nm-.js} +1 -1
  169. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{D9lljYKQ.js → B443AUzu.js} +1 -1
  170. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{RJiighC3.js → B8AwtY2H.js} +1 -1
  171. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{uuIeMWc-.js → BF15LAsF.js} +1 -1
  172. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{D3k0OPJN.js → BRcwIQNr.js} +1 -1
  173. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CyWMqx4W.js → BV6nKitt.js} +1 -1
  174. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CiIAseT4.js → BViJ8lZt.js} +5 -5
  175. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CBBdVcY8.js → BcQ-Q0FE.js} +1 -1
  176. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BovzEFCE.js → Bpyvgze_.js} +1 -1
  177. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BzTRqg-z.js +1 -0
  178. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C0Fr8dve.js +1 -0
  179. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{eNVUfhuA.js → C3rbW_a-.js} +1 -1
  180. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{GYwsonyD.js → C8WYN38h.js} +1 -1
  181. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BIF9m_hv.js → C9I8FlXH.js} +1 -1
  182. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{B0uc0UOD.js → CIQcWgO2.js} +3 -3
  183. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Be7GpZd6.js → CIctN7YN.js} +1 -1
  184. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Bh0LDWpI.js → CKrS_JZW.js} +2 -2
  185. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DUrLdbGD.js → CR6P9C4A.js} +1 -1
  186. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{B7xVLGWV.js → CRRR9MD_.js} +1 -1
  187. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRcR2DqT.js +334 -0
  188. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Dhb8PKl3.js → CSXtMOf0.js} +1 -1
  189. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BPYeabCQ.js → CT-sbxSk.js} +1 -1
  190. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{sQeU3Y1z.js → CWm6DJsp.js} +1 -1
  191. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CnA0NrzZ.js → CpqQ1Kzn.js} +1 -1
  192. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C4B-KCzX.js → D2nGpDRe.js} +1 -1
  193. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DGkLK5U1.js → D9iCMida.js} +1 -1
  194. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BofRWZRR.js → D9ykgMoY.js} +1 -1
  195. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DmxopI1J.js → DL2Ldur1.js} +1 -1
  196. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C30mlcqg.js → DPfltzjH.js} +1 -1
  197. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Vzk33B_K.js → DR8nis88.js} +2 -2
  198. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DI7hHRFL.js → DUliQN2b.js} +1 -1
  199. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C4JcI4KD.js → DXlhR01x.js} +1 -1
  200. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{bT1r9zLR.js → D_lyTybS.js} +1 -1
  201. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DZX00Y4g.js → DngoTTgh.js} +1 -1
  202. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CzZX-COe.js → DqkmHtDC.js} +1 -1
  203. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{B7RN905-.js → DsDh8EYs.js} +1 -1
  204. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DLVjFsZ3.js → DypDmXgd.js} +1 -1
  205. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{iEWssX7S.js → IPYC-LnN.js} +1 -1
  206. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JTLiF7dt.js +24 -0
  207. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DaimHw_p.js → JpevfAFt.js} +1 -1
  208. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DY1XQ8fi.js → R8CEIRAd.js} +1 -1
  209. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Dle-35c7.js → Zxy7qc-l.js} +2 -2
  210. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/q9Hm6zAU.js +1 -0
  211. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C_Usid8X.js → qtd3IeO4.js} +2 -2
  212. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CzeYkLYB.js → ulBFON_C.js} +2 -2
  213. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Cfqx1Qun.js → wQVh1CoA.js} +1 -1
  214. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/{app.D6-I5TpK.js → app.Dr7t0z2J.js} +2 -2
  215. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.BGhZHUS3.js +1 -0
  216. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/{0.m1gL8KXf.js → 0.RgBboRvH.js} +1 -1
  217. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/{1.CgNOuw-d.js → 1.DG-KkbDf.js} +1 -1
  218. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.D_jnf-x6.js +1 -0
  219. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -1
  220. claude_mpm/dashboard/static/svelte-build/index.html +9 -9
  221. claude_mpm/experimental/cli_enhancements.py +2 -1
  222. claude_mpm/hooks/claude_hooks/INTEGRATION_EXAMPLE.md +243 -0
  223. claude_mpm/hooks/claude_hooks/README_AUTO_PAUSE.md +403 -0
  224. claude_mpm/hooks/claude_hooks/auto_pause_handler.py +485 -0
  225. claude_mpm/hooks/claude_hooks/event_handlers.py +466 -136
  226. claude_mpm/hooks/claude_hooks/hook_handler.py +204 -104
  227. claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
  228. claude_mpm/hooks/claude_hooks/installer.py +291 -59
  229. claude_mpm/hooks/claude_hooks/memory_integration.py +52 -32
  230. claude_mpm/hooks/claude_hooks/response_tracking.py +43 -60
  231. claude_mpm/hooks/claude_hooks/services/__init__.py +21 -0
  232. claude_mpm/hooks/claude_hooks/services/connection_manager.py +41 -26
  233. claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +38 -105
  234. claude_mpm/hooks/claude_hooks/services/container.py +326 -0
  235. claude_mpm/hooks/claude_hooks/services/protocols.py +328 -0
  236. claude_mpm/hooks/claude_hooks/services/state_manager.py +25 -38
  237. claude_mpm/hooks/claude_hooks/services/subagent_processor.py +75 -77
  238. claude_mpm/hooks/session_resume_hook.py +89 -1
  239. claude_mpm/hooks/templates/pre_tool_use_simple.py +6 -6
  240. claude_mpm/hooks/templates/pre_tool_use_template.py +16 -8
  241. claude_mpm/init.py +22 -15
  242. claude_mpm/mcp/__init__.py +9 -0
  243. claude_mpm/mcp/google_workspace_server.py +610 -0
  244. claude_mpm/scripts/claude-hook-handler.sh +46 -19
  245. claude_mpm/services/agents/agent_recommendation_service.py +8 -8
  246. claude_mpm/services/agents/agent_selection_service.py +2 -2
  247. claude_mpm/services/agents/cache_git_manager.py +1 -1
  248. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +3 -0
  249. claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
  250. claude_mpm/services/agents/single_tier_deployment_service.py +4 -4
  251. claude_mpm/services/cli/__init__.py +3 -0
  252. claude_mpm/services/cli/incremental_pause_manager.py +561 -0
  253. claude_mpm/services/cli/session_resume_helper.py +10 -2
  254. claude_mpm/services/command_deployment_service.py +44 -26
  255. claude_mpm/services/delegation_detector.py +175 -0
  256. claude_mpm/services/diagnostics/checks/agent_sources_check.py +30 -0
  257. claude_mpm/services/diagnostics/checks/configuration_check.py +24 -0
  258. claude_mpm/services/diagnostics/checks/installation_check.py +22 -0
  259. claude_mpm/services/diagnostics/checks/mcp_services_check.py +23 -0
  260. claude_mpm/services/diagnostics/doctor_reporter.py +31 -1
  261. claude_mpm/services/diagnostics/models.py +14 -1
  262. claude_mpm/services/event_log.py +325 -0
  263. claude_mpm/services/hook_installer_service.py +77 -8
  264. claude_mpm/services/infrastructure/__init__.py +4 -0
  265. claude_mpm/services/infrastructure/context_usage_tracker.py +291 -0
  266. claude_mpm/services/infrastructure/resume_log_generator.py +24 -5
  267. claude_mpm/services/mcp_config_manager.py +99 -19
  268. claude_mpm/services/mcp_service_registry.py +294 -0
  269. claude_mpm/services/monitor/daemon_manager.py +15 -4
  270. claude_mpm/services/monitor/management/lifecycle.py +8 -2
  271. claude_mpm/services/monitor/server.py +111 -16
  272. claude_mpm/services/pm_skills_deployer.py +261 -87
  273. claude_mpm/services/skills/git_skill_source_manager.py +130 -10
  274. claude_mpm/services/skills/selective_skill_deployer.py +142 -16
  275. claude_mpm/services/skills/skill_discovery_service.py +74 -4
  276. claude_mpm/services/skills_deployer.py +31 -5
  277. claude_mpm/services/socketio/handlers/hook.py +14 -7
  278. claude_mpm/services/socketio/server/main.py +12 -4
  279. claude_mpm/skills/__init__.py +2 -1
  280. claude_mpm/skills/bundled/pm/mpm/SKILL.md +38 -0
  281. claude_mpm/skills/bundled/pm/mpm-agent-update-workflow/SKILL.md +75 -0
  282. claude_mpm/skills/bundled/pm/mpm-circuit-breaker-enforcement/SKILL.md +476 -0
  283. claude_mpm/skills/bundled/pm/mpm-config/SKILL.md +29 -0
  284. claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -0
  285. claude_mpm/skills/bundled/pm/mpm-help/SKILL.md +35 -0
  286. claude_mpm/skills/bundled/pm/mpm-init/SKILL.md +125 -0
  287. claude_mpm/skills/bundled/pm/mpm-monitor/SKILL.md +32 -0
  288. claude_mpm/skills/bundled/pm/mpm-organize/SKILL.md +121 -0
  289. claude_mpm/skills/bundled/pm/mpm-postmortem/SKILL.md +22 -0
  290. claude_mpm/skills/bundled/pm/mpm-session-management/SKILL.md +312 -0
  291. claude_mpm/skills/bundled/pm/mpm-session-pause/SKILL.md +170 -0
  292. claude_mpm/skills/bundled/pm/mpm-session-resume/SKILL.md +31 -0
  293. claude_mpm/skills/bundled/pm/mpm-status/SKILL.md +37 -0
  294. claude_mpm/skills/bundled/pm/{pm-teaching-mode → mpm-teaching-mode}/SKILL.md +2 -2
  295. claude_mpm/skills/bundled/pm/mpm-ticket-view/SKILL.md +110 -0
  296. claude_mpm/skills/bundled/pm/mpm-tool-usage-guide/SKILL.md +386 -0
  297. claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
  298. claude_mpm/skills/registry.py +295 -90
  299. claude_mpm/skills/skill_manager.py +4 -4
  300. claude_mpm-5.6.76.dist-info/METADATA +416 -0
  301. {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/RECORD +312 -175
  302. {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/WHEEL +1 -1
  303. {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/entry_points.txt +2 -0
  304. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.DWzvg0-y.css +0 -1
  305. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.ThTw9_ym.css +0 -1
  306. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/4TdZjIqw.js +0 -1
  307. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/5shd3_w0.js +0 -24
  308. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BKjSRqUr.js +0 -1
  309. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Da0KfYnO.js +0 -1
  310. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Dfy6j1xT.js +0 -323
  311. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.NWzMBYRp.js +0 -1
  312. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.C0GcWctS.js +0 -1
  313. claude_mpm-5.4.85.dist-info/METADATA +0 -1023
  314. /claude_mpm/skills/bundled/pm/{pm-bug-reporting/pm-bug-reporting.md → mpm-bug-reporting/SKILL.md} +0 -0
  315. /claude_mpm/skills/bundled/pm/{pm-delegation-patterns → mpm-delegation-patterns}/SKILL.md +0 -0
  316. /claude_mpm/skills/bundled/pm/{pm-git-file-tracking → mpm-git-file-tracking}/SKILL.md +0 -0
  317. /claude_mpm/skills/bundled/pm/{pm-pr-workflow → mpm-pr-workflow}/SKILL.md +0 -0
  318. /claude_mpm/skills/bundled/pm/{pm-ticketing-integration → mpm-ticketing-integration}/SKILL.md +0 -0
  319. /claude_mpm/skills/bundled/pm/{pm-verification-protocols → mpm-verification-protocols}/SKILL.md +0 -0
  320. {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/licenses/LICENSE +0 -0
  321. {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/licenses/LICENSE-FAQ.md +0 -0
  322. {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,121 @@
1
+ """FastAPI application for MPM Commander REST API.
2
+
3
+ This module defines the main FastAPI application instance with CORS,
4
+ lifecycle management, and route registration.
5
+ """
6
+
7
+ from contextlib import asynccontextmanager
8
+ from pathlib import Path
9
+ from typing import AsyncGenerator
10
+
11
+ from fastapi import FastAPI
12
+ from fastapi.middleware.cors import CORSMiddleware
13
+ from fastapi.responses import FileResponse
14
+ from fastapi.staticfiles import StaticFiles
15
+
16
+ from ..events.manager import EventManager
17
+ from ..inbox import Inbox
18
+ from ..registry import ProjectRegistry
19
+ from ..tmux_orchestrator import TmuxOrchestrator
20
+ from ..workflow import EventHandler
21
+ from .routes import events, inbox as inbox_routes, messages, projects, sessions, work
22
+
23
+
24
+ @asynccontextmanager
25
+ async def lifespan(app: FastAPI) -> AsyncGenerator[None, None]:
26
+ """Manage application lifecycle.
27
+
28
+ Initializes shared resources on startup and cleans up on shutdown.
29
+
30
+ Args:
31
+ app: FastAPI application instance
32
+
33
+ Yields:
34
+ None during application runtime
35
+ """
36
+ # Startup
37
+ import logging
38
+
39
+ logger = logging.getLogger(__name__)
40
+ logger.info("Lifespan starting. Initializing app.state resources...")
41
+
42
+ # Initialize app.state resources (daemon will inject its instances later)
43
+ if not hasattr(app.state, "registry"):
44
+ app.state.registry = ProjectRegistry()
45
+ if not hasattr(app.state, "tmux"):
46
+ app.state.tmux = TmuxOrchestrator()
47
+ if not hasattr(app.state, "event_manager"):
48
+ app.state.event_manager = EventManager()
49
+ if not hasattr(app.state, "inbox"):
50
+ app.state.inbox = Inbox(app.state.event_manager, app.state.registry)
51
+ if not hasattr(app.state, "session_manager"):
52
+ app.state.session_manager = {}
53
+ if not hasattr(app.state, "work_queues"):
54
+ logger.info("work_queues not set, creating new dict")
55
+ app.state.work_queues = {}
56
+ else:
57
+ logger.info(
58
+ f"work_queues already set, preserving id: {id(app.state.work_queues)}"
59
+ )
60
+ if not hasattr(app.state, "daemon_instance"):
61
+ app.state.daemon_instance = None
62
+ if not hasattr(app.state, "event_handler"):
63
+ app.state.event_handler = EventHandler(
64
+ app.state.inbox, app.state.session_manager
65
+ )
66
+
67
+ logger.info(f"Lifespan complete. work_queues id: {id(app.state.work_queues)}")
68
+
69
+ yield
70
+
71
+ # Shutdown
72
+ # No cleanup needed for Phase 1
73
+
74
+
75
+ app = FastAPI(
76
+ title="MPM Commander API",
77
+ description="REST API for MPM Commander - Autonomous AI Orchestration",
78
+ version="1.0.0",
79
+ lifespan=lifespan,
80
+ )
81
+
82
+ # CORS for local development
83
+ app.add_middleware(
84
+ CORSMiddleware,
85
+ allow_origins=["http://localhost:*"],
86
+ allow_credentials=True,
87
+ allow_methods=["*"],
88
+ allow_headers=["*"],
89
+ )
90
+
91
+ # Include routers
92
+ app.include_router(projects.router, prefix="/api", tags=["projects"])
93
+ app.include_router(sessions.router, prefix="/api", tags=["sessions"])
94
+ app.include_router(messages.router, prefix="/api", tags=["messages"])
95
+ app.include_router(inbox_routes.router, prefix="/api", tags=["inbox"])
96
+ app.include_router(events.router, prefix="/api", tags=["events"])
97
+ app.include_router(work.router, prefix="/api", tags=["work"])
98
+
99
+ # Mount static files
100
+ static_path = Path(__file__).parent.parent / "web" / "static"
101
+ app.mount("/static", StaticFiles(directory=str(static_path)), name="static")
102
+
103
+
104
+ @app.get("/api/health")
105
+ async def health_check() -> dict:
106
+ """Health check endpoint.
107
+
108
+ Returns:
109
+ Status and version information
110
+ """
111
+ return {"status": "ok", "version": "1.0.0"}
112
+
113
+
114
+ @app.get("/")
115
+ async def root() -> FileResponse:
116
+ """Serve the web UI index page.
117
+
118
+ Returns:
119
+ HTML page for the web UI
120
+ """
121
+ return FileResponse(static_path / "index.html")
@@ -0,0 +1,133 @@
1
+ """Error handling for MPM Commander API.
2
+
3
+ This module defines custom exception classes that map to HTTP error responses
4
+ with structured error codes and messages.
5
+ """
6
+
7
+ from fastapi import HTTPException
8
+
9
+
10
+ class CommanderAPIError(HTTPException):
11
+ """Base exception for all Commander API errors.
12
+
13
+ Provides consistent error response format with code and message.
14
+
15
+ Attributes:
16
+ code: Machine-readable error code
17
+ message: Human-readable error message
18
+ status_code: HTTP status code
19
+ """
20
+
21
+ def __init__(self, code: str, message: str, status_code: int = 400):
22
+ """Initialize API error.
23
+
24
+ Args:
25
+ code: Error code (e.g., "PROJECT_NOT_FOUND")
26
+ message: Descriptive error message
27
+ status_code: HTTP status code (default: 400)
28
+ """
29
+ super().__init__(
30
+ status_code=status_code,
31
+ detail={"error": {"code": code, "message": message}},
32
+ )
33
+
34
+
35
+ class ProjectNotFoundError(CommanderAPIError):
36
+ """Project with given ID does not exist."""
37
+
38
+ def __init__(self, project_id: str):
39
+ """Initialize project not found error.
40
+
41
+ Args:
42
+ project_id: The project ID that was not found
43
+ """
44
+ super().__init__(
45
+ "PROJECT_NOT_FOUND",
46
+ f"Project not found: {project_id}",
47
+ 404,
48
+ )
49
+
50
+
51
+ class ProjectAlreadyExistsError(CommanderAPIError):
52
+ """Project already registered at given path."""
53
+
54
+ def __init__(self, path: str):
55
+ """Initialize project already exists error.
56
+
57
+ Args:
58
+ path: The path that is already registered
59
+ """
60
+ super().__init__(
61
+ "PROJECT_ALREADY_EXISTS",
62
+ f"Project already registered: {path}",
63
+ 409,
64
+ )
65
+
66
+
67
+ class InvalidPathError(CommanderAPIError):
68
+ """Path does not exist or is not a directory."""
69
+
70
+ def __init__(self, path: str):
71
+ """Initialize invalid path error.
72
+
73
+ Args:
74
+ path: The invalid path
75
+ """
76
+ super().__init__(
77
+ "INVALID_PATH",
78
+ f"Path does not exist or is not a directory: {path}",
79
+ 400,
80
+ )
81
+
82
+
83
+ class SessionNotFoundError(CommanderAPIError):
84
+ """Session with given ID does not exist."""
85
+
86
+ def __init__(self, session_id: str):
87
+ """Initialize session not found error.
88
+
89
+ Args:
90
+ session_id: The session ID that was not found
91
+ """
92
+ super().__init__(
93
+ "SESSION_NOT_FOUND",
94
+ f"Session not found: {session_id}",
95
+ 404,
96
+ )
97
+
98
+
99
+ class InvalidRuntimeError(CommanderAPIError):
100
+ """Invalid runtime adapter specified."""
101
+
102
+ def __init__(self, runtime: str):
103
+ """Initialize invalid runtime error.
104
+
105
+ Args:
106
+ runtime: The invalid runtime name
107
+ """
108
+ super().__init__(
109
+ "INVALID_RUNTIME",
110
+ f"Invalid runtime: {runtime}",
111
+ 400,
112
+ )
113
+
114
+
115
+ class TmuxNoSpaceError(CommanderAPIError):
116
+ """Raised when tmux has no space for a new pane."""
117
+
118
+ def __init__(self, message: str | None = None):
119
+ """Initialize tmux no space error.
120
+
121
+ Args:
122
+ message: Custom error message (optional)
123
+ """
124
+ default_msg = (
125
+ "Unable to create session: tmux has no space for new pane. "
126
+ "Try closing some sessions or resize your terminal window. "
127
+ "You can also create a new tmux window with `tmux new-window`."
128
+ )
129
+ super().__init__(
130
+ "TMUX_NO_SPACE",
131
+ message or default_msg,
132
+ 409,
133
+ )
@@ -0,0 +1,8 @@
1
+ """API route modules for MPM Commander.
2
+
3
+ This package exports all route modules for registration with the FastAPI app.
4
+ """
5
+
6
+ from . import messages, projects, sessions, work
7
+
8
+ __all__ = ["messages", "projects", "sessions", "work"]
@@ -0,0 +1,184 @@
1
+ """REST API routes for event resolution.
2
+
3
+ Provides HTTP endpoints for resolving events and managing event responses.
4
+ """
5
+
6
+ from typing import List, Optional
7
+
8
+ from fastapi import APIRouter, Depends, HTTPException, Query
9
+ from pydantic import BaseModel
10
+
11
+ from ...events.manager import EventManager
12
+
13
+ router = APIRouter()
14
+
15
+
16
+ def get_event_manager() -> EventManager:
17
+ """Dependency to get the global event manager instance.
18
+
19
+ Returns:
20
+ The global event manager instance
21
+
22
+ Raises:
23
+ RuntimeError: If event manager is not initialized
24
+ """
25
+ from ..app import event_manager
26
+
27
+ if event_manager is None:
28
+ raise RuntimeError("Event manager not initialized")
29
+ return event_manager
30
+
31
+
32
+ def get_event_handler():
33
+ """Dependency to get the global event handler instance.
34
+
35
+ Returns:
36
+ The global event handler instance
37
+
38
+ Raises:
39
+ RuntimeError: If event handler is not initialized
40
+ """
41
+ from ..app import event_handler
42
+
43
+ if event_handler is None:
44
+ raise RuntimeError("Event handler not initialized")
45
+ return event_handler
46
+
47
+
48
+ class EventResponse(BaseModel):
49
+ """Request model for event resolution.
50
+
51
+ Attributes:
52
+ response: User's response to the event
53
+ """
54
+
55
+ response: str
56
+
57
+
58
+ class EventResolutionResponse(BaseModel):
59
+ """Response model for event resolution.
60
+
61
+ Attributes:
62
+ event_id: ID of resolved event
63
+ status: Resolution status
64
+ session_resumed: Whether project session was resumed
65
+ """
66
+
67
+ event_id: str
68
+ status: str
69
+ session_resumed: bool
70
+
71
+
72
+ class PendingEventResponse(BaseModel):
73
+ """Response model for pending event.
74
+
75
+ Attributes:
76
+ event_id: Unique event identifier
77
+ project_id: Project that raised this event
78
+ event_type: Type of event
79
+ priority: Urgency level
80
+ status: Current lifecycle status
81
+ title: Short event summary
82
+ content: Detailed event message
83
+ options: For DECISION_NEEDED events, list of choices
84
+ is_blocking: Whether event blocks progress
85
+ """
86
+
87
+ event_id: str
88
+ project_id: str
89
+ event_type: str
90
+ priority: str
91
+ status: str
92
+ title: str
93
+ content: str
94
+ options: Optional[List[str]]
95
+ is_blocking: bool
96
+
97
+
98
+ @router.post("/events/{event_id}/resolve", response_model=EventResolutionResponse)
99
+ async def resolve_event(
100
+ event_id: str,
101
+ response: EventResponse,
102
+ event_handler=Depends(get_event_handler),
103
+ ) -> EventResolutionResponse:
104
+ """Resolve an event with user response.
105
+
106
+ Marks the event as resolved and resumes the project session if it was
107
+ paused for this event.
108
+
109
+ Args:
110
+ event_id: ID of event to resolve
111
+ response: User's response to the event
112
+
113
+ Returns:
114
+ Resolution status and whether session was resumed
115
+
116
+ Raises:
117
+ HTTPException: If event not found (404) or resolution fails (500)
118
+
119
+ Example:
120
+ POST /api/events/evt_123/resolve
121
+ {
122
+ "response": "Use authlib for OAuth2"
123
+ }
124
+
125
+ Response:
126
+ {
127
+ "event_id": "evt_123",
128
+ "status": "resolved",
129
+ "session_resumed": true
130
+ }
131
+ """
132
+ try:
133
+ session_resumed = await event_handler.resolve_event(event_id, response.response)
134
+ return EventResolutionResponse(
135
+ event_id=event_id,
136
+ status="resolved",
137
+ session_resumed=session_resumed,
138
+ )
139
+ except KeyError as e:
140
+ raise HTTPException(
141
+ status_code=404, detail=f"Event not found: {event_id}"
142
+ ) from e
143
+ except Exception as e:
144
+ raise HTTPException(
145
+ status_code=500, detail=f"Failed to resolve event: {e!s}"
146
+ ) from e
147
+
148
+
149
+ @router.get("/events/pending", response_model=List[PendingEventResponse])
150
+ async def get_pending_events(
151
+ project_id: Optional[str] = Query(None, description="Filter by project ID"),
152
+ event_handler=Depends(get_event_handler),
153
+ ) -> List[PendingEventResponse]:
154
+ """Get pending events requiring resolution.
155
+
156
+ Returns all unresolved events, optionally filtered by project.
157
+ Events are sorted by priority (high to low) then created time (old to new).
158
+
159
+ Args:
160
+ project_id: If provided, only return events for this project
161
+
162
+ Returns:
163
+ List of pending events
164
+
165
+ Example:
166
+ GET /api/events/pending
167
+ GET /api/events/pending?project_id=proj_123
168
+ """
169
+ events = await event_handler.get_pending_events(project_id)
170
+
171
+ return [
172
+ PendingEventResponse(
173
+ event_id=event.id,
174
+ project_id=event.project_id,
175
+ event_type=event.type.value,
176
+ priority=event.priority.value,
177
+ status=event.status.value,
178
+ title=event.title,
179
+ content=event.content,
180
+ options=event.options,
181
+ is_blocking=event_handler.is_blocking(event),
182
+ )
183
+ for event in events
184
+ ]
@@ -0,0 +1,171 @@
1
+ """REST API routes for inbox system.
2
+
3
+ Provides HTTP endpoints for querying and managing the event inbox.
4
+ """
5
+
6
+ from datetime import datetime
7
+ from typing import List, Optional
8
+
9
+ from fastapi import APIRouter, Depends, Query
10
+ from pydantic import BaseModel
11
+
12
+ from ...inbox import Inbox
13
+ from ...models.events import EventPriority, EventType
14
+
15
+ router = APIRouter()
16
+
17
+
18
+ def get_inbox() -> Inbox:
19
+ """Dependency to get the global inbox instance.
20
+
21
+ Returns:
22
+ The global inbox instance
23
+
24
+ Raises:
25
+ RuntimeError: If inbox is not initialized
26
+ """
27
+ from ..app import inbox
28
+
29
+ if inbox is None:
30
+ raise RuntimeError("Inbox not initialized")
31
+ return inbox
32
+
33
+
34
+ class InboxItemResponse(BaseModel):
35
+ """Response model for inbox item.
36
+
37
+ Attributes:
38
+ event_id: Unique event identifier
39
+ project_id: Project that raised this event
40
+ project_name: Human-readable project name
41
+ project_path: Filesystem path to project
42
+ event_type: Type of event (decision, error, status, etc.)
43
+ priority: Urgency level (critical, high, normal, low, info)
44
+ status: Current lifecycle status
45
+ title: Short event summary
46
+ content: Detailed event message
47
+ options: For DECISION_NEEDED events, list of choices
48
+ age_display: Human-readable age (e.g., "5m ago")
49
+ created_at: When event was created
50
+ session_runtime: Optional session runtime identifier
51
+ """
52
+
53
+ event_id: str
54
+ project_id: str
55
+ project_name: str
56
+ project_path: str
57
+ event_type: str
58
+ priority: str
59
+ status: str
60
+ title: str
61
+ content: str
62
+ options: Optional[List[str]]
63
+ age_display: str
64
+ created_at: datetime
65
+ session_runtime: Optional[str]
66
+
67
+
68
+ class InboxCountsResponse(BaseModel):
69
+ """Response model for inbox counts.
70
+
71
+ Attributes:
72
+ critical: Count of CRITICAL priority events
73
+ high: Count of HIGH priority events
74
+ normal: Count of NORMAL priority events
75
+ low: Count of LOW priority events
76
+ info: Count of INFO priority events
77
+ total: Total count of all pending events
78
+ """
79
+
80
+ critical: int
81
+ high: int
82
+ normal: int
83
+ low: int
84
+ info: int
85
+ total: int
86
+
87
+
88
+ @router.get("/inbox", response_model=List[InboxItemResponse])
89
+ async def get_inbox_items(
90
+ limit: int = Query(50, ge=1, le=100, description="Maximum items to return"),
91
+ offset: int = Query(0, ge=0, description="Number of items to skip"),
92
+ priority: Optional[str] = Query(None, description="Filter by priority level"),
93
+ project_id: Optional[str] = Query(None, description="Filter by project ID"),
94
+ event_type: Optional[str] = Query(None, description="Filter by event type"),
95
+ inbox: Inbox = Depends(get_inbox),
96
+ ) -> List[InboxItemResponse]:
97
+ """Get inbox items with optional filtering and pagination.
98
+
99
+ Returns a list of inbox items sorted by priority (high to low) then
100
+ created time (old to new). Supports filtering by priority, project,
101
+ and event type.
102
+
103
+ Args:
104
+ limit: Maximum number of items to return (1-100, default: 50)
105
+ offset: Number of items to skip for pagination (default: 0)
106
+ priority: Filter by priority (critical, high, normal, low, info)
107
+ project_id: Filter by project ID
108
+ event_type: Filter by event type (decision_needed, error, etc.)
109
+
110
+ Returns:
111
+ List of inbox items matching the filters
112
+
113
+ Example:
114
+ GET /api/inbox?limit=20&priority=critical
115
+ GET /api/inbox?project_id=proj_123&event_type=error
116
+ GET /api/inbox?offset=50&limit=50 # Pagination
117
+ """
118
+ # Parse enum values if provided
119
+ pri = EventPriority(priority) if priority else None
120
+ evt = EventType(event_type) if event_type else None
121
+
122
+ items = inbox.get_items(
123
+ limit=limit,
124
+ offset=offset,
125
+ priority=pri,
126
+ project_id=project_id,
127
+ event_type=evt,
128
+ )
129
+
130
+ return [
131
+ InboxItemResponse(
132
+ event_id=item.event.id,
133
+ project_id=item.event.project_id,
134
+ project_name=item.project_name,
135
+ project_path=item.project_path,
136
+ event_type=item.event.type.value,
137
+ priority=item.event.priority.value,
138
+ status=item.event.status.value,
139
+ title=item.event.title,
140
+ content=item.event.content,
141
+ options=item.event.options,
142
+ age_display=item.age_display,
143
+ created_at=item.event.created_at,
144
+ session_runtime=item.session_runtime,
145
+ )
146
+ for item in items
147
+ ]
148
+
149
+
150
+ @router.get("/inbox/counts", response_model=InboxCountsResponse)
151
+ async def get_inbox_counts(
152
+ project_id: Optional[str] = Query(None, description="Filter by project ID"),
153
+ inbox: Inbox = Depends(get_inbox),
154
+ ) -> InboxCountsResponse:
155
+ """Get count of pending events by priority.
156
+
157
+ Returns summary statistics showing how many events exist at each
158
+ priority level. Optionally filtered to a specific project.
159
+
160
+ Args:
161
+ project_id: If provided, only count events for this project
162
+
163
+ Returns:
164
+ Breakdown of event counts by priority
165
+
166
+ Example:
167
+ GET /api/inbox/counts
168
+ GET /api/inbox/counts?project_id=proj_123
169
+ """
170
+ counts = inbox.get_counts(project_id)
171
+ return InboxCountsResponse(**counts.__dict__)