savepoint 1.0.4 → 1.0.6

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 (425) hide show
  1. package/package.json +10 -3
  2. package/savepoint.exe +0 -0
  3. package/.claude/settings.local.json +0 -38
  4. package/.github/workflows/ci.yml +0 -20
  5. package/.golangci.yml +0 -11
  6. package/.prettierignore +0 -4
  7. package/.savepoint/Design.md +0 -207
  8. package/.savepoint/PRD.md +0 -58
  9. package/.savepoint/config.yml +0 -27
  10. package/.savepoint/releases/v1/epics/E01-go-setup/E01-Detail.md +0 -39
  11. package/.savepoint/releases/v1/epics/E01-go-setup/tasks/T001-init-module.md +0 -42
  12. package/.savepoint/releases/v1/epics/E01-go-setup/tasks/T002-entrypoint.md +0 -23
  13. package/.savepoint/releases/v1/epics/E01-go-setup/tasks/T003-directory-structure.md +0 -24
  14. package/.savepoint/releases/v1/epics/E01-go-setup/tasks/T004-makefile.md +0 -23
  15. package/.savepoint/releases/v1/epics/E02-data-readers/E02-Detail.md +0 -61
  16. package/.savepoint/releases/v1/epics/E02-data-readers/tasks/T001-task-struct.md +0 -29
  17. package/.savepoint/releases/v1/epics/E02-data-readers/tasks/T002-frontmatter-parser.md +0 -30
  18. package/.savepoint/releases/v1/epics/E02-data-readers/tasks/T003-router-reader.md +0 -29
  19. package/.savepoint/releases/v1/epics/E02-data-readers/tasks/T004-config-reader.md +0 -29
  20. package/.savepoint/releases/v1/epics/E02-data-readers/tasks/T005-discovery.md +0 -30
  21. package/.savepoint/releases/v1/epics/E03-board-tui-core/E03-Detail.md +0 -38
  22. package/.savepoint/releases/v1/epics/E03-board-tui-core/tasks/T001-model.md +0 -29
  23. package/.savepoint/releases/v1/epics/E03-board-tui-core/tasks/T002-update-loop.md +0 -30
  24. package/.savepoint/releases/v1/epics/E03-board-tui-core/tasks/T003-view.md +0 -34
  25. package/.savepoint/releases/v1/epics/E03-board-tui-core/tasks/T004-styles.md +0 -29
  26. package/.savepoint/releases/v1/epics/E03-board-tui-core/tasks/T005-layout.md +0 -42
  27. package/.savepoint/releases/v1/epics/E04-board-components/E04-Detail.md +0 -44
  28. package/.savepoint/releases/v1/epics/E04-board-components/tasks/T001-column.md +0 -34
  29. package/.savepoint/releases/v1/epics/E04-board-components/tasks/T002-card.md +0 -33
  30. package/.savepoint/releases/v1/epics/E04-board-components/tasks/T003-epic-panel.md +0 -49
  31. package/.savepoint/releases/v1/epics/E04-board-components/tasks/T004-detail-overlay.md +0 -40
  32. package/.savepoint/releases/v1/epics/E04-board-components/tasks/T005-release-dropdown.md +0 -33
  33. package/.savepoint/releases/v1/epics/E04-board-components/tasks/T006-help-overlay.md +0 -34
  34. package/.savepoint/releases/v1/epics/E05-phase-transitions/E05-Detail.md +0 -38
  35. package/.savepoint/releases/v1/epics/E05-phase-transitions/tasks/T001-phase-stepping.md +0 -29
  36. package/.savepoint/releases/v1/epics/E05-phase-transitions/tasks/T002-gates.md +0 -31
  37. package/.savepoint/releases/v1/epics/E05-phase-transitions/tasks/T003-write-task.md +0 -31
  38. package/.savepoint/releases/v1/epics/E05-phase-transitions/tasks/T004-write-router.md +0 -31
  39. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/E06-Detail.md +0 -62
  40. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T001-color-system.md +0 -39
  41. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T002-header-and-dividers.md +0 -52
  42. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T003-footer-status-bar.md +0 -52
  43. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T004-component-refinement.md +0 -53
  44. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T005-restore-nav-hints.md +0 -39
  45. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T007-detail-card-fixes.md +0 -36
  46. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T008-checkbox-states.md +0 -40
  47. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T009-router-priority-marker.md +0 -48
  48. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T010-auto-refresh-watcher.md +0 -66
  49. package/.savepoint/releases/v1/epics/_archived/E01-archive-and-reset/Design.md +0 -39
  50. package/.savepoint/releases/v1/epics/_archived/E01-archive-and-reset/tasks/T001-archive-epics.md +0 -20
  51. package/.savepoint/releases/v1/epics/_archived/E01-archive-and-reset/tasks/T002-rewrite-prd.md +0 -22
  52. package/.savepoint/releases/v1/epics/_archived/E01-archive-and-reset/tasks/T003-create-epic-stubs.md +0 -24
  53. package/.savepoint/releases/v1/epics/_archived/E01-archive-and-reset/tasks/T004-update-router.md +0 -22
  54. package/.savepoint/releases/v1/epics/_archived/E01-scaffolding/Design.md +0 -118
  55. package/.savepoint/releases/v1/epics/_archived/E01-scaffolding/handoff.md +0 -9
  56. package/.savepoint/releases/v1/epics/_archived/E01-scaffolding/tasks/T001-package-baseline.md +0 -45
  57. package/.savepoint/releases/v1/epics/_archived/E01-scaffolding/tasks/T002-typescript-build.md +0 -48
  58. package/.savepoint/releases/v1/epics/_archived/E01-scaffolding/tasks/T003-vitest-smoke.md +0 -43
  59. package/.savepoint/releases/v1/epics/_archived/E01-scaffolding/tasks/T004-lint-format-gates.md +0 -45
  60. package/.savepoint/releases/v1/epics/_archived/E01-scaffolding/tasks/T005-scaffold-verification.md +0 -40
  61. package/.savepoint/releases/v1/epics/_archived/E02-data-model/Design.md +0 -142
  62. package/.savepoint/releases/v1/epics/_archived/E02-data-model/tasks/T001-domain-ids-status.md +0 -27
  63. package/.savepoint/releases/v1/epics/_archived/E02-data-model/tasks/T002-markdown-frontmatter-boundary.md +0 -28
  64. package/.savepoint/releases/v1/epics/_archived/E02-data-model/tasks/T003-task-documents.md +0 -29
  65. package/.savepoint/releases/v1/epics/_archived/E02-data-model/tasks/T004-release-epic-router-config-readers.md +0 -30
  66. package/.savepoint/releases/v1/epics/_archived/E02-data-model/tasks/T005-dependency-validation.md +0 -29
  67. package/.savepoint/releases/v1/epics/_archived/E02-data-model/tasks/T006-epic-task-set-reader.md +0 -29
  68. package/.savepoint/releases/v1/epics/_archived/E02-data-model/tasks/T007-quality-gates.md +0 -31
  69. package/.savepoint/releases/v1/epics/_archived/E02-domain-phase-model/Design.md +0 -40
  70. package/.savepoint/releases/v1/epics/_archived/E02-domain-phase-model/tasks/T001-phase-types.md +0 -27
  71. package/.savepoint/releases/v1/epics/_archived/E02-domain-phase-model/tasks/T002-phase-frontmatter.md +0 -25
  72. package/.savepoint/releases/v1/epics/_archived/E02-domain-phase-model/tasks/T003-simplify-config.md +0 -26
  73. package/.savepoint/releases/v1/epics/_archived/E02-domain-phase-model/tasks/T004-simplify-router-domain.md +0 -24
  74. package/.savepoint/releases/v1/epics/_archived/E03-cli-foundation/Design.md +0 -122
  75. package/.savepoint/releases/v1/epics/_archived/E03-cli-foundation/tasks/T001-argument-parser-contract.md +0 -28
  76. package/.savepoint/releases/v1/epics/_archived/E03-cli-foundation/tasks/T002-help-text-generation.md +0 -28
  77. package/.savepoint/releases/v1/epics/_archived/E03-cli-foundation/tasks/T003-terminal-environment-detection.md +0 -27
  78. package/.savepoint/releases/v1/epics/_archived/E03-cli-foundation/tasks/T004-command-stub-modules.md +0 -29
  79. package/.savepoint/releases/v1/epics/_archived/E03-cli-foundation/tasks/T005-cli-runner-dispatch.md +0 -34
  80. package/.savepoint/releases/v1/epics/_archived/E03-cli-foundation/tasks/T006-entrypoint-quality-gates.md +0 -32
  81. package/.savepoint/releases/v1/epics/_archived/E03-cli-simplify/Design.md +0 -43
  82. package/.savepoint/releases/v1/epics/_archived/E03-cli-simplify/tasks/T001-strip-args.md +0 -26
  83. package/.savepoint/releases/v1/epics/_archived/E03-cli-simplify/tasks/T002-strip-help.md +0 -23
  84. package/.savepoint/releases/v1/epics/_archived/E03-cli-simplify/tasks/T003-strip-run.md +0 -23
  85. package/.savepoint/releases/v1/epics/_archived/E03-cli-simplify/tasks/T004-delete-commands.md +0 -24
  86. package/.savepoint/releases/v1/epics/_archived/E03-cli-simplify/tasks/T005-update-cli-tests.md +0 -22
  87. package/.savepoint/releases/v1/epics/_archived/E04-board-phase-integration/Design.md +0 -48
  88. package/.savepoint/releases/v1/epics/_archived/E04-board-phase-integration/tasks/T001-board-data-phases.md +0 -26
  89. package/.savepoint/releases/v1/epics/_archived/E04-board-phase-integration/tasks/T002-phase-rendering.md +0 -28
  90. package/.savepoint/releases/v1/epics/_archived/E04-board-phase-integration/tasks/T003-detail-pane-phases.md +0 -27
  91. package/.savepoint/releases/v1/epics/_archived/E04-board-phase-integration/tasks/T004-phase-transitions.md +0 -42
  92. package/.savepoint/releases/v1/epics/_archived/E04-board-phase-integration/tasks/T005-phase-gates.md +0 -24
  93. package/.savepoint/releases/v1/epics/_archived/E04-board-phase-integration/tasks/T006-phase-write-back.md +0 -24
  94. package/.savepoint/releases/v1/epics/_archived/E04-board-phase-integration/tasks/T007-remove-audit-flow.md +0 -27
  95. package/.savepoint/releases/v1/epics/_archived/E04-board-phase-integration/tasks/T008-board-tests.md +0 -25
  96. package/.savepoint/releases/v1/epics/_archived/E04-templates-and-prompts/Design.md +0 -85
  97. package/.savepoint/releases/v1/epics/_archived/E04-templates-and-prompts/tasks/T001-project-template-assets.md +0 -17
  98. package/.savepoint/releases/v1/epics/_archived/E04-templates-and-prompts/tasks/T002-release-and-prompt-assets.md +0 -20
  99. package/.savepoint/releases/v1/epics/_archived/E04-templates-and-prompts/tasks/T003-template-registry-renderer.md +0 -22
  100. package/.savepoint/releases/v1/epics/_archived/E04-templates-and-prompts/tasks/T004-template-integrity-tests.md +0 -17
  101. package/.savepoint/releases/v1/epics/_archived/E04-templates-and-prompts/tasks/T005-template-closeout-quality-gates.md +0 -16
  102. package/.savepoint/releases/v1/epics/_archived/E05-init-command/Design.md +0 -88
  103. package/.savepoint/releases/v1/epics/_archived/E05-init-command/tasks/T001-init-cli-contract.md +0 -22
  104. package/.savepoint/releases/v1/epics/_archived/E05-init-command/tasks/T002-target-validation.md +0 -23
  105. package/.savepoint/releases/v1/epics/_archived/E05-init-command/tasks/T003-scaffold-writer.md +0 -24
  106. package/.savepoint/releases/v1/epics/_archived/E05-init-command/tasks/T004-magic-prompt-and-clipboard.md +0 -23
  107. package/.savepoint/releases/v1/epics/_archived/E05-init-command/tasks/T005-dev-deps-install-option.md +0 -24
  108. package/.savepoint/releases/v1/epics/_archived/E05-init-command/tasks/T006-init-command-integration.md +0 -28
  109. package/.savepoint/releases/v1/epics/_archived/E05-project-cleanup/Design.md +0 -53
  110. package/.savepoint/releases/v1/epics/_archived/E05-project-cleanup/tasks/T001-delete-dead-src.md +0 -23
  111. package/.savepoint/releases/v1/epics/_archived/E05-project-cleanup/tasks/T002-delete-dead-tests.md +0 -26
  112. package/.savepoint/releases/v1/epics/_archived/E05-project-cleanup/tasks/T003-delete-assets.md +0 -25
  113. package/.savepoint/releases/v1/epics/_archived/E05-project-cleanup/tasks/T004-clean-savepoint.md +0 -28
  114. package/.savepoint/releases/v1/epics/_archived/E05-project-cleanup/tasks/T005-rewrite-agents-md.md +0 -28
  115. package/.savepoint/releases/v1/epics/_archived/E05-project-cleanup/tasks/T006-clean-package-json.md +0 -23
  116. package/.savepoint/releases/v1/epics/_archived/E05-project-cleanup/tasks/T007-verify.md +0 -25
  117. package/.savepoint/releases/v1/epics/_archived/E06-tui-board/Design.md +0 -104
  118. package/.savepoint/releases/v1/epics/_archived/E06-tui-board/tasks/T001-board-command-data.md +0 -23
  119. package/.savepoint/releases/v1/epics/_archived/E06-tui-board/tasks/T002-board-view-state.md +0 -24
  120. package/.savepoint/releases/v1/epics/_archived/E06-tui-board/tasks/T003-transition-gates-and-writes.md +0 -25
  121. package/.savepoint/releases/v1/epics/_archived/E06-tui-board/tasks/T004-terminal-theme.md +0 -23
  122. package/.savepoint/releases/v1/epics/_archived/E06-tui-board/tasks/T005-ink-board-ui.md +0 -26
  123. package/.savepoint/releases/v1/epics/_archived/E06-tui-board/tasks/T006-board-integration-audit-entry.md +0 -24
  124. package/.savepoint/releases/v1/epics/_archived/E07-audit-pipeline/Design.md +0 -88
  125. package/.savepoint/releases/v1/epics/_archived/E07-audit-pipeline/tasks/T001-audit-cli-contract.md +0 -23
  126. package/.savepoint/releases/v1/epics/_archived/E07-audit-pipeline/tasks/T002-quality-gate-runner.md +0 -23
  127. package/.savepoint/releases/v1/epics/_archived/E07-audit-pipeline/tasks/T003-snapshot-and-prompt.md +0 -23
  128. package/.savepoint/releases/v1/epics/_archived/E07-audit-pipeline/tasks/T004-audit-orchestration-router.md +0 -27
  129. package/.savepoint/releases/v1/epics/_archived/E07-audit-pipeline/tasks/T005-proposal-validation-apply.md +0 -25
  130. package/.savepoint/releases/v1/epics/_archived/E07-audit-pipeline/tasks/T006-audit-review-state.md +0 -24
  131. package/.savepoint/releases/v1/epics/_archived/E07-audit-pipeline/tasks/T007-audit-review-ui.md +0 -26
  132. package/.savepoint/releases/v1/epics/_archived/E07-audit-pipeline/tasks/T008-audit-pipeline-integration.md +0 -24
  133. package/.savepoint/releases/v1/epics/_archived/E08-board-workflow-cleanup/Design.md +0 -103
  134. package/.savepoint/releases/v1/epics/_archived/E08-board-workflow-cleanup/tasks/T001-acceptance-criteria-model.md +0 -30
  135. package/.savepoint/releases/v1/epics/_archived/E08-board-workflow-cleanup/tasks/T002-release-task-set-reader.md +0 -33
  136. package/.savepoint/releases/v1/epics/_archived/E08-board-workflow-cleanup/tasks/T003-board-data-and-plain-output.md +0 -34
  137. package/.savepoint/releases/v1/epics/_archived/E08-board-workflow-cleanup/tasks/T004-board-selection-state.md +0 -33
  138. package/.savepoint/releases/v1/epics/_archived/E08-board-workflow-cleanup/tasks/T005-ink-board-layout-cleanup.md +0 -37
  139. package/.savepoint/releases/v1/epics/_archived/E08-board-workflow-cleanup/tasks/T006-task-detail-popup.md +0 -36
  140. package/.savepoint/releases/v1/epics/_archived/E08-board-workflow-cleanup/tasks/T007-templates-acceptance-criteria.md +0 -34
  141. package/.savepoint/releases/v1/epics/_archived/E08-board-workflow-cleanup/tasks/T008-board-workflow-integration.md +0 -41
  142. package/.savepoint/releases/v1/epics/_archived/E09-doctor-command/Design.md +0 -70
  143. package/.savepoint/releases/v1/epics/_archived/E10-docs-and-packaging/Design.md +0 -68
  144. package/.savepoint/releases/v1/epics/_archived/E11-release-validation/Design.md +0 -68
  145. package/.savepoint/releases/v1/v1-PRD.md +0 -66
  146. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/E01-Detail.md +0 -40
  147. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T001-next-activity-header.md +0 -56
  148. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T002-rename-epic-design-files.md +0 -38
  149. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T003-rename-release-prd.md +0 -28
  150. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T004-update-instruction-files.md +0 -51
  151. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T005-update-cross-references.md +0 -45
  152. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T006-column-and-detail-scrolling.md +0 -68
  153. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T007-column-focus-border-stability.md +0 -57
  154. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/E02-Audit.md +0 -124
  155. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/E02-Detail.md +0 -49
  156. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/tasks/T001-fix-makefile.md +0 -37
  157. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/tasks/T002-linux-build-target.md +0 -38
  158. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/tasks/T003-macos-build-target.md +0 -36
  159. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/tasks/T004-smoke-tests-and-artifacts.md +0 -59
  160. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/E03-Audit.md +0 -195
  161. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/E03-Detail.md +0 -45
  162. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/tasks/T001-border-resize-fix.md +0 -40
  163. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/tasks/T002-next-activity-below-header.md +0 -64
  164. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/tasks/T003-checkbox-rendering-fix.md +0 -56
  165. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/tasks/T005-unify-status-glyphs.md +0 -65
  166. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/tasks/T006-forced-256-color-profile.md +0 -36
  167. package/.savepoint/releases/v1.1/epics/E04-epic-navigation/E04-Audit.md +0 -167
  168. package/.savepoint/releases/v1.1/epics/E04-epic-navigation/E04-Detail.md +0 -51
  169. package/.savepoint/releases/v1.1/epics/E04-epic-navigation/tasks/T001-sidebar-focusable-navigation.md +0 -65
  170. package/.savepoint/releases/v1.1/epics/E04-epic-navigation/tasks/T002-epic-detail-overlay.md +0 -73
  171. package/.savepoint/releases/v1.1/epics/E04-epic-navigation/tasks/T003-epic-status-glyphs.md +0 -73
  172. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/E05-Audit.md +0 -237
  173. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/E05-Detail.md +0 -54
  174. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T001-update-agents-md.md +0 -45
  175. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T002-update-router-md.md +0 -40
  176. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T003-update-design-md.md +0 -47
  177. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T004-implement-m-hotkey.md +0 -98
  178. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T005-update-help-overlay.md +0 -33
  179. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T006-tests-and-quality-gates.md +0 -62
  180. package/.savepoint/releases/v1.1/epics/E06-audit-command/E06-Audit.md +0 -56
  181. package/.savepoint/releases/v1.1/epics/E06-audit-command/E06-Detail.md +0 -63
  182. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T005-proposals.md +0 -44
  183. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T007-apply-close.md +0 -35
  184. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T009-integration.md +0 -40
  185. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T010-audit-file-migration.md +0 -45
  186. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T011-model-tab-state.md +0 -26
  187. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T012-epic-audit-render.md +0 -33
  188. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T013-handle-tab-keys.md +0 -34
  189. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T014-tab-indicator.md +0 -33
  190. package/.savepoint/releases/v1.1/epics/E07-init-command/E07-Audit.md +0 -336
  191. package/.savepoint/releases/v1.1/epics/E07-init-command/E07-Detail.md +0 -61
  192. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T001-cli-entrypoint.md +0 -37
  193. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T002-target-validation.md +0 -28
  194. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T003-scaffold-writer.md +0 -46
  195. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T004-atomic-writes.md +0 -27
  196. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T005-magic-prompt.md +0 -25
  197. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T006-clipboard.md +0 -26
  198. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T007-integration-test.md +0 -26
  199. package/.savepoint/releases/v1.1/epics/E08-board-command/E08-Audit.md +0 -333
  200. package/.savepoint/releases/v1.1/epics/E08-board-command/E08-Detail.md +0 -68
  201. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T001-cli-entrypoint.md +0 -26
  202. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T002-non-tty-fallback.md +0 -27
  203. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T003-tui-app-shell.md +0 -28
  204. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T004-board-model.md +0 -29
  205. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T005-detail-pane.md +0 -27
  206. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T006-status-transitions.md +0 -29
  207. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T007-theme-fallbacks.md +0 -29
  208. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T008-integration-test.md +0 -27
  209. package/.savepoint/releases/v1.1/epics/E09-doctor-command/E09-Audit.md +0 -207
  210. package/.savepoint/releases/v1.1/epics/E09-doctor-command/E09-Detail.md +0 -65
  211. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T001-cli-entrypoint.md +0 -24
  212. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T002-config-router-validation.md +0 -28
  213. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T003-structure-checks.md +0 -29
  214. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T004-dependency-checks.md +0 -27
  215. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T005-audit-orphan-checks.md +0 -28
  216. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T006-quality-gates-report.md +0 -31
  217. package/.savepoint/releases/v1.1/epics/E11-board-refresh-fix/E11-Detail.md +0 -36
  218. package/.savepoint/releases/v1.1/epics/E11-board-refresh-fix/tasks/T001-debug-logging.md +0 -25
  219. package/.savepoint/releases/v1.1/epics/E11-board-refresh-fix/tasks/T002-increase-debounce.md +0 -21
  220. package/.savepoint/releases/v1.1/epics/E11-board-refresh-fix/tasks/T003-error-handling.md +0 -22
  221. package/.savepoint/releases/v1.1/epics/E11-board-refresh-fix/tasks/T004-test-verify.md +0 -29
  222. package/.savepoint/releases/v1.1/epics/E12-validation-fix/E12-Audit.md +0 -444
  223. package/.savepoint/releases/v1.1/epics/E12-validation-fix/E12-Detail.md +0 -45
  224. package/.savepoint/releases/v1.1/epics/E12-validation-fix/tasks/T001-default-phase.md +0 -35
  225. package/.savepoint/releases/v1.1/epics/E12-validation-fix/tasks/T002-default-status.md +0 -19
  226. package/.savepoint/releases/v1.1/epics/E12-validation-fix/tasks/T003-better-errors.md +0 -29
  227. package/.savepoint/releases/v1.1/epics/E12-validation-fix/tasks/T004-validate-on-write.md +0 -25
  228. package/.savepoint/releases/v1.1/epics/E12-validation-fix/tasks/T005-tests.md +0 -37
  229. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/E13-Audit.md +0 -118
  230. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/E13-Detail.md +0 -73
  231. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T001-safe-cleanup.md +0 -66
  232. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T002-bug-fixes.md +0 -35
  233. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T003-centralize-duplication.md +0 -60
  234. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T004-infrastructure.md +0 -33
  235. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T005-decompose-update.md +0 -37
  236. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T006-async-io.md +0 -40
  237. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T007-test-coverage.md +0 -37
  238. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/E14-Audit.md +0 -267
  239. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/E14-Detail.md +0 -54
  240. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T001-group-model.md +0 -39
  241. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T002-data-interfaces.md +0 -42
  242. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T003-discover-orphans.md +0 -33
  243. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T004-epic-panel-headings.md +0 -35
  244. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T005-shell-tokenization.md +0 -27
  245. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T006-unify-enums.md +0 -29
  246. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T007-testutil-package.md +0 -28
  247. package/.savepoint/releases/v1.1/epics/E15-hardening/E15-Audit.md +0 -272
  248. package/.savepoint/releases/v1.1/epics/E15-hardening/E15-Detail.md +0 -60
  249. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T001-benchmarks.md +0 -31
  250. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T002-fuzz-targets.md +0 -34
  251. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T003-debug-flag.md +0 -30
  252. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T004-dist-checksums.md +0 -27
  253. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T005-windows-targets.md +0 -28
  254. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T006-abbreviation-splitting.md +0 -26
  255. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T007-root-test-allowlist.md +0 -33
  256. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T008-ci-and-release-automation.md +0 -46
  257. package/.savepoint/releases/v1.1/epics/_archived/T001-cli-entrypoint.md +0 -25
  258. package/.savepoint/releases/v1.1/epics/_archived/T002-quality-gates.md +0 -27
  259. package/.savepoint/releases/v1.1/epics/_archived/T003-snapshot.md +0 -27
  260. package/.savepoint/releases/v1.1/epics/_archived/T004-ai-reconcile.md +0 -29
  261. package/.savepoint/releases/v1.1/epics/_archived/T006-tui-review.md +0 -31
  262. package/.savepoint/releases/v1.1/epics/_archived/T008-skip-handling.md +0 -34
  263. package/.savepoint/releases/v1.1/v1.1-PRD.md +0 -139
  264. package/.savepoint/router.md +0 -57
  265. package/.savepoint/visual-identity.md +0 -125
  266. package/AGENTS.md +0 -99
  267. package/CLAUDE.md +0 -1
  268. package/GEMINI.md +0 -1
  269. package/Makefile +0 -31
  270. package/agent-skills/ink-tui-design/SKILL.md +0 -309
  271. package/agent-skills/ink-tui-design/references/component-patterns.md +0 -371
  272. package/agent-skills/ink-tui-design/references/hooks-guide.md +0 -436
  273. package/agent-skills/ink-tui-design/references/ink-gotchas.md +0 -330
  274. package/agent-skills/ink-tui-design/references/testing-patterns.md +0 -384
  275. package/agent-skills/savepoint-audit/SKILL.md +0 -87
  276. package/agent-skills/savepoint-build-task/SKILL.md +0 -44
  277. package/agent-skills/savepoint-create-plan/SKILL.md +0 -33
  278. package/agent-skills/savepoint-create-task/SKILL.md +0 -44
  279. package/agent-skills/savepoint-draft-prd/SKILL.md +0 -37
  280. package/agent-skills/savepoint-system-design/SKILL.md +0 -38
  281. package/agent-skills/superpowers/brainstorming/SKILL.md +0 -165
  282. package/agent-skills/superpowers/brainstorming/visual-companion.md +0 -304
  283. package/agent-skills/superpowers/dispatching-parallel-agents/SKILL.md +0 -193
  284. package/agent-skills/superpowers/executing-plans/SKILL.md +0 -77
  285. package/agent-skills/superpowers/finishing-a-development-branch/SKILL.md +0 -213
  286. package/agent-skills/superpowers/receiving-code-review/SKILL.md +0 -226
  287. package/agent-skills/superpowers/requesting-code-review/SKILL.md +0 -115
  288. package/agent-skills/superpowers/requesting-code-review/code-reviewer.md +0 -160
  289. package/agent-skills/superpowers/subagent-driven-development/SKILL.md +0 -292
  290. package/agent-skills/superpowers/subagent-driven-development/code-quality-reviewer-prompt.md +0 -27
  291. package/agent-skills/superpowers/subagent-driven-development/implementer-prompt.md +0 -113
  292. package/agent-skills/superpowers/subagent-driven-development/spec-reviewer-prompt.md +0 -61
  293. package/agent-skills/superpowers/systematic-debugging/SKILL.md +0 -305
  294. package/agent-skills/superpowers/systematic-debugging/condition-based-waiting.md +0 -122
  295. package/agent-skills/superpowers/systematic-debugging/defense-in-depth.md +0 -130
  296. package/agent-skills/superpowers/systematic-debugging/root-cause-tracing.md +0 -183
  297. package/agent-skills/superpowers/test-driven-development/SKILL.md +0 -389
  298. package/agent-skills/superpowers/test-driven-development/testing-anti-patterns.md +0 -317
  299. package/agent-skills/superpowers/verification-before-completion/SKILL.md +0 -147
  300. package/agent-skills/superpowers/writing-plans/SKILL.md +0 -159
  301. package/agent-skills/superpowers/writing-plans/plan-document-reviewer-prompt.md +0 -49
  302. package/agent_skills_test.go +0 -91
  303. package/assets/banner.png +0 -0
  304. package/assets/logo.png +0 -0
  305. package/assets/strawman.png +0 -0
  306. package/cmd/board.go +0 -59
  307. package/cmd/board_test.go +0 -137
  308. package/cmd/doctor.go +0 -53
  309. package/cmd/doctor_test.go +0 -146
  310. package/cmd/init.go +0 -63
  311. package/cmd/init_test.go +0 -104
  312. package/go.mod +0 -36
  313. package/go.sum +0 -75
  314. package/internal/board/board.go +0 -181
  315. package/internal/board/board_test.go +0 -168
  316. package/internal/board/card.go +0 -129
  317. package/internal/board/card_test.go +0 -287
  318. package/internal/board/column.go +0 -156
  319. package/internal/board/column_test.go +0 -210
  320. package/internal/board/debug.go +0 -26
  321. package/internal/board/debug_test.go +0 -108
  322. package/internal/board/detail.go +0 -218
  323. package/internal/board/detail_test.go +0 -388
  324. package/internal/board/epic_panel.go +0 -264
  325. package/internal/board/epic_panel_test.go +0 -869
  326. package/internal/board/help.go +0 -41
  327. package/internal/board/help_test.go +0 -86
  328. package/internal/board/integration_test.go +0 -266
  329. package/internal/board/interfaces.go +0 -65
  330. package/internal/board/interfaces_test.go +0 -114
  331. package/internal/board/io.go +0 -93
  332. package/internal/board/layout.go +0 -68
  333. package/internal/board/layout_test.go +0 -106
  334. package/internal/board/model.go +0 -235
  335. package/internal/board/model_test.go +0 -67
  336. package/internal/board/plain.go +0 -88
  337. package/internal/board/plain_test.go +0 -117
  338. package/internal/board/release.go +0 -34
  339. package/internal/board/release_test.go +0 -177
  340. package/internal/board/render_policy_test.go +0 -77
  341. package/internal/board/status.go +0 -23
  342. package/internal/board/theme.go +0 -24
  343. package/internal/board/theme_test.go +0 -31
  344. package/internal/board/transitions.go +0 -113
  345. package/internal/board/transitions_test.go +0 -164
  346. package/internal/board/tui.go +0 -32
  347. package/internal/board/update.go +0 -575
  348. package/internal/board/update_test.go +0 -602
  349. package/internal/board/util.go +0 -76
  350. package/internal/board/view.go +0 -317
  351. package/internal/board/view_test.go +0 -377
  352. package/internal/board/watch.go +0 -136
  353. package/internal/buildtool/main.go +0 -249
  354. package/internal/buildtool/main_test.go +0 -224
  355. package/internal/data/config.go +0 -101
  356. package/internal/data/config_test.go +0 -122
  357. package/internal/data/discover.go +0 -178
  358. package/internal/data/discover_test.go +0 -130
  359. package/internal/data/errors.go +0 -13
  360. package/internal/data/fuzz_test.go +0 -75
  361. package/internal/data/lifecycle.go +0 -44
  362. package/internal/data/lifecycle_test.go +0 -41
  363. package/internal/data/parser.go +0 -243
  364. package/internal/data/parser_test.go +0 -281
  365. package/internal/data/router.go +0 -52
  366. package/internal/data/router_test.go +0 -35
  367. package/internal/data/task.go +0 -57
  368. package/internal/data/task_test.go +0 -51
  369. package/internal/data/testdata/fuzz/FuzzSplitFrontmatterBody/68eb66b0fe91e7e3 +0 -2
  370. package/internal/data/write.go +0 -221
  371. package/internal/data/write_test.go +0 -623
  372. package/internal/doctor/checks.go +0 -567
  373. package/internal/doctor/checks_test.go +0 -716
  374. package/internal/doctor/gates.go +0 -193
  375. package/internal/doctor/gates_test.go +0 -166
  376. package/internal/doctor/interfaces.go +0 -64
  377. package/internal/doctor/interfaces_test.go +0 -104
  378. package/internal/doctor/repairs.go +0 -80
  379. package/internal/doctor/repairs_test.go +0 -81
  380. package/internal/doctor/report.go +0 -157
  381. package/internal/doctor/report_test.go +0 -89
  382. package/internal/init/clipboard.go +0 -146
  383. package/internal/init/clipboard_test.go +0 -74
  384. package/internal/init/install.go +0 -16
  385. package/internal/init/integration_test.go +0 -197
  386. package/internal/init/prompt.go +0 -14
  387. package/internal/init/prompt_test.go +0 -77
  388. package/internal/init/scaffold.go +0 -59
  389. package/internal/init/scaffold_test.go +0 -179
  390. package/internal/init/template_freshness_test.go +0 -56
  391. package/internal/init/validate.go +0 -85
  392. package/internal/init/validate_test.go +0 -141
  393. package/internal/init/write.go +0 -73
  394. package/internal/init/write_test.go +0 -91
  395. package/internal/styles/palette.go +0 -49
  396. package/internal/styles/styles.go +0 -139
  397. package/internal/styles/styles_test.go +0 -133
  398. package/internal/testutil/fixture.go +0 -113
  399. package/internal/testutil/fs.go +0 -26
  400. package/main.go +0 -136
  401. package/project-audit/audit_report_glm_5.1.md +0 -411
  402. package/project-audit/audit_report_opus_4.6.md +0 -406
  403. package/project-audit/consolidated-audit-report.md +0 -456
  404. package/scripts/vitest-preload.cjs +0 -95
  405. package/templates/project/.savepoint/Design.md +0 -47
  406. package/templates/project/.savepoint/PRD.md +0 -34
  407. package/templates/project/.savepoint/config.yml +0 -27
  408. package/templates/project/.savepoint/router.md +0 -153
  409. package/templates/project/.savepoint/visual-identity.md +0 -122
  410. package/templates/project/AGENTS.md +0 -88
  411. package/templates/project/agent-skills/savepoint-audit/SKILL.md +0 -87
  412. package/templates/project/agent-skills/savepoint-build-task/SKILL.md +0 -44
  413. package/templates/project/agent-skills/savepoint-create-plan/SKILL.md +0 -33
  414. package/templates/project/agent-skills/savepoint-create-task/SKILL.md +0 -44
  415. package/templates/project/agent-skills/savepoint-draft-prd/SKILL.md +0 -37
  416. package/templates/project/agent-skills/savepoint-system-design/SKILL.md +0 -38
  417. package/templates/prompts/audit-reconciliation.prompt.md +0 -72
  418. package/templates/prompts/design.prompt.md +0 -45
  419. package/templates/prompts/epic-design.prompt.md +0 -43
  420. package/templates/prompts/magic-prompt.prompt.md +0 -7
  421. package/templates/prompts/prd.prompt.md +0 -42
  422. package/templates/prompts/task-breakdown.prompt.md +0 -54
  423. package/templates/prompts/task-building.prompt.md +0 -38
  424. package/templates/prompts/task-planning.prompt.md +0 -53
  425. package/templates/release/v1/PRD.md +0 -37
@@ -1,716 +0,0 @@
1
- package doctor
2
-
3
- import (
4
- "fmt"
5
- "path/filepath"
6
- "strings"
7
- "testing"
8
-
9
- "github.com/opencode/savepoint/internal/testutil"
10
- )
11
-
12
- // --- CheckConfig ---
13
-
14
- func TestCheckConfigMissing(t *testing.T) {
15
- root := t.TempDir()
16
- err := CheckConfig(root)
17
- if err == nil || !strings.Contains(err.Error(), "not found") {
18
- t.Fatalf("CheckConfig() = %v, want not found error", err)
19
- }
20
- }
21
-
22
- func TestCheckConfigInvalidYAML(t *testing.T) {
23
- root := t.TempDir()
24
- testutil.WriteFile(t, filepath.Join(root, "config.yml"), "theme: [broken")
25
- err := CheckConfig(root)
26
- if err == nil || !strings.Contains(err.Error(), "invalid YAML") {
27
- t.Fatalf("CheckConfig() = %v, want invalid YAML error", err)
28
- }
29
- }
30
-
31
- func TestCheckConfigMissingQualityGates(t *testing.T) {
32
- root := t.TempDir()
33
- testutil.WriteFile(t, filepath.Join(root, "config.yml"), "theme:\n bg: \"#000\"\n")
34
- err := CheckConfig(root)
35
- if err == nil || !strings.Contains(err.Error(), "quality_gates") {
36
- t.Fatalf("CheckConfig() = %v, want quality_gates error", err)
37
- }
38
- }
39
-
40
- func TestCheckConfigMissingTheme(t *testing.T) {
41
- root := t.TempDir()
42
- testutil.WriteFile(t, filepath.Join(root, "config.yml"), "quality_gates:\n block_on_failure: true\n")
43
- err := CheckConfig(root)
44
- if err == nil || !strings.Contains(err.Error(), "theme") {
45
- t.Fatalf("CheckConfig() = %v, want theme error", err)
46
- }
47
- }
48
-
49
- func TestCheckConfigValid(t *testing.T) {
50
- root := t.TempDir()
51
- testutil.WriteFile(t, filepath.Join(root, "config.yml"), "quality_gates:\n block_on_failure: true\ntheme:\n bg: \"#000\"\n")
52
- if err := CheckConfig(root); err != nil {
53
- t.Fatalf("CheckConfig() = %v, want nil", err)
54
- }
55
- }
56
-
57
- // --- CheckRouter ---
58
-
59
- func TestCheckRouterMissing(t *testing.T) {
60
- root := t.TempDir()
61
- err := CheckRouter(root, "")
62
- if err == nil || !strings.Contains(err.Error(), "not found") {
63
- t.Fatalf("CheckRouter() = %v, want not found error", err)
64
- }
65
- }
66
-
67
- func TestCheckRouterInvalidStateBlock(t *testing.T) {
68
- root := t.TempDir()
69
- testutil.WriteFile(t, filepath.Join(root, "router.md"), "# no state block")
70
- err := CheckRouter(root, "")
71
- if err == nil || !strings.Contains(err.Error(), "invalid state block") {
72
- t.Fatalf("CheckRouter() = %v, want invalid state block error", err)
73
- }
74
- }
75
-
76
- func TestCheckRouterPreImplementation(t *testing.T) {
77
- root := t.TempDir()
78
- testutil.WriteFile(t, filepath.Join(root, "router.md"), routerContent("pre-implementation", "none", "none"))
79
- if err := CheckRouter(root, ""); err != nil {
80
- t.Fatalf("CheckRouter() = %v, want nil", err)
81
- }
82
- }
83
-
84
- func TestCheckRouterMissingReleaseDir(t *testing.T) {
85
- root := t.TempDir()
86
- testutil.WriteFile(t, filepath.Join(root, "router.md"), routerContent("task-building", "v1", "none"))
87
- err := CheckRouter(root, "")
88
- if err == nil || !strings.Contains(err.Error(), "release") {
89
- t.Fatalf("CheckRouter() = %v, want release directory error", err)
90
- }
91
- }
92
-
93
- func TestCheckRouterMissingEpicDir(t *testing.T) {
94
- root := t.TempDir()
95
- testutil.MkdirAll(t, filepath.Join(root, "releases", "v1"))
96
- testutil.WriteFile(t, filepath.Join(root, "router.md"), routerContent("task-building", "v1", "E03-foo"))
97
- err := CheckRouter(root, "")
98
- if err == nil || !strings.Contains(err.Error(), "epic") {
99
- t.Fatalf("CheckRouter() = %v, want epic directory error", err)
100
- }
101
- }
102
-
103
- func TestCheckRouterValidWithDirs(t *testing.T) {
104
- root := t.TempDir()
105
- testutil.MkdirAll(t, filepath.Join(root, "releases", "v1", "epics", "E03-foo"))
106
- testutil.WriteFile(t, filepath.Join(root, "router.md"), routerContent("task-building", "v1", "E03-foo"))
107
- if err := CheckRouter(root, ""); err != nil {
108
- t.Fatalf("CheckRouter() = %v, want nil", err)
109
- }
110
- }
111
-
112
- func TestCheckRouterEpicFilterSkip(t *testing.T) {
113
- root := t.TempDir()
114
- // release dir missing — would fail without filter
115
- testutil.WriteFile(t, filepath.Join(root, "router.md"), routerContent("task-building", "v1", "E03-foo"))
116
- // filter doesn't match router epic → skip dir checks
117
- if err := CheckRouter(root, "E99-other"); err != nil {
118
- t.Fatalf("CheckRouter() = %v, want nil (filter skip)", err)
119
- }
120
- }
121
-
122
- // --- CheckStructure ---
123
-
124
- func TestCheckStructure_MissingReleasesDir(t *testing.T) {
125
- root := t.TempDir()
126
- problems := CheckStructure(root, "")
127
- if len(problems) != 1 || !strings.Contains(problems[0].Message, "releases directory not found") {
128
- t.Fatalf("CheckStructure() = %v, want releases directory error", problems)
129
- }
130
- }
131
-
132
- func TestCheckStructure_EmptyReleases(t *testing.T) {
133
- root := t.TempDir()
134
- testutil.MkdirAll(t, filepath.Join(root, "releases"))
135
- problems := CheckStructure(root, "")
136
- if len(problems) != 1 || !strings.Contains(problems[0].Message, "no release directories found") {
137
- t.Fatalf("CheckStructure() = %v, want no releases error", problems)
138
- }
139
- }
140
-
141
- func TestCheckStructure_MissingReleasePRD(t *testing.T) {
142
- root := t.TempDir()
143
- testutil.MkdirAll(t, filepath.Join(root, "releases", "v1", "epics"))
144
- problems := CheckStructure(root, "")
145
- found := false
146
- for _, p := range problems {
147
- if strings.Contains(p.Message, "release PRD file not found") {
148
- found = true
149
- break
150
- }
151
- }
152
- if !found {
153
- t.Fatalf("CheckStructure() = %v, want release PRD file not found problem", problems)
154
- }
155
- }
156
-
157
- func TestCheckStructure_ReleasePRDValid(t *testing.T) {
158
- root := t.TempDir()
159
- testutil.MkdirAll(t, filepath.Join(root, "releases", "v1", "epics"))
160
- testutil.WriteReleasePRD(t, filepath.Join(root, "releases", "v1"))
161
- problems := CheckStructure(root, "")
162
- for _, p := range problems {
163
- if strings.Contains(p.File, "v1-PRD.md") {
164
- t.Fatalf("CheckStructure() unexpected PRD problem: %v", p)
165
- }
166
- }
167
- }
168
-
169
- func TestCheckStructure_ReleasePRDCorruptYAML(t *testing.T) {
170
- root := t.TempDir()
171
- testutil.MkdirAll(t, filepath.Join(root, "releases", "v1", "epics"))
172
- testutil.WriteFile(t, filepath.Join(root, "releases", "v1", "v1-PRD.md"), "---\ntype: [broken\n---\n")
173
- problems := CheckStructure(root, "")
174
- found := false
175
- for _, p := range problems {
176
- if strings.Contains(p.File, "v1-PRD.md") && p.Line > 0 {
177
- found = true
178
- break
179
- }
180
- }
181
- if !found {
182
- t.Fatalf("CheckStructure() = %v, want corrupt YAML with line in v1-PRD.md", problems)
183
- }
184
- }
185
-
186
- func TestCheckStructure_ValidEpicDetail(t *testing.T) {
187
- root := t.TempDir()
188
- releasePath := filepath.Join(root, "releases", "v1")
189
- epicPath := filepath.Join(releasePath, "epics", "E01-foo")
190
- testutil.MkdirAll(t, epicPath)
191
- testutil.WriteReleasePRD(t, releasePath)
192
- testutil.WriteFile(t, filepath.Join(epicPath, "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
193
- problems := CheckStructure(root, "")
194
- for _, p := range problems {
195
- if strings.Contains(p.File, "Detail.md") {
196
- t.Fatalf("CheckStructure() unexpected Detail.md problem: %v", p)
197
- }
198
- }
199
- }
200
-
201
- func TestCheckStructure_MissingEpicDetail(t *testing.T) {
202
- root := t.TempDir()
203
- releasePath := filepath.Join(root, "releases", "v1")
204
- epicPath := filepath.Join(releasePath, "epics", "E01-foo")
205
- testutil.MkdirAll(t, epicPath)
206
- testutil.WriteReleasePRD(t, releasePath)
207
- problems := CheckStructure(root, "")
208
- found := false
209
- for _, p := range problems {
210
- if strings.Contains(p.Message, "epic detail file not found") {
211
- found = true
212
- break
213
- }
214
- }
215
- if !found {
216
- t.Fatalf("CheckStructure() = %v, want epic detail file not found problem", problems)
217
- }
218
- }
219
-
220
- func TestCheckStructure_ValidTask(t *testing.T) {
221
- root := t.TempDir()
222
- releasePath := filepath.Join(root, "releases", "v1")
223
- epicPath := filepath.Join(releasePath, "epics", "E01-foo")
224
- tasksPath := filepath.Join(epicPath, "tasks")
225
- testutil.MkdirAll(t, tasksPath)
226
- testutil.WriteReleasePRD(t, releasePath)
227
- testutil.WriteFile(t, filepath.Join(epicPath, "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
228
- testutil.WriteFile(t, filepath.Join(tasksPath, "T001-task.md"), "---\nid: E01-foo/T001-task\nstatus: planned\nobjective: \"Do the thing\"\ndepends_on: []\n---\n\n# T001: Task\n\n## Acceptance Criteria\n\n- It works\n")
229
- problems := CheckStructure(root, "")
230
- if len(problems) > 0 {
231
- t.Fatalf("CheckStructure() = %v, want no problems", problems)
232
- }
233
- }
234
-
235
- func TestCheckStructure_TaskMissingRequiredField(t *testing.T) {
236
- root := t.TempDir()
237
- releasePath := filepath.Join(root, "releases", "v1")
238
- epicPath := filepath.Join(releasePath, "epics", "E01-foo")
239
- tasksPath := filepath.Join(epicPath, "tasks")
240
- testutil.MkdirAll(t, tasksPath)
241
- testutil.WriteReleasePRD(t, releasePath)
242
- testutil.WriteFile(t, filepath.Join(epicPath, "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
243
- testutil.WriteFile(t, filepath.Join(tasksPath, "T001-task.md"), "---\nid: E01-foo/T001-task\nobjective: \"Do the thing\"\n---\n\n# T001: Task\n\n## Acceptance Criteria\n\n- It works\n")
244
- problems := CheckStructure(root, "")
245
- found := false
246
- for _, p := range problems {
247
- if strings.Contains(p.Message, "status") && strings.Contains(p.Message, "missing") {
248
- found = true
249
- break
250
- }
251
- }
252
- if !found {
253
- t.Fatalf("CheckStructure() = %v, want missing status field problem", problems)
254
- }
255
- }
256
-
257
- func TestCheckStructure_TaskMissingAcceptanceCriteria(t *testing.T) {
258
- root := t.TempDir()
259
- releasePath := filepath.Join(root, "releases", "v1")
260
- epicPath := filepath.Join(releasePath, "epics", "E01-foo")
261
- tasksPath := filepath.Join(epicPath, "tasks")
262
- testutil.MkdirAll(t, tasksPath)
263
- testutil.WriteReleasePRD(t, releasePath)
264
- testutil.WriteFile(t, filepath.Join(epicPath, "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
265
- testutil.WriteFile(t, filepath.Join(tasksPath, "T001-task.md"), "---\nid: E01-foo/T001-task\nstatus: planned\nobjective: \"Do the thing\"\n---\n\n# T001: Task\n")
266
- problems := CheckStructure(root, "")
267
- found := false
268
- for _, p := range problems {
269
- if strings.Contains(p.Message, "Acceptance Criteria") {
270
- found = true
271
- break
272
- }
273
- }
274
- if !found {
275
- t.Fatalf("CheckStructure() = %v, want missing acceptance criteria problem", problems)
276
- }
277
- }
278
-
279
- func TestCheckStructure_TaskCorruptYAML(t *testing.T) {
280
- root := t.TempDir()
281
- releasePath := filepath.Join(root, "releases", "v1")
282
- epicPath := filepath.Join(releasePath, "epics", "E01-foo")
283
- tasksPath := filepath.Join(epicPath, "tasks")
284
- testutil.MkdirAll(t, tasksPath)
285
- testutil.WriteReleasePRD(t, releasePath)
286
- testutil.WriteFile(t, filepath.Join(epicPath, "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
287
- testutil.WriteFile(t, filepath.Join(tasksPath, "T001-task.md"), "---\nid: \"unclosed\nstatus: planned\n---\n")
288
- problems := CheckStructure(root, "")
289
- foundLine := false
290
- for _, p := range problems {
291
- if strings.Contains(p.File, "T001-task.md") && p.Line > 0 {
292
- foundLine = true
293
- break
294
- }
295
- }
296
- if !foundLine {
297
- t.Fatalf("CheckStructure() = %v, want corrupt YAML with line number in task", problems)
298
- }
299
- }
300
-
301
- func TestCheckStructure_EpicFilter(t *testing.T) {
302
- root := t.TempDir()
303
- releasePath := filepath.Join(root, "releases", "v1")
304
- epic1Path := filepath.Join(releasePath, "epics", "E01-foo")
305
- epic2Path := filepath.Join(releasePath, "epics", "E02-bar")
306
- testutil.MkdirAll(t, epic1Path)
307
- testutil.MkdirAll(t, epic2Path)
308
- testutil.WriteReleasePRD(t, releasePath)
309
- testutil.WriteFile(t, filepath.Join(epic1Path, "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
310
- // E02 has no detail file — should not appear when filtering to E01
311
- problems := CheckStructure(root, "E01-foo")
312
- for _, p := range problems {
313
- if strings.Contains(p.Message, "E02") {
314
- t.Fatalf("CheckStructure() with epicFilter=E01-foo should skip E02, got: %v", p)
315
- }
316
- }
317
- }
318
-
319
- func TestCheckStructure_EpicFilterByPrefix(t *testing.T) {
320
- root := t.TempDir()
321
- releasePath := filepath.Join(root, "releases", "v1")
322
- epicPath := filepath.Join(releasePath, "epics", "E01-foo")
323
- testutil.MkdirAll(t, epicPath)
324
- testutil.WriteReleasePRD(t, releasePath)
325
- testutil.WriteFile(t, filepath.Join(epicPath, "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
326
- testutil.MkdirAll(t, filepath.Join(epicPath, "tasks"))
327
- problems := CheckStructure(root, "E01")
328
- if len(problems) > 0 {
329
- t.Fatalf("CheckStructure() with epicFilter=E01 prefix = %v, want no problems", problems)
330
- }
331
- }
332
-
333
- // --- CheckDependencies ---
334
-
335
- func TestCheckDependencies_NoReleases(t *testing.T) {
336
- root := t.TempDir()
337
- problems := CheckDependencies(root, "")
338
- if len(problems) == 0 {
339
- t.Fatal("CheckDependencies() = no problems, want error about releases")
340
- }
341
- }
342
-
343
- func TestCheckDependencies_NoDeps(t *testing.T) {
344
- root := t.TempDir()
345
- setupMinimalProject(t, root, "v1", "E01-foo", nil)
346
- problems := CheckDependencies(root, "")
347
- if len(problems) > 0 {
348
- t.Fatalf("CheckDependencies() = %v, want no problems", problems)
349
- }
350
- }
351
-
352
- func TestCheckDependencies_ValidDeps(t *testing.T) {
353
- root := t.TempDir()
354
- setupMinimalProject(t, root, "v1", "E01-foo", []taskSpec{
355
- {id: "E01-foo/T001-task", deps: []string{}},
356
- {id: "E01-foo/T002-task", deps: []string{"E01-foo/T001-task"}},
357
- {id: "E01-foo/T003-task", deps: []string{"E01-foo/T002-task"}},
358
- })
359
- problems := CheckDependencies(root, "")
360
- if len(problems) > 0 {
361
- t.Fatalf("CheckDependencies() = %v, want no problems", problems)
362
- }
363
- }
364
-
365
- func TestCheckDependencies_MissingDep(t *testing.T) {
366
- root := t.TempDir()
367
- setupMinimalProject(t, root, "v1", "E01-foo", []taskSpec{
368
- {id: "E01-foo/T001-task", deps: []string{}},
369
- {id: "E01-foo/T002-task", deps: []string{"E01-foo/T999-nonexistent"}},
370
- })
371
- problems := CheckDependencies(root, "")
372
- found := false
373
- for _, p := range problems {
374
- if strings.Contains(p.Message, "non-existent") {
375
- found = true
376
- break
377
- }
378
- }
379
- if !found {
380
- t.Fatalf("CheckDependencies() = %v, want missing dependency problem", problems)
381
- }
382
- }
383
-
384
- func TestCheckDependencies_DuplicateIDs(t *testing.T) {
385
- root := t.TempDir()
386
- releasePath := filepath.Join(root, "releases", "v1")
387
- epicPath := filepath.Join(releasePath, "epics", "E01-foo")
388
- tasksPath := filepath.Join(epicPath, "tasks")
389
- testutil.MkdirAll(t, tasksPath)
390
- testutil.WriteReleasePRD(t, releasePath)
391
- testutil.WriteFile(t, filepath.Join(epicPath, "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
392
-
393
- // Two epics, same task ID
394
- epic2Path := filepath.Join(releasePath, "epics", "E02-bar")
395
- tasks2Path := filepath.Join(epic2Path, "tasks")
396
- testutil.MkdirAll(t, tasks2Path)
397
- testutil.WriteFile(t, filepath.Join(epic2Path, "E02-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E02: Bar\n")
398
-
399
- testutil.WriteFile(t, filepath.Join(tasksPath, "T001-task.md"), "---\nid: E01-foo/T001-task\nstatus: planned\nobjective: \"A\"\ndepends_on: []\n---\n\n# T001\n\n## Acceptance Criteria\n\n- it works\n")
400
- testutil.WriteFile(t, filepath.Join(tasks2Path, "T001-task.md"), "---\nid: E01-foo/T001-task\nstatus: planned\nobjective: \"A\"\ndepends_on: []\n---\n\n# T001\n\n## Acceptance Criteria\n\n- it works\n")
401
-
402
- problems := CheckDependencies(root, "")
403
- found := false
404
- for _, p := range problems {
405
- if strings.Contains(p.Message, "duplicate task ID") {
406
- found = true
407
- break
408
- }
409
- }
410
- if !found {
411
- t.Fatalf("CheckDependencies() = %v, want duplicate task ID problem", problems)
412
- }
413
- }
414
-
415
- func TestCheckDependencies_Cycle(t *testing.T) {
416
- root := t.TempDir()
417
- setupMinimalProject(t, root, "v1", "E01-foo", []taskSpec{
418
- {id: "E01-foo/T001-task", deps: []string{"E01-foo/T003-task"}},
419
- {id: "E01-foo/T002-task", deps: []string{"E01-foo/T001-task"}},
420
- {id: "E01-foo/T003-task", deps: []string{"E01-foo/T002-task"}},
421
- })
422
- problems := CheckDependencies(root, "")
423
- found := false
424
- for _, p := range problems {
425
- if strings.Contains(p.Message, "cycle") {
426
- found = true
427
- break
428
- }
429
- }
430
- if !found {
431
- t.Fatalf("CheckDependencies() = %v, want cycle problem", problems)
432
- }
433
- }
434
-
435
- func TestCheckDependencies_CycleAccuratePath(t *testing.T) {
436
- root := t.TempDir()
437
- setupMinimalProject(t, root, "v1", "E01-foo", []taskSpec{
438
- {id: "E01-foo/T001-task", deps: []string{"E01-foo/T002-task"}},
439
- {id: "E01-foo/T002-task", deps: []string{"E01-foo/T003-task"}},
440
- {id: "E01-foo/T003-task", deps: []string{"E01-foo/T001-task"}},
441
- })
442
- problems := CheckDependencies(root, "")
443
- var cycleMsg string
444
- for _, p := range problems {
445
- if strings.Contains(p.Message, "cycle") {
446
- cycleMsg = p.Message
447
- break
448
- }
449
- }
450
- if cycleMsg == "" {
451
- t.Fatal("CheckDependencies() = no cycle problem, want one")
452
- }
453
- // The cycle path should contain T001, T002, T003 in the correct order
454
- if !strings.Contains(cycleMsg, "T001") || !strings.Contains(cycleMsg, "T002") || !strings.Contains(cycleMsg, "T003") {
455
- t.Fatalf("CheckDependencies() cycle path = %q, should contain all three tasks", cycleMsg)
456
- }
457
- // Each arrow should separate consecutive nodes in the cycle
458
- if !strings.Contains(cycleMsg, "T001-task") || !strings.Contains(cycleMsg, "T002-task") || !strings.Contains(cycleMsg, "T003-task") {
459
- t.Fatalf("CheckDependencies() cycle path = %q, should reference task files", cycleMsg)
460
- }
461
- }
462
-
463
- func TestCheckDependencies_SelfReference(t *testing.T) {
464
- root := t.TempDir()
465
- setupMinimalProject(t, root, "v1", "E01-foo", []taskSpec{
466
- {id: "E01-foo/T001-task", deps: []string{"E01-foo/T001-task"}},
467
- })
468
- problems := CheckDependencies(root, "")
469
- found := false
470
- for _, p := range problems {
471
- if strings.Contains(p.Message, "cycle") {
472
- found = true
473
- break
474
- }
475
- }
476
- if !found {
477
- t.Fatalf("CheckDependencies() = %v, want cycle problem (self-reference)", problems)
478
- }
479
- }
480
-
481
- func TestCheckDependencies_EpicFilter(t *testing.T) {
482
- root := t.TempDir()
483
- releasePath := filepath.Join(root, "releases", "v1")
484
- testutil.MkdirAll(t, filepath.Join(releasePath, "epics", "E01-foo", "tasks"))
485
- testutil.MkdirAll(t, filepath.Join(releasePath, "epics", "E02-bar", "tasks"))
486
- testutil.WriteReleasePRD(t, releasePath)
487
- testutil.WriteFile(t, filepath.Join(releasePath, "epics", "E01-foo", "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
488
- testutil.WriteFile(t, filepath.Join(releasePath, "epics", "E02-bar", "E02-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E02: Bar\n")
489
-
490
- // E02 has a missing dep — should be invisible with filter
491
- taskE2 := `---\nid: E02-bar/T001-task\nstatus: planned\nobjective: \"B\"\ndepends_on: [\"E02-bar/T999-nonexistent\"]\n---\n\n# T001\n\n## Acceptance Criteria\n\n- it works\n`
492
- testutil.WriteFile(t, filepath.Join(releasePath, "epics", "E01-foo", "tasks", "T001-task.md"), "---\nid: E01-foo/T001-task\nstatus: planned\nobjective: \"A\"\ndepends_on: []\n---\n\n# T001\n\n## Acceptance Criteria\n\n- it works\n")
493
- testutil.WriteFile(t, filepath.Join(releasePath, "epics", "E02-bar", "tasks", "T001-task.md"), strings.ReplaceAll(taskE2, "\\n", "\n"))
494
-
495
- problems := CheckDependencies(root, "E01-foo")
496
- for _, p := range problems {
497
- if strings.Contains(p.Message, "E02-bar") {
498
- t.Fatalf("CheckDependencies() with epicFilter=E01-foo should skip E02, got: %v", p)
499
- }
500
- }
501
- }
502
-
503
- // --- CheckAuditState ---
504
-
505
- func TestCheckAuditState_NoAuditFiles(t *testing.T) {
506
- root := t.TempDir()
507
- testutil.MkdirAll(t, filepath.Join(root, "releases", "v1", "epics", "E01-foo"))
508
- testutil.WriteFile(t, filepath.Join(root, "router.md"), routerContent("task-building", "v1", "E01-foo"))
509
- problems := CheckAuditState(root)
510
- if len(problems) > 0 {
511
- t.Fatalf("CheckAuditState() = %v, want no problems", problems)
512
- }
513
- }
514
-
515
- func TestCheckAuditState_MatchesRouter(t *testing.T) {
516
- root := t.TempDir()
517
- epicPath := filepath.Join(root, "releases", "v1", "epics", "E01-foo")
518
- testutil.MkdirAll(t, epicPath)
519
- testutil.WriteFile(t, filepath.Join(epicPath, "E01-Audit.md"), "---\ntype: audit-findings\n---\n\n# Audit\n")
520
- testutil.WriteFile(t, filepath.Join(root, "router.md"), routerContent("audit-pending", "v1", "E01-foo"))
521
- problems := CheckAuditState(root)
522
- if len(problems) > 0 {
523
- t.Fatalf("CheckAuditState() = %v, want no problems when router matches", problems)
524
- }
525
- }
526
-
527
- func TestCheckAuditState_ProposalWithoutPending(t *testing.T) {
528
- root := t.TempDir()
529
- epicPath := filepath.Join(root, "releases", "v1", "epics", "E01-foo")
530
- testutil.MkdirAll(t, epicPath)
531
- testutil.WriteFile(t, filepath.Join(epicPath, "E01-Audit.md"), "---\ntype: audit-findings\n---\n\n# Audit\n")
532
- testutil.WriteFile(t, filepath.Join(root, "router.md"), routerContent("task-building", "v1", "E01-foo"))
533
- problems := CheckAuditState(root)
534
- if len(problems) != 1 {
535
- t.Fatalf("CheckAuditState() = %v, want 1 problem (audit file without audit-pending)", problems)
536
- }
537
- if !strings.Contains(problems[0].Message, "audit proposal exists") {
538
- t.Fatalf("CheckAuditState() = %v, want 'audit proposal exists' message", problems)
539
- }
540
- }
541
-
542
- func TestCheckAuditState_DifferentEpicInRouter(t *testing.T) {
543
- root := t.TempDir()
544
- epic1Path := filepath.Join(root, "releases", "v1", "epics", "E01-foo")
545
- epic2Path := filepath.Join(root, "releases", "v1", "epics", "E02-bar")
546
- testutil.MkdirAll(t, epic1Path)
547
- testutil.MkdirAll(t, epic2Path)
548
- testutil.WriteFile(t, filepath.Join(epic1Path, "E01-Audit.md"), "---\ntype: audit-findings\n---\n\n# Audit\n")
549
- testutil.WriteFile(t, filepath.Join(root, "router.md"), routerContent("audit-pending", "v1", "E02-bar"))
550
- problems := CheckAuditState(root)
551
- if len(problems) != 1 {
552
- t.Fatalf("CheckAuditState() = %v, want 1 problem (E01 audit but E02 in router)", problems)
553
- }
554
- if !strings.Contains(problems[0].Message, "E01") {
555
- t.Fatalf("CheckAuditState() = %v, want problem mentioning E01", problems)
556
- }
557
- }
558
-
559
- func TestCheckAuditState_MultipleStale(t *testing.T) {
560
- root := t.TempDir()
561
- epic1Path := filepath.Join(root, "releases", "v1", "epics", "E01-foo")
562
- epic2Path := filepath.Join(root, "releases", "v1", "epics", "E02-bar")
563
- testutil.MkdirAll(t, epic1Path)
564
- testutil.MkdirAll(t, epic2Path)
565
- testutil.WriteFile(t, filepath.Join(epic1Path, "E01-Audit.md"), "---\ntype: audit-findings\n---\n\n# Audit\n")
566
- testutil.WriteFile(t, filepath.Join(epic2Path, "E02-Audit.md"), "---\ntype: audit-findings\n---\n\n# Audit\n")
567
- testutil.WriteFile(t, filepath.Join(root, "router.md"), routerContent("task-building", "v1", "E03-baz"))
568
- problems := CheckAuditState(root)
569
- if len(problems) != 2 {
570
- t.Fatalf("CheckAuditState() = %v, want 2 problems (both audit files stale)", problems)
571
- }
572
- }
573
-
574
- // --- CheckOrphans ---
575
-
576
- func TestCheckOrphans_NoOrphans(t *testing.T) {
577
- root := t.TempDir()
578
- setupMinimalProject(t, root, "v1", "E01-foo", []taskSpec{
579
- {id: "E01-foo/T001-task", deps: []string{}},
580
- {id: "E01-foo/T002-task", deps: []string{}},
581
- })
582
- problems := CheckOrphans(root)
583
- if len(problems) > 0 {
584
- t.Fatalf("CheckOrphans() = %v, want no problems", problems)
585
- }
586
- }
587
-
588
- func TestCheckOrphans_TaskRefersNonexistentEpic(t *testing.T) {
589
- root := t.TempDir()
590
- setupMinimalProject(t, root, "v1", "E01-foo", []taskSpec{
591
- {id: "E99-ghost/T001-task", deps: []string{}},
592
- })
593
- problems := CheckOrphans(root)
594
- found := false
595
- for _, p := range problems {
596
- if strings.Contains(p.Message, "orphaned") && strings.Contains(p.Message, "E99-ghost") {
597
- found = true
598
- break
599
- }
600
- }
601
- if !found {
602
- t.Fatalf("CheckOrphans() = %v, want orphaned task problem for E99-ghost", problems)
603
- }
604
- }
605
-
606
- func TestCheckOrphans_CrossReleaseEpicRef(t *testing.T) {
607
- root := t.TempDir()
608
- releasePath := filepath.Join(root, "releases", "v1")
609
- epicPath := filepath.Join(releasePath, "epics", "E01-foo")
610
- tasksPath := filepath.Join(epicPath, "tasks")
611
- testutil.MkdirAll(t, tasksPath)
612
- testutil.WriteReleasePRD(t, releasePath)
613
- testutil.WriteFile(t, filepath.Join(epicPath, "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
614
- testutil.WriteFile(t, filepath.Join(tasksPath, "T001-task.md"), "---\nid: E02-bar/T001-task\nstatus: planned\nobjective: \"Task\"\ndepends_on: []\n---\n\n# T001\n\n## Acceptance Criteria\n\n- it works\n")
615
- // E02-bar does not exist in any release
616
- problems := CheckOrphans(root)
617
- found := false
618
- for _, p := range problems {
619
- if strings.Contains(p.Message, "orphaned") {
620
- found = true
621
- break
622
- }
623
- }
624
- if !found {
625
- t.Fatalf("CheckOrphans() = %v, want orphaned task problem for E02-bar", problems)
626
- }
627
- }
628
-
629
- func TestCheckOrphans_ValidCrossReleaseRef(t *testing.T) {
630
- root := t.TempDir()
631
- releasePath := filepath.Join(root, "releases", "v1")
632
- epic1Path := filepath.Join(releasePath, "epics", "E01-foo")
633
- epic2Path := filepath.Join(releasePath, "epics", "E02-bar")
634
- testutil.MkdirAll(t, filepath.Join(epic1Path, "tasks"))
635
- testutil.MkdirAll(t, filepath.Join(epic2Path, "tasks"))
636
- testutil.WriteReleasePRD(t, releasePath)
637
- testutil.WriteFile(t, filepath.Join(epic1Path, "E01-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E01: Foo\n")
638
- testutil.WriteFile(t, filepath.Join(epic2Path, "E02-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# E02: Bar\n")
639
- testutil.WriteFile(t, filepath.Join(epic1Path, "tasks", "T001-task.md"), "---\nid: E02-bar/T001-task\nstatus: planned\nobjective: \"Task\"\ndepends_on: []\n---\n\n# T001\n\n## Acceptance Criteria\n\n- it works\n")
640
- // E02-bar exists
641
- problems := CheckOrphans(root)
642
- for _, p := range problems {
643
- if strings.Contains(p.Message, "orphaned") {
644
- t.Fatalf("CheckOrphans() = %v, want no orphan problems for cross-epic ref that exists", problems)
645
- }
646
- }
647
- }
648
-
649
- func TestCheckOrphans_EmptyID(t *testing.T) {
650
- root := t.TempDir()
651
- setupMinimalProject(t, root, "v1", "E01-foo", []taskSpec{
652
- {id: "E01-foo/T001-task", deps: []string{}},
653
- })
654
- // Write a task with empty ID
655
- tasksPath := filepath.Join(root, "releases", "v1", "epics", "E01-foo", "tasks")
656
- testutil.WriteFile(t, filepath.Join(tasksPath, "T002-bad.md"), "---\nstatus: planned\nobjective: \"No ID\"\ndepends_on: []\n---\n\n# T002\n\n## Acceptance Criteria\n\n- it works\n")
657
- problems := CheckOrphans(root)
658
- // Should not crash, should handle missing ID gracefully
659
- if len(problems) > 0 {
660
- // Only allow non-orphan problems (e.g. missing ID)
661
- for _, p := range problems {
662
- if strings.Contains(p.Message, "orphaned") {
663
- t.Fatalf("CheckOrphans() = %v, want no orphan problems for task with missing ID", problems)
664
- }
665
- }
666
- }
667
- }
668
-
669
- func TestCheckOrphans_NoReleasesDir(t *testing.T) {
670
- root := t.TempDir()
671
- problems := CheckOrphans(root)
672
- // Should report releases dir problem, not crash
673
- if len(problems) == 0 {
674
- t.Fatal("CheckOrphans() = no problems, want error about missing releases")
675
- }
676
- }
677
-
678
- // helpers
679
-
680
- type taskSpec struct {
681
- id string
682
- deps []string
683
- }
684
-
685
- func setupMinimalProject(t *testing.T, root, releaseID, epicID string, tasks []taskSpec) {
686
- t.Helper()
687
- releasePath := filepath.Join(root, "releases", releaseID)
688
- epicPath := filepath.Join(releasePath, "epics", epicID)
689
- tasksPath := filepath.Join(epicPath, "tasks")
690
- testutil.MkdirAll(t, tasksPath)
691
-
692
- prefix := epicID
693
- if idx := strings.IndexByte(epicID, '-'); idx != -1 {
694
- prefix = epicID[:idx]
695
- }
696
-
697
- testutil.WriteReleasePRD(t, releasePath)
698
- testutil.WriteFile(t, filepath.Join(epicPath, prefix+"-Detail.md"), "---\ntype: epic-design\nstatus: planned\n---\n\n# Epic\n")
699
-
700
- for i, ts := range tasks {
701
- depsYAML := "[]"
702
- if len(ts.deps) > 0 {
703
- quoted := make([]string, len(ts.deps))
704
- for j, d := range ts.deps {
705
- quoted[j] = fmt.Sprintf("%q", d)
706
- }
707
- depsYAML = "[" + strings.Join(quoted, ", ") + "]"
708
- }
709
- content := fmt.Sprintf("---\nid: %s\nstatus: planned\nobjective: \"Task %d\"\ndepends_on: %s\n---\n\n# T%03d\n\n## Acceptance Criteria\n\n- it works\n", ts.id, i, depsYAML, i+1)
710
- testutil.WriteFile(t, filepath.Join(tasksPath, fmt.Sprintf("T%03d-task.md", i+1)), content)
711
- }
712
- }
713
-
714
- func routerContent(state, release, epic string) string {
715
- return "## Current state\n\n```yaml\nstate: " + state + "\nrelease: " + release + "\nepic: " + epic + "\ntask: none\nnext_action: \"\"\n```\n"
716
- }