claude-mpm 4.7.4__py3-none-any.whl → 4.18.2__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- claude_mpm/VERSION +1 -1
- claude_mpm/agents/BASE_AGENT_TEMPLATE.md +118 -0
- claude_mpm/agents/BASE_ENGINEER.md +286 -0
- claude_mpm/agents/BASE_PM.md +106 -1
- claude_mpm/agents/OUTPUT_STYLE.md +329 -11
- claude_mpm/agents/PM_INSTRUCTIONS.md +397 -459
- claude_mpm/agents/agent_loader.py +17 -5
- claude_mpm/agents/frontmatter_validator.py +284 -253
- claude_mpm/agents/templates/README.md +465 -0
- claude_mpm/agents/templates/agent-manager.json +4 -1
- claude_mpm/agents/templates/agentic-coder-optimizer.json +13 -3
- claude_mpm/agents/templates/api_qa.json +11 -2
- claude_mpm/agents/templates/circuit_breakers.md +638 -0
- claude_mpm/agents/templates/clerk-ops.json +12 -2
- claude_mpm/agents/templates/code_analyzer.json +8 -2
- claude_mpm/agents/templates/content-agent.json +358 -0
- claude_mpm/agents/templates/dart_engineer.json +15 -2
- claude_mpm/agents/templates/data_engineer.json +15 -2
- claude_mpm/agents/templates/documentation.json +10 -2
- claude_mpm/agents/templates/engineer.json +21 -1
- claude_mpm/agents/templates/gcp_ops_agent.json +12 -2
- claude_mpm/agents/templates/git_file_tracking.md +584 -0
- claude_mpm/agents/templates/golang_engineer.json +270 -0
- claude_mpm/agents/templates/imagemagick.json +4 -1
- claude_mpm/agents/templates/java_engineer.json +346 -0
- claude_mpm/agents/templates/local_ops_agent.json +1227 -6
- claude_mpm/agents/templates/memory_manager.json +4 -1
- claude_mpm/agents/templates/nextjs_engineer.json +141 -133
- claude_mpm/agents/templates/ops.json +12 -2
- claude_mpm/agents/templates/php-engineer.json +270 -174
- claude_mpm/agents/templates/pm_examples.md +474 -0
- claude_mpm/agents/templates/pm_red_flags.md +240 -0
- claude_mpm/agents/templates/product_owner.json +338 -0
- claude_mpm/agents/templates/project_organizer.json +14 -4
- claude_mpm/agents/templates/prompt-engineer.json +13 -2
- claude_mpm/agents/templates/python_engineer.json +174 -81
- claude_mpm/agents/templates/qa.json +11 -2
- claude_mpm/agents/templates/react_engineer.json +16 -3
- claude_mpm/agents/templates/refactoring_engineer.json +12 -2
- claude_mpm/agents/templates/research.json +34 -21
- claude_mpm/agents/templates/response_format.md +583 -0
- claude_mpm/agents/templates/ruby-engineer.json +129 -192
- claude_mpm/agents/templates/rust_engineer.json +270 -0
- claude_mpm/agents/templates/security.json +10 -2
- claude_mpm/agents/templates/svelte-engineer.json +225 -0
- claude_mpm/agents/templates/ticketing.json +10 -2
- claude_mpm/agents/templates/typescript_engineer.json +116 -125
- claude_mpm/agents/templates/validation_templates.md +312 -0
- claude_mpm/agents/templates/vercel_ops_agent.json +12 -2
- claude_mpm/agents/templates/version_control.json +12 -2
- claude_mpm/agents/templates/web_qa.json +11 -2
- claude_mpm/agents/templates/web_ui.json +15 -2
- claude_mpm/cli/__init__.py +34 -614
- claude_mpm/cli/commands/agent_manager.py +25 -12
- claude_mpm/cli/commands/agent_state_manager.py +186 -0
- claude_mpm/cli/commands/agents.py +235 -148
- claude_mpm/cli/commands/agents_detect.py +380 -0
- claude_mpm/cli/commands/agents_recommend.py +309 -0
- claude_mpm/cli/commands/aggregate.py +7 -3
- claude_mpm/cli/commands/analyze.py +9 -4
- claude_mpm/cli/commands/analyze_code.py +7 -2
- claude_mpm/cli/commands/auto_configure.py +570 -0
- claude_mpm/cli/commands/config.py +47 -13
- claude_mpm/cli/commands/configure.py +419 -1571
- claude_mpm/cli/commands/configure_agent_display.py +261 -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 +167 -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/local_deploy.py +537 -0
- claude_mpm/cli/commands/memory.py +54 -20
- claude_mpm/cli/commands/mpm_init.py +585 -196
- claude_mpm/cli/commands/mpm_init_handler.py +37 -3
- claude_mpm/cli/commands/search.py +170 -4
- claude_mpm/cli/commands/upgrade.py +152 -0
- claude_mpm/cli/executor.py +202 -0
- claude_mpm/cli/helpers.py +105 -0
- claude_mpm/cli/interactive/__init__.py +3 -0
- claude_mpm/cli/interactive/skills_wizard.py +491 -0
- claude_mpm/cli/parsers/__init__.py +7 -1
- claude_mpm/cli/parsers/agents_parser.py +9 -0
- claude_mpm/cli/parsers/auto_configure_parser.py +245 -0
- claude_mpm/cli/parsers/base_parser.py +110 -3
- claude_mpm/cli/parsers/local_deploy_parser.py +227 -0
- claude_mpm/cli/parsers/mpm_init_parser.py +65 -5
- claude_mpm/cli/shared/output_formatters.py +28 -19
- claude_mpm/cli/startup.py +481 -0
- claude_mpm/cli/utils.py +52 -1
- claude_mpm/commands/mpm-agents-detect.md +168 -0
- claude_mpm/commands/mpm-agents-recommend.md +214 -0
- claude_mpm/commands/mpm-agents.md +75 -1
- claude_mpm/commands/mpm-auto-configure.md +217 -0
- claude_mpm/commands/mpm-help.md +163 -0
- claude_mpm/commands/mpm-init.md +148 -3
- claude_mpm/commands/mpm-version.md +113 -0
- claude_mpm/commands/mpm.md +1 -0
- claude_mpm/config/agent_config.py +2 -2
- claude_mpm/config/model_config.py +428 -0
- claude_mpm/constants.py +1 -0
- claude_mpm/core/base_service.py +13 -12
- claude_mpm/core/enums.py +452 -0
- claude_mpm/core/factories.py +1 -1
- claude_mpm/core/instruction_reinforcement_hook.py +2 -1
- claude_mpm/core/interactive_session.py +9 -3
- claude_mpm/core/log_manager.py +2 -0
- claude_mpm/core/logging_config.py +6 -2
- claude_mpm/core/oneshot_session.py +8 -4
- claude_mpm/core/optimized_agent_loader.py +3 -3
- claude_mpm/core/output_style_manager.py +12 -192
- claude_mpm/core/service_registry.py +5 -1
- claude_mpm/core/types.py +2 -9
- claude_mpm/core/typing_utils.py +7 -6
- claude_mpm/dashboard/static/js/dashboard.js +0 -14
- claude_mpm/dashboard/templates/index.html +3 -41
- claude_mpm/hooks/__init__.py +20 -0
- claude_mpm/hooks/claude_hooks/event_handlers.py +4 -2
- claude_mpm/hooks/claude_hooks/response_tracking.py +35 -1
- claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +23 -2
- claude_mpm/hooks/failure_learning/__init__.py +60 -0
- claude_mpm/hooks/failure_learning/failure_detection_hook.py +235 -0
- claude_mpm/hooks/failure_learning/fix_detection_hook.py +217 -0
- claude_mpm/hooks/failure_learning/learning_extraction_hook.py +286 -0
- claude_mpm/hooks/instruction_reinforcement.py +7 -2
- claude_mpm/hooks/kuzu_enrichment_hook.py +263 -0
- claude_mpm/hooks/kuzu_memory_hook.py +37 -12
- claude_mpm/hooks/kuzu_response_hook.py +183 -0
- claude_mpm/models/resume_log.py +340 -0
- claude_mpm/services/agents/__init__.py +18 -5
- claude_mpm/services/agents/auto_config_manager.py +796 -0
- claude_mpm/services/agents/deployment/agent_configuration_manager.py +1 -1
- claude_mpm/services/agents/deployment/agent_record_service.py +1 -1
- claude_mpm/services/agents/deployment/agent_validator.py +17 -1
- claude_mpm/services/agents/deployment/async_agent_deployment.py +1 -1
- claude_mpm/services/agents/deployment/interface_adapter.py +3 -2
- claude_mpm/services/agents/deployment/local_template_deployment.py +1 -1
- claude_mpm/services/agents/deployment/pipeline/steps/agent_processing_step.py +7 -6
- claude_mpm/services/agents/deployment/pipeline/steps/base_step.py +7 -16
- claude_mpm/services/agents/deployment/pipeline/steps/configuration_step.py +4 -3
- claude_mpm/services/agents/deployment/pipeline/steps/target_directory_step.py +5 -3
- claude_mpm/services/agents/deployment/pipeline/steps/validation_step.py +6 -5
- claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +9 -6
- claude_mpm/services/agents/deployment/validation/__init__.py +3 -1
- claude_mpm/services/agents/deployment/validation/validation_result.py +1 -9
- claude_mpm/services/agents/local_template_manager.py +1 -1
- claude_mpm/services/agents/memory/agent_memory_manager.py +5 -2
- claude_mpm/services/agents/observers.py +547 -0
- claude_mpm/services/agents/recommender.py +568 -0
- claude_mpm/services/agents/registry/modification_tracker.py +5 -2
- claude_mpm/services/command_handler_service.py +11 -5
- claude_mpm/services/core/__init__.py +33 -1
- claude_mpm/services/core/interfaces/__init__.py +90 -3
- claude_mpm/services/core/interfaces/agent.py +184 -0
- claude_mpm/services/core/interfaces/health.py +172 -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/stability.py +260 -0
- claude_mpm/services/core/memory_manager.py +11 -24
- claude_mpm/services/core/models/__init__.py +79 -0
- claude_mpm/services/core/models/agent_config.py +381 -0
- claude_mpm/services/core/models/health.py +162 -0
- claude_mpm/services/core/models/process.py +235 -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 +23 -7
- claude_mpm/services/diagnostics/__init__.py +2 -2
- claude_mpm/services/diagnostics/checks/agent_check.py +25 -24
- claude_mpm/services/diagnostics/checks/claude_code_check.py +24 -23
- claude_mpm/services/diagnostics/checks/common_issues_check.py +25 -24
- claude_mpm/services/diagnostics/checks/configuration_check.py +24 -23
- claude_mpm/services/diagnostics/checks/filesystem_check.py +18 -17
- claude_mpm/services/diagnostics/checks/installation_check.py +30 -29
- claude_mpm/services/diagnostics/checks/instructions_check.py +20 -19
- claude_mpm/services/diagnostics/checks/mcp_check.py +50 -36
- claude_mpm/services/diagnostics/checks/mcp_services_check.py +38 -33
- claude_mpm/services/diagnostics/checks/monitor_check.py +23 -22
- claude_mpm/services/diagnostics/checks/startup_log_check.py +9 -8
- claude_mpm/services/diagnostics/diagnostic_runner.py +6 -5
- claude_mpm/services/diagnostics/doctor_reporter.py +28 -25
- claude_mpm/services/diagnostics/models.py +19 -24
- claude_mpm/services/infrastructure/monitoring/__init__.py +1 -1
- claude_mpm/services/infrastructure/monitoring/aggregator.py +12 -12
- claude_mpm/services/infrastructure/monitoring/base.py +5 -13
- claude_mpm/services/infrastructure/monitoring/network.py +7 -6
- claude_mpm/services/infrastructure/monitoring/process.py +13 -12
- claude_mpm/services/infrastructure/monitoring/resources.py +7 -6
- claude_mpm/services/infrastructure/monitoring/service.py +16 -15
- claude_mpm/services/infrastructure/resume_log_generator.py +439 -0
- claude_mpm/services/local_ops/__init__.py +163 -0
- claude_mpm/services/local_ops/crash_detector.py +257 -0
- claude_mpm/services/local_ops/health_checks/__init__.py +28 -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 +430 -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 +9 -4
- claude_mpm/services/mcp_gateway/core/__init__.py +1 -2
- claude_mpm/services/mcp_gateway/core/base.py +18 -31
- claude_mpm/services/mcp_gateway/main.py +30 -0
- claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +206 -32
- claude_mpm/services/mcp_gateway/tools/health_check_tool.py +30 -28
- claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +25 -5
- claude_mpm/services/mcp_service_verifier.py +1 -1
- claude_mpm/services/memory/failure_tracker.py +563 -0
- claude_mpm/services/memory_hook_service.py +165 -4
- 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 +453 -0
- claude_mpm/services/model/ollama_provider.py +415 -0
- claude_mpm/services/monitor/daemon_manager.py +3 -2
- claude_mpm/services/monitor/handlers/dashboard.py +2 -1
- claude_mpm/services/monitor/handlers/hooks.py +2 -1
- claude_mpm/services/monitor/management/lifecycle.py +3 -2
- claude_mpm/services/monitor/server.py +2 -1
- claude_mpm/services/project/__init__.py +23 -0
- claude_mpm/services/project/detection_strategies.py +719 -0
- claude_mpm/services/project/toolchain_analyzer.py +581 -0
- claude_mpm/services/self_upgrade_service.py +342 -0
- claude_mpm/services/session_management_service.py +3 -2
- claude_mpm/services/session_manager.py +205 -1
- claude_mpm/services/shared/async_service_base.py +16 -27
- claude_mpm/services/shared/lifecycle_service_base.py +1 -14
- claude_mpm/services/socketio/handlers/__init__.py +5 -2
- claude_mpm/services/socketio/handlers/hook.py +13 -2
- claude_mpm/services/socketio/handlers/registry.py +4 -2
- claude_mpm/services/socketio/server/main.py +10 -8
- claude_mpm/services/subprocess_launcher_service.py +14 -5
- claude_mpm/services/unified/analyzer_strategies/code_analyzer.py +8 -7
- claude_mpm/services/unified/analyzer_strategies/dependency_analyzer.py +6 -5
- claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +8 -7
- claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +7 -6
- claude_mpm/services/unified/analyzer_strategies/structure_analyzer.py +5 -4
- claude_mpm/services/unified/config_strategies/validation_strategy.py +13 -9
- claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +10 -3
- claude_mpm/services/unified/deployment_strategies/local.py +6 -5
- claude_mpm/services/unified/deployment_strategies/utils.py +6 -5
- claude_mpm/services/unified/deployment_strategies/vercel.py +7 -6
- claude_mpm/services/unified/interfaces.py +3 -1
- claude_mpm/services/unified/unified_analyzer.py +14 -10
- claude_mpm/services/unified/unified_config.py +2 -1
- claude_mpm/services/unified/unified_deployment.py +9 -4
- claude_mpm/services/version_service.py +104 -1
- claude_mpm/skills/__init__.py +21 -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/json-data-handling.md +223 -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 +567 -0
- claude_mpm/skills/bundled/refactoring-patterns.md +180 -0
- claude_mpm/skills/bundled/security-scanning.md +327 -0
- claude_mpm/skills/bundled/systematic-debugging.md +473 -0
- claude_mpm/skills/bundled/test-driven-development.md +378 -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 +310 -0
- claude_mpm/storage/state_storage.py +15 -15
- claude_mpm/tools/code_tree_analyzer.py +177 -141
- claude_mpm/tools/code_tree_events.py +4 -2
- claude_mpm/utils/agent_dependency_loader.py +40 -20
- claude_mpm/utils/display_helper.py +260 -0
- claude_mpm/utils/git_analyzer.py +407 -0
- claude_mpm/utils/robust_installer.py +73 -19
- {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/METADATA +129 -12
- {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/RECORD +295 -193
- claude_mpm/dashboard/static/css/code-tree.css +0 -1639
- claude_mpm/dashboard/static/index-hub-backup.html +0 -713
- claude_mpm/dashboard/static/js/components/code-tree/tree-breadcrumb.js +0 -353
- claude_mpm/dashboard/static/js/components/code-tree/tree-constants.js +0 -235
- claude_mpm/dashboard/static/js/components/code-tree/tree-search.js +0 -409
- claude_mpm/dashboard/static/js/components/code-tree/tree-utils.js +0 -435
- claude_mpm/dashboard/static/js/components/code-tree.js +0 -5869
- claude_mpm/dashboard/static/js/components/code-viewer.js +0 -1386
- claude_mpm/hooks/claude_hooks/hook_handler_eventbus.py +0 -425
- claude_mpm/hooks/claude_hooks/hook_handler_original.py +0 -1041
- claude_mpm/hooks/claude_hooks/hook_handler_refactored.py +0 -347
- claude_mpm/services/agents/deployment/agent_lifecycle_manager_refactored.py +0 -575
- claude_mpm/services/project/analyzer_refactored.py +0 -450
- {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/WHEEL +0 -0
- {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/entry_points.txt +0 -0
- {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,310 @@
|
|
|
1
|
+
"""Skills manager - integrates skills with agents."""
|
|
2
|
+
|
|
3
|
+
import json
|
|
4
|
+
from pathlib import Path
|
|
5
|
+
from typing import Dict, List, Optional
|
|
6
|
+
|
|
7
|
+
from claude_mpm.core.logging_utils import get_logger
|
|
8
|
+
|
|
9
|
+
from .registry import Skill, get_registry
|
|
10
|
+
|
|
11
|
+
logger = get_logger(__name__)
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class SkillManager:
|
|
15
|
+
"""Manages skills and their integration with agents."""
|
|
16
|
+
|
|
17
|
+
def __init__(self):
|
|
18
|
+
"""Initialize the skill manager."""
|
|
19
|
+
self.registry = get_registry()
|
|
20
|
+
self.agent_skill_mapping: Dict[str, List[str]] = {}
|
|
21
|
+
self._load_agent_mappings()
|
|
22
|
+
|
|
23
|
+
def _load_agent_mappings(self):
|
|
24
|
+
"""Load skill mappings from agent templates."""
|
|
25
|
+
# Load mappings from agent JSON templates that have 'skills' field
|
|
26
|
+
agent_templates_dir = Path(__file__).parent.parent / "agents" / "templates"
|
|
27
|
+
|
|
28
|
+
if not agent_templates_dir.exists():
|
|
29
|
+
logger.warning(
|
|
30
|
+
f"Agent templates directory not found: {agent_templates_dir}"
|
|
31
|
+
)
|
|
32
|
+
return
|
|
33
|
+
|
|
34
|
+
mapping_count = 0
|
|
35
|
+
for template_file in agent_templates_dir.glob("*.json"):
|
|
36
|
+
try:
|
|
37
|
+
with open(template_file, encoding="utf-8") as f:
|
|
38
|
+
agent_data = json.load(f)
|
|
39
|
+
|
|
40
|
+
agent_id = agent_data.get("agent_id") or agent_data.get("agent_type")
|
|
41
|
+
if not agent_id:
|
|
42
|
+
continue
|
|
43
|
+
|
|
44
|
+
# Extract skills list if present
|
|
45
|
+
skills = agent_data.get("skills", [])
|
|
46
|
+
if skills:
|
|
47
|
+
self.agent_skill_mapping[agent_id] = skills
|
|
48
|
+
mapping_count += 1
|
|
49
|
+
logger.debug(
|
|
50
|
+
f"Agent '{agent_id}' mapped to {len(skills)} skills: {', '.join(skills)}"
|
|
51
|
+
)
|
|
52
|
+
|
|
53
|
+
except Exception as e:
|
|
54
|
+
logger.error(f"Error loading agent mapping from {template_file}: {e}")
|
|
55
|
+
|
|
56
|
+
if mapping_count > 0:
|
|
57
|
+
logger.info(f"Loaded skill mappings for {mapping_count} agents")
|
|
58
|
+
|
|
59
|
+
def get_agent_skills(self, agent_type: str) -> List[Skill]:
|
|
60
|
+
"""
|
|
61
|
+
Get all skills for an agent (bundled + discovered).
|
|
62
|
+
|
|
63
|
+
Args:
|
|
64
|
+
agent_type: Agent type/ID (e.g., 'engineer', 'python_engineer')
|
|
65
|
+
|
|
66
|
+
Returns:
|
|
67
|
+
List of Skill objects for this agent
|
|
68
|
+
"""
|
|
69
|
+
skill_names = self.agent_skill_mapping.get(agent_type, [])
|
|
70
|
+
|
|
71
|
+
# Get skills from registry
|
|
72
|
+
skills = []
|
|
73
|
+
for name in skill_names:
|
|
74
|
+
skill = self.registry.get_skill(name)
|
|
75
|
+
if skill:
|
|
76
|
+
skills.append(skill)
|
|
77
|
+
else:
|
|
78
|
+
logger.warning(
|
|
79
|
+
f"Skill '{name}' referenced by agent '{agent_type}' not found"
|
|
80
|
+
)
|
|
81
|
+
|
|
82
|
+
# Also include skills that have no agent restriction
|
|
83
|
+
# or explicitly list this agent type
|
|
84
|
+
additional_skills = self.registry.get_skills_for_agent(agent_type)
|
|
85
|
+
for skill in additional_skills:
|
|
86
|
+
if skill not in skills:
|
|
87
|
+
skills.append(skill)
|
|
88
|
+
|
|
89
|
+
return skills
|
|
90
|
+
|
|
91
|
+
def enhance_agent_prompt(
|
|
92
|
+
self, agent_type: str, base_prompt: str, include_all: bool = False
|
|
93
|
+
) -> str:
|
|
94
|
+
"""
|
|
95
|
+
Enhance agent prompt with available skills.
|
|
96
|
+
|
|
97
|
+
Args:
|
|
98
|
+
agent_type: Agent type/ID
|
|
99
|
+
base_prompt: Original agent prompt
|
|
100
|
+
include_all: If True, include all available skills regardless of mapping
|
|
101
|
+
|
|
102
|
+
Returns:
|
|
103
|
+
Enhanced prompt with skills section appended
|
|
104
|
+
"""
|
|
105
|
+
if include_all:
|
|
106
|
+
skills = self.registry.list_skills()
|
|
107
|
+
else:
|
|
108
|
+
skills = self.get_agent_skills(agent_type)
|
|
109
|
+
|
|
110
|
+
if not skills:
|
|
111
|
+
return base_prompt
|
|
112
|
+
|
|
113
|
+
# Build skills section
|
|
114
|
+
skills_section = "\n\n" + "=" * 80 + "\n"
|
|
115
|
+
skills_section += "## 🎯 Available Skills\n\n"
|
|
116
|
+
skills_section += f"You have access to {len(skills)} specialized skills:\n\n"
|
|
117
|
+
|
|
118
|
+
for skill in skills:
|
|
119
|
+
skills_section += f"### 📚 {skill.name.replace('-', ' ').title()}\n\n"
|
|
120
|
+
skills_section += f"**Source:** {skill.source}\n"
|
|
121
|
+
if skill.description:
|
|
122
|
+
skills_section += f"**Description:** {skill.description}\n"
|
|
123
|
+
skills_section += "\n```\n"
|
|
124
|
+
skills_section += skill.content
|
|
125
|
+
skills_section += "\n```\n\n"
|
|
126
|
+
|
|
127
|
+
skills_section += "=" * 80 + "\n"
|
|
128
|
+
|
|
129
|
+
return base_prompt + skills_section
|
|
130
|
+
|
|
131
|
+
def list_agent_skill_mappings(self) -> Dict[str, List[str]]:
|
|
132
|
+
"""
|
|
133
|
+
Get all agent-to-skill mappings.
|
|
134
|
+
|
|
135
|
+
Returns:
|
|
136
|
+
Dictionary mapping agent IDs to lists of skill names
|
|
137
|
+
"""
|
|
138
|
+
return self.agent_skill_mapping.copy()
|
|
139
|
+
|
|
140
|
+
def add_skill_to_agent(self, agent_type: str, skill_name: str) -> bool:
|
|
141
|
+
"""
|
|
142
|
+
Add a skill to an agent's mapping.
|
|
143
|
+
|
|
144
|
+
Args:
|
|
145
|
+
agent_type: Agent type/ID
|
|
146
|
+
skill_name: Name of the skill to add
|
|
147
|
+
|
|
148
|
+
Returns:
|
|
149
|
+
True if successful, False if skill not found
|
|
150
|
+
"""
|
|
151
|
+
skill = self.registry.get_skill(skill_name)
|
|
152
|
+
if not skill:
|
|
153
|
+
logger.error(f"Cannot add skill '{skill_name}': skill not found")
|
|
154
|
+
return False
|
|
155
|
+
|
|
156
|
+
if agent_type not in self.agent_skill_mapping:
|
|
157
|
+
self.agent_skill_mapping[agent_type] = []
|
|
158
|
+
|
|
159
|
+
if skill_name not in self.agent_skill_mapping[agent_type]:
|
|
160
|
+
self.agent_skill_mapping[agent_type].append(skill_name)
|
|
161
|
+
logger.info(f"Added skill '{skill_name}' to agent '{agent_type}'")
|
|
162
|
+
|
|
163
|
+
return True
|
|
164
|
+
|
|
165
|
+
def remove_skill_from_agent(self, agent_type: str, skill_name: str) -> bool:
|
|
166
|
+
"""
|
|
167
|
+
Remove a skill from an agent's mapping.
|
|
168
|
+
|
|
169
|
+
Args:
|
|
170
|
+
agent_type: Agent type/ID
|
|
171
|
+
skill_name: Name of the skill to remove
|
|
172
|
+
|
|
173
|
+
Returns:
|
|
174
|
+
True if successful, False if not found
|
|
175
|
+
"""
|
|
176
|
+
if agent_type not in self.agent_skill_mapping:
|
|
177
|
+
return False
|
|
178
|
+
|
|
179
|
+
if skill_name in self.agent_skill_mapping[agent_type]:
|
|
180
|
+
self.agent_skill_mapping[agent_type].remove(skill_name)
|
|
181
|
+
logger.info(f"Removed skill '{skill_name}' from agent '{agent_type}'")
|
|
182
|
+
return True
|
|
183
|
+
|
|
184
|
+
return False
|
|
185
|
+
|
|
186
|
+
def reload(self):
|
|
187
|
+
"""Reload skills and agent mappings."""
|
|
188
|
+
logger.info("Reloading skill manager...")
|
|
189
|
+
self.registry.reload()
|
|
190
|
+
self.agent_skill_mapping.clear()
|
|
191
|
+
self._load_agent_mappings()
|
|
192
|
+
logger.info("Skill manager reloaded")
|
|
193
|
+
|
|
194
|
+
def infer_agents_for_skill(self, skill) -> List[str]:
|
|
195
|
+
"""Infer which agents should have this skill based on tags/name.
|
|
196
|
+
|
|
197
|
+
Args:
|
|
198
|
+
skill: Skill object to analyze
|
|
199
|
+
|
|
200
|
+
Returns:
|
|
201
|
+
List of agent IDs that should have this skill
|
|
202
|
+
"""
|
|
203
|
+
agents = []
|
|
204
|
+
content_lower = skill.content.lower()
|
|
205
|
+
name_lower = skill.name.lower()
|
|
206
|
+
|
|
207
|
+
# Python-related
|
|
208
|
+
if any(
|
|
209
|
+
tag in content_lower or tag in name_lower
|
|
210
|
+
for tag in ["python", "django", "flask", "fastapi"]
|
|
211
|
+
):
|
|
212
|
+
agents.append("python-engineer")
|
|
213
|
+
|
|
214
|
+
# TypeScript/JavaScript-related
|
|
215
|
+
if any(
|
|
216
|
+
tag in content_lower or tag in name_lower
|
|
217
|
+
for tag in ["typescript", "javascript", "react", "next", "vue", "node"]
|
|
218
|
+
):
|
|
219
|
+
agents.extend(["typescript-engineer", "react-engineer", "nextjs-engineer"])
|
|
220
|
+
|
|
221
|
+
# Go-related
|
|
222
|
+
if any(tag in content_lower or tag in name_lower for tag in ["golang", "go "]):
|
|
223
|
+
agents.append("golang-engineer")
|
|
224
|
+
|
|
225
|
+
# Ops-related
|
|
226
|
+
if any(
|
|
227
|
+
tag in content_lower or tag in name_lower
|
|
228
|
+
for tag in ["docker", "kubernetes", "deploy", "devops", "ops"]
|
|
229
|
+
):
|
|
230
|
+
agents.extend(["ops", "devops", "local-ops"])
|
|
231
|
+
|
|
232
|
+
# Testing/QA-related
|
|
233
|
+
if any(
|
|
234
|
+
tag in content_lower or tag in name_lower
|
|
235
|
+
for tag in ["test", "qa", "quality", "assert"]
|
|
236
|
+
):
|
|
237
|
+
agents.extend(["qa", "web-qa", "api-qa"])
|
|
238
|
+
|
|
239
|
+
# Documentation-related
|
|
240
|
+
if any(
|
|
241
|
+
tag in content_lower or tag in name_lower
|
|
242
|
+
for tag in ["documentation", "docs", "api doc", "openapi"]
|
|
243
|
+
):
|
|
244
|
+
agents.extend(["docs", "documentation", "technical-writer"])
|
|
245
|
+
|
|
246
|
+
# Remove duplicates
|
|
247
|
+
return list(set(agents))
|
|
248
|
+
|
|
249
|
+
def save_mappings_to_config(self, config_path: Optional[Path] = None):
|
|
250
|
+
"""Save current agent-skill mappings to configuration file.
|
|
251
|
+
|
|
252
|
+
Args:
|
|
253
|
+
config_path: Path to configuration file. If None, uses default.
|
|
254
|
+
"""
|
|
255
|
+
import json
|
|
256
|
+
|
|
257
|
+
if config_path is None:
|
|
258
|
+
config_path = Path.cwd() / ".claude-mpm" / "skills_config.json"
|
|
259
|
+
|
|
260
|
+
# Ensure directory exists
|
|
261
|
+
config_path.parent.mkdir(parents=True, exist_ok=True)
|
|
262
|
+
|
|
263
|
+
# Save mappings
|
|
264
|
+
with open(config_path, "w", encoding="utf-8") as f:
|
|
265
|
+
json.dump(self.agent_skill_mapping, f, indent=2)
|
|
266
|
+
|
|
267
|
+
logger.info(f"Saved skill mappings to {config_path}")
|
|
268
|
+
|
|
269
|
+
def load_mappings_from_config(self, config_path: Optional[Path] = None):
|
|
270
|
+
"""Load agent-skill mappings from configuration file.
|
|
271
|
+
|
|
272
|
+
Args:
|
|
273
|
+
config_path: Path to configuration file. If None, uses default.
|
|
274
|
+
"""
|
|
275
|
+
import json
|
|
276
|
+
|
|
277
|
+
if config_path is None:
|
|
278
|
+
config_path = Path.cwd() / ".claude-mpm" / "skills_config.json"
|
|
279
|
+
|
|
280
|
+
if not config_path.exists():
|
|
281
|
+
logger.debug(f"No skill mappings config found at {config_path}")
|
|
282
|
+
return
|
|
283
|
+
|
|
284
|
+
try:
|
|
285
|
+
with open(config_path, encoding="utf-8") as f:
|
|
286
|
+
loaded_mappings = json.load(f)
|
|
287
|
+
|
|
288
|
+
# Merge with existing mappings
|
|
289
|
+
for agent_id, skills in loaded_mappings.items():
|
|
290
|
+
if agent_id not in self.agent_skill_mapping:
|
|
291
|
+
self.agent_skill_mapping[agent_id] = []
|
|
292
|
+
for skill in skills:
|
|
293
|
+
if skill not in self.agent_skill_mapping[agent_id]:
|
|
294
|
+
self.agent_skill_mapping[agent_id].append(skill)
|
|
295
|
+
|
|
296
|
+
logger.info(f"Loaded skill mappings from {config_path}")
|
|
297
|
+
except Exception as e:
|
|
298
|
+
logger.error(f"Error loading skill mappings from {config_path}: {e}")
|
|
299
|
+
|
|
300
|
+
|
|
301
|
+
# Global manager instance (singleton pattern)
|
|
302
|
+
_manager: Optional[SkillManager] = None
|
|
303
|
+
|
|
304
|
+
|
|
305
|
+
def get_manager() -> SkillManager:
|
|
306
|
+
"""Get the global skill manager (singleton)."""
|
|
307
|
+
global _manager
|
|
308
|
+
if _manager is None:
|
|
309
|
+
_manager = SkillManager()
|
|
310
|
+
return _manager
|
|
@@ -86,7 +86,7 @@ class StateStorage:
|
|
|
86
86
|
return True
|
|
87
87
|
|
|
88
88
|
except Exception as e:
|
|
89
|
-
|
|
89
|
+
logger.error(f"Failed to write JSON to {file_path}: {e}")
|
|
90
90
|
self.error_count += 1
|
|
91
91
|
return False
|
|
92
92
|
|
|
@@ -106,7 +106,7 @@ class StateStorage:
|
|
|
106
106
|
file_path = Path(file_path)
|
|
107
107
|
|
|
108
108
|
if not file_path.exists():
|
|
109
|
-
|
|
109
|
+
logger.debug(f"File not found: {file_path}")
|
|
110
110
|
return None
|
|
111
111
|
|
|
112
112
|
# Auto-detect compression
|
|
@@ -125,7 +125,7 @@ class StateStorage:
|
|
|
125
125
|
return data
|
|
126
126
|
|
|
127
127
|
except Exception as e:
|
|
128
|
-
|
|
128
|
+
logger.error(f"Failed to read JSON from {file_path}: {e}")
|
|
129
129
|
self.error_count += 1
|
|
130
130
|
return None
|
|
131
131
|
|
|
@@ -166,7 +166,7 @@ class StateStorage:
|
|
|
166
166
|
return True
|
|
167
167
|
|
|
168
168
|
except Exception as e:
|
|
169
|
-
|
|
169
|
+
logger.error(f"Failed to write pickle to {file_path}: {e}")
|
|
170
170
|
self.error_count += 1
|
|
171
171
|
return False
|
|
172
172
|
|
|
@@ -186,7 +186,7 @@ class StateStorage:
|
|
|
186
186
|
file_path = Path(file_path)
|
|
187
187
|
|
|
188
188
|
if not file_path.exists():
|
|
189
|
-
|
|
189
|
+
logger.debug(f"File not found: {file_path}")
|
|
190
190
|
return None
|
|
191
191
|
|
|
192
192
|
# Auto-detect compression
|
|
@@ -205,7 +205,7 @@ class StateStorage:
|
|
|
205
205
|
return data
|
|
206
206
|
|
|
207
207
|
except Exception as e:
|
|
208
|
-
|
|
208
|
+
logger.error(f"Failed to read pickle from {file_path}: {e}")
|
|
209
209
|
self.error_count += 1
|
|
210
210
|
return None
|
|
211
211
|
|
|
@@ -262,7 +262,7 @@ class StateStorage:
|
|
|
262
262
|
Path(temp_path).replace(file_path)
|
|
263
263
|
|
|
264
264
|
self.write_count += 1
|
|
265
|
-
|
|
265
|
+
logger.debug(f"Atomic write successful: {file_path}")
|
|
266
266
|
return True
|
|
267
267
|
|
|
268
268
|
finally:
|
|
@@ -271,7 +271,7 @@ class StateStorage:
|
|
|
271
271
|
Path(temp_path).unlink()
|
|
272
272
|
|
|
273
273
|
except Exception as e:
|
|
274
|
-
|
|
274
|
+
logger.error(f"Atomic write failed for {file_path}: {e}")
|
|
275
275
|
self.error_count += 1
|
|
276
276
|
return False
|
|
277
277
|
|
|
@@ -339,7 +339,7 @@ class StateStorage:
|
|
|
339
339
|
f.write(checksum)
|
|
340
340
|
|
|
341
341
|
except Exception as e:
|
|
342
|
-
|
|
342
|
+
logger.warning(f"Could not add checksum: {e}")
|
|
343
343
|
|
|
344
344
|
def verify_checksum(self, file_path: Union[str, Path]) -> bool:
|
|
345
345
|
"""Verify file checksum for integrity.
|
|
@@ -370,13 +370,13 @@ class StateStorage:
|
|
|
370
370
|
actual = hasher.hexdigest()
|
|
371
371
|
|
|
372
372
|
if actual != expected:
|
|
373
|
-
|
|
373
|
+
logger.error(f"Checksum mismatch for {file_path}")
|
|
374
374
|
return False
|
|
375
375
|
|
|
376
376
|
return True
|
|
377
377
|
|
|
378
378
|
except Exception as e:
|
|
379
|
-
|
|
379
|
+
logger.warning(f"Could not verify checksum: {e}")
|
|
380
380
|
return True # Assume valid if can't verify
|
|
381
381
|
|
|
382
382
|
def cleanup_temp_files(self) -> int:
|
|
@@ -396,7 +396,7 @@ class StateStorage:
|
|
|
396
396
|
if age > 3600:
|
|
397
397
|
temp_file.unlink()
|
|
398
398
|
cleaned += 1
|
|
399
|
-
|
|
399
|
+
logger.debug(f"Cleaned up temp file: {temp_file}")
|
|
400
400
|
except Exception:
|
|
401
401
|
pass
|
|
402
402
|
|
|
@@ -408,12 +408,12 @@ class StateStorage:
|
|
|
408
408
|
cleaned += 1
|
|
409
409
|
|
|
410
410
|
if cleaned > 0:
|
|
411
|
-
|
|
411
|
+
logger.info(f"Cleaned up {cleaned} temporary files")
|
|
412
412
|
|
|
413
413
|
return cleaned
|
|
414
414
|
|
|
415
415
|
except Exception as e:
|
|
416
|
-
|
|
416
|
+
logger.error(f"Error cleaning up temp files: {e}")
|
|
417
417
|
return 0
|
|
418
418
|
|
|
419
419
|
def get_storage_info(self) -> Dict[str, Any]:
|
|
@@ -447,7 +447,7 @@ class StateStorage:
|
|
|
447
447
|
}
|
|
448
448
|
|
|
449
449
|
except Exception as e:
|
|
450
|
-
|
|
450
|
+
logger.error(f"Error getting storage info: {e}")
|
|
451
451
|
return {"storage_directory": str(self.storage_dir), "error": str(e)}
|
|
452
452
|
|
|
453
453
|
|