claude-mpm 4.24.0__py3-none-any.whl → 5.4.41__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/__init__.py +4 -0
- claude_mpm/agents/BASE_AGENT.md +164 -0
- claude_mpm/agents/{OUTPUT_STYLE.md → CLAUDE_MPM_OUTPUT_STYLE.md} +3 -48
- claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +2002 -0
- claude_mpm/agents/MEMORY.md +1 -1
- claude_mpm/agents/PM_INSTRUCTIONS.md +735 -925
- claude_mpm/agents/WORKFLOW.md +5 -254
- claude_mpm/agents/__init__.py +6 -0
- claude_mpm/agents/agent_loader.py +14 -48
- claude_mpm/agents/base_agent.json +7 -4
- claude_mpm/agents/frontmatter_validator.py +71 -3
- claude_mpm/agents/templates/circuit-breakers.md +1391 -0
- claude_mpm/agents/templates/context-management-examples.md +544 -0
- claude_mpm/agents/templates/{pm_red_flags.md → pm-red-flags.md} +48 -0
- claude_mpm/agents/templates/pr-workflow-examples.md +427 -0
- claude_mpm/agents/templates/research-gate-examples.md +669 -0
- claude_mpm/agents/templates/structured-questions-examples.md +615 -0
- claude_mpm/agents/templates/ticket-completeness-examples.md +139 -0
- claude_mpm/agents/templates/ticketing-examples.md +277 -0
- claude_mpm/cli/__init__.py +37 -2
- claude_mpm/cli/__main__.py +4 -0
- claude_mpm/cli/chrome_devtools_installer.py +175 -0
- claude_mpm/cli/commands/__init__.py +2 -0
- claude_mpm/cli/commands/agent_source.py +774 -0
- claude_mpm/cli/commands/agent_state_manager.py +180 -31
- claude_mpm/cli/commands/agents.py +1116 -55
- claude_mpm/cli/commands/agents_cleanup.py +210 -0
- claude_mpm/cli/commands/agents_discover.py +338 -0
- claude_mpm/cli/commands/aggregate.py +1 -1
- claude_mpm/cli/commands/analyze.py +3 -3
- claude_mpm/cli/commands/auto_configure.py +725 -242
- claude_mpm/cli/commands/config.py +95 -6
- claude_mpm/cli/commands/configure.py +1875 -46
- claude_mpm/cli/commands/configure_agent_display.py +29 -10
- claude_mpm/cli/commands/configure_navigation.py +63 -46
- claude_mpm/cli/commands/debug.py +12 -12
- claude_mpm/cli/commands/doctor.py +10 -2
- claude_mpm/cli/commands/hook_errors.py +277 -0
- claude_mpm/cli/commands/local_deploy.py +1 -4
- claude_mpm/cli/commands/mcp_install_commands.py +1 -1
- claude_mpm/cli/commands/mpm_init/core.py +229 -2
- claude_mpm/cli/commands/mpm_init/git_activity.py +10 -10
- claude_mpm/cli/commands/mpm_init/knowledge_extractor.py +481 -0
- claude_mpm/cli/commands/mpm_init/prompts.py +286 -6
- claude_mpm/cli/commands/postmortem.py +401 -0
- claude_mpm/cli/commands/profile.py +277 -0
- claude_mpm/cli/commands/run.py +123 -165
- claude_mpm/cli/commands/skill_source.py +694 -0
- claude_mpm/cli/commands/skills.py +782 -20
- claude_mpm/cli/commands/summarize.py +413 -0
- claude_mpm/cli/executor.py +96 -3
- claude_mpm/cli/interactive/agent_wizard.py +1030 -45
- claude_mpm/cli/parsers/agent_source_parser.py +171 -0
- claude_mpm/cli/parsers/agents_parser.py +307 -10
- claude_mpm/cli/parsers/auto_configure_parser.py +13 -138
- claude_mpm/cli/parsers/base_parser.py +65 -0
- claude_mpm/cli/parsers/config_parser.py +162 -39
- claude_mpm/cli/parsers/profile_parser.py +148 -0
- claude_mpm/cli/parsers/skill_source_parser.py +169 -0
- claude_mpm/cli/parsers/skills_parser.py +146 -0
- claude_mpm/cli/parsers/source_parser.py +138 -0
- claude_mpm/cli/startup.py +1280 -118
- claude_mpm/cli/startup_display.py +480 -0
- claude_mpm/cli/utils.py +1 -1
- claude_mpm/cli_module/commands.py +1 -1
- claude_mpm/commands/mpm-config.md +21 -134
- claude_mpm/commands/mpm-doctor.md +16 -20
- claude_mpm/commands/mpm-help.md +13 -283
- claude_mpm/commands/mpm-init.md +88 -489
- claude_mpm/commands/mpm-monitor.md +23 -401
- claude_mpm/commands/mpm-organize.md +72 -247
- claude_mpm/commands/mpm-postmortem.md +21 -0
- claude_mpm/commands/mpm-session-resume.md +30 -0
- claude_mpm/commands/mpm-status.md +13 -68
- claude_mpm/commands/mpm-ticket-view.md +109 -0
- claude_mpm/commands/mpm-version.md +13 -106
- claude_mpm/commands/mpm.md +10 -0
- claude_mpm/config/agent_presets.py +488 -0
- claude_mpm/config/agent_sources.py +352 -0
- claude_mpm/config/skill_presets.py +392 -0
- claude_mpm/config/skill_sources.py +590 -0
- claude_mpm/constants.py +13 -0
- claude_mpm/core/claude_runner.py +5 -34
- claude_mpm/core/config.py +15 -1
- claude_mpm/core/constants.py +1 -1
- claude_mpm/core/framework/__init__.py +3 -16
- claude_mpm/core/framework/formatters/content_formatter.py +3 -13
- claude_mpm/core/framework/loaders/agent_loader.py +8 -5
- claude_mpm/core/framework/loaders/file_loader.py +54 -101
- claude_mpm/core/framework/loaders/instruction_loader.py +66 -5
- claude_mpm/core/framework_loader.py +4 -2
- claude_mpm/core/hook_error_memory.py +381 -0
- claude_mpm/core/hook_manager.py +41 -2
- claude_mpm/core/interactive_session.py +91 -10
- claude_mpm/core/logger.py +16 -1
- claude_mpm/core/oneshot_session.py +71 -8
- claude_mpm/core/optimized_startup.py +59 -0
- claude_mpm/core/output_style_manager.py +173 -43
- claude_mpm/core/protocols/__init__.py +23 -0
- claude_mpm/core/protocols/runner_protocol.py +103 -0
- claude_mpm/core/protocols/session_protocol.py +131 -0
- claude_mpm/core/shared/config_loader.py +1 -1
- claude_mpm/core/shared/singleton_manager.py +11 -4
- claude_mpm/core/socketio_pool.py +3 -3
- claude_mpm/core/system_context.py +38 -0
- claude_mpm/core/unified_agent_registry.py +134 -16
- claude_mpm/core/unified_config.py +22 -0
- claude_mpm/dashboard/static/svelte-build/_app/env.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.B_FtCwCQ.css +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.Cl_eSA4x.css +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BgChzWQ1.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIXEwuWe.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWc5urbQ.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DMkZpdF2.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DjhvlsAc.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/N4qtv3Hx.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/uj46x2Wr.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.DTL5mJO-.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.DzuEhzqh.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/0.CAGBuiOw.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DFLC8jdE.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.DPvEihJJ.js +10 -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/experimental/cli_enhancements.py +1 -5
- claude_mpm/hooks/claude_hooks/__pycache__/__init__.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__/installer.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/correlation_manager.py +60 -0
- claude_mpm/hooks/claude_hooks/event_handlers.py +214 -79
- claude_mpm/hooks/claude_hooks/hook_handler.py +155 -1
- claude_mpm/hooks/claude_hooks/installer.py +33 -10
- claude_mpm/hooks/claude_hooks/memory_integration.py +28 -0
- claude_mpm/hooks/claude_hooks/response_tracking.py +2 -3
- claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager.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__/duplicate_detector.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 +30 -6
- claude_mpm/hooks/failure_learning/__init__.py +2 -8
- claude_mpm/hooks/failure_learning/failure_detection_hook.py +1 -6
- claude_mpm/hooks/failure_learning/fix_detection_hook.py +1 -6
- claude_mpm/hooks/failure_learning/learning_extraction_hook.py +1 -6
- claude_mpm/hooks/kuzu_response_hook.py +1 -5
- claude_mpm/hooks/memory_integration_hook.py +46 -1
- claude_mpm/init.py +63 -19
- claude_mpm/models/agent_definition.py +7 -0
- claude_mpm/models/git_repository.py +198 -0
- claude_mpm/scripts/claude-hook-handler.sh +60 -20
- claude_mpm/scripts/launch_monitor.py +93 -13
- claude_mpm/scripts/start_activity_logging.py +3 -1
- claude_mpm/services/agents/agent_builder.py +48 -12
- claude_mpm/services/agents/agent_preset_service.py +238 -0
- claude_mpm/services/agents/agent_recommendation_service.py +278 -0
- claude_mpm/services/agents/agent_review_service.py +280 -0
- claude_mpm/services/agents/agent_selection_service.py +484 -0
- claude_mpm/services/agents/auto_deploy_index_parser.py +569 -0
- claude_mpm/services/agents/cache_git_manager.py +621 -0
- claude_mpm/services/agents/deployment/agent_deployment.py +148 -2
- claude_mpm/services/agents/deployment/agent_discovery_service.py +104 -73
- claude_mpm/services/agents/deployment/agent_format_converter.py +1 -1
- claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +1 -5
- claude_mpm/services/agents/deployment/agent_metrics_collector.py +3 -3
- claude_mpm/services/agents/deployment/agent_restore_handler.py +1 -4
- claude_mpm/services/agents/deployment/agent_template_builder.py +238 -15
- claude_mpm/services/agents/deployment/agents_directory_resolver.py +101 -15
- claude_mpm/services/agents/deployment/async_agent_deployment.py +2 -1
- claude_mpm/services/agents/deployment/facade/deployment_facade.py +3 -3
- claude_mpm/services/agents/deployment/multi_source_deployment_service.py +422 -31
- claude_mpm/services/agents/deployment/pipeline/pipeline_executor.py +2 -2
- claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +1 -4
- claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +841 -0
- claude_mpm/services/agents/deployment/single_agent_deployer.py +2 -2
- claude_mpm/services/agents/deployment/system_instructions_deployer.py +168 -46
- claude_mpm/services/agents/deployment/validation/deployment_validator.py +2 -2
- claude_mpm/services/agents/git_source_manager.py +663 -0
- claude_mpm/services/agents/loading/base_agent_manager.py +1 -13
- claude_mpm/services/agents/loading/framework_agent_loader.py +9 -12
- claude_mpm/services/agents/local_template_manager.py +50 -10
- claude_mpm/services/agents/recommender.py +5 -3
- claude_mpm/services/agents/single_tier_deployment_service.py +696 -0
- claude_mpm/services/agents/sources/__init__.py +13 -0
- claude_mpm/services/agents/sources/agent_sync_state.py +516 -0
- claude_mpm/services/agents/sources/git_source_sync_service.py +1094 -0
- claude_mpm/services/agents/startup_sync.py +259 -0
- claude_mpm/services/agents/toolchain_detector.py +478 -0
- claude_mpm/services/analysis/__init__.py +35 -0
- claude_mpm/services/analysis/clone_detector.py +1030 -0
- claude_mpm/services/analysis/postmortem_reporter.py +474 -0
- claude_mpm/services/analysis/postmortem_service.py +765 -0
- claude_mpm/services/cli/session_pause_manager.py +1 -1
- claude_mpm/services/command_deployment_service.py +271 -6
- claude_mpm/services/core/base.py +7 -2
- claude_mpm/services/core/interfaces/__init__.py +1 -3
- claude_mpm/services/core/interfaces/health.py +1 -4
- claude_mpm/services/core/models/__init__.py +2 -11
- claude_mpm/services/diagnostics/checks/__init__.py +4 -0
- claude_mpm/services/diagnostics/checks/agent_check.py +2 -4
- claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
- claude_mpm/services/diagnostics/checks/instructions_check.py +1 -2
- claude_mpm/services/diagnostics/checks/mcp_check.py +0 -1
- claude_mpm/services/diagnostics/checks/mcp_services_check.py +7 -15
- claude_mpm/services/diagnostics/checks/monitor_check.py +0 -1
- claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
- claude_mpm/services/diagnostics/diagnostic_runner.py +9 -0
- claude_mpm/services/diagnostics/doctor_reporter.py +40 -10
- claude_mpm/services/event_bus/config.py +3 -1
- claude_mpm/services/event_bus/direct_relay.py +3 -3
- claude_mpm/services/events/consumers/logging.py +1 -2
- claude_mpm/services/git/__init__.py +21 -0
- claude_mpm/services/git/git_operations_service.py +579 -0
- claude_mpm/services/github/__init__.py +21 -0
- claude_mpm/services/github/github_cli_service.py +397 -0
- claude_mpm/services/infrastructure/monitoring/__init__.py +1 -5
- claude_mpm/services/infrastructure/monitoring/aggregator.py +1 -6
- claude_mpm/services/instructions/__init__.py +9 -0
- claude_mpm/services/instructions/instruction_cache_service.py +374 -0
- claude_mpm/services/local_ops/__init__.py +3 -13
- claude_mpm/services/local_ops/health_checks/__init__.py +1 -3
- claude_mpm/services/local_ops/health_manager.py +1 -4
- claude_mpm/services/local_ops/resource_monitor.py +1 -1
- claude_mpm/services/mcp_config_manager.py +75 -145
- claude_mpm/services/mcp_service_verifier.py +6 -3
- claude_mpm/services/model/model_router.py +1 -2
- claude_mpm/services/monitor/daemon.py +38 -11
- claude_mpm/services/monitor/daemon_manager.py +134 -21
- claude_mpm/services/monitor/management/lifecycle.py +8 -1
- claude_mpm/services/monitor/server.py +700 -24
- claude_mpm/services/pm_skills_deployer.py +676 -0
- claude_mpm/services/port_manager.py +1 -1
- claude_mpm/services/pr/__init__.py +14 -0
- claude_mpm/services/pr/pr_template_service.py +329 -0
- claude_mpm/services/profile_manager.py +331 -0
- claude_mpm/services/project/documentation_manager.py +2 -1
- claude_mpm/services/project/project_organizer.py +4 -0
- claude_mpm/services/project/toolchain_analyzer.py +3 -1
- claude_mpm/services/runner_configuration_service.py +16 -3
- claude_mpm/services/self_upgrade_service.py +120 -12
- claude_mpm/services/session_management_service.py +16 -4
- claude_mpm/services/skills/__init__.py +21 -0
- claude_mpm/services/skills/git_skill_source_manager.py +1297 -0
- claude_mpm/services/skills/selective_skill_deployer.py +704 -0
- claude_mpm/services/skills/skill_discovery_service.py +568 -0
- claude_mpm/services/skills/skill_to_agent_mapper.py +406 -0
- claude_mpm/services/skills_config.py +547 -0
- claude_mpm/services/skills_deployer.py +1072 -0
- claude_mpm/services/socketio/dashboard_server.py +1 -0
- claude_mpm/services/socketio/event_normalizer.py +51 -6
- claude_mpm/services/socketio/handlers/connection.py +1 -1
- claude_mpm/services/socketio/handlers/git.py +1 -1
- claude_mpm/services/socketio/server/core.py +387 -112
- claude_mpm/services/socketio/server/main.py +1 -3
- claude_mpm/services/system_instructions_service.py +1 -3
- claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +0 -3
- claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +0 -1
- claude_mpm/services/unified/deployment_strategies/vercel.py +1 -5
- claude_mpm/services/unified/unified_deployment.py +1 -5
- claude_mpm/services/version_control/conflict_resolution.py +6 -4
- claude_mpm/services/version_control/git_operations.py +103 -0
- claude_mpm/services/visualization/__init__.py +1 -5
- claude_mpm/services/visualization/mermaid_generator.py +2 -3
- claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +2 -2
- claude_mpm/skills/skill_manager.py +92 -3
- claude_mpm/skills/skills_registry.py +0 -1
- claude_mpm/templates/questions/__init__.py +38 -0
- claude_mpm/templates/questions/base.py +193 -0
- claude_mpm/templates/questions/pr_strategy.py +311 -0
- claude_mpm/templates/questions/project_init.py +385 -0
- claude_mpm/templates/questions/ticket_mgmt.py +394 -0
- claude_mpm/tools/__main__.py +8 -8
- claude_mpm/utils/agent_dependency_loader.py +91 -12
- claude_mpm/utils/agent_filters.py +261 -0
- claude_mpm/utils/dependency_cache.py +3 -1
- claude_mpm/utils/gitignore.py +244 -0
- claude_mpm/utils/migration.py +372 -0
- claude_mpm/utils/progress.py +387 -0
- claude_mpm/utils/robust_installer.py +49 -7
- claude_mpm/utils/structured_questions.py +619 -0
- {claude_mpm-4.24.0.dist-info → claude_mpm-5.4.41.dist-info}/METADATA +445 -122
- {claude_mpm-4.24.0.dist-info → claude_mpm-5.4.41.dist-info}/RECORD +298 -503
- claude_mpm-5.4.41.dist-info/entry_points.txt +5 -0
- claude_mpm-5.4.41.dist-info/licenses/LICENSE +94 -0
- claude_mpm-5.4.41.dist-info/licenses/LICENSE-FAQ.md +153 -0
- claude_mpm/agents/BASE_AGENT_TEMPLATE.md +0 -292
- claude_mpm/agents/BASE_DOCUMENTATION.md +0 -53
- claude_mpm/agents/BASE_OPS.md +0 -219
- claude_mpm/agents/BASE_PM.md +0 -468
- claude_mpm/agents/BASE_PROMPT_ENGINEER.md +0 -787
- claude_mpm/agents/BASE_QA.md +0 -167
- claude_mpm/agents/BASE_RESEARCH.md +0 -53
- claude_mpm/agents/base_agent_loader.py +0 -626
- claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -17
- claude_mpm/agents/templates/.claude-mpm/memories/engineer_memories.md +0 -3
- claude_mpm/agents/templates/agent-manager.json +0 -273
- claude_mpm/agents/templates/agentic-coder-optimizer.json +0 -248
- claude_mpm/agents/templates/api_qa.json +0 -183
- claude_mpm/agents/templates/circuit_breakers.md +0 -638
- claude_mpm/agents/templates/clerk-ops.json +0 -235
- claude_mpm/agents/templates/code_analyzer.json +0 -101
- claude_mpm/agents/templates/content-agent.json +0 -358
- claude_mpm/agents/templates/dart_engineer.json +0 -307
- claude_mpm/agents/templates/data_engineer.json +0 -225
- claude_mpm/agents/templates/documentation.json +0 -238
- claude_mpm/agents/templates/engineer.json +0 -210
- claude_mpm/agents/templates/gcp_ops_agent.json +0 -253
- claude_mpm/agents/templates/golang_engineer.json +0 -270
- claude_mpm/agents/templates/imagemagick.json +0 -264
- claude_mpm/agents/templates/java_engineer.json +0 -346
- claude_mpm/agents/templates/javascript_engineer_agent.json +0 -380
- claude_mpm/agents/templates/local_ops_agent.json +0 -1840
- claude_mpm/agents/templates/logs/prompts/agent_engineer_20250826_014258_728.md +0 -39
- claude_mpm/agents/templates/logs/prompts/agent_engineer_20250901_010124_142.md +0 -400
- claude_mpm/agents/templates/memory_manager.json +0 -158
- claude_mpm/agents/templates/nextjs_engineer.json +0 -285
- claude_mpm/agents/templates/ops.json +0 -185
- claude_mpm/agents/templates/php-engineer.json +0 -287
- claude_mpm/agents/templates/product_owner.json +0 -338
- claude_mpm/agents/templates/project_organizer.json +0 -144
- claude_mpm/agents/templates/prompt-engineer.json +0 -737
- claude_mpm/agents/templates/python_engineer.json +0 -387
- claude_mpm/agents/templates/qa.json +0 -243
- claude_mpm/agents/templates/react_engineer.json +0 -239
- claude_mpm/agents/templates/refactoring_engineer.json +0 -276
- claude_mpm/agents/templates/research.json +0 -188
- claude_mpm/agents/templates/ruby-engineer.json +0 -280
- claude_mpm/agents/templates/rust_engineer.json +0 -275
- claude_mpm/agents/templates/security.json +0 -202
- claude_mpm/agents/templates/svelte-engineer.json +0 -225
- claude_mpm/agents/templates/tauri_engineer.json +0 -274
- claude_mpm/agents/templates/ticketing.json +0 -178
- claude_mpm/agents/templates/typescript_engineer.json +0 -285
- claude_mpm/agents/templates/vercel_ops_agent.json +0 -412
- claude_mpm/agents/templates/version_control.json +0 -159
- claude_mpm/agents/templates/web_qa.json +0 -400
- claude_mpm/agents/templates/web_ui.json +0 -189
- claude_mpm/cli/commands/agents_detect.py +0 -380
- claude_mpm/cli/commands/agents_recommend.py +0 -309
- claude_mpm/cli/ticket_cli.py +0 -35
- claude_mpm/commands/mpm-agents-detect.md +0 -168
- claude_mpm/commands/mpm-agents-recommend.md +0 -214
- claude_mpm/commands/mpm-agents.md +0 -122
- claude_mpm/commands/mpm-auto-configure.md +0 -269
- claude_mpm/commands/mpm-resume.md +0 -372
- claude_mpm/commands/mpm-tickets.md +0 -151
- claude_mpm/dashboard/.claude-mpm/socketio-instances.json +0 -1
- claude_mpm/dashboard/analysis_runner.py +0 -455
- claude_mpm/dashboard/index.html +0 -13
- claude_mpm/dashboard/open_dashboard.py +0 -66
- claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +0 -188
- claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +0 -156
- claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +0 -38
- claude_mpm/dashboard/react/components/shared/FilterBar.module.css +0 -92
- claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +0 -248
- claude_mpm/dashboard/static/archive/activity_dashboard_test.html +0 -61
- claude_mpm/dashboard/static/archive/test_activity_connection.html +0 -179
- claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +0 -68
- claude_mpm/dashboard/static/archive/test_dashboard.html +0 -409
- claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +0 -519
- claude_mpm/dashboard/static/archive/test_dashboard_verification.html +0 -181
- claude_mpm/dashboard/static/archive/test_file_data.html +0 -315
- claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +0 -243
- claude_mpm/dashboard/static/archive/test_file_tree_fix.html +0 -234
- claude_mpm/dashboard/static/archive/test_file_tree_rename.html +0 -117
- claude_mpm/dashboard/static/archive/test_file_tree_tab.html +0 -115
- claude_mpm/dashboard/static/archive/test_file_viewer.html +0 -224
- claude_mpm/dashboard/static/archive/test_final_activity.html +0 -220
- claude_mpm/dashboard/static/archive/test_tab_fix.html +0 -139
- claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +0 -1
- claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
- claude_mpm/dashboard/static/built/components/agent-hierarchy.js +0 -777
- claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
- claude_mpm/dashboard/static/built/components/build-tracker.js +0 -333
- claude_mpm/dashboard/static/built/components/code-simple.js +0 -857
- claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +0 -353
- claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +0 -235
- claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +0 -409
- claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +0 -435
- claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
- claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
- claude_mpm/dashboard/static/built/components/connection-debug.js +0 -654
- claude_mpm/dashboard/static/built/components/diff-viewer.js +0 -891
- claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
- claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
- claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
- claude_mpm/dashboard/static/built/components/file-change-tracker.js +0 -443
- claude_mpm/dashboard/static/built/components/file-change-viewer.js +0 -690
- claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
- claude_mpm/dashboard/static/built/components/file-viewer.js +0 -2
- claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
- claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
- claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
- claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
- claude_mpm/dashboard/static/built/components/nav-bar.js +0 -145
- claude_mpm/dashboard/static/built/components/page-structure.js +0 -429
- claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
- claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
- claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
- claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
- claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
- claude_mpm/dashboard/static/built/connection-manager.js +0 -536
- claude_mpm/dashboard/static/built/dashboard.js +0 -2
- claude_mpm/dashboard/static/built/extension-error-handler.js +0 -164
- claude_mpm/dashboard/static/built/react/events.js +0 -30
- claude_mpm/dashboard/static/built/shared/dom-helpers.js +0 -396
- claude_mpm/dashboard/static/built/shared/event-bus.js +0 -330
- claude_mpm/dashboard/static/built/shared/event-filter-service.js +0 -540
- claude_mpm/dashboard/static/built/shared/logger.js +0 -385
- claude_mpm/dashboard/static/built/shared/page-structure.js +0 -249
- claude_mpm/dashboard/static/built/shared/tooltip-service.js +0 -253
- claude_mpm/dashboard/static/built/socket-client.js +0 -2
- claude_mpm/dashboard/static/built/tab-isolation-fix.js +0 -185
- claude_mpm/dashboard/static/css/activity.css +0 -1958
- claude_mpm/dashboard/static/css/connection-status.css +0 -370
- claude_mpm/dashboard/static/css/dashboard.css +0 -4701
- claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +0 -1
- claude_mpm/dashboard/static/dist/components/activity-tree.js +0 -2
- claude_mpm/dashboard/static/dist/components/agent-inference.js +0 -2
- claude_mpm/dashboard/static/dist/components/code-tree.js +0 -2
- claude_mpm/dashboard/static/dist/components/code-viewer.js +0 -2
- claude_mpm/dashboard/static/dist/components/event-processor.js +0 -2
- claude_mpm/dashboard/static/dist/components/event-viewer.js +0 -2
- claude_mpm/dashboard/static/dist/components/export-manager.js +0 -2
- claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +0 -2
- claude_mpm/dashboard/static/dist/components/file-viewer.js +0 -2
- claude_mpm/dashboard/static/dist/components/hud-library-loader.js +0 -2
- claude_mpm/dashboard/static/dist/components/hud-manager.js +0 -2
- claude_mpm/dashboard/static/dist/components/hud-visualizer.js +0 -2
- claude_mpm/dashboard/static/dist/components/module-viewer.js +0 -2
- claude_mpm/dashboard/static/dist/components/session-manager.js +0 -2
- claude_mpm/dashboard/static/dist/components/socket-manager.js +0 -2
- claude_mpm/dashboard/static/dist/components/ui-state-manager.js +0 -2
- claude_mpm/dashboard/static/dist/components/unified-data-viewer.js +0 -2
- claude_mpm/dashboard/static/dist/components/working-directory.js +0 -2
- claude_mpm/dashboard/static/dist/dashboard.js +0 -2
- claude_mpm/dashboard/static/dist/react/events.js +0 -30
- claude_mpm/dashboard/static/dist/socket-client.js +0 -2
- claude_mpm/dashboard/static/events.html +0 -607
- claude_mpm/dashboard/static/index.html +0 -635
- claude_mpm/dashboard/static/js/components/activity-tree.js +0 -1871
- claude_mpm/dashboard/static/js/components/agent-hierarchy.js +0 -777
- claude_mpm/dashboard/static/js/components/agent-inference.js +0 -956
- claude_mpm/dashboard/static/js/components/build-tracker.js +0 -333
- claude_mpm/dashboard/static/js/components/code-simple.js +0 -857
- claude_mpm/dashboard/static/js/components/connection-debug.js +0 -654
- claude_mpm/dashboard/static/js/components/diff-viewer.js +0 -891
- claude_mpm/dashboard/static/js/components/event-processor.js +0 -542
- claude_mpm/dashboard/static/js/components/event-viewer.js +0 -1155
- claude_mpm/dashboard/static/js/components/export-manager.js +0 -368
- claude_mpm/dashboard/static/js/components/file-change-tracker.js +0 -443
- claude_mpm/dashboard/static/js/components/file-change-viewer.js +0 -690
- claude_mpm/dashboard/static/js/components/file-tool-tracker.js +0 -724
- claude_mpm/dashboard/static/js/components/file-viewer.js +0 -580
- claude_mpm/dashboard/static/js/components/hud-library-loader.js +0 -211
- claude_mpm/dashboard/static/js/components/hud-manager.js +0 -671
- claude_mpm/dashboard/static/js/components/hud-visualizer.js +0 -1718
- claude_mpm/dashboard/static/js/components/module-viewer.js +0 -2764
- claude_mpm/dashboard/static/js/components/session-manager.js +0 -579
- claude_mpm/dashboard/static/js/components/socket-manager.js +0 -368
- claude_mpm/dashboard/static/js/components/ui-state-manager.js +0 -749
- claude_mpm/dashboard/static/js/components/unified-data-viewer.js +0 -1824
- claude_mpm/dashboard/static/js/components/working-directory.js +0 -920
- claude_mpm/dashboard/static/js/connection-manager.js +0 -536
- claude_mpm/dashboard/static/js/dashboard.js +0 -1896
- claude_mpm/dashboard/static/js/extension-error-handler.js +0 -164
- claude_mpm/dashboard/static/js/shared/dom-helpers.js +0 -396
- claude_mpm/dashboard/static/js/shared/event-bus.js +0 -330
- claude_mpm/dashboard/static/js/shared/logger.js +0 -385
- claude_mpm/dashboard/static/js/shared/tooltip-service.js +0 -253
- claude_mpm/dashboard/static/js/socket-client.js +0 -1457
- claude_mpm/dashboard/static/js/stores/dashboard-store.js +0 -562
- claude_mpm/dashboard/static/js/tab-isolation-fix.js +0 -185
- claude_mpm/dashboard/static/legacy/activity.html +0 -736
- claude_mpm/dashboard/static/legacy/agents.html +0 -786
- claude_mpm/dashboard/static/legacy/files.html +0 -747
- claude_mpm/dashboard/static/legacy/tools.html +0 -831
- claude_mpm/dashboard/static/monitors.html +0 -431
- claude_mpm/dashboard/static/production/events.html +0 -659
- claude_mpm/dashboard/static/production/main.html +0 -698
- claude_mpm/dashboard/static/production/monitors.html +0 -483
- claude_mpm/dashboard/static/socket.io.min.js +0 -7
- claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +0 -7
- claude_mpm/dashboard/static/test-archive/dashboard.html +0 -635
- claude_mpm/dashboard/static/test-archive/debug-events.html +0 -147
- claude_mpm/dashboard/static/test-archive/test-navigation.html +0 -256
- claude_mpm/dashboard/static/test-archive/test-react-exports.html +0 -180
- claude_mpm/dashboard/static/test-archive/test_debug.html +0 -25
- claude_mpm/dashboard/templates/code_simple.html +0 -153
- claude_mpm/dashboard/templates/index.html +0 -606
- claude_mpm/dashboard/test_dashboard.html +0 -372
- claude_mpm/scripts/mcp_server.py +0 -75
- claude_mpm/scripts/mcp_wrapper.py +0 -39
- claude_mpm/services/mcp_gateway/__init__.py +0 -159
- claude_mpm/services/mcp_gateway/auto_configure.py +0 -369
- claude_mpm/services/mcp_gateway/config/__init__.py +0 -17
- claude_mpm/services/mcp_gateway/config/config_loader.py +0 -296
- claude_mpm/services/mcp_gateway/config/config_schema.py +0 -243
- claude_mpm/services/mcp_gateway/config/configuration.py +0 -429
- claude_mpm/services/mcp_gateway/core/__init__.py +0 -43
- claude_mpm/services/mcp_gateway/core/base.py +0 -312
- claude_mpm/services/mcp_gateway/core/exceptions.py +0 -253
- claude_mpm/services/mcp_gateway/core/interfaces.py +0 -443
- claude_mpm/services/mcp_gateway/core/process_pool.py +0 -971
- claude_mpm/services/mcp_gateway/core/singleton_manager.py +0 -315
- claude_mpm/services/mcp_gateway/core/startup_verification.py +0 -316
- claude_mpm/services/mcp_gateway/main.py +0 -589
- claude_mpm/services/mcp_gateway/registry/__init__.py +0 -12
- claude_mpm/services/mcp_gateway/registry/service_registry.py +0 -412
- claude_mpm/services/mcp_gateway/registry/tool_registry.py +0 -489
- claude_mpm/services/mcp_gateway/server/__init__.py +0 -15
- claude_mpm/services/mcp_gateway/server/mcp_gateway.py +0 -419
- claude_mpm/services/mcp_gateway/server/stdio_handler.py +0 -372
- claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -714
- claude_mpm/services/mcp_gateway/tools/__init__.py +0 -36
- claude_mpm/services/mcp_gateway/tools/base_adapter.py +0 -485
- claude_mpm/services/mcp_gateway/tools/document_summarizer.py +0 -789
- claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +0 -654
- claude_mpm/services/mcp_gateway/tools/health_check_tool.py +0 -456
- claude_mpm/services/mcp_gateway/tools/hello_world.py +0 -551
- claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +0 -551
- claude_mpm/services/mcp_gateway/utils/__init__.py +0 -14
- claude_mpm/services/mcp_gateway/utils/package_version_checker.py +0 -160
- claude_mpm/services/mcp_gateway/utils/update_preferences.py +0 -170
- claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +0 -79
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +0 -178
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +0 -577
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +0 -467
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +0 -537
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +0 -730
- claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +0 -112
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +0 -146
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +0 -412
- claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +0 -81
- claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +0 -362
- claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +0 -312
- claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +0 -152
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +0 -668
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +0 -587
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +0 -438
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +0 -391
- claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +0 -119
- claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +0 -148
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +0 -483
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +0 -452
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +0 -449
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +0 -411
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +0 -14
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +0 -58
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +0 -68
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +0 -69
- claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +0 -131
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +0 -325
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +0 -490
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +0 -425
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +0 -499
- claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +0 -86
- claude_mpm/skills/bundled/main/internal-comms/SKILL.md +0 -43
- claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +0 -47
- claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +0 -65
- claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +0 -30
- claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +0 -16
- claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +0 -160
- claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +0 -412
- claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +0 -602
- claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +0 -915
- claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +0 -916
- claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +0 -752
- claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +0 -1237
- claude_mpm/skills/bundled/main/skill-creator/SKILL.md +0 -189
- claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +0 -500
- claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +0 -464
- claude_mpm/skills/bundled/main/skill-creator/references/examples.md +0 -619
- claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +0 -437
- claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +0 -231
- claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +0 -170
- claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +0 -602
- claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +0 -821
- claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +0 -742
- claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +0 -726
- claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +0 -764
- claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +0 -831
- claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +0 -226
- claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +0 -901
- claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +0 -901
- claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +0 -775
- claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +0 -937
- claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +0 -770
- claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +0 -961
- claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +0 -119
- claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +0 -253
- claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +0 -145
- claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +0 -543
- claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +0 -741
- claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +0 -470
- claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +0 -458
- claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +0 -639
- claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +0 -140
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +0 -572
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +0 -411
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +0 -569
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +0 -695
- claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +0 -184
- claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +0 -459
- claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +0 -479
- claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +0 -687
- claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +0 -758
- claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +0 -868
- claude_mpm-4.24.0.dist-info/entry_points.txt +0 -10
- claude_mpm-4.24.0.dist-info/licenses/LICENSE +0 -21
- /claude_mpm/agents/templates/{git_file_tracking.md → git-file-tracking.md} +0 -0
- /claude_mpm/agents/templates/{pm_examples.md → pm-examples.md} +0 -0
- /claude_mpm/agents/templates/{response_format.md → response-format.md} +0 -0
- /claude_mpm/agents/templates/{validation_templates.md → validation-templates.md} +0 -0
- {claude_mpm-4.24.0.dist-info → claude_mpm-5.4.41.dist-info}/WHEEL +0 -0
- {claude_mpm-4.24.0.dist-info → claude_mpm-5.4.41.dist-info}/top_level.txt +0 -0
|
@@ -12,7 +12,6 @@ MCP service installations.
|
|
|
12
12
|
import json
|
|
13
13
|
import subprocess
|
|
14
14
|
import sys
|
|
15
|
-
from datetime import datetime, timezone
|
|
16
15
|
from enum import Enum
|
|
17
16
|
from pathlib import Path
|
|
18
17
|
from typing import Dict, Optional, Tuple
|
|
@@ -698,175 +697,106 @@ class MCPConfigManager:
|
|
|
698
697
|
|
|
699
698
|
return config
|
|
700
699
|
|
|
701
|
-
def
|
|
700
|
+
def check_mcp_services_available(self) -> Tuple[bool, str]:
|
|
702
701
|
"""
|
|
703
|
-
|
|
702
|
+
Check if required MCP services are available in ~/.claude.json (READ-ONLY).
|
|
704
703
|
|
|
705
|
-
This method
|
|
706
|
-
|
|
707
|
-
|
|
708
|
-
2. Fix incorrect configurations
|
|
709
|
-
3. Update all projects, not just the current one
|
|
704
|
+
This method performs a READ-ONLY check of MCP service availability.
|
|
705
|
+
It does NOT modify ~/.claude.json. Users should install and configure
|
|
706
|
+
MCP services themselves via pip, npx, or Claude Desktop.
|
|
710
707
|
|
|
711
708
|
Returns:
|
|
712
|
-
Tuple of (
|
|
709
|
+
Tuple of (all_available: bool, message: str)
|
|
713
710
|
"""
|
|
714
|
-
|
|
715
|
-
|
|
716
|
-
added_services = []
|
|
717
|
-
|
|
718
|
-
# Load existing Claude config or create minimal structure
|
|
719
|
-
claude_config = {}
|
|
720
|
-
if self.claude_config_path.exists():
|
|
721
|
-
try:
|
|
722
|
-
with self.claude_config_path.open() as f:
|
|
723
|
-
claude_config = json.load(f)
|
|
724
|
-
except Exception as e:
|
|
725
|
-
self.logger.error(f"Error reading {self.claude_config_path}: {e}")
|
|
726
|
-
return False, f"Failed to read Claude config: {e}"
|
|
711
|
+
# Get services Claude MPM expects to use (from ~/.claude-mpm/config/)
|
|
712
|
+
expected_services = self.get_filtered_services()
|
|
727
713
|
|
|
728
|
-
|
|
729
|
-
|
|
730
|
-
claude_config["projects"] = {}
|
|
731
|
-
updated = True
|
|
714
|
+
if not expected_services:
|
|
715
|
+
return True, "No MCP services configured in Claude MPM"
|
|
732
716
|
|
|
733
|
-
#
|
|
734
|
-
|
|
717
|
+
# Load Claude config (read-only)
|
|
718
|
+
if not self.claude_config_path.exists():
|
|
719
|
+
return False, f"Claude config not found at {self.claude_config_path}"
|
|
735
720
|
|
|
736
|
-
|
|
737
|
-
|
|
721
|
+
try:
|
|
722
|
+
with self.claude_config_path.open() as f:
|
|
723
|
+
claude_config = json.load(f)
|
|
724
|
+
except Exception as e:
|
|
725
|
+
return False, f"Failed to read Claude config: {e}"
|
|
738
726
|
|
|
739
|
-
#
|
|
727
|
+
# Check current project
|
|
740
728
|
current_project_key = str(self.project_root)
|
|
741
|
-
|
|
742
|
-
projects_to_update.append(current_project_key)
|
|
743
|
-
# Initialize new project structure
|
|
744
|
-
claude_config["projects"][current_project_key] = {
|
|
745
|
-
"allowedTools": [],
|
|
746
|
-
"history": [],
|
|
747
|
-
"mcpContextUris": [],
|
|
748
|
-
"mcpServers": {},
|
|
749
|
-
"enabledMcpjsonServers": [],
|
|
750
|
-
"disabledMcpjsonServers": [],
|
|
751
|
-
"hasTrustDialogAccepted": False,
|
|
752
|
-
"projectOnboardingSeenCount": 0,
|
|
753
|
-
"hasClaudeMdExternalIncludesApproved": False,
|
|
754
|
-
"hasClaudeMdExternalIncludesWarningShown": False,
|
|
755
|
-
}
|
|
756
|
-
updated = True
|
|
757
|
-
|
|
758
|
-
# Update each project's MCP configurations
|
|
759
|
-
for project_key in projects_to_update:
|
|
760
|
-
project_config = claude_config["projects"][project_key]
|
|
761
|
-
|
|
762
|
-
# Ensure mcpServers section exists
|
|
763
|
-
if "mcpServers" not in project_config:
|
|
764
|
-
project_config["mcpServers"] = {}
|
|
765
|
-
updated = True
|
|
766
|
-
|
|
767
|
-
# Check and fix each service configuration - now filtered by startup config
|
|
768
|
-
services_to_configure = self.get_filtered_services()
|
|
769
|
-
|
|
770
|
-
for service_name, correct_config in services_to_configure.items():
|
|
771
|
-
# Check if service exists and has correct configuration
|
|
772
|
-
existing_config = project_config["mcpServers"].get(service_name)
|
|
773
|
-
|
|
774
|
-
# Determine if we need to update
|
|
775
|
-
needs_update = False
|
|
776
|
-
if not existing_config:
|
|
777
|
-
# Service is missing
|
|
778
|
-
needs_update = True
|
|
779
|
-
added_services.append(f"{service_name} in {Path(project_key).name}")
|
|
780
|
-
# Service exists, check if configuration is correct
|
|
781
|
-
# Compare command and args (the most critical parts)
|
|
782
|
-
elif existing_config.get("command") != correct_config.get(
|
|
783
|
-
"command"
|
|
784
|
-
) or existing_config.get("args") != correct_config.get("args"):
|
|
785
|
-
needs_update = True
|
|
786
|
-
fixed_services.append(f"{service_name} in {Path(project_key).name}")
|
|
787
|
-
|
|
788
|
-
# Update configuration if needed
|
|
789
|
-
if needs_update:
|
|
790
|
-
project_config["mcpServers"][service_name] = correct_config
|
|
791
|
-
updated = True
|
|
792
|
-
self.logger.debug(
|
|
793
|
-
f"Updated MCP service config for {service_name} in project {Path(project_key).name}"
|
|
794
|
-
)
|
|
729
|
+
project_config = claude_config.get("projects", {}).get(current_project_key)
|
|
795
730
|
|
|
796
|
-
|
|
797
|
-
|
|
798
|
-
|
|
799
|
-
|
|
731
|
+
if not project_config:
|
|
732
|
+
missing = list(expected_services.keys())
|
|
733
|
+
return (
|
|
734
|
+
False,
|
|
735
|
+
f"Current project not configured in Claude. Missing services: {', '.join(missing)}",
|
|
736
|
+
)
|
|
800
737
|
|
|
801
|
-
|
|
802
|
-
|
|
803
|
-
|
|
804
|
-
|
|
805
|
-
|
|
806
|
-
# Remove services that are not in the enabled list
|
|
807
|
-
services_to_remove = []
|
|
808
|
-
for service_name in project_config["mcpServers"]:
|
|
809
|
-
if service_name not in enabled_services:
|
|
810
|
-
services_to_remove.append(service_name)
|
|
811
|
-
|
|
812
|
-
for service_name in services_to_remove:
|
|
813
|
-
del project_config["mcpServers"][service_name]
|
|
814
|
-
updated = True
|
|
815
|
-
self.logger.debug(
|
|
816
|
-
f"Removed disabled service {service_name} from project {Path(project_key).name}"
|
|
817
|
-
)
|
|
818
|
-
|
|
819
|
-
# Write updated config if changes were made
|
|
820
|
-
if updated:
|
|
821
|
-
try:
|
|
822
|
-
# Create backup if file exists and is large (> 100KB)
|
|
823
|
-
if self.claude_config_path.exists():
|
|
824
|
-
file_size = self.claude_config_path.stat().st_size
|
|
825
|
-
if file_size > 100000: # 100KB
|
|
826
|
-
backup_path = self.claude_config_path.with_suffix(
|
|
827
|
-
f".backup.{datetime.now(timezone.utc).strftime('%Y%m%d_%H%M%S')}.json"
|
|
828
|
-
)
|
|
829
|
-
import shutil
|
|
738
|
+
# Check which services are missing
|
|
739
|
+
mcp_servers = project_config.get("mcpServers", {})
|
|
740
|
+
missing_services = [
|
|
741
|
+
name for name in expected_services if name not in mcp_servers
|
|
742
|
+
]
|
|
830
743
|
|
|
831
|
-
|
|
832
|
-
|
|
744
|
+
if missing_services:
|
|
745
|
+
msg = (
|
|
746
|
+
f"Missing MCP services: {', '.join(missing_services)}. "
|
|
747
|
+
f"Install via: pip install {' '.join(missing_services)} "
|
|
748
|
+
f"or configure in Claude Desktop"
|
|
749
|
+
)
|
|
750
|
+
return False, msg
|
|
833
751
|
|
|
834
|
-
|
|
835
|
-
|
|
836
|
-
|
|
752
|
+
return (
|
|
753
|
+
True,
|
|
754
|
+
f"All required MCP services available ({len(expected_services)} services)",
|
|
755
|
+
)
|
|
837
756
|
|
|
838
|
-
|
|
839
|
-
|
|
840
|
-
|
|
841
|
-
f"Added MCP services: {', '.join(added_services[:3])}"
|
|
842
|
-
)
|
|
843
|
-
if fixed_services:
|
|
844
|
-
messages.append(
|
|
845
|
-
f"Fixed MCP services: {', '.join(fixed_services[:3])}"
|
|
846
|
-
)
|
|
757
|
+
def ensure_mcp_services_configured(self) -> Tuple[bool, str]:
|
|
758
|
+
"""
|
|
759
|
+
DEPRECATED: Auto-configuring ~/.claude.json is no longer supported.
|
|
847
760
|
|
|
848
|
-
|
|
849
|
-
|
|
850
|
-
|
|
851
|
-
|
|
852
|
-
|
|
853
|
-
return False, f"Failed to write configuration: {e}"
|
|
761
|
+
As of v4.15.0+, MCP services are user-controlled. Users should install
|
|
762
|
+
and configure MCP services themselves via:
|
|
763
|
+
- pip install <service-name>
|
|
764
|
+
- npx @modelcontextprotocol/...
|
|
765
|
+
- Claude Desktop UI
|
|
854
766
|
|
|
855
|
-
|
|
767
|
+
This method now only performs a read-only check and logs a deprecation warning.
|
|
768
|
+
Use check_mcp_services_available() for read-only checks.
|
|
769
|
+
|
|
770
|
+
Returns:
|
|
771
|
+
Tuple of (success, message)
|
|
772
|
+
"""
|
|
773
|
+
import warnings
|
|
774
|
+
|
|
775
|
+
warnings.warn(
|
|
776
|
+
"ensure_mcp_services_configured() is deprecated and will be removed in v6.0.0. "
|
|
777
|
+
"MCP services are now user-controlled. Use check_mcp_services_available() instead.",
|
|
778
|
+
DeprecationWarning,
|
|
779
|
+
stacklevel=2,
|
|
780
|
+
)
|
|
781
|
+
|
|
782
|
+
# Delegate to read-only check
|
|
783
|
+
return self.check_mcp_services_available()
|
|
856
784
|
|
|
857
785
|
def update_mcp_config(self, force_pipx: bool = True) -> Tuple[bool, str]:
|
|
858
786
|
"""
|
|
859
|
-
|
|
787
|
+
DEPRECATED: Check MCP configuration in ~/.claude.json (READ-ONLY).
|
|
788
|
+
|
|
789
|
+
This method no longer modifies ~/.claude.json. Users should install
|
|
790
|
+
and configure MCP services themselves.
|
|
860
791
|
|
|
861
792
|
Args:
|
|
862
|
-
force_pipx:
|
|
793
|
+
force_pipx: Ignored (kept for backward compatibility)
|
|
863
794
|
|
|
864
795
|
Returns:
|
|
865
|
-
Tuple of (success, message)
|
|
796
|
+
Tuple of (success, message) from read-only check
|
|
866
797
|
"""
|
|
867
|
-
#
|
|
868
|
-
|
|
869
|
-
return self.ensure_mcp_services_configured()
|
|
798
|
+
# Delegate to read-only check
|
|
799
|
+
return self.check_mcp_services_available()
|
|
870
800
|
|
|
871
801
|
def update_project_mcp_config(self, force_pipx: bool = True) -> Tuple[bool, str]:
|
|
872
802
|
"""
|
|
@@ -620,12 +620,15 @@ class MCPServiceVerifier:
|
|
|
620
620
|
return True
|
|
621
621
|
|
|
622
622
|
if "claude-mpm configure" in diagnostic.fix_command:
|
|
623
|
-
#
|
|
623
|
+
# Check if services are available (read-only)
|
|
624
624
|
from .mcp_config_manager import MCPConfigManager
|
|
625
625
|
|
|
626
626
|
manager = MCPConfigManager()
|
|
627
|
-
|
|
628
|
-
|
|
627
|
+
available, message = manager.check_mcp_services_available()
|
|
628
|
+
if not available:
|
|
629
|
+
# Cannot auto-fix - user must install services manually
|
|
630
|
+
self.logger.warning(f"Cannot auto-fix: {message}")
|
|
631
|
+
return available
|
|
629
632
|
|
|
630
633
|
except Exception as e:
|
|
631
634
|
self.logger.error(f"Auto-fix failed for {service_name}: {e}")
|
|
@@ -356,8 +356,7 @@ class ModelRouter(BaseService, IModelRouter):
|
|
|
356
356
|
if require_success and not await self.ollama_provider.is_available():
|
|
357
357
|
if self.strategy == RoutingStrategy.PRIVACY_FIRST:
|
|
358
358
|
error_msg = (
|
|
359
|
-
"Ollama not available. Privacy mode enabled - "
|
|
360
|
-
"not sending to cloud."
|
|
359
|
+
"Ollama not available. Privacy mode enabled - not sending to cloud."
|
|
361
360
|
)
|
|
362
361
|
else:
|
|
363
362
|
error_msg = "Ollama not available and required by configuration"
|
|
@@ -44,6 +44,7 @@ class UnifiedMonitorDaemon:
|
|
|
44
44
|
daemon_mode: bool = False,
|
|
45
45
|
pid_file: Optional[str] = None,
|
|
46
46
|
log_file: Optional[str] = None,
|
|
47
|
+
enable_hot_reload: bool = False,
|
|
47
48
|
):
|
|
48
49
|
"""Initialize the unified monitor daemon.
|
|
49
50
|
|
|
@@ -53,10 +54,12 @@ class UnifiedMonitorDaemon:
|
|
|
53
54
|
daemon_mode: Whether to run as background daemon
|
|
54
55
|
pid_file: Path to PID file for daemon mode
|
|
55
56
|
log_file: Path to log file for daemon mode
|
|
57
|
+
enable_hot_reload: Enable file watching and hot reload for development
|
|
56
58
|
"""
|
|
57
59
|
self.host = host
|
|
58
60
|
self.port = port
|
|
59
61
|
self.daemon_mode = daemon_mode
|
|
62
|
+
self.enable_hot_reload = enable_hot_reload
|
|
60
63
|
self.logger = get_logger(__name__)
|
|
61
64
|
|
|
62
65
|
# Use new consolidated DaemonManager for all daemon operations
|
|
@@ -75,7 +78,9 @@ class UnifiedMonitorDaemon:
|
|
|
75
78
|
)
|
|
76
79
|
|
|
77
80
|
# Core server
|
|
78
|
-
self.server = UnifiedMonitorServer(
|
|
81
|
+
self.server = UnifiedMonitorServer(
|
|
82
|
+
host=host, port=port, enable_hot_reload=enable_hot_reload
|
|
83
|
+
)
|
|
79
84
|
|
|
80
85
|
# Health monitoring
|
|
81
86
|
self.health_monitor = HealthMonitor(port=port)
|
|
@@ -88,11 +93,12 @@ class UnifiedMonitorDaemon:
|
|
|
88
93
|
self.shutdown_event = threading.Event()
|
|
89
94
|
|
|
90
95
|
def _get_default_pid_file(self) -> str:
|
|
91
|
-
"""Get default PID file path."""
|
|
96
|
+
"""Get default PID file path with port number to support multiple daemons."""
|
|
92
97
|
project_root = Path.cwd()
|
|
93
98
|
claude_mpm_dir = project_root / ".claude-mpm"
|
|
94
99
|
claude_mpm_dir.mkdir(exist_ok=True)
|
|
95
|
-
|
|
100
|
+
# Include port in filename to support multiple daemon instances
|
|
101
|
+
return str(claude_mpm_dir / f"monitor-daemon-{self.port}.pid")
|
|
96
102
|
|
|
97
103
|
def start(self, force_restart: bool = False) -> bool:
|
|
98
104
|
"""Start the unified monitor daemon.
|
|
@@ -145,16 +151,26 @@ class UnifiedMonitorDaemon:
|
|
|
145
151
|
if self.daemon_manager.is_running():
|
|
146
152
|
existing_pid = self.daemon_manager.get_pid()
|
|
147
153
|
if not force_restart:
|
|
148
|
-
|
|
154
|
+
msg = f"Daemon already running on port {self.port} with PID {existing_pid}"
|
|
155
|
+
self.logger.warning(msg)
|
|
156
|
+
# If we're in subprocess mode, this is an error - we should have cleaned up
|
|
157
|
+
if os.environ.get("CLAUDE_MPM_SUBPROCESS_DAEMON") == "1":
|
|
158
|
+
self.logger.error(
|
|
159
|
+
f"SUBPROCESS ERROR: {msg} - This should not happen in subprocess mode!"
|
|
160
|
+
)
|
|
149
161
|
return False
|
|
150
162
|
# Force restart was already handled above
|
|
151
163
|
|
|
152
164
|
# Check for our service on the port
|
|
153
165
|
is_ours, pid = self.daemon_manager.is_our_service()
|
|
154
166
|
if is_ours and pid and not force_restart:
|
|
155
|
-
self.
|
|
156
|
-
|
|
157
|
-
|
|
167
|
+
msg = f"Our service already running on port {self.port} (PID: {pid})"
|
|
168
|
+
self.logger.warning(msg)
|
|
169
|
+
# If we're in subprocess mode, this is an error - we should have cleaned up
|
|
170
|
+
if os.environ.get("CLAUDE_MPM_SUBPROCESS_DAEMON") == "1":
|
|
171
|
+
self.logger.error(
|
|
172
|
+
f"SUBPROCESS ERROR: {msg} - This should not happen in subprocess mode!"
|
|
173
|
+
)
|
|
158
174
|
return False
|
|
159
175
|
|
|
160
176
|
# Use subprocess approach for clean daemon startup (v4.2.40)
|
|
@@ -169,20 +185,29 @@ class UnifiedMonitorDaemon:
|
|
|
169
185
|
if os.environ.get("CLAUDE_MPM_SUBPROCESS_DAEMON") == "1":
|
|
170
186
|
# We're in a subprocess started by start_daemon_subprocess
|
|
171
187
|
# We need to write the PID file ourselves since parent didn't
|
|
172
|
-
self.logger.info("Running in subprocess daemon mode
|
|
188
|
+
self.logger.info(f"Running in subprocess daemon mode on port {self.port}")
|
|
189
|
+
self.logger.info(f"Subprocess PID: {os.getpid()}")
|
|
190
|
+
self.logger.info(f"PID file path: {self.daemon_manager.pid_file}")
|
|
173
191
|
self.daemon_manager.write_pid_file()
|
|
192
|
+
self.logger.info("PID file written successfully")
|
|
174
193
|
|
|
175
194
|
# Setup signal handlers for graceful shutdown
|
|
176
195
|
self._setup_signal_handlers()
|
|
196
|
+
self.logger.info("Signal handlers configured")
|
|
177
197
|
|
|
178
198
|
# Start the server (this will run until shutdown)
|
|
199
|
+
self.logger.info("Starting server in subprocess mode...")
|
|
179
200
|
try:
|
|
180
201
|
result = self._run_server()
|
|
181
202
|
if not result:
|
|
182
203
|
self.logger.error("Failed to start server in subprocess mode")
|
|
204
|
+
return result
|
|
205
|
+
self.logger.info("Server started successfully in subprocess mode")
|
|
183
206
|
return result
|
|
184
207
|
except Exception as e:
|
|
185
|
-
self.logger.error(
|
|
208
|
+
self.logger.error(
|
|
209
|
+
f"Server startup exception in subprocess: {e}", exc_info=True
|
|
210
|
+
)
|
|
186
211
|
raise
|
|
187
212
|
else:
|
|
188
213
|
# Legacy fork approach (kept for compatibility but not used by default)
|
|
@@ -479,7 +504,7 @@ class UnifiedMonitorDaemon:
|
|
|
479
504
|
except OSError:
|
|
480
505
|
if i < 9:
|
|
481
506
|
self.logger.debug(
|
|
482
|
-
f"Port {self.port} still in use, waiting... ({i+1}/10)"
|
|
507
|
+
f"Port {self.port} still in use, waiting... ({i + 1}/10)"
|
|
483
508
|
)
|
|
484
509
|
time.sleep(1)
|
|
485
510
|
else:
|
|
@@ -490,7 +515,9 @@ class UnifiedMonitorDaemon:
|
|
|
490
515
|
|
|
491
516
|
# Recreate the server and health monitor after stop() sets them to None
|
|
492
517
|
self.logger.info(f"Recreating server components for {self.host}:{self.port}")
|
|
493
|
-
self.server = UnifiedMonitorServer(
|
|
518
|
+
self.server = UnifiedMonitorServer(
|
|
519
|
+
host=self.host, port=self.port, enable_hot_reload=self.enable_hot_reload
|
|
520
|
+
)
|
|
494
521
|
self.health_monitor = HealthMonitor(port=self.port)
|
|
495
522
|
|
|
496
523
|
# Reset the shutdown event for the new run
|
|
@@ -34,6 +34,11 @@ from typing import Optional, Tuple
|
|
|
34
34
|
from ...core.enums import OperationResult
|
|
35
35
|
from ...core.logging_config import get_logger
|
|
36
36
|
|
|
37
|
+
# Exit code constants for signal handling
|
|
38
|
+
EXIT_NORMAL = 0
|
|
39
|
+
EXIT_SIGKILL = 137 # 128 + SIGKILL(9) - forced termination
|
|
40
|
+
EXIT_SIGTERM = 143 # 128 + SIGTERM(15) - graceful shutdown
|
|
41
|
+
|
|
37
42
|
|
|
38
43
|
class DaemonManager:
|
|
39
44
|
"""Centralized manager for all daemon lifecycle operations.
|
|
@@ -80,18 +85,20 @@ class DaemonManager:
|
|
|
80
85
|
self.startup_status_file = None
|
|
81
86
|
|
|
82
87
|
def _get_default_pid_file(self) -> Path:
|
|
83
|
-
"""Get default PID file path."""
|
|
88
|
+
"""Get default PID file path with port number to support multiple daemons."""
|
|
84
89
|
project_root = Path.cwd()
|
|
85
90
|
claude_mpm_dir = project_root / ".claude-mpm"
|
|
86
91
|
claude_mpm_dir.mkdir(exist_ok=True)
|
|
87
|
-
|
|
92
|
+
# Include port in filename to support multiple daemon instances
|
|
93
|
+
return claude_mpm_dir / f"monitor-daemon-{self.port}.pid"
|
|
88
94
|
|
|
89
95
|
def _get_default_log_file(self) -> Path:
|
|
90
|
-
"""Get default log file path."""
|
|
96
|
+
"""Get default log file path with port number to support multiple daemons."""
|
|
91
97
|
project_root = Path.cwd()
|
|
92
98
|
claude_mpm_dir = project_root / ".claude-mpm"
|
|
93
99
|
claude_mpm_dir.mkdir(exist_ok=True)
|
|
94
|
-
|
|
100
|
+
# Include port in filename to support multiple daemon instances
|
|
101
|
+
return claude_mpm_dir / f"monitor-daemon-{self.port}.log"
|
|
95
102
|
|
|
96
103
|
def cleanup_port_conflicts(self, max_retries: int = 3) -> bool:
|
|
97
104
|
"""Clean up any processes using the daemon port.
|
|
@@ -471,6 +478,55 @@ class DaemonManager:
|
|
|
471
478
|
|
|
472
479
|
return None
|
|
473
480
|
|
|
481
|
+
def _verify_daemon_health(self, max_attempts: int = 3) -> bool:
|
|
482
|
+
"""Verify daemon is healthy by checking HTTP health endpoint.
|
|
483
|
+
|
|
484
|
+
Args:
|
|
485
|
+
max_attempts: Maximum number of connection attempts
|
|
486
|
+
|
|
487
|
+
Returns:
|
|
488
|
+
True if health check passes, False otherwise
|
|
489
|
+
"""
|
|
490
|
+
try:
|
|
491
|
+
import requests
|
|
492
|
+
|
|
493
|
+
for attempt in range(max_attempts):
|
|
494
|
+
try:
|
|
495
|
+
# Try to connect to health endpoint
|
|
496
|
+
response = requests.get(
|
|
497
|
+
f"http://{self.host}:{self.port}/health", timeout=2
|
|
498
|
+
)
|
|
499
|
+
|
|
500
|
+
if response.status_code == 200:
|
|
501
|
+
self.logger.debug(
|
|
502
|
+
f"Health check passed on attempt {attempt + 1}/{max_attempts}"
|
|
503
|
+
)
|
|
504
|
+
return True
|
|
505
|
+
|
|
506
|
+
self.logger.debug(
|
|
507
|
+
f"Health check returned status {response.status_code} on attempt {attempt + 1}/{max_attempts}"
|
|
508
|
+
)
|
|
509
|
+
|
|
510
|
+
except requests.exceptions.RequestException as e:
|
|
511
|
+
self.logger.debug(
|
|
512
|
+
f"Health check attempt {attempt + 1}/{max_attempts} failed: {e}"
|
|
513
|
+
)
|
|
514
|
+
|
|
515
|
+
# Wait before retry (except on last attempt)
|
|
516
|
+
if attempt < max_attempts - 1:
|
|
517
|
+
time.sleep(1)
|
|
518
|
+
|
|
519
|
+
self.logger.debug(f"Health check failed after {max_attempts} attempts")
|
|
520
|
+
return False
|
|
521
|
+
|
|
522
|
+
except ImportError:
|
|
523
|
+
# requests not available, skip health check
|
|
524
|
+
self.logger.debug("requests library not available, skipping health check")
|
|
525
|
+
return True
|
|
526
|
+
except Exception as e:
|
|
527
|
+
self.logger.debug(f"Health check error: {e}")
|
|
528
|
+
return False
|
|
529
|
+
|
|
474
530
|
def start_daemon(self, force_restart: bool = False) -> bool:
|
|
475
531
|
"""Start the daemon with automatic cleanup and retry.
|
|
476
532
|
|
|
@@ -505,7 +561,7 @@ class DaemonManager:
|
|
|
505
561
|
# Use subprocess for clean daemon startup (v4.2.40)
|
|
506
562
|
# This avoids fork() issues with Python threading
|
|
507
563
|
if self.use_subprocess_daemon():
|
|
508
|
-
return self.start_daemon_subprocess()
|
|
564
|
+
return self.start_daemon_subprocess(force_restart=force_restart)
|
|
509
565
|
# Fallback to traditional fork (kept for compatibility)
|
|
510
566
|
return self.daemonize()
|
|
511
567
|
|
|
@@ -523,12 +579,15 @@ class DaemonManager:
|
|
|
523
579
|
# Otherwise, use subprocess for monitor daemon to avoid threading issues
|
|
524
580
|
return True
|
|
525
581
|
|
|
526
|
-
def start_daemon_subprocess(self) -> bool:
|
|
582
|
+
def start_daemon_subprocess(self, force_restart: bool = False) -> bool:
|
|
527
583
|
"""Start daemon using subprocess.Popen for clean process isolation.
|
|
528
584
|
|
|
529
585
|
This avoids all the fork() + threading issues by starting the monitor
|
|
530
586
|
in a completely fresh process with no inherited threads or locks.
|
|
531
587
|
|
|
588
|
+
Args:
|
|
589
|
+
force_restart: Whether this is a force restart (helps interpret exit codes)
|
|
590
|
+
|
|
532
591
|
Returns:
|
|
533
592
|
True if daemon started successfully
|
|
534
593
|
"""
|
|
@@ -588,36 +647,90 @@ class DaemonManager:
|
|
|
588
647
|
pid = process.pid
|
|
589
648
|
self.logger.info(f"Monitor subprocess started with PID {pid}")
|
|
590
649
|
|
|
591
|
-
# Wait for the subprocess to write its PID file
|
|
650
|
+
# Wait for the subprocess to write its PID file and bind to port
|
|
592
651
|
# The subprocess will write the PID file after it starts successfully
|
|
593
652
|
max_wait = 10 # seconds
|
|
594
653
|
start_time = time.time()
|
|
654
|
+
pid_file_found = False
|
|
655
|
+
port_bound = False
|
|
656
|
+
|
|
657
|
+
self.logger.debug(f"Waiting up to {max_wait}s for daemon to start...")
|
|
595
658
|
|
|
596
659
|
while time.time() - start_time < max_wait:
|
|
597
660
|
# Check if process is still running
|
|
598
|
-
|
|
599
|
-
|
|
661
|
+
returncode = process.poll()
|
|
662
|
+
if returncode is not None:
|
|
663
|
+
# Process exited - interpret exit code with context
|
|
664
|
+
# Exit codes 137 (SIGKILL) and 143 (SIGTERM) are common during daemon replacement
|
|
665
|
+
if returncode == EXIT_SIGKILL:
|
|
666
|
+
# SIGKILL - process was forcefully terminated
|
|
667
|
+
if force_restart:
|
|
668
|
+
# This is expected during force restart - old daemon was killed
|
|
669
|
+
self.logger.info(
|
|
670
|
+
f"Previous monitor instance replaced (exit {EXIT_SIGKILL}: SIGKILL during force restart)"
|
|
671
|
+
)
|
|
672
|
+
else:
|
|
673
|
+
# Unexpected SIGKILL - something else killed our new daemon
|
|
674
|
+
self.logger.warning(
|
|
675
|
+
f"Monitor subprocess terminated unexpectedly (exit {EXIT_SIGKILL}: SIGKILL). "
|
|
676
|
+
f"Check {self.log_file} for details."
|
|
677
|
+
)
|
|
678
|
+
return False
|
|
679
|
+
if returncode == EXIT_SIGTERM:
|
|
680
|
+
# SIGTERM - graceful shutdown requested
|
|
681
|
+
self.logger.info(
|
|
682
|
+
f"Monitor subprocess cleanly terminated (exit {EXIT_SIGTERM}: SIGTERM, graceful shutdown)"
|
|
683
|
+
)
|
|
684
|
+
return False
|
|
685
|
+
if returncode == EXIT_NORMAL:
|
|
686
|
+
# Normal exit
|
|
687
|
+
self.logger.info(
|
|
688
|
+
f"Monitor subprocess exited normally (exit code {EXIT_NORMAL})"
|
|
689
|
+
)
|
|
690
|
+
return False
|
|
691
|
+
# Unexpected exit code - this IS an error
|
|
692
|
+
self.logger.error(
|
|
693
|
+
f"Monitor daemon subprocess exited prematurely with code {returncode}"
|
|
694
|
+
)
|
|
600
695
|
self.logger.error(
|
|
601
|
-
f"
|
|
696
|
+
f"Port {self.port} daemon failed to start. Check {self.log_file} for details."
|
|
602
697
|
)
|
|
603
698
|
return False
|
|
604
699
|
|
|
605
700
|
# Check if PID file was written
|
|
606
|
-
if self.pid_file.exists():
|
|
701
|
+
if not pid_file_found and self.pid_file.exists():
|
|
607
702
|
try:
|
|
608
703
|
with self.pid_file.open() as f:
|
|
609
704
|
written_pid = int(f.read().strip())
|
|
610
705
|
if written_pid == pid:
|
|
611
|
-
|
|
612
|
-
|
|
613
|
-
|
|
614
|
-
)
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
|
|
618
|
-
|
|
619
|
-
|
|
620
|
-
|
|
706
|
+
pid_file_found = True
|
|
707
|
+
self.logger.debug(
|
|
708
|
+
f"PID file found with correct PID {pid}"
|
|
709
|
+
)
|
|
710
|
+
except Exception as e:
|
|
711
|
+
self.logger.debug(f"Error reading PID file: {e}")
|
|
712
|
+
|
|
713
|
+
# Check if port is bound (health check)
|
|
714
|
+
if not port_bound and not self._is_port_available():
|
|
715
|
+
# Port NOT available means it's in use (good!)
|
|
716
|
+
port_bound = True
|
|
717
|
+
self.logger.debug(f"Port {self.port} is now bound")
|
|
718
|
+
|
|
719
|
+
# Success criteria: both PID file exists and port is bound
|
|
720
|
+
if pid_file_found and port_bound:
|
|
721
|
+
self.logger.info(
|
|
722
|
+
f"Monitor daemon successfully started on port {self.port} (PID: {pid})"
|
|
723
|
+
)
|
|
724
|
+
# Additional health check: verify we can connect
|
|
725
|
+
if self._verify_daemon_health(max_attempts=3):
|
|
726
|
+
self.logger.info("Daemon health check passed")
|
|
727
|
+
return True
|
|
728
|
+
self.logger.warning(
|
|
729
|
+
"Daemon started but health check failed - may still be initializing"
|
|
730
|
+
)
|
|
731
|
+
return (
|
|
732
|
+
True # Return success anyway if PID file and port are good
|
|
733
|
+
)
|
|
621
734
|
|
|
622
735
|
time.sleep(0.5)
|
|
623
736
|
|