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,148 @@
|
|
|
1
|
+
"""Centralized network port configuration for Claude MPM.
|
|
2
|
+
|
|
3
|
+
This module provides the single source of truth for all network port defaults
|
|
4
|
+
and environment variable names used throughout the MPM system.
|
|
5
|
+
|
|
6
|
+
WHY: Previously, port defaults were hardcoded in multiple locations (config.py,
|
|
7
|
+
constants.py, commander/config.py, CLI parsers), leading to inconsistencies and
|
|
8
|
+
difficulty maintaining different defaults per service.
|
|
9
|
+
|
|
10
|
+
USAGE:
|
|
11
|
+
from claude_mpm.core.network_config import NetworkPorts
|
|
12
|
+
|
|
13
|
+
# Get default ports
|
|
14
|
+
monitor_port = NetworkPorts.MONITOR_DEFAULT
|
|
15
|
+
commander_port = NetworkPorts.COMMANDER_DEFAULT
|
|
16
|
+
|
|
17
|
+
# Get from environment with fallback
|
|
18
|
+
port = NetworkPorts.get_monitor_port()
|
|
19
|
+
"""
|
|
20
|
+
|
|
21
|
+
import os
|
|
22
|
+
from typing import Optional
|
|
23
|
+
|
|
24
|
+
|
|
25
|
+
class NetworkPorts:
|
|
26
|
+
"""Network port configuration with different defaults for each service.
|
|
27
|
+
|
|
28
|
+
Service Default Ports:
|
|
29
|
+
- Monitor: 8765 (user's preferred default)
|
|
30
|
+
- Commander: 8766
|
|
31
|
+
- Dashboard: 8767
|
|
32
|
+
- SocketIO: 8768
|
|
33
|
+
|
|
34
|
+
Port Range: 8765-8785 (21 ports available)
|
|
35
|
+
|
|
36
|
+
Environment Variables:
|
|
37
|
+
- CLAUDE_MPM_MONITOR_PORT: Override monitor port
|
|
38
|
+
- CLAUDE_MPM_COMMANDER_PORT: Override commander port
|
|
39
|
+
- CLAUDE_MPM_DASHBOARD_PORT: Override dashboard port
|
|
40
|
+
- CLAUDE_MPM_SOCKETIO_PORT: Override socketio port
|
|
41
|
+
- CLAUDE_MPM_DEFAULT_HOST: Override default host (default: 127.0.0.1)
|
|
42
|
+
"""
|
|
43
|
+
|
|
44
|
+
# Default ports for each service
|
|
45
|
+
MONITOR_DEFAULT = 8765
|
|
46
|
+
COMMANDER_DEFAULT = 8766
|
|
47
|
+
DASHBOARD_DEFAULT = 8767
|
|
48
|
+
SOCKETIO_DEFAULT = 8768
|
|
49
|
+
|
|
50
|
+
# Port range configuration
|
|
51
|
+
PORT_RANGE_START = 8765
|
|
52
|
+
PORT_RANGE_END = 8785
|
|
53
|
+
|
|
54
|
+
# Default host
|
|
55
|
+
DEFAULT_HOST = "127.0.0.1"
|
|
56
|
+
|
|
57
|
+
# Environment variable names
|
|
58
|
+
ENV_MONITOR_PORT = "CLAUDE_MPM_MONITOR_PORT"
|
|
59
|
+
ENV_COMMANDER_PORT = "CLAUDE_MPM_COMMANDER_PORT"
|
|
60
|
+
ENV_DASHBOARD_PORT = "CLAUDE_MPM_DASHBOARD_PORT"
|
|
61
|
+
ENV_SOCKETIO_PORT = "CLAUDE_MPM_SOCKETIO_PORT"
|
|
62
|
+
ENV_DEFAULT_HOST = "CLAUDE_MPM_DEFAULT_HOST"
|
|
63
|
+
|
|
64
|
+
@classmethod
|
|
65
|
+
def get_monitor_port(cls, default: Optional[int] = None) -> int:
|
|
66
|
+
"""Get monitor port from environment or default.
|
|
67
|
+
|
|
68
|
+
Args:
|
|
69
|
+
default: Optional override default (if not provided, uses MONITOR_DEFAULT)
|
|
70
|
+
|
|
71
|
+
Returns:
|
|
72
|
+
Port number from environment or default
|
|
73
|
+
"""
|
|
74
|
+
if default is None:
|
|
75
|
+
default = cls.MONITOR_DEFAULT
|
|
76
|
+
return int(os.getenv(cls.ENV_MONITOR_PORT, default))
|
|
77
|
+
|
|
78
|
+
@classmethod
|
|
79
|
+
def get_commander_port(cls, default: Optional[int] = None) -> int:
|
|
80
|
+
"""Get commander port from environment or default.
|
|
81
|
+
|
|
82
|
+
Args:
|
|
83
|
+
default: Optional override default (if not provided, uses COMMANDER_DEFAULT)
|
|
84
|
+
|
|
85
|
+
Returns:
|
|
86
|
+
Port number from environment or default
|
|
87
|
+
"""
|
|
88
|
+
if default is None:
|
|
89
|
+
default = cls.COMMANDER_DEFAULT
|
|
90
|
+
return int(os.getenv(cls.ENV_COMMANDER_PORT, default))
|
|
91
|
+
|
|
92
|
+
@classmethod
|
|
93
|
+
def get_dashboard_port(cls, default: Optional[int] = None) -> int:
|
|
94
|
+
"""Get dashboard port from environment or default.
|
|
95
|
+
|
|
96
|
+
Args:
|
|
97
|
+
default: Optional override default (if not provided, uses DASHBOARD_DEFAULT)
|
|
98
|
+
|
|
99
|
+
Returns:
|
|
100
|
+
Port number from environment or default
|
|
101
|
+
"""
|
|
102
|
+
if default is None:
|
|
103
|
+
default = cls.DASHBOARD_DEFAULT
|
|
104
|
+
return int(os.getenv(cls.ENV_DASHBOARD_PORT, default))
|
|
105
|
+
|
|
106
|
+
@classmethod
|
|
107
|
+
def get_socketio_port(cls, default: Optional[int] = None) -> int:
|
|
108
|
+
"""Get socketio port from environment or default.
|
|
109
|
+
|
|
110
|
+
Args:
|
|
111
|
+
default: Optional override default (if not provided, uses SOCKETIO_DEFAULT)
|
|
112
|
+
|
|
113
|
+
Returns:
|
|
114
|
+
Port number from environment or default
|
|
115
|
+
"""
|
|
116
|
+
if default is None:
|
|
117
|
+
default = cls.SOCKETIO_DEFAULT
|
|
118
|
+
return int(os.getenv(cls.ENV_SOCKETIO_PORT, default))
|
|
119
|
+
|
|
120
|
+
@classmethod
|
|
121
|
+
def get_default_host(cls) -> str:
|
|
122
|
+
"""Get default host from environment or default.
|
|
123
|
+
|
|
124
|
+
Returns:
|
|
125
|
+
Host address from environment or DEFAULT_HOST
|
|
126
|
+
"""
|
|
127
|
+
return os.getenv(cls.ENV_DEFAULT_HOST, cls.DEFAULT_HOST)
|
|
128
|
+
|
|
129
|
+
@classmethod
|
|
130
|
+
def get_port_range(cls) -> range:
|
|
131
|
+
"""Get the valid port range.
|
|
132
|
+
|
|
133
|
+
Returns:
|
|
134
|
+
Range object from PORT_RANGE_START to PORT_RANGE_END (inclusive)
|
|
135
|
+
"""
|
|
136
|
+
return range(cls.PORT_RANGE_START, cls.PORT_RANGE_END + 1)
|
|
137
|
+
|
|
138
|
+
@classmethod
|
|
139
|
+
def is_port_in_range(cls, port: int) -> bool:
|
|
140
|
+
"""Check if port is within valid range.
|
|
141
|
+
|
|
142
|
+
Args:
|
|
143
|
+
port: Port number to check
|
|
144
|
+
|
|
145
|
+
Returns:
|
|
146
|
+
True if port is in valid range, False otherwise
|
|
147
|
+
"""
|
|
148
|
+
return cls.PORT_RANGE_START <= port <= cls.PORT_RANGE_END
|
|
@@ -11,7 +11,7 @@ defines the interface it needs.
|
|
|
11
11
|
|
|
12
12
|
import contextlib
|
|
13
13
|
import os
|
|
14
|
-
import subprocess
|
|
14
|
+
import subprocess # nosec B404
|
|
15
15
|
import tempfile
|
|
16
16
|
import time
|
|
17
17
|
import uuid
|
|
@@ -86,11 +86,12 @@ class OneshotSession:
|
|
|
86
86
|
Returns:
|
|
87
87
|
True if successful, False otherwise
|
|
88
88
|
"""
|
|
89
|
-
#
|
|
90
|
-
|
|
91
|
-
|
|
89
|
+
# NOTE: System agents are deployed via reconciliation during startup.
|
|
90
|
+
# The reconciliation process respects user configuration and handles
|
|
91
|
+
# both native and custom mode deployment. No need to call setup_agents() here.
|
|
92
92
|
|
|
93
|
-
# Deploy project-specific agents
|
|
93
|
+
# Deploy project-specific agents from .claude-mpm/agents/
|
|
94
|
+
# This is separate from system agents and handles user-defined agents
|
|
94
95
|
self.runner.deploy_project_agents_to_claude()
|
|
95
96
|
|
|
96
97
|
return True
|
|
@@ -225,7 +226,7 @@ class OneshotSession:
|
|
|
225
226
|
if len(cmd) > 5:
|
|
226
227
|
self.logger.debug(f"Command has {len(cmd)} arguments total")
|
|
227
228
|
|
|
228
|
-
result = subprocess.run(
|
|
229
|
+
result = subprocess.run( # nosec B603
|
|
229
230
|
cmd, capture_output=True, text=True, env=env, check=False
|
|
230
231
|
)
|
|
231
232
|
|
|
@@ -27,7 +27,9 @@ _CACHED_CLAUDE_VERSION: Optional[str] = None
|
|
|
27
27
|
_VERSION_DETECTED: bool = False
|
|
28
28
|
|
|
29
29
|
# Output style types
|
|
30
|
-
OutputStyleType = Literal[
|
|
30
|
+
OutputStyleType = Literal[
|
|
31
|
+
"professional", "teaching", "research", "founders"
|
|
32
|
+
] # "founders" is deprecated, use "research"
|
|
31
33
|
|
|
32
34
|
|
|
33
35
|
class StyleConfig(TypedDict):
|
|
@@ -44,7 +46,7 @@ class OutputStyleManager:
|
|
|
44
46
|
Supports three output styles:
|
|
45
47
|
- professional: Default Claude MPM style (claude-mpm.md)
|
|
46
48
|
- teaching: Adaptive teaching mode (claude-mpm-teacher.md)
|
|
47
|
-
-
|
|
49
|
+
- research: Codebase research mode for founders, PMs, and developers (claude-mpm-research.md)
|
|
48
50
|
"""
|
|
49
51
|
|
|
50
52
|
def __init__(self) -> None:
|
|
@@ -72,12 +74,20 @@ class OutputStyleManager:
|
|
|
72
74
|
target=self.output_style_dir / "claude-mpm-teacher.md",
|
|
73
75
|
name="Claude MPM Teacher",
|
|
74
76
|
),
|
|
77
|
+
"research": StyleConfig(
|
|
78
|
+
source=Path(__file__).parent.parent
|
|
79
|
+
/ "agents"
|
|
80
|
+
/ "CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md",
|
|
81
|
+
target=self.output_style_dir / "claude-mpm-research.md",
|
|
82
|
+
name="Claude MPM Research",
|
|
83
|
+
),
|
|
84
|
+
# Backward compatibility alias (deprecated)
|
|
75
85
|
"founders": StyleConfig(
|
|
76
86
|
source=Path(__file__).parent.parent
|
|
77
87
|
/ "agents"
|
|
78
|
-
/ "
|
|
79
|
-
target=self.output_style_dir / "claude-mpm-
|
|
80
|
-
name="Claude MPM
|
|
88
|
+
/ "CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md",
|
|
89
|
+
target=self.output_style_dir / "claude-mpm-research.md",
|
|
90
|
+
name="Claude MPM Research",
|
|
81
91
|
),
|
|
82
92
|
}
|
|
83
93
|
|
|
@@ -287,6 +297,9 @@ class OutputStyleManager:
|
|
|
287
297
|
target_path = style_config["target"]
|
|
288
298
|
style_name = style_config["name"]
|
|
289
299
|
|
|
300
|
+
# Check if this is a fresh install (file doesn't exist yet)
|
|
301
|
+
is_fresh_install = not target_path.exists()
|
|
302
|
+
|
|
290
303
|
# If content not provided, read from source
|
|
291
304
|
if content is None:
|
|
292
305
|
content = self.extract_output_style_content(style=style)
|
|
@@ -300,7 +313,9 @@ class OutputStyleManager:
|
|
|
300
313
|
|
|
301
314
|
# Activate the style if requested
|
|
302
315
|
if activate:
|
|
303
|
-
self._activate_output_style(
|
|
316
|
+
self._activate_output_style(
|
|
317
|
+
style_name, is_fresh_install=is_fresh_install
|
|
318
|
+
)
|
|
304
319
|
|
|
305
320
|
return True
|
|
306
321
|
|
|
@@ -308,12 +323,21 @@ class OutputStyleManager:
|
|
|
308
323
|
self.logger.error(f"Failed to deploy {style} style: {e}")
|
|
309
324
|
return False
|
|
310
325
|
|
|
311
|
-
def _activate_output_style(
|
|
326
|
+
def _activate_output_style(
|
|
327
|
+
self, style_name: str = "Claude MPM", is_fresh_install: bool = False
|
|
328
|
+
) -> bool:
|
|
312
329
|
"""
|
|
313
330
|
Update Claude Code settings to activate a specific output style.
|
|
314
331
|
|
|
332
|
+
Only activates the style if:
|
|
333
|
+
1. No active style is currently set (first deployment), OR
|
|
334
|
+
2. This is a fresh install (style file didn't exist before deployment)
|
|
335
|
+
|
|
336
|
+
This preserves user preferences if they've manually changed their active style.
|
|
337
|
+
|
|
315
338
|
Args:
|
|
316
339
|
style_name: Name of the style to activate (e.g., "Claude MPM", "Claude MPM Teacher")
|
|
340
|
+
is_fresh_install: Whether this is a fresh install (style file didn't exist before)
|
|
317
341
|
|
|
318
342
|
Returns:
|
|
319
343
|
True if activated successfully, False otherwise
|
|
@@ -332,8 +356,15 @@ class OutputStyleManager:
|
|
|
332
356
|
# Check current active style
|
|
333
357
|
current_style = settings.get("activeOutputStyle")
|
|
334
358
|
|
|
335
|
-
#
|
|
336
|
-
|
|
359
|
+
# Only set activeOutputStyle if:
|
|
360
|
+
# 1. No active style is set (first deployment), OR
|
|
361
|
+
# 2. Current style is "default" (not a real user preference), OR
|
|
362
|
+
# 3. This is a fresh install (file didn't exist before deployment)
|
|
363
|
+
should_activate = (
|
|
364
|
+
current_style is None or current_style == "default" or is_fresh_install
|
|
365
|
+
)
|
|
366
|
+
|
|
367
|
+
if should_activate and current_style != style_name:
|
|
337
368
|
settings["activeOutputStyle"] = style_name
|
|
338
369
|
|
|
339
370
|
# Ensure settings directory exists
|
|
@@ -348,7 +379,10 @@ class OutputStyleManager:
|
|
|
348
379
|
f"✅ Activated {style_name} output style (was: {current_style or 'none'})"
|
|
349
380
|
)
|
|
350
381
|
else:
|
|
351
|
-
self.logger.debug(
|
|
382
|
+
self.logger.debug(
|
|
383
|
+
f"Preserving user preference: {current_style or 'none'} "
|
|
384
|
+
f"(skipping activation of {style_name})"
|
|
385
|
+
)
|
|
352
386
|
|
|
353
387
|
return True
|
|
354
388
|
|
|
@@ -442,6 +476,10 @@ class OutputStyleManager:
|
|
|
442
476
|
"""
|
|
443
477
|
results: Dict[str, bool] = {}
|
|
444
478
|
|
|
479
|
+
# Check if professional style exists BEFORE deployment
|
|
480
|
+
# This determines if this is a fresh install
|
|
481
|
+
professional_style_existed = self.styles["professional"]["target"].exists()
|
|
482
|
+
|
|
445
483
|
for style_type_key in self.styles:
|
|
446
484
|
# Deploy without activation
|
|
447
485
|
# Cast is safe because we know self.styles keys are OutputStyleType
|
|
@@ -449,9 +487,11 @@ class OutputStyleManager:
|
|
|
449
487
|
success = self.deploy_output_style(style=style_type, activate=False)
|
|
450
488
|
results[style_type] = success
|
|
451
489
|
|
|
452
|
-
# Activate the default style if requested
|
|
490
|
+
# Activate the default style if requested AND this is first deployment
|
|
453
491
|
if activate_default and results.get("professional", False):
|
|
454
|
-
self._activate_output_style(
|
|
492
|
+
self._activate_output_style(
|
|
493
|
+
"Claude MPM", is_fresh_install=not professional_style_existed
|
|
494
|
+
)
|
|
455
495
|
|
|
456
496
|
return results
|
|
457
497
|
|
claude_mpm/core/socketio_pool.py
CHANGED
|
@@ -1,4 +1,3 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
1
|
"""Socket.IO connection pool for efficient client connection management.
|
|
3
2
|
|
|
4
3
|
This module provides a connection pool to reuse Socket.IO client connections,
|
|
@@ -31,12 +30,21 @@ except ImportError:
|
|
|
31
30
|
# Import constants for configuration
|
|
32
31
|
try:
|
|
33
32
|
from claude_mpm.core.constants import NetworkConfig
|
|
33
|
+
from claude_mpm.core.network_config import NetworkPorts
|
|
34
34
|
except ImportError:
|
|
35
35
|
# Fallback if constants module not available
|
|
36
|
+
class NetworkPorts:
|
|
37
|
+
MONITOR_DEFAULT = 8765
|
|
38
|
+
COMMANDER_DEFAULT = 8766
|
|
39
|
+
DASHBOARD_DEFAULT = 8767
|
|
40
|
+
SOCKETIO_DEFAULT = 8768
|
|
41
|
+
PORT_RANGE_START = 8765
|
|
42
|
+
PORT_RANGE_END = 8785
|
|
43
|
+
|
|
36
44
|
class NetworkConfig:
|
|
37
|
-
DEFAULT_DASHBOARD_PORT =
|
|
45
|
+
DEFAULT_DASHBOARD_PORT = 8767
|
|
38
46
|
SOCKETIO_PORT_RANGE = (8765, 8785)
|
|
39
|
-
DEFAULT_SOCKETIO_PORT =
|
|
47
|
+
DEFAULT_SOCKETIO_PORT = 8768
|
|
40
48
|
|
|
41
49
|
socketio = None
|
|
42
50
|
|
|
@@ -184,9 +192,14 @@ class SocketIOConnectionPool:
|
|
|
184
192
|
self.health_running = False
|
|
185
193
|
self.last_health_check = datetime.now(timezone.utc)
|
|
186
194
|
|
|
187
|
-
# Server configuration
|
|
188
|
-
self.
|
|
189
|
-
|
|
195
|
+
# Server configuration - use default immediately, update async
|
|
196
|
+
self.server_port = int(
|
|
197
|
+
os.environ.get(
|
|
198
|
+
"CLAUDE_MPM_SOCKETIO_PORT", str(NetworkConfig.DEFAULT_SOCKETIO_PORT)
|
|
199
|
+
)
|
|
200
|
+
)
|
|
201
|
+
self.server_url = f"http://localhost:{self.server_port}"
|
|
202
|
+
self._port_detection_complete = False
|
|
190
203
|
|
|
191
204
|
# Pool lifecycle
|
|
192
205
|
self._running = False
|
|
@@ -200,7 +213,10 @@ class SocketIOConnectionPool:
|
|
|
200
213
|
return
|
|
201
214
|
|
|
202
215
|
self._running = True
|
|
203
|
-
|
|
216
|
+
|
|
217
|
+
# Start async port detection in background (non-blocking)
|
|
218
|
+
# Default port is already set in __init__, this just updates if a better one is found
|
|
219
|
+
self._detect_server_async()
|
|
204
220
|
|
|
205
221
|
# Start batch processing thread
|
|
206
222
|
self.batch_running = True
|
|
@@ -266,14 +282,29 @@ class SocketIOConnectionPool:
|
|
|
266
282
|
|
|
267
283
|
self.logger.info("Socket.IO connection pool stopped")
|
|
268
284
|
|
|
285
|
+
def _detect_server_async(self):
|
|
286
|
+
"""Start server detection in background thread.
|
|
287
|
+
|
|
288
|
+
This runs port scanning asynchronously to avoid blocking the main thread.
|
|
289
|
+
The default port is already set in __init__, so this just updates if a better one is found.
|
|
290
|
+
"""
|
|
291
|
+
threading.Thread(
|
|
292
|
+
target=self._detect_server, daemon=True, name="port-detect"
|
|
293
|
+
).start()
|
|
294
|
+
|
|
269
295
|
def _detect_server(self):
|
|
270
|
-
"""Detect Socket.IO server configuration.
|
|
271
|
-
|
|
296
|
+
"""Detect Socket.IO server configuration.
|
|
297
|
+
|
|
298
|
+
This method scans ports to find a running Socket.IO server.
|
|
299
|
+
It's designed to be run in a background thread to avoid blocking.
|
|
300
|
+
"""
|
|
301
|
+
# Check environment variable first - if set, use it and skip detection
|
|
272
302
|
env_port = os.environ.get("CLAUDE_MPM_SOCKETIO_PORT")
|
|
273
303
|
if env_port:
|
|
274
304
|
try:
|
|
275
305
|
self.server_port = int(env_port)
|
|
276
306
|
self.server_url = f"http://localhost:{self.server_port}"
|
|
307
|
+
self._port_detection_complete = True
|
|
277
308
|
self.logger.debug(
|
|
278
309
|
f"Using Socket.IO server from environment: {self.server_url}"
|
|
279
310
|
)
|
|
@@ -302,19 +333,20 @@ class SocketIOConnectionPool:
|
|
|
302
333
|
for port in common_ports:
|
|
303
334
|
try:
|
|
304
335
|
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
|
|
305
|
-
|
|
336
|
+
# Use 10ms timeout (reduced from 50ms) for faster scanning
|
|
337
|
+
s.settimeout(0.01)
|
|
306
338
|
result = s.connect_ex(("localhost", port))
|
|
307
339
|
if result == 0:
|
|
308
340
|
self.server_port = port
|
|
309
341
|
self.server_url = f"http://localhost:{port}"
|
|
342
|
+
self._port_detection_complete = True
|
|
310
343
|
self.logger.debug(f"Detected Socket.IO server on port {port}")
|
|
311
344
|
return
|
|
312
|
-
except Exception:
|
|
345
|
+
except Exception: # nosec B112 - intentional: skip ports that fail
|
|
313
346
|
continue
|
|
314
347
|
|
|
315
|
-
#
|
|
316
|
-
self.
|
|
317
|
-
self.server_url = f"http://localhost:{self.server_port}"
|
|
348
|
+
# Keep default port set in __init__, mark detection complete
|
|
349
|
+
self._port_detection_complete = True
|
|
318
350
|
self.logger.debug(f"Using default Socket.IO server: {self.server_url}")
|
|
319
351
|
|
|
320
352
|
def _create_client(self) -> Optional[socketio.AsyncClient]:
|
|
@@ -579,7 +611,7 @@ class SocketIOConnectionPool:
|
|
|
579
611
|
loop.stop()
|
|
580
612
|
loop.run_until_complete(loop.shutdown_asyncgens())
|
|
581
613
|
loop.close()
|
|
582
|
-
except Exception:
|
|
614
|
+
except Exception: # nosec B110 - intentional: cleanup best-effort
|
|
583
615
|
pass
|
|
584
616
|
|
|
585
617
|
async def _connect_client(self, client: socketio.AsyncClient):
|
|
@@ -73,15 +73,19 @@ class AgentConfig(BaseModel):
|
|
|
73
73
|
)
|
|
74
74
|
|
|
75
75
|
# Required agents that are always deployed
|
|
76
|
+
# Standard 7 core agents for essential PM workflow functionality
|
|
77
|
+
# These are auto-deployed when no agents are specified in configuration
|
|
76
78
|
required: List[str] = Field(
|
|
77
79
|
default_factory=lambda: [
|
|
78
|
-
"
|
|
79
|
-
"
|
|
80
|
-
"
|
|
81
|
-
"
|
|
82
|
-
"
|
|
80
|
+
"engineer", # General-purpose implementation
|
|
81
|
+
"research", # Codebase exploration and analysis
|
|
82
|
+
"qa", # Testing and quality assurance
|
|
83
|
+
"web-qa", # Browser-based testing specialist
|
|
84
|
+
"documentation", # Documentation generation
|
|
85
|
+
"ops", # Basic deployment operations
|
|
86
|
+
"ticketing", # Ticket tracking (essential for PM workflow)
|
|
83
87
|
],
|
|
84
|
-
description="Agents that are always deployed (core
|
|
88
|
+
description="Agents that are always deployed (standard 7 core agents)",
|
|
85
89
|
)
|
|
86
90
|
|
|
87
91
|
include_universal: bool = Field(
|
claude_mpm/core/unified_paths.py
CHANGED
|
@@ -76,6 +76,7 @@ class DeploymentContext(Enum):
|
|
|
76
76
|
EDITABLE_INSTALL = "editable_install"
|
|
77
77
|
PIP_INSTALL = "pip_install"
|
|
78
78
|
PIPX_INSTALL = "pipx_install"
|
|
79
|
+
UV_TOOLS = "uv_tools"
|
|
79
80
|
SYSTEM_PACKAGE = "system_package"
|
|
80
81
|
|
|
81
82
|
|
|
@@ -190,113 +191,100 @@ class PathContext:
|
|
|
190
191
|
|
|
191
192
|
Priority order:
|
|
192
193
|
1. Environment variable override (CLAUDE_MPM_DEV_MODE)
|
|
193
|
-
2.
|
|
194
|
-
3.
|
|
195
|
-
|
|
194
|
+
2. Package installation path (uv tools, pipx, site-packages, editable)
|
|
195
|
+
3. Current working directory (opt-in with CLAUDE_MPM_PREFER_LOCAL_SOURCE)
|
|
196
|
+
|
|
197
|
+
This ensures installed packages use their installation paths rather than
|
|
198
|
+
accidentally picking up development paths from CWD.
|
|
196
199
|
"""
|
|
197
|
-
#
|
|
200
|
+
# 1. Explicit environment variable override
|
|
198
201
|
if os.environ.get("CLAUDE_MPM_DEV_MODE", "").lower() in ("1", "true", "yes"):
|
|
199
202
|
logger.debug(
|
|
200
203
|
"Development mode forced via CLAUDE_MPM_DEV_MODE environment variable"
|
|
201
204
|
)
|
|
202
205
|
return DeploymentContext.DEVELOPMENT
|
|
203
206
|
|
|
204
|
-
# Check
|
|
205
|
-
# This handles the case where pipx claude-mpm is run from within the dev directory
|
|
206
|
-
cwd = _safe_cwd()
|
|
207
|
-
current = cwd
|
|
208
|
-
for _ in range(5): # Check up to 5 levels up from current directory
|
|
209
|
-
if (current / "pyproject.toml").exists() and (
|
|
210
|
-
current / "src" / "claude_mpm"
|
|
211
|
-
).exists():
|
|
212
|
-
# Check if this is the claude-mpm project
|
|
213
|
-
try:
|
|
214
|
-
pyproject_content = (current / "pyproject.toml").read_text()
|
|
215
|
-
if (
|
|
216
|
-
'name = "claude-mpm"' in pyproject_content
|
|
217
|
-
or '"claude-mpm"' in pyproject_content
|
|
218
|
-
):
|
|
219
|
-
logger.debug(
|
|
220
|
-
f"Detected claude-mpm development directory at {current}"
|
|
221
|
-
)
|
|
222
|
-
logger.debug(
|
|
223
|
-
"Using development mode for local source preference"
|
|
224
|
-
)
|
|
225
|
-
return DeploymentContext.DEVELOPMENT
|
|
226
|
-
except Exception: # nosec B110
|
|
227
|
-
pass
|
|
228
|
-
if current == current.parent:
|
|
229
|
-
break
|
|
230
|
-
current = current.parent
|
|
231
|
-
|
|
207
|
+
# 2. Check where the actual package is installed
|
|
232
208
|
try:
|
|
233
209
|
import claude_mpm
|
|
234
210
|
|
|
235
211
|
module_path = Path(claude_mpm.__file__).parent
|
|
212
|
+
package_str = str(module_path)
|
|
236
213
|
|
|
237
|
-
#
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
# Check if we should use development paths
|
|
242
|
-
# This could be because we're in a src/ directory or running from dev directory
|
|
243
|
-
if module_path.parent.name == "src":
|
|
244
|
-
return DeploymentContext.DEVELOPMENT
|
|
245
|
-
if "pipx" in str(module_path):
|
|
246
|
-
# Running via pipx but from within a development directory
|
|
247
|
-
# Use development mode to prefer local source over pipx installation
|
|
248
|
-
cwd = _safe_cwd()
|
|
249
|
-
current = cwd
|
|
250
|
-
for _ in range(5):
|
|
251
|
-
if (current / "src" / "claude_mpm").exists() and (
|
|
252
|
-
current / "pyproject.toml"
|
|
253
|
-
).exists():
|
|
254
|
-
logger.debug(
|
|
255
|
-
"Running pipx from development directory, using development mode"
|
|
256
|
-
)
|
|
257
|
-
return DeploymentContext.DEVELOPMENT
|
|
258
|
-
if current == current.parent:
|
|
259
|
-
break
|
|
260
|
-
current = current.parent
|
|
261
|
-
return DeploymentContext.EDITABLE_INSTALL
|
|
262
|
-
return DeploymentContext.EDITABLE_INSTALL
|
|
214
|
+
# UV tools installation (~/.local/share/uv/tools/)
|
|
215
|
+
if "/.local/share/uv/tools/" in package_str:
|
|
216
|
+
logger.debug(f"Detected uv tools installation at {module_path}")
|
|
217
|
+
return DeploymentContext.UV_TOOLS
|
|
263
218
|
|
|
264
|
-
#
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
)
|
|
219
|
+
# pipx installation (~/.local/pipx/venvs/)
|
|
220
|
+
if "/.local/pipx/venvs/" in package_str or "/pipx/" in package_str:
|
|
221
|
+
logger.debug(f"Detected pipx installation at {module_path}")
|
|
222
|
+
return DeploymentContext.PIPX_INSTALL
|
|
223
|
+
|
|
224
|
+
# site-packages (pip install) - but not editable
|
|
225
|
+
if "/site-packages/" in package_str and "/src/" not in package_str:
|
|
226
|
+
logger.debug(f"Detected pip installation at {module_path}")
|
|
227
|
+
return DeploymentContext.PIP_INSTALL
|
|
228
|
+
|
|
229
|
+
# Editable install (pip install -e) - module in src/
|
|
230
|
+
if module_path.parent.name == "src":
|
|
231
|
+
# Check if this is truly an editable install
|
|
232
|
+
if PathContext._is_editable_install():
|
|
233
|
+
logger.debug(f"Detected editable installation at {module_path}")
|
|
234
|
+
return DeploymentContext.EDITABLE_INSTALL
|
|
235
|
+
# Module in src/ but not editable - development mode
|
|
270
236
|
logger.debug(
|
|
271
237
|
f"Detected development mode via directory structure at {module_path}"
|
|
272
238
|
)
|
|
273
239
|
return DeploymentContext.DEVELOPMENT
|
|
274
240
|
|
|
275
|
-
#
|
|
276
|
-
if "
|
|
277
|
-
logger.debug(f"Detected pipx installation at {module_path}")
|
|
278
|
-
return DeploymentContext.PIPX_INSTALL
|
|
279
|
-
|
|
280
|
-
# Check for system package
|
|
281
|
-
if "dist-packages" in str(module_path):
|
|
241
|
+
# dist-packages (system package manager)
|
|
242
|
+
if "dist-packages" in package_str:
|
|
282
243
|
logger.debug(f"Detected system package installation at {module_path}")
|
|
283
244
|
return DeploymentContext.SYSTEM_PACKAGE
|
|
284
245
|
|
|
285
|
-
#
|
|
286
|
-
if "site-packages" in str(module_path):
|
|
287
|
-
# Already checked for editable above, so this is a regular pip install
|
|
288
|
-
logger.debug(f"Detected pip installation at {module_path}")
|
|
289
|
-
return DeploymentContext.PIP_INSTALL
|
|
290
|
-
|
|
291
|
-
# Default to pip install
|
|
246
|
+
# Default to pip install for any other installation
|
|
292
247
|
logger.debug(f"Defaulting to pip installation for {module_path}")
|
|
293
248
|
return DeploymentContext.PIP_INSTALL
|
|
294
249
|
|
|
295
250
|
except ImportError:
|
|
296
251
|
logger.debug(
|
|
297
|
-
"ImportError during
|
|
252
|
+
"ImportError during module path detection, checking CWD as fallback"
|
|
298
253
|
)
|
|
299
|
-
|
|
254
|
+
|
|
255
|
+
# 3. CWD-based detection (OPT-IN ONLY for explicit development work)
|
|
256
|
+
# Only use CWD if explicitly requested or no package installation found
|
|
257
|
+
if os.environ.get("CLAUDE_MPM_PREFER_LOCAL_SOURCE", "").lower() in (
|
|
258
|
+
"1",
|
|
259
|
+
"true",
|
|
260
|
+
"yes",
|
|
261
|
+
):
|
|
262
|
+
cwd = _safe_cwd()
|
|
263
|
+
current = cwd
|
|
264
|
+
for _ in range(5): # Check up to 5 levels up from current directory
|
|
265
|
+
if (current / "pyproject.toml").exists() and (
|
|
266
|
+
current / "src" / "claude_mpm"
|
|
267
|
+
).exists():
|
|
268
|
+
# Check if this is the claude-mpm project
|
|
269
|
+
try:
|
|
270
|
+
pyproject_content = (current / "pyproject.toml").read_text()
|
|
271
|
+
if (
|
|
272
|
+
'name = "claude-mpm"' in pyproject_content
|
|
273
|
+
or '"claude-mpm"' in pyproject_content
|
|
274
|
+
):
|
|
275
|
+
logger.debug(
|
|
276
|
+
f"CLAUDE_MPM_PREFER_LOCAL_SOURCE: Using development directory at {current}"
|
|
277
|
+
)
|
|
278
|
+
return DeploymentContext.DEVELOPMENT
|
|
279
|
+
except Exception: # nosec B110
|
|
280
|
+
pass
|
|
281
|
+
if current == current.parent:
|
|
282
|
+
break
|
|
283
|
+
current = current.parent
|
|
284
|
+
|
|
285
|
+
# Final fallback: assume development mode
|
|
286
|
+
logger.debug("No installation detected, defaulting to development mode")
|
|
287
|
+
return DeploymentContext.DEVELOPMENT
|
|
300
288
|
|
|
301
289
|
|
|
302
290
|
class UnifiedPathManager:
|