xtrm-tools 0.5.0

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 (333) hide show
  1. package/CHANGELOG.md +504 -0
  2. package/README.md +201 -0
  3. package/cli/dist/index.cjs +57378 -0
  4. package/cli/dist/index.cjs.map +1 -0
  5. package/cli/dist/index.d.cts +2 -0
  6. package/cli/package.json +47 -0
  7. package/config/.env.example +40 -0
  8. package/config/hooks.json +72 -0
  9. package/config/instructions/agents-top.md +30 -0
  10. package/config/instructions/claude-top.md +30 -0
  11. package/config/mcp_servers.json +57 -0
  12. package/config/mcp_servers_optional.json +53 -0
  13. package/config/pi/auth.json.template +14 -0
  14. package/config/pi/extensions/auto-session-name/index.ts +29 -0
  15. package/config/pi/extensions/auto-session-name/package.json +16 -0
  16. package/config/pi/extensions/auto-update/index.ts +71 -0
  17. package/config/pi/extensions/auto-update/package.json +16 -0
  18. package/config/pi/extensions/beads/index.ts +166 -0
  19. package/config/pi/extensions/beads/package.json +16 -0
  20. package/config/pi/extensions/bg-process/index.ts +230 -0
  21. package/config/pi/extensions/bg-process/package.json +16 -0
  22. package/config/pi/extensions/compact-header/index.ts +69 -0
  23. package/config/pi/extensions/compact-header/package.json +16 -0
  24. package/config/pi/extensions/core/adapter.ts +52 -0
  25. package/config/pi/extensions/core/guard-rules.ts +102 -0
  26. package/config/pi/extensions/core/lib.ts +3 -0
  27. package/config/pi/extensions/core/logger.ts +45 -0
  28. package/config/pi/extensions/core/runner.ts +71 -0
  29. package/config/pi/extensions/core/session-state.ts +59 -0
  30. package/config/pi/extensions/custom-footer/index.ts +160 -0
  31. package/config/pi/extensions/custom-footer/package.json +16 -0
  32. package/config/pi/extensions/custom-provider-qwen-cli/index.ts +363 -0
  33. package/config/pi/extensions/custom-provider-qwen-cli/package.json +1 -0
  34. package/config/pi/extensions/git-checkpoint/index.ts +53 -0
  35. package/config/pi/extensions/git-checkpoint/package.json +16 -0
  36. package/config/pi/extensions/minimal-mode/index.ts +201 -0
  37. package/config/pi/extensions/minimal-mode/package.json +16 -0
  38. package/config/pi/extensions/plan-mode/README.md +65 -0
  39. package/config/pi/extensions/plan-mode/index.ts +417 -0
  40. package/config/pi/extensions/plan-mode/package.json +12 -0
  41. package/config/pi/extensions/plan-mode/utils.ts +324 -0
  42. package/config/pi/extensions/quality-gates/index.ts +67 -0
  43. package/config/pi/extensions/quality-gates/package.json +16 -0
  44. package/config/pi/extensions/service-skills/index.ts +108 -0
  45. package/config/pi/extensions/service-skills/package.json +16 -0
  46. package/config/pi/extensions/session-flow/index.ts +131 -0
  47. package/config/pi/extensions/session-flow/package.json +16 -0
  48. package/config/pi/extensions/todo/index.ts +299 -0
  49. package/config/pi/extensions/todo/package.json +16 -0
  50. package/config/pi/extensions/xtrm-loader/index.ts +89 -0
  51. package/config/pi/extensions/xtrm-loader/package.json +16 -0
  52. package/config/pi/install-schema.json +44 -0
  53. package/config/pi/models.json.template +76 -0
  54. package/config/pi/pi-worktrees-settings.json +6 -0
  55. package/config/pi/settings.json.template +16 -0
  56. package/config/settings.json +70 -0
  57. package/hooks/README.md +75 -0
  58. package/hooks/agent_context.py +105 -0
  59. package/hooks/beads-claim-sync.mjs +166 -0
  60. package/hooks/beads-commit-gate.mjs +55 -0
  61. package/hooks/beads-compact-restore.mjs +69 -0
  62. package/hooks/beads-compact-save.mjs +51 -0
  63. package/hooks/beads-edit-gate.mjs +45 -0
  64. package/hooks/beads-gate-core.mjs +215 -0
  65. package/hooks/beads-gate-messages.mjs +87 -0
  66. package/hooks/beads-gate-utils.mjs +185 -0
  67. package/hooks/beads-memory-gate.mjs +61 -0
  68. package/hooks/beads-stop-gate.mjs +32 -0
  69. package/hooks/branch-state.mjs +39 -0
  70. package/hooks/gitnexus/gitnexus-hook.cjs +222 -0
  71. package/hooks/guard-rules.mjs +118 -0
  72. package/hooks/hooks.json +116 -0
  73. package/hooks/main-guard-post-push.mjs +71 -0
  74. package/hooks/main-guard.mjs +119 -0
  75. package/hooks/quality-check.cjs +1286 -0
  76. package/hooks/quality-check.py +345 -0
  77. package/hooks/serena-workflow-reminder.py +74 -0
  78. package/package.json +77 -0
  79. package/project-skills/quality-gates/.claude/hooks/hook-config.json +66 -0
  80. package/project-skills/quality-gates/.claude/hooks/quality-check.cjs +1286 -0
  81. package/project-skills/quality-gates/.claude/hooks/quality-check.py +334 -0
  82. package/project-skills/quality-gates/.claude/settings.json +3 -0
  83. package/project-skills/quality-gates/.claude/skills/using-quality-gates/SKILL.md +254 -0
  84. package/project-skills/quality-gates/README.md +109 -0
  85. package/project-skills/quality-gates/evals/evals.json +181 -0
  86. package/project-skills/quality-gates/workspace/iteration-1/FINAL-EVAL-SUMMARY.md +75 -0
  87. package/project-skills/quality-gates/workspace/iteration-1/edge-case-auto-fix-verification/with_skill/outputs/response.md +59 -0
  88. package/project-skills/quality-gates/workspace/iteration-1/edge-case-mixed-language-project/with_skill/outputs/response.md +60 -0
  89. package/project-skills/quality-gates/workspace/iteration-1/eval-summary.md +105 -0
  90. package/project-skills/quality-gates/workspace/iteration-1/partial-install-python-only/with_skill/outputs/response.md +93 -0
  91. package/project-skills/quality-gates/workspace/iteration-1/python-refactor-request/with_skill/outputs/response.md +104 -0
  92. package/project-skills/quality-gates/workspace/iteration-1/quality-gate-error-fix/with_skill/outputs/response.md +74 -0
  93. package/project-skills/quality-gates/workspace/iteration-1/should-not-trigger-general-chat/with_skill/outputs/response.md +18 -0
  94. package/project-skills/quality-gates/workspace/iteration-1/should-not-trigger-math-question/with_skill/outputs/response.md +18 -0
  95. package/project-skills/quality-gates/workspace/iteration-1/should-not-trigger-unrelated-coding/with_skill/outputs/response.md +56 -0
  96. package/project-skills/quality-gates/workspace/iteration-1/tdd-guard-blocking-confusion/with_skill/outputs/response.md +67 -0
  97. package/project-skills/quality-gates/workspace/iteration-1/typescript-feature-with-tests/with_skill/outputs/response.md +97 -0
  98. package/project-skills/service-skills-set/.claude/git-hooks/doc_reminder.py +67 -0
  99. package/project-skills/service-skills-set/.claude/git-hooks/skill_staleness.py +194 -0
  100. package/project-skills/service-skills-set/.claude/service-registry.json +4 -0
  101. package/project-skills/service-skills-set/.claude/settings.json +37 -0
  102. package/project-skills/service-skills-set/.claude/skills/creating-service-skills/SKILL.md +433 -0
  103. package/project-skills/service-skills-set/.claude/skills/creating-service-skills/references/script_quality_standards.md +425 -0
  104. package/project-skills/service-skills-set/.claude/skills/creating-service-skills/references/service_skill_system_guide.md +278 -0
  105. package/project-skills/service-skills-set/.claude/skills/creating-service-skills/scripts/bootstrap.py +308 -0
  106. package/project-skills/service-skills-set/.claude/skills/creating-service-skills/scripts/deep_dive.py +304 -0
  107. package/project-skills/service-skills-set/.claude/skills/creating-service-skills/scripts/scaffolder.py +482 -0
  108. package/project-skills/service-skills-set/.claude/skills/scoping-service-skills/SKILL.md +231 -0
  109. package/project-skills/service-skills-set/.claude/skills/scoping-service-skills/scripts/scope.py +74 -0
  110. package/project-skills/service-skills-set/.claude/skills/updating-service-skills/SKILL.md +136 -0
  111. package/project-skills/service-skills-set/.claude/skills/updating-service-skills/scripts/drift_detector.py +222 -0
  112. package/project-skills/service-skills-set/.claude/skills/using-service-skills/SKILL.md +108 -0
  113. package/project-skills/service-skills-set/.claude/skills/using-service-skills/scripts/cataloger.py +74 -0
  114. package/project-skills/service-skills-set/.claude/skills/using-service-skills/scripts/skill_activator.py +152 -0
  115. package/project-skills/service-skills-set/README.md +93 -0
  116. package/project-skills/service-skills-set/install-service-skills.py +193 -0
  117. package/project-skills/service-skills-set/service-skills-readme.md +236 -0
  118. package/skills/README.txt +31 -0
  119. package/skills/clean-code/SKILL.md +201 -0
  120. package/skills/creating-service-skills/SKILL.md +433 -0
  121. package/skills/creating-service-skills/references/script_quality_standards.md +425 -0
  122. package/skills/creating-service-skills/references/service_skill_system_guide.md +278 -0
  123. package/skills/creating-service-skills/scripts/bootstrap.py +326 -0
  124. package/skills/creating-service-skills/scripts/deep_dive.py +304 -0
  125. package/skills/creating-service-skills/scripts/scaffolder.py +482 -0
  126. package/skills/delegating/SKILL.md +196 -0
  127. package/skills/delegating/config.yaml +210 -0
  128. package/skills/delegating/references/orchestration-protocols.md +41 -0
  129. package/skills/docker-expert/SKILL.md +409 -0
  130. package/skills/documenting/CHANGELOG.md +23 -0
  131. package/skills/documenting/README.md +148 -0
  132. package/skills/documenting/SKILL.md +113 -0
  133. package/skills/documenting/examples/example_pattern.md +70 -0
  134. package/skills/documenting/examples/example_reference.md +70 -0
  135. package/skills/documenting/examples/example_ssot_analytics.md +64 -0
  136. package/skills/documenting/examples/example_workflow.md +141 -0
  137. package/skills/documenting/references/changelog-format.md +97 -0
  138. package/skills/documenting/references/metadata-schema.md +136 -0
  139. package/skills/documenting/references/taxonomy.md +81 -0
  140. package/skills/documenting/references/versioning-rules.md +78 -0
  141. package/skills/documenting/scripts/bump_version.sh +60 -0
  142. package/skills/documenting/scripts/changelog/__init__.py +0 -0
  143. package/skills/documenting/scripts/changelog/add_entry.py +216 -0
  144. package/skills/documenting/scripts/changelog/bump_release.py +117 -0
  145. package/skills/documenting/scripts/changelog/init_changelog.py +54 -0
  146. package/skills/documenting/scripts/changelog/validate_changelog.py +128 -0
  147. package/skills/documenting/scripts/drift_detector.py +266 -0
  148. package/skills/documenting/scripts/generate_template.py +311 -0
  149. package/skills/documenting/scripts/list_by_category.sh +84 -0
  150. package/skills/documenting/scripts/orchestrator.py +255 -0
  151. package/skills/documenting/scripts/validate_metadata.py +242 -0
  152. package/skills/documenting/templates/CHANGELOG.md.template +13 -0
  153. package/skills/find-skills/SKILL.md +133 -0
  154. package/skills/gitnexus-debugging/SKILL.md +85 -0
  155. package/skills/gitnexus-exploring/SKILL.md +75 -0
  156. package/skills/gitnexus-impact-analysis/SKILL.md +94 -0
  157. package/skills/gitnexus-refactoring/SKILL.md +113 -0
  158. package/skills/hook-development/SKILL.md +797 -0
  159. package/skills/hook-development/examples/load-context.sh +55 -0
  160. package/skills/hook-development/examples/quality-check.js +1168 -0
  161. package/skills/hook-development/examples/validate-bash.sh +43 -0
  162. package/skills/hook-development/examples/validate-write.sh +38 -0
  163. package/skills/hook-development/references/advanced.md +527 -0
  164. package/skills/hook-development/references/migration.md +369 -0
  165. package/skills/hook-development/references/patterns.md +412 -0
  166. package/skills/hook-development/scripts/README.md +164 -0
  167. package/skills/hook-development/scripts/hook-linter.sh +153 -0
  168. package/skills/hook-development/scripts/test-hook.sh +252 -0
  169. package/skills/hook-development/scripts/validate-hook-schema.sh +159 -0
  170. package/skills/obsidian-cli/SKILL.md +106 -0
  171. package/skills/orchestrating-agents/SKILL.md +135 -0
  172. package/skills/orchestrating-agents/config.yaml +45 -0
  173. package/skills/orchestrating-agents/references/agent-context-integration.md +37 -0
  174. package/skills/orchestrating-agents/references/examples.md +45 -0
  175. package/skills/orchestrating-agents/references/handover-protocol.md +31 -0
  176. package/skills/orchestrating-agents/references/workflows.md +42 -0
  177. package/skills/orchestrating-agents/scripts/detect_neighbors.py +23 -0
  178. package/skills/prompt-improving/README.md +162 -0
  179. package/skills/prompt-improving/SKILL.md +74 -0
  180. package/skills/prompt-improving/references/analysis_commands.md +24 -0
  181. package/skills/prompt-improving/references/chain_of_thought.md +24 -0
  182. package/skills/prompt-improving/references/mcp_definitions.md +20 -0
  183. package/skills/prompt-improving/references/multishot.md +23 -0
  184. package/skills/prompt-improving/references/xml_core.md +60 -0
  185. package/skills/python-testing/SKILL.md +815 -0
  186. package/skills/scoping-service-skills/SKILL.md +231 -0
  187. package/skills/scoping-service-skills/scripts/scope.py +74 -0
  188. package/skills/senior-backend/SKILL.md +209 -0
  189. package/skills/senior-backend/references/api_design_patterns.md +103 -0
  190. package/skills/senior-backend/references/backend_security_practices.md +103 -0
  191. package/skills/senior-backend/references/database_optimization_guide.md +103 -0
  192. package/skills/senior-backend/scripts/api_load_tester.py +114 -0
  193. package/skills/senior-backend/scripts/api_scaffolder.py +114 -0
  194. package/skills/senior-backend/scripts/database_migration_tool.py +114 -0
  195. package/skills/senior-data-scientist/SKILL.md +226 -0
  196. package/skills/senior-data-scientist/references/experiment_design_frameworks.md +80 -0
  197. package/skills/senior-data-scientist/references/feature_engineering_patterns.md +80 -0
  198. package/skills/senior-data-scientist/references/statistical_methods_advanced.md +80 -0
  199. package/skills/senior-data-scientist/scripts/experiment_designer.py +100 -0
  200. package/skills/senior-data-scientist/scripts/feature_engineering_pipeline.py +100 -0
  201. package/skills/senior-data-scientist/scripts/model_evaluation_suite.py +100 -0
  202. package/skills/senior-devops/SKILL.md +209 -0
  203. package/skills/senior-devops/references/cicd_pipeline_guide.md +103 -0
  204. package/skills/senior-devops/references/deployment_strategies.md +103 -0
  205. package/skills/senior-devops/references/infrastructure_as_code.md +103 -0
  206. package/skills/senior-devops/scripts/deployment_manager.py +114 -0
  207. package/skills/senior-devops/scripts/pipeline_generator.py +114 -0
  208. package/skills/senior-devops/scripts/terraform_scaffolder.py +114 -0
  209. package/skills/senior-security/SKILL.md +209 -0
  210. package/skills/senior-security/references/cryptography_implementation.md +103 -0
  211. package/skills/senior-security/references/penetration_testing_guide.md +103 -0
  212. package/skills/senior-security/references/security_architecture_patterns.md +103 -0
  213. package/skills/senior-security/scripts/pentest_automator.py +114 -0
  214. package/skills/senior-security/scripts/security_auditor.py +114 -0
  215. package/skills/senior-security/scripts/threat_modeler.py +114 -0
  216. package/skills/skill-creator/LICENSE.txt +202 -0
  217. package/skills/skill-creator/SKILL.md +479 -0
  218. package/skills/skill-creator/agents/analyzer.md +274 -0
  219. package/skills/skill-creator/agents/comparator.md +202 -0
  220. package/skills/skill-creator/agents/grader.md +223 -0
  221. package/skills/skill-creator/assets/eval_review.html +146 -0
  222. package/skills/skill-creator/eval-viewer/generate_review.py +471 -0
  223. package/skills/skill-creator/eval-viewer/viewer.html +1325 -0
  224. package/skills/skill-creator/references/schemas.md +430 -0
  225. package/skills/skill-creator/scripts/__init__.py +0 -0
  226. package/skills/skill-creator/scripts/aggregate_benchmark.py +401 -0
  227. package/skills/skill-creator/scripts/generate_report.py +326 -0
  228. package/skills/skill-creator/scripts/improve_description.py +248 -0
  229. package/skills/skill-creator/scripts/package_skill.py +136 -0
  230. package/skills/skill-creator/scripts/quick_validate.py +103 -0
  231. package/skills/skill-creator/scripts/run_eval.py +310 -0
  232. package/skills/skill-creator/scripts/run_loop.py +332 -0
  233. package/skills/skill-creator/scripts/utils.py +47 -0
  234. package/skills/sync-docs/SKILL.md +132 -0
  235. package/skills/sync-docs/evals/evals.json +89 -0
  236. package/skills/sync-docs/references/doc-structure.md +99 -0
  237. package/skills/sync-docs/references/schema.md +103 -0
  238. package/skills/sync-docs/scripts/changelog/add_entry.py +216 -0
  239. package/skills/sync-docs/scripts/context_gatherer.py +240 -0
  240. package/skills/sync-docs/scripts/doc_structure_analyzer.py +495 -0
  241. package/skills/sync-docs/scripts/drift_detector.py +327 -0
  242. package/skills/sync-docs/scripts/validate_doc.py +365 -0
  243. package/skills/sync-docs/scripts/validate_metadata.py +185 -0
  244. package/skills/sync-docs-workspace/iteration-1/benchmark.json +293 -0
  245. package/skills/sync-docs-workspace/iteration-1/benchmark.md +13 -0
  246. package/skills/sync-docs-workspace/iteration-1/eval-doc-audit/eval_metadata.json +27 -0
  247. package/skills/sync-docs-workspace/iteration-1/eval-doc-audit/with_skill/outputs/result.md +210 -0
  248. package/skills/sync-docs-workspace/iteration-1/eval-doc-audit/with_skill/run-1/grading.json +28 -0
  249. package/skills/sync-docs-workspace/iteration-1/eval-doc-audit/with_skill/run-1/timing.json +1 -0
  250. package/skills/sync-docs-workspace/iteration-1/eval-doc-audit/without_skill/outputs/result.md +101 -0
  251. package/skills/sync-docs-workspace/iteration-1/eval-doc-audit/without_skill/run-1/grading.json +28 -0
  252. package/skills/sync-docs-workspace/iteration-1/eval-doc-audit/without_skill/run-1/timing.json +5 -0
  253. package/skills/sync-docs-workspace/iteration-1/eval-doc-audit/without_skill/timing.json +5 -0
  254. package/skills/sync-docs-workspace/iteration-1/eval-fix-mode/eval_metadata.json +27 -0
  255. package/skills/sync-docs-workspace/iteration-1/eval-fix-mode/with_skill/outputs/result.md +198 -0
  256. package/skills/sync-docs-workspace/iteration-1/eval-fix-mode/with_skill/run-1/grading.json +28 -0
  257. package/skills/sync-docs-workspace/iteration-1/eval-fix-mode/with_skill/run-1/timing.json +1 -0
  258. package/skills/sync-docs-workspace/iteration-1/eval-fix-mode/without_skill/outputs/result.md +94 -0
  259. package/skills/sync-docs-workspace/iteration-1/eval-fix-mode/without_skill/run-1/grading.json +28 -0
  260. package/skills/sync-docs-workspace/iteration-1/eval-fix-mode/without_skill/run-1/timing.json +1 -0
  261. package/skills/sync-docs-workspace/iteration-1/eval-sprint-closeout/eval_metadata.json +27 -0
  262. package/skills/sync-docs-workspace/iteration-1/eval-sprint-closeout/with_skill/outputs/result.md +237 -0
  263. package/skills/sync-docs-workspace/iteration-1/eval-sprint-closeout/with_skill/run-1/grading.json +28 -0
  264. package/skills/sync-docs-workspace/iteration-1/eval-sprint-closeout/with_skill/run-1/timing.json +1 -0
  265. package/skills/sync-docs-workspace/iteration-1/eval-sprint-closeout/without_skill/outputs/result.md +134 -0
  266. package/skills/sync-docs-workspace/iteration-1/eval-sprint-closeout/without_skill/run-1/grading.json +28 -0
  267. package/skills/sync-docs-workspace/iteration-1/eval-sprint-closeout/without_skill/run-1/timing.json +1 -0
  268. package/skills/sync-docs-workspace/iteration-2/benchmark.json +297 -0
  269. package/skills/sync-docs-workspace/iteration-2/benchmark.md +13 -0
  270. package/skills/sync-docs-workspace/iteration-2/eval-doc-audit/eval_metadata.json +27 -0
  271. package/skills/sync-docs-workspace/iteration-2/eval-doc-audit/with_skill/outputs/result.md +137 -0
  272. package/skills/sync-docs-workspace/iteration-2/eval-doc-audit/with_skill/run-1/grading.json +92 -0
  273. package/skills/sync-docs-workspace/iteration-2/eval-doc-audit/with_skill/run-1/timing.json +1 -0
  274. package/skills/sync-docs-workspace/iteration-2/eval-doc-audit/without_skill/outputs/result.md +134 -0
  275. package/skills/sync-docs-workspace/iteration-2/eval-doc-audit/without_skill/run-1/grading.json +86 -0
  276. package/skills/sync-docs-workspace/iteration-2/eval-doc-audit/without_skill/run-1/timing.json +1 -0
  277. package/skills/sync-docs-workspace/iteration-2/eval-fix-mode/eval_metadata.json +27 -0
  278. package/skills/sync-docs-workspace/iteration-2/eval-fix-mode/with_skill/outputs/result.md +193 -0
  279. package/skills/sync-docs-workspace/iteration-2/eval-fix-mode/with_skill/run-1/grading.json +72 -0
  280. package/skills/sync-docs-workspace/iteration-2/eval-fix-mode/with_skill/run-1/timing.json +1 -0
  281. package/skills/sync-docs-workspace/iteration-2/eval-fix-mode/without_skill/outputs/result.md +211 -0
  282. package/skills/sync-docs-workspace/iteration-2/eval-fix-mode/without_skill/run-1/grading.json +91 -0
  283. package/skills/sync-docs-workspace/iteration-2/eval-fix-mode/without_skill/run-1/timing.json +5 -0
  284. package/skills/sync-docs-workspace/iteration-2/eval-sprint-closeout/eval_metadata.json +27 -0
  285. package/skills/sync-docs-workspace/iteration-2/eval-sprint-closeout/with_skill/outputs/result.md +182 -0
  286. package/skills/sync-docs-workspace/iteration-2/eval-sprint-closeout/with_skill/run-1/grading.json +95 -0
  287. package/skills/sync-docs-workspace/iteration-2/eval-sprint-closeout/with_skill/run-1/timing.json +1 -0
  288. package/skills/sync-docs-workspace/iteration-2/eval-sprint-closeout/without_skill/outputs/result.md +222 -0
  289. package/skills/sync-docs-workspace/iteration-2/eval-sprint-closeout/without_skill/run-1/grading.json +88 -0
  290. package/skills/sync-docs-workspace/iteration-2/eval-sprint-closeout/without_skill/run-1/timing.json +5 -0
  291. package/skills/sync-docs-workspace/iteration-3/benchmark.json +298 -0
  292. package/skills/sync-docs-workspace/iteration-3/benchmark.md +13 -0
  293. package/skills/sync-docs-workspace/iteration-3/eval-doc-audit/eval_metadata.json +27 -0
  294. package/skills/sync-docs-workspace/iteration-3/eval-doc-audit/with_skill/outputs/result.md +125 -0
  295. package/skills/sync-docs-workspace/iteration-3/eval-doc-audit/with_skill/run-1/grading.json +97 -0
  296. package/skills/sync-docs-workspace/iteration-3/eval-doc-audit/with_skill/run-1/timing.json +5 -0
  297. package/skills/sync-docs-workspace/iteration-3/eval-doc-audit/without_skill/outputs/result.md +144 -0
  298. package/skills/sync-docs-workspace/iteration-3/eval-doc-audit/without_skill/run-1/grading.json +78 -0
  299. package/skills/sync-docs-workspace/iteration-3/eval-doc-audit/without_skill/run-1/timing.json +5 -0
  300. package/skills/sync-docs-workspace/iteration-3/eval-fix-mode/eval_metadata.json +27 -0
  301. package/skills/sync-docs-workspace/iteration-3/eval-fix-mode/with_skill/outputs/result.md +104 -0
  302. package/skills/sync-docs-workspace/iteration-3/eval-fix-mode/with_skill/run-1/grading.json +91 -0
  303. package/skills/sync-docs-workspace/iteration-3/eval-fix-mode/with_skill/run-1/timing.json +5 -0
  304. package/skills/sync-docs-workspace/iteration-3/eval-fix-mode/without_skill/outputs/result.md +79 -0
  305. package/skills/sync-docs-workspace/iteration-3/eval-fix-mode/without_skill/run-1/grading.json +82 -0
  306. package/skills/sync-docs-workspace/iteration-3/eval-fix-mode/without_skill/run-1/timing.json +5 -0
  307. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/eval_metadata.json +27 -0
  308. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/phase1_context.json +302 -0
  309. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/phase2_drift.txt +33 -0
  310. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/phase3_analysis.json +114 -0
  311. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/phase4_fix.txt +118 -0
  312. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/phase5_validate.txt +38 -0
  313. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/result.md +158 -0
  314. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/run-1/grading.json +95 -0
  315. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/run-1/timing.json +5 -0
  316. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/without_skill/outputs/result.md +71 -0
  317. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/without_skill/run-1/grading.json +90 -0
  318. package/skills/sync-docs-workspace/iteration-3/eval-sprint-closeout/without_skill/run-1/timing.json +5 -0
  319. package/skills/test-planning/SKILL.md +208 -0
  320. package/skills/test-planning/evals/evals.json +23 -0
  321. package/skills/updating-service-skills/SKILL.md +136 -0
  322. package/skills/updating-service-skills/scripts/drift_detector.py +222 -0
  323. package/skills/using-TDD/SKILL.md +410 -0
  324. package/skills/using-quality-gates/SKILL.md +254 -0
  325. package/skills/using-serena-lsp/README.md +8 -0
  326. package/skills/using-serena-lsp/REFERENCE.md +194 -0
  327. package/skills/using-serena-lsp/SKILL.md +82 -0
  328. package/skills/using-service-skills/SKILL.md +108 -0
  329. package/skills/using-service-skills/scripts/cataloger.py +74 -0
  330. package/skills/using-service-skills/scripts/skill_activator.py +152 -0
  331. package/skills/using-service-skills/scripts/test_skill_activator.py +58 -0
  332. package/skills/using-xtrm/SKILL.md +245 -0
  333. package/skills/xt-end/SKILL.md +128 -0
@@ -0,0 +1,327 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Detect documentation drift between docs/ files and git-modified files.
4
+
5
+ A docs file is considered stale when it declares source globs in frontmatter
6
+ (`source_of_truth_for` or `tracks`) and recent commits modified matching files.
7
+
8
+ Subcommands:
9
+ scan [--since N] [--json] — scan all docs files (default N=30 commits)
10
+ check <docs-file> [--since N] [--json] — check one docs file
11
+ hook [--json] — check current uncommitted changes
12
+ """
13
+
14
+ import sys
15
+ import re
16
+ import json
17
+ import fnmatch
18
+ import subprocess
19
+ from pathlib import Path
20
+ from typing import Any
21
+
22
+ try:
23
+ import yaml
24
+ except ImportError: # pragma: no cover
25
+ yaml = None
26
+
27
+
28
+ def find_project_root() -> Path:
29
+ """Walk up from cwd looking for docs/ and .git."""
30
+ p = Path.cwd()
31
+ for parent in [p, *p.parents]:
32
+ if (parent / ".git").exists():
33
+ return parent
34
+ return p
35
+
36
+
37
+ def get_docs_files(project_root: Path) -> list[Path]:
38
+ docs_dir = project_root / "docs"
39
+ if not docs_dir.exists():
40
+ return []
41
+ return sorted(docs_dir.rglob("*.md"))
42
+
43
+
44
+ def extract_frontmatter(content: str) -> dict[str, Any]:
45
+ match = re.match(r"^---\n(.*?)\n---\n", content, re.DOTALL)
46
+ if not match:
47
+ return {}
48
+
49
+ raw = match.group(1)
50
+ if yaml is not None:
51
+ try:
52
+ return yaml.safe_load(raw) or {}
53
+ except Exception:
54
+ return {}
55
+
56
+ # Minimal fallback parser for environments without pyyaml
57
+ fm: dict[str, Any] = {}
58
+ current_key: str | None = None
59
+ for line in raw.splitlines():
60
+ if not line.strip() or line.strip().startswith("#"):
61
+ continue
62
+ if re.match(r"^[A-Za-z0-9_\-]+:\s*", line):
63
+ key, value = line.split(":", 1)
64
+ key = key.strip()
65
+ value = value.strip()
66
+ if not value:
67
+ fm[key] = []
68
+ current_key = key
69
+ else:
70
+ fm[key] = value.strip('"')
71
+ current_key = None
72
+ elif current_key and line.strip().startswith("-"):
73
+ item = line.strip()[1:].strip().strip('"')
74
+ if isinstance(fm.get(current_key), list):
75
+ fm[current_key].append(item)
76
+ return fm
77
+
78
+
79
+ def extract_globs(content: str) -> list[str]:
80
+ fm = extract_frontmatter(content)
81
+ source = fm.get("source_of_truth_for", [])
82
+ tracks = fm.get("tracks", [])
83
+
84
+ globs: list[str] = []
85
+ if isinstance(source, list):
86
+ globs.extend(str(x) for x in source)
87
+ if isinstance(tracks, list):
88
+ for item in tracks:
89
+ s = str(item)
90
+ if s not in globs:
91
+ globs.append(s)
92
+
93
+ return [g for g in globs if g.strip()]
94
+
95
+
96
+ def extract_updated(content: str) -> str:
97
+ fm = extract_frontmatter(content)
98
+ return str(fm.get("updated", ""))
99
+
100
+
101
+ def _match_glob(path: str, pattern: str) -> bool:
102
+ path_parts = Path(path).as_posix().split("/")
103
+ pattern_parts = Path(pattern).as_posix().split("/")
104
+
105
+ def _match(pp: list[str], pat: list[str]) -> bool:
106
+ if not pat:
107
+ return not pp
108
+ if pat[0] == "**":
109
+ for i in range(len(pp) + 1):
110
+ if _match(pp[i:], pat[1:]):
111
+ return True
112
+ return False
113
+ if not pp:
114
+ return False
115
+ return fnmatch.fnmatch(pp[0], pat[0]) and _match(pp[1:], pat[1:])
116
+
117
+ return _match(path_parts, pattern_parts)
118
+
119
+
120
+ def match_files_to_globs(files: list[str], globs: list[str]) -> list[str]:
121
+ matched: list[str] = []
122
+ for file_path in files:
123
+ for pattern in globs:
124
+ if _match_glob(file_path, pattern):
125
+ matched.append(file_path)
126
+ break
127
+ return matched
128
+
129
+
130
+ def get_recent_modified_files(project_root: Path, since_n_commits: int = 30) -> list[str]:
131
+ try:
132
+ result = subprocess.run(
133
+ ["git", "log", f"-{since_n_commits}", "--name-only", "--format="],
134
+ cwd=project_root,
135
+ capture_output=True,
136
+ text=True,
137
+ timeout=10,
138
+ )
139
+ if result.returncode != 0:
140
+ return []
141
+ return [l.strip() for l in result.stdout.splitlines() if l.strip()]
142
+ except Exception:
143
+ return []
144
+
145
+
146
+ def get_session_written_files(project_root: Path) -> list[str]:
147
+ try:
148
+ unstaged = subprocess.run(
149
+ ["git", "diff", "HEAD", "--name-only"],
150
+ cwd=project_root,
151
+ capture_output=True,
152
+ text=True,
153
+ timeout=10,
154
+ )
155
+ staged = subprocess.run(
156
+ ["git", "diff", "--cached", "--name-only"],
157
+ cwd=project_root,
158
+ capture_output=True,
159
+ text=True,
160
+ timeout=10,
161
+ )
162
+ files = unstaged.stdout.splitlines() + staged.stdout.splitlines()
163
+ return sorted({f.strip() for f in files if f.strip()})
164
+ except Exception:
165
+ return []
166
+
167
+
168
+ def scan_docs(project_root: Path, changed_files: list[str]) -> list[dict[str, Any]]:
169
+ stale: list[dict[str, Any]] = []
170
+ for doc_path in get_docs_files(project_root):
171
+ content = doc_path.read_text(encoding="utf-8", errors="replace")
172
+ globs = extract_globs(content)
173
+ if not globs:
174
+ continue
175
+
176
+ matched = match_files_to_globs(changed_files, globs)
177
+ if matched:
178
+ stale.append(
179
+ {
180
+ "doc": str(doc_path.relative_to(project_root)),
181
+ "updated": extract_updated(content),
182
+ "matched_files": matched[:10],
183
+ "globs": globs,
184
+ }
185
+ )
186
+ return stale
187
+
188
+
189
+ def print_human_report(stale: list[dict[str, Any]], source: str) -> None:
190
+ if not stale:
191
+ print(f"[Docs Drift] All docs up to date ({source}).")
192
+ return
193
+
194
+ print(f"[Drift Report] {len(stale)} stale doc(s) detected from {source}:\n")
195
+ for item in stale:
196
+ print(f" {item['doc']}")
197
+ print(f" Last updated: {item['updated'] or 'unknown'}")
198
+ for file_path in item["matched_files"][:3]:
199
+ print(f" Modified: {file_path}")
200
+ print("")
201
+ print("Run /sync-docs to review and update stale docs.")
202
+
203
+
204
+ def cmd_scan(args: list[str]) -> None:
205
+ since = 30
206
+ as_json = "--json" in args
207
+ if "--since" in args:
208
+ idx = args.index("--since")
209
+ if idx + 1 < len(args):
210
+ since = int(args[idx + 1])
211
+
212
+ project_root = find_project_root()
213
+ changed = get_recent_modified_files(project_root, since)
214
+ stale = scan_docs(project_root, changed)
215
+
216
+ if as_json:
217
+ print(
218
+ json.dumps(
219
+ {
220
+ "mode": "scan",
221
+ "since": since,
222
+ "count": len(stale),
223
+ "stale": stale,
224
+ },
225
+ indent=2,
226
+ )
227
+ )
228
+ else:
229
+ print_human_report(stale, f"last {since} commits")
230
+
231
+ sys.exit(1 if stale else 0)
232
+
233
+
234
+ def cmd_check(args: list[str]) -> None:
235
+ if not args:
236
+ print("Usage: drift_detector.py check <docs-file> [--since N] [--json]")
237
+ sys.exit(1)
238
+
239
+ target = args[0]
240
+ as_json = "--json" in args
241
+ since = 30
242
+ if "--since" in args:
243
+ idx = args.index("--since")
244
+ if idx + 1 < len(args):
245
+ since = int(args[idx + 1])
246
+
247
+ project_root = find_project_root()
248
+ doc_path = (project_root / target).resolve()
249
+ if not doc_path.exists():
250
+ print(f"Doc not found: {target}")
251
+ sys.exit(1)
252
+
253
+ changed = get_recent_modified_files(project_root, since)
254
+ content = doc_path.read_text(encoding="utf-8")
255
+ globs = extract_globs(content)
256
+ matched = match_files_to_globs(changed, globs) if globs else []
257
+
258
+ payload = {
259
+ "mode": "check",
260
+ "doc": str(doc_path.relative_to(project_root)),
261
+ "since": since,
262
+ "stale": bool(matched),
263
+ "matched_files": matched[:10],
264
+ "globs": globs,
265
+ }
266
+
267
+ if as_json:
268
+ print(json.dumps(payload, indent=2))
269
+ else:
270
+ if matched:
271
+ print(f"{payload['doc']}: STALE")
272
+ for f in matched[:5]:
273
+ print(f" Modified: {f}")
274
+ else:
275
+ print(f"{payload['doc']}: up to date")
276
+
277
+ sys.exit(1 if matched else 0)
278
+
279
+
280
+ def cmd_hook(args: list[str]) -> None:
281
+ as_json = "--json" in args
282
+ project_root = find_project_root()
283
+ changed = get_session_written_files(project_root)
284
+ if not changed:
285
+ sys.exit(0)
286
+
287
+ stale = scan_docs(project_root, changed)
288
+ if not stale:
289
+ sys.exit(0)
290
+
291
+ if as_json:
292
+ print(
293
+ json.dumps(
294
+ {
295
+ "hookSpecificOutput": {
296
+ "hookEventName": "Stop",
297
+ "additionalContext": (
298
+ f"[Docs Drift] {len(stale)} docs may need updates. "
299
+ "Run /sync-docs to review."
300
+ ),
301
+ }
302
+ }
303
+ )
304
+ )
305
+ else:
306
+ print_human_report(stale, "current session changes")
307
+
308
+ sys.exit(1)
309
+
310
+
311
+ SUBCOMMANDS = {"scan": cmd_scan, "check": cmd_check, "hook": cmd_hook}
312
+
313
+
314
+ def main() -> None:
315
+ args = sys.argv[1:]
316
+ if not args or args[0] not in SUBCOMMANDS:
317
+ print("Usage: drift_detector.py <scan|check|hook> [options]")
318
+ print(" scan [--since N] [--json] scan all docs files")
319
+ print(" check <docs-file> [--since N] check one docs file")
320
+ print(" hook [--json] stop-hook mode")
321
+ sys.exit(1)
322
+
323
+ SUBCOMMANDS[args[0]](args[1:])
324
+
325
+
326
+ if __name__ == "__main__":
327
+ main()
@@ -0,0 +1,365 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Validate and generate schema-compliant docs/ files.
4
+
5
+ Schema for docs/ files:
6
+ Required: title, scope, category, version, updated
7
+ Optional: description, source_of_truth_for (glob list), domain (tag list)
8
+
9
+ Usage:
10
+ validate_doc.py <file_or_dir> # validate one file or all *.md in dir
11
+ validate_doc.py --generate <path> # generate scaffold + required flags below
12
+ --title="..."
13
+ --scope="..."
14
+ --category="reference|guide|architecture|api"
15
+ --source-for="glob1,glob2" # optional
16
+ --description="..." # optional
17
+ """
18
+
19
+ import sys
20
+ import re
21
+ import json
22
+ from pathlib import Path
23
+ from datetime import date
24
+
25
+ # ── Schema ────────────────────────────────────────────────────────────────────
26
+
27
+ REQUIRED_FIELDS = ["title", "scope", "category", "version", "updated"]
28
+ VALID_CATEGORIES = ["reference", "guide", "architecture", "api", "plan", "overview"]
29
+
30
+ CATEGORY_DESCRIPTIONS = {
31
+ "reference": "Look-up table, cheat sheet, or technical specification",
32
+ "guide": "How-to documentation with step-by-step instructions",
33
+ "architecture": "System design, component relationships, high-level overview",
34
+ "api": "API contracts, interfaces, or data schemas",
35
+ "plan": "Implementation plan or roadmap",
36
+ "overview": "Summary introduction to a subsystem",
37
+ }
38
+
39
+
40
+ # ── Frontmatter helpers ───────────────────────────────────────────────────────
41
+
42
+ def extract_frontmatter(content: str) -> dict | None:
43
+ """Parse simple YAML frontmatter without external dependencies.
44
+
45
+ Handles scalar values, quoted strings, and list fields (- item syntax).
46
+ """
47
+ m = re.match(r"^---\n(.*?)\n---\n", content, re.DOTALL)
48
+ if not m:
49
+ return None
50
+
51
+ result: dict[str, object] = {}
52
+ current_key: str | None = None
53
+ current_list: list[str] | None = None
54
+
55
+ for line in m.group(1).splitlines():
56
+ # List item under current key
57
+ if current_list is not None and re.match(r"^\s+-\s+", line):
58
+ current_list.append(re.sub(r"^\s+-\s+", "", line).strip().strip('"\''))
59
+ continue
60
+
61
+ # New key: value line
62
+ kv = re.match(r'^([a-zA-Z_][a-zA-Z0-9_]*)\s*:\s*(.*)', line)
63
+ if kv:
64
+ # Flush previous list
65
+ if current_key is not None and current_list is not None:
66
+ result[current_key] = current_list
67
+
68
+ current_key = kv.group(1)
69
+ raw_val = kv.group(2).strip()
70
+
71
+ if raw_val == "" or raw_val == ">-":
72
+ # Value on following lines (list or multiline) — start list
73
+ current_list = []
74
+ elif raw_val.startswith("["):
75
+ # Inline list: [a, b, c]
76
+ current_list = None
77
+ inner = raw_val.strip("[]")
78
+ result[current_key] = [v.strip().strip('"\'') for v in inner.split(",") if v.strip()]
79
+ else:
80
+ current_list = None
81
+ result[current_key] = raw_val.strip('"\'')
82
+ else:
83
+ # Continuation line for multiline scalar — ignore for our purposes
84
+ pass
85
+
86
+ # Flush trailing list
87
+ if current_key is not None and current_list is not None:
88
+ result[current_key] = current_list
89
+
90
+ return result
91
+
92
+
93
+ def extract_headings(content: str) -> list[tuple[str, str]]:
94
+ """Return (heading, first_sentence) for every ## section."""
95
+ results = []
96
+ lines = content.splitlines()
97
+ i = 0
98
+ while i < len(lines):
99
+ line = lines[i]
100
+ if line.startswith("## ") and not line.startswith("### "):
101
+ heading = line[3:].strip()
102
+ summary = ""
103
+ j = i + 1
104
+ in_code = False
105
+ while j < len(lines):
106
+ ln = lines[j].strip()
107
+ if ln.startswith("```"):
108
+ in_code = not in_code
109
+ j += 1
110
+ continue
111
+ if not in_code and ln and not ln.startswith("#"):
112
+ summary = ln.split(".")[0].strip()[:120]
113
+ break
114
+ j += 1
115
+ results.append((heading, summary))
116
+ i += 1
117
+ return results
118
+
119
+
120
+ def make_anchor(heading: str) -> str:
121
+ """Generate a GitHub-compatible anchor from a heading string."""
122
+ anchor = heading.lower()
123
+ anchor = re.sub(r"\s+", "-", anchor) # spaces → hyphens
124
+ anchor = re.sub(r"[^a-z0-9\-]", "", anchor) # strip non-alphanumeric (except -)
125
+ anchor = re.sub(r"-+", "-", anchor) # collapse runs
126
+ return anchor.strip("-")
127
+
128
+
129
+ def generate_index_table(headings: list[tuple[str, str]]) -> str:
130
+ rows = ["| Section | Summary |", "|---|---|"]
131
+ for heading, summary in headings:
132
+ rows.append(f"| [{heading}](#{make_anchor(heading)}) | {summary or '_no summary_'} |")
133
+ return "\n".join(rows) + "\n"
134
+
135
+
136
+ def inject_index(content: str, table: str) -> str:
137
+ header = "<!-- INDEX: auto-generated by validate_doc.py — do not edit manually -->\n"
138
+ footer = "<!-- END INDEX -->"
139
+ block = f"{header}{table}{footer}"
140
+
141
+ existing = re.search(r"<!-- INDEX:.*?-->.*?<!-- END INDEX -->", content, re.DOTALL)
142
+ if existing:
143
+ return content[: existing.start()] + block + content[existing.end() :]
144
+
145
+ fm_match = re.match(r"^(---\n.*?\n---\n)(.*)", content, re.DOTALL)
146
+ if fm_match:
147
+ return fm_match.group(1) + "\n" + block + "\n" + fm_match.group(2)
148
+
149
+ return block + "\n" + content
150
+
151
+
152
+ # ── Validation ────────────────────────────────────────────────────────────────
153
+
154
+ def validate_file(path: Path) -> tuple[bool, list[str], list[str]]:
155
+ """Returns (passed, errors, warnings)."""
156
+ errors: list[str] = []
157
+ warnings: list[str] = []
158
+
159
+ if not path.exists():
160
+ return False, [f"File not found: {path}"], []
161
+
162
+ content = path.read_text(encoding="utf-8")
163
+ fm = extract_frontmatter(content)
164
+
165
+ if fm is None:
166
+ errors.append("Missing or invalid YAML frontmatter (wrap in --- markers)")
167
+ return False, errors, warnings
168
+
169
+ # Required fields
170
+ for field in REQUIRED_FIELDS:
171
+ if field not in fm:
172
+ errors.append(f"Missing required field: {field}")
173
+
174
+ # version format
175
+ if "version" in fm:
176
+ if not re.match(r"^\d+\.\d+\.\d+$", str(fm["version"])):
177
+ errors.append(f"version must be semver (x.y.z), got: {fm['version']}")
178
+
179
+ # updated format
180
+ if "updated" in fm:
181
+ if not re.match(r"^\d{4}-\d{2}-\d{2}", str(fm["updated"])):
182
+ warnings.append(f"updated should be ISO date (YYYY-MM-DD), got: {fm['updated']}")
183
+
184
+ # category valid
185
+ if "category" in fm and fm["category"] not in VALID_CATEGORIES:
186
+ errors.append(
187
+ f"category '{fm['category']}' not valid. Choose from: {', '.join(VALID_CATEGORIES)}"
188
+ )
189
+
190
+ # domain is list
191
+ if "domain" in fm and not isinstance(fm["domain"], list):
192
+ errors.append("domain must be a list, e.g. [hooks, claude]")
193
+
194
+ # source_of_truth_for and tracks are lists of globs
195
+ if "source_of_truth_for" in fm and not isinstance(fm["source_of_truth_for"], list):
196
+ errors.append("source_of_truth_for must be a list of glob patterns")
197
+ if "tracks" in fm and not isinstance(fm["tracks"], list):
198
+ errors.append("tracks must be a list of glob patterns")
199
+
200
+ # Regenerate INDEX if valid
201
+ if not errors:
202
+ headings = extract_headings(content)
203
+ if headings:
204
+ table = generate_index_table(headings)
205
+ new_content = inject_index(content, table)
206
+ if new_content != content:
207
+ path.write_text(new_content, encoding="utf-8")
208
+ warnings.append("INDEX regenerated")
209
+
210
+ return len(errors) == 0, errors, warnings
211
+
212
+
213
+ def validate_directory(docs_dir: Path) -> dict:
214
+ results = {}
215
+ for md_file in sorted(docs_dir.glob("*.md")):
216
+ passed, errors, warnings = validate_file(md_file)
217
+ results[str(md_file.relative_to(docs_dir.parent))] = {
218
+ "passed": passed,
219
+ "errors": errors,
220
+ "warnings": warnings,
221
+ }
222
+ return results
223
+
224
+
225
+ def print_file_result(path: str, passed: bool, errors: list[str], warnings: list[str]) -> None:
226
+ status = "PASS" if passed else "FAIL"
227
+ mark = "" if passed else ""
228
+ print(f"\n{mark} {path} [{status}]")
229
+ for e in errors:
230
+ print(f" ERROR: {e}")
231
+ for w in warnings:
232
+ print(f" WARN: {w}")
233
+ if passed and not warnings:
234
+ print(" All checks passed.")
235
+
236
+
237
+ # ── Generator ─────────────────────────────────────────────────────────────────
238
+
239
+ SCAFFOLD_TEMPLATE = """\
240
+ ---
241
+ title: {title}
242
+ scope: {scope}
243
+ category: {category}
244
+ version: 1.0.0
245
+ updated: {today}
246
+ {source_field}{tracks_field}{description_field}domain: []
247
+ ---
248
+
249
+ <!-- INDEX: auto-generated by validate_doc.py — do not edit manually -->
250
+ <!-- END INDEX -->
251
+
252
+ # {title}
253
+
254
+ > {category_desc}
255
+
256
+ ## Overview
257
+
258
+ _Describe what this document covers._
259
+
260
+ """
261
+
262
+
263
+ def generate_scaffold(output_path: Path, title: str, scope: str, category: str,
264
+ source_for: list[str], description: str) -> None:
265
+ source_field = ""
266
+ tracks_field = ""
267
+ if source_for:
268
+ items = "\n".join(f' - "{g}"' for g in source_for)
269
+ source_field = f"source_of_truth_for:\n{items}\n"
270
+ # tracks: mirrors source_of_truth_for so drift_detector.py picks up changes
271
+ tracks_field = f"tracks:\n{items}\n"
272
+
273
+ desc_field = f'description: "{description}"\n' if description else ""
274
+ category_desc = CATEGORY_DESCRIPTIONS.get(category, category)
275
+
276
+ content = SCAFFOLD_TEMPLATE.format(
277
+ title=title,
278
+ scope=scope,
279
+ category=category,
280
+ today=date.today().isoformat(),
281
+ source_field=source_field,
282
+ tracks_field=tracks_field,
283
+ description_field=desc_field,
284
+ category_desc=category_desc,
285
+ )
286
+
287
+ output_path.parent.mkdir(parents=True, exist_ok=True)
288
+ output_path.write_text(content, encoding="utf-8")
289
+ print(f"Generated: {output_path}")
290
+
291
+
292
+ # ── Entry point ───────────────────────────────────────────────────────────────
293
+
294
+ def main() -> None:
295
+ args = sys.argv[1:]
296
+
297
+ if not args:
298
+ print("Usage:")
299
+ print(" validate_doc.py <file_or_dir>")
300
+ print(" validate_doc.py --generate <path> --title=... --scope=... --category=...")
301
+ sys.exit(1)
302
+
303
+ # Generate mode
304
+ if "--generate" in args:
305
+ idx = args.index("--generate")
306
+ if idx + 1 >= len(args):
307
+ print("ERROR: --generate requires a path argument")
308
+ sys.exit(1)
309
+ output_path = Path(args[idx + 1])
310
+
311
+ kw: dict[str, str] = {}
312
+ source_for: list[str] = []
313
+ for arg in args:
314
+ if arg.startswith("--title="):
315
+ kw["title"] = arg.split("=", 1)[1]
316
+ elif arg.startswith("--scope="):
317
+ kw["scope"] = arg.split("=", 1)[1]
318
+ elif arg.startswith("--category="):
319
+ kw["category"] = arg.split("=", 1)[1]
320
+ elif arg.startswith("--source-for="):
321
+ source_for = [g.strip() for g in arg.split("=", 1)[1].split(",")]
322
+ elif arg.startswith("--description="):
323
+ kw["description"] = arg.split("=", 1)[1]
324
+
325
+ for req in ["title", "scope", "category"]:
326
+ if req not in kw:
327
+ print(f"ERROR: --{req} is required for --generate")
328
+ sys.exit(1)
329
+
330
+ generate_scaffold(
331
+ output_path,
332
+ title=kw["title"],
333
+ scope=kw["scope"],
334
+ category=kw["category"],
335
+ source_for=source_for,
336
+ description=kw.get("description", ""),
337
+ )
338
+ sys.exit(0)
339
+
340
+ # Validate mode
341
+ target = Path(args[0])
342
+ all_passed = True
343
+
344
+ if target.is_dir():
345
+ results = validate_directory(target)
346
+ for path_str, res in results.items():
347
+ print_file_result(path_str, res["passed"], res["errors"], res["warnings"])
348
+ if not res["passed"]:
349
+ all_passed = False
350
+ if results:
351
+ total = len(results)
352
+ passed = sum(1 for r in results.values() if r["passed"])
353
+ print(f"\nResult: {passed}/{total} files passed")
354
+ else:
355
+ print(f"No .md files found in {target}")
356
+ else:
357
+ passed, errors, warnings = validate_file(target)
358
+ print_file_result(str(target), passed, errors, warnings)
359
+ all_passed = passed
360
+
361
+ sys.exit(0 if all_passed else 1)
362
+
363
+
364
+ if __name__ == "__main__":
365
+ main()