savepoint 1.0.1 → 1.0.3

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 (278) hide show
  1. package/.claude/settings.local.json +15 -1
  2. package/.golangci.yml +11 -0
  3. package/.savepoint/Design.md +52 -46
  4. package/.savepoint/releases/v1/epics/E01-go-setup/tasks/T001-init-module.md +1 -1
  5. package/.savepoint/releases/v1/epics/E03-board-tui-core/tasks/T005-layout.md +1 -1
  6. package/.savepoint/releases/v1/epics/E04-board-components/tasks/T002-card.md +1 -1
  7. package/.savepoint/releases/v1/epics/E04-board-components/tasks/T006-help-overlay.md +1 -1
  8. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/{Design.md → E06-Detail.md} +5 -3
  9. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T002-header-and-dividers.md +1 -1
  10. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T003-footer-status-bar.md +1 -1
  11. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T004-component-refinement.md +1 -1
  12. package/.savepoint/releases/v1/epics/E06-atari-noir-layout/tasks/T010-auto-refresh-watcher.md +2 -0
  13. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/{Design.md → E01-Detail.md} +9 -1
  14. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/{T007-next-activity-header.md → T001-next-activity-header.md} +13 -12
  15. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T002-rename-epic-design-files.md +9 -9
  16. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T003-rename-release-prd.md +2 -2
  17. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T004-update-instruction-files.md +13 -12
  18. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T005-update-cross-references.md +14 -13
  19. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T006-column-and-detail-scrolling.md +25 -15
  20. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T007-column-focus-border-stability.md +57 -0
  21. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/E02-Audit.md +124 -0
  22. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/{Design.md → E02-Detail.md} +12 -3
  23. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/tasks/T001-fix-makefile.md +11 -8
  24. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/tasks/T002-linux-build-target.md +12 -7
  25. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/tasks/T003-macos-build-target.md +9 -5
  26. package/.savepoint/releases/v1.1/epics/E02-cross-platform-compatibility/tasks/T004-smoke-tests-and-artifacts.md +30 -9
  27. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/E03-Audit.md +195 -0
  28. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/E03-Detail.md +45 -0
  29. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/tasks/T001-border-resize-fix.md +40 -0
  30. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/tasks/T002-next-activity-below-header.md +64 -0
  31. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/tasks/T003-checkbox-rendering-fix.md +56 -0
  32. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/tasks/T005-unify-status-glyphs.md +65 -0
  33. package/.savepoint/releases/v1.1/epics/E03-ui-visual-refinement/tasks/T006-forced-256-color-profile.md +36 -0
  34. package/.savepoint/releases/v1.1/epics/E04-epic-navigation/E04-Audit.md +167 -0
  35. package/.savepoint/releases/v1.1/epics/E04-epic-navigation/E04-Detail.md +51 -0
  36. package/.savepoint/releases/v1.1/epics/E04-epic-navigation/tasks/T001-sidebar-focusable-navigation.md +65 -0
  37. package/.savepoint/releases/v1.1/epics/E04-epic-navigation/tasks/T002-epic-detail-overlay.md +73 -0
  38. package/.savepoint/releases/v1.1/epics/E04-epic-navigation/tasks/T003-epic-status-glyphs.md +73 -0
  39. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/E05-Audit.md +237 -0
  40. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/E05-Detail.md +54 -0
  41. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T001-update-agents-md.md +45 -0
  42. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T002-update-router-md.md +40 -0
  43. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T003-update-design-md.md +47 -0
  44. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T004-implement-m-hotkey.md +98 -0
  45. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T005-update-help-overlay.md +33 -0
  46. package/.savepoint/releases/v1.1/epics/E05-tasking-permissions/tasks/T006-tests-and-quality-gates.md +62 -0
  47. package/.savepoint/releases/v1.1/epics/E06-audit-command/E06-Audit.md +56 -0
  48. package/.savepoint/releases/v1.1/epics/E06-audit-command/E06-Detail.md +63 -0
  49. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T005-proposals.md +44 -0
  50. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T007-apply-close.md +35 -0
  51. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T009-integration.md +40 -0
  52. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T010-audit-file-migration.md +45 -0
  53. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T011-model-tab-state.md +26 -0
  54. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T012-epic-audit-render.md +33 -0
  55. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T013-handle-tab-keys.md +34 -0
  56. package/.savepoint/releases/v1.1/epics/E06-audit-command/tasks/T014-tab-indicator.md +33 -0
  57. package/.savepoint/releases/v1.1/epics/E07-init-command/E07-Audit.md +336 -0
  58. package/.savepoint/releases/v1.1/epics/E07-init-command/E07-Detail.md +61 -0
  59. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T001-cli-entrypoint.md +37 -0
  60. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T002-target-validation.md +28 -0
  61. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T003-scaffold-writer.md +46 -0
  62. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T004-atomic-writes.md +27 -0
  63. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T005-magic-prompt.md +25 -0
  64. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T006-clipboard.md +26 -0
  65. package/.savepoint/releases/v1.1/epics/E07-init-command/tasks/T007-integration-test.md +26 -0
  66. package/.savepoint/releases/v1.1/epics/E08-board-command/E08-Audit.md +333 -0
  67. package/.savepoint/releases/v1.1/epics/E08-board-command/E08-Detail.md +68 -0
  68. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T001-cli-entrypoint.md +26 -0
  69. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T002-non-tty-fallback.md +27 -0
  70. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T003-tui-app-shell.md +28 -0
  71. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T004-board-model.md +29 -0
  72. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T005-detail-pane.md +27 -0
  73. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T006-status-transitions.md +29 -0
  74. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T007-theme-fallbacks.md +29 -0
  75. package/.savepoint/releases/v1.1/epics/E08-board-command/tasks/T008-integration-test.md +27 -0
  76. package/.savepoint/releases/v1.1/epics/E09-doctor-command/E09-Audit.md +207 -0
  77. package/.savepoint/releases/v1.1/epics/E09-doctor-command/E09-Detail.md +65 -0
  78. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T001-cli-entrypoint.md +24 -0
  79. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T002-config-router-validation.md +28 -0
  80. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T003-structure-checks.md +29 -0
  81. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T004-dependency-checks.md +27 -0
  82. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T005-audit-orphan-checks.md +28 -0
  83. package/.savepoint/releases/v1.1/epics/E09-doctor-command/tasks/T006-quality-gates-report.md +31 -0
  84. package/.savepoint/releases/v1.1/epics/E11-board-refresh-fix/E11-Detail.md +36 -0
  85. package/.savepoint/releases/v1.1/epics/E11-board-refresh-fix/tasks/T001-debug-logging.md +25 -0
  86. package/.savepoint/releases/v1.1/epics/E11-board-refresh-fix/tasks/T002-increase-debounce.md +21 -0
  87. package/.savepoint/releases/v1.1/epics/E11-board-refresh-fix/tasks/T003-error-handling.md +22 -0
  88. package/.savepoint/releases/v1.1/epics/E11-board-refresh-fix/tasks/T004-test-verify.md +29 -0
  89. package/.savepoint/releases/v1.1/epics/E12-validation-fix/E12-Audit.md +444 -0
  90. package/.savepoint/releases/v1.1/epics/E12-validation-fix/E12-Detail.md +45 -0
  91. package/.savepoint/releases/v1.1/epics/E12-validation-fix/tasks/T001-default-phase.md +35 -0
  92. package/.savepoint/releases/v1.1/epics/E12-validation-fix/tasks/T002-default-status.md +19 -0
  93. package/.savepoint/releases/v1.1/epics/E12-validation-fix/tasks/T003-better-errors.md +29 -0
  94. package/.savepoint/releases/v1.1/epics/E12-validation-fix/tasks/T004-validate-on-write.md +25 -0
  95. package/.savepoint/releases/v1.1/epics/E12-validation-fix/tasks/T005-tests.md +37 -0
  96. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/E13-Audit.md +118 -0
  97. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/E13-Detail.md +73 -0
  98. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T001-safe-cleanup.md +66 -0
  99. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T002-bug-fixes.md +35 -0
  100. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T003-centralize-duplication.md +60 -0
  101. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T004-infrastructure.md +33 -0
  102. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T005-decompose-update.md +37 -0
  103. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T006-async-io.md +40 -0
  104. package/.savepoint/releases/v1.1/epics/E13-audit-remediation/tasks/T007-test-coverage.md +37 -0
  105. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/E14-Audit.md +267 -0
  106. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/E14-Detail.md +54 -0
  107. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T001-group-model.md +39 -0
  108. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T002-data-interfaces.md +42 -0
  109. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T003-discover-orphans.md +33 -0
  110. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T004-epic-panel-headings.md +35 -0
  111. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T005-shell-tokenization.md +27 -0
  112. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T006-unify-enums.md +29 -0
  113. package/.savepoint/releases/v1.1/epics/E14-structural-improvements/tasks/T007-testutil-package.md +28 -0
  114. package/.savepoint/releases/v1.1/epics/E15-hardening/E15-Detail.md +43 -0
  115. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T001-benchmarks.md +31 -0
  116. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T002-fuzz-targets.md +28 -0
  117. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T003-debug-flag.md +30 -0
  118. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T004-dist-checksums.md +27 -0
  119. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T005-windows-targets.md +28 -0
  120. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T006-abbreviation-splitting.md +26 -0
  121. package/.savepoint/releases/v1.1/epics/E15-hardening/tasks/T007-root-test-allowlist.md +28 -0
  122. package/.savepoint/releases/v1.1/epics/_archived/T001-cli-entrypoint.md +25 -0
  123. package/.savepoint/releases/v1.1/epics/_archived/T002-quality-gates.md +27 -0
  124. package/.savepoint/releases/v1.1/epics/_archived/T003-snapshot.md +27 -0
  125. package/.savepoint/releases/v1.1/epics/_archived/T004-ai-reconcile.md +29 -0
  126. package/.savepoint/releases/v1.1/epics/_archived/T006-tui-review.md +31 -0
  127. package/.savepoint/releases/v1.1/epics/_archived/T008-skip-handling.md +34 -0
  128. package/.savepoint/releases/v1.1/v1.1-PRD.md +139 -0
  129. package/.savepoint/router.md +29 -108
  130. package/AGENTS.md +69 -111
  131. package/Makefile +19 -3
  132. package/README.md +6 -6
  133. package/agent-skills/savepoint-audit/SKILL.md +87 -35
  134. package/agent-skills/savepoint-build-task/SKILL.md +9 -4
  135. package/agent-skills/savepoint-create-plan/SKILL.md +10 -5
  136. package/agent-skills/savepoint-create-task/SKILL.md +44 -31
  137. package/agent-skills/savepoint-draft-prd/SKILL.md +8 -3
  138. package/agent-skills/savepoint-system-design/SKILL.md +8 -3
  139. package/agent_skills_test.go +91 -0
  140. package/cmd/board.go +59 -0
  141. package/cmd/board_test.go +137 -0
  142. package/cmd/doctor.go +53 -0
  143. package/cmd/doctor_test.go +146 -0
  144. package/cmd/init.go +63 -0
  145. package/cmd/init_test.go +104 -0
  146. package/internal/board/board.go +69 -49
  147. package/internal/board/board_test.go +83 -67
  148. package/internal/board/card.go +71 -20
  149. package/internal/board/card_test.go +141 -12
  150. package/internal/board/column.go +77 -11
  151. package/internal/board/column_test.go +63 -13
  152. package/internal/board/detail.go +107 -72
  153. package/internal/board/detail_test.go +117 -26
  154. package/internal/board/epic_panel.go +211 -18
  155. package/internal/board/epic_panel_test.go +637 -14
  156. package/internal/board/help.go +1 -0
  157. package/internal/board/help_test.go +1 -0
  158. package/internal/board/integration_test.go +266 -0
  159. package/internal/board/interfaces.go +65 -0
  160. package/internal/board/interfaces_test.go +114 -0
  161. package/internal/board/io.go +93 -0
  162. package/internal/board/layout.go +12 -2
  163. package/internal/board/layout_test.go +17 -0
  164. package/internal/board/model.go +130 -52
  165. package/internal/board/plain.go +88 -0
  166. package/internal/board/plain_test.go +117 -0
  167. package/internal/board/release.go +1 -9
  168. package/internal/board/release_test.go +6 -6
  169. package/internal/board/render_policy_test.go +77 -0
  170. package/internal/board/status.go +23 -0
  171. package/internal/board/theme.go +24 -0
  172. package/internal/board/theme_test.go +31 -0
  173. package/internal/board/transitions.go +113 -88
  174. package/internal/board/transitions_test.go +164 -141
  175. package/internal/board/tui.go +32 -0
  176. package/internal/board/update.go +472 -94
  177. package/internal/board/update_test.go +447 -0
  178. package/internal/board/util.go +76 -0
  179. package/internal/board/view.go +139 -22
  180. package/internal/board/view_test.go +171 -3
  181. package/internal/board/watch.go +57 -9
  182. package/internal/buildtool/main.go +211 -0
  183. package/internal/buildtool/main_test.go +46 -0
  184. package/internal/data/config.go +17 -3
  185. package/internal/data/config_test.go +49 -0
  186. package/internal/data/discover.go +26 -0
  187. package/internal/data/discover_test.go +34 -10
  188. package/internal/data/errors.go +4 -0
  189. package/internal/data/lifecycle.go +13 -6
  190. package/internal/data/lifecycle_test.go +14 -11
  191. package/internal/data/parser.go +29 -6
  192. package/internal/data/parser_test.go +66 -7
  193. package/internal/data/task.go +1 -0
  194. package/internal/data/write.go +85 -11
  195. package/internal/data/write_test.go +167 -0
  196. package/internal/doctor/checks.go +567 -0
  197. package/internal/doctor/checks_test.go +716 -0
  198. package/internal/doctor/gates.go +193 -0
  199. package/internal/doctor/gates_test.go +166 -0
  200. package/internal/doctor/interfaces.go +64 -0
  201. package/internal/doctor/interfaces_test.go +104 -0
  202. package/internal/doctor/repairs.go +80 -0
  203. package/internal/doctor/repairs_test.go +81 -0
  204. package/internal/doctor/report.go +157 -0
  205. package/internal/doctor/report_test.go +89 -0
  206. package/internal/init/clipboard.go +146 -0
  207. package/internal/init/clipboard_test.go +74 -0
  208. package/internal/init/install.go +16 -0
  209. package/internal/init/integration_test.go +197 -0
  210. package/internal/init/prompt.go +14 -0
  211. package/internal/init/prompt_test.go +77 -0
  212. package/internal/init/scaffold.go +59 -0
  213. package/internal/init/scaffold_test.go +179 -0
  214. package/internal/init/template_freshness_test.go +56 -0
  215. package/internal/init/validate.go +85 -0
  216. package/internal/init/validate_test.go +141 -0
  217. package/internal/init/write.go +73 -0
  218. package/internal/init/write_test.go +91 -0
  219. package/internal/styles/palette.go +3 -3
  220. package/internal/styles/styles.go +39 -12
  221. package/internal/styles/styles_test.go +133 -0
  222. package/internal/testutil/fixture.go +113 -0
  223. package/internal/testutil/fs.go +26 -0
  224. package/main.go +107 -1
  225. package/package.json +2 -2
  226. package/project-audit/audit_report_glm_5.1.md +411 -0
  227. package/project-audit/audit_report_opus_4.6 +406 -0
  228. package/project-audit/consolidated-audit-report.md +456 -0
  229. package/savepoint +0 -0
  230. package/templates/project/.savepoint/Design.md +2 -2
  231. package/templates/project/.savepoint/router.md +15 -14
  232. package/templates/project/AGENTS.md +56 -98
  233. package/templates/project/agent-skills/savepoint-audit/SKILL.md +87 -0
  234. package/templates/project/agent-skills/savepoint-build-task/SKILL.md +44 -0
  235. package/templates/project/agent-skills/savepoint-create-plan/SKILL.md +33 -0
  236. package/templates/project/agent-skills/savepoint-create-task/SKILL.md +44 -0
  237. package/templates/project/agent-skills/savepoint-draft-prd/SKILL.md +37 -0
  238. package/templates/project/agent-skills/savepoint-system-design/SKILL.md +38 -0
  239. package/templates/prompts/audit-reconciliation.prompt.md +35 -30
  240. package/templates/prompts/design.prompt.md +3 -1
  241. package/templates/prompts/epic-design.prompt.md +3 -3
  242. package/templates/prompts/task-breakdown.prompt.md +1 -1
  243. package/templates/prompts/task-building.prompt.md +1 -1
  244. package/templates/prompts/task-planning.prompt.md +1 -1
  245. package/.savepoint/audit/E01-go-setup/proposals.md +0 -166
  246. package/.savepoint/audit/E01-go-setup/snapshot.md +0 -71
  247. package/.savepoint/audit/E01-scaffolding/proposals/AGENTS.md +0 -66
  248. package/.savepoint/audit/E01-scaffolding/proposals/Design.md +0 -210
  249. package/.savepoint/audit/E01-scaffolding/proposals/epic-Design.md +0 -117
  250. package/.savepoint/audit/E01-scaffolding/proposals/quality-review.md +0 -101
  251. package/.savepoint/audit/E01-scaffolding/snapshot.md +0 -54
  252. package/.savepoint/audit/E02-data-model/snapshot.md +0 -128
  253. package/.savepoint/audit/E02-data-readers/proposals.md +0 -123
  254. package/.savepoint/audit/E02-data-readers/snapshot.md +0 -54
  255. package/.savepoint/audit/E03-board-tui-core/proposals.md +0 -146
  256. package/.savepoint/audit/E03-board-tui-core/snapshot.md +0 -57
  257. package/.savepoint/audit/E03-cli-foundation/snapshot.md +0 -106
  258. package/.savepoint/audit/E04-board-components/proposals.md +0 -118
  259. package/.savepoint/audit/E04-board-components/snapshot.md +0 -77
  260. package/.savepoint/audit/E04-templates-and-prompts/snapshot.md +0 -115
  261. package/.savepoint/audit/E05-init-command/snapshot.md +0 -125
  262. package/.savepoint/audit/E05-phase-transitions/proposals.md +0 -83
  263. package/.savepoint/audit/E05-phase-transitions/snapshot.md +0 -36
  264. package/.savepoint/audit/E06-atari-noir-layout/proposals.md +0 -130
  265. package/.savepoint/audit/E06-atari-noir-layout/snapshot.md +0 -84
  266. package/.savepoint/audit/E06-tui-board/snapshot.md +0 -64
  267. package/.savepoint/audit/E07-audit-pipeline/snapshot.md +0 -165
  268. package/.savepoint/audit/E08-board-workflow-cleanup/snapshot.md +0 -65
  269. package/.savepoint/releases/v1.1/epics/E01-tui-optimisation/tasks/T001-border-resize-fix.md +0 -36
  270. package/ink-cli-ui-design.zip +0 -0
  271. package/main.exe +0 -0
  272. package/savepoint.exe +0 -0
  273. /package/.savepoint/releases/v1/epics/E01-go-setup/{Design.md → E01-Detail.md} +0 -0
  274. /package/.savepoint/releases/v1/epics/E02-data-readers/{Design.md → E02-Detail.md} +0 -0
  275. /package/.savepoint/releases/v1/epics/E03-board-tui-core/{Design.md → E03-Detail.md} +0 -0
  276. /package/.savepoint/releases/v1/epics/E04-board-components/{Design.md → E04-Detail.md} +0 -0
  277. /package/.savepoint/releases/v1/epics/E05-phase-transitions/{Design.md → E05-Detail.md} +0 -0
  278. /package/.savepoint/releases/v1/{PRD.md → v1-PRD.md} +0 -0
@@ -1,6 +1,8 @@
1
1
  package board
2
2
 
3
3
  import (
4
+ "os"
5
+ "path/filepath"
4
6
  "strings"
5
7
  "testing"
6
8
 
@@ -9,22 +11,29 @@ import (
9
11
  )
10
12
 
11
13
  func TestRenderEpicSidebar_containsEpicsHeader(t *testing.T) {
12
- got := RenderEpicSidebar([]string{"E01", "E02"}, "E01", 28)
14
+ got := RenderEpicSidebar([]string{"E01", "E02"}, "E01", 28, false, 0, nil, 999)
13
15
  if !strings.Contains(got, "EPICS") {
14
16
  t.Error("RenderEpicSidebar missing EPICS header")
15
17
  }
16
18
  }
17
19
 
18
20
  func TestRenderEpicSidebar_activeEpicMarked(t *testing.T) {
19
- got := RenderEpicSidebar([]string{"E01", "E02"}, "E01", 28)
21
+ got := RenderEpicSidebar([]string{"E01", "E02"}, "E01", 28, false, 0, nil, 999)
20
22
  if !strings.Contains(got, epicActiveMarker) {
21
23
  t.Errorf("RenderEpicSidebar missing active marker %q", epicActiveMarker)
22
24
  }
23
25
  }
24
26
 
27
+ func TestRenderEpicSidebar_focusedCursorMarked(t *testing.T) {
28
+ got := RenderEpicSidebar([]string{"E01", "E02"}, "E01", 28, true, 1, nil, 999)
29
+ if !strings.Contains(got, epicActiveMarker+" E02") {
30
+ t.Errorf("RenderEpicSidebar focused cursor missing marker, got %q", got)
31
+ }
32
+ }
33
+
25
34
  func TestRenderEpicSidebar_allEpicsPresent(t *testing.T) {
26
35
  epics := []string{"E01-foo", "E02-bar", "E03-baz"}
27
- got := RenderEpicSidebar(epics, "E01-foo", 32)
36
+ got := RenderEpicSidebar(epics, "E01-foo", 32, false, 0, nil, 999)
28
37
  for _, e := range epics {
29
38
  if !strings.Contains(got, e) {
30
39
  t.Errorf("RenderEpicSidebar missing epic %q", e)
@@ -33,14 +42,14 @@ func TestRenderEpicSidebar_allEpicsPresent(t *testing.T) {
33
42
  }
34
43
 
35
44
  func TestRenderEpicSidebar_emptyEpicsFallback(t *testing.T) {
36
- got := RenderEpicSidebar(nil, "E03", 28)
45
+ got := RenderEpicSidebar(nil, "E03", 28, false, 0, nil, 999)
37
46
  if !strings.Contains(got, "E03") {
38
47
  t.Error("RenderEpicSidebar with empty list should show selected epic")
39
48
  }
40
49
  }
41
50
 
42
51
  func TestRenderEpicSidebar_emptyBothShowsNone(t *testing.T) {
43
- got := RenderEpicSidebar(nil, "", 28)
52
+ got := RenderEpicSidebar(nil, "", 28, false, 0, nil, 999)
44
53
  if !strings.Contains(got, "(none)") {
45
54
  t.Error("RenderEpicSidebar with no epics and no selected should show (none)")
46
55
  }
@@ -74,22 +83,22 @@ func TestRenderEpicDropdown_emptyShowsNone(t *testing.T) {
74
83
  }
75
84
  }
76
85
 
77
- func TestEpicIndex_found(t *testing.T) {
86
+ func TestSliceIndex_found(t *testing.T) {
78
87
  epics := []string{"E01", "E02", "E03"}
79
- if got := epicIndex(epics, "E02"); got != 1 {
80
- t.Errorf("epicIndex = %d, want 1", got)
88
+ if got := sliceIndex(epics, "E02"); got != 1 {
89
+ t.Errorf("sliceIndex = %d, want 1", got)
81
90
  }
82
91
  }
83
92
 
84
- func TestEpicIndex_notFound(t *testing.T) {
85
- if got := epicIndex([]string{"E01"}, "E99"); got != 0 {
86
- t.Errorf("epicIndex not-found = %d, want 0", got)
93
+ func TestSliceIndex_notFound(t *testing.T) {
94
+ if got := sliceIndex([]string{"E01"}, "E99"); got != 0 {
95
+ t.Errorf("sliceIndex not-found = %d, want 0", got)
87
96
  }
88
97
  }
89
98
 
90
- func TestEpicIndex_empty(t *testing.T) {
91
- if got := epicIndex(nil, "E01"); got != 0 {
92
- t.Errorf("epicIndex empty = %d, want 0", got)
99
+ func TestSliceIndex_empty(t *testing.T) {
100
+ if got := sliceIndex(nil, "E01"); got != 0 {
101
+ t.Errorf("sliceIndex empty = %d, want 0", got)
93
102
  }
94
103
  }
95
104
 
@@ -211,6 +220,211 @@ func TestUpdate_overlayBlocksColumnNav(t *testing.T) {
211
220
  }
212
221
  }
213
222
 
223
+ func TestUpdate_leftFromPlannedFocusesEpicPanelWide(t *testing.T) {
224
+ m := NewModel(nil, "v1", "E02")
225
+ m.Width = 120
226
+ m.Epics = []string{"E01", "E02", "E03"}
227
+
228
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune("h")})
229
+ updated := requireModel(t, got)
230
+
231
+ if !updated.EpicPanelFocus {
232
+ t.Fatal("EpicPanelFocus = false, want true")
233
+ }
234
+ if updated.EpicPanelCursor != 1 {
235
+ t.Errorf("EpicPanelCursor = %d, want 1", updated.EpicPanelCursor)
236
+ }
237
+ if updated.FocusedColumn != data.ColumnPlanned {
238
+ t.Errorf("FocusedColumn = %q, want planned", updated.FocusedColumn)
239
+ }
240
+ }
241
+
242
+ func TestUpdate_leftFromPlannedDoesNotFocusEpicPanelNarrow(t *testing.T) {
243
+ m := NewModel(nil, "v1", "E02")
244
+ m.Width = 100
245
+ m.Epics = []string{"E01", "E02"}
246
+
247
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune("h")})
248
+ updated := requireModel(t, got)
249
+
250
+ if updated.EpicPanelFocus {
251
+ t.Fatal("EpicPanelFocus = true, want false")
252
+ }
253
+ if updated.FocusedColumn != data.ColumnDone {
254
+ t.Errorf("FocusedColumn = %q, want done", updated.FocusedColumn)
255
+ }
256
+ }
257
+
258
+ func TestUpdate_leftFromPlannedDoesNotFocusEmptyEpicPanel(t *testing.T) {
259
+ m := NewModel(nil, "v1", "")
260
+ m.Width = 120
261
+
262
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune("h")})
263
+ updated := requireModel(t, got)
264
+
265
+ if updated.EpicPanelFocus {
266
+ t.Fatal("EpicPanelFocus = true, want false with no epics")
267
+ }
268
+ if updated.EpicPanelCursor != 0 {
269
+ t.Errorf("EpicPanelCursor = %d, want 0", updated.EpicPanelCursor)
270
+ }
271
+ }
272
+
273
+ func TestUpdate_windowResizeClearsEpicPanelFocusWhenHidden(t *testing.T) {
274
+ m := NewModel(nil, "v1", "E01")
275
+ m.Width = 120
276
+ m.Epics = []string{"E01"}
277
+ m.EpicPanelFocus = true
278
+
279
+ got, _ := m.Update(tea.WindowSizeMsg{Width: 100, Height: 24})
280
+ updated := requireModel(t, got)
281
+
282
+ if updated.EpicPanelFocus {
283
+ t.Fatal("EpicPanelFocus = true, want false when panel is hidden")
284
+ }
285
+ }
286
+
287
+ func TestUpdate_epicPanelDownUpClamped(t *testing.T) {
288
+ m := NewModel(nil, "v1", "E01")
289
+ m.Width = 120
290
+ m.EpicPanelFocus = true
291
+ m.Epics = []string{"E01", "E02"}
292
+
293
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyDown})
294
+ updated := requireModel(t, got)
295
+ if updated.EpicPanelCursor != 1 {
296
+ t.Errorf("EpicPanelCursor after down = %d, want 1", updated.EpicPanelCursor)
297
+ }
298
+
299
+ got, _ = updated.Update(tea.KeyMsg{Type: tea.KeyDown})
300
+ updated = requireModel(t, got)
301
+ if updated.EpicPanelCursor != 1 {
302
+ t.Errorf("EpicPanelCursor after clamped down = %d, want 1", updated.EpicPanelCursor)
303
+ }
304
+
305
+ got, _ = updated.Update(tea.KeyMsg{Type: tea.KeyUp})
306
+ updated = requireModel(t, got)
307
+ if updated.EpicPanelCursor != 0 {
308
+ t.Errorf("EpicPanelCursor after up = %d, want 0", updated.EpicPanelCursor)
309
+ }
310
+ }
311
+
312
+ func TestUpdate_epicPanelEnterOpensDetailOverlay(t *testing.T) {
313
+ tasks := []data.Task{
314
+ {ID: "T1", Epic: "E01", Release: "v1", Column: data.ColumnPlanned},
315
+ {ID: "T2", Epic: "E02", Release: "v1", Column: data.ColumnPlanned},
316
+ }
317
+ m := NewModel(tasks, "v1", "E01")
318
+ m.Width = 120
319
+ m.Epics = []string{"E01", "E02"}
320
+ m.EpicPanelFocus = true
321
+ m.EpicPanelCursor = 1
322
+ m.FocusedTask = 3
323
+ m.DetailOffset = 2
324
+
325
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyEnter})
326
+ updated := requireModel(t, got)
327
+
328
+ if updated.Overlay != OverlayEpicDetail {
329
+ t.Errorf("Overlay = %q, want %q", updated.Overlay, OverlayEpicDetail)
330
+ }
331
+ if updated.EpicDetailOffset != 0 {
332
+ t.Errorf("EpicDetailOffset = %d, want 0", updated.EpicDetailOffset)
333
+ }
334
+ if !updated.EpicPanelFocus {
335
+ t.Error("EpicPanelFocus should remain true after enter")
336
+ }
337
+ // SelectedEpic unchanged; Enter now opens detail, not selects
338
+ if updated.SelectedEpic != "E01" {
339
+ t.Errorf("SelectedEpic = %q, want E01 (unchanged)", updated.SelectedEpic)
340
+ }
341
+ }
342
+
343
+ func TestUpdate_epicPanelRightReturnsToPlanned(t *testing.T) {
344
+ m := NewModel(nil, "v1", "E01")
345
+ m.Width = 120
346
+ m.EpicPanelFocus = true
347
+ m.Epics = []string{"E01"}
348
+ m.FocusedColumn = data.ColumnDone
349
+ m.FocusedTask = 2
350
+
351
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune("l")})
352
+ updated := requireModel(t, got)
353
+
354
+ if updated.EpicPanelFocus {
355
+ t.Fatal("EpicPanelFocus = true, want false")
356
+ }
357
+ if updated.FocusedColumn != data.ColumnPlanned {
358
+ t.Errorf("FocusedColumn = %q, want planned", updated.FocusedColumn)
359
+ }
360
+ if updated.FocusedTask != 0 {
361
+ t.Errorf("FocusedTask = %d, want 0", updated.FocusedTask)
362
+ }
363
+ }
364
+
365
+ func TestUpdate_overlayBlocksEpicPanelNav(t *testing.T) {
366
+ m := NewModel(nil, "v1", "E01")
367
+ m.Width = 120
368
+ m.EpicPanelFocus = true
369
+ m.Epics = []string{"E01", "E02"}
370
+ m.Overlay = OverlayHelp
371
+
372
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyDown})
373
+ updated := requireModel(t, got)
374
+
375
+ if updated.EpicPanelCursor != 0 {
376
+ t.Errorf("EpicPanelCursor = %d, want 0 while overlay open", updated.EpicPanelCursor)
377
+ }
378
+ }
379
+
380
+ func TestUpdate_epicPanelFocusAllowsGlobalQuit(t *testing.T) {
381
+ m := NewModel(nil, "v1", "E01")
382
+ m.Width = 120
383
+ m.EpicPanelFocus = true
384
+ m.Epics = []string{"E01"}
385
+
386
+ _, cmd := m.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune("q")})
387
+ if cmd == nil {
388
+ t.Fatal("expected tea.Quit cmd from q while epic panel focused, got nil")
389
+ }
390
+ }
391
+
392
+ func TestUpdate_epicPanelFocusAllowsEpicDropdown(t *testing.T) {
393
+ m := NewModel(nil, "v1", "E02")
394
+ m.Width = 120
395
+ m.EpicPanelFocus = true
396
+ m.Epics = []string{"E01", "E02"}
397
+
398
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune("e")})
399
+ updated := requireModel(t, got)
400
+
401
+ if updated.Overlay != OverlayEpic {
402
+ t.Errorf("Overlay = %q, want %q", updated.Overlay, OverlayEpic)
403
+ }
404
+ if updated.EpicCursor != 1 {
405
+ t.Errorf("EpicCursor = %d, want 1", updated.EpicCursor)
406
+ }
407
+ }
408
+
409
+ func TestUpdate_epicPanelFocusAllowsReleaseDropdown(t *testing.T) {
410
+ m := NewModel(nil, "v1", "E01")
411
+ m.Width = 120
412
+ m.EpicPanelFocus = true
413
+ m.Epics = []string{"E01"}
414
+ m.Releases = []string{"v1", "v1.1"}
415
+ m.SelectedRelease = "v1.1"
416
+
417
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune("r")})
418
+ updated := requireModel(t, got)
419
+
420
+ if updated.Overlay != OverlayRelease {
421
+ t.Errorf("Overlay = %q, want %q", updated.Overlay, OverlayRelease)
422
+ }
423
+ if updated.ReleaseCursor != 1 {
424
+ t.Errorf("ReleaseCursor = %d, want 1", updated.ReleaseCursor)
425
+ }
426
+ }
427
+
214
428
  func TestView_epicDropdownOverlayRendered(t *testing.T) {
215
429
  m := NewModel(nil, "v1", "E01")
216
430
  m.Width = 80
@@ -244,3 +458,412 @@ func TestView_epicSidebarOnWide(t *testing.T) {
244
458
  t.Error("View() at width>=120 missing EPICS header in sidebar")
245
459
  }
246
460
  }
461
+
462
+ func TestUpdate_epicDetailOverlayEscCloses(t *testing.T) {
463
+ m := NewModel(nil, "v1", "E01")
464
+ m.Width = 120
465
+ m.EpicPanelFocus = true
466
+ m.Epics = []string{"E01"}
467
+ m.Overlay = OverlayEpicDetail
468
+ m.EpicDetailContent = "# E01 Detail"
469
+ m.EpicDetailOffset = 3
470
+
471
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyEsc})
472
+ updated := requireModel(t, got)
473
+
474
+ if updated.Overlay != OverlayNone {
475
+ t.Errorf("Overlay = %q, want none after esc", updated.Overlay)
476
+ }
477
+ if !updated.EpicPanelFocus {
478
+ t.Error("EpicPanelFocus should remain true after closing detail overlay")
479
+ }
480
+ }
481
+
482
+ func TestUpdate_epicDetailOverlayScrollUpDown(t *testing.T) {
483
+ m := NewModel(nil, "v1", "E01")
484
+ m.Width = 120
485
+ m.EpicPanelFocus = true
486
+ m.Epics = []string{"E01"}
487
+ m.Overlay = OverlayEpicDetail
488
+ m.EpicDetailOffset = 2
489
+
490
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyDown})
491
+ updated := requireModel(t, got)
492
+ if updated.EpicDetailOffset != 3 {
493
+ t.Errorf("EpicDetailOffset after down = %d, want 3", updated.EpicDetailOffset)
494
+ }
495
+
496
+ got, _ = updated.Update(tea.KeyMsg{Type: tea.KeyUp})
497
+ updated = requireModel(t, got)
498
+ if updated.EpicDetailOffset != 2 {
499
+ t.Errorf("EpicDetailOffset after up = %d, want 2", updated.EpicDetailOffset)
500
+ }
501
+
502
+ got, _ = updated.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune("k")})
503
+ updated = requireModel(t, got)
504
+ if updated.EpicDetailOffset != 1 {
505
+ t.Errorf("EpicDetailOffset after k = %d, want 1", updated.EpicDetailOffset)
506
+ }
507
+
508
+ // Clamp at 0
509
+ updated.EpicDetailOffset = 0
510
+ got, _ = updated.Update(tea.KeyMsg{Type: tea.KeyUp})
511
+ updated = requireModel(t, got)
512
+ if updated.EpicDetailOffset != 0 {
513
+ t.Errorf("EpicDetailOffset should not go below 0, got %d", updated.EpicDetailOffset)
514
+ }
515
+ }
516
+
517
+ func TestUpdate_epicDetailOverlayPgUpDown(t *testing.T) {
518
+ m := NewModel(nil, "v1", "E01")
519
+ m.Width = 120
520
+ m.Height = 30
521
+ m.Epics = []string{"E01"}
522
+ m.Overlay = OverlayEpicDetail
523
+ m.EpicDetailOffset = 20
524
+
525
+ got, _ := m.Update(tea.KeyMsg{Type: tea.KeyPgUp})
526
+ updated := requireModel(t, got)
527
+ if updated.EpicDetailOffset >= 20 {
528
+ t.Errorf("EpicDetailOffset after pgup = %d, should decrease from 20", updated.EpicDetailOffset)
529
+ }
530
+ if updated.EpicDetailOffset < 0 {
531
+ t.Errorf("EpicDetailOffset = %d, should not go below 0", updated.EpicDetailOffset)
532
+ }
533
+
534
+ updated.EpicDetailOffset = 0
535
+ got, _ = updated.Update(tea.KeyMsg{Type: tea.KeyPgDown})
536
+ updated = requireModel(t, got)
537
+ if updated.EpicDetailOffset <= 0 {
538
+ t.Errorf("EpicDetailOffset after pgdown = %d, should increase from 0", updated.EpicDetailOffset)
539
+ }
540
+ }
541
+
542
+ func TestView_epicDetailOverlayRendered(t *testing.T) {
543
+ m := NewModel(nil, "v1", "E01")
544
+ m.Width = 120
545
+ m.Height = 30
546
+ m.Epics = []string{"E01"}
547
+ m.Overlay = OverlayEpicDetail
548
+ m.EpicDetailContent = "# My Epic\n\n## Purpose\nDoes things."
549
+
550
+ got := m.View()
551
+ if !strings.Contains(got, "EPIC DETAIL") {
552
+ t.Error("View() with OverlayEpicDetail missing EPIC DETAIL header")
553
+ }
554
+ }
555
+
556
+ func TestView_epicDetailOverlayNoContent(t *testing.T) {
557
+ m := NewModel(nil, "v1", "E01")
558
+ m.Width = 120
559
+ m.Height = 30
560
+ m.Epics = []string{"E01"}
561
+ m.Overlay = OverlayEpicDetail
562
+ m.EpicDetailContent = "(no detail available)"
563
+
564
+ got := m.View()
565
+ if !strings.Contains(got, "no detail available") {
566
+ t.Error("View() with missing epic detail should show 'no detail available'")
567
+ }
568
+ }
569
+
570
+ func TestRenderEpicDetail_stripsMarkdownHeadings(t *testing.T) {
571
+ content := "---\ntype: epic-design\n---\n# Epic E01\n\n## Purpose\nDoes things."
572
+ got := RenderEpicDetail("E01-test", content, 60, 40, 0, 0)
573
+ if !strings.Contains(got, "EPIC DETAIL") {
574
+ t.Error("RenderEpicDetail missing EPIC DETAIL header")
575
+ }
576
+ if strings.Contains(got, "# Epic E01") {
577
+ t.Error("RenderEpicDetail should strip raw markdown heading prefix")
578
+ }
579
+ }
580
+
581
+ func TestRenderEpicDetail_noDetailFallback(t *testing.T) {
582
+ got := RenderEpicDetail("E01-test", "(no detail available)", 60, 40, 0, 0)
583
+ if !strings.Contains(got, "no detail available") {
584
+ t.Error("RenderEpicDetail fallback message missing")
585
+ }
586
+ }
587
+
588
+ func TestRenderEpicDetail_tabIndicatorDetailActive(t *testing.T) {
589
+ got := RenderEpicDetail("E01-test", "content", 60, 40, 0, 0)
590
+ if !strings.Contains(got, "DETAIL [1]") {
591
+ t.Error("RenderEpicDetail tab=0: missing DETAIL [1] indicator")
592
+ }
593
+ if !strings.Contains(got, "AUDIT [2]") {
594
+ t.Error("RenderEpicDetail tab=0: missing AUDIT [2] indicator")
595
+ }
596
+ }
597
+
598
+ func TestRenderEpicDetail_tabIndicatorAuditActive(t *testing.T) {
599
+ got := RenderEpicDetail("E01-test", "content", 60, 40, 0, 1)
600
+ if !strings.Contains(got, "DETAIL [1]") {
601
+ t.Error("RenderEpicDetail tab=1: missing DETAIL [1] indicator")
602
+ }
603
+ if !strings.Contains(got, "AUDIT [2]") {
604
+ t.Error("RenderEpicDetail tab=1: missing AUDIT [2] indicator")
605
+ }
606
+ }
607
+
608
+ func TestRenderEpicAuditTab_header(t *testing.T) {
609
+ got := RenderEpicAuditTab("E06-test", "# Audit\n\n## Main Findings\nAll good.", 60, 40, 0, 1)
610
+ if !strings.Contains(got, "EPIC AUDIT") {
611
+ t.Error("RenderEpicAuditTab missing EPIC AUDIT header")
612
+ }
613
+ }
614
+
615
+ func TestRenderEpicAuditTab_noContent(t *testing.T) {
616
+ got := RenderEpicAuditTab("E06-test", "(no audit available)", 60, 40, 0, 1)
617
+ if !strings.Contains(got, "no audit available") {
618
+ t.Error("RenderEpicAuditTab fallback message missing")
619
+ }
620
+ }
621
+
622
+ func TestRenderEpicAuditTab_emptyContent(t *testing.T) {
623
+ got := RenderEpicAuditTab("E06-test", "", 60, 40, 0, 1)
624
+ if !strings.Contains(got, "no audit available") {
625
+ t.Error("RenderEpicAuditTab empty content should show fallback")
626
+ }
627
+ }
628
+
629
+ func TestRenderEpicAuditTab_stripsFrontmatter(t *testing.T) {
630
+ content := "---\ntype: audit\n---\n# E06 Audit\n\n## Main Findings\nLooks good."
631
+ got := RenderEpicAuditTab("E06-test", content, 60, 40, 0, 1)
632
+ if strings.Contains(got, "type: audit") {
633
+ t.Error("RenderEpicAuditTab should strip frontmatter")
634
+ }
635
+ if !strings.Contains(got, "EPIC AUDIT") {
636
+ t.Error("RenderEpicAuditTab missing header after frontmatter strip")
637
+ }
638
+ }
639
+
640
+ func TestRenderEpicAuditTab_checkboxDonePresent(t *testing.T) {
641
+ content := "## Code Style Review\n- [x] One job per file\n- [ ] One-sentence functions"
642
+ got := RenderEpicAuditTab("E06-test", content, 60, 40, 0, 1)
643
+ if !strings.Contains(got, "One job per file") {
644
+ t.Error("RenderEpicAuditTab missing done checkbox text")
645
+ }
646
+ if !strings.Contains(got, "One-sentence functions") {
647
+ t.Error("RenderEpicAuditTab missing undone checkbox text")
648
+ }
649
+ }
650
+
651
+ func TestRenderEpicAuditTab_scrollFooter(t *testing.T) {
652
+ got := RenderEpicAuditTab("E06-test", "# Audit", 60, 40, 0, 1)
653
+ if !strings.Contains(got, "esc:close") {
654
+ t.Error("RenderEpicAuditTab missing esc:close footer")
655
+ }
656
+ }
657
+
658
+ func TestRenderEpicAuditTab_tabIndicator(t *testing.T) {
659
+ got := RenderEpicAuditTab("E06-test", "# Audit", 60, 40, 0, 1)
660
+ if !strings.Contains(got, "DETAIL [1]") {
661
+ t.Error("RenderEpicAuditTab missing DETAIL [1] indicator")
662
+ }
663
+ if !strings.Contains(got, "AUDIT [2]") {
664
+ t.Error("RenderEpicAuditTab missing AUDIT [2] indicator")
665
+ }
666
+ }
667
+
668
+ func TestRenderEpicAuditTab_mainFindingsVisible(t *testing.T) {
669
+ content := "## Main Findings\nAudit summary is visible.\n\n## Proposed Changes\n### Target File\nAGENTS.md\n"
670
+ got := RenderEpicAuditTab("E06-test", content, 80, 50, 0, 1)
671
+ if !strings.Contains(got, "Audit summary is visible") {
672
+ t.Error("RenderEpicAuditTab should render Main Findings body")
673
+ }
674
+ if strings.Contains(got, "Target File") || strings.Contains(got, "AGENTS.md") {
675
+ t.Error("RenderEpicAuditTab should not render Proposed Changes admin blocks")
676
+ }
677
+ }
678
+
679
+ func TestRenderEpicAuditTab_qualityReviewHidden(t *testing.T) {
680
+ content := "## Quality Review\nOld quality section.\n\n## Code Style Review\n- [ ] One job per file\n"
681
+ got := RenderEpicAuditTab("E06-test", content, 80, 50, 0, 1)
682
+ if strings.Contains(got, "Old quality section") {
683
+ t.Error("RenderEpicAuditTab should not render superseded Quality Review section")
684
+ }
685
+ if !strings.Contains(got, "One job per file") {
686
+ t.Error("RenderEpicAuditTab should render Code Style Review")
687
+ }
688
+ }
689
+
690
+ func TestRenderEpicAuditTab_hiddenHeadingsRequireExactMatch(t *testing.T) {
691
+ content := "## Proposed Changes Appendix\nNear-match section is visible.\n\n## Proposed Changes\nHidden admin section.\n"
692
+ got := RenderEpicAuditTab("E06-test", content, 80, 50, 0, 1)
693
+ if !strings.Contains(got, "Near-match section is visible") {
694
+ t.Error("RenderEpicAuditTab should render headings that only partially match hidden headings")
695
+ }
696
+ if strings.Contains(got, "Hidden admin section") {
697
+ t.Error("RenderEpicAuditTab should hide exact Proposed Changes section")
698
+ }
699
+ }
700
+
701
+ func TestRenderEpicAuditTab_allCodeStyleRules(t *testing.T) {
702
+ rules := []string{
703
+ "One job per file",
704
+ "One-sentence functions",
705
+ "Test branches",
706
+ "Types are documentation",
707
+ "Build, don't speculate",
708
+ "Errors at boundaries",
709
+ "One source of truth",
710
+ "Comments explain WHY",
711
+ "Content in data files",
712
+ "Small diffs",
713
+ }
714
+ content := "## Code Style Review\n"
715
+ for _, r := range rules {
716
+ content += "- [ ] " + r + "\n"
717
+ }
718
+ got := RenderEpicAuditTab("E06-test", content, 80, 50, 0, 1)
719
+ for _, r := range rules {
720
+ if !strings.Contains(got, r) {
721
+ t.Errorf("RenderEpicAuditTab missing code style rule %q", r)
722
+ }
723
+ }
724
+ }
725
+
726
+ // TestView_epicAuditTabRendered verifies View() uses RenderEpicAuditTab when EpicDetailTab=1.
727
+ func TestView_epicAuditTabRendered(t *testing.T) {
728
+ m := NewModel(nil, "v1.1", "E06-audit-command")
729
+ m.Width = 120
730
+ m.Height = 30
731
+ m.Epics = []string{"E06-audit-command"}
732
+ m.Overlay = OverlayEpicDetail
733
+ m.EpicDetailTab = 1
734
+ m.EpicAuditContent = "# Audit Findings: E06\n\n## Main Findings\nAll good.\n\n## Code Style Review\n- [x] One job per file\n"
735
+
736
+ got := m.View()
737
+ if !strings.Contains(got, "EPIC AUDIT") {
738
+ t.Error("View() with EpicDetailTab=1 missing EPIC AUDIT header")
739
+ }
740
+ if strings.Contains(got, "EPIC DETAIL") {
741
+ t.Error("View() with EpicDetailTab=1 should not render EPIC DETAIL header")
742
+ }
743
+ }
744
+
745
+ // TestAuditWorkflow_fullEndToEnd exercises the full audit workflow:
746
+ // create E##-Audit.md on disk, open overlay, press 2, verify content loads and renders.
747
+ func TestAuditWorkflow_fullEndToEnd(t *testing.T) {
748
+ root := t.TempDir()
749
+ epicSlug := "E06-audit-command"
750
+ epicDir := filepath.Join(root, "releases", "v1.1", "epics", epicSlug)
751
+ if err := os.MkdirAll(epicDir, 0755); err != nil {
752
+ t.Fatal(err)
753
+ }
754
+
755
+ auditContent := `---
756
+ type: audit-findings
757
+ audited: 2026-05-03
758
+ ---
759
+ # Audit Findings: E06 Agent Audit + Audit Tab
760
+
761
+ ## Main Findings
762
+ All acceptance criteria met.
763
+
764
+ ## Code Style Review
765
+ - [x] One job per file
766
+ - [x] One-sentence functions
767
+ - [x] Test branches
768
+ - [x] Types are documentation
769
+ - [x] Build, don't speculate
770
+ - [x] Errors at boundaries
771
+ - [x] One source of truth
772
+ - [x] Comments explain WHY
773
+ - [x] Content in data files
774
+ - [x] Small diffs
775
+ `
776
+ if err := os.WriteFile(filepath.Join(epicDir, "E06-Audit.md"), []byte(auditContent), 0644); err != nil {
777
+ t.Fatal(err)
778
+ }
779
+
780
+ tasks := []data.Task{
781
+ {ID: "E06-audit-command/T009-integration", Release: "v1.1", Epic: epicSlug, Column: data.ColumnPlanned},
782
+ }
783
+ m := NewModel(tasks, "v1.1", epicSlug)
784
+ m.Root = root
785
+ m.Epics = []string{epicSlug}
786
+ m.EpicPanelCursor = 0
787
+ m.Width = 120
788
+ m.Height = 40
789
+
790
+ // Open detail overlay (tab=0)
791
+ m.openEpicDetailOverlay()
792
+ if m.Overlay != OverlayEpicDetail {
793
+ t.Fatal("overlay not opened")
794
+ }
795
+ if m.EpicDetailTab != 0 {
796
+ t.Errorf("EpicDetailTab = %d, want 0 on open", m.EpicDetailTab)
797
+ }
798
+
799
+ // Press 2 → switch to audit tab, load content
800
+ got, cmd := m.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune("2")})
801
+ updated := requireModel(t, got)
802
+
803
+ if updated.EpicDetailTab != 1 {
804
+ t.Errorf("EpicDetailTab = %d, want 1 after pressing 2", updated.EpicDetailTab)
805
+ }
806
+
807
+ msg := cmd()
808
+ got2, _ := updated.Update(msg)
809
+ updated2 := requireModel(t, got2)
810
+ if updated2.EpicAuditContent == "" || updated2.EpicAuditContent == "(no audit available)" {
811
+ t.Errorf("EpicAuditContent not loaded: %q", updated2.EpicAuditContent)
812
+ }
813
+
814
+ // Verify View() renders audit content
815
+ view := updated2.View()
816
+ if !strings.Contains(view, "EPIC AUDIT") {
817
+ t.Error("View() after tab switch missing EPIC AUDIT")
818
+ }
819
+ if !strings.Contains(view, "One job per file") {
820
+ t.Error("View() after tab switch missing code style rule")
821
+ }
822
+
823
+ // Press 1 → switch back to detail tab
824
+ got, _ = updated2.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune("1")})
825
+ updated = requireModel(t, got)
826
+ if updated.EpicDetailTab != 0 {
827
+ t.Errorf("EpicDetailTab = %d, want 0 after pressing 1", updated.EpicDetailTab)
828
+ }
829
+
830
+ // Press esc → overlay closes
831
+ got, _ = updated.Update(tea.KeyMsg{Type: tea.KeyEsc})
832
+ updated = requireModel(t, got)
833
+ if updated.Overlay != OverlayNone {
834
+ t.Errorf("Overlay = %q, want none after esc", updated.Overlay)
835
+ }
836
+ }
837
+
838
+ func TestRenderEpicAuditTab_v11AuditFiles(t *testing.T) {
839
+ files := []struct {
840
+ path string
841
+ want string
842
+ }{
843
+ {filepath.Join("..", "..", ".savepoint", "releases", "v1.1", "epics", "E02-cross-platform-compatibility", "E02-Audit.md"), "cross-platform build work"},
844
+ {filepath.Join("..", "..", ".savepoint", "releases", "v1.1", "epics", "E03-ui-visual-refinement", "E03-Audit.md"), "visual refinement work"},
845
+ {filepath.Join("..", "..", ".savepoint", "releases", "v1.1", "epics", "E04-epic-navigation", "E04-Audit.md"), "wide-screen epic navigation"},
846
+ {filepath.Join("..", "..", ".savepoint", "releases", "v1.1", "epics", "E05-tasking-permissions", "E05-Audit.md"), "tasking-permissions shift"},
847
+ {filepath.Join("..", "..", ".savepoint", "releases", "v1.1", "epics", "E06-audit-command", "E06-Audit.md"), "agent-led"},
848
+ }
849
+
850
+ for _, tt := range files {
851
+ content, err := os.ReadFile(tt.path)
852
+ if err != nil {
853
+ t.Fatalf("read %s: %v", tt.path, err)
854
+ }
855
+ if !strings.Contains(string(content), tt.want) {
856
+ t.Fatalf("fixture %s missing %q", tt.path, tt.want)
857
+ }
858
+ got := RenderEpicAuditTab(filepath.Base(filepath.Dir(tt.path)), string(content), 80, 40, 0, 1)
859
+ if !strings.Contains(got, tt.want) {
860
+ t.Errorf("RenderEpicAuditTab(%s) missing %q", tt.path, tt.want)
861
+ }
862
+ if strings.Contains(got, "Target File") {
863
+ t.Errorf("RenderEpicAuditTab(%s) should not render Proposed Changes", tt.path)
864
+ }
865
+ if strings.Contains(got, "Boundaries") || strings.Contains(got, "Implemented as") || strings.Contains(got, "Implemented As") {
866
+ t.Errorf("RenderEpicAuditTab(%s) should only render visible audit sections", tt.path)
867
+ }
868
+ }
869
+ }