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.
- claude_mpm/VERSION +1 -1
- claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +8 -5
- claude_mpm/agents/{CLAUDE_MPM_FOUNDERS_OUTPUT_STYLE.md → CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md} +14 -6
- claude_mpm/agents/PM_INSTRUCTIONS.md +109 -706
- claude_mpm/agents/WORKFLOW.md +2 -0
- claude_mpm/agents/templates/circuit-breakers.md +26 -17
- claude_mpm/auth/__init__.py +35 -0
- claude_mpm/auth/callback_server.py +328 -0
- claude_mpm/auth/models.py +104 -0
- claude_mpm/auth/oauth_manager.py +266 -0
- claude_mpm/auth/providers/__init__.py +12 -0
- claude_mpm/auth/providers/base.py +165 -0
- claude_mpm/auth/providers/google.py +261 -0
- claude_mpm/auth/token_storage.py +252 -0
- claude_mpm/cli/commands/autotodos.py +566 -0
- claude_mpm/cli/commands/commander.py +216 -0
- claude_mpm/cli/commands/hook_errors.py +60 -60
- claude_mpm/cli/commands/mcp.py +29 -17
- claude_mpm/cli/commands/mcp_command_router.py +39 -0
- claude_mpm/cli/commands/mcp_service_commands.py +304 -0
- claude_mpm/cli/commands/monitor.py +2 -2
- claude_mpm/cli/commands/mpm_init/core.py +2 -2
- claude_mpm/cli/commands/oauth.py +481 -0
- claude_mpm/cli/commands/run.py +35 -3
- claude_mpm/cli/commands/skill_source.py +51 -2
- claude_mpm/cli/commands/skills.py +5 -3
- claude_mpm/cli/executor.py +128 -16
- claude_mpm/cli/helpers.py +1 -1
- claude_mpm/cli/parsers/base_parser.py +84 -1
- claude_mpm/cli/parsers/commander_parser.py +116 -0
- claude_mpm/cli/parsers/mcp_parser.py +79 -0
- claude_mpm/cli/parsers/oauth_parser.py +165 -0
- claude_mpm/cli/parsers/run_parser.py +10 -0
- claude_mpm/cli/parsers/skill_source_parser.py +4 -0
- claude_mpm/cli/parsers/skills_parser.py +5 -0
- claude_mpm/cli/startup.py +345 -40
- claude_mpm/cli/startup_display.py +76 -7
- claude_mpm/cli/startup_logging.py +2 -2
- claude_mpm/cli/startup_migrations.py +236 -0
- claude_mpm/cli/utils.py +7 -3
- claude_mpm/commander/__init__.py +78 -0
- claude_mpm/commander/adapters/__init__.py +60 -0
- claude_mpm/commander/adapters/auggie.py +260 -0
- claude_mpm/commander/adapters/base.py +288 -0
- claude_mpm/commander/adapters/claude_code.py +392 -0
- claude_mpm/commander/adapters/codex.py +237 -0
- claude_mpm/commander/adapters/communication.py +366 -0
- claude_mpm/commander/adapters/example_usage.py +310 -0
- claude_mpm/commander/adapters/mpm.py +389 -0
- claude_mpm/commander/adapters/registry.py +204 -0
- claude_mpm/commander/api/__init__.py +16 -0
- claude_mpm/commander/api/app.py +121 -0
- claude_mpm/commander/api/errors.py +133 -0
- claude_mpm/commander/api/routes/__init__.py +8 -0
- claude_mpm/commander/api/routes/events.py +184 -0
- claude_mpm/commander/api/routes/inbox.py +171 -0
- claude_mpm/commander/api/routes/messages.py +148 -0
- claude_mpm/commander/api/routes/projects.py +271 -0
- claude_mpm/commander/api/routes/sessions.py +226 -0
- claude_mpm/commander/api/routes/work.py +296 -0
- claude_mpm/commander/api/schemas.py +186 -0
- claude_mpm/commander/chat/__init__.py +7 -0
- claude_mpm/commander/chat/cli.py +149 -0
- claude_mpm/commander/chat/commands.py +124 -0
- claude_mpm/commander/chat/repl.py +1957 -0
- claude_mpm/commander/config.py +51 -0
- claude_mpm/commander/config_loader.py +115 -0
- claude_mpm/commander/core/__init__.py +10 -0
- claude_mpm/commander/core/block_manager.py +325 -0
- claude_mpm/commander/core/response_manager.py +323 -0
- claude_mpm/commander/daemon.py +603 -0
- claude_mpm/commander/env_loader.py +59 -0
- claude_mpm/commander/events/__init__.py +26 -0
- claude_mpm/commander/events/manager.py +392 -0
- claude_mpm/commander/frameworks/__init__.py +12 -0
- claude_mpm/commander/frameworks/base.py +233 -0
- claude_mpm/commander/frameworks/claude_code.py +58 -0
- claude_mpm/commander/frameworks/mpm.py +57 -0
- claude_mpm/commander/git/__init__.py +5 -0
- claude_mpm/commander/git/worktree_manager.py +212 -0
- claude_mpm/commander/inbox/__init__.py +16 -0
- claude_mpm/commander/inbox/dedup.py +128 -0
- claude_mpm/commander/inbox/inbox.py +224 -0
- claude_mpm/commander/inbox/models.py +70 -0
- claude_mpm/commander/instance_manager.py +868 -0
- claude_mpm/commander/llm/__init__.py +6 -0
- claude_mpm/commander/llm/openrouter_client.py +167 -0
- claude_mpm/commander/llm/summarizer.py +70 -0
- claude_mpm/commander/memory/__init__.py +45 -0
- claude_mpm/commander/memory/compression.py +347 -0
- claude_mpm/commander/memory/embeddings.py +230 -0
- claude_mpm/commander/memory/entities.py +310 -0
- claude_mpm/commander/memory/example_usage.py +290 -0
- claude_mpm/commander/memory/integration.py +325 -0
- claude_mpm/commander/memory/search.py +381 -0
- claude_mpm/commander/memory/store.py +657 -0
- claude_mpm/commander/models/__init__.py +18 -0
- claude_mpm/commander/models/events.py +127 -0
- claude_mpm/commander/models/project.py +162 -0
- claude_mpm/commander/models/work.py +214 -0
- claude_mpm/commander/parsing/__init__.py +20 -0
- claude_mpm/commander/parsing/extractor.py +132 -0
- claude_mpm/commander/parsing/output_parser.py +270 -0
- claude_mpm/commander/parsing/patterns.py +100 -0
- claude_mpm/commander/persistence/__init__.py +11 -0
- claude_mpm/commander/persistence/event_store.py +274 -0
- claude_mpm/commander/persistence/state_store.py +403 -0
- claude_mpm/commander/persistence/work_store.py +164 -0
- claude_mpm/commander/polling/__init__.py +13 -0
- claude_mpm/commander/polling/event_detector.py +104 -0
- claude_mpm/commander/polling/output_buffer.py +49 -0
- claude_mpm/commander/polling/output_poller.py +153 -0
- claude_mpm/commander/project_session.py +268 -0
- claude_mpm/commander/proxy/__init__.py +12 -0
- claude_mpm/commander/proxy/formatter.py +89 -0
- claude_mpm/commander/proxy/output_handler.py +191 -0
- claude_mpm/commander/proxy/relay.py +155 -0
- claude_mpm/commander/registry.py +410 -0
- claude_mpm/commander/runtime/__init__.py +10 -0
- claude_mpm/commander/runtime/executor.py +191 -0
- claude_mpm/commander/runtime/monitor.py +346 -0
- claude_mpm/commander/session/__init__.py +6 -0
- claude_mpm/commander/session/context.py +81 -0
- claude_mpm/commander/session/manager.py +59 -0
- claude_mpm/commander/tmux_orchestrator.py +362 -0
- claude_mpm/commander/web/__init__.py +1 -0
- claude_mpm/commander/work/__init__.py +30 -0
- claude_mpm/commander/work/executor.py +207 -0
- claude_mpm/commander/work/queue.py +405 -0
- claude_mpm/commander/workflow/__init__.py +27 -0
- claude_mpm/commander/workflow/event_handler.py +241 -0
- claude_mpm/commander/workflow/notifier.py +146 -0
- claude_mpm/commands/mpm-config.md +8 -0
- claude_mpm/commands/mpm-doctor.md +8 -0
- claude_mpm/commands/mpm-help.md +8 -0
- claude_mpm/commands/mpm-init.md +8 -0
- claude_mpm/commands/mpm-monitor.md +8 -0
- claude_mpm/commands/mpm-organize.md +8 -0
- claude_mpm/commands/mpm-postmortem.md +8 -0
- claude_mpm/commands/mpm-session-resume.md +9 -1
- claude_mpm/commands/mpm-status.md +8 -0
- claude_mpm/commands/mpm-ticket-view.md +8 -0
- claude_mpm/commands/mpm-version.md +8 -0
- claude_mpm/commands/mpm.md +8 -0
- claude_mpm/config/agent_presets.py +8 -7
- claude_mpm/config/skill_sources.py +16 -0
- claude_mpm/constants.py +5 -0
- claude_mpm/core/claude_runner.py +152 -0
- claude_mpm/core/config.py +35 -22
- claude_mpm/core/config_constants.py +74 -9
- claude_mpm/core/constants.py +56 -12
- claude_mpm/core/hook_manager.py +53 -4
- claude_mpm/core/interactive_session.py +5 -4
- claude_mpm/core/logger.py +26 -9
- claude_mpm/core/logging_utils.py +39 -13
- claude_mpm/core/network_config.py +148 -0
- claude_mpm/core/oneshot_session.py +7 -6
- claude_mpm/core/output_style_manager.py +52 -12
- claude_mpm/core/socketio_pool.py +47 -15
- claude_mpm/core/unified_config.py +10 -6
- claude_mpm/core/unified_paths.py +68 -80
- claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.C33zOoyM.css +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.CW1J-YuA.css +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Cs_tUR18.js → 1WZnGYqX.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CDuw-vjf.js → 67pF3qNn.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{bTOqqlTd.js → 6RxdMKe4.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DwBR2MJi.js → 8cZrfX0h.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{ZGh7QtNv.js → 9a6T2nm-.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{D9lljYKQ.js → B443AUzu.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{RJiighC3.js → B8AwtY2H.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{uuIeMWc-.js → BF15LAsF.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{D3k0OPJN.js → BRcwIQNr.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CyWMqx4W.js → BV6nKitt.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CiIAseT4.js → BViJ8lZt.js} +5 -5
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CBBdVcY8.js → BcQ-Q0FE.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BovzEFCE.js → Bpyvgze_.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BzTRqg-z.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C0Fr8dve.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{eNVUfhuA.js → C3rbW_a-.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{GYwsonyD.js → C8WYN38h.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BIF9m_hv.js → C9I8FlXH.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{B0uc0UOD.js → CIQcWgO2.js} +3 -3
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Be7GpZd6.js → CIctN7YN.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Bh0LDWpI.js → CKrS_JZW.js} +2 -2
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DUrLdbGD.js → CR6P9C4A.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{B7xVLGWV.js → CRRR9MD_.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRcR2DqT.js +334 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Dhb8PKl3.js → CSXtMOf0.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BPYeabCQ.js → CT-sbxSk.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{sQeU3Y1z.js → CWm6DJsp.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CnA0NrzZ.js → CpqQ1Kzn.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C4B-KCzX.js → D2nGpDRe.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DGkLK5U1.js → D9iCMida.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BofRWZRR.js → D9ykgMoY.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DmxopI1J.js → DL2Ldur1.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C30mlcqg.js → DPfltzjH.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Vzk33B_K.js → DR8nis88.js} +2 -2
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DI7hHRFL.js → DUliQN2b.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C4JcI4KD.js → DXlhR01x.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{bT1r9zLR.js → D_lyTybS.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DZX00Y4g.js → DngoTTgh.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CzZX-COe.js → DqkmHtDC.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{B7RN905-.js → DsDh8EYs.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DLVjFsZ3.js → DypDmXgd.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{iEWssX7S.js → IPYC-LnN.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JTLiF7dt.js +24 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DaimHw_p.js → JpevfAFt.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DY1XQ8fi.js → R8CEIRAd.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Dle-35c7.js → Zxy7qc-l.js} +2 -2
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/q9Hm6zAU.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C_Usid8X.js → qtd3IeO4.js} +2 -2
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CzeYkLYB.js → ulBFON_C.js} +2 -2
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Cfqx1Qun.js → wQVh1CoA.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/{app.D6-I5TpK.js → app.Dr7t0z2J.js} +2 -2
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.BGhZHUS3.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/{0.m1gL8KXf.js → 0.RgBboRvH.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/{1.CgNOuw-d.js → 1.DG-KkbDf.js} +1 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.D_jnf-x6.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -1
- claude_mpm/dashboard/static/svelte-build/index.html +9 -9
- claude_mpm/experimental/cli_enhancements.py +2 -1
- claude_mpm/hooks/claude_hooks/INTEGRATION_EXAMPLE.md +243 -0
- claude_mpm/hooks/claude_hooks/README_AUTO_PAUSE.md +403 -0
- claude_mpm/hooks/claude_hooks/auto_pause_handler.py +485 -0
- claude_mpm/hooks/claude_hooks/event_handlers.py +466 -136
- claude_mpm/hooks/claude_hooks/hook_handler.py +204 -104
- claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
- claude_mpm/hooks/claude_hooks/installer.py +291 -59
- claude_mpm/hooks/claude_hooks/memory_integration.py +52 -32
- claude_mpm/hooks/claude_hooks/response_tracking.py +43 -60
- claude_mpm/hooks/claude_hooks/services/__init__.py +21 -0
- claude_mpm/hooks/claude_hooks/services/connection_manager.py +41 -26
- claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +38 -105
- claude_mpm/hooks/claude_hooks/services/container.py +326 -0
- claude_mpm/hooks/claude_hooks/services/protocols.py +328 -0
- claude_mpm/hooks/claude_hooks/services/state_manager.py +25 -38
- claude_mpm/hooks/claude_hooks/services/subagent_processor.py +75 -77
- claude_mpm/hooks/session_resume_hook.py +89 -1
- claude_mpm/hooks/templates/pre_tool_use_simple.py +6 -6
- claude_mpm/hooks/templates/pre_tool_use_template.py +16 -8
- claude_mpm/init.py +22 -15
- claude_mpm/mcp/__init__.py +9 -0
- claude_mpm/mcp/google_workspace_server.py +610 -0
- claude_mpm/scripts/claude-hook-handler.sh +46 -19
- claude_mpm/services/agents/agent_recommendation_service.py +8 -8
- claude_mpm/services/agents/agent_selection_service.py +2 -2
- claude_mpm/services/agents/cache_git_manager.py +1 -1
- claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +3 -0
- claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
- claude_mpm/services/agents/single_tier_deployment_service.py +4 -4
- claude_mpm/services/cli/__init__.py +3 -0
- claude_mpm/services/cli/incremental_pause_manager.py +561 -0
- claude_mpm/services/cli/session_resume_helper.py +10 -2
- claude_mpm/services/command_deployment_service.py +44 -26
- claude_mpm/services/delegation_detector.py +175 -0
- claude_mpm/services/diagnostics/checks/agent_sources_check.py +30 -0
- claude_mpm/services/diagnostics/checks/configuration_check.py +24 -0
- claude_mpm/services/diagnostics/checks/installation_check.py +22 -0
- claude_mpm/services/diagnostics/checks/mcp_services_check.py +23 -0
- claude_mpm/services/diagnostics/doctor_reporter.py +31 -1
- claude_mpm/services/diagnostics/models.py +14 -1
- claude_mpm/services/event_log.py +325 -0
- claude_mpm/services/hook_installer_service.py +77 -8
- claude_mpm/services/infrastructure/__init__.py +4 -0
- claude_mpm/services/infrastructure/context_usage_tracker.py +291 -0
- claude_mpm/services/infrastructure/resume_log_generator.py +24 -5
- claude_mpm/services/mcp_config_manager.py +99 -19
- claude_mpm/services/mcp_service_registry.py +294 -0
- claude_mpm/services/monitor/daemon_manager.py +15 -4
- claude_mpm/services/monitor/management/lifecycle.py +8 -2
- claude_mpm/services/monitor/server.py +111 -16
- claude_mpm/services/pm_skills_deployer.py +261 -87
- claude_mpm/services/skills/git_skill_source_manager.py +130 -10
- claude_mpm/services/skills/selective_skill_deployer.py +142 -16
- claude_mpm/services/skills/skill_discovery_service.py +74 -4
- claude_mpm/services/skills_deployer.py +31 -5
- claude_mpm/services/socketio/handlers/hook.py +14 -7
- claude_mpm/services/socketio/server/main.py +12 -4
- claude_mpm/skills/__init__.py +2 -1
- claude_mpm/skills/bundled/pm/mpm/SKILL.md +38 -0
- claude_mpm/skills/bundled/pm/mpm-agent-update-workflow/SKILL.md +75 -0
- claude_mpm/skills/bundled/pm/mpm-circuit-breaker-enforcement/SKILL.md +476 -0
- claude_mpm/skills/bundled/pm/mpm-config/SKILL.md +29 -0
- claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -0
- claude_mpm/skills/bundled/pm/mpm-help/SKILL.md +35 -0
- claude_mpm/skills/bundled/pm/mpm-init/SKILL.md +125 -0
- claude_mpm/skills/bundled/pm/mpm-monitor/SKILL.md +32 -0
- claude_mpm/skills/bundled/pm/mpm-organize/SKILL.md +121 -0
- claude_mpm/skills/bundled/pm/mpm-postmortem/SKILL.md +22 -0
- claude_mpm/skills/bundled/pm/mpm-session-management/SKILL.md +312 -0
- claude_mpm/skills/bundled/pm/mpm-session-pause/SKILL.md +170 -0
- claude_mpm/skills/bundled/pm/mpm-session-resume/SKILL.md +31 -0
- claude_mpm/skills/bundled/pm/mpm-status/SKILL.md +37 -0
- claude_mpm/skills/bundled/pm/{pm-teaching-mode → mpm-teaching-mode}/SKILL.md +2 -2
- claude_mpm/skills/bundled/pm/mpm-ticket-view/SKILL.md +110 -0
- claude_mpm/skills/bundled/pm/mpm-tool-usage-guide/SKILL.md +386 -0
- claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
- claude_mpm/skills/registry.py +295 -90
- claude_mpm/skills/skill_manager.py +4 -4
- claude_mpm-5.6.76.dist-info/METADATA +416 -0
- {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/RECORD +312 -175
- {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/WHEEL +1 -1
- {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/entry_points.txt +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.DWzvg0-y.css +0 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.ThTw9_ym.css +0 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/4TdZjIqw.js +0 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/5shd3_w0.js +0 -24
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BKjSRqUr.js +0 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Da0KfYnO.js +0 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Dfy6j1xT.js +0 -323
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.NWzMBYRp.js +0 -1
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.C0GcWctS.js +0 -1
- claude_mpm-5.4.85.dist-info/METADATA +0 -1023
- /claude_mpm/skills/bundled/pm/{pm-bug-reporting/pm-bug-reporting.md → mpm-bug-reporting/SKILL.md} +0 -0
- /claude_mpm/skills/bundled/pm/{pm-delegation-patterns → mpm-delegation-patterns}/SKILL.md +0 -0
- /claude_mpm/skills/bundled/pm/{pm-git-file-tracking → mpm-git-file-tracking}/SKILL.md +0 -0
- /claude_mpm/skills/bundled/pm/{pm-pr-workflow → mpm-pr-workflow}/SKILL.md +0 -0
- /claude_mpm/skills/bundled/pm/{pm-ticketing-integration → mpm-ticketing-integration}/SKILL.md +0 -0
- /claude_mpm/skills/bundled/pm/{pm-verification-protocols → mpm-verification-protocols}/SKILL.md +0 -0
- {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-5.4.85.dist-info → claude_mpm-5.6.76.dist-info}/licenses/LICENSE-FAQ.md +0 -0
- {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,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__)
|