attune-ai 2.0.0__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.
- attune/__init__.py +358 -0
- attune/adaptive/__init__.py +13 -0
- attune/adaptive/task_complexity.py +127 -0
- attune/agent_monitoring.py +414 -0
- attune/cache/__init__.py +117 -0
- attune/cache/base.py +166 -0
- attune/cache/dependency_manager.py +256 -0
- attune/cache/hash_only.py +251 -0
- attune/cache/hybrid.py +457 -0
- attune/cache/storage.py +285 -0
- attune/cache_monitor.py +356 -0
- attune/cache_stats.py +298 -0
- attune/cli/__init__.py +152 -0
- attune/cli/__main__.py +12 -0
- attune/cli/commands/__init__.py +1 -0
- attune/cli/commands/batch.py +264 -0
- attune/cli/commands/cache.py +248 -0
- attune/cli/commands/help.py +331 -0
- attune/cli/commands/info.py +140 -0
- attune/cli/commands/inspect.py +436 -0
- attune/cli/commands/inspection.py +57 -0
- attune/cli/commands/memory.py +48 -0
- attune/cli/commands/metrics.py +92 -0
- attune/cli/commands/orchestrate.py +184 -0
- attune/cli/commands/patterns.py +207 -0
- attune/cli/commands/profiling.py +202 -0
- attune/cli/commands/provider.py +98 -0
- attune/cli/commands/routing.py +285 -0
- attune/cli/commands/setup.py +96 -0
- attune/cli/commands/status.py +235 -0
- attune/cli/commands/sync.py +166 -0
- attune/cli/commands/tier.py +121 -0
- attune/cli/commands/utilities.py +114 -0
- attune/cli/commands/workflow.py +579 -0
- attune/cli/core.py +32 -0
- attune/cli/parsers/__init__.py +68 -0
- attune/cli/parsers/batch.py +118 -0
- attune/cli/parsers/cache.py +65 -0
- attune/cli/parsers/help.py +41 -0
- attune/cli/parsers/info.py +26 -0
- attune/cli/parsers/inspect.py +66 -0
- attune/cli/parsers/metrics.py +42 -0
- attune/cli/parsers/orchestrate.py +61 -0
- attune/cli/parsers/patterns.py +54 -0
- attune/cli/parsers/provider.py +40 -0
- attune/cli/parsers/routing.py +110 -0
- attune/cli/parsers/setup.py +42 -0
- attune/cli/parsers/status.py +47 -0
- attune/cli/parsers/sync.py +31 -0
- attune/cli/parsers/tier.py +33 -0
- attune/cli/parsers/workflow.py +77 -0
- attune/cli/utils/__init__.py +1 -0
- attune/cli/utils/data.py +242 -0
- attune/cli/utils/helpers.py +68 -0
- attune/cli_legacy.py +3957 -0
- attune/cli_minimal.py +1159 -0
- attune/cli_router.py +437 -0
- attune/cli_unified.py +814 -0
- attune/config/__init__.py +66 -0
- attune/config/xml_config.py +286 -0
- attune/config.py +545 -0
- attune/coordination.py +870 -0
- attune/core.py +1511 -0
- attune/core_modules/__init__.py +15 -0
- attune/cost_tracker.py +626 -0
- attune/dashboard/__init__.py +41 -0
- attune/dashboard/app.py +512 -0
- attune/dashboard/simple_server.py +435 -0
- attune/dashboard/standalone_server.py +547 -0
- attune/discovery.py +306 -0
- attune/emergence.py +306 -0
- attune/exceptions.py +123 -0
- attune/feedback_loops.py +373 -0
- attune/hot_reload/README.md +473 -0
- attune/hot_reload/__init__.py +62 -0
- attune/hot_reload/config.py +83 -0
- attune/hot_reload/integration.py +229 -0
- attune/hot_reload/reloader.py +298 -0
- attune/hot_reload/watcher.py +183 -0
- attune/hot_reload/websocket.py +177 -0
- attune/levels.py +577 -0
- attune/leverage_points.py +441 -0
- attune/logging_config.py +261 -0
- attune/mcp/__init__.py +10 -0
- attune/mcp/server.py +506 -0
- attune/memory/__init__.py +237 -0
- attune/memory/claude_memory.py +469 -0
- attune/memory/config.py +224 -0
- attune/memory/control_panel.py +1290 -0
- attune/memory/control_panel_support.py +145 -0
- attune/memory/cross_session.py +845 -0
- attune/memory/edges.py +179 -0
- attune/memory/encryption.py +159 -0
- attune/memory/file_session.py +770 -0
- attune/memory/graph.py +570 -0
- attune/memory/long_term.py +913 -0
- attune/memory/long_term_types.py +99 -0
- attune/memory/mixins/__init__.py +25 -0
- attune/memory/mixins/backend_init_mixin.py +249 -0
- attune/memory/mixins/capabilities_mixin.py +208 -0
- attune/memory/mixins/handoff_mixin.py +208 -0
- attune/memory/mixins/lifecycle_mixin.py +49 -0
- attune/memory/mixins/long_term_mixin.py +352 -0
- attune/memory/mixins/promotion_mixin.py +109 -0
- attune/memory/mixins/short_term_mixin.py +182 -0
- attune/memory/nodes.py +179 -0
- attune/memory/redis_bootstrap.py +540 -0
- attune/memory/security/__init__.py +31 -0
- attune/memory/security/audit_logger.py +932 -0
- attune/memory/security/pii_scrubber.py +640 -0
- attune/memory/security/secrets_detector.py +678 -0
- attune/memory/short_term.py +2192 -0
- attune/memory/simple_storage.py +302 -0
- attune/memory/storage/__init__.py +15 -0
- attune/memory/storage_backend.py +167 -0
- attune/memory/summary_index.py +583 -0
- attune/memory/types.py +446 -0
- attune/memory/unified.py +182 -0
- attune/meta_workflows/__init__.py +74 -0
- attune/meta_workflows/agent_creator.py +248 -0
- attune/meta_workflows/builtin_templates.py +567 -0
- attune/meta_workflows/cli_commands/__init__.py +56 -0
- attune/meta_workflows/cli_commands/agent_commands.py +321 -0
- attune/meta_workflows/cli_commands/analytics_commands.py +442 -0
- attune/meta_workflows/cli_commands/config_commands.py +232 -0
- attune/meta_workflows/cli_commands/memory_commands.py +182 -0
- attune/meta_workflows/cli_commands/template_commands.py +354 -0
- attune/meta_workflows/cli_commands/workflow_commands.py +382 -0
- attune/meta_workflows/cli_meta_workflows.py +59 -0
- attune/meta_workflows/form_engine.py +292 -0
- attune/meta_workflows/intent_detector.py +409 -0
- attune/meta_workflows/models.py +569 -0
- attune/meta_workflows/pattern_learner.py +738 -0
- attune/meta_workflows/plan_generator.py +384 -0
- attune/meta_workflows/session_context.py +397 -0
- attune/meta_workflows/template_registry.py +229 -0
- attune/meta_workflows/workflow.py +984 -0
- attune/metrics/__init__.py +12 -0
- attune/metrics/collector.py +31 -0
- attune/metrics/prompt_metrics.py +194 -0
- attune/models/__init__.py +172 -0
- attune/models/__main__.py +13 -0
- attune/models/adaptive_routing.py +437 -0
- attune/models/auth_cli.py +444 -0
- attune/models/auth_strategy.py +450 -0
- attune/models/cli.py +655 -0
- attune/models/empathy_executor.py +354 -0
- attune/models/executor.py +257 -0
- attune/models/fallback.py +762 -0
- attune/models/provider_config.py +282 -0
- attune/models/registry.py +472 -0
- attune/models/tasks.py +359 -0
- attune/models/telemetry/__init__.py +71 -0
- attune/models/telemetry/analytics.py +594 -0
- attune/models/telemetry/backend.py +196 -0
- attune/models/telemetry/data_models.py +431 -0
- attune/models/telemetry/storage.py +489 -0
- attune/models/token_estimator.py +420 -0
- attune/models/validation.py +280 -0
- attune/monitoring/__init__.py +52 -0
- attune/monitoring/alerts.py +946 -0
- attune/monitoring/alerts_cli.py +448 -0
- attune/monitoring/multi_backend.py +271 -0
- attune/monitoring/otel_backend.py +362 -0
- attune/optimization/__init__.py +19 -0
- attune/optimization/context_optimizer.py +272 -0
- attune/orchestration/__init__.py +67 -0
- attune/orchestration/agent_templates.py +707 -0
- attune/orchestration/config_store.py +499 -0
- attune/orchestration/execution_strategies.py +2111 -0
- attune/orchestration/meta_orchestrator.py +1168 -0
- attune/orchestration/pattern_learner.py +696 -0
- attune/orchestration/real_tools.py +931 -0
- attune/pattern_cache.py +187 -0
- attune/pattern_library.py +542 -0
- attune/patterns/debugging/all_patterns.json +81 -0
- attune/patterns/debugging/workflow_20260107_1770825e.json +77 -0
- attune/patterns/refactoring_memory.json +89 -0
- attune/persistence.py +564 -0
- attune/platform_utils.py +265 -0
- attune/plugins/__init__.py +28 -0
- attune/plugins/base.py +361 -0
- attune/plugins/registry.py +268 -0
- attune/project_index/__init__.py +32 -0
- attune/project_index/cli.py +335 -0
- attune/project_index/index.py +667 -0
- attune/project_index/models.py +504 -0
- attune/project_index/reports.py +474 -0
- attune/project_index/scanner.py +777 -0
- attune/project_index/scanner_parallel.py +291 -0
- attune/prompts/__init__.py +61 -0
- attune/prompts/config.py +77 -0
- attune/prompts/context.py +177 -0
- attune/prompts/parser.py +285 -0
- attune/prompts/registry.py +313 -0
- attune/prompts/templates.py +208 -0
- attune/redis_config.py +302 -0
- attune/redis_memory.py +799 -0
- attune/resilience/__init__.py +56 -0
- attune/resilience/circuit_breaker.py +256 -0
- attune/resilience/fallback.py +179 -0
- attune/resilience/health.py +300 -0
- attune/resilience/retry.py +209 -0
- attune/resilience/timeout.py +135 -0
- attune/routing/__init__.py +43 -0
- attune/routing/chain_executor.py +433 -0
- attune/routing/classifier.py +217 -0
- attune/routing/smart_router.py +234 -0
- attune/routing/workflow_registry.py +343 -0
- attune/scaffolding/README.md +589 -0
- attune/scaffolding/__init__.py +35 -0
- attune/scaffolding/__main__.py +14 -0
- attune/scaffolding/cli.py +240 -0
- attune/scaffolding/templates/base_wizard.py.jinja2 +121 -0
- attune/scaffolding/templates/coach_wizard.py.jinja2 +321 -0
- attune/scaffolding/templates/domain_wizard.py.jinja2 +408 -0
- attune/scaffolding/templates/linear_flow_wizard.py.jinja2 +203 -0
- attune/socratic/__init__.py +256 -0
- attune/socratic/ab_testing.py +958 -0
- attune/socratic/blueprint.py +533 -0
- attune/socratic/cli.py +703 -0
- attune/socratic/collaboration.py +1114 -0
- attune/socratic/domain_templates.py +924 -0
- attune/socratic/embeddings.py +738 -0
- attune/socratic/engine.py +794 -0
- attune/socratic/explainer.py +682 -0
- attune/socratic/feedback.py +772 -0
- attune/socratic/forms.py +629 -0
- attune/socratic/generator.py +732 -0
- attune/socratic/llm_analyzer.py +637 -0
- attune/socratic/mcp_server.py +702 -0
- attune/socratic/session.py +312 -0
- attune/socratic/storage.py +667 -0
- attune/socratic/success.py +730 -0
- attune/socratic/visual_editor.py +860 -0
- attune/socratic/web_ui.py +958 -0
- attune/telemetry/__init__.py +39 -0
- attune/telemetry/agent_coordination.py +475 -0
- attune/telemetry/agent_tracking.py +367 -0
- attune/telemetry/approval_gates.py +545 -0
- attune/telemetry/cli.py +1231 -0
- attune/telemetry/commands/__init__.py +14 -0
- attune/telemetry/commands/dashboard_commands.py +696 -0
- attune/telemetry/event_streaming.py +409 -0
- attune/telemetry/feedback_loop.py +567 -0
- attune/telemetry/usage_tracker.py +591 -0
- attune/templates.py +754 -0
- attune/test_generator/__init__.py +38 -0
- attune/test_generator/__main__.py +14 -0
- attune/test_generator/cli.py +234 -0
- attune/test_generator/generator.py +355 -0
- attune/test_generator/risk_analyzer.py +216 -0
- attune/test_generator/templates/unit_test.py.jinja2 +272 -0
- attune/tier_recommender.py +384 -0
- attune/tools.py +183 -0
- attune/trust/__init__.py +28 -0
- attune/trust/circuit_breaker.py +579 -0
- attune/trust_building.py +527 -0
- attune/validation/__init__.py +19 -0
- attune/validation/xml_validator.py +281 -0
- attune/vscode_bridge.py +173 -0
- attune/workflow_commands.py +780 -0
- attune/workflow_patterns/__init__.py +33 -0
- attune/workflow_patterns/behavior.py +249 -0
- attune/workflow_patterns/core.py +76 -0
- attune/workflow_patterns/output.py +99 -0
- attune/workflow_patterns/registry.py +255 -0
- attune/workflow_patterns/structural.py +288 -0
- attune/workflows/__init__.py +539 -0
- attune/workflows/autonomous_test_gen.py +1268 -0
- attune/workflows/base.py +2667 -0
- attune/workflows/batch_processing.py +342 -0
- attune/workflows/bug_predict.py +1084 -0
- attune/workflows/builder.py +273 -0
- attune/workflows/caching.py +253 -0
- attune/workflows/code_review.py +1048 -0
- attune/workflows/code_review_adapters.py +312 -0
- attune/workflows/code_review_pipeline.py +722 -0
- attune/workflows/config.py +645 -0
- attune/workflows/dependency_check.py +644 -0
- attune/workflows/document_gen/__init__.py +25 -0
- attune/workflows/document_gen/config.py +30 -0
- attune/workflows/document_gen/report_formatter.py +162 -0
- attune/workflows/document_gen/workflow.py +1426 -0
- attune/workflows/document_manager.py +216 -0
- attune/workflows/document_manager_README.md +134 -0
- attune/workflows/documentation_orchestrator.py +1205 -0
- attune/workflows/history.py +510 -0
- attune/workflows/keyboard_shortcuts/__init__.py +39 -0
- attune/workflows/keyboard_shortcuts/generators.py +391 -0
- attune/workflows/keyboard_shortcuts/parsers.py +416 -0
- attune/workflows/keyboard_shortcuts/prompts.py +295 -0
- attune/workflows/keyboard_shortcuts/schema.py +193 -0
- attune/workflows/keyboard_shortcuts/workflow.py +509 -0
- attune/workflows/llm_base.py +363 -0
- attune/workflows/manage_docs.py +87 -0
- attune/workflows/manage_docs_README.md +134 -0
- attune/workflows/manage_documentation.py +821 -0
- attune/workflows/new_sample_workflow1.py +149 -0
- attune/workflows/new_sample_workflow1_README.md +150 -0
- attune/workflows/orchestrated_health_check.py +849 -0
- attune/workflows/orchestrated_release_prep.py +600 -0
- attune/workflows/output.py +413 -0
- attune/workflows/perf_audit.py +863 -0
- attune/workflows/pr_review.py +762 -0
- attune/workflows/progress.py +785 -0
- attune/workflows/progress_server.py +322 -0
- attune/workflows/progressive/README 2.md +454 -0
- attune/workflows/progressive/README.md +454 -0
- attune/workflows/progressive/__init__.py +82 -0
- attune/workflows/progressive/cli.py +219 -0
- attune/workflows/progressive/core.py +488 -0
- attune/workflows/progressive/orchestrator.py +723 -0
- attune/workflows/progressive/reports.py +520 -0
- attune/workflows/progressive/telemetry.py +274 -0
- attune/workflows/progressive/test_gen.py +495 -0
- attune/workflows/progressive/workflow.py +589 -0
- attune/workflows/refactor_plan.py +694 -0
- attune/workflows/release_prep.py +895 -0
- attune/workflows/release_prep_crew.py +969 -0
- attune/workflows/research_synthesis.py +404 -0
- attune/workflows/routing.py +168 -0
- attune/workflows/secure_release.py +593 -0
- attune/workflows/security_adapters.py +297 -0
- attune/workflows/security_audit.py +1329 -0
- attune/workflows/security_audit_phase3.py +355 -0
- attune/workflows/seo_optimization.py +633 -0
- attune/workflows/step_config.py +234 -0
- attune/workflows/telemetry_mixin.py +269 -0
- attune/workflows/test5.py +125 -0
- attune/workflows/test5_README.md +158 -0
- attune/workflows/test_coverage_boost_crew.py +849 -0
- attune/workflows/test_gen/__init__.py +52 -0
- attune/workflows/test_gen/ast_analyzer.py +249 -0
- attune/workflows/test_gen/config.py +88 -0
- attune/workflows/test_gen/data_models.py +38 -0
- attune/workflows/test_gen/report_formatter.py +289 -0
- attune/workflows/test_gen/test_templates.py +381 -0
- attune/workflows/test_gen/workflow.py +655 -0
- attune/workflows/test_gen.py +54 -0
- attune/workflows/test_gen_behavioral.py +477 -0
- attune/workflows/test_gen_parallel.py +341 -0
- attune/workflows/test_lifecycle.py +526 -0
- attune/workflows/test_maintenance.py +627 -0
- attune/workflows/test_maintenance_cli.py +590 -0
- attune/workflows/test_maintenance_crew.py +840 -0
- attune/workflows/test_runner.py +622 -0
- attune/workflows/tier_tracking.py +531 -0
- attune/workflows/xml_enhanced_crew.py +285 -0
- attune_ai-2.0.0.dist-info/METADATA +1026 -0
- attune_ai-2.0.0.dist-info/RECORD +457 -0
- attune_ai-2.0.0.dist-info/WHEEL +5 -0
- attune_ai-2.0.0.dist-info/entry_points.txt +26 -0
- attune_ai-2.0.0.dist-info/licenses/LICENSE +201 -0
- attune_ai-2.0.0.dist-info/licenses/LICENSE_CHANGE_ANNOUNCEMENT.md +101 -0
- attune_ai-2.0.0.dist-info/top_level.txt +5 -0
- attune_healthcare/__init__.py +13 -0
- attune_healthcare/monitors/__init__.py +9 -0
- attune_healthcare/monitors/clinical_protocol_monitor.py +315 -0
- attune_healthcare/monitors/monitoring/__init__.py +44 -0
- attune_healthcare/monitors/monitoring/protocol_checker.py +300 -0
- attune_healthcare/monitors/monitoring/protocol_loader.py +214 -0
- attune_healthcare/monitors/monitoring/sensor_parsers.py +306 -0
- attune_healthcare/monitors/monitoring/trajectory_analyzer.py +389 -0
- attune_llm/README.md +553 -0
- attune_llm/__init__.py +28 -0
- attune_llm/agent_factory/__init__.py +53 -0
- attune_llm/agent_factory/adapters/__init__.py +85 -0
- attune_llm/agent_factory/adapters/autogen_adapter.py +312 -0
- attune_llm/agent_factory/adapters/crewai_adapter.py +483 -0
- attune_llm/agent_factory/adapters/haystack_adapter.py +298 -0
- attune_llm/agent_factory/adapters/langchain_adapter.py +362 -0
- attune_llm/agent_factory/adapters/langgraph_adapter.py +333 -0
- attune_llm/agent_factory/adapters/native.py +228 -0
- attune_llm/agent_factory/adapters/wizard_adapter.py +423 -0
- attune_llm/agent_factory/base.py +305 -0
- attune_llm/agent_factory/crews/__init__.py +67 -0
- attune_llm/agent_factory/crews/code_review.py +1113 -0
- attune_llm/agent_factory/crews/health_check.py +1262 -0
- attune_llm/agent_factory/crews/refactoring.py +1128 -0
- attune_llm/agent_factory/crews/security_audit.py +1018 -0
- attune_llm/agent_factory/decorators.py +287 -0
- attune_llm/agent_factory/factory.py +558 -0
- attune_llm/agent_factory/framework.py +193 -0
- attune_llm/agent_factory/memory_integration.py +328 -0
- attune_llm/agent_factory/resilient.py +320 -0
- attune_llm/agents_md/__init__.py +22 -0
- attune_llm/agents_md/loader.py +218 -0
- attune_llm/agents_md/parser.py +271 -0
- attune_llm/agents_md/registry.py +307 -0
- attune_llm/claude_memory.py +466 -0
- attune_llm/cli/__init__.py +8 -0
- attune_llm/cli/sync_claude.py +487 -0
- attune_llm/code_health.py +1313 -0
- attune_llm/commands/__init__.py +51 -0
- attune_llm/commands/context.py +375 -0
- attune_llm/commands/loader.py +301 -0
- attune_llm/commands/models.py +231 -0
- attune_llm/commands/parser.py +371 -0
- attune_llm/commands/registry.py +429 -0
- attune_llm/config/__init__.py +29 -0
- attune_llm/config/unified.py +291 -0
- attune_llm/context/__init__.py +22 -0
- attune_llm/context/compaction.py +455 -0
- attune_llm/context/manager.py +434 -0
- attune_llm/contextual_patterns.py +361 -0
- attune_llm/core.py +907 -0
- attune_llm/git_pattern_extractor.py +435 -0
- attune_llm/hooks/__init__.py +24 -0
- attune_llm/hooks/config.py +306 -0
- attune_llm/hooks/executor.py +289 -0
- attune_llm/hooks/registry.py +302 -0
- attune_llm/hooks/scripts/__init__.py +39 -0
- attune_llm/hooks/scripts/evaluate_session.py +201 -0
- attune_llm/hooks/scripts/first_time_init.py +285 -0
- attune_llm/hooks/scripts/pre_compact.py +207 -0
- attune_llm/hooks/scripts/session_end.py +183 -0
- attune_llm/hooks/scripts/session_start.py +163 -0
- attune_llm/hooks/scripts/suggest_compact.py +225 -0
- attune_llm/learning/__init__.py +30 -0
- attune_llm/learning/evaluator.py +438 -0
- attune_llm/learning/extractor.py +514 -0
- attune_llm/learning/storage.py +560 -0
- attune_llm/levels.py +227 -0
- attune_llm/pattern_confidence.py +414 -0
- attune_llm/pattern_resolver.py +272 -0
- attune_llm/pattern_summary.py +350 -0
- attune_llm/providers.py +967 -0
- attune_llm/routing/__init__.py +32 -0
- attune_llm/routing/model_router.py +362 -0
- attune_llm/security/IMPLEMENTATION_SUMMARY.md +413 -0
- attune_llm/security/PHASE2_COMPLETE.md +384 -0
- attune_llm/security/PHASE2_SECRETS_DETECTOR_COMPLETE.md +271 -0
- attune_llm/security/QUICK_REFERENCE.md +316 -0
- attune_llm/security/README.md +262 -0
- attune_llm/security/__init__.py +62 -0
- attune_llm/security/audit_logger.py +929 -0
- attune_llm/security/audit_logger_example.py +152 -0
- attune_llm/security/pii_scrubber.py +640 -0
- attune_llm/security/secrets_detector.py +678 -0
- attune_llm/security/secrets_detector_example.py +304 -0
- attune_llm/security/secure_memdocs.py +1192 -0
- attune_llm/security/secure_memdocs_example.py +278 -0
- attune_llm/session_status.py +745 -0
- attune_llm/state.py +246 -0
- attune_llm/utils/__init__.py +5 -0
- attune_llm/utils/tokens.py +349 -0
- attune_software/SOFTWARE_PLUGIN_README.md +57 -0
- attune_software/__init__.py +13 -0
- attune_software/cli/__init__.py +120 -0
- attune_software/cli/inspect.py +362 -0
- attune_software/cli.py +574 -0
- attune_software/plugin.py +188 -0
- workflow_scaffolding/__init__.py +11 -0
- workflow_scaffolding/__main__.py +12 -0
- workflow_scaffolding/cli.py +206 -0
- workflow_scaffolding/generator.py +265 -0
attune/cli_router.py
ADDED
|
@@ -0,0 +1,437 @@
|
|
|
1
|
+
"""Hybrid CLI Router - Skills + Natural Language
|
|
2
|
+
|
|
3
|
+
Routes keywords and natural language to Claude Code skill invocations:
|
|
4
|
+
- Skills: /dev, /testing, /workflows (Claude Code Skill tool)
|
|
5
|
+
- Keywords: commit, test, security (maps to skills)
|
|
6
|
+
- Natural language: "commit my changes" (SmartRouter classification)
|
|
7
|
+
|
|
8
|
+
Copyright 2025 Smart-AI-Memory
|
|
9
|
+
Licensed under Fair Source License 0.9
|
|
10
|
+
"""
|
|
11
|
+
|
|
12
|
+
from dataclasses import dataclass
|
|
13
|
+
from pathlib import Path
|
|
14
|
+
from typing import Any
|
|
15
|
+
|
|
16
|
+
import yaml
|
|
17
|
+
|
|
18
|
+
from attune.routing import SmartRouter
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
@dataclass
|
|
22
|
+
class RoutingPreference:
|
|
23
|
+
"""User's learned routing preferences."""
|
|
24
|
+
|
|
25
|
+
keyword: str
|
|
26
|
+
skill: str
|
|
27
|
+
args: str = ""
|
|
28
|
+
usage_count: int = 0
|
|
29
|
+
confidence: float = 1.0
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
class HybridRouter:
|
|
33
|
+
"""Routes user input to Claude Code skill invocations.
|
|
34
|
+
|
|
35
|
+
Supports three input modes:
|
|
36
|
+
1. Skills: /dev, /testing (returns skill invocation metadata)
|
|
37
|
+
2. Keywords: commit, test (maps to skill invocations)
|
|
38
|
+
3. Natural language: "I need to commit" (uses SmartRouter)
|
|
39
|
+
|
|
40
|
+
Example:
|
|
41
|
+
router = HybridRouter()
|
|
42
|
+
|
|
43
|
+
# Skill invocation
|
|
44
|
+
result = await router.route("/dev")
|
|
45
|
+
# → {type: "skill", skill: "dev", args: "", instruction: "Use Skill tool..."}
|
|
46
|
+
|
|
47
|
+
# Keyword to skill
|
|
48
|
+
result = await router.route("commit")
|
|
49
|
+
# → {type: "skill", skill: "dev", args: "commit", instruction: "Use Skill tool..."}
|
|
50
|
+
|
|
51
|
+
# Natural language
|
|
52
|
+
result = await router.route("I want to commit my changes")
|
|
53
|
+
# → {type: "skill", skill: "dev", args: "commit", reasoning: "..."}
|
|
54
|
+
"""
|
|
55
|
+
|
|
56
|
+
def __init__(self, preferences_path: str | None = None):
|
|
57
|
+
"""Initialize hybrid router.
|
|
58
|
+
|
|
59
|
+
Args:
|
|
60
|
+
preferences_path: Path to user preferences YAML
|
|
61
|
+
Default: .attune/routing_preferences.yaml
|
|
62
|
+
"""
|
|
63
|
+
self.preferences_path = Path(
|
|
64
|
+
preferences_path or Path.home() / ".empathy" / "routing_preferences.yaml"
|
|
65
|
+
)
|
|
66
|
+
self.smart_router = SmartRouter()
|
|
67
|
+
self.preferences: dict[str, RoutingPreference] = {}
|
|
68
|
+
|
|
69
|
+
# Keyword to skill mapping: keyword → (skill_name, args)
|
|
70
|
+
self._keyword_to_skill = {
|
|
71
|
+
# Dev commands → /dev skill
|
|
72
|
+
"commit": ("dev", "commit"),
|
|
73
|
+
"review": ("dev", "review"),
|
|
74
|
+
"review-pr": ("dev", "review"),
|
|
75
|
+
"refactor": ("dev", "refactor"),
|
|
76
|
+
"perf": ("dev", "perf-audit"),
|
|
77
|
+
"perf-audit": ("dev", "perf-audit"),
|
|
78
|
+
"debug": ("dev", "debug"),
|
|
79
|
+
# Testing commands → /testing skill
|
|
80
|
+
"test": ("testing", "run"),
|
|
81
|
+
"tests": ("testing", "run"),
|
|
82
|
+
"coverage": ("testing", "coverage"),
|
|
83
|
+
"generate-tests": ("testing", "gen"),
|
|
84
|
+
"test-gen": ("testing", "gen"),
|
|
85
|
+
"benchmark": ("testing", "benchmark"),
|
|
86
|
+
# Learning commands → /learning skill
|
|
87
|
+
"evaluate": ("learning", "evaluate"),
|
|
88
|
+
"patterns": ("learning", "patterns"),
|
|
89
|
+
"improve": ("learning", "improve"),
|
|
90
|
+
# Workflow commands → /workflows skill
|
|
91
|
+
"security": ("workflows", "run security-audit"),
|
|
92
|
+
"security-audit": ("workflows", "run security-audit"),
|
|
93
|
+
"bug-predict": ("workflows", "run bug-predict"),
|
|
94
|
+
"bugs": ("workflows", "run bug-predict"),
|
|
95
|
+
"perf-workflow": ("workflows", "run perf-audit"),
|
|
96
|
+
# Context commands → /context skill
|
|
97
|
+
"status": ("context", "status"),
|
|
98
|
+
"memory": ("context", "memory"),
|
|
99
|
+
"state": ("context", "state"),
|
|
100
|
+
# Doc commands → /docs skill
|
|
101
|
+
"explain": ("docs", "explain"),
|
|
102
|
+
"document": ("docs", "generate"),
|
|
103
|
+
"overview": ("docs", "overview"),
|
|
104
|
+
"docs-seo": ("workflows", "run seo-optimization --mode audit"),
|
|
105
|
+
# SEO commands → /workflows skill
|
|
106
|
+
"seo": ("workflows", "run seo-optimization --mode audit"),
|
|
107
|
+
"seo-audit": ("workflows", "run seo-optimization --mode audit"),
|
|
108
|
+
"seo-optimize": ("workflows", "run seo-optimization --mode fix"),
|
|
109
|
+
"optimize-seo": ("workflows", "run seo-optimization --mode suggest"),
|
|
110
|
+
"meta-tags": ("workflows", "run seo-optimization --mode audit"),
|
|
111
|
+
"check-seo": ("workflows", "run seo-optimization --mode audit"),
|
|
112
|
+
# Plan commands → /plan skill
|
|
113
|
+
"plan": ("plan", ""),
|
|
114
|
+
"tdd": ("plan", "tdd"),
|
|
115
|
+
# Release commands → /release skill
|
|
116
|
+
"release": ("release", "prep"),
|
|
117
|
+
"ship": ("release", "prep"),
|
|
118
|
+
# Authentication strategy commands (CLI)
|
|
119
|
+
"auth-setup": ("workflows", "python -m attune.models.auth_cli setup"),
|
|
120
|
+
"auth-status": ("workflows", "python -m attune.models.auth_cli status"),
|
|
121
|
+
"auth-recommend": ("workflows", "python -m attune.models.auth_cli recommend"),
|
|
122
|
+
"auth-reset": ("workflows", "python -m attune.models.auth_cli reset"),
|
|
123
|
+
"auth": ("workflows", "python -m attune.models.auth_cli status"),
|
|
124
|
+
# Agent dashboard commands
|
|
125
|
+
"dashboard": ("workflows", "python examples/dashboard_demo.py"),
|
|
126
|
+
"agent-dashboard": ("workflows", "python examples/dashboard_demo.py"),
|
|
127
|
+
# Batch test generation (enhanced)
|
|
128
|
+
"batch-tests": ("testing", "generate --batch"),
|
|
129
|
+
"bulk-tests": ("testing", "generate --batch"),
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
# Hub descriptions for disambiguation
|
|
133
|
+
self._hub_descriptions = {
|
|
134
|
+
"dev": "Development tools (commits, reviews, refactoring)",
|
|
135
|
+
"testing": "Test generation and coverage analysis",
|
|
136
|
+
"learning": "Session evaluation and pattern learning",
|
|
137
|
+
"workflows": "AI-powered workflows (security, bugs, performance)",
|
|
138
|
+
"context": "Memory and state management",
|
|
139
|
+
"docs": "Documentation generation",
|
|
140
|
+
"plan": "Development planning and architecture",
|
|
141
|
+
"release": "Release preparation and publishing",
|
|
142
|
+
"utilities": "Utility tools (profiling, dependencies)",
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
self._load_preferences()
|
|
146
|
+
|
|
147
|
+
def _load_preferences(self) -> None:
|
|
148
|
+
"""Load user routing preferences from disk."""
|
|
149
|
+
if not self.preferences_path.exists():
|
|
150
|
+
return
|
|
151
|
+
|
|
152
|
+
try:
|
|
153
|
+
with open(self.preferences_path) as f:
|
|
154
|
+
data = yaml.safe_load(f) or {}
|
|
155
|
+
|
|
156
|
+
for keyword, pref_data in data.get("preferences", {}).items():
|
|
157
|
+
# Handle backward compatibility: old format had "slash_command"
|
|
158
|
+
if "slash_command" in pref_data:
|
|
159
|
+
# Migrate old format: "/dev commit" → skill="dev", args="commit"
|
|
160
|
+
slash_cmd = pref_data["slash_command"].lstrip("/")
|
|
161
|
+
parts = slash_cmd.split(maxsplit=1)
|
|
162
|
+
skill = parts[0] if parts else "help"
|
|
163
|
+
args = parts[1] if len(parts) > 1 else ""
|
|
164
|
+
else:
|
|
165
|
+
# New format
|
|
166
|
+
skill = pref_data["skill"]
|
|
167
|
+
args = pref_data.get("args", "")
|
|
168
|
+
|
|
169
|
+
self.preferences[keyword] = RoutingPreference(
|
|
170
|
+
keyword=keyword,
|
|
171
|
+
skill=skill,
|
|
172
|
+
args=args,
|
|
173
|
+
usage_count=pref_data.get("usage_count", 0),
|
|
174
|
+
confidence=pref_data.get("confidence", 1.0),
|
|
175
|
+
)
|
|
176
|
+
except Exception as e:
|
|
177
|
+
print(f"Warning: Could not load routing preferences: {e}")
|
|
178
|
+
|
|
179
|
+
def _save_preferences(self) -> None:
|
|
180
|
+
"""Save user routing preferences to disk."""
|
|
181
|
+
self.preferences_path.parent.mkdir(parents=True, exist_ok=True)
|
|
182
|
+
|
|
183
|
+
data = {
|
|
184
|
+
"preferences": {
|
|
185
|
+
pref.keyword: {
|
|
186
|
+
"skill": pref.skill,
|
|
187
|
+
"args": pref.args,
|
|
188
|
+
"usage_count": pref.usage_count,
|
|
189
|
+
"confidence": pref.confidence,
|
|
190
|
+
}
|
|
191
|
+
for pref in self.preferences.values()
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
with open(self.preferences_path, "w") as f:
|
|
196
|
+
yaml.dump(data, f, default_flow_style=False)
|
|
197
|
+
|
|
198
|
+
async def route(
|
|
199
|
+
self, user_input: str, context: dict[str, Any] | None = None
|
|
200
|
+
) -> dict[str, Any]:
|
|
201
|
+
"""Route user input to appropriate command or workflow.
|
|
202
|
+
|
|
203
|
+
Args:
|
|
204
|
+
user_input: User's input (slash command, keyword, or natural language)
|
|
205
|
+
context: Optional context (current file, project info, etc.)
|
|
206
|
+
|
|
207
|
+
Returns:
|
|
208
|
+
Routing result with type, command/workflow, and metadata
|
|
209
|
+
"""
|
|
210
|
+
user_input = user_input.strip()
|
|
211
|
+
|
|
212
|
+
# Level 1: Slash command (direct execution)
|
|
213
|
+
if user_input.startswith("/"):
|
|
214
|
+
return self._route_slash_command(user_input)
|
|
215
|
+
|
|
216
|
+
# Level 2: Single word or known command (inference)
|
|
217
|
+
words = user_input.split()
|
|
218
|
+
if len(words) <= 2:
|
|
219
|
+
inferred = self._infer_command(user_input)
|
|
220
|
+
if inferred:
|
|
221
|
+
return inferred
|
|
222
|
+
|
|
223
|
+
# Level 3: Natural language (SmartRouter)
|
|
224
|
+
return await self._route_natural_language(user_input, context)
|
|
225
|
+
|
|
226
|
+
def _route_slash_command(self, command: str) -> dict[str, Any]:
|
|
227
|
+
"""Route slash command to skill invocation.
|
|
228
|
+
|
|
229
|
+
Args:
|
|
230
|
+
command: Slash command like "/dev" or "/dev commit"
|
|
231
|
+
|
|
232
|
+
Returns:
|
|
233
|
+
Skill invocation instructions
|
|
234
|
+
"""
|
|
235
|
+
parts = command[1:].split(maxsplit=1) # Remove leading /
|
|
236
|
+
skill = parts[0] if parts else "help"
|
|
237
|
+
args = parts[1] if len(parts) > 1 else ""
|
|
238
|
+
|
|
239
|
+
return {
|
|
240
|
+
"type": "skill",
|
|
241
|
+
"skill": skill,
|
|
242
|
+
"args": args,
|
|
243
|
+
"original": command,
|
|
244
|
+
"confidence": 1.0,
|
|
245
|
+
"instruction": f"Use Skill tool with skill='{skill}'" + (f", args='{args}'" if args else ""),
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
def _infer_command(self, keyword: str) -> dict[str, Any] | None:
|
|
249
|
+
"""Infer skill invocation from keyword or short phrase.
|
|
250
|
+
|
|
251
|
+
Args:
|
|
252
|
+
keyword: Single word or short phrase
|
|
253
|
+
|
|
254
|
+
Returns:
|
|
255
|
+
Skill invocation instructions if inference successful, None otherwise
|
|
256
|
+
"""
|
|
257
|
+
keyword_lower = keyword.lower().strip()
|
|
258
|
+
|
|
259
|
+
# Check learned preferences first
|
|
260
|
+
if keyword_lower in self.preferences:
|
|
261
|
+
pref = self.preferences[keyword_lower]
|
|
262
|
+
|
|
263
|
+
# Update usage count
|
|
264
|
+
pref.usage_count += 1
|
|
265
|
+
self._save_preferences()
|
|
266
|
+
|
|
267
|
+
return {
|
|
268
|
+
"type": "skill",
|
|
269
|
+
"skill": pref.skill,
|
|
270
|
+
"args": pref.args,
|
|
271
|
+
"original": keyword,
|
|
272
|
+
"confidence": pref.confidence,
|
|
273
|
+
"source": "learned",
|
|
274
|
+
"instruction": f"Use Skill tool with skill='{pref.skill}'" + (f", args='{pref.args}'" if pref.args else ""),
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
# Check built-in keyword map
|
|
278
|
+
if keyword_lower in self._keyword_to_skill:
|
|
279
|
+
skill, args = self._keyword_to_skill[keyword_lower]
|
|
280
|
+
return {
|
|
281
|
+
"type": "skill",
|
|
282
|
+
"skill": skill,
|
|
283
|
+
"args": args,
|
|
284
|
+
"original": keyword,
|
|
285
|
+
"confidence": 0.9,
|
|
286
|
+
"source": "builtin",
|
|
287
|
+
"instruction": f"Use Skill tool with skill='{skill}'" + (f", args='{args}'" if args else ""),
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
# Check for hub names (show hub menu)
|
|
291
|
+
if keyword_lower in self._hub_descriptions:
|
|
292
|
+
return {
|
|
293
|
+
"type": "skill",
|
|
294
|
+
"skill": keyword_lower,
|
|
295
|
+
"args": "",
|
|
296
|
+
"original": keyword,
|
|
297
|
+
"confidence": 1.0,
|
|
298
|
+
"source": "hub",
|
|
299
|
+
"instruction": f"Use Skill tool with skill='{keyword_lower}'",
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
return None
|
|
303
|
+
|
|
304
|
+
async def _route_natural_language(
|
|
305
|
+
self, text: str, context: dict[str, Any] | None = None
|
|
306
|
+
) -> dict[str, Any]:
|
|
307
|
+
"""Route natural language input using SmartRouter.
|
|
308
|
+
|
|
309
|
+
Args:
|
|
310
|
+
text: Natural language input
|
|
311
|
+
context: Optional context
|
|
312
|
+
|
|
313
|
+
Returns:
|
|
314
|
+
Skill invocation instructions based on SmartRouter decision
|
|
315
|
+
"""
|
|
316
|
+
# Use SmartRouter for classification
|
|
317
|
+
decision = await self.smart_router.route(text, context)
|
|
318
|
+
|
|
319
|
+
# Map workflow to skill invocation
|
|
320
|
+
skill, args = self._workflow_to_skill(decision.primary_workflow)
|
|
321
|
+
|
|
322
|
+
return {
|
|
323
|
+
"type": "skill",
|
|
324
|
+
"skill": skill,
|
|
325
|
+
"args": args,
|
|
326
|
+
"workflow": decision.primary_workflow,
|
|
327
|
+
"secondary_workflows": decision.secondary_workflows,
|
|
328
|
+
"confidence": decision.confidence,
|
|
329
|
+
"reasoning": decision.reasoning,
|
|
330
|
+
"original": text,
|
|
331
|
+
"source": "natural_language",
|
|
332
|
+
"instruction": f"Use Skill tool with skill='{skill}'" + (f", args='{args}'" if args else ""),
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
def _workflow_to_skill(self, workflow: str) -> tuple[str, str]:
|
|
336
|
+
"""Map workflow name to skill invocation.
|
|
337
|
+
|
|
338
|
+
Args:
|
|
339
|
+
workflow: Workflow name (e.g., "security-audit")
|
|
340
|
+
|
|
341
|
+
Returns:
|
|
342
|
+
Tuple of (skill_name, args)
|
|
343
|
+
"""
|
|
344
|
+
# Workflow to skill mapping
|
|
345
|
+
workflow_map = {
|
|
346
|
+
"security-audit": ("workflows", "run security-audit"),
|
|
347
|
+
"bug-predict": ("workflows", "run bug-predict"),
|
|
348
|
+
"code-review": ("dev", "review"),
|
|
349
|
+
"test-gen": ("testing", "gen"),
|
|
350
|
+
"perf-audit": ("workflows", "run perf-audit"),
|
|
351
|
+
"seo-optimization": ("workflows", "run seo-optimization"),
|
|
352
|
+
"commit": ("dev", "commit"),
|
|
353
|
+
"refactor": ("dev", "refactor"),
|
|
354
|
+
"debug": ("dev", "debug"),
|
|
355
|
+
"explain": ("docs", "explain"),
|
|
356
|
+
"plan": ("plan", ""),
|
|
357
|
+
}
|
|
358
|
+
|
|
359
|
+
return workflow_map.get(workflow, ("workflows", f"run {workflow}"))
|
|
360
|
+
|
|
361
|
+
def learn_preference(self, keyword: str, skill: str, args: str = "") -> None:
|
|
362
|
+
"""Learn user's routing preference.
|
|
363
|
+
|
|
364
|
+
Args:
|
|
365
|
+
keyword: Keyword user typed
|
|
366
|
+
skill: Skill name that was invoked
|
|
367
|
+
args: Arguments passed to skill
|
|
368
|
+
"""
|
|
369
|
+
if keyword in self.preferences:
|
|
370
|
+
pref = self.preferences[keyword]
|
|
371
|
+
pref.usage_count += 1
|
|
372
|
+
# Increase confidence with repeated usage
|
|
373
|
+
pref.confidence = min(1.0, pref.confidence + 0.05)
|
|
374
|
+
else:
|
|
375
|
+
self.preferences[keyword] = RoutingPreference(
|
|
376
|
+
keyword=keyword,
|
|
377
|
+
skill=skill,
|
|
378
|
+
args=args,
|
|
379
|
+
usage_count=1,
|
|
380
|
+
confidence=0.8,
|
|
381
|
+
)
|
|
382
|
+
|
|
383
|
+
self._save_preferences()
|
|
384
|
+
|
|
385
|
+
def get_suggestions(self, partial: str) -> list[str]:
|
|
386
|
+
"""Get command suggestions based on partial input.
|
|
387
|
+
|
|
388
|
+
Args:
|
|
389
|
+
partial: Partial command input
|
|
390
|
+
|
|
391
|
+
Returns:
|
|
392
|
+
List of suggested keywords and skills
|
|
393
|
+
"""
|
|
394
|
+
suggestions = []
|
|
395
|
+
partial_lower = partial.lower()
|
|
396
|
+
|
|
397
|
+
# Suggest keywords
|
|
398
|
+
for keyword in self._keyword_to_skill.keys():
|
|
399
|
+
if partial_lower in keyword:
|
|
400
|
+
skill, args = self._keyword_to_skill[keyword]
|
|
401
|
+
suggestions.append(f"{keyword} → /{skill} {args}".strip())
|
|
402
|
+
|
|
403
|
+
# Suggest learned preferences
|
|
404
|
+
for pref in self.preferences.values():
|
|
405
|
+
if partial_lower in pref.keyword.lower():
|
|
406
|
+
suggestions.append(f"{pref.keyword} → /{pref.skill} {pref.args}".strip())
|
|
407
|
+
|
|
408
|
+
return suggestions[:5] # Top 5 suggestions
|
|
409
|
+
|
|
410
|
+
|
|
411
|
+
# Convenience functions
|
|
412
|
+
async def route_user_input(
|
|
413
|
+
user_input: str, context: dict[str, Any] | None = None
|
|
414
|
+
) -> dict[str, Any]:
|
|
415
|
+
"""Quick routing helper.
|
|
416
|
+
|
|
417
|
+
Args:
|
|
418
|
+
user_input: User's input
|
|
419
|
+
context: Optional context
|
|
420
|
+
|
|
421
|
+
Returns:
|
|
422
|
+
Routing result
|
|
423
|
+
"""
|
|
424
|
+
router = HybridRouter()
|
|
425
|
+
return await router.route(user_input, context)
|
|
426
|
+
|
|
427
|
+
|
|
428
|
+
def is_slash_command(text: str) -> bool:
|
|
429
|
+
"""Check if text is a slash command.
|
|
430
|
+
|
|
431
|
+
Args:
|
|
432
|
+
text: Input text
|
|
433
|
+
|
|
434
|
+
Returns:
|
|
435
|
+
True if slash command, False otherwise
|
|
436
|
+
"""
|
|
437
|
+
return text.strip().startswith("/")
|