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
|
@@ -0,0 +1,436 @@
|
|
|
1
|
+
"""Inspect commands for patterns, metrics, and interactive REPL.
|
|
2
|
+
|
|
3
|
+
Copyright 2025 Smart-AI-Memory
|
|
4
|
+
Licensed under Fair Source License 0.9
|
|
5
|
+
"""
|
|
6
|
+
|
|
7
|
+
import heapq
|
|
8
|
+
import sys
|
|
9
|
+
import time
|
|
10
|
+
|
|
11
|
+
from attune.config import EmpathyConfig, _validate_file_path, load_config
|
|
12
|
+
from attune.core import EmpathyOS
|
|
13
|
+
from attune.logging_config import get_logger
|
|
14
|
+
from attune.metrics.collector import MetricsCollector
|
|
15
|
+
from attune.pattern_library import PatternLibrary
|
|
16
|
+
from attune.persistence import PatternPersistence, StateManager
|
|
17
|
+
|
|
18
|
+
logger = get_logger(__name__)
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
def cmd_run(args):
|
|
22
|
+
"""Interactive REPL for testing empathy interactions.
|
|
23
|
+
|
|
24
|
+
Starts an interactive session for testing empathy levels and features.
|
|
25
|
+
|
|
26
|
+
Args:
|
|
27
|
+
args: Namespace object from argparse with attributes:
|
|
28
|
+
- config (str | None): Path to configuration file.
|
|
29
|
+
- user_id (str | None): User ID (default: cli_user).
|
|
30
|
+
- level (int): Target empathy level (1-5).
|
|
31
|
+
|
|
32
|
+
Returns:
|
|
33
|
+
None: Runs interactive REPL until user exits.
|
|
34
|
+
"""
|
|
35
|
+
config_file = args.config
|
|
36
|
+
user_id = args.user_id or "cli_user"
|
|
37
|
+
level = args.level
|
|
38
|
+
|
|
39
|
+
print("🧠 Empathy Framework - Interactive Mode")
|
|
40
|
+
print("=" * 50)
|
|
41
|
+
|
|
42
|
+
# Load configuration
|
|
43
|
+
if config_file:
|
|
44
|
+
config = load_config(filepath=config_file)
|
|
45
|
+
print(f"✓ Loaded config from: {config_file}")
|
|
46
|
+
else:
|
|
47
|
+
config = EmpathyConfig(user_id=user_id, target_level=level)
|
|
48
|
+
print("✓ Using default configuration")
|
|
49
|
+
|
|
50
|
+
print(f"\nUser ID: {config.user_id}")
|
|
51
|
+
print(f"Target Level: {config.target_level}")
|
|
52
|
+
print(f"Confidence Threshold: {config.confidence_threshold:.0%}")
|
|
53
|
+
|
|
54
|
+
# Create EmpathyOS instance
|
|
55
|
+
try:
|
|
56
|
+
empathy = EmpathyOS(
|
|
57
|
+
user_id=config.user_id,
|
|
58
|
+
target_level=config.target_level,
|
|
59
|
+
confidence_threshold=config.confidence_threshold,
|
|
60
|
+
persistence_enabled=config.persistence_enabled,
|
|
61
|
+
)
|
|
62
|
+
print("✓ Empathy OS initialized")
|
|
63
|
+
except ValueError as e:
|
|
64
|
+
# Invalid configuration parameters
|
|
65
|
+
print(f"✗ Configuration error: {e}")
|
|
66
|
+
sys.exit(1)
|
|
67
|
+
except (OSError, FileNotFoundError, PermissionError) as e:
|
|
68
|
+
# Cannot access required files/directories
|
|
69
|
+
print(f"✗ File system error: {e}")
|
|
70
|
+
sys.exit(1)
|
|
71
|
+
except Exception as e:
|
|
72
|
+
# Unexpected initialization failure
|
|
73
|
+
logger.exception(f"Unexpected error initializing Empathy OS: {e}")
|
|
74
|
+
print(f"✗ Failed to initialize Empathy OS: {e}")
|
|
75
|
+
sys.exit(1)
|
|
76
|
+
|
|
77
|
+
print("\n" + "=" * 50)
|
|
78
|
+
print("Type your input (or 'exit'/'quit' to stop)")
|
|
79
|
+
print("Type 'help' for available commands")
|
|
80
|
+
print("=" * 50 + "\n")
|
|
81
|
+
|
|
82
|
+
# Interactive loop
|
|
83
|
+
while True:
|
|
84
|
+
try:
|
|
85
|
+
user_input = input("You: ").strip()
|
|
86
|
+
|
|
87
|
+
if not user_input:
|
|
88
|
+
continue
|
|
89
|
+
|
|
90
|
+
if user_input.lower() in ["exit", "quit", "q"]:
|
|
91
|
+
print("\n👋 Goodbye!")
|
|
92
|
+
break
|
|
93
|
+
|
|
94
|
+
if user_input.lower() == "help":
|
|
95
|
+
print("\nAvailable commands:")
|
|
96
|
+
print(" exit, quit, q - Exit the program")
|
|
97
|
+
print(" help - Show this help message")
|
|
98
|
+
print(" trust - Show current trust level")
|
|
99
|
+
print(" stats - Show session statistics")
|
|
100
|
+
print(" level - Show current empathy level")
|
|
101
|
+
print()
|
|
102
|
+
continue
|
|
103
|
+
|
|
104
|
+
if user_input.lower() == "trust":
|
|
105
|
+
trust = empathy.collaboration_state.trust_level
|
|
106
|
+
print(f"\n Current trust level: {trust:.0%}\n")
|
|
107
|
+
continue
|
|
108
|
+
|
|
109
|
+
if user_input.lower() == "level":
|
|
110
|
+
current_level = empathy.collaboration_state.current_level
|
|
111
|
+
print(f"\n Current empathy level: {current_level}\n")
|
|
112
|
+
continue
|
|
113
|
+
|
|
114
|
+
if user_input.lower() == "stats":
|
|
115
|
+
print("\n Session Statistics:")
|
|
116
|
+
print(f" Trust: {empathy.collaboration_state.trust_level:.0%}")
|
|
117
|
+
print(f" Current Level: {empathy.collaboration_state.current_level}")
|
|
118
|
+
print(f" Target Level: {config.target_level}")
|
|
119
|
+
print()
|
|
120
|
+
continue
|
|
121
|
+
|
|
122
|
+
# Process interaction
|
|
123
|
+
start_time = time.time()
|
|
124
|
+
response = empathy.interact(user_id=config.user_id, user_input=user_input, context={})
|
|
125
|
+
duration = (time.time() - start_time) * 1000
|
|
126
|
+
|
|
127
|
+
# Display response with level indicator
|
|
128
|
+
level_indicators = ["❌", "🔵", "🟢", "🟡", "🔮"]
|
|
129
|
+
level_indicator = level_indicators[response.level]
|
|
130
|
+
|
|
131
|
+
print(f"\nBot {level_indicator} [L{response.level}]: {response.response}")
|
|
132
|
+
|
|
133
|
+
# Show predictions if Level 4
|
|
134
|
+
if response.predictions:
|
|
135
|
+
print("\n🔮 Predictions:")
|
|
136
|
+
for pred in response.predictions:
|
|
137
|
+
print(f" • {pred}")
|
|
138
|
+
|
|
139
|
+
conf = f"{response.confidence:.0%}"
|
|
140
|
+
print(f"\n Level: {response.level} | Confidence: {conf} | Time: {duration:.0f}ms")
|
|
141
|
+
print()
|
|
142
|
+
|
|
143
|
+
# Ask for feedback
|
|
144
|
+
feedback = input("Was this helpful? (y/n/skip): ").strip().lower()
|
|
145
|
+
if feedback == "y":
|
|
146
|
+
empathy.record_success(success=True)
|
|
147
|
+
trust = empathy.collaboration_state.trust_level
|
|
148
|
+
print(f" ✓ Trust increased to {trust:.0%}\n")
|
|
149
|
+
elif feedback == "n":
|
|
150
|
+
empathy.record_success(success=False)
|
|
151
|
+
trust = empathy.collaboration_state.trust_level
|
|
152
|
+
print(f" ✗ Trust decreased to {trust:.0%}\n")
|
|
153
|
+
|
|
154
|
+
except KeyboardInterrupt:
|
|
155
|
+
print("\n\n👋 Goodbye!")
|
|
156
|
+
break
|
|
157
|
+
except (ValueError, KeyError) as e:
|
|
158
|
+
# Invalid input or response structure
|
|
159
|
+
print(f"\n✗ Input error: {e}\n")
|
|
160
|
+
except Exception as e:
|
|
161
|
+
# Unexpected errors in interactive loop - log and continue
|
|
162
|
+
logger.exception(f"Unexpected error in interactive loop: {e}")
|
|
163
|
+
print(f"\n✗ Error: {e}\n")
|
|
164
|
+
|
|
165
|
+
|
|
166
|
+
def cmd_inspect(args):
|
|
167
|
+
"""Unified inspection command for patterns, metrics, and state.
|
|
168
|
+
|
|
169
|
+
Inspect various framework data including patterns, user metrics, and states.
|
|
170
|
+
|
|
171
|
+
Args:
|
|
172
|
+
args: Namespace object from argparse with attributes:
|
|
173
|
+
- type (str): What to inspect ('patterns', 'metrics', or 'state').
|
|
174
|
+
- user_id (str | None): Filter by user ID.
|
|
175
|
+
- db (str | None): Database path (default: .attune/patterns.db).
|
|
176
|
+
- state_dir (str | None): State directory for state inspection.
|
|
177
|
+
|
|
178
|
+
Returns:
|
|
179
|
+
None: Prints inspection results. Exits with code 1 on failure.
|
|
180
|
+
"""
|
|
181
|
+
inspect_type = args.type
|
|
182
|
+
user_id = args.user_id
|
|
183
|
+
db_path = args.db or ".attune/patterns.db"
|
|
184
|
+
|
|
185
|
+
print(f"🔍 Inspecting: {inspect_type}")
|
|
186
|
+
print("=" * 50)
|
|
187
|
+
|
|
188
|
+
if inspect_type == "patterns":
|
|
189
|
+
try:
|
|
190
|
+
# Determine file format from extension
|
|
191
|
+
if db_path.endswith(".json"):
|
|
192
|
+
library = PatternPersistence.load_from_json(db_path)
|
|
193
|
+
else:
|
|
194
|
+
library = PatternPersistence.load_from_sqlite(db_path)
|
|
195
|
+
|
|
196
|
+
patterns = list(library.patterns.values())
|
|
197
|
+
|
|
198
|
+
# Filter by user_id if specified
|
|
199
|
+
if user_id:
|
|
200
|
+
patterns = [p for p in patterns if p.agent_id == user_id]
|
|
201
|
+
|
|
202
|
+
print(f"\nPatterns for {'user ' + user_id if user_id else 'all users'}:")
|
|
203
|
+
print(f" Total patterns: {len(patterns)}")
|
|
204
|
+
|
|
205
|
+
if patterns:
|
|
206
|
+
print("\n Top patterns:")
|
|
207
|
+
# Sort by confidence
|
|
208
|
+
top_patterns = heapq.nlargest(10, patterns, key=lambda p: p.confidence)
|
|
209
|
+
for i, pattern in enumerate(top_patterns, 1):
|
|
210
|
+
print(f"\n {i}. {pattern.name}")
|
|
211
|
+
print(f" Confidence: {pattern.confidence:.0%}")
|
|
212
|
+
print(f" Used: {pattern.usage_count} times")
|
|
213
|
+
print(f" Success rate: {pattern.success_rate:.0%}")
|
|
214
|
+
except FileNotFoundError:
|
|
215
|
+
print(f"✗ Pattern library not found: {db_path}")
|
|
216
|
+
print(" Tip: Use 'empathy-framework workflow' to set up your first project")
|
|
217
|
+
sys.exit(1)
|
|
218
|
+
except (ValueError, KeyError) as e:
|
|
219
|
+
# Invalid pattern data format
|
|
220
|
+
print(f"✗ Invalid pattern data: {e}")
|
|
221
|
+
sys.exit(1)
|
|
222
|
+
except Exception as e:
|
|
223
|
+
# Unexpected errors loading patterns
|
|
224
|
+
logger.exception(f"Unexpected error loading patterns: {e}")
|
|
225
|
+
print(f"✗ Failed to load patterns: {e}")
|
|
226
|
+
sys.exit(1)
|
|
227
|
+
|
|
228
|
+
elif inspect_type == "metrics":
|
|
229
|
+
if not user_id:
|
|
230
|
+
print("✗ User ID required for metrics inspection")
|
|
231
|
+
print(" Usage: empathy-framework inspect metrics --user-id USER_ID")
|
|
232
|
+
sys.exit(1)
|
|
233
|
+
|
|
234
|
+
try:
|
|
235
|
+
collector = MetricsCollector(db_path=db_path)
|
|
236
|
+
stats = collector.get_user_stats(user_id)
|
|
237
|
+
|
|
238
|
+
print(f"\nMetrics for user: {user_id}")
|
|
239
|
+
print(f" Total operations: {stats.get('total_operations', 0)}")
|
|
240
|
+
print(f" Success rate: {stats.get('success_rate', 0):.0%}")
|
|
241
|
+
print(f" Average response time: {stats.get('avg_response_time_ms', 0):.0f}ms")
|
|
242
|
+
print("\n Empathy level usage:")
|
|
243
|
+
for level in range(1, 6):
|
|
244
|
+
count = stats.get(f"level_{level}_count", 0)
|
|
245
|
+
print(f" Level {level}: {count} times")
|
|
246
|
+
except (OSError, FileNotFoundError) as e:
|
|
247
|
+
# Database file not found
|
|
248
|
+
print(f"✗ Metrics database not found: {e}")
|
|
249
|
+
sys.exit(1)
|
|
250
|
+
except KeyError as e:
|
|
251
|
+
# User not found
|
|
252
|
+
print(f"✗ User {user_id} not found: {e}")
|
|
253
|
+
sys.exit(1)
|
|
254
|
+
except Exception as e:
|
|
255
|
+
# Unexpected errors loading metrics
|
|
256
|
+
logger.exception(f"Unexpected error loading metrics: {e}")
|
|
257
|
+
print(f"✗ Failed to load metrics: {e}")
|
|
258
|
+
sys.exit(1)
|
|
259
|
+
|
|
260
|
+
elif inspect_type == "state":
|
|
261
|
+
state_dir = args.state_dir or ".attune/state"
|
|
262
|
+
try:
|
|
263
|
+
manager = StateManager(state_dir)
|
|
264
|
+
users = manager.list_users()
|
|
265
|
+
|
|
266
|
+
print("\nSaved states:")
|
|
267
|
+
print(f" Total users: {len(users)}")
|
|
268
|
+
|
|
269
|
+
if users:
|
|
270
|
+
print("\n Users:")
|
|
271
|
+
for uid in users:
|
|
272
|
+
print(f" • {uid}")
|
|
273
|
+
except (OSError, FileNotFoundError) as e:
|
|
274
|
+
# State directory not found
|
|
275
|
+
print(f"✗ State directory not found: {e}")
|
|
276
|
+
sys.exit(1)
|
|
277
|
+
except Exception as e:
|
|
278
|
+
# Unexpected errors loading state
|
|
279
|
+
logger.exception(f"Unexpected error loading state: {e}")
|
|
280
|
+
print(f"✗ Failed to load state: {e}")
|
|
281
|
+
sys.exit(1)
|
|
282
|
+
|
|
283
|
+
print()
|
|
284
|
+
|
|
285
|
+
|
|
286
|
+
def cmd_export(args):
|
|
287
|
+
"""Export patterns to file for sharing/backup.
|
|
288
|
+
|
|
289
|
+
Args:
|
|
290
|
+
args: Namespace object from argparse with attributes:
|
|
291
|
+
- output (str): Output file path.
|
|
292
|
+
- user_id (str | None): Filter patterns by user ID.
|
|
293
|
+
- db (str | None): Source database path.
|
|
294
|
+
- format (str): Output format ('json').
|
|
295
|
+
|
|
296
|
+
Returns:
|
|
297
|
+
None: Exports patterns to file. Exits with code 1 on failure.
|
|
298
|
+
|
|
299
|
+
Raises:
|
|
300
|
+
ValueError: If output path is invalid or unsafe.
|
|
301
|
+
"""
|
|
302
|
+
output_file = args.output
|
|
303
|
+
user_id = args.user_id
|
|
304
|
+
db_path = args.db or ".attune/patterns.db"
|
|
305
|
+
format_type = args.format
|
|
306
|
+
|
|
307
|
+
print(f"📦 Exporting patterns to: {output_file}")
|
|
308
|
+
print("=" * 50)
|
|
309
|
+
|
|
310
|
+
try:
|
|
311
|
+
# Load pattern library from source file
|
|
312
|
+
if db_path.endswith(".json"):
|
|
313
|
+
library = PatternPersistence.load_from_json(db_path)
|
|
314
|
+
else:
|
|
315
|
+
library = PatternPersistence.load_from_sqlite(db_path)
|
|
316
|
+
|
|
317
|
+
patterns = list(library.patterns.values())
|
|
318
|
+
|
|
319
|
+
# Filter by user_id if specified
|
|
320
|
+
if user_id:
|
|
321
|
+
patterns = [p for p in patterns if p.agent_id == user_id]
|
|
322
|
+
|
|
323
|
+
print(f" Found {len(patterns)} patterns")
|
|
324
|
+
|
|
325
|
+
# Validate output path
|
|
326
|
+
validated_output = _validate_file_path(output_file)
|
|
327
|
+
|
|
328
|
+
if format_type == "json":
|
|
329
|
+
# Create filtered library if user_id specified
|
|
330
|
+
if user_id:
|
|
331
|
+
filtered_library = PatternLibrary()
|
|
332
|
+
for pattern in patterns:
|
|
333
|
+
filtered_library.contribute_pattern(pattern.agent_id, pattern)
|
|
334
|
+
else:
|
|
335
|
+
filtered_library = library
|
|
336
|
+
|
|
337
|
+
# Export as JSON
|
|
338
|
+
PatternPersistence.save_to_json(filtered_library, str(validated_output))
|
|
339
|
+
print(f" ✓ Exported {len(patterns)} patterns to {output_file}")
|
|
340
|
+
else:
|
|
341
|
+
print(f"✗ Unsupported format: {format_type}")
|
|
342
|
+
sys.exit(1)
|
|
343
|
+
|
|
344
|
+
except FileNotFoundError:
|
|
345
|
+
print(f"✗ Source file not found: {db_path}")
|
|
346
|
+
print(" Tip: Patterns are saved automatically when using the framework")
|
|
347
|
+
sys.exit(1)
|
|
348
|
+
except (OSError, PermissionError) as e:
|
|
349
|
+
# Cannot write output file
|
|
350
|
+
print(f"✗ Cannot write to file: {e}")
|
|
351
|
+
sys.exit(1)
|
|
352
|
+
except (ValueError, KeyError) as e:
|
|
353
|
+
# Invalid pattern data
|
|
354
|
+
print(f"✗ Invalid pattern data: {e}")
|
|
355
|
+
sys.exit(1)
|
|
356
|
+
except Exception as e:
|
|
357
|
+
# Unexpected errors during export
|
|
358
|
+
logger.exception(f"Unexpected error exporting patterns: {e}")
|
|
359
|
+
print(f"✗ Export failed: {e}")
|
|
360
|
+
sys.exit(1)
|
|
361
|
+
|
|
362
|
+
print()
|
|
363
|
+
|
|
364
|
+
|
|
365
|
+
def cmd_import(args):
|
|
366
|
+
"""Import patterns from file (local dev only - SQLite/JSON).
|
|
367
|
+
|
|
368
|
+
Merges imported patterns into existing pattern library.
|
|
369
|
+
|
|
370
|
+
Args:
|
|
371
|
+
args: Namespace object from argparse with attributes:
|
|
372
|
+
- input (str): Input file path.
|
|
373
|
+
- db (str | None): Target database path (default: .attune/patterns.db).
|
|
374
|
+
|
|
375
|
+
Returns:
|
|
376
|
+
None: Imports and merges patterns. Exits with code 1 on failure.
|
|
377
|
+
"""
|
|
378
|
+
input_file = args.input
|
|
379
|
+
db_path = args.db or ".attune/patterns.db"
|
|
380
|
+
|
|
381
|
+
print(f"📥 Importing patterns from: {input_file}")
|
|
382
|
+
print("=" * 50)
|
|
383
|
+
|
|
384
|
+
try:
|
|
385
|
+
# Load patterns from input file
|
|
386
|
+
if input_file.endswith(".json"):
|
|
387
|
+
imported_library = PatternPersistence.load_from_json(input_file)
|
|
388
|
+
else:
|
|
389
|
+
imported_library = PatternPersistence.load_from_sqlite(input_file)
|
|
390
|
+
|
|
391
|
+
pattern_count = len(imported_library.patterns)
|
|
392
|
+
print(f" Found {pattern_count} patterns in file")
|
|
393
|
+
|
|
394
|
+
# Load existing library if it exists, otherwise create new one
|
|
395
|
+
try:
|
|
396
|
+
if db_path.endswith(".json"):
|
|
397
|
+
existing_library = PatternPersistence.load_from_json(db_path)
|
|
398
|
+
else:
|
|
399
|
+
existing_library = PatternPersistence.load_from_sqlite(db_path)
|
|
400
|
+
|
|
401
|
+
print(f" Existing library has {len(existing_library.patterns)} patterns")
|
|
402
|
+
except FileNotFoundError:
|
|
403
|
+
existing_library = PatternLibrary()
|
|
404
|
+
print(" Creating new pattern library")
|
|
405
|
+
|
|
406
|
+
# Merge imported patterns into existing library
|
|
407
|
+
for pattern in imported_library.patterns.values():
|
|
408
|
+
existing_library.contribute_pattern(pattern.agent_id, pattern)
|
|
409
|
+
|
|
410
|
+
# Save merged library (SQLite for local dev)
|
|
411
|
+
if db_path.endswith(".json"):
|
|
412
|
+
PatternPersistence.save_to_json(existing_library, db_path)
|
|
413
|
+
else:
|
|
414
|
+
PatternPersistence.save_to_sqlite(existing_library, db_path)
|
|
415
|
+
|
|
416
|
+
print(f" ✓ Imported {pattern_count} patterns")
|
|
417
|
+
print(f" ✓ Total patterns in library: {len(existing_library.patterns)}")
|
|
418
|
+
|
|
419
|
+
except FileNotFoundError:
|
|
420
|
+
print(f"✗ Input file not found: {input_file}")
|
|
421
|
+
sys.exit(1)
|
|
422
|
+
except (ValueError, KeyError) as e:
|
|
423
|
+
# Invalid pattern data format
|
|
424
|
+
print(f"✗ Invalid pattern data: {e}")
|
|
425
|
+
sys.exit(1)
|
|
426
|
+
except (OSError, PermissionError) as e:
|
|
427
|
+
# Cannot read input or write to database
|
|
428
|
+
print(f"✗ File access error: {e}")
|
|
429
|
+
sys.exit(1)
|
|
430
|
+
except Exception as e:
|
|
431
|
+
# Unexpected errors during import
|
|
432
|
+
logger.exception(f"Unexpected error importing patterns: {e}")
|
|
433
|
+
print(f"✗ Import failed: {e}")
|
|
434
|
+
sys.exit(1)
|
|
435
|
+
|
|
436
|
+
print()
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
"""Code inspection commands.
|
|
2
|
+
|
|
3
|
+
Commands for scanning and inspecting codebases for issues.
|
|
4
|
+
|
|
5
|
+
Copyright 2025 Smart-AI-Memory
|
|
6
|
+
Licensed under Fair Source License 0.9
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
import subprocess
|
|
10
|
+
from pathlib import Path
|
|
11
|
+
|
|
12
|
+
from attune.cli.core import console
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
def scan(
|
|
16
|
+
path: Path = Path("."),
|
|
17
|
+
format_out: str = "text",
|
|
18
|
+
fix: bool = False,
|
|
19
|
+
staged: bool = False,
|
|
20
|
+
) -> None:
|
|
21
|
+
"""Scan codebase for issues using ruff and bandit."""
|
|
22
|
+
console.print(f"[bold blue]Scanning {path}...[/bold blue]\n")
|
|
23
|
+
|
|
24
|
+
# Run ruff for linting
|
|
25
|
+
console.print("[bold]Running ruff (linting)...[/bold]")
|
|
26
|
+
ruff_args = ["ruff", "check", str(path)]
|
|
27
|
+
if fix:
|
|
28
|
+
ruff_args.append("--fix")
|
|
29
|
+
subprocess.run(ruff_args, check=False)
|
|
30
|
+
|
|
31
|
+
# Run bandit for security (if available)
|
|
32
|
+
console.print("\n[bold]Running bandit (security)...[/bold]")
|
|
33
|
+
bandit_args = ["bandit", "-r", str(path), "-q"]
|
|
34
|
+
if format_out == "json":
|
|
35
|
+
bandit_args.extend(["-f", "json"])
|
|
36
|
+
result = subprocess.run(bandit_args, check=False, capture_output=True)
|
|
37
|
+
if result.returncode == 0:
|
|
38
|
+
console.print("[green]No security issues found[/green]")
|
|
39
|
+
elif result.stdout:
|
|
40
|
+
console.print(result.stdout.decode())
|
|
41
|
+
|
|
42
|
+
console.print("\n[bold green]Scan complete![/bold green]")
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
def inspect_cmd(
|
|
46
|
+
path: Path = Path("."),
|
|
47
|
+
format_out: str = "text",
|
|
48
|
+
) -> None:
|
|
49
|
+
"""Deep inspection with code analysis."""
|
|
50
|
+
args = ["empathy-inspect", str(path)]
|
|
51
|
+
if format_out != "text":
|
|
52
|
+
args.extend(["--format", format_out])
|
|
53
|
+
|
|
54
|
+
result = subprocess.run(args, check=False, capture_output=False)
|
|
55
|
+
if result.returncode != 0:
|
|
56
|
+
console.print("[yellow]Note: empathy-inspect may not be installed[/yellow]")
|
|
57
|
+
console.print("Install with: pip install empathy-framework[software]")
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
"""Memory system control panel commands.
|
|
2
|
+
|
|
3
|
+
Commands for managing Redis-backed short-term memory system.
|
|
4
|
+
|
|
5
|
+
Copyright 2025 Smart-AI-Memory
|
|
6
|
+
Licensed under Fair Source License 0.9
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
import subprocess
|
|
10
|
+
import sys
|
|
11
|
+
|
|
12
|
+
import typer
|
|
13
|
+
|
|
14
|
+
# Create the memory Typer app
|
|
15
|
+
memory_app = typer.Typer(help="Memory system control panel")
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
@memory_app.command("status")
|
|
19
|
+
def memory_status() -> None:
|
|
20
|
+
"""Check memory system status (Redis, patterns, stats)."""
|
|
21
|
+
subprocess.run([sys.executable, "-m", "attune.memory.control_panel", "status"], check=False)
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
@memory_app.command("start")
|
|
25
|
+
def memory_start() -> None:
|
|
26
|
+
"""Start Redis server for short-term memory."""
|
|
27
|
+
subprocess.run([sys.executable, "-m", "attune.memory.control_panel", "start"], check=False)
|
|
28
|
+
|
|
29
|
+
|
|
30
|
+
@memory_app.command("stop")
|
|
31
|
+
def memory_stop() -> None:
|
|
32
|
+
"""Stop Redis server."""
|
|
33
|
+
subprocess.run([sys.executable, "-m", "attune.memory.control_panel", "stop"], check=False)
|
|
34
|
+
|
|
35
|
+
|
|
36
|
+
@memory_app.command("stats")
|
|
37
|
+
def memory_stats() -> None:
|
|
38
|
+
"""Show memory statistics."""
|
|
39
|
+
subprocess.run([sys.executable, "-m", "attune.memory.control_panel", "stats"], check=False)
|
|
40
|
+
|
|
41
|
+
|
|
42
|
+
@memory_app.command("patterns")
|
|
43
|
+
def memory_patterns() -> None:
|
|
44
|
+
"""List stored patterns."""
|
|
45
|
+
subprocess.run(
|
|
46
|
+
[sys.executable, "-m", "attune.memory.control_panel", "patterns", "--list"],
|
|
47
|
+
check=False,
|
|
48
|
+
)
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
"""Metrics commands for user statistics.
|
|
2
|
+
|
|
3
|
+
Copyright 2025 Smart-AI-Memory
|
|
4
|
+
Licensed under Fair Source License 0.9
|
|
5
|
+
"""
|
|
6
|
+
|
|
7
|
+
import sys
|
|
8
|
+
|
|
9
|
+
from attune.logging_config import get_logger
|
|
10
|
+
from attune.metrics.collector import MetricsCollector
|
|
11
|
+
from attune.persistence import StateManager
|
|
12
|
+
|
|
13
|
+
logger = get_logger(__name__)
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
def cmd_metrics_show(args):
|
|
17
|
+
"""Display metrics for a user.
|
|
18
|
+
|
|
19
|
+
Args:
|
|
20
|
+
args: Namespace object from argparse with attributes:
|
|
21
|
+
- user (str): User ID to retrieve metrics for.
|
|
22
|
+
- db (str): Path to metrics database (default: ./metrics.db).
|
|
23
|
+
|
|
24
|
+
Returns:
|
|
25
|
+
None: Prints user metrics to stdout. Exits with code 1 on failure.
|
|
26
|
+
"""
|
|
27
|
+
db_path = args.db
|
|
28
|
+
user_id = args.user
|
|
29
|
+
|
|
30
|
+
logger.info(f"Retrieving metrics for user: {user_id} from {db_path}")
|
|
31
|
+
|
|
32
|
+
collector = MetricsCollector(db_path)
|
|
33
|
+
|
|
34
|
+
try:
|
|
35
|
+
stats = collector.get_user_stats(user_id)
|
|
36
|
+
|
|
37
|
+
logger.info(f"Successfully retrieved metrics for user: {user_id}")
|
|
38
|
+
logger.info(f"=== Metrics for User: {user_id} ===\n")
|
|
39
|
+
logger.info(f"Total Operations: {stats['total_operations']}")
|
|
40
|
+
logger.info(f"Success Rate: {stats['success_rate']:.1%}")
|
|
41
|
+
logger.info(f"Average Response Time: {stats.get('avg_response_time_ms', 0):.0f} ms")
|
|
42
|
+
logger.info(f"\nFirst Use: {stats['first_use']}")
|
|
43
|
+
logger.info(f"Last Use: {stats['last_use']}")
|
|
44
|
+
|
|
45
|
+
logger.info("\nEmpathy Level Usage:")
|
|
46
|
+
logger.info(f" Level 1: {stats.get('level_1_count', 0)} uses")
|
|
47
|
+
logger.info(f" Level 2: {stats.get('level_2_count', 0)} uses")
|
|
48
|
+
logger.info(f" Level 3: {stats.get('level_3_count', 0)} uses")
|
|
49
|
+
logger.info(f" Level 4: {stats.get('level_4_count', 0)} uses")
|
|
50
|
+
logger.info(f" Level 5: {stats.get('level_5_count', 0)} uses")
|
|
51
|
+
except (OSError, FileNotFoundError) as e:
|
|
52
|
+
# Database file not found
|
|
53
|
+
logger.error(f"Metrics database error: {e}")
|
|
54
|
+
logger.error(f"✗ Cannot read metrics database: {e}")
|
|
55
|
+
sys.exit(1)
|
|
56
|
+
except KeyError as e:
|
|
57
|
+
# User not found in database
|
|
58
|
+
logger.error(f"User not found in metrics: {e}")
|
|
59
|
+
logger.error(f"✗ User {user_id} not found: {e}")
|
|
60
|
+
sys.exit(1)
|
|
61
|
+
except Exception as e:
|
|
62
|
+
# Unexpected errors retrieving metrics
|
|
63
|
+
logger.exception(f"Unexpected error retrieving metrics for user {user_id}: {e}")
|
|
64
|
+
logger.error(f"✗ Failed to retrieve metrics: {e}")
|
|
65
|
+
sys.exit(1)
|
|
66
|
+
|
|
67
|
+
|
|
68
|
+
def cmd_state_list(args):
|
|
69
|
+
"""List saved user states.
|
|
70
|
+
|
|
71
|
+
Args:
|
|
72
|
+
args: Namespace object from argparse with attributes:
|
|
73
|
+
- state_dir (str): Directory containing state files.
|
|
74
|
+
|
|
75
|
+
Returns:
|
|
76
|
+
None: Prints list of users with saved states.
|
|
77
|
+
"""
|
|
78
|
+
state_dir = args.state_dir
|
|
79
|
+
|
|
80
|
+
logger.info(f"Listing saved user states from: {state_dir}")
|
|
81
|
+
|
|
82
|
+
manager = StateManager(state_dir)
|
|
83
|
+
users = manager.list_users()
|
|
84
|
+
|
|
85
|
+
logger.info(f"Found {len(users)} saved user states")
|
|
86
|
+
logger.info(f"=== Saved User States: {state_dir} ===\n")
|
|
87
|
+
logger.info(f"Total users: {len(users)}")
|
|
88
|
+
|
|
89
|
+
if users:
|
|
90
|
+
logger.info("\nUsers:")
|
|
91
|
+
for user_id in users:
|
|
92
|
+
logger.info(f" - {user_id}")
|