gsd-pi 2.18.0 → 2.20.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (289) hide show
  1. package/README.md +5 -1
  2. package/dist/cli.js +3 -3
  3. package/dist/onboarding.d.ts +3 -1
  4. package/dist/onboarding.js +77 -3
  5. package/dist/remote-questions-config.d.ts +1 -1
  6. package/dist/resources/extensions/google-search/index.ts +164 -47
  7. package/dist/resources/extensions/gsd/auto-dashboard.ts +14 -2
  8. package/dist/resources/extensions/gsd/auto-prompts.ts +148 -39
  9. package/dist/resources/extensions/gsd/auto-worktree.ts +93 -9
  10. package/dist/resources/extensions/gsd/auto.ts +690 -39
  11. package/dist/resources/extensions/gsd/captures.ts +384 -0
  12. package/dist/resources/extensions/gsd/commands.ts +654 -36
  13. package/dist/resources/extensions/gsd/complexity-classifier.ts +322 -0
  14. package/dist/resources/extensions/gsd/context-budget.ts +243 -0
  15. package/dist/resources/extensions/gsd/context-store.ts +195 -0
  16. package/dist/resources/extensions/gsd/dashboard-overlay.ts +51 -3
  17. package/dist/resources/extensions/gsd/db-writer.ts +341 -0
  18. package/dist/resources/extensions/gsd/debug-logger.ts +178 -0
  19. package/dist/resources/extensions/gsd/dispatch-guard.ts +0 -1
  20. package/dist/resources/extensions/gsd/docs/preferences-reference.md +54 -0
  21. package/dist/resources/extensions/gsd/doctor-proactive.ts +286 -0
  22. package/dist/resources/extensions/gsd/doctor.ts +283 -2
  23. package/dist/resources/extensions/gsd/export.ts +81 -2
  24. package/dist/resources/extensions/gsd/files.ts +39 -9
  25. package/dist/resources/extensions/gsd/git-service.ts +6 -0
  26. package/dist/resources/extensions/gsd/gsd-db.ts +752 -0
  27. package/dist/resources/extensions/gsd/guided-flow.ts +26 -1
  28. package/dist/resources/extensions/gsd/history.ts +0 -1
  29. package/dist/resources/extensions/gsd/index.ts +277 -1
  30. package/dist/resources/extensions/gsd/md-importer.ts +526 -0
  31. package/dist/resources/extensions/gsd/metrics.ts +84 -0
  32. package/dist/resources/extensions/gsd/model-cost-table.ts +65 -0
  33. package/dist/resources/extensions/gsd/model-router.ts +256 -0
  34. package/dist/resources/extensions/gsd/notifications.ts +0 -1
  35. package/dist/resources/extensions/gsd/post-unit-hooks.ts +72 -2
  36. package/dist/resources/extensions/gsd/preferences.ts +198 -150
  37. package/dist/resources/extensions/gsd/prompt-loader.ts +45 -9
  38. package/dist/resources/extensions/gsd/prompts/execute-task.md +3 -5
  39. package/dist/resources/extensions/gsd/prompts/heal-skill.md +45 -0
  40. package/dist/resources/extensions/gsd/prompts/plan-slice.md +5 -1
  41. package/dist/resources/extensions/gsd/prompts/quick-task.md +48 -0
  42. package/dist/resources/extensions/gsd/prompts/reassess-roadmap.md +6 -0
  43. package/dist/resources/extensions/gsd/prompts/replan-slice.md +8 -0
  44. package/dist/resources/extensions/gsd/prompts/system.md +2 -1
  45. package/dist/resources/extensions/gsd/prompts/triage-captures.md +62 -0
  46. package/dist/resources/extensions/gsd/quick.ts +156 -0
  47. package/dist/resources/extensions/gsd/skill-discovery.ts +5 -3
  48. package/dist/resources/extensions/gsd/skill-health.ts +417 -0
  49. package/dist/resources/extensions/gsd/skill-telemetry.ts +127 -0
  50. package/dist/resources/extensions/gsd/state.ts +30 -0
  51. package/dist/resources/extensions/gsd/templates/preferences.md +1 -0
  52. package/dist/resources/extensions/gsd/tests/captures.test.ts +438 -0
  53. package/dist/resources/extensions/gsd/tests/complexity-classifier.test.ts +181 -0
  54. package/dist/resources/extensions/gsd/tests/context-budget.test.ts +283 -0
  55. package/dist/resources/extensions/gsd/tests/context-compression.test.ts +1 -1
  56. package/dist/resources/extensions/gsd/tests/context-store.test.ts +462 -0
  57. package/dist/resources/extensions/gsd/tests/continue-here.test.ts +204 -0
  58. package/dist/resources/extensions/gsd/tests/dashboard-budget.test.ts +346 -0
  59. package/dist/resources/extensions/gsd/tests/db-writer.test.ts +602 -0
  60. package/dist/resources/extensions/gsd/tests/debug-logger.test.ts +185 -0
  61. package/dist/resources/extensions/gsd/tests/derive-state-db.test.ts +406 -0
  62. package/dist/resources/extensions/gsd/tests/dispatch-guard.test.ts +0 -1
  63. package/dist/resources/extensions/gsd/tests/dist-redirect.mjs +22 -0
  64. package/dist/resources/extensions/gsd/tests/doctor-proactive.test.ts +244 -0
  65. package/dist/resources/extensions/gsd/tests/doctor-runtime.test.ts +303 -0
  66. package/dist/resources/extensions/gsd/tests/feature-branch-lifecycle-integration.test.ts +434 -0
  67. package/dist/resources/extensions/gsd/tests/gsd-db.test.ts +353 -0
  68. package/dist/resources/extensions/gsd/tests/gsd-inspect.test.ts +125 -0
  69. package/dist/resources/extensions/gsd/tests/gsd-tools.test.ts +326 -0
  70. package/dist/resources/extensions/gsd/tests/integration-edge.test.ts +228 -0
  71. package/dist/resources/extensions/gsd/tests/integration-lifecycle.test.ts +277 -0
  72. package/dist/resources/extensions/gsd/tests/md-importer.test.ts +411 -0
  73. package/dist/resources/extensions/gsd/tests/metrics.test.ts +197 -0
  74. package/dist/resources/extensions/gsd/tests/milestone-transition-worktree.test.ts +144 -0
  75. package/dist/resources/extensions/gsd/tests/model-cost-table.test.ts +69 -0
  76. package/dist/resources/extensions/gsd/tests/model-isolation.test.ts +99 -0
  77. package/dist/resources/extensions/gsd/tests/model-router.test.ts +167 -0
  78. package/dist/resources/extensions/gsd/tests/parsers.test.ts +40 -0
  79. package/dist/resources/extensions/gsd/tests/post-unit-hooks.test.ts +41 -1
  80. package/dist/resources/extensions/gsd/tests/preferences-git.test.ts +0 -1
  81. package/dist/resources/extensions/gsd/tests/preferences-hooks.test.ts +0 -1
  82. package/dist/resources/extensions/gsd/tests/preferences-mode.test.ts +110 -0
  83. package/dist/resources/extensions/gsd/tests/preferences-models.test.ts +0 -1
  84. package/dist/resources/extensions/gsd/tests/prompt-budget-enforcement.test.ts +464 -0
  85. package/dist/resources/extensions/gsd/tests/prompt-db.test.ts +385 -0
  86. package/dist/resources/extensions/gsd/tests/remote-questions.test.ts +488 -1
  87. package/dist/resources/extensions/gsd/tests/resolve-ts-hooks.mjs +17 -29
  88. package/dist/resources/extensions/gsd/tests/resolve-ts.mjs +2 -8
  89. package/dist/resources/extensions/gsd/tests/routing-history.test.ts +215 -62
  90. package/dist/resources/extensions/gsd/tests/skill-lifecycle.test.ts +126 -0
  91. package/dist/resources/extensions/gsd/tests/stop-auto-remote.test.ts +31 -8
  92. package/dist/resources/extensions/gsd/tests/token-savings.test.ts +366 -0
  93. package/dist/resources/extensions/gsd/tests/triage-dispatch.test.ts +224 -0
  94. package/dist/resources/extensions/gsd/tests/triage-resolution.test.ts +215 -0
  95. package/dist/resources/extensions/gsd/tests/unit-runtime.test.ts +25 -1
  96. package/dist/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +145 -0
  97. package/dist/resources/extensions/gsd/tests/visualizer-data.test.ts +290 -0
  98. package/dist/resources/extensions/gsd/tests/visualizer-overlay.test.ts +120 -0
  99. package/dist/resources/extensions/gsd/tests/visualizer-views.test.ts +478 -0
  100. package/dist/resources/extensions/gsd/tests/worktree-db-integration.test.ts +205 -0
  101. package/dist/resources/extensions/gsd/tests/worktree-db.test.ts +442 -0
  102. package/dist/resources/extensions/gsd/tests/worktree-post-create-hook.test.ts +165 -0
  103. package/dist/resources/extensions/gsd/triage-resolution.ts +200 -0
  104. package/dist/resources/extensions/gsd/triage-ui.ts +175 -0
  105. package/dist/resources/extensions/gsd/types.ts +29 -0
  106. package/dist/resources/extensions/gsd/undo.ts +0 -1
  107. package/dist/resources/extensions/gsd/unit-runtime.ts +5 -1
  108. package/dist/resources/extensions/gsd/visualizer-data.ts +505 -0
  109. package/dist/resources/extensions/gsd/visualizer-overlay.ts +337 -0
  110. package/dist/resources/extensions/gsd/visualizer-views.ts +755 -0
  111. package/dist/resources/extensions/gsd/worktree-command.ts +18 -0
  112. package/dist/resources/extensions/gsd/worktree-manager.ts +11 -4
  113. package/dist/resources/extensions/remote-questions/config.ts +4 -2
  114. package/dist/resources/extensions/remote-questions/discord-adapter.ts +35 -4
  115. package/dist/resources/extensions/remote-questions/format.ts +166 -14
  116. package/dist/resources/extensions/remote-questions/manager.ts +14 -4
  117. package/dist/resources/extensions/remote-questions/remote-command.ts +100 -4
  118. package/dist/resources/extensions/remote-questions/slack-adapter.ts +58 -2
  119. package/dist/resources/extensions/remote-questions/telegram-adapter.ts +161 -0
  120. package/dist/resources/extensions/remote-questions/types.ts +2 -1
  121. package/dist/resources/extensions/ttsr/ttsr-manager.ts +26 -0
  122. package/dist/resources/extensions/voice/index.ts +4 -3
  123. package/package.json +1 -1
  124. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  125. package/packages/pi-coding-agent/dist/core/agent-session.js +12 -1
  126. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  127. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  128. package/packages/pi-coding-agent/dist/core/extensions/loader.js +5 -0
  129. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  130. package/packages/pi-coding-agent/dist/core/lsp/client.d.ts +6 -0
  131. package/packages/pi-coding-agent/dist/core/lsp/client.d.ts.map +1 -1
  132. package/packages/pi-coding-agent/dist/core/lsp/client.js +25 -0
  133. package/packages/pi-coding-agent/dist/core/lsp/client.js.map +1 -1
  134. package/packages/pi-coding-agent/dist/core/lsp/index.d.ts +2 -0
  135. package/packages/pi-coding-agent/dist/core/lsp/index.d.ts.map +1 -1
  136. package/packages/pi-coding-agent/dist/core/lsp/index.js +106 -3
  137. package/packages/pi-coding-agent/dist/core/lsp/index.js.map +1 -1
  138. package/packages/pi-coding-agent/dist/core/lsp/lsp.md +6 -0
  139. package/packages/pi-coding-agent/dist/core/lsp/types.d.ts +35 -0
  140. package/packages/pi-coding-agent/dist/core/lsp/types.d.ts.map +1 -1
  141. package/packages/pi-coding-agent/dist/core/lsp/types.js +6 -0
  142. package/packages/pi-coding-agent/dist/core/lsp/types.js.map +1 -1
  143. package/packages/pi-coding-agent/dist/core/lsp/utils.d.ts +3 -1
  144. package/packages/pi-coding-agent/dist/core/lsp/utils.d.ts.map +1 -1
  145. package/packages/pi-coding-agent/dist/core/lsp/utils.js +45 -0
  146. package/packages/pi-coding-agent/dist/core/lsp/utils.js.map +1 -1
  147. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +6 -0
  148. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
  149. package/packages/pi-coding-agent/dist/core/settings-manager.js +43 -11
  150. package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
  151. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  152. package/packages/pi-coding-agent/dist/core/system-prompt.js +7 -1
  153. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  154. package/packages/pi-coding-agent/dist/core/tools/edit.d.ts.map +1 -1
  155. package/packages/pi-coding-agent/dist/core/tools/edit.js +5 -0
  156. package/packages/pi-coding-agent/dist/core/tools/edit.js.map +1 -1
  157. package/packages/pi-coding-agent/dist/core/tools/index.d.ts +2 -0
  158. package/packages/pi-coding-agent/dist/core/tools/index.d.ts.map +1 -1
  159. package/packages/pi-coding-agent/dist/core/tools/write.d.ts.map +1 -1
  160. package/packages/pi-coding-agent/dist/core/tools/write.js +5 -0
  161. package/packages/pi-coding-agent/dist/core/tools/write.js.map +1 -1
  162. package/packages/pi-coding-agent/src/core/agent-session.ts +13 -1
  163. package/packages/pi-coding-agent/src/core/extensions/loader.ts +6 -0
  164. package/packages/pi-coding-agent/src/core/lsp/client.ts +26 -0
  165. package/packages/pi-coding-agent/src/core/lsp/index.ts +157 -2
  166. package/packages/pi-coding-agent/src/core/lsp/lsp.md +6 -0
  167. package/packages/pi-coding-agent/src/core/lsp/types.ts +53 -0
  168. package/packages/pi-coding-agent/src/core/lsp/utils.ts +56 -0
  169. package/packages/pi-coding-agent/src/core/settings-manager.ts +41 -11
  170. package/packages/pi-coding-agent/src/core/system-prompt.ts +7 -1
  171. package/packages/pi-coding-agent/src/core/tools/edit.ts +3 -0
  172. package/packages/pi-coding-agent/src/core/tools/write.ts +3 -0
  173. package/src/resources/extensions/google-search/index.ts +164 -47
  174. package/src/resources/extensions/gsd/auto-dashboard.ts +14 -2
  175. package/src/resources/extensions/gsd/auto-prompts.ts +148 -39
  176. package/src/resources/extensions/gsd/auto-worktree.ts +93 -9
  177. package/src/resources/extensions/gsd/auto.ts +690 -39
  178. package/src/resources/extensions/gsd/captures.ts +384 -0
  179. package/src/resources/extensions/gsd/commands.ts +654 -36
  180. package/src/resources/extensions/gsd/complexity-classifier.ts +322 -0
  181. package/src/resources/extensions/gsd/context-budget.ts +243 -0
  182. package/src/resources/extensions/gsd/context-store.ts +195 -0
  183. package/src/resources/extensions/gsd/dashboard-overlay.ts +51 -3
  184. package/src/resources/extensions/gsd/db-writer.ts +341 -0
  185. package/src/resources/extensions/gsd/debug-logger.ts +178 -0
  186. package/src/resources/extensions/gsd/dispatch-guard.ts +0 -1
  187. package/src/resources/extensions/gsd/docs/preferences-reference.md +54 -0
  188. package/src/resources/extensions/gsd/doctor-proactive.ts +286 -0
  189. package/src/resources/extensions/gsd/doctor.ts +283 -2
  190. package/src/resources/extensions/gsd/export.ts +81 -2
  191. package/src/resources/extensions/gsd/files.ts +39 -9
  192. package/src/resources/extensions/gsd/git-service.ts +6 -0
  193. package/src/resources/extensions/gsd/gsd-db.ts +752 -0
  194. package/src/resources/extensions/gsd/guided-flow.ts +26 -1
  195. package/src/resources/extensions/gsd/history.ts +0 -1
  196. package/src/resources/extensions/gsd/index.ts +277 -1
  197. package/src/resources/extensions/gsd/md-importer.ts +526 -0
  198. package/src/resources/extensions/gsd/metrics.ts +84 -0
  199. package/src/resources/extensions/gsd/model-cost-table.ts +65 -0
  200. package/src/resources/extensions/gsd/model-router.ts +256 -0
  201. package/src/resources/extensions/gsd/notifications.ts +0 -1
  202. package/src/resources/extensions/gsd/post-unit-hooks.ts +72 -2
  203. package/src/resources/extensions/gsd/preferences.ts +198 -150
  204. package/src/resources/extensions/gsd/prompt-loader.ts +45 -9
  205. package/src/resources/extensions/gsd/prompts/execute-task.md +3 -5
  206. package/src/resources/extensions/gsd/prompts/heal-skill.md +45 -0
  207. package/src/resources/extensions/gsd/prompts/plan-slice.md +5 -1
  208. package/src/resources/extensions/gsd/prompts/quick-task.md +48 -0
  209. package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +6 -0
  210. package/src/resources/extensions/gsd/prompts/replan-slice.md +8 -0
  211. package/src/resources/extensions/gsd/prompts/system.md +2 -1
  212. package/src/resources/extensions/gsd/prompts/triage-captures.md +62 -0
  213. package/src/resources/extensions/gsd/quick.ts +156 -0
  214. package/src/resources/extensions/gsd/skill-discovery.ts +5 -3
  215. package/src/resources/extensions/gsd/skill-health.ts +417 -0
  216. package/src/resources/extensions/gsd/skill-telemetry.ts +127 -0
  217. package/src/resources/extensions/gsd/state.ts +30 -0
  218. package/src/resources/extensions/gsd/templates/preferences.md +1 -0
  219. package/src/resources/extensions/gsd/tests/captures.test.ts +438 -0
  220. package/src/resources/extensions/gsd/tests/complexity-classifier.test.ts +181 -0
  221. package/src/resources/extensions/gsd/tests/context-budget.test.ts +283 -0
  222. package/src/resources/extensions/gsd/tests/context-compression.test.ts +1 -1
  223. package/src/resources/extensions/gsd/tests/context-store.test.ts +462 -0
  224. package/src/resources/extensions/gsd/tests/continue-here.test.ts +204 -0
  225. package/src/resources/extensions/gsd/tests/dashboard-budget.test.ts +346 -0
  226. package/src/resources/extensions/gsd/tests/db-writer.test.ts +602 -0
  227. package/src/resources/extensions/gsd/tests/debug-logger.test.ts +185 -0
  228. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +406 -0
  229. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +0 -1
  230. package/src/resources/extensions/gsd/tests/dist-redirect.mjs +22 -0
  231. package/src/resources/extensions/gsd/tests/doctor-proactive.test.ts +244 -0
  232. package/src/resources/extensions/gsd/tests/doctor-runtime.test.ts +303 -0
  233. package/src/resources/extensions/gsd/tests/feature-branch-lifecycle-integration.test.ts +434 -0
  234. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +353 -0
  235. package/src/resources/extensions/gsd/tests/gsd-inspect.test.ts +125 -0
  236. package/src/resources/extensions/gsd/tests/gsd-tools.test.ts +326 -0
  237. package/src/resources/extensions/gsd/tests/integration-edge.test.ts +228 -0
  238. package/src/resources/extensions/gsd/tests/integration-lifecycle.test.ts +277 -0
  239. package/src/resources/extensions/gsd/tests/md-importer.test.ts +411 -0
  240. package/src/resources/extensions/gsd/tests/metrics.test.ts +197 -0
  241. package/src/resources/extensions/gsd/tests/milestone-transition-worktree.test.ts +144 -0
  242. package/src/resources/extensions/gsd/tests/model-cost-table.test.ts +69 -0
  243. package/src/resources/extensions/gsd/tests/model-isolation.test.ts +99 -0
  244. package/src/resources/extensions/gsd/tests/model-router.test.ts +167 -0
  245. package/src/resources/extensions/gsd/tests/parsers.test.ts +40 -0
  246. package/src/resources/extensions/gsd/tests/post-unit-hooks.test.ts +41 -1
  247. package/src/resources/extensions/gsd/tests/preferences-git.test.ts +0 -1
  248. package/src/resources/extensions/gsd/tests/preferences-hooks.test.ts +0 -1
  249. package/src/resources/extensions/gsd/tests/preferences-mode.test.ts +110 -0
  250. package/src/resources/extensions/gsd/tests/preferences-models.test.ts +0 -1
  251. package/src/resources/extensions/gsd/tests/prompt-budget-enforcement.test.ts +464 -0
  252. package/src/resources/extensions/gsd/tests/prompt-db.test.ts +385 -0
  253. package/src/resources/extensions/gsd/tests/remote-questions.test.ts +488 -1
  254. package/src/resources/extensions/gsd/tests/resolve-ts-hooks.mjs +17 -29
  255. package/src/resources/extensions/gsd/tests/resolve-ts.mjs +2 -8
  256. package/src/resources/extensions/gsd/tests/routing-history.test.ts +215 -62
  257. package/src/resources/extensions/gsd/tests/skill-lifecycle.test.ts +126 -0
  258. package/src/resources/extensions/gsd/tests/stop-auto-remote.test.ts +31 -8
  259. package/src/resources/extensions/gsd/tests/token-savings.test.ts +366 -0
  260. package/src/resources/extensions/gsd/tests/triage-dispatch.test.ts +224 -0
  261. package/src/resources/extensions/gsd/tests/triage-resolution.test.ts +215 -0
  262. package/src/resources/extensions/gsd/tests/unit-runtime.test.ts +25 -1
  263. package/src/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +145 -0
  264. package/src/resources/extensions/gsd/tests/visualizer-data.test.ts +290 -0
  265. package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +120 -0
  266. package/src/resources/extensions/gsd/tests/visualizer-views.test.ts +478 -0
  267. package/src/resources/extensions/gsd/tests/worktree-db-integration.test.ts +205 -0
  268. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +442 -0
  269. package/src/resources/extensions/gsd/tests/worktree-post-create-hook.test.ts +165 -0
  270. package/src/resources/extensions/gsd/triage-resolution.ts +200 -0
  271. package/src/resources/extensions/gsd/triage-ui.ts +175 -0
  272. package/src/resources/extensions/gsd/types.ts +29 -0
  273. package/src/resources/extensions/gsd/undo.ts +0 -1
  274. package/src/resources/extensions/gsd/unit-runtime.ts +5 -1
  275. package/src/resources/extensions/gsd/visualizer-data.ts +505 -0
  276. package/src/resources/extensions/gsd/visualizer-overlay.ts +337 -0
  277. package/src/resources/extensions/gsd/visualizer-views.ts +755 -0
  278. package/src/resources/extensions/gsd/worktree-command.ts +18 -0
  279. package/src/resources/extensions/gsd/worktree-manager.ts +11 -4
  280. package/src/resources/extensions/remote-questions/config.ts +4 -2
  281. package/src/resources/extensions/remote-questions/discord-adapter.ts +35 -4
  282. package/src/resources/extensions/remote-questions/format.ts +166 -14
  283. package/src/resources/extensions/remote-questions/manager.ts +14 -4
  284. package/src/resources/extensions/remote-questions/remote-command.ts +100 -4
  285. package/src/resources/extensions/remote-questions/slack-adapter.ts +58 -2
  286. package/src/resources/extensions/remote-questions/telegram-adapter.ts +161 -0
  287. package/src/resources/extensions/remote-questions/types.ts +2 -1
  288. package/src/resources/extensions/ttsr/ttsr-manager.ts +26 -0
  289. package/src/resources/extensions/voice/index.ts +4 -3
@@ -0,0 +1,110 @@
1
+ // GSD Workflow Mode Tests — validates mode defaults, overrides, and validation
2
+
3
+ import { createTestContext } from "./test-helpers.ts";
4
+ import { validatePreferences, applyModeDefaults } from "../preferences.ts";
5
+ import type { GSDPreferences } from "../preferences.ts";
6
+
7
+ const { assertEq, assertTrue, report } = createTestContext();
8
+
9
+ async function main(): Promise<void> {
10
+ console.log("\n=== mode: solo defaults ===");
11
+
12
+ {
13
+ const prefs: GSDPreferences = { mode: "solo" };
14
+ const result = applyModeDefaults("solo", prefs);
15
+ assertEq(result.git?.auto_push, true, "solo — auto_push defaults to true");
16
+ assertEq(result.git?.push_branches, false, "solo — push_branches defaults to false");
17
+ assertEq(result.git?.pre_merge_check, false, "solo — pre_merge_check defaults to false");
18
+ assertEq(result.git?.merge_strategy, "squash", "solo — merge_strategy defaults to squash");
19
+ assertEq(result.git?.isolation, "worktree", "solo — isolation defaults to worktree");
20
+ assertEq(result.git?.commit_docs, true, "solo — commit_docs defaults to true");
21
+ assertEq(result.unique_milestone_ids, false, "solo — unique_milestone_ids defaults to false");
22
+ }
23
+
24
+ console.log("\n=== mode: team defaults ===");
25
+
26
+ {
27
+ const prefs: GSDPreferences = { mode: "team" };
28
+ const result = applyModeDefaults("team", prefs);
29
+ assertEq(result.git?.auto_push, false, "team — auto_push defaults to false");
30
+ assertEq(result.git?.push_branches, true, "team — push_branches defaults to true");
31
+ assertEq(result.git?.pre_merge_check, true, "team — pre_merge_check defaults to true");
32
+ assertEq(result.git?.merge_strategy, "squash", "team — merge_strategy defaults to squash");
33
+ assertEq(result.git?.isolation, "worktree", "team — isolation defaults to worktree");
34
+ assertEq(result.git?.commit_docs, true, "team — commit_docs defaults to true");
35
+ assertEq(result.unique_milestone_ids, true, "team — unique_milestone_ids defaults to true");
36
+ }
37
+
38
+ console.log("\n=== explicit override wins over mode default ===");
39
+
40
+ {
41
+ const prefs: GSDPreferences = {
42
+ mode: "solo",
43
+ git: { auto_push: false },
44
+ };
45
+ const result = applyModeDefaults("solo", prefs);
46
+ assertEq(result.git?.auto_push, false, "solo + explicit auto_push=false — override wins");
47
+ assertEq(result.git?.push_branches, false, "solo + override — other defaults still apply");
48
+ assertEq(result.git?.merge_strategy, "squash", "solo + override — merge_strategy still defaults");
49
+ }
50
+
51
+ console.log("\n=== no mode set — no defaults injected ===");
52
+
53
+ {
54
+ const prefs: GSDPreferences = { git: { auto_push: true } };
55
+ const { preferences } = validatePreferences(prefs);
56
+ assertEq(preferences.mode, undefined, "no mode — mode is undefined");
57
+ assertEq(preferences.git?.push_branches, undefined, "no mode — push_branches not injected");
58
+ assertEq(preferences.unique_milestone_ids, undefined, "no mode — unique_milestone_ids not injected");
59
+ }
60
+
61
+ console.log("\n=== invalid mode value → validation error ===");
62
+
63
+ {
64
+ const { errors } = validatePreferences({ mode: "invalid" as any });
65
+ assertTrue(errors.length > 0, "invalid mode — produces error");
66
+ assertTrue(errors[0].includes("solo, team"), "invalid mode — error mentions valid values");
67
+ }
68
+
69
+ console.log("\n=== valid mode values pass validation ===");
70
+
71
+ {
72
+ const { errors: soloErrors, preferences: soloPrefs } = validatePreferences({ mode: "solo" });
73
+ assertEq(soloErrors.length, 0, "mode: solo — no errors");
74
+ assertEq(soloPrefs.mode, "solo", "mode: solo — value preserved");
75
+ }
76
+ {
77
+ const { errors: teamErrors, preferences: teamPrefs } = validatePreferences({ mode: "team" });
78
+ assertEq(teamErrors.length, 0, "mode: team — no errors");
79
+ assertEq(teamPrefs.mode, "team", "mode: team — value preserved");
80
+ }
81
+
82
+ console.log("\n=== deep merge: mode + explicit git.remote ===");
83
+
84
+ {
85
+ const prefs: GSDPreferences = {
86
+ mode: "team",
87
+ git: { remote: "upstream" },
88
+ };
89
+ const result = applyModeDefaults("team", prefs);
90
+ assertEq(result.git?.remote, "upstream", "team + git.remote — custom remote preserved");
91
+ assertEq(result.git?.auto_push, false, "team + git.remote — team auto_push default applied");
92
+ assertEq(result.git?.push_branches, true, "team + git.remote — team push_branches default applied");
93
+ }
94
+
95
+ console.log("\n=== mode + unique_milestone_ids explicit override ===");
96
+
97
+ {
98
+ const prefs: GSDPreferences = {
99
+ mode: "team",
100
+ unique_milestone_ids: false,
101
+ };
102
+ const result = applyModeDefaults("team", prefs);
103
+ assertEq(result.unique_milestone_ids, false, "team + explicit unique_milestone_ids=false — override wins");
104
+ assertEq(result.git?.push_branches, true, "team + override — other team defaults still apply");
105
+ }
106
+
107
+ report();
108
+ }
109
+
110
+ main();
@@ -1,5 +1,4 @@
1
1
  // GSD Extension — Model Preferences Parsing Tests
2
- // Copyright (c) 2026 Jeremy McSpadden <jeremy@fluxlabs.net>
3
2
 
4
3
  import test from "node:test";
5
4
  import assert from "node:assert/strict";
@@ -0,0 +1,464 @@
1
+ /**
2
+ * Prompt budget enforcement tests — verifies that budget-aware prompt builders
3
+ * truncate content at section boundaries and that plan-slice includes executor
4
+ * context constraints.
5
+ *
6
+ * Tests:
7
+ * 1. inlineDependencySummaries() truncates when budget is small, passes through when large
8
+ * 2. plan-slice.md template includes {{executorContextConstraints}} placeholder
9
+ * 3. Executor constraints formatting varies with context window size
10
+ * 4. Different context windows produce different budget-constrained outputs
11
+ */
12
+
13
+ import { describe, it, beforeEach, afterEach } from "node:test";
14
+ import assert from "node:assert/strict";
15
+ import { mkdtempSync, mkdirSync, rmSync, writeFileSync, readFileSync } from "node:fs";
16
+ import { join, dirname } from "node:path";
17
+ import { tmpdir } from "node:os";
18
+ import { fileURLToPath } from "node:url";
19
+
20
+ import { inlineDependencySummaries } from "../auto-prompts.js";
21
+ import { computeBudgets, truncateAtSectionBoundary } from "../context-budget.js";
22
+
23
+ const __dirname = dirname(fileURLToPath(import.meta.url));
24
+
25
+ // ─── Fixture helpers ──────────────────────────────────────────────────────────
26
+
27
+ function createFixtureBase(): string {
28
+ return mkdtempSync(join(tmpdir(), "gsd-prompt-budget-test-"));
29
+ }
30
+
31
+ function cleanup(base: string): void {
32
+ rmSync(base, { recursive: true, force: true });
33
+ }
34
+
35
+ /**
36
+ * Set up a minimal milestone with a roadmap declaring slice dependencies and
37
+ * dependency slice summaries on disk.
38
+ */
39
+ function setupDependencyFixture(
40
+ base: string,
41
+ mid: string,
42
+ sid: string,
43
+ deps: string[],
44
+ summaries: Record<string, string>,
45
+ ): void {
46
+ const msDir = join(base, ".gsd", "milestones", mid);
47
+ mkdirSync(msDir, { recursive: true });
48
+
49
+ // Build roadmap content — sid depends on deps
50
+ const depStr = deps.join(", ");
51
+ const sliceLines = [
52
+ `- [x] **${deps[0]}: Done dep** \`risk:low\` \`depends:[]\``,
53
+ `- [ ] **${sid}: Current slice** \`risk:medium\` \`depends:[${depStr}]\``,
54
+ ];
55
+ // Add any extra deps as completed slices
56
+ for (let i = 1; i < deps.length; i++) {
57
+ sliceLines.unshift(`- [x] **${deps[i]}: Another dep** \`risk:low\` \`depends:[]\``);
58
+ }
59
+ const roadmapContent = [
60
+ "# Roadmap",
61
+ "",
62
+ "## Slices",
63
+ "",
64
+ ...sliceLines,
65
+ ].join("\n");
66
+ writeFileSync(join(msDir, `${mid}-ROADMAP.md`), roadmapContent);
67
+
68
+ // Write dependency slice summaries
69
+ for (const [depId, content] of Object.entries(summaries)) {
70
+ const sliceDir = join(msDir, "slices", depId);
71
+ mkdirSync(sliceDir, { recursive: true });
72
+ writeFileSync(join(sliceDir, `${depId}-SUMMARY.md`), content);
73
+ }
74
+
75
+ // Ensure target slice dir exists
76
+ const targetSliceDir = join(msDir, "slices", sid);
77
+ mkdirSync(targetSliceDir, { recursive: true });
78
+ }
79
+
80
+ // ─── inlineDependencySummaries truncation ─────────────────────────────────────
81
+
82
+ describe("prompt-budget: inlineDependencySummaries truncation", () => {
83
+ let base: string;
84
+
85
+ beforeEach(() => {
86
+ base = createFixtureBase();
87
+ });
88
+
89
+ afterEach(() => {
90
+ cleanup(base);
91
+ });
92
+
93
+ it("passes through all content when budget is larger than total", async () => {
94
+ const summaryContent = "### Results\n\nEverything works.\n\n### Forward Intelligence\n\nWatch out for X.";
95
+ setupDependencyFixture(base, "M001", "S02", ["S01"], {
96
+ S01: summaryContent,
97
+ });
98
+
99
+ const result = await inlineDependencySummaries("M001", "S02", base, 100_000);
100
+ assert.ok(result.includes("Everything works."), "should include full summary content");
101
+ assert.ok(result.includes("Watch out for X."), "should include forward intelligence");
102
+ assert.ok(!result.includes("[...truncated"), "should not have truncation marker");
103
+ });
104
+
105
+ it("truncates at section boundaries when budget is small", async () => {
106
+ // Create a large summary with multiple sections
107
+ const sections = [];
108
+ for (let i = 0; i < 10; i++) {
109
+ sections.push(`### Section ${i}\n\n${"Lorem ipsum dolor sit amet. ".repeat(50)}`);
110
+ }
111
+ const largeSummary = sections.join("\n\n");
112
+
113
+ setupDependencyFixture(base, "M001", "S02", ["S01"], {
114
+ S01: largeSummary,
115
+ });
116
+
117
+ // Use a budget smaller than total content
118
+ const result = await inlineDependencySummaries("M001", "S02", base, 500);
119
+ assert.ok(result.includes("[...truncated"), "should have truncation marker when over budget");
120
+ assert.ok(result.length <= 600, `result should be near budget limit, got ${result.length}`);
121
+ });
122
+
123
+ it("returns content unchanged when no budget is provided (backward compat)", async () => {
124
+ const sections = [];
125
+ for (let i = 0; i < 5; i++) {
126
+ sections.push(`### Section ${i}\n\n${"Content block. ".repeat(30)}`);
127
+ }
128
+ const largeSummary = sections.join("\n\n");
129
+
130
+ setupDependencyFixture(base, "M001", "S02", ["S01"], {
131
+ S01: largeSummary,
132
+ });
133
+
134
+ // No budget parameter — backward-compatible behavior
135
+ const result = await inlineDependencySummaries("M001", "S02", base);
136
+ assert.ok(!result.includes("[...truncated"), "should not truncate without budget");
137
+ assert.ok(result.includes("Section 4"), "should include all sections");
138
+ });
139
+
140
+ it("handles multiple dependency summaries with truncation", async () => {
141
+ const summary1 = "### S01 Results\n\nFirst dep done.\n\n### S01 Notes\n\nSome notes.";
142
+ const summary2 = "### S02 Results\n\nSecond dep done.\n\n### S02 Notes\n\nMore notes.";
143
+ setupDependencyFixture(base, "M001", "S03", ["S01", "S02"], {
144
+ S01: summary1,
145
+ S02: summary2,
146
+ });
147
+
148
+ // Budget large enough for all content
149
+ const fullResult = await inlineDependencySummaries("M001", "S03", base, 100_000);
150
+ assert.ok(fullResult.includes("First dep done."), "should have S01 content");
151
+ assert.ok(fullResult.includes("Second dep done."), "should have S02 content");
152
+
153
+ // Budget too small for all
154
+ const truncResult = await inlineDependencySummaries("M001", "S03", base, 200);
155
+ assert.ok(truncResult.includes("[...truncated"), "should truncate when budget is small");
156
+ });
157
+
158
+ it("returns no-dependencies marker when slice has no deps", async () => {
159
+ const msDir = join(base, ".gsd", "milestones", "M001");
160
+ mkdirSync(msDir, { recursive: true });
161
+ const roadmap = "# Roadmap\n\n## Slices\n\n- [ ] **S01: Solo** `risk:low` `depends:[]`\n";
162
+ writeFileSync(join(msDir, "M001-ROADMAP.md"), roadmap);
163
+
164
+ const result = await inlineDependencySummaries("M001", "S01", base, 1000);
165
+ assert.equal(result, "- (no dependencies)");
166
+ });
167
+ });
168
+
169
+ // ─── plan-slice template includes executor constraints placeholder ────────────
170
+
171
+ describe("prompt-budget: plan-slice template", () => {
172
+ it("contains {{executorContextConstraints}} placeholder", () => {
173
+ const templatePath = join(__dirname, "..", "prompts", "plan-slice.md");
174
+ const template = readFileSync(templatePath, "utf-8");
175
+ assert.ok(
176
+ template.includes("{{executorContextConstraints}}"),
177
+ "plan-slice.md should contain {{executorContextConstraints}} placeholder",
178
+ );
179
+ });
180
+ });
181
+
182
+ // ─── Executor constraints formatting ──────────────────────────────────────────
183
+
184
+ describe("prompt-budget: executor constraints formatting", () => {
185
+ it("128K window produces different constraints than 1M window", () => {
186
+ const budget128K = computeBudgets(128_000);
187
+ const budget1M = computeBudgets(1_000_000);
188
+
189
+ // Task count ranges should differ
190
+ assert.notEqual(
191
+ budget128K.taskCountRange.max,
192
+ budget1M.taskCountRange.max,
193
+ "128K and 1M should have different max task counts",
194
+ );
195
+
196
+ // Inline context budgets should differ
197
+ assert.ok(
198
+ budget1M.inlineContextBudgetChars > budget128K.inlineContextBudgetChars,
199
+ "1M should have larger inline context budget than 128K",
200
+ );
201
+
202
+ // Format constraint blocks and verify they differ
203
+ const format = (b: ReturnType<typeof computeBudgets>, windowTokens: number) => {
204
+ const { min, max } = b.taskCountRange;
205
+ const execWindowK = Math.round(windowTokens / 1000);
206
+ const perTaskBudgetK = Math.round(b.inlineContextBudgetChars / 1000);
207
+ return [
208
+ `## Executor Context Constraints`,
209
+ ``,
210
+ `The agent that executes each task has a **${execWindowK}K token** context window.`,
211
+ `- Recommended task count for this slice: **${min}–${max} tasks**`,
212
+ `- Each task gets ~${perTaskBudgetK}K chars of inline context (plans, code, decisions)`,
213
+ `- Keep individual tasks completable within a single context window — if a task needs more context than fits, split it`,
214
+ ].join("\n");
215
+ };
216
+
217
+ const constraints128K = format(budget128K, 128_000);
218
+ const constraints1M = format(budget1M, 1_000_000);
219
+
220
+ assert.ok(constraints128K.includes("128K token"), "128K constraints should reference 128K");
221
+ assert.ok(constraints1M.includes("1000K token"), "1M constraints should reference 1000K");
222
+ assert.ok(constraints128K.includes("2–5 tasks"), "128K should recommend 2–5 tasks");
223
+ assert.ok(constraints1M.includes("2–8 tasks"), "1M should recommend 2–8 tasks");
224
+ assert.notEqual(constraints128K, constraints1M, "constraint blocks should differ");
225
+ });
226
+
227
+ it("undefined context window falls back to 200K defaults", () => {
228
+ // computeBudgets(0) defaults to 200K (D002)
229
+ const budgetDefault = computeBudgets(0);
230
+ const budget200K = computeBudgets(200_000);
231
+
232
+ assert.equal(budgetDefault.summaryBudgetChars, budget200K.summaryBudgetChars);
233
+ assert.equal(budgetDefault.inlineContextBudgetChars, budget200K.inlineContextBudgetChars);
234
+ assert.equal(budgetDefault.taskCountRange.max, budget200K.taskCountRange.max);
235
+ });
236
+ });
237
+
238
+ // ─── Budget-constrained output varies with context window ─────────────────────
239
+
240
+ describe("prompt-budget: different context windows produce different outputs", () => {
241
+ it("small window truncates content that large window preserves", () => {
242
+ // Simulate assembled inlinedContext with multiple sections
243
+ const sections = [];
244
+ for (let i = 0; i < 20; i++) {
245
+ sections.push(`### Section ${i}: Important Context\n\n${"Detailed content for this section. ".repeat(100)}`);
246
+ }
247
+ const largeContent = `## Inlined Context\n\n${sections.join("\n\n---\n\n")}`;
248
+
249
+ // 128K context window budget
250
+ const budget128K = computeBudgets(128_000);
251
+ const r128K = truncateAtSectionBoundary(largeContent, budget128K.inlineContextBudgetChars);
252
+
253
+ // 1M context window budget
254
+ const budget1M = computeBudgets(1_000_000);
255
+ const r1M = truncateAtSectionBoundary(largeContent, budget1M.inlineContextBudgetChars);
256
+
257
+ // The large content (~70K chars) should fit in 1M budget (~1.6M chars) but
258
+ // if we make content bigger, the 128K budget (~204K chars) would truncate
259
+ assert.ok(
260
+ r128K.content.length <= budget128K.inlineContextBudgetChars + 100, // +100 for truncation marker
261
+ "128K result should respect budget",
262
+ );
263
+ assert.ok(
264
+ r1M.content.length <= budget1M.inlineContextBudgetChars + 100,
265
+ "1M result should respect budget",
266
+ );
267
+
268
+ // With content smaller than both budgets, both should pass through unchanged
269
+ const smallContent = "### One Section\n\nSmall content.";
270
+ const small128K = truncateAtSectionBoundary(smallContent, budget128K.inlineContextBudgetChars);
271
+ const small1M = truncateAtSectionBoundary(smallContent, budget1M.inlineContextBudgetChars);
272
+ assert.equal(small128K.content, smallContent, "small content unchanged for 128K");
273
+ assert.equal(small128K.droppedSections, 0);
274
+ assert.equal(small1M.content, smallContent, "small content unchanged for 1M");
275
+ assert.equal(small1M.droppedSections, 0);
276
+ });
277
+
278
+ it("128K budget truncates very large content while 1M preserves it", () => {
279
+ // Create content that exceeds 128K budget (~204K chars) but fits in 1M (~1.6M chars)
280
+ const sections = [];
281
+ for (let i = 0; i < 100; i++) {
282
+ sections.push(`### Section ${i}\n\n${"X".repeat(3000)}`);
283
+ }
284
+ const content = sections.join("\n\n");
285
+ // ~310K chars total
286
+
287
+ const budget128K = computeBudgets(128_000);
288
+ const result128K = truncateAtSectionBoundary(content, budget128K.inlineContextBudgetChars);
289
+
290
+ const budget1M = computeBudgets(1_000_000);
291
+ const result1M = truncateAtSectionBoundary(content, budget1M.inlineContextBudgetChars);
292
+
293
+ assert.ok(result128K.content.includes("[...truncated"), "128K should truncate ~310K content");
294
+ assert.ok(result128K.droppedSections > 0, "128K should report dropped sections");
295
+ assert.ok(!result1M.content.includes("[...truncated"), "1M should preserve ~310K content");
296
+ assert.equal(result1M.droppedSections, 0);
297
+ assert.ok(result128K.content.length < result1M.content.length, "128K result should be shorter than 1M result");
298
+ });
299
+ });
300
+
301
+ // ─── execute-task template includes verificationBudget placeholder ─────────
302
+
303
+ describe("prompt-budget: execute-task template", () => {
304
+ it("contains {{verificationBudget}} placeholder", () => {
305
+ const templatePath = join(__dirname, "..", "prompts", "execute-task.md");
306
+ const template = readFileSync(templatePath, "utf-8");
307
+ assert.ok(
308
+ template.includes("{{verificationBudget}}"),
309
+ "execute-task.md should contain {{verificationBudget}} placeholder",
310
+ );
311
+ });
312
+
313
+ it("verificationBudget format varies with context window size", () => {
314
+ const budget128K = computeBudgets(128_000);
315
+ const budget1M = computeBudgets(1_000_000);
316
+
317
+ const format128K = `~${Math.round(budget128K.verificationBudgetChars / 1000)}K chars`;
318
+ const format1M = `~${Math.round(budget1M.verificationBudgetChars / 1000)}K chars`;
319
+
320
+ assert.notEqual(format128K, format1M, "128K and 1M should produce different verification budget strings");
321
+ assert.ok(format128K.includes("~51K"), `128K should produce ~51K, got ${format128K}`);
322
+ assert.ok(format1M.includes("~400K"), `1M should produce ~400K, got ${format1M}`);
323
+ });
324
+ });
325
+
326
+ // ─── buildCompleteSlicePrompt budget enforcement (simulated) ─────────────────
327
+
328
+ describe("prompt-budget: complete-slice builder truncation pattern", () => {
329
+ it("truncateAtSectionBoundary truncates assembled inlinedContext for complete-slice pattern", () => {
330
+ // Simulate buildCompleteSlicePrompt: roadmap + slice plan + task summaries
331
+ const inlined: string[] = [];
332
+ inlined.push("### Milestone Roadmap\n\nRoadmap content here.");
333
+ inlined.push("### Slice Plan\n\nSlice plan content here.");
334
+ // Add many task summaries that push past budget
335
+ for (let i = 0; i < 50; i++) {
336
+ inlined.push(`### Task Summary: T${String(i).padStart(2, "0")}\nSource: \`tasks/T${String(i).padStart(2, "0")}-SUMMARY.md\`\n\n${"Task result details. ".repeat(200)}`);
337
+ }
338
+
339
+ const assembledContent = `## Inlined Context (preloaded — do not re-read these files)\n\n${inlined.join("\n\n---\n\n")}`;
340
+
341
+ // Small context window (128K) should truncate
342
+ const budget128K = computeBudgets(128_000);
343
+ const result128K = truncateAtSectionBoundary(assembledContent, budget128K.inlineContextBudgetChars);
344
+ assert.ok(result128K.content.includes("[...truncated"), "128K should truncate many task summaries");
345
+ assert.ok(result128K.content.includes("### Milestone Roadmap"), "should preserve early sections");
346
+ assert.ok(result128K.droppedSections > 0, "128K should report dropped sections");
347
+
348
+ // Large context window (1M) should preserve all
349
+ const budget1M = computeBudgets(1_000_000);
350
+ const result1M = truncateAtSectionBoundary(assembledContent, budget1M.inlineContextBudgetChars);
351
+ assert.ok(!result1M.content.includes("[...truncated"), "1M should preserve all task summaries");
352
+ assert.equal(result1M.droppedSections, 0);
353
+ });
354
+
355
+ it("small content passes through unchanged at any context window size", () => {
356
+ const smallContent = "## Inlined Context\n\n### Roadmap\n\nSmall roadmap.\n\n---\n\n### Plan\n\nSmall plan.";
357
+
358
+ const budget128K = computeBudgets(128_000);
359
+ const result128K = truncateAtSectionBoundary(smallContent, budget128K.inlineContextBudgetChars);
360
+ assert.equal(result128K.content, smallContent, "small content unchanged for 128K");
361
+ assert.equal(result128K.droppedSections, 0);
362
+
363
+ const budget1M = computeBudgets(1_000_000);
364
+ const result1M = truncateAtSectionBoundary(smallContent, budget1M.inlineContextBudgetChars);
365
+ assert.equal(result1M.content, smallContent, "small content unchanged for 1M");
366
+ assert.equal(result1M.droppedSections, 0);
367
+ });
368
+ });
369
+
370
+ // ─── buildCompleteMilestonePrompt budget enforcement (simulated) ─────────────
371
+
372
+ describe("prompt-budget: complete-milestone builder truncation pattern", () => {
373
+ it("truncateAtSectionBoundary truncates assembled inlinedContext for complete-milestone pattern", () => {
374
+ // Simulate buildCompleteMilestonePrompt: roadmap + slice summaries + root files
375
+ const inlined: string[] = [];
376
+ inlined.push("### Milestone Roadmap\n\nRoadmap content here.");
377
+ // Add many slice summaries that push past budget
378
+ for (let i = 0; i < 30; i++) {
379
+ inlined.push(`### S${String(i).padStart(2, "0")} Summary\n\n${"Slice summary with detailed results and forward intelligence. ".repeat(200)}`);
380
+ }
381
+ inlined.push("### Requirements\n\nProject requirements.");
382
+ inlined.push("### Decisions\n\nProject decisions.");
383
+
384
+ const assembledContent = `## Inlined Context (preloaded — do not re-read these files)\n\n${inlined.join("\n\n---\n\n")}`;
385
+
386
+ // Small context window (128K) should truncate
387
+ const budget128K = computeBudgets(128_000);
388
+ const result128K = truncateAtSectionBoundary(assembledContent, budget128K.inlineContextBudgetChars);
389
+ assert.ok(result128K.content.includes("[...truncated"), "128K should truncate many slice summaries");
390
+ assert.ok(result128K.droppedSections > 0);
391
+
392
+ // Large context window (1M) should preserve all
393
+ const budget1M = computeBudgets(1_000_000);
394
+ const result1M = truncateAtSectionBoundary(assembledContent, budget1M.inlineContextBudgetChars);
395
+ assert.ok(!result1M.content.includes("[...truncated"), "1M should preserve all slice summaries");
396
+ assert.equal(result1M.droppedSections, 0);
397
+ });
398
+
399
+ it("different context windows produce different truncation for milestone completion", () => {
400
+ // Create content that exceeds 128K budget but not 200K budget
401
+ const inlined: string[] = [];
402
+ inlined.push("### Roadmap\n\nRoadmap.");
403
+ for (let i = 0; i < 15; i++) {
404
+ inlined.push(`### S${i} Summary\n\n${"X".repeat(15000)}`);
405
+ }
406
+ const content = `## Inlined Context\n\n${inlined.join("\n\n---\n\n")}`;
407
+ // ~225K chars total
408
+
409
+ const budget128K = computeBudgets(128_000);
410
+ const budget200K = computeBudgets(200_000);
411
+ const budget1M = computeBudgets(1_000_000);
412
+
413
+ const result128K = truncateAtSectionBoundary(content, budget128K.inlineContextBudgetChars);
414
+ const result200K = truncateAtSectionBoundary(content, budget200K.inlineContextBudgetChars);
415
+ const result1M = truncateAtSectionBoundary(content, budget1M.inlineContextBudgetChars);
416
+
417
+ // 128K (budget ~204K) should truncate ~225K content
418
+ assert.ok(result128K.content.includes("[...truncated"), "128K should truncate ~225K content");
419
+ assert.ok(result128K.droppedSections > 0);
420
+ // 200K (budget ~320K) should not truncate ~225K content
421
+ assert.ok(!result200K.content.includes("[...truncated"), "200K should preserve ~225K content");
422
+ assert.equal(result200K.droppedSections, 0);
423
+ // 1M should not truncate
424
+ assert.ok(!result1M.content.includes("[...truncated"), "1M should preserve ~225K content");
425
+ assert.equal(result1M.droppedSections, 0);
426
+ // 128K result should be shorter
427
+ assert.ok(result128K.content.length < result200K.content.length, "128K result should be shorter than 200K");
428
+ });
429
+ });
430
+
431
+ // ─── buildExecuteTaskPrompt budget enforcement (simulated) ───────────────────
432
+
433
+ describe("prompt-budget: execute-task builder truncation pattern", () => {
434
+ it("truncateAtSectionBoundary truncates assembled carry-forward + task plan + slice excerpt", () => {
435
+ // Simulate the assembled content from buildExecuteTaskPrompt
436
+ const carryForward = "## Carry-Forward Context\n" + Array.from({ length: 20 }, (_, i) =>
437
+ `- \`tasks/T${String(i).padStart(2, "0")}-SUMMARY.md\` — ${"Summary details. ".repeat(100)}`
438
+ ).join("\n");
439
+
440
+ const taskPlan = "## Inlined Task Plan\n\n" + Array.from({ length: 10 }, (_, i) =>
441
+ `### Step ${i}\n\n${"Implementation step details. ".repeat(200)}`
442
+ ).join("\n\n");
443
+
444
+ const sliceExcerpt = "## Slice Plan Excerpt\n\n" + "Slice goal and verification details. ".repeat(100);
445
+
446
+ const assembled = [carryForward, taskPlan, sliceExcerpt].join("\n\n---\n\n");
447
+
448
+ // Small context window should truncate
449
+ const budget128K = computeBudgets(128_000);
450
+ const result = truncateAtSectionBoundary(assembled, budget128K.inlineContextBudgetChars);
451
+
452
+ // Content should respect budget
453
+ assert.ok(
454
+ result.content.length <= budget128K.inlineContextBudgetChars + 100,
455
+ `result should respect 128K budget, got ${result.content.length} chars vs budget ${budget128K.inlineContextBudgetChars}`,
456
+ );
457
+
458
+ // Large content should be truncated
459
+ if (assembled.length > budget128K.inlineContextBudgetChars) {
460
+ assert.ok(result.content.includes("[...truncated"), "should truncate when content exceeds 128K budget");
461
+ assert.ok(result.droppedSections > 0, "should report dropped sections");
462
+ }
463
+ });
464
+ });