moai-adk 0.15.0__py3-none-any.whl → 0.25.4__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.

Potentially problematic release.


This version of moai-adk might be problematic. Click here for more details.

Files changed (417) hide show
  1. moai_adk/__init__.py +1 -2
  2. moai_adk/__main__.py +85 -2
  3. moai_adk/cli/__init__.py +0 -1
  4. moai_adk/cli/commands/__init__.py +0 -1
  5. moai_adk/cli/commands/analyze.py +127 -0
  6. moai_adk/cli/commands/backup.py +5 -3
  7. moai_adk/cli/commands/doctor.py +35 -11
  8. moai_adk/cli/commands/improve_user_experience.py +348 -0
  9. moai_adk/cli/commands/init.py +150 -23
  10. moai_adk/cli/commands/language.py +269 -0
  11. moai_adk/cli/commands/migrate.py +158 -0
  12. moai_adk/cli/commands/status.py +13 -12
  13. moai_adk/cli/commands/update.py +364 -60
  14. moai_adk/cli/commands/validate_links.py +118 -0
  15. moai_adk/cli/main.py +3 -2
  16. moai_adk/cli/prompts/init_prompts.py +79 -82
  17. moai_adk/core/__init__.py +0 -1
  18. moai_adk/core/analysis/__init__.py +9 -0
  19. moai_adk/core/analysis/session_analyzer.py +439 -0
  20. moai_adk/core/claude_integration.py +421 -0
  21. moai_adk/core/command_helpers.py +270 -0
  22. moai_adk/core/config/__init__.py +6 -0
  23. moai_adk/core/config/auto_spec_config.py +346 -0
  24. moai_adk/core/config/migration.py +133 -12
  25. moai_adk/core/context_manager.py +279 -0
  26. moai_adk/core/diagnostics/slash_commands.py +0 -1
  27. moai_adk/core/error_recovery_system.py +1289 -0
  28. moai_adk/core/git/__init__.py +0 -1
  29. moai_adk/core/git/branch.py +0 -1
  30. moai_adk/core/git/branch_manager.py +4 -4
  31. moai_adk/core/git/checkpoint.py +1 -5
  32. moai_adk/core/git/commit.py +0 -1
  33. moai_adk/core/git/event_detector.py +3 -5
  34. moai_adk/core/git/manager.py +0 -1
  35. moai_adk/core/hooks/post_tool_auto_spec_completion.py +925 -0
  36. moai_adk/core/integration/__init__.py +22 -0
  37. moai_adk/core/integration/engine.py +169 -0
  38. moai_adk/core/integration/integration_tester.py +225 -0
  39. moai_adk/core/integration/models.py +88 -0
  40. moai_adk/core/integration/utils.py +211 -0
  41. moai_adk/core/issue_creator.py +28 -18
  42. moai_adk/core/language_config.py +202 -0
  43. moai_adk/core/language_validator.py +556 -0
  44. moai_adk/core/mcp/setup.py +113 -0
  45. moai_adk/core/migration/__init__.py +18 -0
  46. moai_adk/core/migration/backup_manager.py +208 -0
  47. moai_adk/core/migration/file_migrator.py +218 -0
  48. moai_adk/core/migration/version_detector.py +143 -0
  49. moai_adk/core/migration/version_migrator.py +228 -0
  50. moai_adk/core/performance/__init__.py +6 -0
  51. moai_adk/core/performance/cache_system.py +318 -0
  52. moai_adk/core/performance/parallel_processor.py +116 -0
  53. moai_adk/core/project/__init__.py +0 -1
  54. moai_adk/core/project/backup_utils.py +2 -7
  55. moai_adk/core/project/checker.py +3 -3
  56. moai_adk/core/project/detector.py +20 -40
  57. moai_adk/core/project/initializer.py +42 -17
  58. moai_adk/core/project/phase_executor.py +415 -58
  59. moai_adk/core/project/validator.py +6 -25
  60. moai_adk/core/quality/__init__.py +1 -1
  61. moai_adk/core/quality/trust_checker.py +64 -110
  62. moai_adk/core/quality/validators/__init__.py +1 -1
  63. moai_adk/core/quality/validators/base_validator.py +1 -1
  64. moai_adk/core/rollback_manager.py +993 -0
  65. moai_adk/core/session_manager.py +667 -0
  66. moai_adk/core/spec/confidence_scoring.py +749 -0
  67. moai_adk/core/spec/ears_template_engine.py +1182 -0
  68. moai_adk/core/spec/quality_validator.py +721 -0
  69. moai_adk/core/spec_status_manager.py +488 -0
  70. moai_adk/core/template/__init__.py +0 -1
  71. moai_adk/core/template/backup.py +41 -1
  72. moai_adk/core/template/config.py +11 -12
  73. moai_adk/core/template/languages.py +0 -1
  74. moai_adk/core/template/merger.py +79 -22
  75. moai_adk/core/template/processor.py +614 -40
  76. moai_adk/core/template_engine.py +36 -27
  77. moai_adk/foundation/git/commit_templates.py +565 -0
  78. moai_adk/foundation/trust/trust_principles.py +725 -0
  79. moai_adk/foundation/trust/validation_checklist.py +1678 -0
  80. moai_adk/statusline/__init__.py +38 -0
  81. moai_adk/statusline/alfred_detector.py +107 -0
  82. moai_adk/statusline/config.py +364 -0
  83. moai_adk/statusline/enhanced_output_style_detector.py +364 -0
  84. moai_adk/statusline/git_collector.py +190 -0
  85. moai_adk/statusline/main.py +228 -0
  86. moai_adk/statusline/metrics_tracker.py +78 -0
  87. moai_adk/statusline/renderer.py +327 -0
  88. moai_adk/statusline/update_checker.py +135 -0
  89. moai_adk/statusline/version_reader.py +647 -0
  90. moai_adk/templates/.git-hooks/pre-commit +66 -0
  91. moai_adk/templates/.git-hooks/pre-push +116 -4
  92. moai_adk/templates/.github/workflows/moai-gitflow.yml +1 -7
  93. moai_adk/templates/.github/workflows/spec-issue-sync.yml +0 -1
  94. moai_adk/templates/.gitignore +44 -0
  95. moai_adk/templates/.mcp.json +22 -0
  96. moai_adk/templates/CLAUDE.md +450 -1071
  97. moai_adk/utils/__init__.py +0 -1
  98. moai_adk/utils/banner.py +0 -1
  99. moai_adk/utils/common.py +308 -0
  100. moai_adk/utils/link_validator.py +249 -0
  101. moai_adk/utils/logger.py +4 -9
  102. moai_adk/utils/safe_file_reader.py +210 -0
  103. moai_adk/utils/user_experience.py +531 -0
  104. moai_adk-0.25.4.dist-info/METADATA +2279 -0
  105. moai_adk-0.25.4.dist-info/RECORD +112 -0
  106. moai_adk/core/tags/__init__.py +0 -86
  107. moai_adk/core/tags/ci_validator.py +0 -463
  108. moai_adk/core/tags/cli.py +0 -283
  109. moai_adk/core/tags/generator.py +0 -109
  110. moai_adk/core/tags/inserter.py +0 -99
  111. moai_adk/core/tags/mapper.py +0 -126
  112. moai_adk/core/tags/parser.py +0 -76
  113. moai_adk/core/tags/pre_commit_validator.py +0 -393
  114. moai_adk/core/tags/reporter.py +0 -956
  115. moai_adk/core/tags/tags.py +0 -149
  116. moai_adk/core/tags/validator.py +0 -897
  117. moai_adk/templates/.claude/agents/alfred/backend-expert.md +0 -319
  118. moai_adk/templates/.claude/agents/alfred/cc-manager.md +0 -316
  119. moai_adk/templates/.claude/agents/alfred/debug-helper.md +0 -208
  120. moai_adk/templates/.claude/agents/alfred/devops-expert.md +0 -464
  121. moai_adk/templates/.claude/agents/alfred/doc-syncer.md +0 -214
  122. moai_adk/templates/.claude/agents/alfred/frontend-expert.md +0 -357
  123. moai_adk/templates/.claude/agents/alfred/git-manager.md +0 -406
  124. moai_adk/templates/.claude/agents/alfred/implementation-planner.md +0 -423
  125. moai_adk/templates/.claude/agents/alfred/project-manager.md +0 -312
  126. moai_adk/templates/.claude/agents/alfred/quality-gate.md +0 -343
  127. moai_adk/templates/.claude/agents/alfred/skill-factory.md +0 -865
  128. moai_adk/templates/.claude/agents/alfred/spec-builder.md +0 -392
  129. moai_adk/templates/.claude/agents/alfred/tag-agent.md +0 -361
  130. moai_adk/templates/.claude/agents/alfred/tdd-implementer.md +0 -428
  131. moai_adk/templates/.claude/agents/alfred/trust-checker.md +0 -375
  132. moai_adk/templates/.claude/agents/alfred/ui-ux-expert.md +0 -571
  133. moai_adk/templates/.claude/commands/alfred/0-project.md +0 -1525
  134. moai_adk/templates/.claude/commands/alfred/1-plan.md +0 -802
  135. moai_adk/templates/.claude/commands/alfred/2-run.md +0 -709
  136. moai_adk/templates/.claude/commands/alfred/3-sync.md +0 -1009
  137. moai_adk/templates/.claude/commands/alfred/9-feedback.md +0 -149
  138. moai_adk/templates/.claude/hooks/alfred/core/project.py +0 -748
  139. moai_adk/templates/.claude/hooks/alfred/core/timeout.py +0 -136
  140. moai_adk/templates/.claude/hooks/alfred/core/ttl_cache.py +0 -108
  141. moai_adk/templates/.claude/hooks/alfred/core/version_cache.py +0 -198
  142. moai_adk/templates/.claude/hooks/alfred/handlers/__init__.py +0 -29
  143. moai_adk/templates/.claude/hooks/alfred/post_tool__log_changes.py +0 -94
  144. moai_adk/templates/.claude/hooks/alfred/pre_tool__auto_checkpoint.py +0 -100
  145. moai_adk/templates/.claude/hooks/alfred/session_end__cleanup.py +0 -94
  146. moai_adk/templates/.claude/hooks/alfred/session_start__show_project_info.py +0 -94
  147. moai_adk/templates/.claude/hooks/alfred/shared/core/__init__.py +0 -170
  148. moai_adk/templates/.claude/hooks/alfred/shared/core/checkpoint.py +0 -271
  149. moai_adk/templates/.claude/hooks/alfred/shared/core/context.py +0 -67
  150. moai_adk/templates/.claude/hooks/alfred/shared/core/project.py +0 -749
  151. moai_adk/templates/.claude/hooks/alfred/shared/core/tags.py +0 -230
  152. moai_adk/templates/.claude/hooks/alfred/shared/core/version_cache.py +0 -198
  153. moai_adk/templates/.claude/hooks/alfred/shared/handlers/__init__.py +0 -21
  154. moai_adk/templates/.claude/hooks/alfred/shared/handlers/notification.py +0 -154
  155. moai_adk/templates/.claude/hooks/alfred/shared/handlers/session.py +0 -174
  156. moai_adk/templates/.claude/hooks/alfred/shared/handlers/tool.py +0 -87
  157. moai_adk/templates/.claude/hooks/alfred/shared/handlers/user.py +0 -61
  158. moai_adk/templates/.claude/hooks/alfred/user_prompt__jit_load_docs.py +0 -112
  159. moai_adk/templates/.claude/hooks/alfred/utils/__init__.py +0 -1
  160. moai_adk/templates/.claude/hooks/alfred/utils/timeout.py +0 -161
  161. moai_adk/templates/.claude/settings.json +0 -144
  162. moai_adk/templates/.claude/skills/moai-alfred-agent-guide/SKILL.md +0 -70
  163. moai_adk/templates/.claude/skills/moai-alfred-agent-guide/examples.md +0 -62
  164. moai_adk/templates/.claude/skills/moai-alfred-agent-guide/reference.md +0 -242
  165. moai_adk/templates/.claude/skills/moai-alfred-config-schema/SKILL.md +0 -56
  166. moai_adk/templates/.claude/skills/moai-alfred-config-schema/examples.md +0 -28
  167. moai_adk/templates/.claude/skills/moai-alfred-config-schema/reference.md +0 -444
  168. moai_adk/templates/.claude/skills/moai-alfred-context-budget/SKILL.md +0 -62
  169. moai_adk/templates/.claude/skills/moai-alfred-context-budget/examples.md +0 -28
  170. moai_adk/templates/.claude/skills/moai-alfred-context-budget/reference.md +0 -405
  171. moai_adk/templates/.claude/skills/moai-alfred-dev-guide/SKILL.md +0 -51
  172. moai_adk/templates/.claude/skills/moai-alfred-dev-guide/examples.md +0 -355
  173. moai_adk/templates/.claude/skills/moai-alfred-dev-guide/reference.md +0 -239
  174. moai_adk/templates/.claude/skills/moai-alfred-ears-authoring/SKILL.md +0 -113
  175. moai_adk/templates/.claude/skills/moai-alfred-ears-authoring/examples.md +0 -29
  176. moai_adk/templates/.claude/skills/moai-alfred-ears-authoring/reference.md +0 -28
  177. moai_adk/templates/.claude/skills/moai-alfred-expertise-detection/SKILL.md +0 -323
  178. moai_adk/templates/.claude/skills/moai-alfred-expertise-detection/examples.md +0 -286
  179. moai_adk/templates/.claude/skills/moai-alfred-expertise-detection/reference.md +0 -126
  180. moai_adk/templates/.claude/skills/moai-alfred-git-workflow/SKILL.md +0 -122
  181. moai_adk/templates/.claude/skills/moai-alfred-git-workflow/examples.md +0 -29
  182. moai_adk/templates/.claude/skills/moai-alfred-git-workflow/reference.md +0 -29
  183. moai_adk/templates/.claude/skills/moai-alfred-gitflow-policy/SKILL.md +0 -74
  184. moai_adk/templates/.claude/skills/moai-alfred-gitflow-policy/examples.md +0 -4
  185. moai_adk/templates/.claude/skills/moai-alfred-gitflow-policy/reference.md +0 -269
  186. moai_adk/templates/.claude/skills/moai-alfred-interactive-questions/SKILL.md +0 -237
  187. moai_adk/templates/.claude/skills/moai-alfred-interactive-questions/examples.md +0 -615
  188. moai_adk/templates/.claude/skills/moai-alfred-interactive-questions/reference.md +0 -653
  189. moai_adk/templates/.claude/skills/moai-alfred-issue-labels/SKILL.md +0 -19
  190. moai_adk/templates/.claude/skills/moai-alfred-issue-labels/examples.md +0 -4
  191. moai_adk/templates/.claude/skills/moai-alfred-issue-labels/reference.md +0 -150
  192. moai_adk/templates/.claude/skills/moai-alfred-language-detection/SKILL.md +0 -113
  193. moai_adk/templates/.claude/skills/moai-alfred-language-detection/examples.md +0 -29
  194. moai_adk/templates/.claude/skills/moai-alfred-language-detection/reference.md +0 -28
  195. moai_adk/templates/.claude/skills/moai-alfred-persona-roles/SKILL.md +0 -198
  196. moai_adk/templates/.claude/skills/moai-alfred-persona-roles/examples.md +0 -431
  197. moai_adk/templates/.claude/skills/moai-alfred-persona-roles/reference.md +0 -141
  198. moai_adk/templates/.claude/skills/moai-alfred-practices/SKILL.md +0 -89
  199. moai_adk/templates/.claude/skills/moai-alfred-practices/examples.md +0 -122
  200. moai_adk/templates/.claude/skills/moai-alfred-practices/reference.md +0 -369
  201. moai_adk/templates/.claude/skills/moai-alfred-proactive-suggestions/SKILL.md +0 -508
  202. moai_adk/templates/.claude/skills/moai-alfred-proactive-suggestions/examples.md +0 -481
  203. moai_adk/templates/.claude/skills/moai-alfred-proactive-suggestions/reference.md +0 -100
  204. moai_adk/templates/.claude/skills/moai-alfred-reporting/SKILL.md +0 -273
  205. moai_adk/templates/.claude/skills/moai-alfred-rules/SKILL.md +0 -77
  206. moai_adk/templates/.claude/skills/moai-alfred-rules/examples.md +0 -265
  207. moai_adk/templates/.claude/skills/moai-alfred-rules/reference.md +0 -539
  208. moai_adk/templates/.claude/skills/moai-alfred-session-state/SKILL.md +0 -19
  209. moai_adk/templates/.claude/skills/moai-alfred-session-state/examples.md +0 -4
  210. moai_adk/templates/.claude/skills/moai-alfred-session-state/reference.md +0 -84
  211. moai_adk/templates/.claude/skills/moai-alfred-spec-authoring/README.md +0 -137
  212. moai_adk/templates/.claude/skills/moai-alfred-spec-authoring/SKILL.md +0 -219
  213. moai_adk/templates/.claude/skills/moai-alfred-spec-authoring/examples/validate-spec.sh +0 -161
  214. moai_adk/templates/.claude/skills/moai-alfred-spec-authoring/examples.md +0 -541
  215. moai_adk/templates/.claude/skills/moai-alfred-spec-authoring/reference.md +0 -622
  216. moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-extended/SKILL.md +0 -115
  217. moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-extended/examples.md +0 -4
  218. moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-extended/reference.md +0 -348
  219. moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-validation/SKILL.md +0 -113
  220. moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-validation/examples.md +0 -29
  221. moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-validation/reference.md +0 -28
  222. moai_adk/templates/.claude/skills/moai-alfred-tag-scanning/SKILL.md +0 -113
  223. moai_adk/templates/.claude/skills/moai-alfred-tag-scanning/examples.md +0 -29
  224. moai_adk/templates/.claude/skills/moai-alfred-tag-scanning/reference.md +0 -28
  225. moai_adk/templates/.claude/skills/moai-alfred-todowrite-pattern/SKILL.md +0 -19
  226. moai_adk/templates/.claude/skills/moai-alfred-todowrite-pattern/examples.md +0 -4
  227. moai_adk/templates/.claude/skills/moai-alfred-todowrite-pattern/reference.md +0 -211
  228. moai_adk/templates/.claude/skills/moai-alfred-trust-validation/SKILL.md +0 -113
  229. moai_adk/templates/.claude/skills/moai-alfred-trust-validation/examples.md +0 -29
  230. moai_adk/templates/.claude/skills/moai-alfred-trust-validation/reference.md +0 -28
  231. moai_adk/templates/.claude/skills/moai-alfred-workflow/SKILL.md +0 -288
  232. moai_adk/templates/.claude/skills/moai-cc-agents/SKILL.md +0 -269
  233. moai_adk/templates/.claude/skills/moai-cc-agents/templates/agent-template.md +0 -32
  234. moai_adk/templates/.claude/skills/moai-cc-claude-md/SKILL.md +0 -298
  235. moai_adk/templates/.claude/skills/moai-cc-claude-md/templates/CLAUDE-template.md +0 -26
  236. moai_adk/templates/.claude/skills/moai-cc-commands/SKILL.md +0 -307
  237. moai_adk/templates/.claude/skills/moai-cc-commands/templates/command-template.md +0 -21
  238. moai_adk/templates/.claude/skills/moai-cc-hooks/SKILL.md +0 -252
  239. moai_adk/templates/.claude/skills/moai-cc-hooks/scripts/pre-bash-check.sh +0 -19
  240. moai_adk/templates/.claude/skills/moai-cc-hooks/scripts/preserve-permissions.sh +0 -19
  241. moai_adk/templates/.claude/skills/moai-cc-hooks/scripts/validate-bash-command.py +0 -24
  242. moai_adk/templates/.claude/skills/moai-cc-mcp-plugins/SKILL.md +0 -199
  243. moai_adk/templates/.claude/skills/moai-cc-mcp-plugins/templates/settings-mcp-template.json +0 -39
  244. moai_adk/templates/.claude/skills/moai-cc-memory/SKILL.md +0 -316
  245. moai_adk/templates/.claude/skills/moai-cc-memory/templates/session-summary-template.md +0 -18
  246. moai_adk/templates/.claude/skills/moai-cc-settings/SKILL.md +0 -263
  247. moai_adk/templates/.claude/skills/moai-cc-settings/templates/settings-complete-template.json +0 -30
  248. moai_adk/templates/.claude/skills/moai-cc-skill-descriptions/SKILL.md +0 -19
  249. moai_adk/templates/.claude/skills/moai-cc-skill-descriptions/examples.md +0 -4
  250. moai_adk/templates/.claude/skills/moai-cc-skill-descriptions/reference.md +0 -218
  251. moai_adk/templates/.claude/skills/moai-cc-skill-factory/CHECKLIST.md +0 -482
  252. moai_adk/templates/.claude/skills/moai-cc-skill-factory/EXAMPLES.md +0 -278
  253. moai_adk/templates/.claude/skills/moai-cc-skill-factory/INTERACTIVE-DISCOVERY.md +0 -524
  254. moai_adk/templates/.claude/skills/moai-cc-skill-factory/METADATA.md +0 -477
  255. moai_adk/templates/.claude/skills/moai-cc-skill-factory/PARALLEL-ANALYSIS-REPORT.md +0 -429
  256. moai_adk/templates/.claude/skills/moai-cc-skill-factory/PYTHON-VERSION-MATRIX.md +0 -391
  257. moai_adk/templates/.claude/skills/moai-cc-skill-factory/SKILL-FACTORY-WORKFLOW.md +0 -431
  258. moai_adk/templates/.claude/skills/moai-cc-skill-factory/SKILL-UPDATE-ADVISOR.md +0 -577
  259. moai_adk/templates/.claude/skills/moai-cc-skill-factory/SKILL.md +0 -271
  260. moai_adk/templates/.claude/skills/moai-cc-skill-factory/STEP-BY-STEP-GUIDE.md +0 -466
  261. moai_adk/templates/.claude/skills/moai-cc-skill-factory/STRUCTURE.md +0 -583
  262. moai_adk/templates/.claude/skills/moai-cc-skill-factory/WEB-RESEARCH.md +0 -526
  263. moai_adk/templates/.claude/skills/moai-cc-skill-factory/reference.md +0 -465
  264. moai_adk/templates/.claude/skills/moai-cc-skill-factory/scripts/generate-structure.sh +0 -328
  265. moai_adk/templates/.claude/skills/moai-cc-skill-factory/scripts/validate-skill.sh +0 -312
  266. moai_adk/templates/.claude/skills/moai-cc-skill-factory/templates/SKILL_TEMPLATE.md +0 -245
  267. moai_adk/templates/.claude/skills/moai-cc-skill-factory/templates/examples-template.md +0 -285
  268. moai_adk/templates/.claude/skills/moai-cc-skill-factory/templates/reference-template.md +0 -278
  269. moai_adk/templates/.claude/skills/moai-cc-skill-factory/templates/scripts-template.sh +0 -303
  270. moai_adk/templates/.claude/skills/moai-cc-skills/SKILL.md +0 -291
  271. moai_adk/templates/.claude/skills/moai-cc-skills/templates/SKILL-template.md +0 -15
  272. moai_adk/templates/.claude/skills/moai-design-systems/SKILL.md +0 -802
  273. moai_adk/templates/.claude/skills/moai-design-systems/examples.md +0 -1238
  274. moai_adk/templates/.claude/skills/moai-design-systems/reference.md +0 -673
  275. moai_adk/templates/.claude/skills/moai-domain-backend/SKILL.md +0 -290
  276. moai_adk/templates/.claude/skills/moai-domain-backend/examples.md +0 -1633
  277. moai_adk/templates/.claude/skills/moai-domain-backend/reference.md +0 -660
  278. moai_adk/templates/.claude/skills/moai-domain-cli-tool/SKILL.md +0 -123
  279. moai_adk/templates/.claude/skills/moai-domain-cli-tool/examples.md +0 -29
  280. moai_adk/templates/.claude/skills/moai-domain-cli-tool/reference.md +0 -30
  281. moai_adk/templates/.claude/skills/moai-domain-data-science/SKILL.md +0 -123
  282. moai_adk/templates/.claude/skills/moai-domain-data-science/examples.md +0 -29
  283. moai_adk/templates/.claude/skills/moai-domain-data-science/reference.md +0 -30
  284. moai_adk/templates/.claude/skills/moai-domain-database/SKILL.md +0 -123
  285. moai_adk/templates/.claude/skills/moai-domain-database/examples.md +0 -29
  286. moai_adk/templates/.claude/skills/moai-domain-database/reference.md +0 -30
  287. moai_adk/templates/.claude/skills/moai-domain-devops/SKILL.md +0 -124
  288. moai_adk/templates/.claude/skills/moai-domain-devops/examples.md +0 -29
  289. moai_adk/templates/.claude/skills/moai-domain-devops/reference.md +0 -31
  290. moai_adk/templates/.claude/skills/moai-domain-frontend/SKILL.md +0 -128
  291. moai_adk/templates/.claude/skills/moai-domain-frontend/examples.md +0 -29
  292. moai_adk/templates/.claude/skills/moai-domain-frontend/reference.md +0 -31
  293. moai_adk/templates/.claude/skills/moai-domain-ml/SKILL.md +0 -123
  294. moai_adk/templates/.claude/skills/moai-domain-ml/examples.md +0 -29
  295. moai_adk/templates/.claude/skills/moai-domain-ml/reference.md +0 -30
  296. moai_adk/templates/.claude/skills/moai-domain-mobile-app/SKILL.md +0 -123
  297. moai_adk/templates/.claude/skills/moai-domain-mobile-app/examples.md +0 -29
  298. moai_adk/templates/.claude/skills/moai-domain-mobile-app/reference.md +0 -30
  299. moai_adk/templates/.claude/skills/moai-domain-security/SKILL.md +0 -123
  300. moai_adk/templates/.claude/skills/moai-domain-security/examples.md +0 -29
  301. moai_adk/templates/.claude/skills/moai-domain-security/reference.md +0 -30
  302. moai_adk/templates/.claude/skills/moai-domain-web-api/SKILL.md +0 -123
  303. moai_adk/templates/.claude/skills/moai-domain-web-api/examples.md +0 -29
  304. moai_adk/templates/.claude/skills/moai-domain-web-api/reference.md +0 -30
  305. moai_adk/templates/.claude/skills/moai-essentials-debug/SKILL.md +0 -303
  306. moai_adk/templates/.claude/skills/moai-essentials-debug/examples.md +0 -1064
  307. moai_adk/templates/.claude/skills/moai-essentials-debug/reference.md +0 -1047
  308. moai_adk/templates/.claude/skills/moai-essentials-perf/SKILL.md +0 -113
  309. moai_adk/templates/.claude/skills/moai-essentials-perf/examples.md +0 -29
  310. moai_adk/templates/.claude/skills/moai-essentials-perf/reference.md +0 -28
  311. moai_adk/templates/.claude/skills/moai-essentials-refactor/SKILL.md +0 -113
  312. moai_adk/templates/.claude/skills/moai-essentials-refactor/examples.md +0 -29
  313. moai_adk/templates/.claude/skills/moai-essentials-refactor/reference.md +0 -28
  314. moai_adk/templates/.claude/skills/moai-essentials-review/SKILL.md +0 -113
  315. moai_adk/templates/.claude/skills/moai-essentials-review/examples.md +0 -29
  316. moai_adk/templates/.claude/skills/moai-essentials-review/reference.md +0 -28
  317. moai_adk/templates/.claude/skills/moai-foundation-ears/SKILL.md +0 -116
  318. moai_adk/templates/.claude/skills/moai-foundation-ears/examples.md +0 -29
  319. moai_adk/templates/.claude/skills/moai-foundation-ears/reference.md +0 -28
  320. moai_adk/templates/.claude/skills/moai-foundation-git/SKILL.md +0 -122
  321. moai_adk/templates/.claude/skills/moai-foundation-git/examples.md +0 -29
  322. moai_adk/templates/.claude/skills/moai-foundation-git/reference.md +0 -29
  323. moai_adk/templates/.claude/skills/moai-foundation-langs/SKILL.md +0 -113
  324. moai_adk/templates/.claude/skills/moai-foundation-langs/examples.md +0 -29
  325. moai_adk/templates/.claude/skills/moai-foundation-langs/reference.md +0 -28
  326. moai_adk/templates/.claude/skills/moai-foundation-specs/SKILL.md +0 -113
  327. moai_adk/templates/.claude/skills/moai-foundation-specs/examples.md +0 -29
  328. moai_adk/templates/.claude/skills/moai-foundation-specs/reference.md +0 -28
  329. moai_adk/templates/.claude/skills/moai-foundation-tags/SKILL.md +0 -113
  330. moai_adk/templates/.claude/skills/moai-foundation-tags/examples.md +0 -29
  331. moai_adk/templates/.claude/skills/moai-foundation-tags/reference.md +0 -28
  332. moai_adk/templates/.claude/skills/moai-foundation-trust/SKILL.md +0 -307
  333. moai_adk/templates/.claude/skills/moai-foundation-trust/examples.md +0 -0
  334. moai_adk/templates/.claude/skills/moai-foundation-trust/reference.md +0 -1099
  335. moai_adk/templates/.claude/skills/moai-lang-c/SKILL.md +0 -124
  336. moai_adk/templates/.claude/skills/moai-lang-c/examples.md +0 -29
  337. moai_adk/templates/.claude/skills/moai-lang-c/reference.md +0 -31
  338. moai_adk/templates/.claude/skills/moai-lang-cpp/SKILL.md +0 -124
  339. moai_adk/templates/.claude/skills/moai-lang-cpp/examples.md +0 -29
  340. moai_adk/templates/.claude/skills/moai-lang-cpp/reference.md +0 -31
  341. moai_adk/templates/.claude/skills/moai-lang-csharp/SKILL.md +0 -123
  342. moai_adk/templates/.claude/skills/moai-lang-csharp/examples.md +0 -29
  343. moai_adk/templates/.claude/skills/moai-lang-csharp/reference.md +0 -30
  344. moai_adk/templates/.claude/skills/moai-lang-dart/SKILL.md +0 -123
  345. moai_adk/templates/.claude/skills/moai-lang-dart/examples.md +0 -29
  346. moai_adk/templates/.claude/skills/moai-lang-dart/reference.md +0 -30
  347. moai_adk/templates/.claude/skills/moai-lang-go/SKILL.md +0 -127
  348. moai_adk/templates/.claude/skills/moai-lang-go/examples.md +0 -29
  349. moai_adk/templates/.claude/skills/moai-lang-go/reference.md +0 -31
  350. moai_adk/templates/.claude/skills/moai-lang-java/SKILL.md +0 -126
  351. moai_adk/templates/.claude/skills/moai-lang-java/examples.md +0 -29
  352. moai_adk/templates/.claude/skills/moai-lang-java/reference.md +0 -31
  353. moai_adk/templates/.claude/skills/moai-lang-javascript/SKILL.md +0 -125
  354. moai_adk/templates/.claude/skills/moai-lang-javascript/examples.md +0 -29
  355. moai_adk/templates/.claude/skills/moai-lang-javascript/reference.md +0 -32
  356. moai_adk/templates/.claude/skills/moai-lang-kotlin/SKILL.md +0 -124
  357. moai_adk/templates/.claude/skills/moai-lang-kotlin/examples.md +0 -29
  358. moai_adk/templates/.claude/skills/moai-lang-kotlin/reference.md +0 -31
  359. moai_adk/templates/.claude/skills/moai-lang-php/SKILL.md +0 -126
  360. moai_adk/templates/.claude/skills/moai-lang-php/examples.md +0 -29
  361. moai_adk/templates/.claude/skills/moai-lang-php/reference.md +0 -30
  362. moai_adk/templates/.claude/skills/moai-lang-python/SKILL.md +0 -433
  363. moai_adk/templates/.claude/skills/moai-lang-python/examples.md +0 -624
  364. moai_adk/templates/.claude/skills/moai-lang-python/reference.md +0 -316
  365. moai_adk/templates/.claude/skills/moai-lang-r/SKILL.md +0 -123
  366. moai_adk/templates/.claude/skills/moai-lang-r/examples.md +0 -29
  367. moai_adk/templates/.claude/skills/moai-lang-r/reference.md +0 -30
  368. moai_adk/templates/.claude/skills/moai-lang-ruby/SKILL.md +0 -124
  369. moai_adk/templates/.claude/skills/moai-lang-ruby/examples.md +0 -29
  370. moai_adk/templates/.claude/skills/moai-lang-ruby/reference.md +0 -31
  371. moai_adk/templates/.claude/skills/moai-lang-rust/SKILL.md +0 -127
  372. moai_adk/templates/.claude/skills/moai-lang-rust/examples.md +0 -29
  373. moai_adk/templates/.claude/skills/moai-lang-rust/reference.md +0 -31
  374. moai_adk/templates/.claude/skills/moai-lang-scala/SKILL.md +0 -125
  375. moai_adk/templates/.claude/skills/moai-lang-scala/examples.md +0 -29
  376. moai_adk/templates/.claude/skills/moai-lang-scala/reference.md +0 -30
  377. moai_adk/templates/.claude/skills/moai-lang-shell/SKILL.md +0 -123
  378. moai_adk/templates/.claude/skills/moai-lang-shell/examples.md +0 -29
  379. moai_adk/templates/.claude/skills/moai-lang-shell/reference.md +0 -30
  380. moai_adk/templates/.claude/skills/moai-lang-sql/SKILL.md +0 -124
  381. moai_adk/templates/.claude/skills/moai-lang-sql/examples.md +0 -29
  382. moai_adk/templates/.claude/skills/moai-lang-sql/reference.md +0 -31
  383. moai_adk/templates/.claude/skills/moai-lang-swift/SKILL.md +0 -123
  384. moai_adk/templates/.claude/skills/moai-lang-swift/examples.md +0 -29
  385. moai_adk/templates/.claude/skills/moai-lang-swift/reference.md +0 -30
  386. moai_adk/templates/.claude/skills/moai-lang-typescript/SKILL.md +0 -133
  387. moai_adk/templates/.claude/skills/moai-lang-typescript/examples.md +0 -29
  388. moai_adk/templates/.claude/skills/moai-lang-typescript/reference.md +0 -34
  389. moai_adk/templates/.claude/skills/moai-project-documentation.md +0 -622
  390. moai_adk/templates/.github/workflows/c-tag-validation.yml +0 -11
  391. moai_adk/templates/.github/workflows/cpp-tag-validation.yml +0 -11
  392. moai_adk/templates/.github/workflows/csharp-tag-validation.yml +0 -11
  393. moai_adk/templates/.github/workflows/dart-tag-validation.yml +0 -11
  394. moai_adk/templates/.github/workflows/go-tag-validation.yml +0 -130
  395. moai_adk/templates/.github/workflows/java-tag-validation.yml +0 -11
  396. moai_adk/templates/.github/workflows/javascript-tag-validation.yml +0 -135
  397. moai_adk/templates/.github/workflows/kotlin-tag-validation.yml +0 -11
  398. moai_adk/templates/.github/workflows/php-tag-validation.yml +0 -11
  399. moai_adk/templates/.github/workflows/python-tag-validation.yml +0 -118
  400. moai_adk/templates/.github/workflows/release.yml +0 -118
  401. moai_adk/templates/.github/workflows/ruby-tag-validation.yml +0 -11
  402. moai_adk/templates/.github/workflows/rust-tag-validation.yml +0 -11
  403. moai_adk/templates/.github/workflows/shell-tag-validation.yml +0 -11
  404. moai_adk/templates/.github/workflows/swift-tag-validation.yml +0 -11
  405. moai_adk/templates/.github/workflows/tag-report.yml +0 -269
  406. moai_adk/templates/.github/workflows/tag-validation.yml +0 -186
  407. moai_adk/templates/.github/workflows/typescript-tag-validation.yml +0 -154
  408. moai_adk/templates/.moai/config.json +0 -115
  409. moai_adk/templates/workflows/go-tag-validation.yml +0 -30
  410. moai_adk/templates/workflows/javascript-tag-validation.yml +0 -41
  411. moai_adk/templates/workflows/python-tag-validation.yml +0 -42
  412. moai_adk/templates/workflows/typescript-tag-validation.yml +0 -31
  413. moai_adk-0.15.0.dist-info/METADATA +0 -3079
  414. moai_adk-0.15.0.dist-info/RECORD +0 -365
  415. {moai_adk-0.15.0.dist-info → moai_adk-0.25.4.dist-info}/WHEEL +0 -0
  416. {moai_adk-0.15.0.dist-info → moai_adk-0.25.4.dist-info}/entry_points.txt +0 -0
  417. {moai_adk-0.15.0.dist-info → moai_adk-0.25.4.dist-info}/licenses/LICENSE +0 -0
@@ -1,230 +0,0 @@
1
- #!/usr/bin/env python3
2
- # @CODE:HOOK-TAG-001 | SPEC: TBD | TEST: tests/hooks/test_tag_validation.py
3
- """TAG validation helpers for MoAI-ADK hooks
4
-
5
- Fast checks used by PreToolUse/PostToolUse to nudge users when
6
- new or modified files are missing required @TAG annotations.
7
-
8
- Configurable rules with sensible defaults:
9
- - Load patterns from .moai/tag-rules.json if present.
10
- - Otherwise, apply default glob patterns (folder names are not hard-coded only).
11
-
12
- Defaults (order matters; first match wins):
13
- 1) SPEC
14
- - .moai/specs/**
15
- - **/SPEC-*/spec.md
16
- 2) TEST
17
- - **/*_test.py, **/test_*.py, **/*.test.* (ts,tsx,js,jsx,go,rs)
18
- - **/*.spec.* (ts,tsx,js,jsx)
19
- - tests/**
20
- 3) DOC
21
- - docs/**/*.md, **/README.md, **/*.api.md
22
- 4) CODE
23
- - Source extensions: .py,.ts,.tsx,.js,.jsx,.go,.rs,.java,.kt,.rb,.php,.c,.cpp,.cs,.swift,.scala
24
- - Excluding TEST patterns
25
-
26
- Notes:
27
- - Best-effort: skip binary/large files and non-target paths
28
- - Do not block execution; return a list of issues for messaging
29
- """
30
-
31
- from __future__ import annotations
32
-
33
- import fnmatch
34
- import json
35
- import subprocess
36
- from dataclasses import dataclass
37
- from pathlib import Path
38
- from typing import Iterable, List, Optional
39
-
40
- DEFAULT_CODE_EXTS = (
41
- ".py",
42
- ".ts",
43
- ".tsx",
44
- ".js",
45
- ".jsx",
46
- ".go",
47
- ".rs",
48
- ".java",
49
- ".kt",
50
- ".rb",
51
- ".php",
52
- ".c",
53
- ".cpp",
54
- ".cs",
55
- ".swift",
56
- ".scala",
57
- )
58
-
59
-
60
- @dataclass
61
- class TagIssue:
62
- path: str
63
- expected: str # one of @SPEC, @TEST, @CODE, @DOC
64
- reason: str
65
-
66
-
67
- @dataclass
68
- class Rule:
69
- include: List[str]
70
- expect: str # '@SPEC:' | '@TEST:' | '@CODE:' | '@DOC:'
71
- exclude: List[str]
72
-
73
-
74
- def _load_rules(cwd: str) -> List[Rule]:
75
- """Load tag rules from .moai/tag-rules.json or return defaults.
76
-
77
- Schema example:
78
- {
79
- "rules": [
80
- {"include": ["**/*_test.py", "**/*.test.ts"], "expect": "@TEST:", "exclude": []},
81
- {"include": ["docs/**/*.md", "**/README.md"], "expect": "@DOC:", "exclude": []}
82
- ]
83
- }
84
- """
85
- cfg = Path(cwd) / ".moai" / "tag-rules.json"
86
- if cfg.exists():
87
- try:
88
- data = json.loads(cfg.read_text(encoding="utf-8"))
89
- items = data.get("rules", [])
90
- rules: List[Rule] = []
91
- for it in items:
92
- include = list(it.get("include", []))
93
- expect = str(it.get("expect", ""))
94
- exclude = list(it.get("exclude", []))
95
- if include and expect in ("@SPEC:", "@TEST:", "@CODE:", "@DOC:"):
96
- rules.append(Rule(include=include, expect=expect, exclude=exclude))
97
- if rules:
98
- return rules
99
- except Exception:
100
- pass
101
-
102
- # Defaults (ordered)
103
- return [
104
- Rule(include=[".moai/specs/**", "**/SPEC-*/spec.md"], expect="@SPEC:", exclude=[]),
105
- Rule(
106
- include=[
107
- "**/*_test.py",
108
- "**/test_*.py",
109
- "**/*.test.ts",
110
- "**/*.test.tsx",
111
- "**/*.test.js",
112
- "**/*.test.jsx",
113
- "**/*.test.go",
114
- "**/*.test.rs",
115
- "**/*.spec.ts",
116
- "**/*.spec.tsx",
117
- "tests/**",
118
- ],
119
- expect="@TEST:",
120
- exclude=[".claude/**"],
121
- ),
122
- Rule(
123
- include=["docs/**/*.md", "**/README.md", "**/*.api.md"],
124
- expect="@DOC:",
125
- exclude=[".claude/**"],
126
- ),
127
- Rule(
128
- include=["**/*"],
129
- expect="@CODE:",
130
- exclude=[
131
- "tests/**",
132
- "docs/**",
133
- ".moai/**",
134
- ".claude/**",
135
- "**/*.md",
136
- "**/*.json",
137
- "**/*.yml",
138
- "**/*.yaml",
139
- "**/*.toml",
140
- "**/*.lock",
141
- "**/*.svg",
142
- "**/*.png",
143
- "**/*.jpg",
144
- "**/*.jpeg",
145
- "**/*.gif",
146
- ],
147
- ),
148
- ]
149
-
150
-
151
- def _match_any(path: str, patterns: List[str]) -> bool:
152
- return any(fnmatch.fnmatch(path, pat) for pat in patterns)
153
-
154
-
155
- def _needs_tag_str(path_str: str, rules: List[Rule]) -> Optional[str]:
156
- p = path_str
157
- for rule in rules:
158
- if _match_any(p, rule.include) and not _match_any(p, rule.exclude):
159
- if rule.expect == "@CODE:":
160
- # CODE: limit to source-like extensions to reduce noise
161
- if not any(p.endswith(ext) for ext in DEFAULT_CODE_EXTS):
162
- continue
163
- return rule.expect
164
- return None
165
-
166
-
167
- def _has_tag(content: str, expected: str) -> bool:
168
- return expected in content
169
-
170
-
171
- def _iter_recent_changes(cwd: str) -> Iterable[Path]:
172
- root = Path(cwd)
173
- try:
174
- # Staged files
175
- r1 = subprocess.run(
176
- ["git", "diff", "--name-only", "--cached"],
177
- cwd=cwd,
178
- capture_output=True,
179
- text=True,
180
- timeout=1,
181
- )
182
- # Modified (unstaged) tracked files
183
- r2 = subprocess.run(
184
- ["git", "ls-files", "-m"], cwd=cwd, capture_output=True, text=True, timeout=1
185
- )
186
- # Untracked (other) files respecting .gitignore
187
- r3 = subprocess.run(
188
- ["git", "ls-files", "-o", "--exclude-standard"],
189
- cwd=cwd,
190
- capture_output=True,
191
- text=True,
192
- timeout=1,
193
- )
194
- names = set()
195
- if r1.returncode == 0:
196
- names.update([line.strip() for line in r1.stdout.splitlines() if line.strip()])
197
- if r2.returncode == 0:
198
- names.update([line.strip() for line in r2.stdout.splitlines() if line.strip()])
199
- if r3.returncode == 0:
200
- names.update([line.strip() for line in r3.stdout.splitlines() if line.strip()])
201
- for n in names:
202
- p = (root / n).resolve()
203
- if p.is_file():
204
- yield p
205
- except Exception:
206
- return []
207
-
208
-
209
- def scan_recent_changes_for_missing_tags(cwd: str) -> list[TagIssue]:
210
- issues: list[TagIssue] = []
211
- rules = _load_rules(cwd)
212
- root = Path(cwd).resolve()
213
- for path in _iter_recent_changes(cwd):
214
- try:
215
- content = path.read_text(encoding="utf-8", errors="ignore")
216
- except Exception:
217
- continue
218
- # compute relative path once and use for matching/excluding
219
- try:
220
- rel = path.resolve().relative_to(root)
221
- rel_s = rel.as_posix()
222
- except Exception:
223
- rel_s = path.name
224
-
225
- expected = _needs_tag_str(rel_s, rules)
226
- if not expected:
227
- continue
228
- if not _has_tag(content, expected):
229
- issues.append(TagIssue(path=rel_s, expected=expected, reason="missing tag"))
230
- return issues
@@ -1,198 +0,0 @@
1
- #!/usr/bin/env python3
2
- # @CODE:VERSION-CACHE-001
3
- """Version information cache with TTL support
4
-
5
- TTL-based caching system for version check results to minimize network calls
6
- during SessionStart hook execution.
7
-
8
- SPEC: SPEC-UPDATE-ENHANCE-001 - SessionStart 버전 체크 시스템 강화
9
- Phase 1: Cache System Implementation
10
- """
11
-
12
- import json
13
- from datetime import datetime, timezone
14
- from pathlib import Path
15
- from typing import Any
16
-
17
-
18
- class VersionCache:
19
- """TTL-based version information cache
20
-
21
- Caches version check results with configurable Time-To-Live (TTL)
22
- to avoid excessive network calls to PyPI during SessionStart events.
23
-
24
- Attributes:
25
- cache_dir: Directory to store cache file
26
- ttl_hours: Time-to-live in hours (default 24)
27
- cache_file: Path to the cache JSON file
28
-
29
- Examples:
30
- >>> cache = VersionCache(Path(".moai/cache"), ttl_hours=24)
31
- >>> cache.save({"current_version": "0.8.1", "latest_version": "0.9.0"})
32
- True
33
- >>> cache.is_valid()
34
- True
35
- >>> data = cache.load()
36
- >>> data["current_version"]
37
- '0.8.1'
38
- """
39
-
40
- def __init__(self, cache_dir: Path, ttl_hours: int = 4):
41
- """Initialize cache with TTL in hours
42
-
43
- Args:
44
- cache_dir: Directory where cache file will be stored
45
- ttl_hours: Time-to-live in hours (default 4)
46
- """
47
- self.cache_dir = Path(cache_dir)
48
- self.ttl_hours = ttl_hours
49
- self.cache_file = self.cache_dir / "version-check.json"
50
-
51
- def _calculate_age_hours(self, last_check_iso: str) -> float:
52
- """Calculate age in hours from ISO timestamp (internal helper)
53
-
54
- Normalizes timezone-aware and naive datetimes for consistent comparison.
55
-
56
- Args:
57
- last_check_iso: ISO format timestamp string
58
-
59
- Returns:
60
- Age in hours
61
-
62
- Raises:
63
- ValueError: If timestamp parsing fails
64
- """
65
- last_check = datetime.fromisoformat(last_check_iso)
66
-
67
- # Normalize to naive datetime (remove timezone for comparison)
68
- if last_check.tzinfo is not None:
69
- last_check = last_check.replace(tzinfo=None)
70
-
71
- now = datetime.now()
72
- return (now - last_check).total_seconds() / 3600
73
-
74
- def is_valid(self) -> bool:
75
- """Check if cache exists and is not expired
76
-
77
- Returns:
78
- True if cache file exists and is within TTL, False otherwise
79
-
80
- Examples:
81
- >>> cache = VersionCache(Path(".moai/cache"))
82
- >>> cache.is_valid()
83
- False # No cache file exists yet
84
- """
85
- if not self.cache_file.exists():
86
- return False
87
-
88
- try:
89
- with open(self.cache_file, "r") as f:
90
- data = json.load(f)
91
-
92
- age_hours = self._calculate_age_hours(data["last_check"])
93
- return age_hours < self.ttl_hours
94
-
95
- except (json.JSONDecodeError, KeyError, ValueError, OSError):
96
- # Corrupted or invalid cache file
97
- return False
98
-
99
- def load(self) -> dict[str, Any] | None:
100
- """Load cached version info if valid
101
-
102
- Returns:
103
- Cached version info dictionary if valid, None otherwise
104
-
105
- Examples:
106
- >>> cache = VersionCache(Path(".moai/cache"))
107
- >>> data = cache.load()
108
- >>> data is None
109
- True # No valid cache exists
110
- """
111
- if not self.is_valid():
112
- return None
113
-
114
- try:
115
- with open(self.cache_file, "r") as f:
116
- return json.load(f)
117
- except (json.JSONDecodeError, OSError):
118
- # Graceful degradation on read errors
119
- return None
120
-
121
- def save(self, version_info: dict[str, Any]) -> bool:
122
- """Save version info to cache file
123
-
124
- Creates cache directory if it doesn't exist.
125
- Updates last_check timestamp to current time if not provided.
126
-
127
- Args:
128
- version_info: Version information dictionary to cache
129
-
130
- Returns:
131
- True on successful save, False on error
132
-
133
- Examples:
134
- >>> cache = VersionCache(Path(".moai/cache"))
135
- >>> cache.save({"current_version": "0.8.1"})
136
- True
137
- """
138
- try:
139
- # Create cache directory if it doesn't exist
140
- self.cache_dir.mkdir(parents=True, exist_ok=True)
141
-
142
- # Update last_check timestamp only if not provided (for testing)
143
- if "last_check" not in version_info:
144
- version_info["last_check"] = datetime.now(timezone.utc).isoformat()
145
-
146
- # Write to cache file
147
- with open(self.cache_file, "w") as f:
148
- json.dump(version_info, f, indent=2)
149
-
150
- return True
151
-
152
- except (OSError, TypeError):
153
- # Graceful degradation on write errors
154
- return False
155
-
156
- def clear(self) -> bool:
157
- """Clear/remove cache file
158
-
159
- Returns:
160
- True if cache file was removed or didn't exist, False on error
161
-
162
- Examples:
163
- >>> cache = VersionCache(Path(".moai/cache"))
164
- >>> cache.clear()
165
- True
166
- """
167
- try:
168
- if self.cache_file.exists():
169
- self.cache_file.unlink()
170
- return True
171
- except OSError:
172
- return False
173
-
174
- def get_age_hours(self) -> float:
175
- """Get age of cache in hours
176
-
177
- Returns:
178
- Age in hours, or 0.0 if cache doesn't exist or is invalid
179
-
180
- Examples:
181
- >>> cache = VersionCache(Path(".moai/cache"))
182
- >>> cache.get_age_hours()
183
- 0.0 # No cache exists
184
- """
185
- if not self.cache_file.exists():
186
- return 0.0
187
-
188
- try:
189
- with open(self.cache_file, "r") as f:
190
- data = json.load(f)
191
-
192
- return self._calculate_age_hours(data["last_check"])
193
-
194
- except (json.JSONDecodeError, KeyError, ValueError, OSError):
195
- return 0.0
196
-
197
-
198
- __all__ = ["VersionCache"]
@@ -1,21 +0,0 @@
1
- #!/usr/bin/env python3
2
- """Event handlers for Alfred Hooks
3
-
4
- Claude Code Event Handlers
5
- """
6
-
7
- from .notification import handle_notification, handle_stop, handle_subagent_stop
8
- from .session import handle_session_end, handle_session_start
9
- from .tool import handle_post_tool_use, handle_pre_tool_use
10
- from .user import handle_user_prompt_submit
11
-
12
- __all__ = [
13
- "handle_session_start",
14
- "handle_session_end",
15
- "handle_user_prompt_submit",
16
- "handle_pre_tool_use",
17
- "handle_post_tool_use",
18
- "handle_notification",
19
- "handle_stop",
20
- "handle_subagent_stop",
21
- ]
@@ -1,154 +0,0 @@
1
- #!/usr/bin/env python3
2
- """Notification and control handlers
3
-
4
- Notification, Stop, SubagentStop event handling
5
- """
6
-
7
- import json
8
- from datetime import datetime
9
- from pathlib import Path
10
-
11
- from core import HookPayload, HookResult
12
-
13
-
14
- def _get_command_state_file(cwd: str) -> Path:
15
- """Get the path to command state tracking file"""
16
- state_dir = Path(cwd) / ".moai" / "memory"
17
- state_dir.mkdir(parents=True, exist_ok=True)
18
- return state_dir / "command-execution-state.json"
19
-
20
-
21
- def _load_command_state(cwd: str) -> dict:
22
- """Load current command execution state"""
23
- try:
24
- state_file = _get_command_state_file(cwd)
25
- if state_file.exists():
26
- with open(state_file, "r", encoding="utf-8") as f:
27
- return json.load(f)
28
- except Exception:
29
- pass
30
- return {"last_command": None, "last_timestamp": None, "is_running": False}
31
-
32
-
33
- def _save_command_state(cwd: str, state: dict) -> None:
34
- """Save command execution state"""
35
- try:
36
- state_file = _get_command_state_file(cwd)
37
- with open(state_file, "w", encoding="utf-8") as f:
38
- json.dump(state, f, indent=2)
39
- except Exception:
40
- pass
41
-
42
-
43
- def _is_duplicate_command(current_cmd: str, last_cmd: str, last_timestamp: str) -> bool:
44
- """Check if current command is a duplicate of the last one within 3 seconds"""
45
- if not last_cmd or not last_timestamp or current_cmd != last_cmd:
46
- return False
47
-
48
- try:
49
- last_time = datetime.fromisoformat(last_timestamp)
50
- current_time = datetime.now()
51
- time_diff = (current_time - last_time).total_seconds()
52
- # Consider it a duplicate if same command within 3 seconds
53
- return time_diff < 3
54
- except Exception:
55
- return False
56
-
57
-
58
- def handle_notification(payload: HookPayload) -> HookResult:
59
- """Notification event handler
60
-
61
- Detects and warns about duplicate command executions
62
- (When the same /alfred: command is triggered multiple times within 3 seconds)
63
- """
64
- cwd = payload.get("cwd", ".")
65
- notification = payload.get("notification", {})
66
-
67
- # Extract command information from notification
68
- current_cmd = None
69
- if isinstance(notification, dict):
70
- # Check if notification contains command information
71
- text = notification.get("text", "") or str(notification)
72
- if "/alfred:" in text:
73
- # Extract /alfred: command
74
- import re
75
-
76
- match = re.search(r"/alfred:\S+", text)
77
- if match:
78
- current_cmd = match.group()
79
-
80
- if not current_cmd:
81
- return HookResult()
82
-
83
- # Load current state
84
- state = _load_command_state(cwd)
85
- last_cmd = state.get("last_command")
86
- last_timestamp = state.get("last_timestamp")
87
-
88
- # Check for duplicate
89
- if _is_duplicate_command(current_cmd, last_cmd, last_timestamp):
90
- warning_msg = (
91
- f"⚠️ Duplicate command detected: '{current_cmd}' "
92
- f"is running multiple times within 3 seconds.\n"
93
- f"This may indicate a system issue. Check logs in `.moai/logs/command-invocations.log`"
94
- )
95
-
96
- # Update state - mark as duplicate detected
97
- state["duplicate_detected"] = True
98
- state["duplicate_command"] = current_cmd
99
- state["duplicate_timestamp"] = datetime.now().isoformat()
100
- _save_command_state(cwd, state)
101
-
102
- return HookResult(system_message=warning_msg, continue_execution=True)
103
-
104
- # Update state with current command
105
- state["last_command"] = current_cmd
106
- state["last_timestamp"] = datetime.now().isoformat()
107
- state["is_running"] = True
108
- state["duplicate_detected"] = False
109
- _save_command_state(cwd, state)
110
-
111
- return HookResult()
112
-
113
-
114
- def handle_stop(payload: HookPayload) -> HookResult:
115
- """Stop event handler
116
-
117
- Marks command execution as complete
118
- """
119
- cwd = payload.get("cwd", ".")
120
- state = _load_command_state(cwd)
121
- state["is_running"] = False
122
- state["last_timestamp"] = datetime.now().isoformat()
123
- _save_command_state(cwd, state)
124
-
125
- return HookResult()
126
-
127
-
128
- def handle_subagent_stop(payload: HookPayload) -> HookResult:
129
- """SubagentStop event handler
130
-
131
- Records when a sub-agent finishes execution
132
- """
133
- cwd = payload.get("cwd", ".")
134
-
135
- # Extract subagent name if available
136
- subagent_name = (
137
- payload.get("subagent", {}).get("name")
138
- if isinstance(payload.get("subagent"), dict)
139
- else None
140
- )
141
-
142
- try:
143
- state_file = _get_command_state_file(cwd).parent / "subagent-execution.log"
144
- timestamp = datetime.now().isoformat()
145
-
146
- with open(state_file, "a", encoding="utf-8") as f:
147
- f.write(f"{timestamp} | Subagent Stop | {subagent_name}\n")
148
- except Exception:
149
- pass
150
-
151
- return HookResult()
152
-
153
-
154
- __all__ = ["handle_notification", "handle_stop", "handle_subagent_stop"]