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,380 @@
|
|
|
1
|
+
"""
|
|
2
|
+
Agents Detect CLI Command for Claude MPM Framework
|
|
3
|
+
===================================================
|
|
4
|
+
|
|
5
|
+
WHY: This module provides a CLI interface for detecting project toolchain
|
|
6
|
+
without making any changes. Useful for debugging and verification of the
|
|
7
|
+
toolchain detection system.
|
|
8
|
+
|
|
9
|
+
DESIGN DECISION: Focused solely on detection and display, with no side effects.
|
|
10
|
+
Supports multiple output formats for different use cases (human-readable,
|
|
11
|
+
JSON for scripting).
|
|
12
|
+
|
|
13
|
+
Part of TSK-0054: Auto-Configuration Feature - Phase 5
|
|
14
|
+
"""
|
|
15
|
+
|
|
16
|
+
import json
|
|
17
|
+
from pathlib import Path
|
|
18
|
+
from typing import Optional
|
|
19
|
+
|
|
20
|
+
try:
|
|
21
|
+
from rich.console import Console
|
|
22
|
+
from rich.table import Table
|
|
23
|
+
|
|
24
|
+
RICH_AVAILABLE = True
|
|
25
|
+
except ImportError:
|
|
26
|
+
RICH_AVAILABLE = False
|
|
27
|
+
|
|
28
|
+
from ...services.project.toolchain_analyzer import ToolchainAnalyzerService
|
|
29
|
+
from ..shared import BaseCommand, CommandResult
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
class AgentsDetectCommand(BaseCommand):
|
|
33
|
+
"""
|
|
34
|
+
Handle agents detect CLI command.
|
|
35
|
+
|
|
36
|
+
This command analyzes the project to detect languages, frameworks,
|
|
37
|
+
and deployment targets without making any configuration changes.
|
|
38
|
+
"""
|
|
39
|
+
|
|
40
|
+
def __init__(self):
|
|
41
|
+
"""Initialize the agents detect command."""
|
|
42
|
+
super().__init__("agents-detect")
|
|
43
|
+
self.console = Console() if RICH_AVAILABLE else None
|
|
44
|
+
self._toolchain_analyzer = None
|
|
45
|
+
|
|
46
|
+
@property
|
|
47
|
+
def toolchain_analyzer(self) -> ToolchainAnalyzerService:
|
|
48
|
+
"""Get toolchain analyzer (lazy loaded)."""
|
|
49
|
+
if self._toolchain_analyzer is None:
|
|
50
|
+
self._toolchain_analyzer = ToolchainAnalyzerService()
|
|
51
|
+
return self._toolchain_analyzer
|
|
52
|
+
|
|
53
|
+
def validate_args(self, args) -> Optional[str]:
|
|
54
|
+
"""Validate command arguments."""
|
|
55
|
+
# Validate project path
|
|
56
|
+
project_path = (
|
|
57
|
+
Path(args.project_path)
|
|
58
|
+
if hasattr(args, "project_path") and args.project_path
|
|
59
|
+
else Path.cwd()
|
|
60
|
+
)
|
|
61
|
+
if not project_path.exists():
|
|
62
|
+
return f"Project path does not exist: {project_path}"
|
|
63
|
+
|
|
64
|
+
return None
|
|
65
|
+
|
|
66
|
+
def run(self, args) -> CommandResult:
|
|
67
|
+
"""
|
|
68
|
+
Execute agents detect command.
|
|
69
|
+
|
|
70
|
+
Returns:
|
|
71
|
+
CommandResult with success status and exit code
|
|
72
|
+
"""
|
|
73
|
+
try:
|
|
74
|
+
# Setup logging
|
|
75
|
+
self.setup_logging(args)
|
|
76
|
+
|
|
77
|
+
# Validate arguments
|
|
78
|
+
error = self.validate_args(args)
|
|
79
|
+
if error:
|
|
80
|
+
return CommandResult.error_result(error)
|
|
81
|
+
|
|
82
|
+
# Get configuration options
|
|
83
|
+
project_path = (
|
|
84
|
+
Path(args.project_path)
|
|
85
|
+
if hasattr(args, "project_path") and args.project_path
|
|
86
|
+
else Path.cwd()
|
|
87
|
+
)
|
|
88
|
+
json_output = args.json if hasattr(args, "json") and args.json else False
|
|
89
|
+
verbose = (
|
|
90
|
+
args.verbose if hasattr(args, "verbose") and args.verbose else False
|
|
91
|
+
)
|
|
92
|
+
|
|
93
|
+
# Analyze toolchain
|
|
94
|
+
if self.console and not json_output:
|
|
95
|
+
with self.console.status("[bold green]Analyzing project toolchain..."):
|
|
96
|
+
analysis = self.toolchain_analyzer.analyze_project(
|
|
97
|
+
str(project_path)
|
|
98
|
+
)
|
|
99
|
+
else:
|
|
100
|
+
analysis = self.toolchain_analyzer.analyze_project(str(project_path))
|
|
101
|
+
|
|
102
|
+
# Output results
|
|
103
|
+
if json_output:
|
|
104
|
+
return self._output_json(analysis, verbose)
|
|
105
|
+
return self._display_results(analysis, verbose)
|
|
106
|
+
|
|
107
|
+
except KeyboardInterrupt:
|
|
108
|
+
if self.console:
|
|
109
|
+
self.console.print("\n\nā Operation cancelled by user")
|
|
110
|
+
else:
|
|
111
|
+
print("\n\nOperation cancelled by user")
|
|
112
|
+
return CommandResult.error_result("Operation cancelled", exit_code=130)
|
|
113
|
+
|
|
114
|
+
except Exception as e:
|
|
115
|
+
self.logger.exception("Toolchain detection failed")
|
|
116
|
+
error_msg = f"Toolchain detection failed: {e!s}"
|
|
117
|
+
if self.console:
|
|
118
|
+
self.console.print(f"\nā {error_msg}")
|
|
119
|
+
else:
|
|
120
|
+
print(f"\n{error_msg}")
|
|
121
|
+
return CommandResult.error_result(error_msg)
|
|
122
|
+
|
|
123
|
+
def _display_results(self, analysis, verbose: bool) -> CommandResult:
|
|
124
|
+
"""Display toolchain analysis results with Rich formatting."""
|
|
125
|
+
if not self.console:
|
|
126
|
+
return self._display_results_plain(analysis, verbose)
|
|
127
|
+
|
|
128
|
+
# Display header
|
|
129
|
+
self.console.print("\nš Project Toolchain Analysis", style="bold blue")
|
|
130
|
+
self.console.print(f"Project: {analysis.project_path}\n")
|
|
131
|
+
|
|
132
|
+
# Display detected languages
|
|
133
|
+
if analysis.languages:
|
|
134
|
+
self.console.print("š¤ Detected Languages:", style="bold green")
|
|
135
|
+
lang_table = Table(show_header=True, header_style="bold")
|
|
136
|
+
lang_table.add_column("Language", style="cyan")
|
|
137
|
+
lang_table.add_column("Version", style="yellow")
|
|
138
|
+
lang_table.add_column("Confidence", style="green")
|
|
139
|
+
if verbose:
|
|
140
|
+
lang_table.add_column("Evidence", style="dim")
|
|
141
|
+
|
|
142
|
+
for lang in analysis.languages:
|
|
143
|
+
confidence_pct = int(lang.confidence * 100)
|
|
144
|
+
bar = "ā" * (confidence_pct // 10) + "ā" * (10 - confidence_pct // 10)
|
|
145
|
+
confidence_str = f"{bar} {confidence_pct}%"
|
|
146
|
+
|
|
147
|
+
if verbose:
|
|
148
|
+
evidence = ", ".join(lang.evidence[:3]) # First 3 items
|
|
149
|
+
if len(lang.evidence) > 3:
|
|
150
|
+
evidence += f" (+{len(lang.evidence) - 3} more)"
|
|
151
|
+
lang_table.add_row(
|
|
152
|
+
lang.language,
|
|
153
|
+
lang.version or "Unknown",
|
|
154
|
+
confidence_str,
|
|
155
|
+
evidence,
|
|
156
|
+
)
|
|
157
|
+
else:
|
|
158
|
+
lang_table.add_row(
|
|
159
|
+
lang.language, lang.version or "Unknown", confidence_str
|
|
160
|
+
)
|
|
161
|
+
|
|
162
|
+
self.console.print(lang_table)
|
|
163
|
+
else:
|
|
164
|
+
self.console.print(" No languages detected", style="yellow")
|
|
165
|
+
|
|
166
|
+
# Display detected frameworks
|
|
167
|
+
if analysis.frameworks:
|
|
168
|
+
self.console.print("\nšļø Detected Frameworks:", style="bold green")
|
|
169
|
+
fw_table = Table(show_header=True, header_style="bold")
|
|
170
|
+
fw_table.add_column("Framework", style="cyan")
|
|
171
|
+
fw_table.add_column("Version", style="yellow")
|
|
172
|
+
fw_table.add_column("Category", style="magenta")
|
|
173
|
+
fw_table.add_column("Confidence", style="green")
|
|
174
|
+
|
|
175
|
+
for fw in analysis.frameworks:
|
|
176
|
+
confidence_pct = int(fw.confidence * 100)
|
|
177
|
+
bar = "ā" * (confidence_pct // 10) + "ā" * (10 - confidence_pct // 10)
|
|
178
|
+
confidence_str = f"{bar} {confidence_pct}%"
|
|
179
|
+
|
|
180
|
+
fw_table.add_row(
|
|
181
|
+
fw.name,
|
|
182
|
+
fw.version or "Unknown",
|
|
183
|
+
(
|
|
184
|
+
fw.category.value
|
|
185
|
+
if hasattr(fw.category, "value")
|
|
186
|
+
else str(fw.category)
|
|
187
|
+
),
|
|
188
|
+
confidence_str,
|
|
189
|
+
)
|
|
190
|
+
|
|
191
|
+
self.console.print(fw_table)
|
|
192
|
+
else:
|
|
193
|
+
self.console.print("\n No frameworks detected", style="yellow")
|
|
194
|
+
|
|
195
|
+
# Display deployment targets
|
|
196
|
+
if analysis.deployment_targets:
|
|
197
|
+
self.console.print("\nš Deployment Targets:", style="bold green")
|
|
198
|
+
dt_table = Table(show_header=True, header_style="bold")
|
|
199
|
+
dt_table.add_column("Target", style="cyan")
|
|
200
|
+
dt_table.add_column("Confidence", style="green")
|
|
201
|
+
if verbose:
|
|
202
|
+
dt_table.add_column("Evidence", style="dim")
|
|
203
|
+
|
|
204
|
+
for target in analysis.deployment_targets:
|
|
205
|
+
confidence_pct = int(target.confidence * 100)
|
|
206
|
+
bar = "ā" * (confidence_pct // 10) + "ā" * (10 - confidence_pct // 10)
|
|
207
|
+
confidence_str = f"{bar} {confidence_pct}%"
|
|
208
|
+
|
|
209
|
+
if verbose:
|
|
210
|
+
evidence = ", ".join(target.evidence[:3])
|
|
211
|
+
if len(target.evidence) > 3:
|
|
212
|
+
evidence += f" (+{len(target.evidence) - 3} more)"
|
|
213
|
+
dt_table.add_row(
|
|
214
|
+
(
|
|
215
|
+
target.target_type.value
|
|
216
|
+
if hasattr(target.target_type, "value")
|
|
217
|
+
else str(target.target_type)
|
|
218
|
+
),
|
|
219
|
+
confidence_str,
|
|
220
|
+
evidence,
|
|
221
|
+
)
|
|
222
|
+
else:
|
|
223
|
+
dt_table.add_row(
|
|
224
|
+
(
|
|
225
|
+
target.target_type.value
|
|
226
|
+
if hasattr(target.target_type, "value")
|
|
227
|
+
else str(target.target_type)
|
|
228
|
+
),
|
|
229
|
+
confidence_str,
|
|
230
|
+
)
|
|
231
|
+
|
|
232
|
+
self.console.print(dt_table)
|
|
233
|
+
else:
|
|
234
|
+
self.console.print("\n No deployment targets detected", style="yellow")
|
|
235
|
+
|
|
236
|
+
# Display all components summary
|
|
237
|
+
if analysis.components:
|
|
238
|
+
self.console.print("\nš¦ All Components:", style="bold blue")
|
|
239
|
+
comp_table = Table(show_header=True, header_style="bold")
|
|
240
|
+
comp_table.add_column("Type", style="cyan")
|
|
241
|
+
comp_table.add_column("Name", style="yellow")
|
|
242
|
+
comp_table.add_column("Version", style="magenta")
|
|
243
|
+
comp_table.add_column("Confidence", style="green")
|
|
244
|
+
|
|
245
|
+
for comp in analysis.components:
|
|
246
|
+
confidence_pct = int(comp.confidence * 100)
|
|
247
|
+
bar = "ā" * (confidence_pct // 10) + "ā" * (10 - confidence_pct // 10)
|
|
248
|
+
confidence_str = f"{bar} {confidence_pct}%"
|
|
249
|
+
|
|
250
|
+
comp_table.add_row(
|
|
251
|
+
comp.type.value if hasattr(comp.type, "value") else str(comp.type),
|
|
252
|
+
comp.name or "-",
|
|
253
|
+
comp.version or "Unknown",
|
|
254
|
+
confidence_str,
|
|
255
|
+
)
|
|
256
|
+
|
|
257
|
+
self.console.print(comp_table)
|
|
258
|
+
|
|
259
|
+
# Summary
|
|
260
|
+
self.console.print(
|
|
261
|
+
f"\nā
Analysis complete: {len(analysis.languages)} language(s), "
|
|
262
|
+
f"{len(analysis.frameworks)} framework(s), "
|
|
263
|
+
f"{len(analysis.deployment_targets)} deployment target(s)",
|
|
264
|
+
style="bold green",
|
|
265
|
+
)
|
|
266
|
+
|
|
267
|
+
return CommandResult.success_result()
|
|
268
|
+
|
|
269
|
+
def _display_results_plain(self, analysis, verbose: bool) -> CommandResult:
|
|
270
|
+
"""Display results in plain text (fallback)."""
|
|
271
|
+
print("\nš Project Toolchain Analysis")
|
|
272
|
+
print(f"Project: {analysis.project_path}\n")
|
|
273
|
+
|
|
274
|
+
# Languages
|
|
275
|
+
if analysis.languages:
|
|
276
|
+
print("Detected Languages:")
|
|
277
|
+
for lang in analysis.languages:
|
|
278
|
+
confidence_pct = int(lang.confidence * 100)
|
|
279
|
+
print(
|
|
280
|
+
f" - {lang.language} {lang.version or 'Unknown'} ({confidence_pct}%)"
|
|
281
|
+
)
|
|
282
|
+
if verbose:
|
|
283
|
+
print(f" Evidence: {', '.join(lang.evidence[:5])}")
|
|
284
|
+
else:
|
|
285
|
+
print("No languages detected")
|
|
286
|
+
|
|
287
|
+
# Frameworks
|
|
288
|
+
if analysis.frameworks:
|
|
289
|
+
print("\nDetected Frameworks:")
|
|
290
|
+
for fw in analysis.frameworks:
|
|
291
|
+
confidence_pct = int(fw.confidence * 100)
|
|
292
|
+
print(
|
|
293
|
+
f" - {fw.name} {fw.version or 'Unknown'} ({fw.category}) ({confidence_pct}%)"
|
|
294
|
+
)
|
|
295
|
+
else:
|
|
296
|
+
print("\nNo frameworks detected")
|
|
297
|
+
|
|
298
|
+
# Deployment targets
|
|
299
|
+
if analysis.deployment_targets:
|
|
300
|
+
print("\nDeployment Targets:")
|
|
301
|
+
for target in analysis.deployment_targets:
|
|
302
|
+
confidence_pct = int(target.confidence * 100)
|
|
303
|
+
print(f" - {target.target_type} ({confidence_pct}%)")
|
|
304
|
+
if verbose:
|
|
305
|
+
print(f" Evidence: {', '.join(target.evidence[:5])}")
|
|
306
|
+
else:
|
|
307
|
+
print("\nNo deployment targets detected")
|
|
308
|
+
|
|
309
|
+
print(
|
|
310
|
+
f"\nAnalysis complete: {len(analysis.languages)} language(s), "
|
|
311
|
+
f"{len(analysis.frameworks)} framework(s), "
|
|
312
|
+
f"{len(analysis.deployment_targets)} deployment target(s)"
|
|
313
|
+
)
|
|
314
|
+
|
|
315
|
+
return CommandResult.success_result()
|
|
316
|
+
|
|
317
|
+
def _output_json(self, analysis, verbose: bool) -> CommandResult:
|
|
318
|
+
"""Output toolchain analysis as JSON."""
|
|
319
|
+
output = {
|
|
320
|
+
"project_path": analysis.project_path,
|
|
321
|
+
"analysis_time": analysis.analysis_time,
|
|
322
|
+
"languages": [
|
|
323
|
+
{
|
|
324
|
+
"language": lang.language,
|
|
325
|
+
"version": lang.version,
|
|
326
|
+
"confidence": lang.confidence,
|
|
327
|
+
"evidence": lang.evidence if verbose else None,
|
|
328
|
+
}
|
|
329
|
+
for lang in analysis.languages
|
|
330
|
+
],
|
|
331
|
+
"frameworks": [
|
|
332
|
+
{
|
|
333
|
+
"name": fw.name,
|
|
334
|
+
"version": fw.version,
|
|
335
|
+
"category": (
|
|
336
|
+
fw.category.value
|
|
337
|
+
if hasattr(fw.category, "value")
|
|
338
|
+
else str(fw.category)
|
|
339
|
+
),
|
|
340
|
+
"confidence": fw.confidence,
|
|
341
|
+
"config_file": fw.config_file,
|
|
342
|
+
}
|
|
343
|
+
for fw in analysis.frameworks
|
|
344
|
+
],
|
|
345
|
+
"deployment_targets": [
|
|
346
|
+
{
|
|
347
|
+
"target_type": (
|
|
348
|
+
target.target_type.value
|
|
349
|
+
if hasattr(target.target_type, "value")
|
|
350
|
+
else str(target.target_type)
|
|
351
|
+
),
|
|
352
|
+
"confidence": target.confidence,
|
|
353
|
+
"evidence": target.evidence if verbose else None,
|
|
354
|
+
}
|
|
355
|
+
for target in analysis.deployment_targets
|
|
356
|
+
],
|
|
357
|
+
"components": [
|
|
358
|
+
{
|
|
359
|
+
"type": (
|
|
360
|
+
comp.type.value
|
|
361
|
+
if hasattr(comp.type, "value")
|
|
362
|
+
else str(comp.type)
|
|
363
|
+
),
|
|
364
|
+
"name": comp.name,
|
|
365
|
+
"version": comp.version,
|
|
366
|
+
"confidence": comp.confidence,
|
|
367
|
+
}
|
|
368
|
+
for comp in analysis.components
|
|
369
|
+
],
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
# Remove None values if not verbose
|
|
373
|
+
if not verbose:
|
|
374
|
+
for lang in output["languages"]:
|
|
375
|
+
lang.pop("evidence", None)
|
|
376
|
+
for target in output["deployment_targets"]:
|
|
377
|
+
target.pop("evidence", None)
|
|
378
|
+
|
|
379
|
+
print(json.dumps(output, indent=2))
|
|
380
|
+
return CommandResult.success_result(data=output)
|
|
@@ -0,0 +1,309 @@
|
|
|
1
|
+
"""
|
|
2
|
+
Agents Recommend CLI Command for Claude MPM Framework
|
|
3
|
+
======================================================
|
|
4
|
+
|
|
5
|
+
WHY: This module provides a CLI interface for getting agent recommendations
|
|
6
|
+
based on project toolchain without deploying anything. Useful for reviewing
|
|
7
|
+
recommendations before committing to deployment.
|
|
8
|
+
|
|
9
|
+
DESIGN DECISION: Focused on recommendation display with detailed reasoning,
|
|
10
|
+
showing users why each agent was recommended. Supports JSON output for
|
|
11
|
+
integration with other tools.
|
|
12
|
+
|
|
13
|
+
Part of TSK-0054: Auto-Configuration Feature - Phase 5
|
|
14
|
+
"""
|
|
15
|
+
|
|
16
|
+
import json
|
|
17
|
+
from pathlib import Path
|
|
18
|
+
from typing import Optional
|
|
19
|
+
|
|
20
|
+
try:
|
|
21
|
+
from rich.console import Console
|
|
22
|
+
from rich.panel import Panel
|
|
23
|
+
from rich.table import Table
|
|
24
|
+
|
|
25
|
+
RICH_AVAILABLE = True
|
|
26
|
+
except ImportError:
|
|
27
|
+
RICH_AVAILABLE = False
|
|
28
|
+
|
|
29
|
+
from ...services.agents.recommender import AgentRecommenderService
|
|
30
|
+
from ...services.agents.registry import AgentRegistry
|
|
31
|
+
from ...services.project.toolchain_analyzer import ToolchainAnalyzerService
|
|
32
|
+
from ..shared import BaseCommand, CommandResult
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
class AgentsRecommendCommand(BaseCommand):
|
|
36
|
+
"""
|
|
37
|
+
Handle agents recommend CLI command.
|
|
38
|
+
|
|
39
|
+
This command analyzes the project toolchain and recommends appropriate
|
|
40
|
+
agents without deploying them. Shows detailed reasoning for each
|
|
41
|
+
recommendation.
|
|
42
|
+
"""
|
|
43
|
+
|
|
44
|
+
def __init__(self):
|
|
45
|
+
"""Initialize the agents recommend command."""
|
|
46
|
+
super().__init__("agents-recommend")
|
|
47
|
+
self.console = Console() if RICH_AVAILABLE else None
|
|
48
|
+
self._toolchain_analyzer = None
|
|
49
|
+
self._agent_recommender = None
|
|
50
|
+
|
|
51
|
+
@property
|
|
52
|
+
def toolchain_analyzer(self) -> ToolchainAnalyzerService:
|
|
53
|
+
"""Get toolchain analyzer (lazy loaded)."""
|
|
54
|
+
if self._toolchain_analyzer is None:
|
|
55
|
+
self._toolchain_analyzer = ToolchainAnalyzerService()
|
|
56
|
+
return self._toolchain_analyzer
|
|
57
|
+
|
|
58
|
+
@property
|
|
59
|
+
def agent_recommender(self) -> AgentRecommenderService:
|
|
60
|
+
"""Get agent recommender (lazy loaded)."""
|
|
61
|
+
if self._agent_recommender is None:
|
|
62
|
+
agent_registry = AgentRegistry()
|
|
63
|
+
self._agent_recommender = AgentRecommenderService(
|
|
64
|
+
agent_registry=agent_registry
|
|
65
|
+
)
|
|
66
|
+
return self._agent_recommender
|
|
67
|
+
|
|
68
|
+
def validate_args(self, args) -> Optional[str]:
|
|
69
|
+
"""Validate command arguments."""
|
|
70
|
+
# Validate project path
|
|
71
|
+
project_path = (
|
|
72
|
+
Path(args.project_path)
|
|
73
|
+
if hasattr(args, "project_path") and args.project_path
|
|
74
|
+
else Path.cwd()
|
|
75
|
+
)
|
|
76
|
+
if not project_path.exists():
|
|
77
|
+
return f"Project path does not exist: {project_path}"
|
|
78
|
+
|
|
79
|
+
# Validate min_confidence range
|
|
80
|
+
if hasattr(args, "min_confidence") and args.min_confidence:
|
|
81
|
+
if not 0.0 <= args.min_confidence <= 1.0:
|
|
82
|
+
return "min_confidence must be between 0.0 and 1.0"
|
|
83
|
+
|
|
84
|
+
return None
|
|
85
|
+
|
|
86
|
+
def run(self, args) -> CommandResult:
|
|
87
|
+
"""
|
|
88
|
+
Execute agents recommend command.
|
|
89
|
+
|
|
90
|
+
Returns:
|
|
91
|
+
CommandResult with success status and exit code
|
|
92
|
+
"""
|
|
93
|
+
try:
|
|
94
|
+
# Setup logging
|
|
95
|
+
self.setup_logging(args)
|
|
96
|
+
|
|
97
|
+
# Validate arguments
|
|
98
|
+
error = self.validate_args(args)
|
|
99
|
+
if error:
|
|
100
|
+
return CommandResult.error_result(error)
|
|
101
|
+
|
|
102
|
+
# Get configuration options
|
|
103
|
+
project_path = (
|
|
104
|
+
Path(args.project_path)
|
|
105
|
+
if hasattr(args, "project_path") and args.project_path
|
|
106
|
+
else Path.cwd()
|
|
107
|
+
)
|
|
108
|
+
min_confidence = (
|
|
109
|
+
args.min_confidence
|
|
110
|
+
if hasattr(args, "min_confidence") and args.min_confidence
|
|
111
|
+
else 0.8
|
|
112
|
+
)
|
|
113
|
+
json_output = args.json if hasattr(args, "json") and args.json else False
|
|
114
|
+
show_reasoning = (
|
|
115
|
+
args.show_reasoning
|
|
116
|
+
if hasattr(args, "show_reasoning") and args.show_reasoning
|
|
117
|
+
else True
|
|
118
|
+
)
|
|
119
|
+
|
|
120
|
+
# Analyze toolchain
|
|
121
|
+
if self.console and not json_output:
|
|
122
|
+
with self.console.status("[bold green]Analyzing project toolchain..."):
|
|
123
|
+
analysis = self.toolchain_analyzer.analyze_project(
|
|
124
|
+
str(project_path)
|
|
125
|
+
)
|
|
126
|
+
else:
|
|
127
|
+
analysis = self.toolchain_analyzer.analyze_project(str(project_path))
|
|
128
|
+
|
|
129
|
+
# Get recommendations
|
|
130
|
+
if self.console and not json_output:
|
|
131
|
+
with self.console.status(
|
|
132
|
+
"[bold green]Generating agent recommendations..."
|
|
133
|
+
):
|
|
134
|
+
recommendations = self.agent_recommender.recommend_agents(
|
|
135
|
+
analysis, min_confidence
|
|
136
|
+
)
|
|
137
|
+
else:
|
|
138
|
+
recommendations = self.agent_recommender.recommend_agents(
|
|
139
|
+
analysis, min_confidence
|
|
140
|
+
)
|
|
141
|
+
|
|
142
|
+
# Output results
|
|
143
|
+
if json_output:
|
|
144
|
+
return self._output_json(recommendations, analysis)
|
|
145
|
+
return self._display_results(recommendations, analysis, show_reasoning)
|
|
146
|
+
|
|
147
|
+
except KeyboardInterrupt:
|
|
148
|
+
if self.console:
|
|
149
|
+
self.console.print("\n\nā Operation cancelled by user")
|
|
150
|
+
else:
|
|
151
|
+
print("\n\nOperation cancelled by user")
|
|
152
|
+
return CommandResult.error_result("Operation cancelled", exit_code=130)
|
|
153
|
+
|
|
154
|
+
except Exception as e:
|
|
155
|
+
self.logger.exception("Agent recommendation failed")
|
|
156
|
+
error_msg = f"Agent recommendation failed: {e!s}"
|
|
157
|
+
if self.console:
|
|
158
|
+
self.console.print(f"\nā {error_msg}")
|
|
159
|
+
else:
|
|
160
|
+
print(f"\n{error_msg}")
|
|
161
|
+
return CommandResult.error_result(error_msg)
|
|
162
|
+
|
|
163
|
+
def _display_results(
|
|
164
|
+
self, recommendations, analysis, show_reasoning: bool
|
|
165
|
+
) -> CommandResult:
|
|
166
|
+
"""Display agent recommendations with Rich formatting."""
|
|
167
|
+
if not self.console:
|
|
168
|
+
return self._display_results_plain(
|
|
169
|
+
recommendations, analysis, show_reasoning
|
|
170
|
+
)
|
|
171
|
+
|
|
172
|
+
# Display header
|
|
173
|
+
self.console.print("\nš¤ Agent Recommendations", style="bold blue")
|
|
174
|
+
self.console.print(f"Project: {analysis.project_path}\n")
|
|
175
|
+
|
|
176
|
+
# Display quick summary
|
|
177
|
+
if recommendations:
|
|
178
|
+
summary_text = (
|
|
179
|
+
f"Found {len(recommendations)} recommended agent(s) "
|
|
180
|
+
f"for your project based on detected toolchain."
|
|
181
|
+
)
|
|
182
|
+
panel = Panel(summary_text, border_style="green")
|
|
183
|
+
self.console.print(panel)
|
|
184
|
+
else:
|
|
185
|
+
panel = Panel(
|
|
186
|
+
"No agents recommended for this project.\n"
|
|
187
|
+
"Try lowering the confidence threshold with --min-confidence",
|
|
188
|
+
border_style="yellow",
|
|
189
|
+
)
|
|
190
|
+
self.console.print(panel)
|
|
191
|
+
return CommandResult.success_result()
|
|
192
|
+
|
|
193
|
+
# Display recommendations table
|
|
194
|
+
self.console.print("\nš Recommended Agents:", style="bold green")
|
|
195
|
+
table = Table(show_header=True, header_style="bold")
|
|
196
|
+
table.add_column("Agent ID", style="cyan")
|
|
197
|
+
table.add_column("Confidence", style="green")
|
|
198
|
+
table.add_column("Priority", style="yellow")
|
|
199
|
+
if show_reasoning:
|
|
200
|
+
table.add_column("Reasoning", style="dim", no_wrap=False)
|
|
201
|
+
|
|
202
|
+
for rec in recommendations:
|
|
203
|
+
confidence_pct = int(rec.confidence * 100)
|
|
204
|
+
bar = "ā" * (confidence_pct // 10) + "ā" * (10 - confidence_pct // 10)
|
|
205
|
+
confidence_str = f"{bar} {confidence_pct}%"
|
|
206
|
+
|
|
207
|
+
if show_reasoning:
|
|
208
|
+
table.add_row(
|
|
209
|
+
rec.agent_id,
|
|
210
|
+
confidence_str,
|
|
211
|
+
str(rec.priority),
|
|
212
|
+
rec.reasoning,
|
|
213
|
+
)
|
|
214
|
+
else:
|
|
215
|
+
table.add_row(rec.agent_id, confidence_str, str(rec.priority))
|
|
216
|
+
|
|
217
|
+
self.console.print(table)
|
|
218
|
+
|
|
219
|
+
# Display match details if reasoning enabled
|
|
220
|
+
if show_reasoning:
|
|
221
|
+
self.console.print("\nš Match Details:", style="bold blue")
|
|
222
|
+
for rec in recommendations:
|
|
223
|
+
self.console.print(f"\n[bold cyan]{rec.agent_id}[/bold cyan]")
|
|
224
|
+
self.console.print(f" Confidence: {int(rec.confidence * 100)}%")
|
|
225
|
+
self.console.print(f" Priority: {rec.priority}")
|
|
226
|
+
self.console.print(f" Reasoning: {rec.reasoning}")
|
|
227
|
+
|
|
228
|
+
if rec.matched_capabilities:
|
|
229
|
+
self.console.print(" Matched capabilities:", style="green")
|
|
230
|
+
for cap in rec.matched_capabilities[:5]:
|
|
231
|
+
self.console.print(f" ⢠{cap}", style="dim")
|
|
232
|
+
if len(rec.matched_capabilities) > 5:
|
|
233
|
+
remaining = len(rec.matched_capabilities) - 5
|
|
234
|
+
self.console.print(f" ... and {remaining} more", style="dim")
|
|
235
|
+
|
|
236
|
+
# Display next steps
|
|
237
|
+
self.console.print("\nš” Next Steps:", style="bold yellow")
|
|
238
|
+
self.console.print(" 1. Review the recommendations and their reasoning")
|
|
239
|
+
self.console.print(
|
|
240
|
+
" 2. Deploy agents with: [bold]claude-mpm auto-configure[/bold]"
|
|
241
|
+
)
|
|
242
|
+
self.console.print(
|
|
243
|
+
" 3. Or preview deployment with: [bold]claude-mpm auto-configure --preview[/bold]"
|
|
244
|
+
)
|
|
245
|
+
|
|
246
|
+
return CommandResult.success_result()
|
|
247
|
+
|
|
248
|
+
def _display_results_plain(
|
|
249
|
+
self, recommendations, analysis, show_reasoning: bool
|
|
250
|
+
) -> CommandResult:
|
|
251
|
+
"""Display results in plain text (fallback)."""
|
|
252
|
+
print("\nš¤ Agent Recommendations")
|
|
253
|
+
print(f"Project: {analysis.project_path}\n")
|
|
254
|
+
|
|
255
|
+
if not recommendations:
|
|
256
|
+
print("No agents recommended for this project.")
|
|
257
|
+
print("Try lowering the confidence threshold with --min-confidence")
|
|
258
|
+
return CommandResult.success_result()
|
|
259
|
+
|
|
260
|
+
print(f"Found {len(recommendations)} recommended agent(s) for your project:\n")
|
|
261
|
+
|
|
262
|
+
for rec in recommendations:
|
|
263
|
+
confidence_pct = int(rec.confidence * 100)
|
|
264
|
+
print(f"⢠{rec.agent_id} ({confidence_pct}% confidence)")
|
|
265
|
+
|
|
266
|
+
if show_reasoning:
|
|
267
|
+
print(f" Priority: {rec.priority}")
|
|
268
|
+
print(f" Reasoning: {rec.reasoning}")
|
|
269
|
+
|
|
270
|
+
if rec.matched_capabilities:
|
|
271
|
+
print(" Matched capabilities:")
|
|
272
|
+
for cap in rec.matched_capabilities[:5]:
|
|
273
|
+
print(f" - {cap}")
|
|
274
|
+
if len(rec.matched_capabilities) > 5:
|
|
275
|
+
remaining = len(rec.matched_capabilities) - 5
|
|
276
|
+
print(f" ... and {remaining} more")
|
|
277
|
+
print()
|
|
278
|
+
|
|
279
|
+
print("\nNext Steps:")
|
|
280
|
+
print(" 1. Review the recommendations and their reasoning")
|
|
281
|
+
print(" 2. Deploy agents with: claude-mpm auto-configure")
|
|
282
|
+
print(" 3. Or preview deployment with: claude-mpm auto-configure --preview")
|
|
283
|
+
|
|
284
|
+
return CommandResult.success_result()
|
|
285
|
+
|
|
286
|
+
def _output_json(self, recommendations, analysis) -> CommandResult:
|
|
287
|
+
"""Output recommendations as JSON."""
|
|
288
|
+
output = {
|
|
289
|
+
"project_path": analysis.project_path,
|
|
290
|
+
"recommendations": [
|
|
291
|
+
{
|
|
292
|
+
"agent_id": rec.agent_id,
|
|
293
|
+
"confidence": rec.confidence,
|
|
294
|
+
"priority": rec.priority,
|
|
295
|
+
"reasoning": rec.reasoning,
|
|
296
|
+
"matched_capabilities": rec.matched_capabilities,
|
|
297
|
+
"requirements": rec.requirements,
|
|
298
|
+
}
|
|
299
|
+
for rec in recommendations
|
|
300
|
+
],
|
|
301
|
+
"toolchain_summary": {
|
|
302
|
+
"languages": len(analysis.languages),
|
|
303
|
+
"frameworks": len(analysis.frameworks),
|
|
304
|
+
"deployment_targets": len(analysis.deployment_targets),
|
|
305
|
+
},
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
print(json.dumps(output, indent=2))
|
|
309
|
+
return CommandResult.success_result(data=output)
|