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_software/cli.py
ADDED
|
@@ -0,0 +1,574 @@
|
|
|
1
|
+
"""Empathy Framework - Software Development CLI
|
|
2
|
+
|
|
3
|
+
Command-line interface for running AI development wizards on your codebase.
|
|
4
|
+
|
|
5
|
+
Usage:
|
|
6
|
+
empathy-software analyze /path/to/project
|
|
7
|
+
empathy-software analyze /path/to/project --wizards prompt,context,collaboration
|
|
8
|
+
empathy-software list-wizards
|
|
9
|
+
empathy-software wizard-info testing
|
|
10
|
+
|
|
11
|
+
Copyright 2025 Smart AI Memory, LLC
|
|
12
|
+
Licensed under Fair Source 0.9
|
|
13
|
+
"""
|
|
14
|
+
|
|
15
|
+
import argparse
|
|
16
|
+
import asyncio
|
|
17
|
+
import json
|
|
18
|
+
import os
|
|
19
|
+
import sys
|
|
20
|
+
from pathlib import Path
|
|
21
|
+
from typing import Any
|
|
22
|
+
|
|
23
|
+
# Add parent directory to path for imports
|
|
24
|
+
sys.path.insert(0, os.path.join(os.path.dirname(__file__), "..", "src"))
|
|
25
|
+
|
|
26
|
+
from attune.logging_config import get_logger
|
|
27
|
+
|
|
28
|
+
logger = get_logger(__name__)
|
|
29
|
+
|
|
30
|
+
|
|
31
|
+
def get_global_registry():
|
|
32
|
+
"""Get the global plugin registry.
|
|
33
|
+
|
|
34
|
+
This wrapper exists so tests can patch this function at the module level
|
|
35
|
+
via `empathy_software_plugin.cli.get_global_registry`.
|
|
36
|
+
|
|
37
|
+
Uses late binding to enable patching in tests.
|
|
38
|
+
"""
|
|
39
|
+
from attune.plugins import get_global_registry as _get_global_registry
|
|
40
|
+
|
|
41
|
+
return _get_global_registry()
|
|
42
|
+
|
|
43
|
+
|
|
44
|
+
# Initialize colorama for cross-platform ANSI color support (especially Windows)
|
|
45
|
+
try:
|
|
46
|
+
import colorama
|
|
47
|
+
|
|
48
|
+
colorama.init()
|
|
49
|
+
except ImportError:
|
|
50
|
+
# Colorama not installed - ANSI colors may not work on Windows CMD
|
|
51
|
+
pass
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
class Colors:
|
|
55
|
+
"""ANSI color codes for terminal output"""
|
|
56
|
+
|
|
57
|
+
HEADER = "\033[95m"
|
|
58
|
+
BLUE = "\033[94m"
|
|
59
|
+
CYAN = "\033[96m"
|
|
60
|
+
GREEN = "\033[92m"
|
|
61
|
+
YELLOW = "\033[93m"
|
|
62
|
+
RED = "\033[91m"
|
|
63
|
+
BOLD = "\033[1m"
|
|
64
|
+
UNDERLINE = "\033[4m"
|
|
65
|
+
END = "\033[0m"
|
|
66
|
+
|
|
67
|
+
|
|
68
|
+
def print_header(text: str):
|
|
69
|
+
"""Print colored header"""
|
|
70
|
+
print(f"\n{Colors.BOLD}{Colors.BLUE}{text}{Colors.END}")
|
|
71
|
+
print("=" * len(text))
|
|
72
|
+
|
|
73
|
+
|
|
74
|
+
def print_alert(text: str):
|
|
75
|
+
"""Print alert message"""
|
|
76
|
+
print(f"{Colors.YELLOW}[ALERT]{Colors.END} {text}")
|
|
77
|
+
|
|
78
|
+
|
|
79
|
+
def print_success(text: str):
|
|
80
|
+
"""Print success message"""
|
|
81
|
+
print(f"{Colors.GREEN}✓{Colors.END} {text}")
|
|
82
|
+
|
|
83
|
+
|
|
84
|
+
def print_error(text: str):
|
|
85
|
+
"""Print error message"""
|
|
86
|
+
print(f"{Colors.RED}✗{Colors.END} {text}")
|
|
87
|
+
|
|
88
|
+
|
|
89
|
+
def print_info(text: str):
|
|
90
|
+
"""Print info message"""
|
|
91
|
+
print(f"{Colors.CYAN}ℹ{Colors.END} {text}")
|
|
92
|
+
|
|
93
|
+
|
|
94
|
+
async def analyze_project(
|
|
95
|
+
project_path: str,
|
|
96
|
+
wizard_names: list[str] = None,
|
|
97
|
+
output_format: str = "text",
|
|
98
|
+
verbose: bool = False,
|
|
99
|
+
):
|
|
100
|
+
"""Analyze a project with AI development wizards.
|
|
101
|
+
|
|
102
|
+
Args:
|
|
103
|
+
project_path: Path to project root
|
|
104
|
+
wizard_names: List of wizard names to run (or None for all)
|
|
105
|
+
output_format: 'text' or 'json'
|
|
106
|
+
verbose: Show detailed output
|
|
107
|
+
|
|
108
|
+
"""
|
|
109
|
+
logger.info(f"Starting project analysis for: {project_path} (format: {output_format})")
|
|
110
|
+
print_header("Empathy Framework - AI Development Analysis")
|
|
111
|
+
print(f"Project: {project_path}\n")
|
|
112
|
+
|
|
113
|
+
# Get registry
|
|
114
|
+
registry = get_global_registry()
|
|
115
|
+
software_plugin = registry.get_plugin("software")
|
|
116
|
+
|
|
117
|
+
if not software_plugin:
|
|
118
|
+
logger.error("Software plugin not found in registry")
|
|
119
|
+
print_error("Software plugin not found. Is it installed?")
|
|
120
|
+
return 1
|
|
121
|
+
|
|
122
|
+
# Determine which wizards to run
|
|
123
|
+
if wizard_names:
|
|
124
|
+
wizards_to_run = wizard_names
|
|
125
|
+
else:
|
|
126
|
+
# Run all AI development wizards
|
|
127
|
+
wizards_to_run = [
|
|
128
|
+
"prompt_engineering",
|
|
129
|
+
"context_window",
|
|
130
|
+
"collaboration_pattern",
|
|
131
|
+
"ai_documentation",
|
|
132
|
+
]
|
|
133
|
+
|
|
134
|
+
# Gather context
|
|
135
|
+
logger.info("Gathering project context...")
|
|
136
|
+
print_info("Gathering project context...")
|
|
137
|
+
context = await gather_project_context(project_path)
|
|
138
|
+
|
|
139
|
+
if verbose:
|
|
140
|
+
logger.debug(f"Found {len(context.get('ai_integration_files', []))} AI integration files")
|
|
141
|
+
logger.debug(f"Found {len(context.get('documentation_files', []))} documentation files")
|
|
142
|
+
print_info(f"Found {len(context.get('ai_integration_files', []))} AI integration files")
|
|
143
|
+
print_info(f"Found {len(context.get('documentation_files', []))} documentation files")
|
|
144
|
+
|
|
145
|
+
# Run wizards
|
|
146
|
+
all_results = {}
|
|
147
|
+
|
|
148
|
+
for wizard_name in wizards_to_run:
|
|
149
|
+
logger.info(f"Running wizard: {wizard_name}")
|
|
150
|
+
print_header(f"Running {wizard_name.replace('_', ' ').title()} Wizard")
|
|
151
|
+
|
|
152
|
+
WizardClass = software_plugin.get_wizard(wizard_name)
|
|
153
|
+
if not WizardClass:
|
|
154
|
+
logger.error(f"Wizard not found: {wizard_name}")
|
|
155
|
+
print_error(f"Wizard '{wizard_name}' not found")
|
|
156
|
+
continue
|
|
157
|
+
|
|
158
|
+
wizard = WizardClass()
|
|
159
|
+
|
|
160
|
+
try:
|
|
161
|
+
# Prepare wizard-specific context
|
|
162
|
+
wizard_context = prepare_wizard_context(wizard_name, context)
|
|
163
|
+
|
|
164
|
+
# Run analysis
|
|
165
|
+
result = await wizard.analyze(wizard_context)
|
|
166
|
+
all_results[wizard_name] = result
|
|
167
|
+
logger.info(f"Wizard {wizard_name} completed successfully")
|
|
168
|
+
|
|
169
|
+
# Display results
|
|
170
|
+
if output_format == "text":
|
|
171
|
+
display_wizard_results(wizard, result, verbose)
|
|
172
|
+
else:
|
|
173
|
+
# JSON output handled at end
|
|
174
|
+
pass
|
|
175
|
+
|
|
176
|
+
except Exception as e:
|
|
177
|
+
logger.error(f"Error running wizard {wizard_name}: {e}")
|
|
178
|
+
print_error(f"Error running wizard: {e}")
|
|
179
|
+
if verbose:
|
|
180
|
+
import traceback
|
|
181
|
+
|
|
182
|
+
traceback.print_exc()
|
|
183
|
+
|
|
184
|
+
# Output results
|
|
185
|
+
if output_format == "json":
|
|
186
|
+
print(json.dumps(all_results, indent=2, default=str))
|
|
187
|
+
else:
|
|
188
|
+
print_summary(all_results)
|
|
189
|
+
|
|
190
|
+
return 0
|
|
191
|
+
|
|
192
|
+
|
|
193
|
+
async def gather_project_context(project_path: str) -> dict[str, Any]:
|
|
194
|
+
"""Gather context about the project.
|
|
195
|
+
|
|
196
|
+
Returns dictionary with all context needed by wizards.
|
|
197
|
+
"""
|
|
198
|
+
project_root = Path(project_path)
|
|
199
|
+
|
|
200
|
+
context = {
|
|
201
|
+
"project_path": str(project_root),
|
|
202
|
+
"ai_integration_files": [],
|
|
203
|
+
"documentation_files": [],
|
|
204
|
+
"prompt_files": [],
|
|
205
|
+
"code_files": [],
|
|
206
|
+
"test_files": [],
|
|
207
|
+
"ai_calls": [],
|
|
208
|
+
"context_sources": [],
|
|
209
|
+
"ai_usage_patterns": [],
|
|
210
|
+
"version_history": [],
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
# Find AI integration files
|
|
214
|
+
for pattern in ["**/*.py", "**/*.js", "**/*.ts"]:
|
|
215
|
+
for file_path in project_root.glob(pattern):
|
|
216
|
+
if file_path.is_file():
|
|
217
|
+
# Check if file has AI integration
|
|
218
|
+
try:
|
|
219
|
+
with open(file_path) as f:
|
|
220
|
+
content = f.read()
|
|
221
|
+
if any(
|
|
222
|
+
ai_lib in content
|
|
223
|
+
for ai_lib in [
|
|
224
|
+
"openai",
|
|
225
|
+
"anthropic",
|
|
226
|
+
"langchain",
|
|
227
|
+
"llama",
|
|
228
|
+
"ai.generate",
|
|
229
|
+
]
|
|
230
|
+
):
|
|
231
|
+
context["ai_integration_files"].append(str(file_path))
|
|
232
|
+
|
|
233
|
+
# Parse AI calls
|
|
234
|
+
context["ai_calls"].extend(parse_ai_calls(str(file_path), content))
|
|
235
|
+
except Exception as e:
|
|
236
|
+
# Best effort: Skip files that can't be parsed (corrupted, binary, etc.)
|
|
237
|
+
logger.debug(f"Could not parse {file_path}: {e}")
|
|
238
|
+
pass
|
|
239
|
+
|
|
240
|
+
# All Python/JS/TS files are code files
|
|
241
|
+
context["code_files"].append(str(file_path))
|
|
242
|
+
|
|
243
|
+
# Find documentation files
|
|
244
|
+
for pattern in ["**/*.md", "**/*.rst", "**/*.txt"]:
|
|
245
|
+
for file_path in project_root.glob(pattern):
|
|
246
|
+
if file_path.is_file() and "node_modules" not in str(file_path):
|
|
247
|
+
context["documentation_files"].append(str(file_path))
|
|
248
|
+
|
|
249
|
+
# Find prompt files (common patterns)
|
|
250
|
+
for pattern in ["**/prompts/**/*", "**/*prompt*.txt", "**/*prompt*.md"]:
|
|
251
|
+
for file_path in project_root.glob(pattern):
|
|
252
|
+
if file_path.is_file():
|
|
253
|
+
context["prompt_files"].append(str(file_path))
|
|
254
|
+
|
|
255
|
+
# Find test files
|
|
256
|
+
for pattern in ["**/test_*.py", "**/*_test.py", "**/tests/**/*.py"]:
|
|
257
|
+
for file_path in project_root.glob(pattern):
|
|
258
|
+
if file_path.is_file():
|
|
259
|
+
context["test_files"].append(str(file_path))
|
|
260
|
+
|
|
261
|
+
# Get git history if available
|
|
262
|
+
try:
|
|
263
|
+
import subprocess
|
|
264
|
+
|
|
265
|
+
result = subprocess.run(
|
|
266
|
+
["git", "log", "--oneline", "--name-only", "-50"],
|
|
267
|
+
check=False,
|
|
268
|
+
cwd=project_path,
|
|
269
|
+
capture_output=True,
|
|
270
|
+
text=True,
|
|
271
|
+
timeout=5,
|
|
272
|
+
)
|
|
273
|
+
if result.returncode == 0:
|
|
274
|
+
context["version_history"] = parse_git_history(result.stdout)
|
|
275
|
+
except Exception as e:
|
|
276
|
+
# Optional: Git history unavailable (not a git repo or git not installed)
|
|
277
|
+
logger.debug(f"Could not fetch git history: {e}")
|
|
278
|
+
pass
|
|
279
|
+
|
|
280
|
+
return context
|
|
281
|
+
|
|
282
|
+
|
|
283
|
+
def parse_ai_calls(file_path: str, content: str) -> list[dict[str, Any]]:
|
|
284
|
+
"""Parse AI API calls from file content"""
|
|
285
|
+
# Simplified parser - in production, use AST
|
|
286
|
+
calls = []
|
|
287
|
+
|
|
288
|
+
# Look for common AI call patterns
|
|
289
|
+
if "openai.chat" in content or "anthropic.messages" in content:
|
|
290
|
+
calls.append(
|
|
291
|
+
{
|
|
292
|
+
"id": f"{file_path}:ai_call",
|
|
293
|
+
"location": file_path,
|
|
294
|
+
"code_snippet": content[:500], # First 500 chars as sample
|
|
295
|
+
"prompt_size": len(content),
|
|
296
|
+
"conversation_id": None, # Could detect from context
|
|
297
|
+
},
|
|
298
|
+
)
|
|
299
|
+
|
|
300
|
+
return calls
|
|
301
|
+
|
|
302
|
+
|
|
303
|
+
def parse_git_history(git_output: str) -> list[dict[str, Any]]:
|
|
304
|
+
"""Parse git log output into structured history"""
|
|
305
|
+
commits = []
|
|
306
|
+
current_commit = None
|
|
307
|
+
|
|
308
|
+
for line in git_output.split("\n"):
|
|
309
|
+
if line and not line.startswith(" "):
|
|
310
|
+
# New commit
|
|
311
|
+
if current_commit:
|
|
312
|
+
commits.append(current_commit)
|
|
313
|
+
current_commit = {"hash": line.split()[0], "files": []}
|
|
314
|
+
elif line and current_commit:
|
|
315
|
+
# File in commit
|
|
316
|
+
current_commit["files"].append(line.strip())
|
|
317
|
+
|
|
318
|
+
if current_commit:
|
|
319
|
+
commits.append(current_commit)
|
|
320
|
+
|
|
321
|
+
return commits
|
|
322
|
+
|
|
323
|
+
|
|
324
|
+
def prepare_wizard_context(wizard_name: str, full_context: dict[str, Any]) -> dict[str, Any]:
|
|
325
|
+
"""Prepare context specific to a wizard's requirements"""
|
|
326
|
+
base_context = {
|
|
327
|
+
"project_path": full_context["project_path"],
|
|
328
|
+
"version_history": full_context.get("version_history", []),
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
if wizard_name == "prompt_engineering":
|
|
332
|
+
return {
|
|
333
|
+
**base_context,
|
|
334
|
+
"prompt_files": full_context.get("prompt_files", []),
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
if wizard_name == "context_window":
|
|
338
|
+
return {
|
|
339
|
+
**base_context,
|
|
340
|
+
"ai_calls": full_context.get("ai_calls", []),
|
|
341
|
+
"context_sources": full_context.get("context_sources", []),
|
|
342
|
+
"ai_provider": "anthropic", # Could detect from code
|
|
343
|
+
"model_name": "claude-3-sonnet",
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
if wizard_name == "collaboration_pattern":
|
|
347
|
+
return {
|
|
348
|
+
**base_context,
|
|
349
|
+
"ai_integration_files": full_context.get("ai_integration_files", []),
|
|
350
|
+
"ai_usage_patterns": full_context.get("ai_usage_patterns", []),
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
if wizard_name == "ai_documentation":
|
|
354
|
+
return {
|
|
355
|
+
**base_context,
|
|
356
|
+
"documentation_files": full_context.get("documentation_files", []),
|
|
357
|
+
"code_files": full_context.get("code_files", []),
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
return base_context
|
|
361
|
+
|
|
362
|
+
|
|
363
|
+
def display_wizard_results(wizard, result: dict[str, Any], verbose: bool):
|
|
364
|
+
"""Display wizard results in human-readable format"""
|
|
365
|
+
# Issues
|
|
366
|
+
issues = result.get("issues", [])
|
|
367
|
+
if issues:
|
|
368
|
+
print(f"\n{Colors.BOLD}Current Issues:{Colors.END}")
|
|
369
|
+
for issue in issues:
|
|
370
|
+
severity = issue.get("severity", "info")
|
|
371
|
+
marker = {
|
|
372
|
+
"error": f"{Colors.RED}✗{Colors.END}",
|
|
373
|
+
"warning": f"{Colors.YELLOW}⚠{Colors.END}",
|
|
374
|
+
"info": f"{Colors.CYAN}ℹ{Colors.END}",
|
|
375
|
+
}.get(severity, "ℹ")
|
|
376
|
+
|
|
377
|
+
print(f" {marker} {issue.get('message', 'No message')}")
|
|
378
|
+
if verbose and "suggestion" in issue:
|
|
379
|
+
print(f" → {issue['suggestion']}")
|
|
380
|
+
|
|
381
|
+
# Predictions (Level 4!)
|
|
382
|
+
predictions = result.get("predictions", [])
|
|
383
|
+
if predictions:
|
|
384
|
+
print(f"\n{Colors.BOLD}Anticipatory Alerts (Level 4):{Colors.END}")
|
|
385
|
+
for pred in predictions:
|
|
386
|
+
print(f"\n {Colors.YELLOW}[ALERT]{Colors.END} {pred.get('alert', '')}")
|
|
387
|
+
|
|
388
|
+
if verbose:
|
|
389
|
+
if "reasoning" in pred:
|
|
390
|
+
print(f" Reasoning: {pred['reasoning']}")
|
|
391
|
+
if "personal_experience" in pred:
|
|
392
|
+
print(f" {Colors.CYAN}Experience:{Colors.END} {pred['personal_experience']}")
|
|
393
|
+
|
|
394
|
+
print(f" {Colors.BOLD}Prevention steps:{Colors.END}")
|
|
395
|
+
for i, step in enumerate(pred.get("prevention_steps", [])[:3], 1):
|
|
396
|
+
print(f" {i}. {step}")
|
|
397
|
+
|
|
398
|
+
# Recommendations
|
|
399
|
+
recommendations = result.get("recommendations", [])
|
|
400
|
+
if recommendations:
|
|
401
|
+
print(f"\n{Colors.BOLD}Recommendations:{Colors.END}")
|
|
402
|
+
for rec in recommendations[:5]: # Top 5
|
|
403
|
+
if rec.strip():
|
|
404
|
+
print(f" • {rec}")
|
|
405
|
+
|
|
406
|
+
# Confidence
|
|
407
|
+
confidence = result.get("confidence", 0)
|
|
408
|
+
print(f"\n{Colors.BOLD}Analysis Confidence:{Colors.END} {confidence:.0%}")
|
|
409
|
+
|
|
410
|
+
|
|
411
|
+
def print_summary(all_results: dict[str, dict[str, Any]]):
|
|
412
|
+
"""Print overall summary of analysis"""
|
|
413
|
+
print_header("Analysis Summary")
|
|
414
|
+
|
|
415
|
+
total_issues = sum(len(r.get("issues", [])) for r in all_results.values())
|
|
416
|
+
total_predictions = sum(len(r.get("predictions", [])) for r in all_results.values())
|
|
417
|
+
|
|
418
|
+
print(f"\nWizards run: {len(all_results)}")
|
|
419
|
+
print(f"Current issues found: {total_issues}")
|
|
420
|
+
print(f"Anticipatory alerts: {total_predictions}")
|
|
421
|
+
|
|
422
|
+
high_impact = sum(
|
|
423
|
+
1
|
|
424
|
+
for r in all_results.values()
|
|
425
|
+
for p in r.get("predictions", [])
|
|
426
|
+
if p.get("impact") == "high"
|
|
427
|
+
)
|
|
428
|
+
|
|
429
|
+
if high_impact > 0:
|
|
430
|
+
print(f"\n{Colors.YELLOW}{Colors.BOLD}High-impact alerts: {high_impact}{Colors.END}")
|
|
431
|
+
print("Review these immediately to prevent future issues.")
|
|
432
|
+
|
|
433
|
+
# Extract patterns
|
|
434
|
+
all_patterns = []
|
|
435
|
+
for result in all_results.values():
|
|
436
|
+
all_patterns.extend(result.get("patterns", []))
|
|
437
|
+
|
|
438
|
+
if all_patterns:
|
|
439
|
+
print(f"\nCross-domain patterns discovered: {len(all_patterns)}")
|
|
440
|
+
print("These patterns can be applied to other domains (Level 5 Systems Empathy)")
|
|
441
|
+
|
|
442
|
+
|
|
443
|
+
def list_wizards():
|
|
444
|
+
"""List available wizards"""
|
|
445
|
+
logger.info("Listing available wizards")
|
|
446
|
+
print_header("Available AI Development Wizards")
|
|
447
|
+
|
|
448
|
+
registry = get_global_registry()
|
|
449
|
+
software_plugin = registry.get_plugin("software")
|
|
450
|
+
|
|
451
|
+
if not software_plugin:
|
|
452
|
+
logger.error("Software plugin not found in registry")
|
|
453
|
+
print_error("Software plugin not found")
|
|
454
|
+
return 1
|
|
455
|
+
|
|
456
|
+
wizards = software_plugin.list_wizards()
|
|
457
|
+
logger.info(f"Found {len(wizards)} available wizards")
|
|
458
|
+
|
|
459
|
+
for wizard_id in wizards:
|
|
460
|
+
info = software_plugin.get_wizard_info(wizard_id)
|
|
461
|
+
if info:
|
|
462
|
+
print(f"\n{Colors.BOLD}{wizard_id}{Colors.END}")
|
|
463
|
+
print(f" Name: {info['name']}")
|
|
464
|
+
print(
|
|
465
|
+
f" Level: {info['empathy_level']} ({'Anticipatory' if info['empathy_level'] == 4 else 'Other'})",
|
|
466
|
+
)
|
|
467
|
+
print(f" Category: {info.get('category', 'N/A')}")
|
|
468
|
+
|
|
469
|
+
return 0
|
|
470
|
+
|
|
471
|
+
|
|
472
|
+
def wizard_info(wizard_id: str):
|
|
473
|
+
"""Show detailed info about a wizard"""
|
|
474
|
+
logger.info(f"Displaying info for wizard: {wizard_id}")
|
|
475
|
+
registry = get_global_registry()
|
|
476
|
+
software_plugin = registry.get_plugin("software")
|
|
477
|
+
|
|
478
|
+
if not software_plugin:
|
|
479
|
+
logger.error("Software plugin not found in registry")
|
|
480
|
+
print_error("Software plugin not found")
|
|
481
|
+
return 1
|
|
482
|
+
|
|
483
|
+
info = software_plugin.get_wizard_info(wizard_id)
|
|
484
|
+
if not info:
|
|
485
|
+
logger.error(f"Wizard not found: {wizard_id}")
|
|
486
|
+
print_error(f"Wizard '{wizard_id}' not found")
|
|
487
|
+
return 1
|
|
488
|
+
|
|
489
|
+
logger.debug(f"Wizard info retrieved: {wizard_id}")
|
|
490
|
+
print_header(f"Wizard: {info['name']}")
|
|
491
|
+
print(f"ID: {wizard_id}")
|
|
492
|
+
print(f"Domain: {info['domain']}")
|
|
493
|
+
print(f"Empathy Level: {info['empathy_level']}")
|
|
494
|
+
print(f"Category: {info.get('category', 'N/A')}")
|
|
495
|
+
|
|
496
|
+
print(f"\n{Colors.BOLD}Required Context:{Colors.END}")
|
|
497
|
+
for ctx in info.get("required_context", []):
|
|
498
|
+
print(f" • {ctx}")
|
|
499
|
+
|
|
500
|
+
return 0
|
|
501
|
+
|
|
502
|
+
|
|
503
|
+
def main():
|
|
504
|
+
"""Main CLI entry point"""
|
|
505
|
+
parser = argparse.ArgumentParser(
|
|
506
|
+
description="Empathy Framework - AI Development Analysis",
|
|
507
|
+
formatter_class=argparse.RawDescriptionHelpFormatter,
|
|
508
|
+
epilog="""
|
|
509
|
+
Examples:
|
|
510
|
+
empathy-software analyze /path/to/project
|
|
511
|
+
empathy-software analyze . --wizards prompt_engineering,context_window
|
|
512
|
+
empathy-software analyze . --verbose --output json
|
|
513
|
+
empathy-software list-wizards
|
|
514
|
+
empathy-software wizard-info prompt_engineering
|
|
515
|
+
""",
|
|
516
|
+
)
|
|
517
|
+
|
|
518
|
+
subparsers = parser.add_subparsers(dest="command", help="Command to run")
|
|
519
|
+
|
|
520
|
+
# Analyze command
|
|
521
|
+
analyze_parser = subparsers.add_parser("analyze", help="Analyze a project")
|
|
522
|
+
analyze_parser.add_argument("path", help="Path to project")
|
|
523
|
+
analyze_parser.add_argument(
|
|
524
|
+
"--wizards",
|
|
525
|
+
help="Comma-separated list of wizards to run",
|
|
526
|
+
default=None,
|
|
527
|
+
)
|
|
528
|
+
analyze_parser.add_argument(
|
|
529
|
+
"--output",
|
|
530
|
+
choices=["text", "json"],
|
|
531
|
+
default="text",
|
|
532
|
+
help="Output format",
|
|
533
|
+
)
|
|
534
|
+
analyze_parser.add_argument("--verbose", action="store_true", help="Show detailed output")
|
|
535
|
+
|
|
536
|
+
# List wizards command
|
|
537
|
+
subparsers.add_parser("list-wizards", help="List available wizards")
|
|
538
|
+
|
|
539
|
+
# Wizard info command
|
|
540
|
+
info_parser = subparsers.add_parser("wizard-info", help="Show wizard details")
|
|
541
|
+
info_parser.add_argument("wizard_id", help="Wizard identifier")
|
|
542
|
+
|
|
543
|
+
args = parser.parse_args()
|
|
544
|
+
|
|
545
|
+
if not args.command:
|
|
546
|
+
parser.print_help()
|
|
547
|
+
return 1
|
|
548
|
+
|
|
549
|
+
# Execute command
|
|
550
|
+
if args.command == "analyze":
|
|
551
|
+
wizard_names = None
|
|
552
|
+
if args.wizards:
|
|
553
|
+
wizard_names = [w.strip() for w in args.wizards.split(",")]
|
|
554
|
+
|
|
555
|
+
return asyncio.run(
|
|
556
|
+
analyze_project(
|
|
557
|
+
args.path,
|
|
558
|
+
wizard_names=wizard_names,
|
|
559
|
+
output_format=args.output,
|
|
560
|
+
verbose=args.verbose,
|
|
561
|
+
),
|
|
562
|
+
)
|
|
563
|
+
|
|
564
|
+
if args.command == "list-wizards":
|
|
565
|
+
return list_wizards()
|
|
566
|
+
|
|
567
|
+
if args.command == "wizard-info":
|
|
568
|
+
return wizard_info(args.wizard_id)
|
|
569
|
+
|
|
570
|
+
return 0
|
|
571
|
+
|
|
572
|
+
|
|
573
|
+
if __name__ == "__main__":
|
|
574
|
+
sys.exit(main())
|