claude-mpm 3.4.10__py3-none-any.whl → 5.4.55__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/BUILD_NUMBER +1 -0
- claude_mpm/VERSION +1 -0
- claude_mpm/__init__.py +50 -12
- claude_mpm/__main__.py +7 -2
- claude_mpm/agents/BASE_AGENT.md +164 -0
- claude_mpm/agents/BASE_ENGINEER.md +658 -0
- claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +290 -0
- claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +2002 -0
- claude_mpm/agents/MEMORY.md +72 -0
- claude_mpm/agents/PM_INSTRUCTIONS.md +1402 -0
- claude_mpm/agents/WORKFLOW.md +111 -0
- claude_mpm/agents/__init__.py +92 -80
- claude_mpm/agents/agent-template.yaml +83 -0
- claude_mpm/agents/agent_loader.py +560 -745
- claude_mpm/agents/agent_loader_integration.py +53 -55
- claude_mpm/agents/agents_metadata.py +186 -27
- claude_mpm/agents/async_agent_loader.py +436 -0
- claude_mpm/agents/base_agent.json +8 -4
- claude_mpm/agents/frontmatter_validator.py +754 -0
- claude_mpm/agents/system_agent_config.py +222 -155
- claude_mpm/agents/templates/README.md +465 -0
- claude_mpm/agents/templates/__init__.py +17 -13
- claude_mpm/agents/templates/circuit-breakers.md +1391 -0
- claude_mpm/agents/templates/context-management-examples.md +544 -0
- claude_mpm/agents/templates/git-file-tracking.md +584 -0
- claude_mpm/agents/templates/pm-examples.md +474 -0
- claude_mpm/agents/templates/pm-red-flags.md +310 -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/response-format.md +583 -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/agents/templates/validation-templates.md +312 -0
- claude_mpm/cli/__init__.py +90 -128
- claude_mpm/cli/__main__.py +33 -0
- claude_mpm/cli/chrome_devtools_installer.py +175 -0
- claude_mpm/cli/commands/__init__.py +36 -12
- claude_mpm/cli/commands/agent_manager.py +1403 -0
- claude_mpm/cli/commands/agent_source.py +774 -0
- claude_mpm/cli/commands/agent_state_manager.py +335 -0
- claude_mpm/cli/commands/agents.py +2503 -168
- claude_mpm/cli/commands/agents_cleanup.py +210 -0
- claude_mpm/cli/commands/agents_discover.py +338 -0
- claude_mpm/cli/commands/aggregate.py +540 -0
- claude_mpm/cli/commands/analyze.py +553 -0
- claude_mpm/cli/commands/analyze_code.py +528 -0
- claude_mpm/cli/commands/auto_configure.py +1053 -0
- claude_mpm/cli/commands/cleanup.py +588 -0
- claude_mpm/cli/commands/cleanup_orphaned_agents.py +150 -0
- claude_mpm/cli/commands/config.py +586 -0
- claude_mpm/cli/commands/configure.py +2654 -0
- claude_mpm/cli/commands/configure_agent_display.py +282 -0
- claude_mpm/cli/commands/configure_behavior_manager.py +204 -0
- claude_mpm/cli/commands/configure_hook_manager.py +225 -0
- claude_mpm/cli/commands/configure_models.py +18 -0
- claude_mpm/cli/commands/configure_navigation.py +184 -0
- claude_mpm/cli/commands/configure_paths.py +104 -0
- claude_mpm/cli/commands/configure_persistence.py +254 -0
- claude_mpm/cli/commands/configure_startup_manager.py +646 -0
- claude_mpm/cli/commands/configure_template_editor.py +497 -0
- claude_mpm/cli/commands/configure_validators.py +73 -0
- claude_mpm/cli/commands/dashboard.py +286 -0
- claude_mpm/cli/commands/debug.py +1386 -0
- claude_mpm/cli/commands/doctor.py +243 -0
- claude_mpm/cli/commands/hook_errors.py +277 -0
- claude_mpm/cli/commands/info.py +195 -74
- claude_mpm/cli/commands/local_deploy.py +534 -0
- claude_mpm/cli/commands/mcp.py +205 -0
- claude_mpm/cli/commands/mcp_command_router.py +161 -0
- claude_mpm/cli/commands/mcp_config.py +154 -0
- claude_mpm/cli/commands/mcp_config_commands.py +20 -0
- claude_mpm/cli/commands/mcp_external_commands.py +249 -0
- claude_mpm/cli/commands/mcp_install_commands.py +346 -0
- claude_mpm/cli/commands/mcp_pipx_config.py +208 -0
- claude_mpm/cli/commands/mcp_server_commands.py +155 -0
- claude_mpm/cli/commands/mcp_setup_external.py +868 -0
- claude_mpm/cli/commands/mcp_tool_commands.py +34 -0
- claude_mpm/cli/commands/memory.py +585 -846
- claude_mpm/cli/commands/monitor.py +228 -310
- claude_mpm/cli/commands/mpm_init/__init__.py +73 -0
- claude_mpm/cli/commands/mpm_init/core.py +759 -0
- claude_mpm/cli/commands/mpm_init/display.py +341 -0
- claude_mpm/cli/commands/mpm_init/git_activity.py +427 -0
- claude_mpm/cli/commands/mpm_init/knowledge_extractor.py +481 -0
- claude_mpm/cli/commands/mpm_init/modes.py +397 -0
- claude_mpm/cli/commands/mpm_init/prompts.py +722 -0
- claude_mpm/cli/commands/mpm_init_cli.py +396 -0
- claude_mpm/cli/commands/mpm_init_handler.py +195 -0
- claude_mpm/cli/commands/postmortem.py +401 -0
- claude_mpm/cli/commands/profile.py +276 -0
- claude_mpm/cli/commands/run.py +910 -488
- claude_mpm/cli/commands/search.py +458 -0
- claude_mpm/cli/commands/skill_source.py +694 -0
- claude_mpm/cli/commands/skills.py +1246 -0
- claude_mpm/cli/commands/summarize.py +413 -0
- claude_mpm/cli/commands/tickets.py +536 -53
- claude_mpm/cli/commands/uninstall.py +176 -0
- claude_mpm/cli/commands/upgrade.py +152 -0
- claude_mpm/cli/commands/verify.py +119 -0
- claude_mpm/cli/executor.py +297 -0
- claude_mpm/cli/helpers.py +105 -0
- claude_mpm/cli/interactive/__init__.py +21 -0
- claude_mpm/cli/interactive/agent_wizard.py +1947 -0
- claude_mpm/cli/interactive/skills_wizard.py +491 -0
- claude_mpm/cli/parser.py +87 -563
- claude_mpm/cli/parsers/__init__.py +35 -0
- claude_mpm/cli/parsers/agent_manager_parser.py +393 -0
- claude_mpm/cli/parsers/agent_source_parser.py +171 -0
- claude_mpm/cli/parsers/agents_parser.py +575 -0
- claude_mpm/cli/parsers/analyze_code_parser.py +170 -0
- claude_mpm/cli/parsers/analyze_parser.py +135 -0
- claude_mpm/cli/parsers/auto_configure_parser.py +120 -0
- claude_mpm/cli/parsers/base_parser.py +644 -0
- claude_mpm/cli/parsers/config_parser.py +208 -0
- claude_mpm/cli/parsers/configure_parser.py +138 -0
- claude_mpm/cli/parsers/dashboard_parser.py +113 -0
- claude_mpm/cli/parsers/debug_parser.py +319 -0
- claude_mpm/cli/parsers/local_deploy_parser.py +227 -0
- claude_mpm/cli/parsers/mcp_parser.py +195 -0
- claude_mpm/cli/parsers/memory_parser.py +138 -0
- claude_mpm/cli/parsers/monitor_parser.py +142 -0
- claude_mpm/cli/parsers/mpm_init_parser.py +311 -0
- claude_mpm/cli/parsers/profile_parser.py +147 -0
- claude_mpm/cli/parsers/run_parser.py +157 -0
- claude_mpm/cli/parsers/search_parser.py +245 -0
- claude_mpm/cli/parsers/skill_source_parser.py +169 -0
- claude_mpm/cli/parsers/skills_parser.py +277 -0
- claude_mpm/cli/parsers/source_parser.py +138 -0
- claude_mpm/cli/parsers/tickets_parser.py +203 -0
- claude_mpm/cli/shared/__init__.py +40 -0
- claude_mpm/cli/shared/argument_patterns.py +205 -0
- claude_mpm/cli/shared/base_command.py +242 -0
- claude_mpm/cli/shared/error_handling.py +242 -0
- claude_mpm/cli/shared/output_formatters.py +241 -0
- claude_mpm/cli/startup.py +1743 -0
- claude_mpm/cli/startup_display.py +480 -0
- claude_mpm/cli/startup_logging.py +839 -0
- claude_mpm/cli/utils.py +136 -47
- claude_mpm/cli_module/__init__.py +6 -6
- claude_mpm/cli_module/args.py +188 -140
- claude_mpm/cli_module/commands.py +79 -70
- claude_mpm/cli_module/migration_example.py +42 -64
- claude_mpm/commands/__init__.py +14 -0
- claude_mpm/commands/mpm-config.md +28 -0
- claude_mpm/commands/mpm-doctor.md +20 -0
- claude_mpm/commands/mpm-help.md +20 -0
- claude_mpm/commands/mpm-init.md +120 -0
- claude_mpm/commands/mpm-monitor.md +31 -0
- claude_mpm/commands/mpm-organize.md +120 -0
- claude_mpm/commands/mpm-postmortem.md +21 -0
- claude_mpm/commands/mpm-session-resume.md +30 -0
- claude_mpm/commands/mpm-status.md +20 -0
- claude_mpm/commands/mpm-ticket-view.md +109 -0
- claude_mpm/commands/mpm-version.md +20 -0
- claude_mpm/commands/mpm.md +31 -0
- claude_mpm/config/__init__.py +42 -2
- claude_mpm/config/agent_config.py +402 -0
- claude_mpm/config/agent_presets.py +488 -0
- claude_mpm/config/agent_sources.py +352 -0
- claude_mpm/config/experimental_features.py +217 -0
- claude_mpm/config/model_config.py +428 -0
- claude_mpm/config/paths.py +258 -0
- claude_mpm/config/skill_presets.py +392 -0
- claude_mpm/config/skill_sources.py +590 -0
- claude_mpm/config/socketio_config.py +125 -83
- claude_mpm/constants.py +132 -22
- claude_mpm/core/__init__.py +62 -36
- claude_mpm/core/agent_name_normalizer.py +71 -73
- claude_mpm/core/agent_registry.py +385 -492
- claude_mpm/core/agent_session_manager.py +81 -70
- claude_mpm/core/api_validator.py +330 -0
- claude_mpm/core/base_service.py +159 -122
- claude_mpm/core/cache.py +560 -0
- claude_mpm/core/claude_runner.py +696 -916
- claude_mpm/core/config.py +613 -122
- claude_mpm/core/config_aliases.py +74 -73
- claude_mpm/core/config_constants.py +314 -0
- claude_mpm/core/constants.py +361 -0
- claude_mpm/core/container.py +646 -104
- claude_mpm/core/enums.py +452 -0
- claude_mpm/core/error_handler.py +623 -0
- claude_mpm/core/exceptions.py +536 -0
- claude_mpm/core/factories.py +105 -109
- claude_mpm/core/file_utils.py +764 -0
- claude_mpm/core/framework/__init__.py +25 -0
- claude_mpm/core/framework/formatters/__init__.py +11 -0
- claude_mpm/core/framework/formatters/capability_generator.py +367 -0
- claude_mpm/core/framework/formatters/content_formatter.py +278 -0
- claude_mpm/core/framework/formatters/context_generator.py +185 -0
- claude_mpm/core/framework/loaders/__init__.py +13 -0
- claude_mpm/core/framework/loaders/agent_loader.py +213 -0
- claude_mpm/core/framework/loaders/file_loader.py +176 -0
- claude_mpm/core/framework/loaders/instruction_loader.py +222 -0
- claude_mpm/core/framework/loaders/packaged_loader.py +232 -0
- claude_mpm/core/framework/processors/__init__.py +11 -0
- claude_mpm/core/framework/processors/memory_processor.py +230 -0
- claude_mpm/core/framework/processors/metadata_processor.py +146 -0
- claude_mpm/core/framework/processors/template_processor.py +244 -0
- claude_mpm/core/framework_loader.py +485 -414
- claude_mpm/core/hook_error_memory.py +381 -0
- claude_mpm/core/hook_manager.py +246 -86
- claude_mpm/core/hook_performance_config.py +147 -0
- claude_mpm/core/injectable_service.py +72 -63
- claude_mpm/core/instruction_reinforcement_hook.py +267 -0
- claude_mpm/core/interactive_session.py +670 -0
- claude_mpm/core/interfaces.py +570 -164
- claude_mpm/core/lazy.py +467 -0
- claude_mpm/core/log_manager.py +707 -0
- claude_mpm/core/logger.py +295 -134
- claude_mpm/core/logging_config.py +474 -0
- claude_mpm/core/logging_utils.py +520 -0
- claude_mpm/core/minimal_framework_loader.py +24 -22
- claude_mpm/core/mixins.py +30 -29
- claude_mpm/core/oneshot_session.py +594 -0
- claude_mpm/core/optimized_agent_loader.py +479 -0
- claude_mpm/core/optimized_startup.py +554 -0
- claude_mpm/core/output_style_manager.py +483 -0
- claude_mpm/core/pm_hook_interceptor.py +197 -82
- 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/service_registry.py +153 -116
- claude_mpm/core/session_manager.py +179 -64
- claude_mpm/core/shared/__init__.py +17 -0
- claude_mpm/core/shared/config_loader.py +326 -0
- claude_mpm/core/shared/path_resolver.py +281 -0
- claude_mpm/core/shared/singleton_manager.py +221 -0
- claude_mpm/core/socketio_pool.py +400 -137
- claude_mpm/core/system_context.py +38 -0
- claude_mpm/core/tool_access_control.py +64 -57
- claude_mpm/core/types.py +307 -0
- claude_mpm/core/typing_utils.py +553 -0
- claude_mpm/core/unified_agent_registry.py +969 -0
- claude_mpm/core/unified_config.py +570 -0
- claude_mpm/core/unified_paths.py +941 -0
- claude_mpm/dashboard/__init__.py +12 -0
- claude_mpm/dashboard/api/simple_directory.py +261 -0
- claude_mpm/dashboard/static/svelte-build/_app/env.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.DWzvg0-y.css +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.ThTw9_ym.css +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/4TdZjIqw.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/5shd3_w0.js +24 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B0uc0UOD.js +36 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B7RN905-.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B7xVLGWV.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BIF9m_hv.js +61 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BKjSRqUr.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BPYeabCQ.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BQaXIfA_.js +331 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BSNlmTZj.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Be7GpZd6.js +7 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Bh0LDWpI.js +145 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BofRWZRR.js +10 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BovzEFCE.js +30 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C30mlcqg.js +165 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C4B-KCzX.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C4JcI4KD.js +122 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CBBdVcY8.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CDuw-vjf.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C_Usid8X.js +15 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Cfqx1Qun.js +10 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CiIAseT4.js +128 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CmKTTxBW.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CnA0NrzZ.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Cs_tUR18.js +24 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Cu_Erd72.js +261 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CyWMqx4W.js +43 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CzZX-COe.js +220 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CzeYkLYB.js +65 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D3k0OPJN.js +4 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9lljYKQ.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DGkLK5U1.js +267 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DI7hHRFL.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DLVjFsZ3.js +139 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DUrLdbGD.js +89 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DVp1hx9R.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DY1XQ8fi.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DZX00Y4g.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Da0KfYnO.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DaimHw_p.js +68 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Dfy6j1xT.js +323 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Dhb8PKl3.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Dle-35c7.js +64 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DmxopI1J.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DwBR2MJi.js +60 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/GYwsonyD.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Gi6I4Gst.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/NqQ1dWOy.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/RJiighC3.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Vzk33B_K.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/ZGh7QtNv.js +7 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/bT1r9zLR.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/bTOqqlTd.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/eNVUfhuA.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/iEWssX7S.js +162 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/sQeU3Y1z.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/uuIeMWc-.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.D6-I5TpK.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.NWzMBYRp.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/0.m1gL8KXf.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.CgNOuw-d.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.C0GcWctS.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -0
- claude_mpm/dashboard/static/svelte-build/favicon.svg +7 -0
- claude_mpm/dashboard/static/svelte-build/index.html +36 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/fonts/generate_fonts.py +58 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_tfms.py +114 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_ttfs.py +122 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/format_json.py +28 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/parse_tfm.py +211 -0
- claude_mpm/experimental/__init__.py +10 -0
- claude_mpm/experimental/cli_enhancements.py +104 -89
- claude_mpm/generators/__init__.py +1 -1
- claude_mpm/generators/agent_profile_generator.py +76 -66
- claude_mpm/hooks/__init__.py +37 -1
- claude_mpm/hooks/base_hook.py +37 -32
- claude_mpm/hooks/claude_hooks/__init__.py +1 -1
- 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/connection_pool.py +250 -0
- claude_mpm/hooks/claude_hooks/correlation_manager.py +60 -0
- claude_mpm/hooks/claude_hooks/event_handlers.py +888 -0
- claude_mpm/hooks/claude_hooks/hook_handler.py +652 -875
- claude_mpm/hooks/claude_hooks/hook_wrapper.sh +10 -7
- claude_mpm/hooks/claude_hooks/installer.py +806 -0
- claude_mpm/hooks/claude_hooks/memory_integration.py +249 -0
- claude_mpm/hooks/claude_hooks/response_tracking.py +412 -0
- claude_mpm/hooks/claude_hooks/services/__init__.py +15 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/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 +229 -0
- claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +254 -0
- claude_mpm/hooks/claude_hooks/services/duplicate_detector.py +106 -0
- claude_mpm/hooks/claude_hooks/services/state_manager.py +284 -0
- claude_mpm/hooks/claude_hooks/services/subagent_processor.py +374 -0
- claude_mpm/hooks/claude_hooks/tool_analysis.py +224 -0
- claude_mpm/hooks/failure_learning/__init__.py +54 -0
- claude_mpm/hooks/failure_learning/failure_detection_hook.py +230 -0
- claude_mpm/hooks/failure_learning/fix_detection_hook.py +212 -0
- claude_mpm/hooks/failure_learning/learning_extraction_hook.py +281 -0
- claude_mpm/hooks/instruction_reinforcement.py +301 -0
- claude_mpm/hooks/kuzu_enrichment_hook.py +263 -0
- claude_mpm/hooks/kuzu_memory_hook.py +386 -0
- claude_mpm/hooks/kuzu_response_hook.py +179 -0
- claude_mpm/hooks/memory_integration_hook.py +201 -107
- claude_mpm/hooks/session_resume_hook.py +121 -0
- claude_mpm/hooks/templates/pre_tool_use_simple.py +78 -0
- claude_mpm/hooks/templates/pre_tool_use_template.py +323 -0
- claude_mpm/hooks/tool_call_interceptor.py +92 -76
- claude_mpm/hooks/validation_hooks.py +62 -54
- claude_mpm/init.py +518 -83
- claude_mpm/models/__init__.py +9 -9
- claude_mpm/models/agent_definition.py +40 -23
- claude_mpm/models/agent_session.py +538 -0
- claude_mpm/models/git_repository.py +198 -0
- claude_mpm/models/resume_log.py +340 -0
- claude_mpm/schemas/__init__.py +12 -0
- claude_mpm/scripts/__init__.py +15 -0
- claude_mpm/scripts/claude-hook-handler.sh +227 -0
- claude_mpm/scripts/launch_monitor.py +165 -0
- claude_mpm/scripts/mpm_doctor.py +322 -0
- claude_mpm/scripts/socketio_daemon.py +189 -200
- claude_mpm/scripts/start_activity_logging.py +91 -0
- claude_mpm/services/__init__.py +208 -39
- claude_mpm/services/agent_capabilities_service.py +266 -0
- claude_mpm/services/agents/__init__.py +89 -0
- claude_mpm/services/agents/agent_builder.py +514 -0
- 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_config_manager.py +796 -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/__init__.py +21 -0
- claude_mpm/services/agents/deployment/agent_config_provider.py +410 -0
- claude_mpm/services/agents/deployment/agent_configuration_manager.py +358 -0
- claude_mpm/services/agents/deployment/agent_definition_factory.py +80 -0
- claude_mpm/services/agents/deployment/agent_deployment.py +1037 -0
- claude_mpm/services/agents/deployment/agent_discovery_service.py +546 -0
- claude_mpm/services/agents/deployment/agent_environment_manager.py +288 -0
- claude_mpm/services/agents/deployment/agent_filesystem_manager.py +383 -0
- claude_mpm/services/agents/deployment/agent_format_converter.py +505 -0
- claude_mpm/services/agents/deployment/agent_frontmatter_validator.py +160 -0
- claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +957 -0
- claude_mpm/services/agents/deployment/agent_metrics_collector.py +273 -0
- claude_mpm/services/agents/deployment/agent_operation_service.py +573 -0
- claude_mpm/services/agents/deployment/agent_record_service.py +418 -0
- claude_mpm/services/agents/deployment/agent_restore_handler.py +84 -0
- claude_mpm/services/agents/deployment/agent_state_service.py +381 -0
- claude_mpm/services/agents/deployment/agent_template_builder.py +1369 -0
- claude_mpm/services/agents/deployment/agent_validator.py +376 -0
- claude_mpm/services/agents/deployment/agent_version_manager.py +322 -0
- claude_mpm/services/{agent_versioning.py → agents/deployment/agent_versioning.py} +10 -13
- claude_mpm/services/agents/deployment/agents_directory_resolver.py +149 -0
- claude_mpm/services/agents/deployment/async_agent_deployment.py +768 -0
- claude_mpm/services/agents/deployment/base_agent_locator.py +132 -0
- claude_mpm/services/agents/deployment/config/__init__.py +13 -0
- claude_mpm/services/agents/deployment/config/deployment_config.py +181 -0
- claude_mpm/services/agents/deployment/config/deployment_config_manager.py +200 -0
- claude_mpm/services/agents/deployment/deployment_config_loader.py +178 -0
- claude_mpm/services/agents/deployment/deployment_results_manager.py +185 -0
- claude_mpm/services/agents/deployment/deployment_type_detector.py +120 -0
- claude_mpm/services/agents/deployment/deployment_wrapper.py +129 -0
- claude_mpm/services/agents/deployment/facade/__init__.py +18 -0
- claude_mpm/services/agents/deployment/facade/async_deployment_executor.py +159 -0
- claude_mpm/services/agents/deployment/facade/deployment_executor.py +70 -0
- claude_mpm/services/agents/deployment/facade/deployment_facade.py +269 -0
- claude_mpm/services/agents/deployment/facade/sync_deployment_executor.py +178 -0
- claude_mpm/services/agents/deployment/interface_adapter.py +226 -0
- claude_mpm/services/agents/deployment/lifecycle_health_checker.py +85 -0
- claude_mpm/services/agents/deployment/lifecycle_performance_tracker.py +100 -0
- claude_mpm/services/agents/deployment/local_template_deployment.py +362 -0
- claude_mpm/services/agents/deployment/multi_source_deployment_service.py +1478 -0
- claude_mpm/services/agents/deployment/pipeline/__init__.py +32 -0
- claude_mpm/services/agents/deployment/pipeline/pipeline_builder.py +158 -0
- claude_mpm/services/agents/deployment/pipeline/pipeline_context.py +162 -0
- claude_mpm/services/agents/deployment/pipeline/pipeline_executor.py +169 -0
- claude_mpm/services/agents/deployment/pipeline/steps/__init__.py +19 -0
- claude_mpm/services/agents/deployment/pipeline/steps/agent_processing_step.py +240 -0
- claude_mpm/services/agents/deployment/pipeline/steps/base_step.py +110 -0
- claude_mpm/services/agents/deployment/pipeline/steps/configuration_step.py +80 -0
- claude_mpm/services/agents/deployment/pipeline/steps/target_directory_step.py +92 -0
- claude_mpm/services/agents/deployment/pipeline/steps/validation_step.py +101 -0
- claude_mpm/services/agents/deployment/processors/__init__.py +15 -0
- claude_mpm/services/agents/deployment/processors/agent_deployment_context.py +102 -0
- claude_mpm/services/agents/deployment/processors/agent_deployment_result.py +235 -0
- claude_mpm/services/agents/deployment/processors/agent_processor.py +269 -0
- claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +311 -0
- claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +862 -0
- claude_mpm/services/agents/deployment/results/__init__.py +13 -0
- claude_mpm/services/agents/deployment/results/deployment_metrics.py +200 -0
- claude_mpm/services/agents/deployment/results/deployment_result_builder.py +249 -0
- claude_mpm/services/agents/deployment/single_agent_deployer.py +315 -0
- claude_mpm/services/agents/deployment/strategies/__init__.py +25 -0
- claude_mpm/services/agents/deployment/strategies/base_strategy.py +113 -0
- claude_mpm/services/agents/deployment/strategies/project_strategy.py +148 -0
- claude_mpm/services/agents/deployment/strategies/strategy_selector.py +117 -0
- claude_mpm/services/agents/deployment/strategies/system_strategy.py +131 -0
- claude_mpm/services/agents/deployment/strategies/user_strategy.py +130 -0
- claude_mpm/services/agents/deployment/system_instructions_deployer.py +228 -0
- claude_mpm/services/agents/deployment/validation/__init__.py +21 -0
- claude_mpm/services/agents/deployment/validation/agent_validator.py +323 -0
- claude_mpm/services/agents/deployment/validation/deployment_validator.py +238 -0
- claude_mpm/services/agents/deployment/validation/template_validator.py +319 -0
- claude_mpm/services/agents/deployment/validation/validation_result.py +214 -0
- claude_mpm/services/agents/git_source_manager.py +682 -0
- claude_mpm/services/agents/loading/__init__.py +11 -0
- claude_mpm/services/{agent_profile_loader.py → agents/loading/agent_profile_loader.py} +306 -228
- claude_mpm/services/{base_agent_manager.py → agents/loading/base_agent_manager.py} +106 -91
- claude_mpm/services/agents/loading/framework_agent_loader.py +433 -0
- claude_mpm/services/agents/local_template_manager.py +784 -0
- claude_mpm/services/agents/management/__init__.py +9 -0
- claude_mpm/services/{agent_capabilities_generator.py → agents/management/agent_capabilities_generator.py} +92 -69
- claude_mpm/services/{agent_management_service.py → agents/management/agent_management_service.py} +219 -168
- claude_mpm/services/agents/memory/__init__.py +22 -0
- claude_mpm/services/agents/memory/agent_memory_manager.py +784 -0
- claude_mpm/services/{agent_persistence_service.py → agents/memory/agent_persistence_service.py} +20 -18
- claude_mpm/services/agents/memory/content_manager.py +470 -0
- claude_mpm/services/agents/memory/memory_categorization_service.py +167 -0
- claude_mpm/services/agents/memory/memory_file_service.py +129 -0
- claude_mpm/services/agents/memory/memory_format_service.py +201 -0
- claude_mpm/services/agents/memory/memory_limits_service.py +101 -0
- claude_mpm/services/agents/memory/template_generator.py +83 -0
- claude_mpm/services/agents/observers.py +547 -0
- claude_mpm/services/agents/recommender.py +617 -0
- claude_mpm/services/agents/registry/__init__.py +30 -0
- claude_mpm/services/agents/registry/deployed_agent_discovery.py +273 -0
- claude_mpm/services/{agent_modification_tracker.py → agents/registry/modification_tracker.py} +370 -295
- 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 +1202 -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/async_session_logger.py +665 -0
- claude_mpm/services/claude_session_logger.py +321 -0
- claude_mpm/services/cli/__init__.py +18 -0
- claude_mpm/services/cli/agent_cleanup_service.py +408 -0
- claude_mpm/services/cli/agent_dependency_service.py +395 -0
- claude_mpm/services/cli/agent_listing_service.py +463 -0
- claude_mpm/services/cli/agent_output_formatter.py +605 -0
- claude_mpm/services/cli/agent_validation_service.py +590 -0
- claude_mpm/services/cli/memory_crud_service.py +622 -0
- claude_mpm/services/cli/memory_output_formatter.py +604 -0
- claude_mpm/services/cli/resume_service.py +617 -0
- claude_mpm/services/cli/session_manager.py +604 -0
- claude_mpm/services/cli/session_pause_manager.py +504 -0
- claude_mpm/services/cli/session_resume_helper.py +372 -0
- claude_mpm/services/cli/startup_checker.py +362 -0
- claude_mpm/services/cli/unified_dashboard_manager.py +439 -0
- claude_mpm/services/command_deployment_service.py +446 -0
- claude_mpm/services/command_handler_service.py +221 -0
- claude_mpm/services/communication/__init__.py +22 -0
- claude_mpm/services/core/__init__.py +108 -0
- claude_mpm/services/core/base.py +269 -0
- claude_mpm/services/core/cache_manager.py +309 -0
- claude_mpm/services/core/interfaces/__init__.py +273 -0
- claude_mpm/services/core/interfaces/agent.py +514 -0
- claude_mpm/services/core/interfaces/communication.py +316 -0
- claude_mpm/services/core/interfaces/health.py +169 -0
- claude_mpm/services/core/interfaces/infrastructure.py +357 -0
- claude_mpm/services/core/interfaces/model.py +281 -0
- claude_mpm/services/core/interfaces/process.py +372 -0
- claude_mpm/services/core/interfaces/project.py +121 -0
- claude_mpm/services/core/interfaces/restart.py +307 -0
- claude_mpm/services/core/interfaces/service.py +405 -0
- claude_mpm/services/core/interfaces/stability.py +260 -0
- claude_mpm/services/core/interfaces.py +81 -0
- claude_mpm/services/core/memory_manager.py +682 -0
- claude_mpm/services/core/models/__init__.py +70 -0
- claude_mpm/services/core/models/agent_config.py +384 -0
- claude_mpm/services/core/models/health.py +162 -0
- claude_mpm/services/core/models/process.py +239 -0
- claude_mpm/services/core/models/restart.py +302 -0
- claude_mpm/services/core/models/stability.py +264 -0
- claude_mpm/services/core/models/toolchain.py +306 -0
- claude_mpm/services/core/path_resolver.py +517 -0
- claude_mpm/services/core/service_container.py +520 -0
- claude_mpm/services/core/service_interfaces.py +436 -0
- claude_mpm/services/diagnostics/__init__.py +18 -0
- claude_mpm/services/diagnostics/checks/__init__.py +38 -0
- claude_mpm/services/diagnostics/checks/agent_check.py +370 -0
- claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
- claude_mpm/services/diagnostics/checks/base_check.py +60 -0
- claude_mpm/services/diagnostics/checks/claude_code_check.py +270 -0
- claude_mpm/services/diagnostics/checks/common_issues_check.py +363 -0
- claude_mpm/services/diagnostics/checks/configuration_check.py +306 -0
- claude_mpm/services/diagnostics/checks/filesystem_check.py +233 -0
- claude_mpm/services/diagnostics/checks/installation_check.py +520 -0
- claude_mpm/services/diagnostics/checks/instructions_check.py +415 -0
- claude_mpm/services/diagnostics/checks/mcp_check.py +330 -0
- claude_mpm/services/diagnostics/checks/mcp_services_check.py +1058 -0
- claude_mpm/services/diagnostics/checks/monitor_check.py +281 -0
- claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
- claude_mpm/services/diagnostics/checks/startup_log_check.py +319 -0
- claude_mpm/services/diagnostics/diagnostic_runner.py +286 -0
- claude_mpm/services/diagnostics/doctor_reporter.py +578 -0
- claude_mpm/services/diagnostics/models.py +138 -0
- claude_mpm/services/event_aggregator.py +582 -0
- claude_mpm/services/event_bus/__init__.py +18 -0
- claude_mpm/services/event_bus/config.py +186 -0
- claude_mpm/services/event_bus/direct_relay.py +312 -0
- claude_mpm/services/event_bus/event_bus.py +396 -0
- claude_mpm/services/event_bus/relay.py +326 -0
- claude_mpm/services/events/__init__.py +44 -0
- claude_mpm/services/events/consumers/__init__.py +18 -0
- claude_mpm/services/events/consumers/dead_letter.py +306 -0
- claude_mpm/services/events/consumers/logging.py +184 -0
- claude_mpm/services/events/consumers/metrics.py +241 -0
- claude_mpm/services/events/consumers/socketio.py +377 -0
- claude_mpm/services/events/core.py +480 -0
- claude_mpm/services/events/interfaces.py +214 -0
- claude_mpm/services/events/producers/__init__.py +14 -0
- claude_mpm/services/events/producers/hook.py +269 -0
- claude_mpm/services/events/producers/system.py +329 -0
- claude_mpm/services/exceptions.py +433 -353
- claude_mpm/services/framework_claude_md_generator/__init__.py +81 -80
- claude_mpm/services/framework_claude_md_generator/content_assembler.py +74 -67
- claude_mpm/services/framework_claude_md_generator/content_validator.py +66 -62
- claude_mpm/services/framework_claude_md_generator/deployment_manager.py +82 -60
- claude_mpm/services/framework_claude_md_generator/section_generators/__init__.py +36 -37
- claude_mpm/services/framework_claude_md_generator/section_generators/agents.py +41 -40
- claude_mpm/services/framework_claude_md_generator/section_generators/claude_pm_init.py +15 -15
- claude_mpm/services/framework_claude_md_generator/section_generators/core_responsibilities.py +5 -4
- claude_mpm/services/framework_claude_md_generator/section_generators/delegation_constraints.py +4 -3
- claude_mpm/services/framework_claude_md_generator/section_generators/environment_config.py +4 -3
- claude_mpm/services/framework_claude_md_generator/section_generators/footer.py +6 -5
- claude_mpm/services/framework_claude_md_generator/section_generators/header.py +8 -7
- claude_mpm/services/framework_claude_md_generator/section_generators/orchestration_principles.py +5 -4
- claude_mpm/services/framework_claude_md_generator/section_generators/role_designation.py +6 -5
- claude_mpm/services/framework_claude_md_generator/section_generators/subprocess_validation.py +9 -8
- claude_mpm/services/framework_claude_md_generator/section_generators/todo_task_tools.py +26 -30
- claude_mpm/services/framework_claude_md_generator/section_generators/troubleshooting.py +6 -5
- claude_mpm/services/framework_claude_md_generator/section_manager.py +28 -27
- claude_mpm/services/framework_claude_md_generator/version_manager.py +31 -30
- 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/hook_installer_service.py +506 -0
- claude_mpm/services/hook_service.py +159 -111
- claude_mpm/services/infrastructure/__init__.py +52 -0
- claude_mpm/services/infrastructure/context_preservation.py +569 -0
- claude_mpm/services/infrastructure/daemon_manager.py +279 -0
- claude_mpm/services/infrastructure/logging.py +209 -0
- claude_mpm/services/infrastructure/monitoring/__init__.py +39 -0
- claude_mpm/services/infrastructure/monitoring/aggregator.py +432 -0
- claude_mpm/services/infrastructure/monitoring/base.py +122 -0
- claude_mpm/services/infrastructure/monitoring/legacy.py +203 -0
- claude_mpm/services/infrastructure/monitoring/network.py +219 -0
- claude_mpm/services/infrastructure/monitoring/process.py +343 -0
- claude_mpm/services/infrastructure/monitoring/resources.py +244 -0
- claude_mpm/services/infrastructure/monitoring/service.py +368 -0
- claude_mpm/services/infrastructure/monitoring.py +71 -0
- claude_mpm/services/infrastructure/resume_log_generator.py +439 -0
- 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 +155 -0
- claude_mpm/services/local_ops/crash_detector.py +257 -0
- claude_mpm/services/local_ops/health_checks/__init__.py +26 -0
- claude_mpm/services/local_ops/health_checks/http_check.py +224 -0
- claude_mpm/services/local_ops/health_checks/process_check.py +236 -0
- claude_mpm/services/local_ops/health_checks/resource_check.py +255 -0
- claude_mpm/services/local_ops/health_manager.py +427 -0
- claude_mpm/services/local_ops/log_monitor.py +396 -0
- claude_mpm/services/local_ops/memory_leak_detector.py +294 -0
- claude_mpm/services/local_ops/process_manager.py +595 -0
- claude_mpm/services/local_ops/resource_monitor.py +331 -0
- claude_mpm/services/local_ops/restart_manager.py +401 -0
- claude_mpm/services/local_ops/restart_policy.py +387 -0
- claude_mpm/services/local_ops/state_manager.py +372 -0
- claude_mpm/services/local_ops/unified_manager.py +600 -0
- claude_mpm/services/mcp_config_manager.py +1542 -0
- claude_mpm/services/mcp_service_verifier.py +732 -0
- claude_mpm/services/memory/__init__.py +19 -0
- claude_mpm/services/{memory_builder.py → memory/builder.py} +465 -373
- claude_mpm/services/memory/cache/__init__.py +14 -0
- claude_mpm/services/{shared_prompt_cache.py → memory/cache/shared_prompt_cache.py} +237 -200
- claude_mpm/services/memory/cache/simple_cache.py +331 -0
- claude_mpm/services/memory/failure_tracker.py +578 -0
- claude_mpm/services/memory/indexed_memory.py +648 -0
- claude_mpm/services/{memory_optimizer.py → memory/optimizer.py} +272 -243
- claude_mpm/services/memory/router.py +951 -0
- claude_mpm/services/memory_hook_service.py +470 -0
- claude_mpm/services/model/__init__.py +147 -0
- claude_mpm/services/model/base_provider.py +365 -0
- claude_mpm/services/model/claude_provider.py +412 -0
- claude_mpm/services/model/model_router.py +452 -0
- claude_mpm/services/model/ollama_provider.py +415 -0
- claude_mpm/services/monitor/__init__.py +20 -0
- claude_mpm/services/monitor/daemon.py +698 -0
- claude_mpm/services/monitor/daemon_manager.py +1076 -0
- claude_mpm/services/monitor/event_emitter.py +350 -0
- claude_mpm/services/monitor/handlers/__init__.py +21 -0
- claude_mpm/services/monitor/handlers/code_analysis.py +332 -0
- claude_mpm/services/monitor/handlers/dashboard.py +299 -0
- claude_mpm/services/monitor/handlers/file.py +264 -0
- claude_mpm/services/monitor/handlers/hooks.py +512 -0
- claude_mpm/services/monitor/management/__init__.py +18 -0
- claude_mpm/services/monitor/management/health.py +124 -0
- claude_mpm/services/monitor/management/lifecycle.py +730 -0
- claude_mpm/services/monitor/server.py +1493 -0
- claude_mpm/services/monitor_build_service.py +349 -0
- claude_mpm/services/native_agent_converter.py +356 -0
- claude_mpm/services/orphan_detection.py +786 -0
- claude_mpm/services/pm_skills_deployer.py +707 -0
- claude_mpm/services/port_manager.py +597 -0
- claude_mpm/services/pr/__init__.py +14 -0
- claude_mpm/services/pr/pr_template_service.py +329 -0
- claude_mpm/services/profile_manager.py +337 -0
- claude_mpm/services/project/__init__.py +44 -0
- claude_mpm/services/{project_analyzer.py → project/analyzer.py} +541 -291
- claude_mpm/services/project/analyzer_v2.py +566 -0
- claude_mpm/services/project/architecture_analyzer.py +461 -0
- claude_mpm/services/project/archive_manager.py +1045 -0
- claude_mpm/services/project/dependency_analyzer.py +462 -0
- claude_mpm/services/project/detection_strategies.py +719 -0
- claude_mpm/services/project/documentation_manager.py +554 -0
- claude_mpm/services/project/enhanced_analyzer.py +572 -0
- claude_mpm/services/project/language_analyzer.py +265 -0
- claude_mpm/services/project/metrics_collector.py +407 -0
- claude_mpm/services/project/project_organizer.py +1009 -0
- claude_mpm/services/project/registry.py +636 -0
- claude_mpm/services/project/toolchain_analyzer.py +583 -0
- claude_mpm/services/project_port_allocator.py +596 -0
- claude_mpm/services/recovery_manager.py +293 -240
- claude_mpm/services/response_tracker.py +267 -0
- claude_mpm/services/runner_configuration_service.py +605 -0
- claude_mpm/services/self_upgrade_service.py +608 -0
- claude_mpm/services/session_management_service.py +314 -0
- claude_mpm/services/session_manager.py +380 -0
- claude_mpm/services/shared/__init__.py +21 -0
- claude_mpm/services/shared/async_service_base.py +216 -0
- claude_mpm/services/shared/config_service_base.py +301 -0
- claude_mpm/services/shared/lifecycle_service_base.py +308 -0
- claude_mpm/services/shared/manager_base.py +315 -0
- claude_mpm/services/shared/service_factory.py +309 -0
- claude_mpm/services/skills/__init__.py +21 -0
- claude_mpm/services/skills/git_skill_source_manager.py +1324 -0
- claude_mpm/services/skills/selective_skill_deployer.py +744 -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 +1168 -0
- claude_mpm/services/socketio/__init__.py +25 -0
- claude_mpm/services/socketio/client_proxy.py +229 -0
- claude_mpm/services/socketio/dashboard_server.py +362 -0
- claude_mpm/services/socketio/event_normalizer.py +798 -0
- claude_mpm/services/socketio/handlers/__init__.py +30 -0
- claude_mpm/services/socketio/handlers/base.py +136 -0
- claude_mpm/services/socketio/handlers/code_analysis.py +682 -0
- claude_mpm/services/socketio/handlers/connection.py +643 -0
- claude_mpm/services/socketio/handlers/connection_handler.py +333 -0
- claude_mpm/services/socketio/handlers/file.py +263 -0
- claude_mpm/services/socketio/handlers/git.py +962 -0
- claude_mpm/services/socketio/handlers/hook.py +211 -0
- claude_mpm/services/socketio/handlers/memory.py +26 -0
- claude_mpm/services/socketio/handlers/project.py +24 -0
- claude_mpm/services/socketio/handlers/registry.py +214 -0
- claude_mpm/services/socketio/migration_utils.py +343 -0
- claude_mpm/services/socketio/monitor_client.py +364 -0
- claude_mpm/services/socketio/server/__init__.py +18 -0
- claude_mpm/services/socketio/server/broadcaster.py +569 -0
- claude_mpm/services/socketio/server/connection_manager.py +579 -0
- claude_mpm/services/socketio/server/core.py +1079 -0
- claude_mpm/services/socketio/server/eventbus_integration.py +245 -0
- claude_mpm/services/socketio/server/main.py +501 -0
- claude_mpm/services/socketio_client_manager.py +173 -143
- claude_mpm/services/socketio_server.py +38 -1657
- claude_mpm/services/subprocess_launcher_service.py +322 -0
- claude_mpm/services/system_instructions_service.py +270 -0
- claude_mpm/services/ticket_manager.py +25 -209
- claude_mpm/services/ticket_services/__init__.py +26 -0
- claude_mpm/services/ticket_services/crud_service.py +328 -0
- claude_mpm/services/ticket_services/formatter_service.py +290 -0
- claude_mpm/services/ticket_services/search_service.py +324 -0
- claude_mpm/services/ticket_services/validation_service.py +303 -0
- claude_mpm/services/ticket_services/workflow_service.py +244 -0
- claude_mpm/services/unified/__init__.py +65 -0
- claude_mpm/services/unified/analyzer_strategies/__init__.py +44 -0
- claude_mpm/services/unified/analyzer_strategies/code_analyzer.py +518 -0
- claude_mpm/services/unified/analyzer_strategies/dependency_analyzer.py +680 -0
- claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +900 -0
- claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +745 -0
- claude_mpm/services/unified/analyzer_strategies/structure_analyzer.py +733 -0
- claude_mpm/services/unified/config_strategies/__init__.py +175 -0
- claude_mpm/services/unified/config_strategies/config_schema.py +731 -0
- claude_mpm/services/unified/config_strategies/context_strategy.py +747 -0
- claude_mpm/services/unified/config_strategies/error_handling_strategy.py +1005 -0
- claude_mpm/services/unified/config_strategies/file_loader_strategy.py +881 -0
- claude_mpm/services/unified/config_strategies/unified_config_service.py +823 -0
- claude_mpm/services/unified/config_strategies/validation_strategy.py +1148 -0
- claude_mpm/services/unified/deployment_strategies/__init__.py +97 -0
- claude_mpm/services/unified/deployment_strategies/base.py +553 -0
- claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +573 -0
- claude_mpm/services/unified/deployment_strategies/local.py +607 -0
- claude_mpm/services/unified/deployment_strategies/utils.py +667 -0
- claude_mpm/services/unified/deployment_strategies/vercel.py +471 -0
- claude_mpm/services/unified/interfaces.py +475 -0
- claude_mpm/services/unified/migration.py +509 -0
- claude_mpm/services/unified/strategies.py +534 -0
- claude_mpm/services/unified/unified_analyzer.py +542 -0
- claude_mpm/services/unified/unified_config.py +691 -0
- claude_mpm/services/unified/unified_deployment.py +466 -0
- claude_mpm/services/utility_service.py +280 -0
- claude_mpm/services/version_control/__init__.py +34 -37
- claude_mpm/services/version_control/branch_strategy.py +26 -17
- claude_mpm/services/version_control/conflict_resolution.py +52 -36
- claude_mpm/services/version_control/git_operations.py +183 -49
- claude_mpm/services/version_control/semantic_versioning.py +172 -61
- claude_mpm/services/version_control/version_parser.py +546 -0
- claude_mpm/services/version_service.py +379 -0
- claude_mpm/services/visualization/__init__.py +15 -0
- claude_mpm/services/visualization/mermaid_generator.py +937 -0
- claude_mpm/skills/__init__.py +42 -0
- claude_mpm/skills/agent_skills_injector.py +324 -0
- claude_mpm/skills/bundled/LICENSE_ATTRIBUTIONS.md +79 -0
- claude_mpm/skills/bundled/__init__.py +6 -0
- claude_mpm/skills/bundled/api-documentation.md +393 -0
- claude_mpm/skills/bundled/async-testing.md +571 -0
- claude_mpm/skills/bundled/code-review.md +143 -0
- claude_mpm/skills/bundled/database-migration.md +199 -0
- claude_mpm/skills/bundled/docker-containerization.md +194 -0
- claude_mpm/skills/bundled/express-local-dev.md +1429 -0
- claude_mpm/skills/bundled/fastapi-local-dev.md +1199 -0
- claude_mpm/skills/bundled/git-workflow.md +414 -0
- claude_mpm/skills/bundled/imagemagick.md +204 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/scripts/validate_env.py +576 -0
- claude_mpm/skills/bundled/json-data-handling.md +223 -0
- claude_mpm/skills/bundled/main/mcp-builder/scripts/connections.py +157 -0
- claude_mpm/skills/bundled/main/mcp-builder/scripts/evaluation.py +425 -0
- claude_mpm/skills/bundled/main/skill-creator/scripts/init_skill.py +303 -0
- claude_mpm/skills/bundled/main/skill-creator/scripts/package_skill.py +113 -0
- claude_mpm/skills/bundled/main/skill-creator/scripts/quick_validate.py +72 -0
- claude_mpm/skills/bundled/nextjs-local-dev.md +807 -0
- claude_mpm/skills/bundled/pdf.md +141 -0
- claude_mpm/skills/bundled/performance-profiling.md +573 -0
- claude_mpm/skills/bundled/refactoring-patterns.md +180 -0
- claude_mpm/skills/bundled/security-scanning.md +439 -0
- claude_mpm/skills/bundled/systematic-debugging.md +473 -0
- claude_mpm/skills/bundled/test-driven-development.md +378 -0
- claude_mpm/skills/bundled/testing/webapp-testing/examples/console_logging.py +35 -0
- claude_mpm/skills/bundled/testing/webapp-testing/examples/element_discovery.py +44 -0
- claude_mpm/skills/bundled/testing/webapp-testing/examples/static_html_automation.py +34 -0
- claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +129 -0
- claude_mpm/skills/bundled/vite-local-dev.md +1061 -0
- claude_mpm/skills/bundled/web-performance-optimization.md +2305 -0
- claude_mpm/skills/bundled/xlsx.md +157 -0
- claude_mpm/skills/registry.py +286 -0
- claude_mpm/skills/skill_manager.py +405 -0
- claude_mpm/skills/skills_registry.py +347 -0
- claude_mpm/skills/skills_service.py +739 -0
- claude_mpm/storage/__init__.py +9 -0
- claude_mpm/storage/state_storage.py +546 -0
- claude_mpm/templates/.pre-commit-config.yaml +112 -0
- 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/ticket_wrapper.py +2 -2
- claude_mpm/tools/__init__.py +10 -0
- claude_mpm/tools/__main__.py +208 -0
- claude_mpm/tools/code_tree_analyzer/__init__.py +45 -0
- claude_mpm/tools/code_tree_analyzer/analysis.py +299 -0
- claude_mpm/tools/code_tree_analyzer/cache.py +131 -0
- claude_mpm/tools/code_tree_analyzer/core.py +380 -0
- claude_mpm/tools/code_tree_analyzer/discovery.py +403 -0
- claude_mpm/tools/code_tree_analyzer/events.py +168 -0
- claude_mpm/tools/code_tree_analyzer/gitignore.py +308 -0
- claude_mpm/tools/code_tree_analyzer/models.py +39 -0
- claude_mpm/tools/code_tree_analyzer/multilang_analyzer.py +224 -0
- claude_mpm/tools/code_tree_analyzer/python_analyzer.py +284 -0
- claude_mpm/tools/code_tree_builder.py +631 -0
- claude_mpm/tools/code_tree_events.py +420 -0
- claude_mpm/tools/socketio_debug.py +671 -0
- claude_mpm/utils/__init__.py +8 -8
- claude_mpm/utils/agent_dependency_loader.py +1090 -0
- claude_mpm/utils/agent_filters.py +261 -0
- claude_mpm/utils/common.py +544 -0
- claude_mpm/utils/config_manager.py +168 -126
- claude_mpm/utils/console.py +11 -0
- claude_mpm/utils/database_connector.py +298 -0
- claude_mpm/utils/dependency_cache.py +373 -0
- claude_mpm/utils/dependency_manager.py +60 -59
- claude_mpm/utils/dependency_strategies.py +381 -0
- claude_mpm/utils/display_helper.py +260 -0
- claude_mpm/utils/environment_context.py +313 -0
- claude_mpm/utils/error_handler.py +78 -66
- claude_mpm/utils/file_utils.py +305 -0
- claude_mpm/utils/framework_detection.py +12 -11
- claude_mpm/utils/git_analyzer.py +407 -0
- claude_mpm/utils/gitignore.py +244 -0
- claude_mpm/utils/import_migration_example.py +12 -60
- claude_mpm/utils/imports.py +48 -45
- claude_mpm/utils/log_cleanup.py +627 -0
- claude_mpm/utils/migration.py +372 -0
- claude_mpm/utils/path_operations.py +110 -104
- claude_mpm/utils/progress.py +387 -0
- claude_mpm/utils/robust_installer.py +823 -0
- claude_mpm/utils/session_logging.py +121 -0
- claude_mpm/utils/structured_questions.py +619 -0
- claude_mpm/utils/subprocess_utils.py +343 -0
- claude_mpm/validation/__init__.py +1 -1
- claude_mpm/validation/agent_validator.py +214 -108
- claude_mpm/validation/frontmatter_validator.py +252 -0
- claude_mpm-5.4.55.dist-info/METADATA +999 -0
- claude_mpm-5.4.55.dist-info/RECORD +868 -0
- {claude_mpm-3.4.10.dist-info → claude_mpm-5.4.55.dist-info}/entry_points.txt +1 -3
- claude_mpm-5.4.55.dist-info/licenses/LICENSE +94 -0
- claude_mpm-5.4.55.dist-info/licenses/LICENSE-FAQ.md +153 -0
- claude_mpm/agents/BASE_AGENT_TEMPLATE.md +0 -88
- claude_mpm/agents/INSTRUCTIONS.md +0 -352
- claude_mpm/agents/backups/INSTRUCTIONS.md +0 -352
- claude_mpm/agents/base_agent_loader.py +0 -529
- claude_mpm/agents/schema/agent_schema.json +0 -314
- claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -36
- claude_mpm/agents/templates/backup/data_engineer_agent_20250726_234551.json +0 -46
- claude_mpm/agents/templates/backup/documentation_agent_20250726_234551.json +0 -45
- claude_mpm/agents/templates/backup/engineer_agent_20250726_234551.json +0 -49
- claude_mpm/agents/templates/backup/ops_agent_20250726_234551.json +0 -46
- claude_mpm/agents/templates/backup/qa_agent_20250726_234551.json +0 -45
- claude_mpm/agents/templates/backup/research_agent_20250726_234551.json +0 -49
- claude_mpm/agents/templates/backup/security_agent_20250726_234551.json +0 -46
- claude_mpm/agents/templates/backup/version_control_agent_20250726_234551.json +0 -46
- claude_mpm/agents/templates/data_engineer.json +0 -110
- claude_mpm/agents/templates/documentation.json +0 -109
- claude_mpm/agents/templates/engineer.json +0 -113
- claude_mpm/agents/templates/ops.json +0 -109
- claude_mpm/agents/templates/pm.json +0 -25
- claude_mpm/agents/templates/qa.json +0 -111
- claude_mpm/agents/templates/research.json +0 -65
- claude_mpm/agents/templates/security.json +0 -113
- claude_mpm/agents/templates/test_integration.json +0 -112
- claude_mpm/agents/templates/version_control.json +0 -107
- claude_mpm/cli/commands/ui.py +0 -57
- claude_mpm/core/simple_runner.py +0 -1046
- claude_mpm/dashboard/open_dashboard.py +0 -34
- claude_mpm/deployment_paths.py +0 -261
- claude_mpm/hooks/builtin/__init__.py +0 -1
- claude_mpm/hooks/builtin/logging_hook_example.py +0 -165
- claude_mpm/hooks/builtin/memory_hooks_example.py +0 -67
- claude_mpm/hooks/builtin/mpm_command_hook.py +0 -125
- claude_mpm/hooks/builtin/post_delegation_hook_example.py +0 -124
- claude_mpm/hooks/builtin/pre_delegation_hook_example.py +0 -125
- claude_mpm/hooks/builtin/submit_hook_example.py +0 -100
- claude_mpm/hooks/builtin/ticket_extraction_hook_example.py +0 -237
- claude_mpm/hooks/builtin/todo_agent_prefix_hook.py +0 -240
- claude_mpm/hooks/builtin/workflow_start_hook.py +0 -181
- claude_mpm/orchestration/__init__.py +0 -6
- claude_mpm/orchestration/archive/direct_orchestrator.py +0 -195
- claude_mpm/orchestration/archive/factory.py +0 -215
- claude_mpm/orchestration/archive/hook_enabled_orchestrator.py +0 -188
- claude_mpm/orchestration/archive/hook_integration_example.py +0 -178
- claude_mpm/orchestration/archive/interactive_subprocess_orchestrator.py +0 -826
- claude_mpm/orchestration/archive/orchestrator.py +0 -501
- claude_mpm/orchestration/archive/pexpect_orchestrator.py +0 -252
- claude_mpm/orchestration/archive/pty_orchestrator.py +0 -270
- claude_mpm/orchestration/archive/simple_orchestrator.py +0 -82
- claude_mpm/orchestration/archive/subprocess_orchestrator.py +0 -801
- claude_mpm/orchestration/archive/system_prompt_orchestrator.py +0 -278
- claude_mpm/orchestration/archive/wrapper_orchestrator.py +0 -187
- claude_mpm/schemas/workflow_validator.py +0 -411
- claude_mpm/services/agent_deployment.py +0 -1534
- claude_mpm/services/agent_lifecycle_manager.py +0 -1169
- claude_mpm/services/agent_memory_manager.py +0 -1415
- claude_mpm/services/agent_registry.py +0 -676
- claude_mpm/services/deployed_agent_discovery.py +0 -226
- claude_mpm/services/framework_agent_loader.py +0 -337
- claude_mpm/services/framework_claude_md_generator.py +0 -621
- claude_mpm/services/health_monitor.py +0 -892
- claude_mpm/services/memory_router.py +0 -538
- claude_mpm/services/parent_directory_manager/__init__.py +0 -577
- claude_mpm/services/parent_directory_manager/backup_manager.py +0 -258
- claude_mpm/services/parent_directory_manager/config_manager.py +0 -210
- claude_mpm/services/parent_directory_manager/deduplication_manager.py +0 -279
- claude_mpm/services/parent_directory_manager/framework_protector.py +0 -143
- claude_mpm/services/parent_directory_manager/operations.py +0 -186
- claude_mpm/services/parent_directory_manager/state_manager.py +0 -624
- claude_mpm/services/parent_directory_manager/template_deployer.py +0 -579
- claude_mpm/services/parent_directory_manager/validation_manager.py +0 -378
- claude_mpm/services/parent_directory_manager/version_control_helper.py +0 -339
- claude_mpm/services/parent_directory_manager/version_manager.py +0 -222
- claude_mpm/services/standalone_socketio_server.py +0 -1300
- claude_mpm/services/ticket_manager_di.py +0 -318
- claude_mpm/services/ticketing_service_original.py +0 -508
- claude_mpm/ui/__init__.py +0 -1
- claude_mpm/ui/rich_terminal_ui.py +0 -295
- claude_mpm/ui/terminal_ui.py +0 -328
- claude_mpm/utils/paths.py +0 -289
- claude_mpm-3.4.10.dist-info/METADATA +0 -183
- claude_mpm-3.4.10.dist-info/RECORD +0 -201
- claude_mpm-3.4.10.dist-info/licenses/LICENSE +0 -21
- {claude_mpm-3.4.10.dist-info → claude_mpm-5.4.55.dist-info}/WHEEL +0 -0
- {claude_mpm-3.4.10.dist-info → claude_mpm-5.4.55.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,1369 @@
|
|
|
1
|
+
"""Agent Template Builder Service
|
|
2
|
+
|
|
3
|
+
This service handles the building and generation of agent templates,
|
|
4
|
+
including YAML and Markdown generation, template merging, and metadata extraction.
|
|
5
|
+
|
|
6
|
+
Extracted from AgentDeploymentService as part of the refactoring to improve
|
|
7
|
+
maintainability and testability.
|
|
8
|
+
"""
|
|
9
|
+
|
|
10
|
+
import json
|
|
11
|
+
import re
|
|
12
|
+
from pathlib import Path
|
|
13
|
+
from typing import Any, Dict, List
|
|
14
|
+
|
|
15
|
+
import yaml
|
|
16
|
+
|
|
17
|
+
from claude_mpm.core.logging_config import get_logger
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
class AgentTemplateBuilder:
|
|
21
|
+
"""Service for building agent templates from JSON and base agent data.
|
|
22
|
+
|
|
23
|
+
This service handles:
|
|
24
|
+
- Building agent markdown files with YAML frontmatter
|
|
25
|
+
- Building agent YAML files
|
|
26
|
+
- Merging narrative and configuration fields
|
|
27
|
+
- Extracting agent metadata
|
|
28
|
+
- Formatting YAML lists
|
|
29
|
+
"""
|
|
30
|
+
|
|
31
|
+
def __init__(self):
|
|
32
|
+
"""Initialize the template builder."""
|
|
33
|
+
self.logger = get_logger(__name__)
|
|
34
|
+
|
|
35
|
+
def normalize_tools_input(self, tools):
|
|
36
|
+
"""Normalize various tool input formats to a consistent list.
|
|
37
|
+
|
|
38
|
+
Handles multiple input formats:
|
|
39
|
+
- None/empty: Returns default tools
|
|
40
|
+
- String: Splits by comma and strips whitespace
|
|
41
|
+
- List: Ensures all items are strings and strips whitespace
|
|
42
|
+
- Dict: Takes enabled tools (where value is True)
|
|
43
|
+
|
|
44
|
+
Args:
|
|
45
|
+
tools: Tools input in various formats (str, list, dict, or None)
|
|
46
|
+
|
|
47
|
+
Returns:
|
|
48
|
+
List of tool names, normalized and cleaned
|
|
49
|
+
"""
|
|
50
|
+
default_tools = ["Read", "Write", "Edit", "Grep", "Glob", "Bash"]
|
|
51
|
+
|
|
52
|
+
# Handle None or empty
|
|
53
|
+
if not tools:
|
|
54
|
+
self.logger.debug("No tools provided, using defaults")
|
|
55
|
+
return default_tools
|
|
56
|
+
|
|
57
|
+
# Convert to list format
|
|
58
|
+
if isinstance(tools, str):
|
|
59
|
+
# Split by comma, strip whitespace
|
|
60
|
+
tool_list = [t.strip() for t in tools.split(",") if t.strip()]
|
|
61
|
+
self.logger.debug(f"Converted string tools '{tools}' to list: {tool_list}")
|
|
62
|
+
elif isinstance(tools, list):
|
|
63
|
+
# Ensure all items are strings and strip whitespace
|
|
64
|
+
tool_list = [str(t).strip() for t in tools if t and str(t).strip()]
|
|
65
|
+
self.logger.debug(f"Normalized list tools: {tool_list}")
|
|
66
|
+
elif isinstance(tools, dict):
|
|
67
|
+
# Handle dict format - take enabled tools
|
|
68
|
+
tool_list = [k for k, v in tools.items() if v]
|
|
69
|
+
self.logger.info(f"Converting dict tools format: {tools} -> {tool_list}")
|
|
70
|
+
else:
|
|
71
|
+
self.logger.warning(f"Unknown tools format: {type(tools)}, using defaults")
|
|
72
|
+
return default_tools
|
|
73
|
+
|
|
74
|
+
# Return processed list or defaults if empty
|
|
75
|
+
if not tool_list:
|
|
76
|
+
self.logger.debug("Tools list empty after processing, using defaults")
|
|
77
|
+
return default_tools
|
|
78
|
+
|
|
79
|
+
return tool_list
|
|
80
|
+
|
|
81
|
+
def _discover_base_agent_templates(self, agent_file: Path) -> List[Path]:
|
|
82
|
+
"""Discover BASE-AGENT.md files in hierarchy from agent file to repository root.
|
|
83
|
+
|
|
84
|
+
This method implements hierarchical BASE template discovery by walking up the
|
|
85
|
+
directory tree from the agent file location and collecting all BASE-AGENT.md
|
|
86
|
+
files found along the way.
|
|
87
|
+
|
|
88
|
+
Composition Order (closest to farthest):
|
|
89
|
+
1. Local BASE-AGENT.md (same directory as agent)
|
|
90
|
+
2. Parent BASE-AGENT.md (parent directory)
|
|
91
|
+
3. Grandparent BASE-AGENT.md (grandparent directory)
|
|
92
|
+
... continuing to repository root
|
|
93
|
+
|
|
94
|
+
Args:
|
|
95
|
+
agent_file: Path to the agent template file
|
|
96
|
+
|
|
97
|
+
Returns:
|
|
98
|
+
List of BASE-AGENT.md paths ordered from closest to farthest
|
|
99
|
+
(same directory to root)
|
|
100
|
+
|
|
101
|
+
Example:
|
|
102
|
+
Given structure:
|
|
103
|
+
repo/
|
|
104
|
+
BASE-AGENT.md # Root (index 2)
|
|
105
|
+
engineering/
|
|
106
|
+
BASE-AGENT.md # Parent (index 1)
|
|
107
|
+
python/
|
|
108
|
+
BASE-AGENT.md # Local (index 0)
|
|
109
|
+
fastapi-engineer.md # Agent file
|
|
110
|
+
|
|
111
|
+
Returns: [
|
|
112
|
+
repo/engineering/python/BASE-AGENT.md,
|
|
113
|
+
repo/engineering/BASE-AGENT.md,
|
|
114
|
+
repo/BASE-AGENT.md
|
|
115
|
+
]
|
|
116
|
+
"""
|
|
117
|
+
base_templates = []
|
|
118
|
+
current_dir = agent_file.parent
|
|
119
|
+
|
|
120
|
+
# Walk up directory tree until we reach root or a reasonable limit
|
|
121
|
+
# Stop at repository root or after 10 levels (safety limit)
|
|
122
|
+
max_depth = 10
|
|
123
|
+
depth = 0
|
|
124
|
+
|
|
125
|
+
while current_dir and depth < max_depth:
|
|
126
|
+
# Check for BASE-AGENT.md in current directory
|
|
127
|
+
base_agent_file = current_dir / "BASE-AGENT.md"
|
|
128
|
+
if base_agent_file.exists() and base_agent_file.is_file():
|
|
129
|
+
base_templates.append(base_agent_file)
|
|
130
|
+
self.logger.debug(f"Found BASE-AGENT.md at: {base_agent_file}")
|
|
131
|
+
|
|
132
|
+
# Stop at git repository root if detected (check AFTER finding BASE-AGENT.md)
|
|
133
|
+
if (current_dir / ".git").exists():
|
|
134
|
+
self.logger.debug(f"Reached git repository root at: {current_dir}")
|
|
135
|
+
break
|
|
136
|
+
|
|
137
|
+
# Stop at common repository root indicators (check AFTER finding BASE-AGENT.md)
|
|
138
|
+
# Stop at cache root or .claude-mpm directory
|
|
139
|
+
if current_dir.name in [".claude-mpm", "cache"]:
|
|
140
|
+
self.logger.debug(
|
|
141
|
+
f"Reached repository root indicator at: {current_dir}"
|
|
142
|
+
)
|
|
143
|
+
break
|
|
144
|
+
|
|
145
|
+
# Move to parent directory
|
|
146
|
+
parent = current_dir.parent
|
|
147
|
+
if parent == current_dir: # Reached filesystem root
|
|
148
|
+
break
|
|
149
|
+
|
|
150
|
+
current_dir = parent
|
|
151
|
+
depth += 1
|
|
152
|
+
|
|
153
|
+
if base_templates:
|
|
154
|
+
self.logger.info(
|
|
155
|
+
f"Discovered {len(base_templates)} BASE-AGENT.md file(s) for {agent_file.name}"
|
|
156
|
+
)
|
|
157
|
+
|
|
158
|
+
return base_templates
|
|
159
|
+
|
|
160
|
+
def _parse_markdown_template(self, template_path: Path) -> dict:
|
|
161
|
+
"""Parse Markdown template with YAML frontmatter.
|
|
162
|
+
|
|
163
|
+
Extracts metadata from YAML frontmatter and content from Markdown body.
|
|
164
|
+
Supports the new agent template format with YAML frontmatter between --- delimiters.
|
|
165
|
+
|
|
166
|
+
Args:
|
|
167
|
+
template_path: Path to the Markdown template file
|
|
168
|
+
|
|
169
|
+
Returns:
|
|
170
|
+
Dictionary containing metadata and instructions
|
|
171
|
+
|
|
172
|
+
Raises:
|
|
173
|
+
ValueError: If frontmatter is missing or malformed
|
|
174
|
+
yaml.YAMLError: If YAML parsing fails
|
|
175
|
+
"""
|
|
176
|
+
content = template_path.read_text(encoding="utf-8")
|
|
177
|
+
|
|
178
|
+
# Split frontmatter and body
|
|
179
|
+
# Format: ---\n<yaml>\n---\n<markdown>
|
|
180
|
+
if not content.startswith("---"):
|
|
181
|
+
raise ValueError(
|
|
182
|
+
f"Markdown template missing YAML frontmatter: {template_path}"
|
|
183
|
+
)
|
|
184
|
+
|
|
185
|
+
# Split by --- delimiters
|
|
186
|
+
parts = content.split("---", 2)
|
|
187
|
+
if len(parts) < 3:
|
|
188
|
+
raise ValueError(f"Malformed YAML frontmatter in template: {template_path}")
|
|
189
|
+
|
|
190
|
+
# parts[0] is empty (before first ---)
|
|
191
|
+
# parts[1] is YAML frontmatter
|
|
192
|
+
# parts[2] is Markdown content
|
|
193
|
+
yaml_content = parts[1].strip()
|
|
194
|
+
markdown_content = parts[2].strip()
|
|
195
|
+
|
|
196
|
+
# Parse YAML frontmatter
|
|
197
|
+
try:
|
|
198
|
+
metadata = yaml.safe_load(yaml_content)
|
|
199
|
+
if not isinstance(metadata, dict):
|
|
200
|
+
raise ValueError(
|
|
201
|
+
f"YAML frontmatter must be a dictionary: {template_path}"
|
|
202
|
+
)
|
|
203
|
+
except yaml.YAMLError as e:
|
|
204
|
+
self.logger.error(
|
|
205
|
+
f"Failed to parse YAML frontmatter in {template_path}: {e}"
|
|
206
|
+
)
|
|
207
|
+
raise
|
|
208
|
+
|
|
209
|
+
# Validate required fields
|
|
210
|
+
required_fields = ["name", "description", "version"]
|
|
211
|
+
missing_fields = [field for field in required_fields if field not in metadata]
|
|
212
|
+
if missing_fields:
|
|
213
|
+
raise ValueError(
|
|
214
|
+
f"Missing required fields in template {template_path}: {missing_fields}"
|
|
215
|
+
)
|
|
216
|
+
|
|
217
|
+
# Add the markdown content as instructions field
|
|
218
|
+
metadata["instructions"] = markdown_content
|
|
219
|
+
|
|
220
|
+
# Normalize metadata structure to match JSON template format
|
|
221
|
+
# JSON templates have these fields at top level, Markdown may have them nested
|
|
222
|
+
self._normalize_metadata_structure(metadata)
|
|
223
|
+
|
|
224
|
+
return metadata
|
|
225
|
+
|
|
226
|
+
def _normalize_metadata_structure(self, metadata: dict) -> None:
|
|
227
|
+
"""Normalize metadata structure to match expected JSON template format.
|
|
228
|
+
|
|
229
|
+
This ensures both Markdown and JSON templates produce the same metadata structure
|
|
230
|
+
for downstream processing.
|
|
231
|
+
|
|
232
|
+
Args:
|
|
233
|
+
metadata: Metadata dictionary to normalize (modified in-place)
|
|
234
|
+
"""
|
|
235
|
+
# Map Markdown frontmatter fields to JSON template structure
|
|
236
|
+
# Handle tags: YAML list vs JSON comma-separated string
|
|
237
|
+
if "tags" in metadata and isinstance(metadata["tags"], list):
|
|
238
|
+
# Keep as list for now, normalize_tools_input will handle both formats
|
|
239
|
+
pass
|
|
240
|
+
|
|
241
|
+
# Map agent_id to name if name is missing
|
|
242
|
+
if "agent_id" in metadata and "name" not in metadata:
|
|
243
|
+
metadata["name"] = metadata["agent_id"]
|
|
244
|
+
|
|
245
|
+
# Ensure capabilities dict exists
|
|
246
|
+
if "capabilities" not in metadata:
|
|
247
|
+
metadata["capabilities"] = {}
|
|
248
|
+
|
|
249
|
+
# Merge top-level capability fields into capabilities dict
|
|
250
|
+
capability_fields = ["memory_limit", "cpu_limit", "network_access"]
|
|
251
|
+
for field in capability_fields:
|
|
252
|
+
if field in metadata:
|
|
253
|
+
metadata["capabilities"][field] = metadata.pop(field)
|
|
254
|
+
|
|
255
|
+
# Add model to capabilities if present at top level
|
|
256
|
+
if "model" in metadata and "model" not in metadata["capabilities"]:
|
|
257
|
+
metadata["capabilities"]["model"] = metadata["model"]
|
|
258
|
+
|
|
259
|
+
def _load_base_agent_instructions(self, agent_type: str) -> str:
|
|
260
|
+
"""Load BASE instructions for a specific agent type.
|
|
261
|
+
|
|
262
|
+
DEPRECATED: This method loads BASE_{TYPE}.md files (old pattern).
|
|
263
|
+
New pattern uses hierarchical BASE-AGENT.md files discovered via
|
|
264
|
+
_discover_base_agent_templates() and composed in build_agent_markdown().
|
|
265
|
+
|
|
266
|
+
Args:
|
|
267
|
+
agent_type: The type of agent (engineer, qa, ops, research, documentation)
|
|
268
|
+
|
|
269
|
+
Returns:
|
|
270
|
+
The BASE instructions content or empty string if not found
|
|
271
|
+
"""
|
|
272
|
+
if not agent_type:
|
|
273
|
+
return ""
|
|
274
|
+
|
|
275
|
+
try:
|
|
276
|
+
# Construct BASE file name
|
|
277
|
+
base_file = f"BASE_{agent_type.upper()}.md"
|
|
278
|
+
|
|
279
|
+
# Try to find BASE file in agents directory
|
|
280
|
+
# First try current working directory structure
|
|
281
|
+
agents_dir = Path(__file__).parent.parent.parent.parent / "agents"
|
|
282
|
+
base_path = agents_dir / base_file
|
|
283
|
+
|
|
284
|
+
if not base_path.exists():
|
|
285
|
+
# Try packaged resources if available
|
|
286
|
+
try:
|
|
287
|
+
from importlib.resources import files
|
|
288
|
+
|
|
289
|
+
agents_package = files("claude_mpm.agents")
|
|
290
|
+
base_resource = agents_package / base_file
|
|
291
|
+
if base_resource.is_file():
|
|
292
|
+
content = base_resource.read_text(encoding="utf-8")
|
|
293
|
+
self.logger.debug(
|
|
294
|
+
f"Loaded BASE instructions from package: {base_file}"
|
|
295
|
+
)
|
|
296
|
+
return content
|
|
297
|
+
except (ImportError, Exception) as e:
|
|
298
|
+
self.logger.debug(
|
|
299
|
+
f"Could not load BASE instructions from package: {e}"
|
|
300
|
+
)
|
|
301
|
+
|
|
302
|
+
# Final fallback - try multiple possible locations
|
|
303
|
+
possible_paths = [
|
|
304
|
+
Path.cwd() / "src" / "claude_mpm" / "agents" / base_file,
|
|
305
|
+
Path(__file__).parent.parent.parent.parent / "agents" / base_file,
|
|
306
|
+
Path.home() / ".claude-mpm" / "agents" / base_file,
|
|
307
|
+
]
|
|
308
|
+
|
|
309
|
+
for path in possible_paths:
|
|
310
|
+
if path.exists():
|
|
311
|
+
base_path = path
|
|
312
|
+
break
|
|
313
|
+
else:
|
|
314
|
+
self.logger.debug(
|
|
315
|
+
f"No BASE instructions found for type: {agent_type}"
|
|
316
|
+
)
|
|
317
|
+
return ""
|
|
318
|
+
|
|
319
|
+
if base_path.exists():
|
|
320
|
+
self.logger.debug(f"Loading BASE instructions from {base_path}")
|
|
321
|
+
return base_path.read_text(encoding="utf-8")
|
|
322
|
+
self.logger.debug(f"No BASE instructions found for type: {agent_type}")
|
|
323
|
+
return ""
|
|
324
|
+
|
|
325
|
+
except Exception as e:
|
|
326
|
+
self.logger.warning(
|
|
327
|
+
f"Error loading BASE instructions for {agent_type}: {e}"
|
|
328
|
+
)
|
|
329
|
+
return ""
|
|
330
|
+
|
|
331
|
+
def build_agent_markdown(
|
|
332
|
+
self,
|
|
333
|
+
agent_name: str,
|
|
334
|
+
template_path: Path,
|
|
335
|
+
base_agent_data: dict,
|
|
336
|
+
source_info: str = "unknown",
|
|
337
|
+
) -> str:
|
|
338
|
+
"""
|
|
339
|
+
Build a complete agent markdown file with YAML frontmatter.
|
|
340
|
+
|
|
341
|
+
Args:
|
|
342
|
+
agent_name: Name of the agent
|
|
343
|
+
template_path: Path to the agent template (JSON or Markdown file)
|
|
344
|
+
base_agent_data: Base agent configuration data
|
|
345
|
+
source_info: Source of the agent (system/project/user)
|
|
346
|
+
|
|
347
|
+
Returns:
|
|
348
|
+
Complete markdown content with YAML frontmatter
|
|
349
|
+
|
|
350
|
+
Raises:
|
|
351
|
+
FileNotFoundError: If template file doesn't exist
|
|
352
|
+
json.JSONDecodeError: If template JSON is invalid
|
|
353
|
+
yaml.YAMLError: If template YAML is invalid
|
|
354
|
+
ValueError: If template format is unsupported or malformed
|
|
355
|
+
"""
|
|
356
|
+
if not template_path.exists():
|
|
357
|
+
raise FileNotFoundError(f"Template file not found: {template_path}")
|
|
358
|
+
|
|
359
|
+
# Format detection: route to appropriate parser
|
|
360
|
+
try:
|
|
361
|
+
if template_path.suffix == ".md":
|
|
362
|
+
# Parse Markdown template with YAML frontmatter
|
|
363
|
+
self.logger.debug(f"Parsing Markdown template: {template_path}")
|
|
364
|
+
template_data = self._parse_markdown_template(template_path)
|
|
365
|
+
elif template_path.suffix == ".json":
|
|
366
|
+
# Parse JSON template (legacy format)
|
|
367
|
+
self.logger.debug(f"Parsing JSON template: {template_path}")
|
|
368
|
+
template_content = template_path.read_text()
|
|
369
|
+
template_data = json.loads(template_content)
|
|
370
|
+
else:
|
|
371
|
+
raise ValueError(
|
|
372
|
+
f"Unsupported template format: {template_path.suffix}. "
|
|
373
|
+
f"Expected .md or .json"
|
|
374
|
+
)
|
|
375
|
+
except json.JSONDecodeError as e:
|
|
376
|
+
self.logger.error(f"Invalid JSON in template {template_path}: {e}")
|
|
377
|
+
raise
|
|
378
|
+
except (yaml.YAMLError, ValueError) as e:
|
|
379
|
+
self.logger.error(f"Invalid template {template_path}: {e}")
|
|
380
|
+
raise
|
|
381
|
+
|
|
382
|
+
# Extract tools from template with fallback
|
|
383
|
+
# Handle both dict and list formats for capabilities (backward compatibility)
|
|
384
|
+
capabilities = template_data.get("capabilities", {})
|
|
385
|
+
capabilities_tools = (
|
|
386
|
+
capabilities.get("tools") if isinstance(capabilities, dict) else None
|
|
387
|
+
)
|
|
388
|
+
|
|
389
|
+
# Get raw tools from various possible locations
|
|
390
|
+
raw_tools = (
|
|
391
|
+
template_data.get("tools")
|
|
392
|
+
or capabilities_tools
|
|
393
|
+
or template_data.get("configuration_fields", {}).get("tools")
|
|
394
|
+
)
|
|
395
|
+
|
|
396
|
+
# Normalize tools to a consistent list format
|
|
397
|
+
tools = self.normalize_tools_input(raw_tools)
|
|
398
|
+
|
|
399
|
+
# Log if we see non-standard tool names (info level, not warning)
|
|
400
|
+
standard_tools = {
|
|
401
|
+
"Read",
|
|
402
|
+
"Write",
|
|
403
|
+
"Edit",
|
|
404
|
+
"MultiEdit", # File operations
|
|
405
|
+
"Grep",
|
|
406
|
+
"Glob",
|
|
407
|
+
"LS", # Search and navigation
|
|
408
|
+
"Bash",
|
|
409
|
+
"BashOutput",
|
|
410
|
+
"KillShell", # Command execution
|
|
411
|
+
"TodoWrite",
|
|
412
|
+
"ExitPlanMode", # Task management
|
|
413
|
+
"WebSearch",
|
|
414
|
+
"WebFetch", # Web operations
|
|
415
|
+
"NotebookRead",
|
|
416
|
+
"NotebookEdit", # Jupyter notebook support
|
|
417
|
+
}
|
|
418
|
+
non_standard = [t for t in tools if t not in standard_tools]
|
|
419
|
+
if non_standard:
|
|
420
|
+
self.logger.info(f"Using non-standard tools: {non_standard}")
|
|
421
|
+
|
|
422
|
+
# Extract model from template (no fallback - preserve None if not specified)
|
|
423
|
+
capabilities_model = (
|
|
424
|
+
capabilities.get("model") if isinstance(capabilities, dict) else None
|
|
425
|
+
)
|
|
426
|
+
|
|
427
|
+
model = (
|
|
428
|
+
template_data.get("model")
|
|
429
|
+
or capabilities_model
|
|
430
|
+
or template_data.get("configuration_fields", {}).get("model")
|
|
431
|
+
# No default fallback - preserve None if not set
|
|
432
|
+
)
|
|
433
|
+
|
|
434
|
+
# Convert tools list to comma-separated string (without spaces for compatibility)
|
|
435
|
+
",".join(tools)
|
|
436
|
+
|
|
437
|
+
# Map model names to Claude Code format (as required)
|
|
438
|
+
model_map = {
|
|
439
|
+
"claude-3-5-sonnet-20241022": "sonnet",
|
|
440
|
+
"claude-3-5-haiku-20241022": "haiku",
|
|
441
|
+
"claude-3-opus-20240229": "opus",
|
|
442
|
+
"claude-3-5-sonnet": "sonnet",
|
|
443
|
+
"claude-3-sonnet": "sonnet",
|
|
444
|
+
"claude-3-haiku": "haiku",
|
|
445
|
+
"claude-3-opus": "opus",
|
|
446
|
+
"sonnet": "sonnet",
|
|
447
|
+
"haiku": "haiku",
|
|
448
|
+
"opus": "opus",
|
|
449
|
+
}
|
|
450
|
+
|
|
451
|
+
# Only map model if it's not None
|
|
452
|
+
if model is not None:
|
|
453
|
+
if model in model_map:
|
|
454
|
+
model = model_map[model]
|
|
455
|
+
# If model is specified but not in map, keep as-is (no default)
|
|
456
|
+
|
|
457
|
+
# Get response format from template or use base agent default
|
|
458
|
+
template_data.get("response", {}).get("format", "structured")
|
|
459
|
+
|
|
460
|
+
# Create Claude Code compatible name (lowercase, hyphens only)
|
|
461
|
+
claude_code_name = agent_name.lower().replace("_", "-")
|
|
462
|
+
|
|
463
|
+
# CRITICAL: NO underscores allowed - they cause silent failures!
|
|
464
|
+
|
|
465
|
+
# Validate the name before proceeding
|
|
466
|
+
if not re.match(r"^[a-z0-9]+(-[a-z0-9]+)*$", claude_code_name):
|
|
467
|
+
self.logger.error(
|
|
468
|
+
f"Invalid agent name '{claude_code_name}' - must match ^[a-z0-9]+(-[a-z0-9]+)*$"
|
|
469
|
+
)
|
|
470
|
+
raise ValueError(
|
|
471
|
+
f"Agent name '{claude_code_name}' does not meet Claude Code requirements"
|
|
472
|
+
)
|
|
473
|
+
|
|
474
|
+
# Extract description from template with fallback
|
|
475
|
+
raw_description = (
|
|
476
|
+
template_data.get("description")
|
|
477
|
+
or template_data.get("metadata", {}).get("description")
|
|
478
|
+
or f"{agent_name.title()} agent for specialized tasks"
|
|
479
|
+
)
|
|
480
|
+
|
|
481
|
+
# Convert to multiline format with examples for Claude Code compatibility
|
|
482
|
+
description = self._create_multiline_description(
|
|
483
|
+
raw_description, agent_name, template_data
|
|
484
|
+
)
|
|
485
|
+
|
|
486
|
+
# Extract custom metadata fields
|
|
487
|
+
metadata = template_data.get("metadata", {})
|
|
488
|
+
agent_version = (
|
|
489
|
+
template_data.get("agent_version")
|
|
490
|
+
or template_data.get("version")
|
|
491
|
+
or metadata.get("version", "1.0.0")
|
|
492
|
+
)
|
|
493
|
+
agent_type = template_data.get("agent_type", "general")
|
|
494
|
+
# Use the capabilities_model we already extracted earlier
|
|
495
|
+
model_type = capabilities_model or "sonnet"
|
|
496
|
+
|
|
497
|
+
# Map our model types to Claude Code format
|
|
498
|
+
if model_type in ["opus", "sonnet", "haiku"]:
|
|
499
|
+
# Use inherit for now - Claude Code seems to prefer this
|
|
500
|
+
pass
|
|
501
|
+
else:
|
|
502
|
+
pass
|
|
503
|
+
|
|
504
|
+
# Determine color - prefer template's color, fallback to type-based defaults
|
|
505
|
+
template_metadata = template_data.get("metadata", {})
|
|
506
|
+
template_color = template_metadata.get("color")
|
|
507
|
+
|
|
508
|
+
if template_color:
|
|
509
|
+
# Use the color specified in the template
|
|
510
|
+
pass
|
|
511
|
+
else:
|
|
512
|
+
# Fallback to default color map based on agent type
|
|
513
|
+
color_map = {
|
|
514
|
+
"engineer": "blue",
|
|
515
|
+
"qa": "green",
|
|
516
|
+
"security": "red",
|
|
517
|
+
"research": "purple",
|
|
518
|
+
"documentation": "cyan", # Changed default to match template preference
|
|
519
|
+
"ops": "gray",
|
|
520
|
+
}
|
|
521
|
+
color_map.get(agent_type, "blue")
|
|
522
|
+
|
|
523
|
+
# Check if we should include tools field (only if significantly restricting)
|
|
524
|
+
# Claude Code approach: omit tools field unless specifically restricting
|
|
525
|
+
|
|
526
|
+
# Convert tools to set for comparison
|
|
527
|
+
agent_tools = set(tools) if isinstance(tools, list) else set(tools.split(","))
|
|
528
|
+
|
|
529
|
+
# Only include tools field if agent is missing several important tools
|
|
530
|
+
# This matches Claude Code's approach of omitting tools for general-purpose agents
|
|
531
|
+
core_tools = {"Read", "Write", "Edit", "Bash", "Grep", "Glob"}
|
|
532
|
+
has_core_tools = len(agent_tools.intersection(core_tools)) >= 5
|
|
533
|
+
|
|
534
|
+
# Include tools field only if agent is clearly restricted (missing core tools or very few tools)
|
|
535
|
+
not has_core_tools or len(agent_tools) < 6
|
|
536
|
+
|
|
537
|
+
# Build YAML frontmatter with all relevant metadata from JSON template
|
|
538
|
+
# Include all fields that are useful for agent management and functionality
|
|
539
|
+
#
|
|
540
|
+
# COMPREHENSIVE AGENT FRONTMATTER FORMAT:
|
|
541
|
+
# - name: kebab-case agent name (required)
|
|
542
|
+
# - description: when/why to use this agent with examples (required, multiline)
|
|
543
|
+
# - model: mapped model name (required)
|
|
544
|
+
# - type: agent type for categorization and functionality (optional but important)
|
|
545
|
+
# - category: organizational category (optional)
|
|
546
|
+
# - color: visual identifier (optional)
|
|
547
|
+
# - version: agent version for update tracking (optional)
|
|
548
|
+
# - author: creator information (optional)
|
|
549
|
+
# - created_at: creation timestamp (optional)
|
|
550
|
+
# - updated_at: last update timestamp (optional)
|
|
551
|
+
# - tags: list of tags for search and categorization (optional)
|
|
552
|
+
frontmatter_lines = [
|
|
553
|
+
"---",
|
|
554
|
+
f"name: {claude_code_name}",
|
|
555
|
+
]
|
|
556
|
+
|
|
557
|
+
# Add description as single-line YAML string with \n escapes
|
|
558
|
+
frontmatter_lines.append(
|
|
559
|
+
f"description: {self._format_description_for_yaml(description)}"
|
|
560
|
+
)
|
|
561
|
+
|
|
562
|
+
# Add model field only if explicitly set (not required for Claude Code)
|
|
563
|
+
if model is not None:
|
|
564
|
+
frontmatter_lines.append(f"model: {model}")
|
|
565
|
+
|
|
566
|
+
# Add type field (important for agent categorization)
|
|
567
|
+
if agent_type and agent_type != "general":
|
|
568
|
+
frontmatter_lines.append(f"type: {agent_type}")
|
|
569
|
+
|
|
570
|
+
# Add optional metadata fields
|
|
571
|
+
if metadata.get("color"):
|
|
572
|
+
frontmatter_lines.append(f"color: {metadata['color']}")
|
|
573
|
+
if metadata.get("category"):
|
|
574
|
+
frontmatter_lines.append(f"category: {metadata['category']}")
|
|
575
|
+
# Always include version field to prevent deployment comparison issues
|
|
576
|
+
if agent_version:
|
|
577
|
+
frontmatter_lines.append(f'version: "{agent_version}"')
|
|
578
|
+
if metadata.get("author"):
|
|
579
|
+
frontmatter_lines.append(f'author: "{metadata["author"]}"')
|
|
580
|
+
if metadata.get("created_at"):
|
|
581
|
+
frontmatter_lines.append(f"created_at: {metadata['created_at']}")
|
|
582
|
+
if metadata.get("updated_at"):
|
|
583
|
+
frontmatter_lines.append(f"updated_at: {metadata['updated_at']}")
|
|
584
|
+
# Add tags as comma-separated string if they exist (consistent with tools format)
|
|
585
|
+
if metadata.get("tags") and isinstance(metadata["tags"], list):
|
|
586
|
+
tags_str = ",".join(metadata["tags"])
|
|
587
|
+
frontmatter_lines.append(f"tags: {tags_str}")
|
|
588
|
+
|
|
589
|
+
frontmatter_lines.extend(
|
|
590
|
+
[
|
|
591
|
+
"---",
|
|
592
|
+
"",
|
|
593
|
+
]
|
|
594
|
+
)
|
|
595
|
+
|
|
596
|
+
frontmatter = "\n".join(frontmatter_lines)
|
|
597
|
+
|
|
598
|
+
# Get agent instructions from template data (primary) or base agent data (fallback)
|
|
599
|
+
raw_instructions = template_data.get("instructions")
|
|
600
|
+
|
|
601
|
+
# Handle dictionary instructions format
|
|
602
|
+
if isinstance(raw_instructions, dict):
|
|
603
|
+
agent_specific_instructions = self._convert_instructions_dict_to_markdown(
|
|
604
|
+
raw_instructions
|
|
605
|
+
)
|
|
606
|
+
else:
|
|
607
|
+
agent_specific_instructions = (
|
|
608
|
+
raw_instructions
|
|
609
|
+
or base_agent_data.get("content")
|
|
610
|
+
or base_agent_data.get("instructions")
|
|
611
|
+
or "# Agent Instructions\n\nThis agent provides specialized assistance."
|
|
612
|
+
)
|
|
613
|
+
|
|
614
|
+
# Compose hierarchical BASE-AGENT.md templates
|
|
615
|
+
# Order: agent-specific + local BASE + parent BASE + ... + root BASE
|
|
616
|
+
content_parts = [agent_specific_instructions]
|
|
617
|
+
|
|
618
|
+
# Discover BASE-AGENT.md files in directory hierarchy
|
|
619
|
+
base_templates = self._discover_base_agent_templates(template_path)
|
|
620
|
+
|
|
621
|
+
# Append each BASE template (order: closest to farthest)
|
|
622
|
+
for base_template_path in base_templates:
|
|
623
|
+
try:
|
|
624
|
+
base_content = base_template_path.read_text(encoding="utf-8")
|
|
625
|
+
if base_content.strip():
|
|
626
|
+
content_parts.append(base_content)
|
|
627
|
+
self.logger.debug(
|
|
628
|
+
f"Composed BASE template: {base_template_path.relative_to(template_path.parent.parent) if template_path.parent.parent in base_template_path.parents else base_template_path.name}"
|
|
629
|
+
)
|
|
630
|
+
except Exception as e:
|
|
631
|
+
self.logger.warning(
|
|
632
|
+
f"Failed to read BASE template {base_template_path}: {e}"
|
|
633
|
+
)
|
|
634
|
+
|
|
635
|
+
# Fallback: Load legacy BASE_{TYPE}.md if no hierarchical templates found
|
|
636
|
+
if len(content_parts) == 1: # Only agent-specific instructions
|
|
637
|
+
legacy_base_instructions = self._load_base_agent_instructions(agent_type)
|
|
638
|
+
if legacy_base_instructions:
|
|
639
|
+
content_parts.append(legacy_base_instructions)
|
|
640
|
+
self.logger.debug(
|
|
641
|
+
f"Using legacy BASE_{agent_type.upper()}.md (no hierarchical BASE-AGENT.md found)"
|
|
642
|
+
)
|
|
643
|
+
|
|
644
|
+
# Join all parts with separator
|
|
645
|
+
content = "\n\n---\n\n".join(content_parts)
|
|
646
|
+
|
|
647
|
+
# Add memory update instructions if not already present
|
|
648
|
+
if "memory-update" not in content and "Remember" not in content:
|
|
649
|
+
memory_instructions = """
|
|
650
|
+
|
|
651
|
+
## Memory Updates
|
|
652
|
+
|
|
653
|
+
When you learn something important about this project that would be useful for future tasks, include it in your response JSON block:
|
|
654
|
+
|
|
655
|
+
```json
|
|
656
|
+
{
|
|
657
|
+
"memory-update": {
|
|
658
|
+
"Project Architecture": ["Key architectural patterns or structures"],
|
|
659
|
+
"Implementation Guidelines": ["Important coding standards or practices"],
|
|
660
|
+
"Current Technical Context": ["Project-specific technical details"]
|
|
661
|
+
}
|
|
662
|
+
}
|
|
663
|
+
```
|
|
664
|
+
|
|
665
|
+
Or use the simpler "remember" field for general learnings:
|
|
666
|
+
|
|
667
|
+
```json
|
|
668
|
+
{
|
|
669
|
+
"remember": ["Learning 1", "Learning 2"]
|
|
670
|
+
}
|
|
671
|
+
```
|
|
672
|
+
|
|
673
|
+
Only include memories that are:
|
|
674
|
+
- Project-specific (not generic programming knowledge)
|
|
675
|
+
- Likely to be useful in future tasks
|
|
676
|
+
- Not already documented elsewhere
|
|
677
|
+
"""
|
|
678
|
+
content = content + memory_instructions
|
|
679
|
+
|
|
680
|
+
# Combine frontmatter and content
|
|
681
|
+
return frontmatter + content
|
|
682
|
+
|
|
683
|
+
def build_agent_yaml(
|
|
684
|
+
self, agent_name: str, template_path: Path, base_agent_data: dict
|
|
685
|
+
) -> str:
|
|
686
|
+
"""
|
|
687
|
+
Build a complete agent YAML file by combining base agent and template.
|
|
688
|
+
Only includes essential fields for Claude Code best practices.
|
|
689
|
+
|
|
690
|
+
Args:
|
|
691
|
+
agent_name: Name of the agent
|
|
692
|
+
template_path: Path to the agent template JSON file
|
|
693
|
+
base_agent_data: Base agent configuration data
|
|
694
|
+
|
|
695
|
+
Returns:
|
|
696
|
+
Complete YAML content
|
|
697
|
+
|
|
698
|
+
Raises:
|
|
699
|
+
FileNotFoundError: If template file doesn't exist
|
|
700
|
+
json.JSONDecodeError: If template JSON is invalid
|
|
701
|
+
"""
|
|
702
|
+
if not template_path.exists():
|
|
703
|
+
raise FileNotFoundError(f"Template file not found: {template_path}")
|
|
704
|
+
|
|
705
|
+
try:
|
|
706
|
+
template_content = template_path.read_text()
|
|
707
|
+
template_data = json.loads(template_content)
|
|
708
|
+
except json.JSONDecodeError as e:
|
|
709
|
+
self.logger.error(f"Invalid JSON in template {template_path}: {e}")
|
|
710
|
+
raise
|
|
711
|
+
|
|
712
|
+
# Merge narrative and configuration fields
|
|
713
|
+
self.merge_narrative_fields(base_agent_data, template_data)
|
|
714
|
+
merged_config = self.merge_configuration_fields(base_agent_data, template_data)
|
|
715
|
+
|
|
716
|
+
# Extract essential fields for Claude Code
|
|
717
|
+
name = template_data.get("name", agent_name)
|
|
718
|
+
description = template_data.get(
|
|
719
|
+
"description", f"{name} agent for specialized tasks"
|
|
720
|
+
)
|
|
721
|
+
|
|
722
|
+
# Get tools and model (no fallback for model)
|
|
723
|
+
raw_tools = merged_config.get("tools")
|
|
724
|
+
tools = self.normalize_tools_input(raw_tools)
|
|
725
|
+
model = merged_config.get("model") # No default - preserve None
|
|
726
|
+
|
|
727
|
+
# Format tools as YAML list
|
|
728
|
+
tools_yaml = self.format_yaml_list(tools, 2)
|
|
729
|
+
|
|
730
|
+
# Build YAML content with only essential fields
|
|
731
|
+
yaml_lines = [
|
|
732
|
+
f"name: {name}",
|
|
733
|
+
f"description: {description}",
|
|
734
|
+
]
|
|
735
|
+
|
|
736
|
+
# Only include model if explicitly set
|
|
737
|
+
if model is not None:
|
|
738
|
+
yaml_lines.append(f"model: {model}")
|
|
739
|
+
|
|
740
|
+
yaml_lines.extend(
|
|
741
|
+
[
|
|
742
|
+
"tools:",
|
|
743
|
+
tools_yaml,
|
|
744
|
+
]
|
|
745
|
+
)
|
|
746
|
+
|
|
747
|
+
return "\n".join(yaml_lines) + "\n"
|
|
748
|
+
|
|
749
|
+
def merge_narrative_fields(self, base_data: dict, template_data: dict) -> dict:
|
|
750
|
+
"""
|
|
751
|
+
Merge narrative fields from base and template, combining arrays.
|
|
752
|
+
|
|
753
|
+
Args:
|
|
754
|
+
base_data: Base agent data
|
|
755
|
+
template_data: Template agent data
|
|
756
|
+
|
|
757
|
+
Returns:
|
|
758
|
+
Merged narrative fields
|
|
759
|
+
"""
|
|
760
|
+
merged = {}
|
|
761
|
+
|
|
762
|
+
# Fields that should be combined (arrays)
|
|
763
|
+
combinable_fields = [
|
|
764
|
+
"when_to_use",
|
|
765
|
+
"specialized_knowledge",
|
|
766
|
+
"unique_capabilities",
|
|
767
|
+
]
|
|
768
|
+
|
|
769
|
+
for field in combinable_fields:
|
|
770
|
+
base_value = base_data.get(field, [])
|
|
771
|
+
template_value = template_data.get(field, [])
|
|
772
|
+
|
|
773
|
+
# Ensure both are lists
|
|
774
|
+
if not isinstance(base_value, list):
|
|
775
|
+
base_value = [base_value] if base_value else []
|
|
776
|
+
if not isinstance(template_value, list):
|
|
777
|
+
template_value = [template_value] if template_value else []
|
|
778
|
+
|
|
779
|
+
# Combine and deduplicate
|
|
780
|
+
combined = list(set(base_value + template_value))
|
|
781
|
+
merged[field] = combined
|
|
782
|
+
|
|
783
|
+
return merged
|
|
784
|
+
|
|
785
|
+
def merge_configuration_fields(self, base_data: dict, template_data: dict) -> dict:
|
|
786
|
+
"""
|
|
787
|
+
Merge configuration fields, with template overriding base.
|
|
788
|
+
|
|
789
|
+
Args:
|
|
790
|
+
base_data: Base agent data
|
|
791
|
+
template_data: Template agent data
|
|
792
|
+
|
|
793
|
+
Returns:
|
|
794
|
+
Merged configuration fields
|
|
795
|
+
"""
|
|
796
|
+
merged = {}
|
|
797
|
+
|
|
798
|
+
# Start with base configuration
|
|
799
|
+
if "configuration_fields" in base_data:
|
|
800
|
+
merged.update(base_data["configuration_fields"])
|
|
801
|
+
|
|
802
|
+
# Override with template configuration
|
|
803
|
+
if "configuration_fields" in template_data:
|
|
804
|
+
merged.update(template_data["configuration_fields"])
|
|
805
|
+
|
|
806
|
+
# Also check for direct fields in template
|
|
807
|
+
direct_fields = ["tools", "model", "timeout", "max_tokens"]
|
|
808
|
+
for field in direct_fields:
|
|
809
|
+
if field in template_data:
|
|
810
|
+
merged[field] = template_data[field]
|
|
811
|
+
|
|
812
|
+
return merged
|
|
813
|
+
|
|
814
|
+
def extract_agent_metadata(self, template_content: str) -> Dict[str, Any]:
|
|
815
|
+
"""
|
|
816
|
+
Extract metadata from simplified agent template content.
|
|
817
|
+
|
|
818
|
+
Args:
|
|
819
|
+
template_content: Agent template markdown content
|
|
820
|
+
|
|
821
|
+
Returns:
|
|
822
|
+
Dictionary containing extracted metadata
|
|
823
|
+
"""
|
|
824
|
+
metadata = {}
|
|
825
|
+
current_section = None
|
|
826
|
+
section_content = []
|
|
827
|
+
|
|
828
|
+
lines = template_content.split("\n")
|
|
829
|
+
|
|
830
|
+
for line in lines:
|
|
831
|
+
line = line.strip()
|
|
832
|
+
|
|
833
|
+
# Check for section headers
|
|
834
|
+
if line.startswith("## "):
|
|
835
|
+
# Save previous section
|
|
836
|
+
if current_section and section_content:
|
|
837
|
+
metadata[current_section] = section_content.copy()
|
|
838
|
+
|
|
839
|
+
# Start new section
|
|
840
|
+
current_section = line[3:].lower().replace(" ", "_")
|
|
841
|
+
section_content = []
|
|
842
|
+
|
|
843
|
+
elif line.startswith("- ") and current_section:
|
|
844
|
+
# Add list item to current section
|
|
845
|
+
section_content.append(line[2:])
|
|
846
|
+
|
|
847
|
+
elif line and current_section and not line.startswith("#"):
|
|
848
|
+
# Add non-empty, non-header line to current section
|
|
849
|
+
section_content.append(line)
|
|
850
|
+
|
|
851
|
+
# Save final section
|
|
852
|
+
if current_section and section_content:
|
|
853
|
+
metadata[current_section] = section_content.copy()
|
|
854
|
+
|
|
855
|
+
# Ensure all required fields have defaults
|
|
856
|
+
metadata.setdefault("when_to_use", [])
|
|
857
|
+
metadata.setdefault("specialized_knowledge", [])
|
|
858
|
+
metadata.setdefault("unique_capabilities", [])
|
|
859
|
+
|
|
860
|
+
return metadata
|
|
861
|
+
|
|
862
|
+
def format_yaml_list(self, items: List[str], indent: int) -> str:
|
|
863
|
+
"""
|
|
864
|
+
Format a list for YAML with proper indentation.
|
|
865
|
+
|
|
866
|
+
Args:
|
|
867
|
+
items: List of items to format
|
|
868
|
+
indent: Number of spaces for indentation
|
|
869
|
+
|
|
870
|
+
Returns:
|
|
871
|
+
Formatted YAML list string
|
|
872
|
+
"""
|
|
873
|
+
if not items:
|
|
874
|
+
return ""
|
|
875
|
+
|
|
876
|
+
indent_str = " " * indent
|
|
877
|
+
formatted_items = []
|
|
878
|
+
|
|
879
|
+
for item in items:
|
|
880
|
+
formatted_items.append(f"{indent_str}- {item}")
|
|
881
|
+
|
|
882
|
+
return "\n".join(formatted_items)
|
|
883
|
+
|
|
884
|
+
def _create_multiline_description(
|
|
885
|
+
self, raw_description: str, agent_name: str, template_data: dict
|
|
886
|
+
) -> str:
|
|
887
|
+
"""
|
|
888
|
+
Create a comprehensive multiline description with examples for Claude Code compatibility.
|
|
889
|
+
Based on Claude's software-engineer.md format: detailed when/why description with examples.
|
|
890
|
+
|
|
891
|
+
Args:
|
|
892
|
+
raw_description: Original single-line description
|
|
893
|
+
agent_name: Name of the agent
|
|
894
|
+
template_data: Template data for extracting examples
|
|
895
|
+
|
|
896
|
+
Returns:
|
|
897
|
+
Formatted multiline description with examples in Claude Code format
|
|
898
|
+
"""
|
|
899
|
+
raw_description = self._format_to_single_line(raw_description)
|
|
900
|
+
|
|
901
|
+
# Get agent type for creating targeted descriptions
|
|
902
|
+
agent_type = template_data.get("agent_type", "general")
|
|
903
|
+
|
|
904
|
+
# Create enhanced description based on agent type
|
|
905
|
+
enhanced_description = self._create_enhanced_description(
|
|
906
|
+
raw_description, agent_name, agent_type, template_data
|
|
907
|
+
)
|
|
908
|
+
|
|
909
|
+
# Add examples
|
|
910
|
+
examples = self._extract_examples_from_template(template_data, agent_name)
|
|
911
|
+
if not examples:
|
|
912
|
+
examples = self._generate_default_examples(agent_name, template_data)
|
|
913
|
+
|
|
914
|
+
# Combine enhanced description with examples
|
|
915
|
+
if examples:
|
|
916
|
+
description_parts = [enhanced_description, "", *examples]
|
|
917
|
+
else:
|
|
918
|
+
description_parts = [enhanced_description]
|
|
919
|
+
|
|
920
|
+
return "\n".join(description_parts)
|
|
921
|
+
|
|
922
|
+
def _format_to_single_line(self, description: str) -> str:
|
|
923
|
+
"""
|
|
924
|
+
Format description to single line by removing line breaks and normalizing whitespace.
|
|
925
|
+
|
|
926
|
+
Args:
|
|
927
|
+
description: Raw description text
|
|
928
|
+
|
|
929
|
+
Returns:
|
|
930
|
+
Single-line formatted description
|
|
931
|
+
"""
|
|
932
|
+
if not description:
|
|
933
|
+
return description
|
|
934
|
+
|
|
935
|
+
# Remove all line breaks and normalize whitespace
|
|
936
|
+
single_line = " ".join(description.strip().split())
|
|
937
|
+
|
|
938
|
+
# Remove redundant spaces around punctuation
|
|
939
|
+
single_line = re.sub(r"\s+([,.!?;:])", r"\1", single_line)
|
|
940
|
+
return re.sub(r"([,.!?;:])\s+", r"\1 ", single_line)
|
|
941
|
+
|
|
942
|
+
def _create_enhanced_description(
|
|
943
|
+
self,
|
|
944
|
+
raw_description: str,
|
|
945
|
+
agent_name: str,
|
|
946
|
+
agent_type: str,
|
|
947
|
+
template_data: dict,
|
|
948
|
+
) -> str:
|
|
949
|
+
"""
|
|
950
|
+
Create an enhanced description based on agent type that follows Claude's format.
|
|
951
|
+
|
|
952
|
+
Args:
|
|
953
|
+
raw_description: Original description
|
|
954
|
+
agent_name: Name of the agent
|
|
955
|
+
agent_type: Type of agent (engineer, qa, research, etc.)
|
|
956
|
+
template_data: Template data for additional context
|
|
957
|
+
|
|
958
|
+
Returns:
|
|
959
|
+
Enhanced description string
|
|
960
|
+
"""
|
|
961
|
+
# Type-specific enhanced descriptions following Claude's software-engineer.md pattern
|
|
962
|
+
enhanced_descriptions = {
|
|
963
|
+
"engineer": "Use this agent when you need to implement new features, write production-quality code, refactor existing code, or solve complex programming challenges. This agent excels at translating requirements into well-architected, maintainable code solutions across various programming languages and frameworks.",
|
|
964
|
+
"qa": "Use this agent when you need comprehensive testing, quality assurance validation, or test automation. This agent specializes in creating robust test suites, identifying edge cases, and ensuring code quality through systematic testing approaches across different testing methodologies.",
|
|
965
|
+
"research": "Use this agent when you need to investigate codebases, analyze system architecture, or gather technical insights. This agent excels at code exploration, pattern identification, and providing comprehensive analysis of existing systems while maintaining strict memory efficiency.",
|
|
966
|
+
"ops": "Use this agent when you need infrastructure management, deployment automation, or operational excellence. This agent specializes in DevOps practices, cloud operations, monitoring setup, and maintaining reliable production systems.",
|
|
967
|
+
"security": "Use this agent when you need security analysis, vulnerability assessment, or secure coding practices. This agent excels at identifying security risks, implementing security best practices, and ensuring applications meet security standards.",
|
|
968
|
+
"documentation": "Use this agent when you need to create, update, or maintain technical documentation. This agent specializes in writing clear, comprehensive documentation including API docs, user guides, and technical specifications.",
|
|
969
|
+
}
|
|
970
|
+
|
|
971
|
+
# Get the enhanced description or fallback to the original with improvements
|
|
972
|
+
if agent_type in enhanced_descriptions:
|
|
973
|
+
return enhanced_descriptions[agent_type]
|
|
974
|
+
# Enhance the raw description if it's a custom type
|
|
975
|
+
if raw_description and len(raw_description) > 10:
|
|
976
|
+
return f"Use this agent when you need specialized assistance with {raw_description.lower()}. This agent provides targeted expertise and follows best practices for {agent_name.replace('-', ' ')} related tasks."
|
|
977
|
+
return f"Use this agent when you need specialized assistance from the {agent_name.replace('-', ' ')} agent. This agent provides targeted expertise and follows established best practices."
|
|
978
|
+
|
|
979
|
+
def _extract_examples_from_template(
|
|
980
|
+
self, template_data: dict, agent_name: str
|
|
981
|
+
) -> List[str]:
|
|
982
|
+
"""
|
|
983
|
+
Extract examples from template data and format with commentary.
|
|
984
|
+
Creates ONE example with commentary from template data.
|
|
985
|
+
|
|
986
|
+
Args:
|
|
987
|
+
template_data: Template data
|
|
988
|
+
agent_name: Name of the agent
|
|
989
|
+
|
|
990
|
+
Returns:
|
|
991
|
+
List of example strings (single example with commentary)
|
|
992
|
+
"""
|
|
993
|
+
examples = []
|
|
994
|
+
|
|
995
|
+
# Check for examples in knowledge section
|
|
996
|
+
knowledge = template_data.get("knowledge", {})
|
|
997
|
+
template_examples = knowledge.get("examples", [])
|
|
998
|
+
|
|
999
|
+
if template_examples:
|
|
1000
|
+
# Take only the first example and add commentary
|
|
1001
|
+
example = template_examples[0]
|
|
1002
|
+
scenario = example.get("scenario", "")
|
|
1003
|
+
approach = example.get("approach", "")
|
|
1004
|
+
commentary = example.get("commentary", "")
|
|
1005
|
+
|
|
1006
|
+
if scenario and approach:
|
|
1007
|
+
examples.extend(
|
|
1008
|
+
[
|
|
1009
|
+
"<example>",
|
|
1010
|
+
f"Context: {scenario}",
|
|
1011
|
+
f'user: "I need help with {scenario.lower()}"',
|
|
1012
|
+
f'assistant: "I\'ll use the {agent_name} agent to {approach.lower()}."',
|
|
1013
|
+
"<commentary>",
|
|
1014
|
+
(
|
|
1015
|
+
commentary
|
|
1016
|
+
if commentary
|
|
1017
|
+
else f"This agent is well-suited for {scenario.lower()} because it specializes in {approach.lower()} with targeted expertise."
|
|
1018
|
+
),
|
|
1019
|
+
"</commentary>",
|
|
1020
|
+
"</example>",
|
|
1021
|
+
]
|
|
1022
|
+
)
|
|
1023
|
+
|
|
1024
|
+
# Check for triggers that can be converted to examples
|
|
1025
|
+
interactions = template_data.get("interactions", {})
|
|
1026
|
+
triggers = interactions.get("triggers", [])
|
|
1027
|
+
|
|
1028
|
+
if triggers and not examples:
|
|
1029
|
+
# Convert first trigger to example with commentary
|
|
1030
|
+
trigger = triggers[0]
|
|
1031
|
+
|
|
1032
|
+
# Handle both string and dict trigger formats
|
|
1033
|
+
if isinstance(trigger, dict):
|
|
1034
|
+
# New format with pattern and confidence
|
|
1035
|
+
trigger_text = trigger.get("pattern", "")
|
|
1036
|
+
else:
|
|
1037
|
+
# Old format with simple string
|
|
1038
|
+
trigger_text = str(trigger)
|
|
1039
|
+
|
|
1040
|
+
# Skip if we don't have valid trigger text
|
|
1041
|
+
if not trigger_text:
|
|
1042
|
+
return examples
|
|
1043
|
+
|
|
1044
|
+
agent_type = template_data.get("agent_type", "general")
|
|
1045
|
+
|
|
1046
|
+
examples.extend(
|
|
1047
|
+
[
|
|
1048
|
+
"<example>",
|
|
1049
|
+
f"Context: When user needs {trigger_text}",
|
|
1050
|
+
f'user: "{trigger_text}"',
|
|
1051
|
+
f'assistant: "I\'ll use the {agent_name} agent for {trigger_text}."',
|
|
1052
|
+
"<commentary>",
|
|
1053
|
+
f"This {agent_type} agent is appropriate because it has specialized capabilities for {trigger_text.lower()} tasks.",
|
|
1054
|
+
"</commentary>",
|
|
1055
|
+
"</example>",
|
|
1056
|
+
]
|
|
1057
|
+
)
|
|
1058
|
+
|
|
1059
|
+
return examples
|
|
1060
|
+
|
|
1061
|
+
def _generate_default_examples(
|
|
1062
|
+
self, agent_name: str, template_data: dict
|
|
1063
|
+
) -> List[str]:
|
|
1064
|
+
"""
|
|
1065
|
+
Generate default examples when none are available in template.
|
|
1066
|
+
Creates ONE example with commentary for each agent type.
|
|
1067
|
+
|
|
1068
|
+
Args:
|
|
1069
|
+
agent_name: Name of the agent
|
|
1070
|
+
template_data: Template data for context
|
|
1071
|
+
|
|
1072
|
+
Returns:
|
|
1073
|
+
List of example strings (single example with commentary)
|
|
1074
|
+
"""
|
|
1075
|
+
agent_type = template_data.get("agent_type", "general")
|
|
1076
|
+
|
|
1077
|
+
# Create type-specific examples with commentary inside
|
|
1078
|
+
type_examples = {
|
|
1079
|
+
"engineer": [
|
|
1080
|
+
"<example>",
|
|
1081
|
+
"Context: When you need to implement new features or write code.",
|
|
1082
|
+
'user: "I need to add authentication to my API"',
|
|
1083
|
+
f'assistant: "I\'ll use the {agent_name} agent to implement a secure authentication system for your API."',
|
|
1084
|
+
"<commentary>",
|
|
1085
|
+
"The engineer agent is ideal for code implementation tasks because it specializes in writing production-quality code, following best practices, and creating well-architected solutions.",
|
|
1086
|
+
"</commentary>",
|
|
1087
|
+
"</example>",
|
|
1088
|
+
],
|
|
1089
|
+
"ops": [
|
|
1090
|
+
"<example>",
|
|
1091
|
+
"Context: When you need to deploy or manage infrastructure.",
|
|
1092
|
+
'user: "I need to deploy my application to the cloud"',
|
|
1093
|
+
f'assistant: "I\'ll use the {agent_name} agent to set up and deploy your application infrastructure."',
|
|
1094
|
+
"<commentary>",
|
|
1095
|
+
"The ops agent excels at infrastructure management and deployment automation, ensuring reliable and scalable production systems.",
|
|
1096
|
+
"</commentary>",
|
|
1097
|
+
"</example>",
|
|
1098
|
+
],
|
|
1099
|
+
"qa": [
|
|
1100
|
+
"<example>",
|
|
1101
|
+
"Context: When you need to test or validate functionality.",
|
|
1102
|
+
'user: "I need to write tests for my new feature"',
|
|
1103
|
+
f'assistant: "I\'ll use the {agent_name} agent to create comprehensive tests for your feature."',
|
|
1104
|
+
"<commentary>",
|
|
1105
|
+
"The QA agent specializes in comprehensive testing strategies, quality assurance validation, and creating robust test suites that ensure code reliability.",
|
|
1106
|
+
"</commentary>",
|
|
1107
|
+
"</example>",
|
|
1108
|
+
],
|
|
1109
|
+
"research": [
|
|
1110
|
+
"<example>",
|
|
1111
|
+
"Context: When you need to investigate or analyze existing codebases.",
|
|
1112
|
+
'user: "I need to understand how the authentication system works in this project"',
|
|
1113
|
+
f'assistant: "I\'ll use the {agent_name} agent to analyze the codebase and explain the authentication implementation."',
|
|
1114
|
+
"<commentary>",
|
|
1115
|
+
"The research agent is perfect for code exploration and analysis tasks, providing thorough investigation of existing systems while maintaining memory efficiency.",
|
|
1116
|
+
"</commentary>",
|
|
1117
|
+
"</example>",
|
|
1118
|
+
],
|
|
1119
|
+
"security": [
|
|
1120
|
+
"<example>",
|
|
1121
|
+
"Context: When you need to review code for security vulnerabilities.",
|
|
1122
|
+
'user: "I need a security review of my authentication implementation"',
|
|
1123
|
+
f'assistant: "I\'ll use the {agent_name} agent to conduct a thorough security analysis of your authentication code."',
|
|
1124
|
+
"<commentary>",
|
|
1125
|
+
"The security agent specializes in identifying security risks, vulnerability assessment, and ensuring applications meet security standards and best practices.",
|
|
1126
|
+
"</commentary>",
|
|
1127
|
+
"</example>",
|
|
1128
|
+
],
|
|
1129
|
+
"documentation": [
|
|
1130
|
+
"<example>",
|
|
1131
|
+
"Context: When you need to create or update technical documentation.",
|
|
1132
|
+
'user: "I need to document this new API endpoint"',
|
|
1133
|
+
f'assistant: "I\'ll use the {agent_name} agent to create comprehensive API documentation."',
|
|
1134
|
+
"<commentary>",
|
|
1135
|
+
"The documentation agent excels at creating clear, comprehensive technical documentation including API docs, user guides, and technical specifications.",
|
|
1136
|
+
"</commentary>",
|
|
1137
|
+
"</example>",
|
|
1138
|
+
],
|
|
1139
|
+
}
|
|
1140
|
+
|
|
1141
|
+
return type_examples.get(
|
|
1142
|
+
agent_type,
|
|
1143
|
+
[
|
|
1144
|
+
"<example>",
|
|
1145
|
+
f"Context: When you need specialized assistance from the {agent_name} agent.",
|
|
1146
|
+
f'user: "I need help with {agent_name.replace("-", " ")} tasks"',
|
|
1147
|
+
f'assistant: "I\'ll use the {agent_name} agent to provide specialized assistance."',
|
|
1148
|
+
"<commentary>",
|
|
1149
|
+
f"This agent provides targeted expertise for {agent_name.replace('-', ' ')} related tasks and follows established best practices.",
|
|
1150
|
+
"</commentary>",
|
|
1151
|
+
"</example>",
|
|
1152
|
+
],
|
|
1153
|
+
)
|
|
1154
|
+
|
|
1155
|
+
def _indent_multiline_text(self, text: str, spaces: int) -> str:
|
|
1156
|
+
"""
|
|
1157
|
+
Indent multiline text with specified number of spaces.
|
|
1158
|
+
|
|
1159
|
+
Args:
|
|
1160
|
+
text: Text to indent
|
|
1161
|
+
spaces: Number of spaces for indentation
|
|
1162
|
+
|
|
1163
|
+
Returns:
|
|
1164
|
+
Indented text
|
|
1165
|
+
"""
|
|
1166
|
+
if not text:
|
|
1167
|
+
return ""
|
|
1168
|
+
|
|
1169
|
+
indent = " " * spaces
|
|
1170
|
+
lines = text.split("\n")
|
|
1171
|
+
indented_lines = []
|
|
1172
|
+
|
|
1173
|
+
for line in lines:
|
|
1174
|
+
if line.strip(): # Non-empty lines get indented
|
|
1175
|
+
indented_lines.append(indent + line)
|
|
1176
|
+
else: # Empty lines stay empty
|
|
1177
|
+
indented_lines.append("")
|
|
1178
|
+
|
|
1179
|
+
return "\n".join(indented_lines)
|
|
1180
|
+
|
|
1181
|
+
def _format_description_for_yaml(self, description: str) -> str:
|
|
1182
|
+
"""Format description as a single-line YAML string with escaped newlines.
|
|
1183
|
+
|
|
1184
|
+
Args:
|
|
1185
|
+
description: Multi-line description text
|
|
1186
|
+
|
|
1187
|
+
Returns:
|
|
1188
|
+
Single-line YAML-formatted string with \n escapes
|
|
1189
|
+
"""
|
|
1190
|
+
if not description:
|
|
1191
|
+
return '""'
|
|
1192
|
+
|
|
1193
|
+
# The description already contains actual newlines, we need to escape them
|
|
1194
|
+
# Replace actual newlines with \n escape sequence
|
|
1195
|
+
escaped = description.replace("\n", "\\n")
|
|
1196
|
+
|
|
1197
|
+
# Escape any quotes in the description
|
|
1198
|
+
escaped = escaped.replace('"', '\\"')
|
|
1199
|
+
|
|
1200
|
+
# Return as quoted string
|
|
1201
|
+
return f'"{escaped}"'
|
|
1202
|
+
|
|
1203
|
+
def _convert_instructions_dict_to_markdown(self, instructions_dict: dict) -> str:
|
|
1204
|
+
"""Convert complex instructions dictionary to markdown format.
|
|
1205
|
+
|
|
1206
|
+
Args:
|
|
1207
|
+
instructions_dict: Dictionary containing structured instructions
|
|
1208
|
+
|
|
1209
|
+
Returns:
|
|
1210
|
+
Formatted markdown string representing the instructions
|
|
1211
|
+
"""
|
|
1212
|
+
if not instructions_dict:
|
|
1213
|
+
return "# Agent Instructions\n\nThis agent provides specialized assistance."
|
|
1214
|
+
|
|
1215
|
+
markdown_parts = []
|
|
1216
|
+
|
|
1217
|
+
# Add primary role
|
|
1218
|
+
if "primary_role" in instructions_dict:
|
|
1219
|
+
markdown_parts.extend(["# Role", "", instructions_dict["primary_role"], ""])
|
|
1220
|
+
|
|
1221
|
+
# Add core identity
|
|
1222
|
+
if "core_identity" in instructions_dict:
|
|
1223
|
+
markdown_parts.extend(
|
|
1224
|
+
["## Core Identity", "", instructions_dict["core_identity"], ""]
|
|
1225
|
+
)
|
|
1226
|
+
|
|
1227
|
+
# Add responsibilities
|
|
1228
|
+
if "responsibilities" in instructions_dict:
|
|
1229
|
+
markdown_parts.extend(["## Responsibilities", ""])
|
|
1230
|
+
|
|
1231
|
+
responsibilities = instructions_dict["responsibilities"]
|
|
1232
|
+
if isinstance(responsibilities, list):
|
|
1233
|
+
for resp in responsibilities:
|
|
1234
|
+
if isinstance(resp, dict):
|
|
1235
|
+
area = resp.get("area", "Unknown Area")
|
|
1236
|
+
tasks = resp.get("tasks", [])
|
|
1237
|
+
|
|
1238
|
+
markdown_parts.extend([f"### {area}", ""])
|
|
1239
|
+
|
|
1240
|
+
if isinstance(tasks, list):
|
|
1241
|
+
for task in tasks:
|
|
1242
|
+
markdown_parts.append(f"- {task}")
|
|
1243
|
+
|
|
1244
|
+
markdown_parts.append("")
|
|
1245
|
+
else:
|
|
1246
|
+
markdown_parts.append(f"- {resp}")
|
|
1247
|
+
|
|
1248
|
+
markdown_parts.append("")
|
|
1249
|
+
|
|
1250
|
+
# Add analytical framework
|
|
1251
|
+
if "analytical_framework" in instructions_dict:
|
|
1252
|
+
framework = instructions_dict["analytical_framework"]
|
|
1253
|
+
if isinstance(framework, dict):
|
|
1254
|
+
markdown_parts.extend(["## Analytical Framework", ""])
|
|
1255
|
+
|
|
1256
|
+
for framework_area, framework_data in framework.items():
|
|
1257
|
+
markdown_parts.extend(
|
|
1258
|
+
[f"### {framework_area.replace('_', ' ').title()}", ""]
|
|
1259
|
+
)
|
|
1260
|
+
|
|
1261
|
+
if isinstance(framework_data, dict):
|
|
1262
|
+
for category, items in framework_data.items():
|
|
1263
|
+
markdown_parts.extend(
|
|
1264
|
+
[f"#### {category.replace('_', ' ').title()}", ""]
|
|
1265
|
+
)
|
|
1266
|
+
|
|
1267
|
+
if isinstance(items, list):
|
|
1268
|
+
for item in items:
|
|
1269
|
+
markdown_parts.append(f"- {item}")
|
|
1270
|
+
elif isinstance(items, str):
|
|
1271
|
+
markdown_parts.append(items)
|
|
1272
|
+
|
|
1273
|
+
markdown_parts.append("")
|
|
1274
|
+
elif isinstance(framework_data, list):
|
|
1275
|
+
for item in framework_data:
|
|
1276
|
+
markdown_parts.append(f"- {item}")
|
|
1277
|
+
markdown_parts.append("")
|
|
1278
|
+
|
|
1279
|
+
# Add methodologies
|
|
1280
|
+
if "methodologies" in instructions_dict:
|
|
1281
|
+
methodologies = instructions_dict["methodologies"]
|
|
1282
|
+
if isinstance(methodologies, dict):
|
|
1283
|
+
markdown_parts.extend(["## Methodologies", ""])
|
|
1284
|
+
|
|
1285
|
+
for method_name, method_data in methodologies.items():
|
|
1286
|
+
markdown_parts.extend(
|
|
1287
|
+
[f"### {method_name.replace('_', ' ').title()}", ""]
|
|
1288
|
+
)
|
|
1289
|
+
|
|
1290
|
+
if isinstance(method_data, dict):
|
|
1291
|
+
for key, value in method_data.items():
|
|
1292
|
+
if isinstance(value, list):
|
|
1293
|
+
markdown_parts.extend(
|
|
1294
|
+
[f"#### {key.replace('_', ' ').title()}", ""]
|
|
1295
|
+
)
|
|
1296
|
+
for item in value:
|
|
1297
|
+
markdown_parts.append(f"- {item}")
|
|
1298
|
+
markdown_parts.append("")
|
|
1299
|
+
elif isinstance(value, str):
|
|
1300
|
+
markdown_parts.extend(
|
|
1301
|
+
[
|
|
1302
|
+
f"**{key.replace('_', ' ').title()}**: {value}",
|
|
1303
|
+
"",
|
|
1304
|
+
]
|
|
1305
|
+
)
|
|
1306
|
+
|
|
1307
|
+
# Add quality standards
|
|
1308
|
+
if "quality_standards" in instructions_dict:
|
|
1309
|
+
standards = instructions_dict["quality_standards"]
|
|
1310
|
+
if isinstance(standards, dict):
|
|
1311
|
+
markdown_parts.extend(["## Quality Standards", ""])
|
|
1312
|
+
|
|
1313
|
+
for standard_area, standard_items in standards.items():
|
|
1314
|
+
markdown_parts.extend(
|
|
1315
|
+
[f"### {standard_area.replace('_', ' ').title()}", ""]
|
|
1316
|
+
)
|
|
1317
|
+
|
|
1318
|
+
if isinstance(standard_items, list):
|
|
1319
|
+
for item in standard_items:
|
|
1320
|
+
markdown_parts.append(f"- {item}")
|
|
1321
|
+
elif isinstance(standard_items, str):
|
|
1322
|
+
markdown_parts.append(standard_items)
|
|
1323
|
+
|
|
1324
|
+
markdown_parts.append("")
|
|
1325
|
+
|
|
1326
|
+
# Add communication style
|
|
1327
|
+
if "communication_style" in instructions_dict:
|
|
1328
|
+
comm_style = instructions_dict["communication_style"]
|
|
1329
|
+
if isinstance(comm_style, dict):
|
|
1330
|
+
markdown_parts.extend(["## Communication Style", ""])
|
|
1331
|
+
|
|
1332
|
+
for style_area, style_items in comm_style.items():
|
|
1333
|
+
markdown_parts.extend(
|
|
1334
|
+
[f"### {style_area.replace('_', ' ').title()}", ""]
|
|
1335
|
+
)
|
|
1336
|
+
|
|
1337
|
+
if isinstance(style_items, list):
|
|
1338
|
+
for item in style_items:
|
|
1339
|
+
markdown_parts.append(f"- {item}")
|
|
1340
|
+
elif isinstance(style_items, str):
|
|
1341
|
+
markdown_parts.append(style_items)
|
|
1342
|
+
|
|
1343
|
+
markdown_parts.append("")
|
|
1344
|
+
|
|
1345
|
+
# If no specific sections were found, convert as generic dict
|
|
1346
|
+
if not markdown_parts:
|
|
1347
|
+
markdown_parts = ["# Agent Instructions", ""]
|
|
1348
|
+
for key, value in instructions_dict.items():
|
|
1349
|
+
key_title = key.replace("_", " ").title()
|
|
1350
|
+
if isinstance(value, str):
|
|
1351
|
+
markdown_parts.extend([f"## {key_title}", "", value, ""])
|
|
1352
|
+
elif isinstance(value, list):
|
|
1353
|
+
markdown_parts.extend([f"## {key_title}", ""])
|
|
1354
|
+
for item in value:
|
|
1355
|
+
markdown_parts.append(f"- {item}")
|
|
1356
|
+
markdown_parts.append("")
|
|
1357
|
+
elif isinstance(value, dict):
|
|
1358
|
+
markdown_parts.extend([f"## {key_title}", ""])
|
|
1359
|
+
# Simple dict formatting
|
|
1360
|
+
for subkey, subvalue in value.items():
|
|
1361
|
+
if isinstance(subvalue, str):
|
|
1362
|
+
markdown_parts.extend(
|
|
1363
|
+
[
|
|
1364
|
+
f"**{subkey.replace('_', ' ').title()}**: {subvalue}",
|
|
1365
|
+
"",
|
|
1366
|
+
]
|
|
1367
|
+
)
|
|
1368
|
+
|
|
1369
|
+
return "\n".join(markdown_parts).strip()
|