claude-mpm 5.4.22__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.
- claude_mpm/VERSION +1 -1
- claude_mpm/agents/BASE_AGENT.md +164 -0
- claude_mpm/agents/BASE_ENGINEER.md +658 -0
- claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +66 -241
- claude_mpm/agents/CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md +413 -0
- claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +109 -1925
- claude_mpm/agents/MEMORY.md +1 -1
- claude_mpm/agents/PM_INSTRUCTIONS.md +374 -1257
- claude_mpm/agents/WORKFLOW.md +6 -253
- claude_mpm/agents/agent_loader.py +1 -1
- claude_mpm/agents/base_agent.json +31 -0
- claude_mpm/agents/frontmatter_validator.py +2 -2
- claude_mpm/agents/templates/circuit-breakers.md +26 -17
- claude_mpm/cli/__init__.py +5 -1
- claude_mpm/cli/commands/agent_state_manager.py +10 -10
- claude_mpm/cli/commands/agents.py +11 -13
- claude_mpm/cli/commands/agents_reconcile.py +197 -0
- claude_mpm/cli/commands/auto_configure.py +4 -4
- claude_mpm/cli/commands/autotodos.py +566 -0
- claude_mpm/cli/commands/commander.py +216 -0
- claude_mpm/cli/commands/configure.py +621 -22
- claude_mpm/cli/commands/configure_agent_display.py +12 -0
- claude_mpm/cli/commands/hook_errors.py +60 -60
- claude_mpm/cli/commands/monitor.py +2 -2
- claude_mpm/cli/commands/mpm_init/core.py +72 -0
- claude_mpm/cli/commands/postmortem.py +1 -1
- claude_mpm/cli/commands/profile.py +276 -0
- claude_mpm/cli/commands/run.py +35 -3
- claude_mpm/cli/commands/skill_source.py +51 -2
- claude_mpm/cli/commands/skills.py +182 -32
- claude_mpm/cli/executor.py +130 -16
- claude_mpm/cli/interactive/__init__.py +10 -0
- claude_mpm/cli/interactive/agent_wizard.py +32 -52
- claude_mpm/cli/interactive/questionary_styles.py +65 -0
- claude_mpm/cli/interactive/skill_selector.py +481 -0
- claude_mpm/cli/parsers/base_parser.py +83 -1
- claude_mpm/cli/parsers/commander_parser.py +116 -0
- claude_mpm/cli/parsers/profile_parser.py +147 -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 +2 -3
- claude_mpm/cli/startup.py +690 -386
- claude_mpm/cli/startup_display.py +74 -6
- claude_mpm/cli/startup_logging.py +2 -2
- 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 +146 -0
- claude_mpm/commander/chat/commands.py +96 -0
- claude_mpm/commander/chat/repl.py +310 -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 +332 -0
- claude_mpm/commander/frameworks/__init__.py +12 -0
- claude_mpm/commander/frameworks/base.py +146 -0
- claude_mpm/commander/frameworks/claude_code.py +58 -0
- claude_mpm/commander/frameworks/mpm.py +62 -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 +450 -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 +121 -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 +309 -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 +361 -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 +20 -249
- claude_mpm/commands/mpm-doctor.md +16 -21
- claude_mpm/commands/mpm-help.md +12 -205
- claude_mpm/commands/mpm-init.md +88 -506
- claude_mpm/commands/mpm-monitor.md +22 -401
- claude_mpm/commands/mpm-organize.md +70 -442
- claude_mpm/commands/mpm-postmortem.md +13 -107
- claude_mpm/commands/mpm-session-resume.md +20 -363
- claude_mpm/commands/mpm-status.md +13 -69
- claude_mpm/commands/mpm-ticket-view.md +60 -495
- claude_mpm/commands/mpm-version.md +13 -107
- 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 +1 -0
- claude_mpm/core/claude_runner.py +154 -2
- claude_mpm/core/config.py +37 -26
- claude_mpm/core/config_constants.py +74 -9
- claude_mpm/core/constants.py +56 -12
- claude_mpm/core/framework/loaders/agent_loader.py +1 -1
- claude_mpm/core/framework/loaders/instruction_loader.py +52 -11
- claude_mpm/core/hook_manager.py +51 -3
- claude_mpm/core/interactive_session.py +12 -11
- 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/optimized_startup.py +61 -0
- claude_mpm/core/output_style_manager.py +66 -18
- claude_mpm/core/shared/config_loader.py +3 -1
- claude_mpm/core/socketio_pool.py +47 -15
- claude_mpm/core/unified_agent_registry.py +1 -1
- claude_mpm/core/unified_config.py +54 -8
- claude_mpm/core/unified_paths.py +95 -90
- claude_mpm/dashboard/static/svelte-build/_app/env.js +1 -0
- 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/1WZnGYqX.js +24 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/67pF3qNn.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/6RxdMKe4.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/8cZrfX0h.js +60 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/9a6T2nm-.js +7 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B443AUzu.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B8AwtY2H.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BF15LAsF.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BQaXIfA_.js +331 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BRcwIQNr.js +4 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BSNlmTZj.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BV6nKitt.js +43 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BViJ8lZt.js +128 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BcQ-Q0FE.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Bpyvgze_.js +30 -0
- 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/C3rbW_a-.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C8WYN38h.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C9I8FlXH.js +61 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIQcWgO2.js +36 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIctN7YN.js +7 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CKrS_JZW.js +145 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CR6P9C4A.js +89 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRRR9MD_.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRcR2DqT.js +334 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CSXtMOf0.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CT-sbxSk.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWm6DJsp.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CmKTTxBW.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CpqQ1Kzn.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Cu_Erd72.js +261 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D2nGpDRe.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9iCMida.js +267 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9ykgMoY.js +10 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DL2Ldur1.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DPfltzjH.js +165 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DR8nis88.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DUliQN2b.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DVp1hx9R.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DXlhR01x.js +122 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D_lyTybS.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DngoTTgh.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DqkmHtDC.js +220 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DsDh8EYs.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DypDmXgd.js +139 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Gi6I4Gst.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/IPYC-LnN.js +162 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JTLiF7dt.js +24 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JpevfAFt.js +68 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/NqQ1dWOy.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/R8CEIRAd.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Zxy7qc-l.js +64 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/q9Hm6zAU.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/qtd3IeO4.js +15 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/ulBFON_C.js +65 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/wQVh1CoA.js +10 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.Dr7t0z2J.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.BGhZHUS3.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/0.RgBboRvH.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DG-KkbDf.js +1 -0
- 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 -0
- claude_mpm/dashboard/static/svelte-build/favicon.svg +7 -0
- claude_mpm/dashboard/static/svelte-build/index.html +36 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/fonts/generate_fonts.py +58 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_tfms.py +114 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_ttfs.py +122 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/format_json.py +28 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/parse_tfm.py +211 -0
- 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/__pycache__/__init__.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/auto_pause_handler.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/auto_pause_handler.py +485 -0
- claude_mpm/hooks/claude_hooks/event_handlers.py +527 -136
- claude_mpm/hooks/claude_hooks/hook_handler.py +313 -99
- claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
- claude_mpm/hooks/claude_hooks/installer.py +206 -36
- 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/__pycache__/__init__.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/container.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/protocols.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/connection_manager.py +67 -32
- claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +38 -105
- claude_mpm/hooks/claude_hooks/services/container.py +310 -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/kuzu_memory_hook.py +5 -5
- 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 +276 -0
- claude_mpm/models/git_repository.py +3 -3
- claude_mpm/scripts/claude-hook-handler.sh +46 -19
- claude_mpm/services/agents/agent_builder.py +3 -3
- 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 +7 -7
- claude_mpm/services/agents/deployment/agent_deployment.py +29 -7
- claude_mpm/services/agents/deployment/agent_discovery_service.py +4 -2
- claude_mpm/services/agents/deployment/agent_format_converter.py +25 -13
- claude_mpm/services/agents/deployment/agent_template_builder.py +39 -19
- claude_mpm/services/agents/deployment/agents_directory_resolver.py +2 -2
- claude_mpm/services/agents/deployment/async_agent_deployment.py +31 -27
- claude_mpm/services/agents/deployment/deployment_reconciler.py +577 -0
- claude_mpm/services/agents/deployment/local_template_deployment.py +3 -1
- claude_mpm/services/agents/deployment/multi_source_deployment_service.py +169 -26
- claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +101 -75
- claude_mpm/services/agents/deployment/startup_reconciliation.py +138 -0
- claude_mpm/services/agents/git_source_manager.py +23 -4
- claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
- claude_mpm/services/agents/recommender.py +5 -3
- claude_mpm/services/agents/single_tier_deployment_service.py +6 -6
- claude_mpm/services/agents/sources/git_source_sync_service.py +121 -10
- claude_mpm/services/agents/startup_sync.py +27 -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_check.py +2 -2
- claude_mpm/services/diagnostics/checks/agent_sources_check.py +31 -1
- 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/git/git_operations_service.py +8 -8
- 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/monitor/daemon_manager.py +15 -4
- claude_mpm/services/monitor/management/lifecycle.py +15 -3
- claude_mpm/services/monitor/server.py +571 -11
- claude_mpm/services/pm_skills_deployer.py +884 -0
- claude_mpm/services/profile_manager.py +337 -0
- claude_mpm/services/skills/git_skill_source_manager.py +281 -20
- claude_mpm/services/skills/selective_skill_deployer.py +211 -46
- claude_mpm/services/skills/skill_discovery_service.py +74 -4
- claude_mpm/services/skills_deployer.py +192 -70
- claude_mpm/services/socketio/dashboard_server.py +1 -0
- claude_mpm/services/socketio/event_normalizer.py +37 -6
- claude_mpm/services/socketio/handlers/hook.py +14 -7
- claude_mpm/services/socketio/server/core.py +262 -123
- claude_mpm/services/socketio/server/main.py +12 -4
- claude_mpm/skills/__init__.py +2 -1
- claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
- claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
- claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
- claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
- claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
- claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
- claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
- claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
- claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
- claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
- claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
- claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
- claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
- 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-bug-reporting/SKILL.md +248 -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-delegation-patterns/SKILL.md +167 -0
- claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -0
- claude_mpm/skills/bundled/pm/mpm-git-file-tracking/SKILL.md +113 -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-pr-workflow/SKILL.md +124 -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/mpm-teaching-mode/SKILL.md +657 -0
- claude_mpm/skills/bundled/pm/mpm-ticket-view/SKILL.md +110 -0
- claude_mpm/skills/bundled/pm/mpm-ticketing-integration/SKILL.md +154 -0
- claude_mpm/skills/bundled/pm/mpm-tool-usage-guide/SKILL.md +386 -0
- claude_mpm/skills/bundled/pm/mpm-verification-protocols/SKILL.md +198 -0
- claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
- claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
- claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
- claude_mpm/skills/bundled/security-scanning.md +112 -0
- claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
- claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
- claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
- claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
- claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
- claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
- claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
- claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
- claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
- claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
- claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
- claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
- claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
- claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
- claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
- claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
- claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
- claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
- claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
- claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
- claude_mpm/skills/registry.py +295 -90
- claude_mpm/skills/skill_manager.py +98 -3
- claude_mpm/templates/.pre-commit-config.yaml +112 -0
- claude_mpm/utils/agent_dependency_loader.py +115 -4
- claude_mpm/utils/agent_filters.py +1 -1
- claude_mpm/utils/migration.py +4 -4
- claude_mpm/utils/robust_installer.py +86 -21
- claude_mpm-5.6.34.dist-info/METADATA +393 -0
- {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/RECORD +486 -145
- claude_mpm-5.4.22.dist-info/METADATA +0 -996
- {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/WHEEL +0 -0
- {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/entry_points.txt +0 -0
- {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/licenses/LICENSE-FAQ.md +0 -0
- {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/top_level.txt +0 -0
|
@@ -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__)
|
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
"""Message and thread management endpoints for MPM Commander API.
|
|
2
|
+
|
|
3
|
+
This module implements REST endpoints for sending messages and retrieving
|
|
4
|
+
conversation threads for projects.
|
|
5
|
+
"""
|
|
6
|
+
|
|
7
|
+
import uuid
|
|
8
|
+
from typing import List
|
|
9
|
+
|
|
10
|
+
from fastapi import APIRouter, Request
|
|
11
|
+
|
|
12
|
+
from ...models import ThreadMessage
|
|
13
|
+
from ..errors import ProjectNotFoundError
|
|
14
|
+
from ..schemas import MessageResponse, SendMessageRequest
|
|
15
|
+
|
|
16
|
+
router = APIRouter()
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
def _get_registry(request: Request):
|
|
20
|
+
"""Get registry instance from app.state."""
|
|
21
|
+
if not hasattr(request.app.state, "registry") or request.app.state.registry is None:
|
|
22
|
+
raise RuntimeError("Registry not initialized")
|
|
23
|
+
return request.app.state.registry
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
def _message_to_response(message: ThreadMessage) -> MessageResponse:
|
|
27
|
+
"""Convert ThreadMessage model to MessageResponse schema.
|
|
28
|
+
|
|
29
|
+
Args:
|
|
30
|
+
message: ThreadMessage instance
|
|
31
|
+
|
|
32
|
+
Returns:
|
|
33
|
+
MessageResponse with message data
|
|
34
|
+
"""
|
|
35
|
+
return MessageResponse(
|
|
36
|
+
id=message.id,
|
|
37
|
+
role=message.role,
|
|
38
|
+
content=message.content,
|
|
39
|
+
session_id=message.session_id,
|
|
40
|
+
timestamp=message.timestamp,
|
|
41
|
+
)
|
|
42
|
+
|
|
43
|
+
|
|
44
|
+
@router.get("/projects/{project_id}/thread", response_model=List[MessageResponse])
|
|
45
|
+
async def get_thread(request: Request, project_id: str) -> List[MessageResponse]:
|
|
46
|
+
"""Get conversation thread for a project.
|
|
47
|
+
|
|
48
|
+
Returns all messages in chronological order.
|
|
49
|
+
|
|
50
|
+
Args:
|
|
51
|
+
project_id: Unique project identifier
|
|
52
|
+
|
|
53
|
+
Returns:
|
|
54
|
+
List of messages in thread (may be empty)
|
|
55
|
+
|
|
56
|
+
Raises:
|
|
57
|
+
ProjectNotFoundError: If project_id doesn't exist
|
|
58
|
+
|
|
59
|
+
Example:
|
|
60
|
+
GET /api/projects/abc-123/thread
|
|
61
|
+
Response: [
|
|
62
|
+
{
|
|
63
|
+
"id": "msg-1",
|
|
64
|
+
"role": "user",
|
|
65
|
+
"content": "Fix the login bug",
|
|
66
|
+
"session_id": null,
|
|
67
|
+
"timestamp": "2025-01-12T10:00:00Z"
|
|
68
|
+
},
|
|
69
|
+
{
|
|
70
|
+
"id": "msg-2",
|
|
71
|
+
"role": "assistant",
|
|
72
|
+
"content": "I'll investigate the login issue",
|
|
73
|
+
"session_id": "sess-456",
|
|
74
|
+
"timestamp": "2025-01-12T10:00:30Z"
|
|
75
|
+
}
|
|
76
|
+
]
|
|
77
|
+
"""
|
|
78
|
+
registry = _get_registry(request)
|
|
79
|
+
project = registry.get(project_id)
|
|
80
|
+
|
|
81
|
+
if project is None:
|
|
82
|
+
raise ProjectNotFoundError(project_id)
|
|
83
|
+
|
|
84
|
+
# Convert thread messages to responses
|
|
85
|
+
return [_message_to_response(m) for m in project.thread]
|
|
86
|
+
|
|
87
|
+
|
|
88
|
+
@router.post(
|
|
89
|
+
"/projects/{project_id}/messages", response_model=MessageResponse, status_code=201
|
|
90
|
+
)
|
|
91
|
+
async def send_message(
|
|
92
|
+
request: Request, project_id: str, req: SendMessageRequest
|
|
93
|
+
) -> MessageResponse:
|
|
94
|
+
"""Send a message to a project's active session.
|
|
95
|
+
|
|
96
|
+
Adds message to conversation thread and sends to specified or active session.
|
|
97
|
+
|
|
98
|
+
Args:
|
|
99
|
+
project_id: Unique project identifier
|
|
100
|
+
req: Message request with content and optional session_id
|
|
101
|
+
|
|
102
|
+
Returns:
|
|
103
|
+
Created message information
|
|
104
|
+
|
|
105
|
+
Raises:
|
|
106
|
+
ProjectNotFoundError: If project_id doesn't exist
|
|
107
|
+
|
|
108
|
+
Example:
|
|
109
|
+
POST /api/projects/abc-123/messages
|
|
110
|
+
Body: {
|
|
111
|
+
"content": "Fix the login bug",
|
|
112
|
+
"session_id": "sess-456"
|
|
113
|
+
}
|
|
114
|
+
Response: {
|
|
115
|
+
"id": "msg-1",
|
|
116
|
+
"role": "user",
|
|
117
|
+
"content": "Fix the login bug",
|
|
118
|
+
"session_id": "sess-456",
|
|
119
|
+
"timestamp": "2025-01-12T10:00:00Z"
|
|
120
|
+
}
|
|
121
|
+
"""
|
|
122
|
+
registry = _get_registry(request)
|
|
123
|
+
project = registry.get(project_id)
|
|
124
|
+
|
|
125
|
+
if project is None:
|
|
126
|
+
raise ProjectNotFoundError(project_id)
|
|
127
|
+
|
|
128
|
+
# Generate message ID
|
|
129
|
+
message_id = str(uuid.uuid4())
|
|
130
|
+
|
|
131
|
+
# Create message object
|
|
132
|
+
message = ThreadMessage(
|
|
133
|
+
id=message_id,
|
|
134
|
+
role="user",
|
|
135
|
+
content=req.content,
|
|
136
|
+
session_id=req.session_id,
|
|
137
|
+
)
|
|
138
|
+
|
|
139
|
+
# Add to project thread
|
|
140
|
+
project.thread.append(message)
|
|
141
|
+
|
|
142
|
+
# Update last activity
|
|
143
|
+
registry.touch(project_id)
|
|
144
|
+
|
|
145
|
+
# TODO: Send to session/runtime adapter (Phase 2)
|
|
146
|
+
# For Phase 1, message is just stored in thread
|
|
147
|
+
|
|
148
|
+
return _message_to_response(message)
|
|
@@ -0,0 +1,271 @@
|
|
|
1
|
+
"""Project management endpoints for MPM Commander API.
|
|
2
|
+
|
|
3
|
+
This module implements REST endpoints for registering, listing, and managing
|
|
4
|
+
projects in the MPM Commander.
|
|
5
|
+
"""
|
|
6
|
+
|
|
7
|
+
from pathlib import Path
|
|
8
|
+
from typing import List
|
|
9
|
+
|
|
10
|
+
from fastapi import APIRouter, Request, Response
|
|
11
|
+
|
|
12
|
+
from ...models import ProjectState
|
|
13
|
+
from ..errors import InvalidPathError, ProjectAlreadyExistsError, ProjectNotFoundError
|
|
14
|
+
from ..schemas import ProjectResponse, RegisterProjectRequest, SessionResponse
|
|
15
|
+
|
|
16
|
+
router = APIRouter()
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
def _get_registry(request: Request):
|
|
20
|
+
"""Get registry instance from app.state."""
|
|
21
|
+
if not hasattr(request.app.state, "registry") or request.app.state.registry is None:
|
|
22
|
+
raise RuntimeError("Registry not initialized")
|
|
23
|
+
return request.app.state.registry
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
def _project_to_response(project) -> ProjectResponse:
|
|
27
|
+
"""Convert Project model to ProjectResponse schema.
|
|
28
|
+
|
|
29
|
+
Args:
|
|
30
|
+
project: Project instance
|
|
31
|
+
|
|
32
|
+
Returns:
|
|
33
|
+
ProjectResponse with all project data
|
|
34
|
+
"""
|
|
35
|
+
# Convert sessions dict to list of SessionResponse
|
|
36
|
+
session_responses = [
|
|
37
|
+
SessionResponse(
|
|
38
|
+
id=session.id,
|
|
39
|
+
project_id=session.project_id,
|
|
40
|
+
runtime=session.runtime,
|
|
41
|
+
tmux_target=session.tmux_target,
|
|
42
|
+
status=session.status,
|
|
43
|
+
created_at=session.created_at,
|
|
44
|
+
)
|
|
45
|
+
for session in project.sessions.values()
|
|
46
|
+
]
|
|
47
|
+
|
|
48
|
+
return ProjectResponse(
|
|
49
|
+
id=project.id,
|
|
50
|
+
path=project.path,
|
|
51
|
+
name=project.name,
|
|
52
|
+
state=project.state.value,
|
|
53
|
+
state_reason=project.state_reason,
|
|
54
|
+
sessions=session_responses,
|
|
55
|
+
pending_events_count=len(project.pending_events),
|
|
56
|
+
last_activity=project.last_activity,
|
|
57
|
+
created_at=project.created_at,
|
|
58
|
+
)
|
|
59
|
+
|
|
60
|
+
|
|
61
|
+
@router.get("/projects", response_model=List[ProjectResponse])
|
|
62
|
+
async def list_projects(request: Request) -> List[ProjectResponse]:
|
|
63
|
+
"""List all registered projects.
|
|
64
|
+
|
|
65
|
+
Returns:
|
|
66
|
+
List of project information (may be empty)
|
|
67
|
+
|
|
68
|
+
Example:
|
|
69
|
+
GET /api/projects
|
|
70
|
+
Response: [
|
|
71
|
+
{
|
|
72
|
+
"id": "abc-123",
|
|
73
|
+
"path": "/Users/user/projects/my-app",
|
|
74
|
+
"name": "my-app",
|
|
75
|
+
"state": "idle",
|
|
76
|
+
"state_reason": null,
|
|
77
|
+
"sessions": [],
|
|
78
|
+
"pending_events_count": 0,
|
|
79
|
+
"last_activity": "2025-01-12T10:00:00Z",
|
|
80
|
+
"created_at": "2025-01-12T09:00:00Z"
|
|
81
|
+
}
|
|
82
|
+
]
|
|
83
|
+
"""
|
|
84
|
+
registry = _get_registry(request)
|
|
85
|
+
projects = registry.list_all()
|
|
86
|
+
return [_project_to_response(p) for p in projects]
|
|
87
|
+
|
|
88
|
+
|
|
89
|
+
@router.get("/projects/{project_id}", response_model=ProjectResponse)
|
|
90
|
+
async def get_project(request: Request, project_id: str) -> ProjectResponse:
|
|
91
|
+
"""Get project details by ID.
|
|
92
|
+
|
|
93
|
+
Args:
|
|
94
|
+
project_id: Unique project identifier
|
|
95
|
+
|
|
96
|
+
Returns:
|
|
97
|
+
Project information
|
|
98
|
+
|
|
99
|
+
Raises:
|
|
100
|
+
ProjectNotFoundError: If project_id doesn't exist
|
|
101
|
+
|
|
102
|
+
Example:
|
|
103
|
+
GET /api/projects/abc-123
|
|
104
|
+
Response: {
|
|
105
|
+
"id": "abc-123",
|
|
106
|
+
"path": "/Users/user/projects/my-app",
|
|
107
|
+
"name": "my-app",
|
|
108
|
+
"state": "idle",
|
|
109
|
+
...
|
|
110
|
+
}
|
|
111
|
+
"""
|
|
112
|
+
registry = _get_registry(request)
|
|
113
|
+
project = registry.get(project_id)
|
|
114
|
+
|
|
115
|
+
if project is None:
|
|
116
|
+
raise ProjectNotFoundError(project_id)
|
|
117
|
+
|
|
118
|
+
return _project_to_response(project)
|
|
119
|
+
|
|
120
|
+
|
|
121
|
+
@router.post("/projects", response_model=ProjectResponse, status_code=201)
|
|
122
|
+
async def register_project(
|
|
123
|
+
request: Request, req: RegisterProjectRequest
|
|
124
|
+
) -> ProjectResponse:
|
|
125
|
+
"""Register a new project.
|
|
126
|
+
|
|
127
|
+
Args:
|
|
128
|
+
req: Registration request with path and optional name
|
|
129
|
+
|
|
130
|
+
Returns:
|
|
131
|
+
Newly registered project information
|
|
132
|
+
|
|
133
|
+
Raises:
|
|
134
|
+
InvalidPathError: If path doesn't exist or isn't a directory
|
|
135
|
+
ProjectAlreadyExistsError: If path already registered
|
|
136
|
+
|
|
137
|
+
Example:
|
|
138
|
+
POST /api/projects
|
|
139
|
+
Body: {
|
|
140
|
+
"path": "/Users/user/projects/my-app",
|
|
141
|
+
"name": "My App"
|
|
142
|
+
}
|
|
143
|
+
Response: {
|
|
144
|
+
"id": "abc-123",
|
|
145
|
+
"path": "/Users/user/projects/my-app",
|
|
146
|
+
"name": "My App",
|
|
147
|
+
"state": "idle",
|
|
148
|
+
...
|
|
149
|
+
}
|
|
150
|
+
"""
|
|
151
|
+
registry = _get_registry(request)
|
|
152
|
+
|
|
153
|
+
# Validate path exists and is directory
|
|
154
|
+
path_obj = Path(req.path)
|
|
155
|
+
if not path_obj.exists() or not path_obj.is_dir():
|
|
156
|
+
raise InvalidPathError(req.path)
|
|
157
|
+
|
|
158
|
+
try:
|
|
159
|
+
project = registry.register(req.path, req.name, req.project_id)
|
|
160
|
+
return _project_to_response(project)
|
|
161
|
+
except ValueError as e:
|
|
162
|
+
# Registry raises ValueError for duplicate registration
|
|
163
|
+
error_msg = str(e)
|
|
164
|
+
if "already registered" in error_msg.lower():
|
|
165
|
+
raise ProjectAlreadyExistsError(req.path) from e
|
|
166
|
+
# Re-raise as InvalidPathError for other validation errors
|
|
167
|
+
raise InvalidPathError(req.path) from e
|
|
168
|
+
|
|
169
|
+
|
|
170
|
+
@router.delete("/projects/{project_id}", status_code=204)
|
|
171
|
+
async def unregister_project(request: Request, project_id: str) -> Response:
|
|
172
|
+
"""Unregister a project.
|
|
173
|
+
|
|
174
|
+
Args:
|
|
175
|
+
project_id: Unique project identifier
|
|
176
|
+
|
|
177
|
+
Returns:
|
|
178
|
+
Empty response with 204 status
|
|
179
|
+
|
|
180
|
+
Raises:
|
|
181
|
+
ProjectNotFoundError: If project_id doesn't exist
|
|
182
|
+
|
|
183
|
+
Example:
|
|
184
|
+
DELETE /api/projects/abc-123
|
|
185
|
+
Response: 204 No Content
|
|
186
|
+
"""
|
|
187
|
+
registry = _get_registry(request)
|
|
188
|
+
|
|
189
|
+
try:
|
|
190
|
+
registry.unregister(project_id)
|
|
191
|
+
return Response(status_code=204)
|
|
192
|
+
except KeyError as e:
|
|
193
|
+
raise ProjectNotFoundError(project_id) from e
|
|
194
|
+
|
|
195
|
+
|
|
196
|
+
@router.post("/projects/{project_id}/pause", response_model=ProjectResponse)
|
|
197
|
+
async def pause_project(request: Request, project_id: str) -> ProjectResponse:
|
|
198
|
+
"""Pause a project.
|
|
199
|
+
|
|
200
|
+
Sets project state to PAUSED to prevent automatic work processing.
|
|
201
|
+
|
|
202
|
+
Args:
|
|
203
|
+
project_id: Unique project identifier
|
|
204
|
+
|
|
205
|
+
Returns:
|
|
206
|
+
Updated project information
|
|
207
|
+
|
|
208
|
+
Raises:
|
|
209
|
+
ProjectNotFoundError: If project_id doesn't exist
|
|
210
|
+
|
|
211
|
+
Example:
|
|
212
|
+
POST /api/projects/abc-123/pause
|
|
213
|
+
Response: {
|
|
214
|
+
"id": "abc-123",
|
|
215
|
+
"state": "paused",
|
|
216
|
+
"state_reason": "Manually paused via API",
|
|
217
|
+
...
|
|
218
|
+
}
|
|
219
|
+
"""
|
|
220
|
+
registry = _get_registry(request)
|
|
221
|
+
project = registry.get(project_id)
|
|
222
|
+
|
|
223
|
+
if project is None:
|
|
224
|
+
raise ProjectNotFoundError(project_id)
|
|
225
|
+
|
|
226
|
+
registry.update_state(
|
|
227
|
+
project_id,
|
|
228
|
+
ProjectState.PAUSED,
|
|
229
|
+
reason="Manually paused via API",
|
|
230
|
+
)
|
|
231
|
+
|
|
232
|
+
return _project_to_response(project)
|
|
233
|
+
|
|
234
|
+
|
|
235
|
+
@router.post("/projects/{project_id}/resume", response_model=ProjectResponse)
|
|
236
|
+
async def resume_project(request: Request, project_id: str) -> ProjectResponse:
|
|
237
|
+
"""Resume a paused project.
|
|
238
|
+
|
|
239
|
+
Sets project state back to IDLE to allow work processing.
|
|
240
|
+
|
|
241
|
+
Args:
|
|
242
|
+
project_id: Unique project identifier
|
|
243
|
+
|
|
244
|
+
Returns:
|
|
245
|
+
Updated project information
|
|
246
|
+
|
|
247
|
+
Raises:
|
|
248
|
+
ProjectNotFoundError: If project_id doesn't exist
|
|
249
|
+
|
|
250
|
+
Example:
|
|
251
|
+
POST /api/projects/abc-123/resume
|
|
252
|
+
Response: {
|
|
253
|
+
"id": "abc-123",
|
|
254
|
+
"state": "idle",
|
|
255
|
+
"state_reason": "Resumed via API",
|
|
256
|
+
...
|
|
257
|
+
}
|
|
258
|
+
"""
|
|
259
|
+
registry = _get_registry(request)
|
|
260
|
+
project = registry.get(project_id)
|
|
261
|
+
|
|
262
|
+
if project is None:
|
|
263
|
+
raise ProjectNotFoundError(project_id)
|
|
264
|
+
|
|
265
|
+
registry.update_state(
|
|
266
|
+
project_id,
|
|
267
|
+
ProjectState.IDLE,
|
|
268
|
+
reason="Resumed via API",
|
|
269
|
+
)
|
|
270
|
+
|
|
271
|
+
return _project_to_response(project)
|