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.
Files changed (457) hide show
  1. attune/__init__.py +358 -0
  2. attune/adaptive/__init__.py +13 -0
  3. attune/adaptive/task_complexity.py +127 -0
  4. attune/agent_monitoring.py +414 -0
  5. attune/cache/__init__.py +117 -0
  6. attune/cache/base.py +166 -0
  7. attune/cache/dependency_manager.py +256 -0
  8. attune/cache/hash_only.py +251 -0
  9. attune/cache/hybrid.py +457 -0
  10. attune/cache/storage.py +285 -0
  11. attune/cache_monitor.py +356 -0
  12. attune/cache_stats.py +298 -0
  13. attune/cli/__init__.py +152 -0
  14. attune/cli/__main__.py +12 -0
  15. attune/cli/commands/__init__.py +1 -0
  16. attune/cli/commands/batch.py +264 -0
  17. attune/cli/commands/cache.py +248 -0
  18. attune/cli/commands/help.py +331 -0
  19. attune/cli/commands/info.py +140 -0
  20. attune/cli/commands/inspect.py +436 -0
  21. attune/cli/commands/inspection.py +57 -0
  22. attune/cli/commands/memory.py +48 -0
  23. attune/cli/commands/metrics.py +92 -0
  24. attune/cli/commands/orchestrate.py +184 -0
  25. attune/cli/commands/patterns.py +207 -0
  26. attune/cli/commands/profiling.py +202 -0
  27. attune/cli/commands/provider.py +98 -0
  28. attune/cli/commands/routing.py +285 -0
  29. attune/cli/commands/setup.py +96 -0
  30. attune/cli/commands/status.py +235 -0
  31. attune/cli/commands/sync.py +166 -0
  32. attune/cli/commands/tier.py +121 -0
  33. attune/cli/commands/utilities.py +114 -0
  34. attune/cli/commands/workflow.py +579 -0
  35. attune/cli/core.py +32 -0
  36. attune/cli/parsers/__init__.py +68 -0
  37. attune/cli/parsers/batch.py +118 -0
  38. attune/cli/parsers/cache.py +65 -0
  39. attune/cli/parsers/help.py +41 -0
  40. attune/cli/parsers/info.py +26 -0
  41. attune/cli/parsers/inspect.py +66 -0
  42. attune/cli/parsers/metrics.py +42 -0
  43. attune/cli/parsers/orchestrate.py +61 -0
  44. attune/cli/parsers/patterns.py +54 -0
  45. attune/cli/parsers/provider.py +40 -0
  46. attune/cli/parsers/routing.py +110 -0
  47. attune/cli/parsers/setup.py +42 -0
  48. attune/cli/parsers/status.py +47 -0
  49. attune/cli/parsers/sync.py +31 -0
  50. attune/cli/parsers/tier.py +33 -0
  51. attune/cli/parsers/workflow.py +77 -0
  52. attune/cli/utils/__init__.py +1 -0
  53. attune/cli/utils/data.py +242 -0
  54. attune/cli/utils/helpers.py +68 -0
  55. attune/cli_legacy.py +3957 -0
  56. attune/cli_minimal.py +1159 -0
  57. attune/cli_router.py +437 -0
  58. attune/cli_unified.py +814 -0
  59. attune/config/__init__.py +66 -0
  60. attune/config/xml_config.py +286 -0
  61. attune/config.py +545 -0
  62. attune/coordination.py +870 -0
  63. attune/core.py +1511 -0
  64. attune/core_modules/__init__.py +15 -0
  65. attune/cost_tracker.py +626 -0
  66. attune/dashboard/__init__.py +41 -0
  67. attune/dashboard/app.py +512 -0
  68. attune/dashboard/simple_server.py +435 -0
  69. attune/dashboard/standalone_server.py +547 -0
  70. attune/discovery.py +306 -0
  71. attune/emergence.py +306 -0
  72. attune/exceptions.py +123 -0
  73. attune/feedback_loops.py +373 -0
  74. attune/hot_reload/README.md +473 -0
  75. attune/hot_reload/__init__.py +62 -0
  76. attune/hot_reload/config.py +83 -0
  77. attune/hot_reload/integration.py +229 -0
  78. attune/hot_reload/reloader.py +298 -0
  79. attune/hot_reload/watcher.py +183 -0
  80. attune/hot_reload/websocket.py +177 -0
  81. attune/levels.py +577 -0
  82. attune/leverage_points.py +441 -0
  83. attune/logging_config.py +261 -0
  84. attune/mcp/__init__.py +10 -0
  85. attune/mcp/server.py +506 -0
  86. attune/memory/__init__.py +237 -0
  87. attune/memory/claude_memory.py +469 -0
  88. attune/memory/config.py +224 -0
  89. attune/memory/control_panel.py +1290 -0
  90. attune/memory/control_panel_support.py +145 -0
  91. attune/memory/cross_session.py +845 -0
  92. attune/memory/edges.py +179 -0
  93. attune/memory/encryption.py +159 -0
  94. attune/memory/file_session.py +770 -0
  95. attune/memory/graph.py +570 -0
  96. attune/memory/long_term.py +913 -0
  97. attune/memory/long_term_types.py +99 -0
  98. attune/memory/mixins/__init__.py +25 -0
  99. attune/memory/mixins/backend_init_mixin.py +249 -0
  100. attune/memory/mixins/capabilities_mixin.py +208 -0
  101. attune/memory/mixins/handoff_mixin.py +208 -0
  102. attune/memory/mixins/lifecycle_mixin.py +49 -0
  103. attune/memory/mixins/long_term_mixin.py +352 -0
  104. attune/memory/mixins/promotion_mixin.py +109 -0
  105. attune/memory/mixins/short_term_mixin.py +182 -0
  106. attune/memory/nodes.py +179 -0
  107. attune/memory/redis_bootstrap.py +540 -0
  108. attune/memory/security/__init__.py +31 -0
  109. attune/memory/security/audit_logger.py +932 -0
  110. attune/memory/security/pii_scrubber.py +640 -0
  111. attune/memory/security/secrets_detector.py +678 -0
  112. attune/memory/short_term.py +2192 -0
  113. attune/memory/simple_storage.py +302 -0
  114. attune/memory/storage/__init__.py +15 -0
  115. attune/memory/storage_backend.py +167 -0
  116. attune/memory/summary_index.py +583 -0
  117. attune/memory/types.py +446 -0
  118. attune/memory/unified.py +182 -0
  119. attune/meta_workflows/__init__.py +74 -0
  120. attune/meta_workflows/agent_creator.py +248 -0
  121. attune/meta_workflows/builtin_templates.py +567 -0
  122. attune/meta_workflows/cli_commands/__init__.py +56 -0
  123. attune/meta_workflows/cli_commands/agent_commands.py +321 -0
  124. attune/meta_workflows/cli_commands/analytics_commands.py +442 -0
  125. attune/meta_workflows/cli_commands/config_commands.py +232 -0
  126. attune/meta_workflows/cli_commands/memory_commands.py +182 -0
  127. attune/meta_workflows/cli_commands/template_commands.py +354 -0
  128. attune/meta_workflows/cli_commands/workflow_commands.py +382 -0
  129. attune/meta_workflows/cli_meta_workflows.py +59 -0
  130. attune/meta_workflows/form_engine.py +292 -0
  131. attune/meta_workflows/intent_detector.py +409 -0
  132. attune/meta_workflows/models.py +569 -0
  133. attune/meta_workflows/pattern_learner.py +738 -0
  134. attune/meta_workflows/plan_generator.py +384 -0
  135. attune/meta_workflows/session_context.py +397 -0
  136. attune/meta_workflows/template_registry.py +229 -0
  137. attune/meta_workflows/workflow.py +984 -0
  138. attune/metrics/__init__.py +12 -0
  139. attune/metrics/collector.py +31 -0
  140. attune/metrics/prompt_metrics.py +194 -0
  141. attune/models/__init__.py +172 -0
  142. attune/models/__main__.py +13 -0
  143. attune/models/adaptive_routing.py +437 -0
  144. attune/models/auth_cli.py +444 -0
  145. attune/models/auth_strategy.py +450 -0
  146. attune/models/cli.py +655 -0
  147. attune/models/empathy_executor.py +354 -0
  148. attune/models/executor.py +257 -0
  149. attune/models/fallback.py +762 -0
  150. attune/models/provider_config.py +282 -0
  151. attune/models/registry.py +472 -0
  152. attune/models/tasks.py +359 -0
  153. attune/models/telemetry/__init__.py +71 -0
  154. attune/models/telemetry/analytics.py +594 -0
  155. attune/models/telemetry/backend.py +196 -0
  156. attune/models/telemetry/data_models.py +431 -0
  157. attune/models/telemetry/storage.py +489 -0
  158. attune/models/token_estimator.py +420 -0
  159. attune/models/validation.py +280 -0
  160. attune/monitoring/__init__.py +52 -0
  161. attune/monitoring/alerts.py +946 -0
  162. attune/monitoring/alerts_cli.py +448 -0
  163. attune/monitoring/multi_backend.py +271 -0
  164. attune/monitoring/otel_backend.py +362 -0
  165. attune/optimization/__init__.py +19 -0
  166. attune/optimization/context_optimizer.py +272 -0
  167. attune/orchestration/__init__.py +67 -0
  168. attune/orchestration/agent_templates.py +707 -0
  169. attune/orchestration/config_store.py +499 -0
  170. attune/orchestration/execution_strategies.py +2111 -0
  171. attune/orchestration/meta_orchestrator.py +1168 -0
  172. attune/orchestration/pattern_learner.py +696 -0
  173. attune/orchestration/real_tools.py +931 -0
  174. attune/pattern_cache.py +187 -0
  175. attune/pattern_library.py +542 -0
  176. attune/patterns/debugging/all_patterns.json +81 -0
  177. attune/patterns/debugging/workflow_20260107_1770825e.json +77 -0
  178. attune/patterns/refactoring_memory.json +89 -0
  179. attune/persistence.py +564 -0
  180. attune/platform_utils.py +265 -0
  181. attune/plugins/__init__.py +28 -0
  182. attune/plugins/base.py +361 -0
  183. attune/plugins/registry.py +268 -0
  184. attune/project_index/__init__.py +32 -0
  185. attune/project_index/cli.py +335 -0
  186. attune/project_index/index.py +667 -0
  187. attune/project_index/models.py +504 -0
  188. attune/project_index/reports.py +474 -0
  189. attune/project_index/scanner.py +777 -0
  190. attune/project_index/scanner_parallel.py +291 -0
  191. attune/prompts/__init__.py +61 -0
  192. attune/prompts/config.py +77 -0
  193. attune/prompts/context.py +177 -0
  194. attune/prompts/parser.py +285 -0
  195. attune/prompts/registry.py +313 -0
  196. attune/prompts/templates.py +208 -0
  197. attune/redis_config.py +302 -0
  198. attune/redis_memory.py +799 -0
  199. attune/resilience/__init__.py +56 -0
  200. attune/resilience/circuit_breaker.py +256 -0
  201. attune/resilience/fallback.py +179 -0
  202. attune/resilience/health.py +300 -0
  203. attune/resilience/retry.py +209 -0
  204. attune/resilience/timeout.py +135 -0
  205. attune/routing/__init__.py +43 -0
  206. attune/routing/chain_executor.py +433 -0
  207. attune/routing/classifier.py +217 -0
  208. attune/routing/smart_router.py +234 -0
  209. attune/routing/workflow_registry.py +343 -0
  210. attune/scaffolding/README.md +589 -0
  211. attune/scaffolding/__init__.py +35 -0
  212. attune/scaffolding/__main__.py +14 -0
  213. attune/scaffolding/cli.py +240 -0
  214. attune/scaffolding/templates/base_wizard.py.jinja2 +121 -0
  215. attune/scaffolding/templates/coach_wizard.py.jinja2 +321 -0
  216. attune/scaffolding/templates/domain_wizard.py.jinja2 +408 -0
  217. attune/scaffolding/templates/linear_flow_wizard.py.jinja2 +203 -0
  218. attune/socratic/__init__.py +256 -0
  219. attune/socratic/ab_testing.py +958 -0
  220. attune/socratic/blueprint.py +533 -0
  221. attune/socratic/cli.py +703 -0
  222. attune/socratic/collaboration.py +1114 -0
  223. attune/socratic/domain_templates.py +924 -0
  224. attune/socratic/embeddings.py +738 -0
  225. attune/socratic/engine.py +794 -0
  226. attune/socratic/explainer.py +682 -0
  227. attune/socratic/feedback.py +772 -0
  228. attune/socratic/forms.py +629 -0
  229. attune/socratic/generator.py +732 -0
  230. attune/socratic/llm_analyzer.py +637 -0
  231. attune/socratic/mcp_server.py +702 -0
  232. attune/socratic/session.py +312 -0
  233. attune/socratic/storage.py +667 -0
  234. attune/socratic/success.py +730 -0
  235. attune/socratic/visual_editor.py +860 -0
  236. attune/socratic/web_ui.py +958 -0
  237. attune/telemetry/__init__.py +39 -0
  238. attune/telemetry/agent_coordination.py +475 -0
  239. attune/telemetry/agent_tracking.py +367 -0
  240. attune/telemetry/approval_gates.py +545 -0
  241. attune/telemetry/cli.py +1231 -0
  242. attune/telemetry/commands/__init__.py +14 -0
  243. attune/telemetry/commands/dashboard_commands.py +696 -0
  244. attune/telemetry/event_streaming.py +409 -0
  245. attune/telemetry/feedback_loop.py +567 -0
  246. attune/telemetry/usage_tracker.py +591 -0
  247. attune/templates.py +754 -0
  248. attune/test_generator/__init__.py +38 -0
  249. attune/test_generator/__main__.py +14 -0
  250. attune/test_generator/cli.py +234 -0
  251. attune/test_generator/generator.py +355 -0
  252. attune/test_generator/risk_analyzer.py +216 -0
  253. attune/test_generator/templates/unit_test.py.jinja2 +272 -0
  254. attune/tier_recommender.py +384 -0
  255. attune/tools.py +183 -0
  256. attune/trust/__init__.py +28 -0
  257. attune/trust/circuit_breaker.py +579 -0
  258. attune/trust_building.py +527 -0
  259. attune/validation/__init__.py +19 -0
  260. attune/validation/xml_validator.py +281 -0
  261. attune/vscode_bridge.py +173 -0
  262. attune/workflow_commands.py +780 -0
  263. attune/workflow_patterns/__init__.py +33 -0
  264. attune/workflow_patterns/behavior.py +249 -0
  265. attune/workflow_patterns/core.py +76 -0
  266. attune/workflow_patterns/output.py +99 -0
  267. attune/workflow_patterns/registry.py +255 -0
  268. attune/workflow_patterns/structural.py +288 -0
  269. attune/workflows/__init__.py +539 -0
  270. attune/workflows/autonomous_test_gen.py +1268 -0
  271. attune/workflows/base.py +2667 -0
  272. attune/workflows/batch_processing.py +342 -0
  273. attune/workflows/bug_predict.py +1084 -0
  274. attune/workflows/builder.py +273 -0
  275. attune/workflows/caching.py +253 -0
  276. attune/workflows/code_review.py +1048 -0
  277. attune/workflows/code_review_adapters.py +312 -0
  278. attune/workflows/code_review_pipeline.py +722 -0
  279. attune/workflows/config.py +645 -0
  280. attune/workflows/dependency_check.py +644 -0
  281. attune/workflows/document_gen/__init__.py +25 -0
  282. attune/workflows/document_gen/config.py +30 -0
  283. attune/workflows/document_gen/report_formatter.py +162 -0
  284. attune/workflows/document_gen/workflow.py +1426 -0
  285. attune/workflows/document_manager.py +216 -0
  286. attune/workflows/document_manager_README.md +134 -0
  287. attune/workflows/documentation_orchestrator.py +1205 -0
  288. attune/workflows/history.py +510 -0
  289. attune/workflows/keyboard_shortcuts/__init__.py +39 -0
  290. attune/workflows/keyboard_shortcuts/generators.py +391 -0
  291. attune/workflows/keyboard_shortcuts/parsers.py +416 -0
  292. attune/workflows/keyboard_shortcuts/prompts.py +295 -0
  293. attune/workflows/keyboard_shortcuts/schema.py +193 -0
  294. attune/workflows/keyboard_shortcuts/workflow.py +509 -0
  295. attune/workflows/llm_base.py +363 -0
  296. attune/workflows/manage_docs.py +87 -0
  297. attune/workflows/manage_docs_README.md +134 -0
  298. attune/workflows/manage_documentation.py +821 -0
  299. attune/workflows/new_sample_workflow1.py +149 -0
  300. attune/workflows/new_sample_workflow1_README.md +150 -0
  301. attune/workflows/orchestrated_health_check.py +849 -0
  302. attune/workflows/orchestrated_release_prep.py +600 -0
  303. attune/workflows/output.py +413 -0
  304. attune/workflows/perf_audit.py +863 -0
  305. attune/workflows/pr_review.py +762 -0
  306. attune/workflows/progress.py +785 -0
  307. attune/workflows/progress_server.py +322 -0
  308. attune/workflows/progressive/README 2.md +454 -0
  309. attune/workflows/progressive/README.md +454 -0
  310. attune/workflows/progressive/__init__.py +82 -0
  311. attune/workflows/progressive/cli.py +219 -0
  312. attune/workflows/progressive/core.py +488 -0
  313. attune/workflows/progressive/orchestrator.py +723 -0
  314. attune/workflows/progressive/reports.py +520 -0
  315. attune/workflows/progressive/telemetry.py +274 -0
  316. attune/workflows/progressive/test_gen.py +495 -0
  317. attune/workflows/progressive/workflow.py +589 -0
  318. attune/workflows/refactor_plan.py +694 -0
  319. attune/workflows/release_prep.py +895 -0
  320. attune/workflows/release_prep_crew.py +969 -0
  321. attune/workflows/research_synthesis.py +404 -0
  322. attune/workflows/routing.py +168 -0
  323. attune/workflows/secure_release.py +593 -0
  324. attune/workflows/security_adapters.py +297 -0
  325. attune/workflows/security_audit.py +1329 -0
  326. attune/workflows/security_audit_phase3.py +355 -0
  327. attune/workflows/seo_optimization.py +633 -0
  328. attune/workflows/step_config.py +234 -0
  329. attune/workflows/telemetry_mixin.py +269 -0
  330. attune/workflows/test5.py +125 -0
  331. attune/workflows/test5_README.md +158 -0
  332. attune/workflows/test_coverage_boost_crew.py +849 -0
  333. attune/workflows/test_gen/__init__.py +52 -0
  334. attune/workflows/test_gen/ast_analyzer.py +249 -0
  335. attune/workflows/test_gen/config.py +88 -0
  336. attune/workflows/test_gen/data_models.py +38 -0
  337. attune/workflows/test_gen/report_formatter.py +289 -0
  338. attune/workflows/test_gen/test_templates.py +381 -0
  339. attune/workflows/test_gen/workflow.py +655 -0
  340. attune/workflows/test_gen.py +54 -0
  341. attune/workflows/test_gen_behavioral.py +477 -0
  342. attune/workflows/test_gen_parallel.py +341 -0
  343. attune/workflows/test_lifecycle.py +526 -0
  344. attune/workflows/test_maintenance.py +627 -0
  345. attune/workflows/test_maintenance_cli.py +590 -0
  346. attune/workflows/test_maintenance_crew.py +840 -0
  347. attune/workflows/test_runner.py +622 -0
  348. attune/workflows/tier_tracking.py +531 -0
  349. attune/workflows/xml_enhanced_crew.py +285 -0
  350. attune_ai-2.0.0.dist-info/METADATA +1026 -0
  351. attune_ai-2.0.0.dist-info/RECORD +457 -0
  352. attune_ai-2.0.0.dist-info/WHEEL +5 -0
  353. attune_ai-2.0.0.dist-info/entry_points.txt +26 -0
  354. attune_ai-2.0.0.dist-info/licenses/LICENSE +201 -0
  355. attune_ai-2.0.0.dist-info/licenses/LICENSE_CHANGE_ANNOUNCEMENT.md +101 -0
  356. attune_ai-2.0.0.dist-info/top_level.txt +5 -0
  357. attune_healthcare/__init__.py +13 -0
  358. attune_healthcare/monitors/__init__.py +9 -0
  359. attune_healthcare/monitors/clinical_protocol_monitor.py +315 -0
  360. attune_healthcare/monitors/monitoring/__init__.py +44 -0
  361. attune_healthcare/monitors/monitoring/protocol_checker.py +300 -0
  362. attune_healthcare/monitors/monitoring/protocol_loader.py +214 -0
  363. attune_healthcare/monitors/monitoring/sensor_parsers.py +306 -0
  364. attune_healthcare/monitors/monitoring/trajectory_analyzer.py +389 -0
  365. attune_llm/README.md +553 -0
  366. attune_llm/__init__.py +28 -0
  367. attune_llm/agent_factory/__init__.py +53 -0
  368. attune_llm/agent_factory/adapters/__init__.py +85 -0
  369. attune_llm/agent_factory/adapters/autogen_adapter.py +312 -0
  370. attune_llm/agent_factory/adapters/crewai_adapter.py +483 -0
  371. attune_llm/agent_factory/adapters/haystack_adapter.py +298 -0
  372. attune_llm/agent_factory/adapters/langchain_adapter.py +362 -0
  373. attune_llm/agent_factory/adapters/langgraph_adapter.py +333 -0
  374. attune_llm/agent_factory/adapters/native.py +228 -0
  375. attune_llm/agent_factory/adapters/wizard_adapter.py +423 -0
  376. attune_llm/agent_factory/base.py +305 -0
  377. attune_llm/agent_factory/crews/__init__.py +67 -0
  378. attune_llm/agent_factory/crews/code_review.py +1113 -0
  379. attune_llm/agent_factory/crews/health_check.py +1262 -0
  380. attune_llm/agent_factory/crews/refactoring.py +1128 -0
  381. attune_llm/agent_factory/crews/security_audit.py +1018 -0
  382. attune_llm/agent_factory/decorators.py +287 -0
  383. attune_llm/agent_factory/factory.py +558 -0
  384. attune_llm/agent_factory/framework.py +193 -0
  385. attune_llm/agent_factory/memory_integration.py +328 -0
  386. attune_llm/agent_factory/resilient.py +320 -0
  387. attune_llm/agents_md/__init__.py +22 -0
  388. attune_llm/agents_md/loader.py +218 -0
  389. attune_llm/agents_md/parser.py +271 -0
  390. attune_llm/agents_md/registry.py +307 -0
  391. attune_llm/claude_memory.py +466 -0
  392. attune_llm/cli/__init__.py +8 -0
  393. attune_llm/cli/sync_claude.py +487 -0
  394. attune_llm/code_health.py +1313 -0
  395. attune_llm/commands/__init__.py +51 -0
  396. attune_llm/commands/context.py +375 -0
  397. attune_llm/commands/loader.py +301 -0
  398. attune_llm/commands/models.py +231 -0
  399. attune_llm/commands/parser.py +371 -0
  400. attune_llm/commands/registry.py +429 -0
  401. attune_llm/config/__init__.py +29 -0
  402. attune_llm/config/unified.py +291 -0
  403. attune_llm/context/__init__.py +22 -0
  404. attune_llm/context/compaction.py +455 -0
  405. attune_llm/context/manager.py +434 -0
  406. attune_llm/contextual_patterns.py +361 -0
  407. attune_llm/core.py +907 -0
  408. attune_llm/git_pattern_extractor.py +435 -0
  409. attune_llm/hooks/__init__.py +24 -0
  410. attune_llm/hooks/config.py +306 -0
  411. attune_llm/hooks/executor.py +289 -0
  412. attune_llm/hooks/registry.py +302 -0
  413. attune_llm/hooks/scripts/__init__.py +39 -0
  414. attune_llm/hooks/scripts/evaluate_session.py +201 -0
  415. attune_llm/hooks/scripts/first_time_init.py +285 -0
  416. attune_llm/hooks/scripts/pre_compact.py +207 -0
  417. attune_llm/hooks/scripts/session_end.py +183 -0
  418. attune_llm/hooks/scripts/session_start.py +163 -0
  419. attune_llm/hooks/scripts/suggest_compact.py +225 -0
  420. attune_llm/learning/__init__.py +30 -0
  421. attune_llm/learning/evaluator.py +438 -0
  422. attune_llm/learning/extractor.py +514 -0
  423. attune_llm/learning/storage.py +560 -0
  424. attune_llm/levels.py +227 -0
  425. attune_llm/pattern_confidence.py +414 -0
  426. attune_llm/pattern_resolver.py +272 -0
  427. attune_llm/pattern_summary.py +350 -0
  428. attune_llm/providers.py +967 -0
  429. attune_llm/routing/__init__.py +32 -0
  430. attune_llm/routing/model_router.py +362 -0
  431. attune_llm/security/IMPLEMENTATION_SUMMARY.md +413 -0
  432. attune_llm/security/PHASE2_COMPLETE.md +384 -0
  433. attune_llm/security/PHASE2_SECRETS_DETECTOR_COMPLETE.md +271 -0
  434. attune_llm/security/QUICK_REFERENCE.md +316 -0
  435. attune_llm/security/README.md +262 -0
  436. attune_llm/security/__init__.py +62 -0
  437. attune_llm/security/audit_logger.py +929 -0
  438. attune_llm/security/audit_logger_example.py +152 -0
  439. attune_llm/security/pii_scrubber.py +640 -0
  440. attune_llm/security/secrets_detector.py +678 -0
  441. attune_llm/security/secrets_detector_example.py +304 -0
  442. attune_llm/security/secure_memdocs.py +1192 -0
  443. attune_llm/security/secure_memdocs_example.py +278 -0
  444. attune_llm/session_status.py +745 -0
  445. attune_llm/state.py +246 -0
  446. attune_llm/utils/__init__.py +5 -0
  447. attune_llm/utils/tokens.py +349 -0
  448. attune_software/SOFTWARE_PLUGIN_README.md +57 -0
  449. attune_software/__init__.py +13 -0
  450. attune_software/cli/__init__.py +120 -0
  451. attune_software/cli/inspect.py +362 -0
  452. attune_software/cli.py +574 -0
  453. attune_software/plugin.py +188 -0
  454. workflow_scaffolding/__init__.py +11 -0
  455. workflow_scaffolding/__main__.py +12 -0
  456. workflow_scaffolding/cli.py +206 -0
  457. 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("/")