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
@@ -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}")