@opengsd/gsd-pi 1.3.0-dev.65546769 → 1.3.0-dev.eed73bea

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 (183) hide show
  1. package/dist/resources/.managed-resources-content-hash +1 -1
  2. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +11 -2
  3. package/dist/resources/extensions/google-cli/stream-adapter.js +82 -15
  4. package/dist/resources/extensions/gsd/auto/orchestrator.js +12 -3
  5. package/dist/resources/extensions/gsd/auto-dispatch.js +17 -14
  6. package/dist/resources/extensions/gsd/auto-prompts.js +43 -12
  7. package/dist/resources/extensions/gsd/auto-recovery.js +13 -6
  8. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +103 -13
  9. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +6 -1
  10. package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +2 -0
  11. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +8 -3
  12. package/dist/resources/extensions/gsd/bootstrap/system-context.js +46 -19
  13. package/dist/resources/extensions/gsd/bootstrap/tool-call-loop-guard.js +75 -1
  14. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +1 -1
  15. package/dist/resources/extensions/gsd/commands-context.js +19 -1
  16. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +16 -10
  17. package/dist/resources/extensions/gsd/commands-worktree.js +12 -10
  18. package/dist/resources/extensions/gsd/dashboard-overlay.js +32 -3
  19. package/dist/resources/extensions/gsd/db/queries.js +60 -0
  20. package/dist/resources/extensions/gsd/doctor-providers.js +92 -8
  21. package/dist/resources/extensions/gsd/exec-sandbox.js +45 -9
  22. package/dist/resources/extensions/gsd/forensics.js +2 -32
  23. package/dist/resources/extensions/gsd/git-service.js +4 -4
  24. package/dist/resources/extensions/gsd/guided-flow-queue.js +59 -5
  25. package/dist/resources/extensions/gsd/health-widget.js +55 -29
  26. package/dist/resources/extensions/gsd/markdown-renderer.js +6 -2
  27. package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +44 -21
  28. package/dist/resources/extensions/gsd/milestone-implementation-evidence.js +26 -20
  29. package/dist/resources/extensions/gsd/quick.js +45 -2
  30. package/dist/resources/extensions/gsd/session-forensics.js +11 -1
  31. package/dist/resources/extensions/gsd/state-reconciliation/drift/stale-render.js +52 -3
  32. package/dist/resources/extensions/gsd/tools/complete-slice.js +34 -3
  33. package/dist/resources/extensions/gsd/tools/complete-task.js +78 -16
  34. package/dist/resources/extensions/gsd/tools/exec-tool.js +7 -2
  35. package/dist/resources/extensions/gsd/unit-context-composer.js +23 -7
  36. package/dist/resources/extensions/gsd/unit-registry.js +25 -3
  37. package/dist/resources/extensions/gsd/unmerged-milestone-guard.js +33 -3
  38. package/dist/resources/extensions/gsd/validation-block-guard.js +9 -4
  39. package/dist/resources/extensions/gsd/workspace-git-preflight.js +30 -1
  40. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  41. package/dist/web/standalone/.next/BUILD_ID +1 -1
  42. package/dist/web/standalone/.next/app-path-routes-manifest.json +14 -14
  43. package/dist/web/standalone/.next/build-manifest.json +3 -3
  44. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  45. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  46. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  47. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  48. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  49. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  50. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  51. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  52. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  53. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  54. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  55. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  56. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  57. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  58. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  59. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  60. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  61. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  62. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  63. package/dist/web/standalone/.next/server/app/index.html +1 -1
  64. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  65. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  66. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  67. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  68. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  69. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  70. package/dist/web/standalone/.next/server/app-paths-manifest.json +14 -14
  71. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  72. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  73. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  74. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  75. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  76. package/dist/web/standalone/.next/static/chunks/{796.e0bdc932325d7e03.js → 796.3976108148518f7d.js} +3 -3
  77. package/dist/web/standalone/.next/static/chunks/{webpack-f46ea08200a0227e.js → webpack-7c1d97e39be2da11.js} +1 -1
  78. package/package.json +1 -1
  79. package/packages/cloud-mcp-gateway/package.json +2 -2
  80. package/packages/contracts/dist/workflow.d.ts +1 -0
  81. package/packages/contracts/dist/workflow.d.ts.map +1 -1
  82. package/packages/contracts/dist/workflow.js +2 -0
  83. package/packages/contracts/dist/workflow.js.map +1 -1
  84. package/packages/contracts/package.json +1 -1
  85. package/packages/daemon/package.json +4 -4
  86. package/packages/gsd-agent-core/package.json +5 -5
  87. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  88. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +21 -9
  89. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  90. package/packages/gsd-agent-modes/package.json +7 -7
  91. package/packages/mcp-server/README.md +1 -1
  92. package/packages/mcp-server/dist/server.d.ts +1 -1
  93. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  94. package/packages/mcp-server/dist/server.js +3 -3
  95. package/packages/mcp-server/dist/server.js.map +1 -1
  96. package/packages/mcp-server/dist/workflow-tools.d.ts +13 -1
  97. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  98. package/packages/mcp-server/dist/workflow-tools.js +34 -20
  99. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  100. package/packages/mcp-server/package.json +4 -4
  101. package/packages/native/package.json +1 -1
  102. package/packages/pi-agent-core/package.json +1 -1
  103. package/packages/pi-ai/package.json +1 -1
  104. package/packages/pi-coding-agent/package.json +7 -7
  105. package/packages/pi-tui/package.json +2 -2
  106. package/packages/rpc-client/package.json +2 -2
  107. package/pkg/package.json +1 -1
  108. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +20 -2
  109. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +80 -0
  110. package/src/resources/extensions/google-cli/stream-adapter.ts +106 -19
  111. package/src/resources/extensions/gsd/auto/orchestrator.ts +25 -11
  112. package/src/resources/extensions/gsd/auto-dispatch.ts +18 -17
  113. package/src/resources/extensions/gsd/auto-prompts.ts +54 -12
  114. package/src/resources/extensions/gsd/auto-recovery.ts +13 -6
  115. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +125 -12
  116. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +6 -1
  117. package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +2 -0
  118. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +9 -3
  119. package/src/resources/extensions/gsd/bootstrap/system-context.ts +52 -18
  120. package/src/resources/extensions/gsd/bootstrap/tool-call-loop-guard.ts +82 -1
  121. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +1 -1
  122. package/src/resources/extensions/gsd/commands-context.ts +18 -1
  123. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +14 -9
  124. package/src/resources/extensions/gsd/commands-worktree.ts +12 -10
  125. package/src/resources/extensions/gsd/dashboard-overlay.ts +32 -3
  126. package/src/resources/extensions/gsd/db/queries.ts +79 -0
  127. package/src/resources/extensions/gsd/doctor-providers.ts +103 -9
  128. package/src/resources/extensions/gsd/exec-sandbox.ts +49 -9
  129. package/src/resources/extensions/gsd/forensics.ts +2 -33
  130. package/src/resources/extensions/gsd/git-service.ts +5 -5
  131. package/src/resources/extensions/gsd/guided-flow-queue.ts +82 -4
  132. package/src/resources/extensions/gsd/health-widget.ts +69 -32
  133. package/src/resources/extensions/gsd/markdown-renderer.ts +6 -1
  134. package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +51 -19
  135. package/src/resources/extensions/gsd/milestone-implementation-evidence.ts +35 -21
  136. package/src/resources/extensions/gsd/quick.ts +43 -2
  137. package/src/resources/extensions/gsd/session-forensics.ts +11 -1
  138. package/src/resources/extensions/gsd/state-reconciliation/drift/stale-render.ts +76 -8
  139. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +111 -1
  140. package/src/resources/extensions/gsd/tests/commands-context.test.ts +26 -0
  141. package/src/resources/extensions/gsd/tests/commands-worktree-clean.test.ts +80 -0
  142. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +11 -0
  143. package/src/resources/extensions/gsd/tests/complete-task-rollback-evidence.test.ts +48 -8
  144. package/src/resources/extensions/gsd/tests/complete-task.test.ts +75 -0
  145. package/src/resources/extensions/gsd/tests/dashboard-overlay.test.ts +55 -2
  146. package/src/resources/extensions/gsd/tests/dispatch-rule-coverage.test.ts +26 -1
  147. package/src/resources/extensions/gsd/tests/doctor-forensics-db-open-regression.test.ts +70 -2
  148. package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +107 -0
  149. package/src/resources/extensions/gsd/tests/exec-graceful-kill.test.ts +38 -0
  150. package/src/resources/extensions/gsd/tests/exec-tool.test.ts +45 -1
  151. package/src/resources/extensions/gsd/tests/forensics-error-filter.test.ts +88 -0
  152. package/src/resources/extensions/gsd/tests/guided-discuss-milestone-prompt-rendering.test.ts +42 -0
  153. package/src/resources/extensions/gsd/tests/health-widget.test.ts +268 -3
  154. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +119 -1
  155. package/src/resources/extensions/gsd/tests/integration/queue-active-milestone-context-budget.test.ts +93 -0
  156. package/src/resources/extensions/gsd/tests/integration/quick-branch-lifecycle.test.ts +56 -9
  157. package/src/resources/extensions/gsd/tests/knowledge-cold-start.test.ts +14 -0
  158. package/src/resources/extensions/gsd/tests/memory-consolidation-scanner.test.ts +78 -0
  159. package/src/resources/extensions/gsd/tests/orchestrator-logs.test.ts +43 -1
  160. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +26 -0
  161. package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +1 -1
  162. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +54 -1
  163. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +195 -1
  164. package/src/resources/extensions/gsd/tests/read-uat-gate-verdict.test.ts +185 -0
  165. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +87 -0
  166. package/src/resources/extensions/gsd/tests/state-reconciliation-drift.test.ts +76 -0
  167. package/src/resources/extensions/gsd/tests/tool-call-loop-guard.test.ts +68 -0
  168. package/src/resources/extensions/gsd/tests/tool-param-optionality.test.ts +26 -0
  169. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +193 -14
  170. package/src/resources/extensions/gsd/tests/unmerged-milestone-guard.test.ts +25 -0
  171. package/src/resources/extensions/gsd/tests/validation-block-guard.test.ts +79 -0
  172. package/src/resources/extensions/gsd/tests/verify-artifact-tightened.test.ts +66 -0
  173. package/src/resources/extensions/gsd/tests/workspace-git-preflight.test.ts +151 -2
  174. package/src/resources/extensions/gsd/tools/complete-slice.ts +30 -3
  175. package/src/resources/extensions/gsd/tools/complete-task.ts +86 -16
  176. package/src/resources/extensions/gsd/tools/exec-tool.ts +7 -3
  177. package/src/resources/extensions/gsd/unit-context-composer.ts +33 -7
  178. package/src/resources/extensions/gsd/unit-registry.ts +25 -3
  179. package/src/resources/extensions/gsd/unmerged-milestone-guard.ts +41 -5
  180. package/src/resources/extensions/gsd/validation-block-guard.ts +13 -7
  181. package/src/resources/extensions/gsd/workspace-git-preflight.ts +31 -0
  182. /package/dist/web/standalone/.next/static/{BTKtGFF1Y-hvVJEGhBRo9 → SzEuqWX37DR9MEpEuQjP1}/_buildManifest.js +0 -0
  183. /package/dist/web/standalone/.next/static/{BTKtGFF1Y-hvVJEGhBRo9 → SzEuqWX37DR9MEpEuQjP1}/_ssgManifest.js +0 -0
@@ -119,3 +119,91 @@ describe("extractTrace error filtering (#2539)", () => {
119
119
  assert.equal(trace.errors.length, 1, "lint error with output should be counted");
120
120
  });
121
121
  });
122
+
123
+ describe("extractTrace pending tool calls (#945)", () => {
124
+ test("tool call without a toolResult is counted as an error", () => {
125
+ const entries = [
126
+ {
127
+ type: "message",
128
+ message: {
129
+ role: "assistant",
130
+ content: [
131
+ {
132
+ type: "toolCall",
133
+ id: "toolu_missing",
134
+ name: "gsd_summary_save",
135
+ arguments: { content: "unfinished summary" },
136
+ },
137
+ ],
138
+ },
139
+ },
140
+ ];
141
+
142
+ const trace = extractTrace(entries);
143
+
144
+ assert.equal(trace.toolCalls.length, 1);
145
+ assert.equal(trace.toolCalls[0]?.name, "gsd_summary_save");
146
+ assert.equal(trace.toolCalls[0]?.isError, true);
147
+ assert.equal(trace.toolCalls[0]?.result, "missing tool result (stream/tool-call abort before execution)");
148
+ assert.equal(trace.errors.length, 1);
149
+ assert.equal(trace.errors[0], "Tool call gsd_summary_save started but no toolResult was recorded");
150
+ });
151
+
152
+ test("pending bash call with no toolResult marks commandsRun entry as failed", () => {
153
+ // A bash tool call that was initiated but never got a toolResult (stream abort).
154
+ const entries = [
155
+ {
156
+ type: "message",
157
+ message: {
158
+ role: "assistant",
159
+ content: [
160
+ {
161
+ type: "toolCall",
162
+ id: "toolu_bash_missing",
163
+ name: "bash",
164
+ arguments: { command: "npm run build" },
165
+ },
166
+ ],
167
+ },
168
+ },
169
+ ];
170
+
171
+ const trace = extractTrace(entries);
172
+
173
+ // The tool call should be recorded as an error.
174
+ assert.equal(trace.toolCalls.length, 1);
175
+ assert.equal(trace.toolCalls[0]?.name, "bash");
176
+ assert.equal(trace.toolCalls[0]?.isError, true);
177
+
178
+ // The commandsRun entry must be consistent: failed: true, not failed: false.
179
+ assert.equal(trace.commandsRun.length, 1);
180
+ assert.equal(trace.commandsRun[0]?.command, "npm run build");
181
+ assert.equal(trace.commandsRun[0]?.failed, true,
182
+ "commandsRun entry should be failed when its bash call never returned a result");
183
+ });
184
+
185
+ test("pending bg_shell call with no toolResult marks commandsRun entry as failed", () => {
186
+ const entries = [
187
+ {
188
+ type: "message",
189
+ message: {
190
+ role: "assistant",
191
+ content: [
192
+ {
193
+ type: "toolCall",
194
+ id: "toolu_bg_missing",
195
+ name: "bg_shell",
196
+ arguments: { command: "sleep 30" },
197
+ },
198
+ ],
199
+ },
200
+ },
201
+ ];
202
+
203
+ const trace = extractTrace(entries);
204
+
205
+ assert.equal(trace.commandsRun.length, 1);
206
+ assert.equal(trace.commandsRun[0]?.failed, true,
207
+ "bg_shell commandsRun entry should be failed when no toolResult was recorded");
208
+ });
209
+ });
@@ -86,3 +86,45 @@ test("guided milestone prompt builder preloads milestone planning context", asyn
86
86
  rmSync(base, { recursive: true, force: true });
87
87
  }
88
88
  });
89
+
90
+ test("guided milestone prompt builder caps prior draft seed before interpolation", async () => {
91
+ const base = mkdtempSync(join(tmpdir(), "gsd-guided-milestone-draft-cap-"));
92
+ const previousGsdHome = process.env.GSD_HOME;
93
+ process.env.GSD_HOME = join(base, ".gsd-home");
94
+
95
+ try {
96
+ const currentDir = join(base, ".gsd", "milestones", "M001");
97
+ mkdirSync(currentDir, { recursive: true });
98
+
99
+ const draftPath = join(currentDir, "M001-CONTEXT-DRAFT.md");
100
+ writeFileSync(draftPath, "# Draft\n\nSMALL-DRAFT-SIGNAL", "utf-8");
101
+ const smallPrompt = await buildDiscussMilestonePrompt("M001", "Draft Resume", base, "true", {
102
+ includeContextMode: false,
103
+ });
104
+
105
+ writeFileSync(
106
+ draftPath,
107
+ ["# Draft", "", "SMALL-DRAFT-SIGNAL", "", "A".repeat(200_000), "", "OVERSIZED-DRAFT-TAIL-SIGNAL"].join("\n"),
108
+ "utf-8",
109
+ );
110
+ const largePrompt = await buildDiscussMilestonePrompt("M001", "Draft Resume", base, "true", {
111
+ includeContextMode: false,
112
+ });
113
+
114
+ const addedChars = largePrompt.length - smallPrompt.length;
115
+
116
+ assert.match(largePrompt, /## Prior Discussion \(Draft Seed\)/);
117
+ assert.match(largePrompt, /### Prior Discussion Draft/);
118
+ assert.match(largePrompt, /SMALL-DRAFT-SIGNAL/);
119
+ assert.doesNotMatch(largePrompt, /OVERSIZED-DRAFT-TAIL-SIGNAL/);
120
+ assert.match(
121
+ largePrompt,
122
+ /Draft seed truncated; read the full draft at `\.gsd\/milestones\/M001\/M001-CONTEXT-DRAFT\.md` if needed\./,
123
+ );
124
+ assert.ok(addedChars < 25_000, `large draft should add bounded seed chars, added ${addedChars}`);
125
+ } finally {
126
+ if (previousGsdHome === undefined) delete process.env.GSD_HOME;
127
+ else process.env.GSD_HOME = previousGsdHome;
128
+ rmSync(base, { recursive: true, force: true });
129
+ }
130
+ });
@@ -3,8 +3,10 @@
3
3
 
4
4
  import test from "node:test";
5
5
  import assert from "node:assert/strict";
6
- import { mkdirSync, rmSync, writeFileSync } from "node:fs";
7
- import { join } from "node:path";
6
+ import { execFileSync } from "node:child_process";
7
+ import { chmodSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
8
+ import { performance } from "node:perf_hooks";
9
+ import { delimiter, join } from "node:path";
8
10
  import { tmpdir } from "node:os";
9
11
  import {
10
12
  buildHealthLines,
@@ -12,8 +14,9 @@ import {
12
14
  formatRelativeTime,
13
15
  type HealthWidgetData,
14
16
  } from "../health-widget-core.ts";
15
- import { HEALTH_WIDGET_ACTIVE_HINTS } from "../health-widget.ts";
17
+ import { HEALTH_WIDGET_ACTIVE_HINTS, getCachedProjectState, initHealthWidget } from "../health-widget.ts";
16
18
  import { registerHooks } from "../bootstrap/register-hooks.ts";
19
+ import { GIT_NO_PROMPT_ENV } from "../git-constants.ts";
17
20
 
18
21
  function makeTempDir(prefix: string): string {
19
22
  const dir = join(
@@ -32,6 +35,58 @@ function cleanup(dir: string): void {
32
35
  }
33
36
  }
34
37
 
38
+ function runGit(cwd: string, ...args: string[]): string {
39
+ return execFileSync("git", args, {
40
+ cwd,
41
+ encoding: "utf-8",
42
+ stdio: ["ignore", "pipe", "pipe"],
43
+ }).trim();
44
+ }
45
+
46
+ function makeTempRepo(prefix: string): string {
47
+ const dir = makeTempDir(prefix);
48
+ runGit(dir, "init");
49
+ runGit(dir, "config", "user.email", "test@test.com");
50
+ runGit(dir, "config", "user.name", "Test");
51
+ writeFileSync(join(dir, "README.md"), "# test\n", "utf-8");
52
+ runGit(dir, "add", "README.md");
53
+ runGit(dir, "commit", "-m", "initial commit");
54
+ return dir;
55
+ }
56
+
57
+ function installSlowGitLogShim(binDir: string): void {
58
+ writeFileSync(
59
+ join(binDir, "git"),
60
+ [
61
+ "#!/bin/sh",
62
+ 'if [ "$1" = "log" ]; then sleep 1; fi',
63
+ 'PATH="$GSD_REAL_PATH"',
64
+ "export PATH",
65
+ 'exec git "$@"',
66
+ "",
67
+ ].join("\n"),
68
+ "utf-8",
69
+ );
70
+ chmodSync(join(binDir, "git"), 0o755);
71
+
72
+ writeFileSync(
73
+ join(binDir, "git.cmd"),
74
+ [
75
+ "@echo off",
76
+ 'if "%1"=="log" powershell -NoProfile -Command "Start-Sleep -Seconds 1"',
77
+ 'set "PATH=%GSD_REAL_PATH%"',
78
+ "git %*",
79
+ "",
80
+ ].join("\r\n"),
81
+ "utf-8",
82
+ );
83
+ }
84
+
85
+ type HealthWidgetFactory = (
86
+ tui: { requestRender(): void },
87
+ theme: { fg(style: string, text: string): string },
88
+ ) => { dispose(): void };
89
+
35
90
  function activeData(overrides: Partial<HealthWidgetData> = {}): HealthWidgetData {
36
91
  return {
37
92
  projectState: "active",
@@ -70,6 +125,86 @@ test("detectHealthWidgetProjectState: milestone without metrics returns active",
70
125
  assert.equal(detectHealthWidgetProjectState(dir), "active");
71
126
  });
72
127
 
128
+ test("getCachedProjectState: reuses project state until the refresh TTL expires", (t) => {
129
+ const dir = makeTempDir("cached-state");
130
+ t.after(() => { cleanup(dir); });
131
+
132
+ let now = 1_000_000;
133
+ const dateNow = t.mock.method(Date, "now", () => now);
134
+ t.after(() => { dateNow.mock.restore(); });
135
+
136
+ mkdirSync(join(dir, ".gsd"), { recursive: true });
137
+ assert.equal(getCachedProjectState(dir), "initialized");
138
+
139
+ mkdirSync(join(dir, ".gsd", "milestones", "M001"), { recursive: true });
140
+ assert.equal(getCachedProjectState(dir), "initialized");
141
+
142
+ now += 60_000;
143
+ assert.equal(getCachedProjectState(dir), "initialized");
144
+
145
+ now += 1;
146
+ assert.equal(getCachedProjectState(dir), "active");
147
+ });
148
+
149
+ test("getCachedProjectState: force=true bypasses TTL and returns fresh state within TTL window", (t) => {
150
+ const dir = makeTempDir("forced-state");
151
+ t.after(() => { cleanup(dir); });
152
+
153
+ let now = 2_000_000;
154
+ const dateNow = t.mock.method(Date, "now", () => now);
155
+ t.after(() => { dateNow.mock.restore(); });
156
+
157
+ mkdirSync(join(dir, ".gsd"), { recursive: true });
158
+ // Prime the cache with "initialized".
159
+ assert.equal(getCachedProjectState(dir), "initialized");
160
+
161
+ // Disk changes within the TTL window.
162
+ mkdirSync(join(dir, ".gsd", "milestones", "M001"), { recursive: true });
163
+ now += 1_000; // well within 60s TTL
164
+
165
+ // Normal call still returns stale cached value.
166
+ assert.equal(getCachedProjectState(dir), "initialized");
167
+
168
+ // force=true bypasses TTL and returns the fresh disk state.
169
+ assert.equal(getCachedProjectState(dir, true), "active");
170
+
171
+ // Subsequent non-forced call also reflects the freshened cache.
172
+ assert.equal(getCachedProjectState(dir), "active");
173
+ });
174
+
175
+ test("initHealthWidget: re-init paints fresh project state within cache TTL", (t) => {
176
+ const dir = makeTempDir("reinit-state");
177
+ t.after(() => { cleanup(dir); });
178
+
179
+ let now = 3_000_000;
180
+ const dateNow = t.mock.method(Date, "now", () => now);
181
+ t.after(() => { dateNow.mock.restore(); });
182
+
183
+ const originalCwd = process.cwd();
184
+ process.chdir(dir);
185
+ t.after(() => { process.chdir(originalCwd); });
186
+
187
+ const initialLineSets: string[][] = [];
188
+ const ctx = {
189
+ hasUI: true,
190
+ ui: {
191
+ setWidget: (_key: string, value: unknown) => {
192
+ if (Array.isArray(value)) initialLineSets.push(value as string[]);
193
+ },
194
+ },
195
+ } as any;
196
+
197
+ mkdirSync(join(dir, ".gsd"), { recursive: true });
198
+ initHealthWidget(ctx);
199
+ assert.equal(initialLineSets.at(-1)?.[0], " GSD Project Initialized");
200
+
201
+ mkdirSync(join(dir, ".gsd", "milestones", "M001"), { recursive: true });
202
+ now += 1_000;
203
+
204
+ initHealthWidget(ctx);
205
+ assert.match(initialLineSets.at(-1)?.[0] ?? "", /System OK/);
206
+ });
207
+
73
208
  test("buildHealthLines: none state shows single onboarding line pointing at /gsd", (t) => {
74
209
  const lines = buildHealthLines(activeData({ projectState: "none" }));
75
210
  assert.equal(lines.length, 1, "renders exactly one line");
@@ -101,6 +236,136 @@ test("health widget active hints include visualization and notifications", () =>
101
236
  assert.match(HEALTH_WIDGET_ACTIVE_HINTS, /\/gsd help/);
102
237
  });
103
238
 
239
+ test("health widget async refresh does not block timers while git log is slow", async (t) => {
240
+ const dir = makeTempRepo("slow-git-log");
241
+ const binDir = makeTempDir("slow-git-log-bin");
242
+ mkdirSync(join(dir, ".gsd", "milestones", "M001"), { recursive: true });
243
+ installSlowGitLogShim(binDir);
244
+
245
+ const originalCwd = process.cwd();
246
+ const originalProcessPath = process.env.PATH;
247
+ const originalEnvPath = GIT_NO_PROMPT_ENV.PATH;
248
+ const originalEnvRealPath = GIT_NO_PROMPT_ENV.GSD_REAL_PATH;
249
+ const shimmedPath = `${binDir}${delimiter}${originalProcessPath ?? ""}`;
250
+
251
+ process.chdir(dir);
252
+ process.env.PATH = shimmedPath;
253
+ GIT_NO_PROMPT_ENV.PATH = shimmedPath;
254
+ GIT_NO_PROMPT_ENV.GSD_REAL_PATH = originalProcessPath ?? "";
255
+
256
+ let factory: HealthWidgetFactory | null = null;
257
+ let resolveRefresh: (() => void) | undefined;
258
+ const refreshed = new Promise<void>((resolve) => { resolveRefresh = resolve; });
259
+ const gaps: number[] = [];
260
+ let lastTick = performance.now();
261
+ let heartbeat: NodeJS.Timeout | undefined;
262
+ let refreshTimeout: NodeJS.Timeout | undefined;
263
+ let widget: { dispose(): void } | undefined;
264
+
265
+ t.after(() => {
266
+ if (widget) widget.dispose();
267
+ if (heartbeat) clearInterval(heartbeat);
268
+ if (refreshTimeout) clearTimeout(refreshTimeout);
269
+ process.chdir(originalCwd);
270
+ if (originalProcessPath === undefined) delete process.env.PATH;
271
+ else process.env.PATH = originalProcessPath;
272
+ if (originalEnvPath === undefined) delete GIT_NO_PROMPT_ENV.PATH;
273
+ else GIT_NO_PROMPT_ENV.PATH = originalEnvPath;
274
+ if (originalEnvRealPath === undefined) delete GIT_NO_PROMPT_ENV.GSD_REAL_PATH;
275
+ else GIT_NO_PROMPT_ENV.GSD_REAL_PATH = originalEnvRealPath;
276
+ cleanup(binDir);
277
+ cleanup(dir);
278
+ });
279
+
280
+ initHealthWidget({
281
+ hasUI: true,
282
+ ui: {
283
+ setWidget: (_key: string, value: unknown) => {
284
+ if (typeof value === "function") factory = value as HealthWidgetFactory;
285
+ },
286
+ },
287
+ } as any);
288
+
289
+ assert.ok(factory, "health widget factory is registered");
290
+
291
+ heartbeat = setInterval(() => {
292
+ const now = performance.now();
293
+ gaps.push(now - lastTick);
294
+ lastTick = now;
295
+ }, 25);
296
+
297
+ // assert.ok above guards at runtime; double-cast is needed because TypeScript
298
+ // cannot track the factory assignment through the `as any` closure call.
299
+ widget = (factory as unknown as HealthWidgetFactory)(
300
+ { requestRender: () => { resolveRefresh?.(); } },
301
+ { fg: (_style: string, text: string) => text },
302
+ );
303
+
304
+ await Promise.race([
305
+ refreshed,
306
+ new Promise<never>((_, reject) => {
307
+ refreshTimeout = setTimeout(() => reject(new Error("health widget refresh did not complete")), 4_000);
308
+ }),
309
+ ]);
310
+ if (refreshTimeout) clearTimeout(refreshTimeout);
311
+
312
+ assert.ok(gaps.length > 0, "heartbeat ran while refresh was in flight");
313
+ const maxGap = Math.max(...gaps);
314
+ assert.ok(maxGap < 750, `slow git log must not starve timers; max gap was ${Math.round(maxGap)}ms`);
315
+ });
316
+
317
+ test("initHealthWidget: synchronous first-paint render never contains last-commit info (regression #964)", (t) => {
318
+ // Before the fix, loadHealthWidgetData with includeChecks:true called
319
+ // loadLastCommitInfo — synchronous native-git-bridge ops (nativeIsRepo,
320
+ // nativeGetCurrentBranch, nativeLastCommitEpoch, nativeCommitSubject) — which
321
+ // froze the TUI on slow repos. The fix removes that synchronous git path
322
+ // entirely: lastCommitEpoch/lastCommitMessage are now always null from the
323
+ // synchronous loader; only the async refresh (loadLastCommitInfoAsync) fills
324
+ // them in. This test guards that contract by verifying that the initial
325
+ // string-array setWidget call never contains "Last commit:" even on a real
326
+ // git repo where native git queries would succeed.
327
+ const dir = makeTempRepo("sync-last-commit-regression");
328
+ mkdirSync(join(dir, ".gsd", "milestones", "M001"), { recursive: true });
329
+
330
+ const originalCwd = process.cwd();
331
+ process.chdir(dir);
332
+
333
+ let widget: { dispose(): void } | undefined;
334
+ t.after(() => {
335
+ if (widget) widget.dispose();
336
+ process.chdir(originalCwd);
337
+ cleanup(dir);
338
+ });
339
+
340
+ const initialRenders: string[][] = [];
341
+
342
+ initHealthWidget({
343
+ hasUI: true,
344
+ ui: {
345
+ setWidget: (_key: string, value: unknown) => {
346
+ if (Array.isArray(value)) {
347
+ initialRenders.push(value as string[]);
348
+ } else if (typeof value === "function") {
349
+ // Instantiate the factory to satisfy dispose(), but do not await the
350
+ // async refresh — we are only inspecting the synchronous first-paint.
351
+ widget = (value as unknown as HealthWidgetFactory)(
352
+ { requestRender: () => {} },
353
+ { fg: (_style: string, text: string) => text },
354
+ );
355
+ }
356
+ },
357
+ },
358
+ } as any);
359
+
360
+ assert.ok(initialRenders.length > 0, "at least one synchronous setWidget call");
361
+
362
+ const combined = initialRenders.flat().join("\n");
363
+ assert.ok(
364
+ !combined.includes("Last commit:"),
365
+ "synchronous first-paint render must not contain 'Last commit:' — sync git path removed (regression #964)",
366
+ );
367
+ });
368
+
104
369
  test("buildHealthLines: active state with budget ceiling shows percent summary", (t) => {
105
370
  const lines = buildHealthLines(activeData({ budgetSpent: 2.5, budgetCeiling: 10 }));
106
371
  assert.equal(lines.length, 1);
@@ -3,7 +3,7 @@
3
3
  import { describe, test } from 'node:test';
4
4
  import assert from 'node:assert/strict';
5
5
  import { mkdtempSync, mkdirSync, writeFileSync, rmSync, existsSync, symlinkSync, readFileSync, chmodSync } from "node:fs";
6
- import { join, dirname } from "node:path";
6
+ import { join, dirname, delimiter } from "node:path";
7
7
  import { tmpdir } from "node:os";
8
8
  import { execFileSync, execSync } from "node:child_process";
9
9
 
@@ -23,6 +23,7 @@ import {
23
23
  type PreMergeCheckResult,
24
24
  type TaskCommitContext,
25
25
  } from "../../git-service.ts";
26
+ import { GIT_NO_PROMPT_ENV } from "../../git-constants.ts";
26
27
  import { nativeAddAllWithExclusions, nativeHasChanges, _resetHasChangesCache } from "../../native-git-bridge.ts";
27
28
  function run(command: string, cwd: string): string {
28
29
  return execSync(command, { cwd, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8" }).trim();
@@ -405,6 +406,49 @@ describe('git-service', async () => {
405
406
  }).trim();
406
407
  }
407
408
 
409
+ function findGitExecutable(): string {
410
+ const command = process.platform === "win32" ? "where git" : "command -v git";
411
+ const output = execSync(command, { encoding: "utf-8" });
412
+ const found = output.split(/\r?\n/).map(line => line.trim()).find(Boolean);
413
+ assert.ok(found, "git executable is available");
414
+ return found;
415
+ }
416
+
417
+ function installGitRmCachedCounterShim(shimDir: string): void {
418
+ const shimScript = join(shimDir, "git-shim.cjs");
419
+ writeFileSync(shimScript, `
420
+ const { appendFileSync } = require("node:fs");
421
+ const { spawnSync } = require("node:child_process");
422
+
423
+ const args = process.argv.slice(2);
424
+ if (args[0] === "rm" && args[1] === "--cached") {
425
+ appendFileSync(process.env.GSD_GIT_SHIM_LOG, args.join("\\0") + "\\n");
426
+ }
427
+
428
+ const result = spawnSync(process.env.GSD_REAL_GIT || "git", args, {
429
+ stdio: "inherit",
430
+ env: process.env,
431
+ });
432
+ if (result.error) {
433
+ console.error(result.error.message);
434
+ process.exit(1);
435
+ }
436
+ process.exit(result.status ?? 0);
437
+ `.trimStart(), "utf-8");
438
+
439
+ const posixShim = join(shimDir, "git");
440
+ const quotedScript = `'${shimScript.replace(/'/g, "'\\''")}'`;
441
+ writeFileSync(posixShim, `#!/bin/sh\nexec node ${quotedScript} "$@"\n`, "utf-8");
442
+ chmodSync(posixShim, 0o755);
443
+
444
+ writeFileSync(join(shimDir, "git.cmd"), "@echo off\r\nnode \"%~dp0git-shim.cjs\" %*\r\n", "utf-8");
445
+ }
446
+
447
+ function countRmCachedInvocations(logFile: string): number {
448
+ if (!existsSync(logFile)) return 0;
449
+ return readFileSync(logFile, "utf-8").split(/\r?\n/).filter(Boolean).length;
450
+ }
451
+
408
452
  // ─── GitServiceImpl: smart staging ─────────────────────────────────────
409
453
 
410
454
  test('GitServiceImpl: smart staging', () => {
@@ -447,6 +491,80 @@ describe('git-service', async () => {
447
491
  rmSync(repo, { recursive: true, force: true });
448
492
  });
449
493
 
494
+ test('GitServiceImpl: runtime cleanup runs once per repo across service instances', () => {
495
+ const repo = initTempRepo();
496
+ const shimDir = mkdtempSync(join(tmpdir(), "gsd-git-rm-shim-"));
497
+ const logFile = join(shimDir, "git-rm-cached.log");
498
+ const originalPath = process.env.PATH;
499
+ const originalRealGit = process.env.GSD_REAL_GIT;
500
+ const originalShimLog = process.env.GSD_GIT_SHIM_LOG;
501
+ const originalSafePath = GIT_NO_PROMPT_ENV.PATH;
502
+ const originalSafeRealGit = GIT_NO_PROMPT_ENV.GSD_REAL_GIT;
503
+ const originalSafeShimLog = GIT_NO_PROMPT_ENV.GSD_GIT_SHIM_LOG;
504
+
505
+ try {
506
+ installGitRmCachedCounterShim(shimDir);
507
+ const realGit = findGitExecutable();
508
+ process.env.GSD_REAL_GIT = realGit;
509
+ process.env.GSD_GIT_SHIM_LOG = logFile;
510
+ process.env.PATH = `${shimDir}${delimiter}${originalPath ?? ""}`;
511
+ GIT_NO_PROMPT_ENV.GSD_REAL_GIT = realGit;
512
+ GIT_NO_PROMPT_ENV.GSD_GIT_SHIM_LOG = logFile;
513
+ GIT_NO_PROMPT_ENV.PATH = process.env.PATH;
514
+
515
+ createFile(repo, "src/first.ts", "first");
516
+ const first = new GitServiceImpl(repo).commit({ message: "test: first cleanup pass" });
517
+ assert.deepStrictEqual(first, "test: first cleanup pass", "first commit succeeds");
518
+ assert.deepStrictEqual(
519
+ countRmCachedInvocations(logFile),
520
+ RUNTIME_EXCLUSION_PATHS.length,
521
+ "first service instance checks each runtime exclusion once",
522
+ );
523
+
524
+ createFile(repo, "src/second.ts", "second");
525
+ const second = new GitServiceImpl(repo).commit({ message: "test: second cleanup pass" });
526
+ assert.deepStrictEqual(second, "test: second cleanup pass", "second commit succeeds");
527
+ assert.deepStrictEqual(
528
+ countRmCachedInvocations(logFile),
529
+ RUNTIME_EXCLUSION_PATHS.length,
530
+ "fresh service instance for the same repo does not rerun runtime cleanup",
531
+ );
532
+ } finally {
533
+ if (originalPath === undefined) {
534
+ delete process.env.PATH;
535
+ } else {
536
+ process.env.PATH = originalPath;
537
+ }
538
+ if (originalRealGit === undefined) {
539
+ delete process.env.GSD_REAL_GIT;
540
+ } else {
541
+ process.env.GSD_REAL_GIT = originalRealGit;
542
+ }
543
+ if (originalShimLog === undefined) {
544
+ delete process.env.GSD_GIT_SHIM_LOG;
545
+ } else {
546
+ process.env.GSD_GIT_SHIM_LOG = originalShimLog;
547
+ }
548
+ if (originalSafePath === undefined) {
549
+ delete GIT_NO_PROMPT_ENV.PATH;
550
+ } else {
551
+ GIT_NO_PROMPT_ENV.PATH = originalSafePath;
552
+ }
553
+ if (originalSafeRealGit === undefined) {
554
+ delete GIT_NO_PROMPT_ENV.GSD_REAL_GIT;
555
+ } else {
556
+ GIT_NO_PROMPT_ENV.GSD_REAL_GIT = originalSafeRealGit;
557
+ }
558
+ if (originalSafeShimLog === undefined) {
559
+ delete GIT_NO_PROMPT_ENV.GSD_GIT_SHIM_LOG;
560
+ } else {
561
+ GIT_NO_PROMPT_ENV.GSD_GIT_SHIM_LOG = originalSafeShimLog;
562
+ }
563
+ rmSync(repo, { recursive: true, force: true });
564
+ rmSync(shimDir, { recursive: true, force: true });
565
+ }
566
+ });
567
+
450
568
  test('GitServiceImpl: task autoCommit skips keyFiles inside submodules', () => {
451
569
  const repo = initTempRepo();
452
570
  const subSrc = mkdtempSync(join(tmpdir(), "gsd-git-submodule-src-"));
@@ -0,0 +1,93 @@
1
+ import { describe, test } from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
4
+ import { join } from "node:path";
5
+ import { tmpdir } from "node:os";
6
+
7
+ import { buildExistingMilestonesContext } from "../../guided-flow-queue.ts";
8
+ import type { GSDState, MilestoneRegistryEntry } from "../../types.ts";
9
+
10
+ const LARGE_BODY = "A".repeat(150_000);
11
+ const LARGE_DRAFT = "D".repeat(150_000);
12
+ const LARGE_ROADMAP = "R".repeat(150_000);
13
+
14
+ function makeState(registry: MilestoneRegistryEntry[]): GSDState {
15
+ return {
16
+ activeMilestone: registry.find(m => m.status === "active") ?? null,
17
+ activeSlice: null,
18
+ activeTask: null,
19
+ phase: "executing",
20
+ recentDecisions: [],
21
+ blockers: [],
22
+ nextAction: "",
23
+ registry,
24
+ };
25
+ }
26
+
27
+ function writeMilestoneArtifact(base: string, mid: string, suffix: string, content: string): void {
28
+ mkdirSync(join(base, ".gsd", "milestones", mid), { recursive: true });
29
+ writeFileSync(join(base, ".gsd", "milestones", mid, `${mid}-${suffix}.md`), content);
30
+ }
31
+
32
+ describe("queue active/pending milestone context budget", () => {
33
+ test("summarizes active and pending artifacts with source paths and bounded excerpts", async () => {
34
+ const tmpBase = mkdtempSync(join(tmpdir(), "gsd-queue-active-budget-"));
35
+ try {
36
+ writeMilestoneArtifact(tmpBase, "M001", "CONTEXT", `# Active context\n\n${LARGE_BODY}\nEND_ACTIVE_CONTEXT`);
37
+ writeMilestoneArtifact(tmpBase, "M001", "ROADMAP", `# Active roadmap\n\n${LARGE_ROADMAP}\nEND_ACTIVE_ROADMAP`);
38
+ writeMilestoneArtifact(tmpBase, "M002", "CONTEXT-DRAFT", `# Pending draft\n\n${LARGE_DRAFT}\nEND_PENDING_DRAFT`);
39
+ writeMilestoneArtifact(tmpBase, "M002", "ROADMAP", `# Pending roadmap\n\n${LARGE_ROADMAP}\nEND_PENDING_ROADMAP`);
40
+
41
+ const registry: MilestoneRegistryEntry[] = [
42
+ { id: "M001", title: "Active milestone", status: "active" },
43
+ { id: "M002", title: "Pending milestone", status: "pending" },
44
+ ];
45
+
46
+ const context = await buildExistingMilestonesContext(tmpBase, ["M001", "M002"], makeState(registry));
47
+
48
+ assert.match(context, /Source: `.gsd\/milestones\/M001\/M001-CONTEXT.md`/);
49
+ assert.match(context, /Source: `.gsd\/milestones\/M001\/M001-ROADMAP.md`/);
50
+ assert.match(context, /Source: `.gsd\/milestones\/M002\/M002-CONTEXT-DRAFT.md`/);
51
+ assert.match(context, /Source: `.gsd\/milestones\/M002\/M002-ROADMAP.md`/);
52
+ assert.match(context, /Read `.gsd\/milestones\/M001\/M001-CONTEXT.md` for full content/);
53
+ assert.equal(context.includes("END_ACTIVE_CONTEXT"), false);
54
+ assert.equal(context.includes("END_ACTIVE_ROADMAP"), false);
55
+ assert.equal(context.includes("END_PENDING_DRAFT"), false);
56
+ assert.equal(context.includes("END_PENDING_ROADMAP"), false);
57
+ } finally {
58
+ rmSync(tmpBase, { recursive: true, force: true });
59
+ }
60
+ });
61
+
62
+ test("caps the total existing milestones context", async () => {
63
+ const tmpBase = mkdtempSync(join(tmpdir(), "gsd-queue-total-budget-"));
64
+ try {
65
+ const registry: MilestoneRegistryEntry[] = [];
66
+ const milestoneIds: string[] = [];
67
+ for (let i = 1; i <= 5; i++) {
68
+ const mid = `M${String(i).padStart(3, "0")}`;
69
+ milestoneIds.push(mid);
70
+ registry.push({ id: mid, title: `Pending milestone ${i}`, status: "pending" });
71
+ writeMilestoneArtifact(tmpBase, mid, "CONTEXT", `# ${mid} context\n\n${LARGE_BODY}\nEND_${mid}_CONTEXT`);
72
+ writeMilestoneArtifact(tmpBase, mid, "ROADMAP", `# ${mid} roadmap\n\n${LARGE_ROADMAP}\nEND_${mid}_ROADMAP`);
73
+ }
74
+
75
+ const context = await buildExistingMilestonesContext(tmpBase, milestoneIds, makeState(registry));
76
+
77
+ assert.ok(
78
+ context.length <= 120_000,
79
+ `expected total context to stay within budget, got ${context.length} chars`,
80
+ );
81
+ assert.match(context, /Existing milestones context truncated/);
82
+ for (let i = 1; i <= 5; i++) {
83
+ const mid = `M${String(i).padStart(3, "0")}`;
84
+ assert.match(context, new RegExp(`### ${mid}: Pending milestone ${i}`));
85
+ assert.match(context, new RegExp(`Source: \`.gsd/milestones/${mid}/${mid}-CONTEXT.md\``));
86
+ assert.match(context, new RegExp(`Source: \`.gsd/milestones/${mid}/${mid}-ROADMAP.md\``));
87
+ }
88
+ assert.equal(context.includes("END_M005_ROADMAP"), false);
89
+ } finally {
90
+ rmSync(tmpBase, { recursive: true, force: true });
91
+ }
92
+ });
93
+ });