@opengsd/gsd-pi 1.2.0-dev.d6c5343c → 1.2.0-dev.ddc97c10
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.
- package/dist/mcp-server.js +2 -1
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/extensions/gsd/auto/orchestrator.js +28 -10
- package/dist/resources/extensions/gsd/auto/phases.js +47 -4
- package/dist/resources/extensions/gsd/auto/session.js +3 -0
- package/dist/resources/extensions/gsd/auto-direct-dispatch.js +3 -2
- package/dist/resources/extensions/gsd/auto-dispatch.js +11 -2
- package/dist/resources/extensions/gsd/auto-model-selection.js +11 -7
- package/dist/resources/extensions/gsd/auto-post-unit.js +18 -6
- package/dist/resources/extensions/gsd/auto-unit-closeout.js +45 -21
- package/dist/resources/extensions/gsd/auto-verification.js +14 -2
- package/dist/resources/extensions/gsd/auto.js +37 -1
- package/dist/resources/extensions/gsd/blocked-models.js +28 -0
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +26 -6
- package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +2 -2
- package/dist/resources/extensions/gsd/closeout-wizard.js +92 -0
- package/dist/resources/extensions/gsd/commands/context.js +16 -2
- package/dist/resources/extensions/gsd/commands-handlers.js +46 -3
- package/dist/resources/extensions/gsd/consent-question.js +16 -0
- package/dist/resources/extensions/gsd/crash-recovery.js +8 -3
- package/dist/resources/extensions/gsd/doctor-engine-checks.js +3 -3
- package/dist/resources/extensions/gsd/doctor-git-checks.js +2 -18
- package/dist/resources/extensions/gsd/gsd-command-home.js +22 -12
- package/dist/resources/extensions/gsd/gsd-db.js +2 -1
- package/dist/resources/extensions/gsd/guided-flow.js +6 -3
- package/dist/resources/extensions/gsd/milestone-closeout.js +73 -2
- package/dist/resources/extensions/gsd/milestone-planning-persistence.js +2 -2
- package/dist/resources/extensions/gsd/projection-flush.js +7 -0
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/execute-task.md +1 -1
- package/dist/resources/extensions/gsd/prompts/plan-milestone.md +1 -1
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/quick-task.md +1 -1
- package/dist/resources/extensions/gsd/prompts/reassess-roadmap.md +1 -1
- package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/replan-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/research-milestone.md +1 -1
- package/dist/resources/extensions/gsd/prompts/research-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
- package/dist/resources/extensions/gsd/prompts/run-uat.md +1 -1
- package/dist/resources/extensions/gsd/prompts/triage-captures.md +1 -1
- package/dist/resources/extensions/gsd/prompts/validate-milestone.md +1 -1
- package/dist/resources/extensions/gsd/roadmap-slices.js +25 -3
- package/dist/resources/extensions/gsd/session-lock.js +1 -1
- package/dist/resources/extensions/gsd/tool-contract.js +14 -3
- package/dist/resources/extensions/gsd/tools/complete-milestone.js +3 -2
- package/dist/resources/extensions/gsd/tools/complete-slice.js +2 -2
- package/dist/resources/extensions/gsd/tools/complete-task.js +3 -2
- package/dist/resources/extensions/gsd/tools/plan-slice.js +2 -2
- package/dist/resources/extensions/gsd/tools/plan-task.js +2 -2
- package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +2 -2
- package/dist/resources/extensions/gsd/tools/reopen-milestone.js +2 -2
- package/dist/resources/extensions/gsd/tools/reopen-slice.js +2 -2
- package/dist/resources/extensions/gsd/tools/reopen-task.js +2 -2
- package/dist/resources/extensions/gsd/tools/replan-slice.js +2 -2
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +67 -2
- package/dist/resources/extensions/gsd/verification-verdict.js +2 -1
- package/dist/resources/extensions/shared/gsd-browser-cli.js +21 -2
- package/dist/resources/shared/gsd-browser-path-sync.js +214 -0
- package/dist/resources/shared/package-manager-detection.js +1 -1
- package/dist/tsconfig.extensions.tsbuildinfo +1 -1
- package/dist/update-check.d.ts +2 -0
- package/dist/update-check.js +24 -1
- package/dist/update-cmd.js +20 -3
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
- package/dist/web/standalone/.next/server/chunks/8357.js +2 -2
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/dist/web/standalone/node_modules/node-pty/build/Makefile +1 -1
- package/package.json +1 -1
- package/packages/cloud-mcp-gateway/package.json +2 -2
- package/packages/contracts/package.json +1 -1
- package/packages/daemon/package.json +4 -4
- package/packages/gsd-agent-core/package.json +5 -5
- package/packages/gsd-agent-modes/package.json +7 -7
- package/packages/mcp-server/dist/cli.js +10 -5
- package/packages/mcp-server/dist/cli.js.map +1 -1
- package/packages/mcp-server/dist/moonshot-tool-schema.d.ts +29 -0
- package/packages/mcp-server/dist/moonshot-tool-schema.d.ts.map +1 -0
- package/packages/mcp-server/dist/moonshot-tool-schema.js +50 -0
- package/packages/mcp-server/dist/moonshot-tool-schema.js.map +1 -0
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +4 -0
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts +18 -18
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +99 -38
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +5 -4
- package/packages/native/package.json +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-ai/dist/index.d.ts +2 -0
- package/packages/pi-ai/dist/index.d.ts.map +1 -1
- package/packages/pi-ai/dist/index.js +2 -0
- package/packages/pi-ai/dist/index.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.js +12 -7
- package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
- package/packages/pi-ai/dist/providers/google-shared.d.ts +5 -0
- package/packages/pi-ai/dist/providers/google-shared.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/google-shared.js +12 -3
- package/packages/pi-ai/dist/providers/google-shared.js.map +1 -1
- package/packages/pi-ai/dist/providers/openai-completions.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/openai-completions.js +7 -3
- package/packages/pi-ai/dist/providers/openai-completions.js.map +1 -1
- package/packages/pi-ai/dist/utils/moonshot-tool-schema.d.ts +9 -0
- package/packages/pi-ai/dist/utils/moonshot-tool-schema.d.ts.map +1 -0
- package/packages/pi-ai/dist/utils/moonshot-tool-schema.js +34 -0
- package/packages/pi-ai/dist/utils/moonshot-tool-schema.js.map +1 -0
- package/packages/pi-ai/dist/utils/oauth/github-copilot.d.ts.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/github-copilot.js +6 -2
- package/packages/pi-ai/dist/utils/oauth/github-copilot.js.map +1 -1
- package/packages/pi-ai/package.json +1 -1
- package/packages/pi-coding-agent/package.json +7 -7
- package/packages/pi-tui/package.json +2 -2
- package/packages/rpc-client/package.json +2 -2
- package/pkg/package.json +1 -1
- package/src/resources/extensions/browser-tools/tests/gsd-browser-launch-config.test.mjs +11 -0
- package/src/resources/extensions/gsd/auto/orchestrator.ts +28 -10
- package/src/resources/extensions/gsd/auto/phases.ts +63 -24
- package/src/resources/extensions/gsd/auto/session.ts +3 -0
- package/src/resources/extensions/gsd/auto-direct-dispatch.ts +10 -16
- package/src/resources/extensions/gsd/auto-dispatch.ts +11 -10
- package/src/resources/extensions/gsd/auto-model-selection.ts +16 -7
- package/src/resources/extensions/gsd/auto-post-unit.ts +21 -6
- package/src/resources/extensions/gsd/auto-unit-closeout.ts +83 -28
- package/src/resources/extensions/gsd/auto-verification.ts +18 -2
- package/src/resources/extensions/gsd/auto.ts +44 -1
- package/src/resources/extensions/gsd/blocked-models.ts +49 -0
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +34 -5
- package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +2 -2
- package/src/resources/extensions/gsd/closeout-wizard.ts +102 -0
- package/src/resources/extensions/gsd/commands/context.ts +16 -2
- package/src/resources/extensions/gsd/commands-handlers.ts +46 -3
- package/src/resources/extensions/gsd/consent-question.ts +15 -0
- package/src/resources/extensions/gsd/crash-recovery.ts +10 -2
- package/src/resources/extensions/gsd/doctor-engine-checks.ts +3 -3
- package/src/resources/extensions/gsd/doctor-git-checks.ts +2 -19
- package/src/resources/extensions/gsd/gsd-command-home.ts +13 -3
- package/src/resources/extensions/gsd/gsd-db.ts +4 -3
- package/src/resources/extensions/gsd/guided-flow.ts +21 -26
- package/src/resources/extensions/gsd/milestone-closeout.ts +97 -2
- package/src/resources/extensions/gsd/milestone-planning-persistence.ts +2 -2
- package/src/resources/extensions/gsd/projection-flush.ts +20 -0
- package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/execute-task.md +1 -1
- package/src/resources/extensions/gsd/prompts/plan-milestone.md +1 -1
- package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/quick-task.md +1 -1
- package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +1 -1
- package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/replan-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/research-milestone.md +1 -1
- package/src/resources/extensions/gsd/prompts/research-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
- package/src/resources/extensions/gsd/prompts/run-uat.md +1 -1
- package/src/resources/extensions/gsd/prompts/triage-captures.md +1 -1
- package/src/resources/extensions/gsd/prompts/validate-milestone.md +1 -1
- package/src/resources/extensions/gsd/roadmap-slices.ts +28 -3
- package/src/resources/extensions/gsd/session-lock.ts +1 -1
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +69 -0
- package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +97 -0
- package/src/resources/extensions/gsd/tests/auto-remote-session-lock-cleanup.test.ts +65 -3
- package/src/resources/extensions/gsd/tests/blocked-models.test.ts +19 -0
- package/src/resources/extensions/gsd/tests/consent-question.test.ts +15 -0
- package/src/resources/extensions/gsd/tests/doctor-git-checks-terminal.test.ts +73 -0
- package/src/resources/extensions/gsd/tests/gsd-command-home.test.ts +120 -0
- package/src/resources/extensions/gsd/tests/guided-dispatch-root.test.ts +2 -6
- package/src/resources/extensions/gsd/tests/milestone-closeout.test.ts +95 -4
- package/src/resources/extensions/gsd/tests/parsers-legacy-importers.test.ts +0 -1
- package/src/resources/extensions/gsd/tests/phases-terminal-complete-idempotent.test.ts +242 -0
- package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +63 -2
- package/src/resources/extensions/gsd/tests/roadmap-slices.test.ts +68 -0
- package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +19 -1
- package/src/resources/extensions/gsd/tests/tool-unavailable-retry.test.ts +33 -0
- package/src/resources/extensions/gsd/tests/transport-gate-double-complete.test.ts +139 -0
- package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +2 -0
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +273 -38
- package/src/resources/extensions/gsd/tool-contract.ts +38 -3
- package/src/resources/extensions/gsd/tools/complete-milestone.ts +3 -2
- package/src/resources/extensions/gsd/tools/complete-slice.ts +2 -2
- package/src/resources/extensions/gsd/tools/complete-task.ts +3 -2
- package/src/resources/extensions/gsd/tools/plan-slice.ts +2 -2
- package/src/resources/extensions/gsd/tools/plan-task.ts +2 -2
- package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +2 -2
- package/src/resources/extensions/gsd/tools/reopen-milestone.ts +2 -2
- package/src/resources/extensions/gsd/tools/reopen-slice.ts +2 -2
- package/src/resources/extensions/gsd/tools/reopen-task.ts +2 -2
- package/src/resources/extensions/gsd/tools/replan-slice.ts +2 -2
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +81 -2
- package/src/resources/extensions/gsd/verification-verdict.ts +4 -2
- package/src/resources/extensions/shared/gsd-browser-cli.ts +23 -2
- package/src/resources/shared/gsd-browser-path-sync.ts +273 -0
- package/src/resources/shared/package-manager-detection.ts +1 -1
- /package/dist/web/standalone/.next/static/{jmTLg6xZmAuq_LIqKOxrH → McokybTayhff1xEVc-d3T}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{jmTLg6xZmAuq_LIqKOxrH → McokybTayhff1xEVc-d3T}/_ssgManifest.js +0 -0
|
@@ -154,6 +154,29 @@ function parseTableSlices(section: string): RoadmapSliceEntry[] {
|
|
|
154
154
|
return slices;
|
|
155
155
|
}
|
|
156
156
|
|
|
157
|
+
function looksLikeTable(section: string): boolean {
|
|
158
|
+
const lines = section.split("\n");
|
|
159
|
+
|
|
160
|
+
// Checkbox format takes precedence — embedded demo tables must not switch mode (#721).
|
|
161
|
+
if (lines.some(line => /^\s*-\s+\[[ xX]\]/.test(line))) {
|
|
162
|
+
return false;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
const pipeLines = lines.filter(line => /^\s*\|/.test(line));
|
|
166
|
+
if (pipeLines.length < 2) return false;
|
|
167
|
+
|
|
168
|
+
const hasSeparatorRow = pipeLines.some((line, index) => {
|
|
169
|
+
if (index === 0) return false;
|
|
170
|
+
const cells = line.split("|").map(c => c.trim()).filter(Boolean);
|
|
171
|
+
return /^\s*\|[\s:-]+\|/.test(line) && cells.length >= 2 && cells.every(c => /^[\s:-]+$/.test(c));
|
|
172
|
+
});
|
|
173
|
+
|
|
174
|
+
if (hasSeparatorRow) return true;
|
|
175
|
+
|
|
176
|
+
// Tables without a separator row still expose slice IDs in data rows.
|
|
177
|
+
return pipeLines.some(line => /\bS\d+\b/.test(line));
|
|
178
|
+
}
|
|
179
|
+
|
|
157
180
|
export function parseRoadmapSlices(content: string): RoadmapSliceEntry[] {
|
|
158
181
|
const slicesSection = extractSlicesSection(content);
|
|
159
182
|
if (!slicesSection) {
|
|
@@ -165,9 +188,11 @@ export function parseRoadmapSlices(content: string): RoadmapSliceEntry[] {
|
|
|
165
188
|
|
|
166
189
|
// Try table format first — if the section contains pipe-delimited rows with
|
|
167
190
|
// slice IDs, parse them as a table (#1736).
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
191
|
+
if (looksLikeTable(slicesSection)) {
|
|
192
|
+
const tableSlices = parseTableSlices(slicesSection);
|
|
193
|
+
if (tableSlices.length > 0) {
|
|
194
|
+
return tableSlices;
|
|
195
|
+
}
|
|
171
196
|
}
|
|
172
197
|
|
|
173
198
|
// Standard checkbox format
|
|
@@ -387,7 +387,7 @@ export function acquireSessionLock(basePath: string): SessionLockResult {
|
|
|
387
387
|
// #3218: Provide actionable workaround when lock recovery fails
|
|
388
388
|
const lockDirPath = lockTarget + ".lock";
|
|
389
389
|
const reason = existingPid
|
|
390
|
-
? `Another auto-mode session (PID ${existingPid}) appears to be running.\
|
|
390
|
+
? `Another auto-mode session (PID ${existingPid}) appears to be running.\nRun \`/gsd stop\` for graceful shutdown, or choose "Force start" from \`/gsd auto\` to terminate it.`
|
|
391
391
|
: `Another auto-mode session lock is stuck on this project.\nRun: rm -rf "${lockDirPath}" && rm -f "${lp}"`;
|
|
392
392
|
|
|
393
393
|
return { acquired: false, reason, existingPid };
|
|
@@ -8,6 +8,7 @@ import { fileURLToPath } from "node:url";
|
|
|
8
8
|
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
9
9
|
|
|
10
10
|
import { ModelPolicyDispatchBlockedError, resolvePreferredModelConfig, resolveModelId, selectAndApplyModel, floorThinkingLevelForUnit } from "../auto-model-selection.js";
|
|
11
|
+
import { blockModelUntil, clearTemporaryModelBlocksForTest } from "../blocked-models.ts";
|
|
11
12
|
|
|
12
13
|
function makeTempDir(prefix: string): string {
|
|
13
14
|
return mkdtempSync(join(tmpdir(), prefix));
|
|
@@ -223,6 +224,74 @@ test("selectAndApplyModel honors explicit phase models without downgrading (#361
|
|
|
223
224
|
}
|
|
224
225
|
});
|
|
225
226
|
|
|
227
|
+
test("selectAndApplyModel skips a rate-limited primary until its reset time", async () => {
|
|
228
|
+
const originalCwd = process.cwd();
|
|
229
|
+
const originalGsdHome = process.env.GSD_HOME;
|
|
230
|
+
const tempProject = makeTempDir("gsd-rate-limit-fallback-project-");
|
|
231
|
+
const tempGsdHome = makeTempDir("gsd-rate-limit-fallback-home-");
|
|
232
|
+
const setModelCalls: string[] = [];
|
|
233
|
+
|
|
234
|
+
try {
|
|
235
|
+
clearTemporaryModelBlocksForTest();
|
|
236
|
+
mkdirSync(join(tempProject, ".gsd"), { recursive: true });
|
|
237
|
+
writeFileSync(
|
|
238
|
+
join(tempProject, ".gsd", "PREFERENCES.md"),
|
|
239
|
+
[
|
|
240
|
+
"---",
|
|
241
|
+
"models:",
|
|
242
|
+
" execution:",
|
|
243
|
+
" model: gpt-5.5",
|
|
244
|
+
" provider: openai-codex",
|
|
245
|
+
" fallbacks:",
|
|
246
|
+
" - anthropic/claude-sonnet-4-6",
|
|
247
|
+
"---",
|
|
248
|
+
].join("\n"),
|
|
249
|
+
"utf-8",
|
|
250
|
+
);
|
|
251
|
+
process.env.GSD_HOME = tempGsdHome;
|
|
252
|
+
process.chdir(tempProject);
|
|
253
|
+
|
|
254
|
+
const availableModels = [
|
|
255
|
+
{ id: "gpt-5.5", provider: "openai-codex", api: "responses" },
|
|
256
|
+
{ id: "claude-sonnet-4-6", provider: "anthropic", api: "anthropic-messages" },
|
|
257
|
+
];
|
|
258
|
+
const ctx = {
|
|
259
|
+
modelRegistry: { getAvailable: () => availableModels },
|
|
260
|
+
sessionManager: { getSessionId: () => "test-session" },
|
|
261
|
+
ui: { notify: () => {} },
|
|
262
|
+
model: { provider: "openai-codex", id: "gpt-5.5", api: "responses" },
|
|
263
|
+
} as any;
|
|
264
|
+
const pi = {
|
|
265
|
+
setModel: async (model: { provider: string; id: string }) => {
|
|
266
|
+
setModelCalls.push(`${model.provider}/${model.id}`);
|
|
267
|
+
return true;
|
|
268
|
+
},
|
|
269
|
+
emitBeforeModelSelect: async () => undefined,
|
|
270
|
+
getActiveTools: () => [],
|
|
271
|
+
emitAdjustToolSet: async () => undefined,
|
|
272
|
+
setActiveTools: () => {},
|
|
273
|
+
} as any;
|
|
274
|
+
|
|
275
|
+
blockModelUntil(tempProject, "openai-codex", "gpt-5.5", Date.now() + 60_000, "session limit");
|
|
276
|
+
await selectAndApplyModel(ctx, pi, "execute-task", "M001/S01/T01", tempProject, undefined, false, { provider: "openai-codex", id: "gpt-5.5" }, undefined, true);
|
|
277
|
+
|
|
278
|
+
blockModelUntil(tempProject, "openai-codex", "gpt-5.5", Date.now() - 1, "expired");
|
|
279
|
+
await selectAndApplyModel(ctx, pi, "execute-task", "M001/S01/T02", tempProject, undefined, false, { provider: "openai-codex", id: "gpt-5.5" }, undefined, true);
|
|
280
|
+
|
|
281
|
+
assert.deepEqual(setModelCalls, [
|
|
282
|
+
"anthropic/claude-sonnet-4-6",
|
|
283
|
+
"openai-codex/gpt-5.5",
|
|
284
|
+
]);
|
|
285
|
+
} finally {
|
|
286
|
+
clearTemporaryModelBlocksForTest();
|
|
287
|
+
process.chdir(originalCwd);
|
|
288
|
+
if (originalGsdHome === undefined) delete process.env.GSD_HOME;
|
|
289
|
+
else process.env.GSD_HOME = originalGsdHome;
|
|
290
|
+
rmSync(tempProject, { recursive: true, force: true });
|
|
291
|
+
rmSync(tempGsdHome, { recursive: true, force: true });
|
|
292
|
+
}
|
|
293
|
+
});
|
|
294
|
+
|
|
226
295
|
test("selectAndApplyModel lets explicit unit models bypass stale cross-provider lock (#116)", async () => {
|
|
227
296
|
const originalCwd = process.cwd();
|
|
228
297
|
const originalGsdHome = process.env.GSD_HOME;
|
|
@@ -1223,6 +1223,103 @@ test("decideOrchestratorDispatch forwards constructor session when advance input
|
|
|
1223
1223
|
}
|
|
1224
1224
|
});
|
|
1225
1225
|
|
|
1226
|
+
test("decideOrchestratorDispatch evaluates deep pre-planning rules without an active milestone", async (t) => {
|
|
1227
|
+
const base = mkdtempSync(join(tmpdir(), "gsd-orchestrator-no-active-"));
|
|
1228
|
+
t.after(() => {
|
|
1229
|
+
resetRegistry();
|
|
1230
|
+
rmSync(base, { recursive: true, force: true });
|
|
1231
|
+
});
|
|
1232
|
+
resetRegistry();
|
|
1233
|
+
mkdirSync(join(base, ".gsd"), { recursive: true });
|
|
1234
|
+
writeFileSync(
|
|
1235
|
+
join(base, ".gsd", "PREFERENCES.md"),
|
|
1236
|
+
[
|
|
1237
|
+
"---",
|
|
1238
|
+
"planning_depth: deep",
|
|
1239
|
+
"workflow_prefs_captured: true",
|
|
1240
|
+
"---",
|
|
1241
|
+
"",
|
|
1242
|
+
].join("\n"),
|
|
1243
|
+
);
|
|
1244
|
+
|
|
1245
|
+
const stateSnapshot: GSDState = {
|
|
1246
|
+
...makeState(),
|
|
1247
|
+
activeMilestone: null,
|
|
1248
|
+
phase: "pre-planning",
|
|
1249
|
+
nextAction: "All remaining milestones are parked (M027). Run /gsd unpark M027 or create a new milestone.",
|
|
1250
|
+
registry: [{ id: "M027", title: "Parked", status: "parked" }],
|
|
1251
|
+
};
|
|
1252
|
+
const ctx = { model: {}, modelRegistry: { getAll: () => [] } } as never;
|
|
1253
|
+
const pi = { getActiveTools: () => [] } as never;
|
|
1254
|
+
const session = {
|
|
1255
|
+
basePath: base,
|
|
1256
|
+
originalBasePath: base,
|
|
1257
|
+
currentMilestoneId: "M027",
|
|
1258
|
+
} as never;
|
|
1259
|
+
|
|
1260
|
+
const result = await decideOrchestratorDispatch(ctx, pi, base, session, { stateSnapshot });
|
|
1261
|
+
|
|
1262
|
+
assert.ok(result && "unitType" in result, `expected project-level dispatch, got ${JSON.stringify(result)}`);
|
|
1263
|
+
assert.equal(result.unitType, "discuss-project");
|
|
1264
|
+
assert.equal(result.unitId, "PROJECT");
|
|
1265
|
+
});
|
|
1266
|
+
|
|
1267
|
+
test("decideOrchestratorDispatch does not replay milestone-scoped verification retry when no milestone is active", async (t) => {
|
|
1268
|
+
const base = mkdtempSync(join(tmpdir(), "gsd-orchestrator-no-active-retry-"));
|
|
1269
|
+
t.after(() => {
|
|
1270
|
+
resetRegistry();
|
|
1271
|
+
rmSync(base, { recursive: true, force: true });
|
|
1272
|
+
});
|
|
1273
|
+
resetRegistry();
|
|
1274
|
+
mkdirSync(join(base, ".gsd"), { recursive: true });
|
|
1275
|
+
writeFileSync(
|
|
1276
|
+
join(base, ".gsd", "PREFERENCES.md"),
|
|
1277
|
+
[
|
|
1278
|
+
"---",
|
|
1279
|
+
"planning_depth: deep",
|
|
1280
|
+
"workflow_prefs_captured: true",
|
|
1281
|
+
"---",
|
|
1282
|
+
"",
|
|
1283
|
+
].join("\n"),
|
|
1284
|
+
);
|
|
1285
|
+
|
|
1286
|
+
const stateSnapshot: GSDState = {
|
|
1287
|
+
...makeState(),
|
|
1288
|
+
activeMilestone: null,
|
|
1289
|
+
phase: "pre-planning",
|
|
1290
|
+
nextAction: "All remaining milestones are parked (M027). Run /gsd unpark M027 or create a new milestone.",
|
|
1291
|
+
registry: [{ id: "M027", title: "Parked", status: "parked" }],
|
|
1292
|
+
};
|
|
1293
|
+
const ctx = { model: {}, modelRegistry: { getAll: () => [] } } as never;
|
|
1294
|
+
const pi = { getActiveTools: () => [] } as never;
|
|
1295
|
+
const stalePendingRetry = {
|
|
1296
|
+
unitType: "execute-task",
|
|
1297
|
+
unitId: "M027.S1.T1",
|
|
1298
|
+
prompt: "stale retry prompt",
|
|
1299
|
+
pauseAfterUatDispatch: false,
|
|
1300
|
+
state: stateSnapshot,
|
|
1301
|
+
mid: "M027",
|
|
1302
|
+
midTitle: "Parked",
|
|
1303
|
+
};
|
|
1304
|
+
const session = {
|
|
1305
|
+
basePath: base,
|
|
1306
|
+
originalBasePath: base,
|
|
1307
|
+
currentMilestoneId: "M027",
|
|
1308
|
+
pendingVerificationRetryDispatch: stalePendingRetry,
|
|
1309
|
+
} as never;
|
|
1310
|
+
|
|
1311
|
+
const result = await decideOrchestratorDispatch(ctx, pi, base, session, { stateSnapshot });
|
|
1312
|
+
|
|
1313
|
+
assert.ok(result && "unitType" in result, `expected project-level dispatch, got ${JSON.stringify(result)}`);
|
|
1314
|
+
assert.equal(result.unitType, "discuss-project");
|
|
1315
|
+
assert.equal(result.unitId, "PROJECT");
|
|
1316
|
+
// The stale retry must be preserved for a future tick, not consumed by this
|
|
1317
|
+
// no-active-milestone path (mirrors pre-#712-fix behavior where !active
|
|
1318
|
+
// returned null before touching the retry).
|
|
1319
|
+
const sess = session as unknown as { pendingVerificationRetryDispatch: unknown };
|
|
1320
|
+
assert.equal(sess.pendingVerificationRetryDispatch, stalePendingRetry);
|
|
1321
|
+
});
|
|
1322
|
+
|
|
1226
1323
|
test("decideOrchestratorDispatch adopts next active milestone after the session milestone is closed", async (t) => {
|
|
1227
1324
|
const base = mkdtempSync(join(tmpdir(), "gsd-orchestrator-milestone-adopt-"));
|
|
1228
1325
|
t.after(() => rmSync(base, { recursive: true, force: true }));
|
|
@@ -1,12 +1,13 @@
|
|
|
1
1
|
import test from "node:test";
|
|
2
2
|
import assert from "node:assert/strict";
|
|
3
|
-
import { mkdtempSync, mkdirSync, rmSync } from "node:fs";
|
|
3
|
+
import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
|
|
4
4
|
import { join } from "node:path";
|
|
5
5
|
import { tmpdir } from "node:os";
|
|
6
6
|
|
|
7
|
-
import { checkRemoteAutoSession } from "../auto.ts";
|
|
7
|
+
import { checkRemoteAutoSession, forceStopAutoRemote } from "../auto.ts";
|
|
8
8
|
import { openDatabase, closeDatabase, _getAdapter } from "../gsd-db.ts";
|
|
9
|
-
import { registerAutoWorker } from "../db/auto-workers.ts";
|
|
9
|
+
import { getAutoWorker, registerAutoWorker } from "../db/auto-workers.ts";
|
|
10
|
+
import { claimMilestoneLease, getMilestoneLease } from "../db/milestone-leases.ts";
|
|
10
11
|
import { normalizeRealPath } from "../paths.ts";
|
|
11
12
|
import { readCrashLock } from "../crash-recovery.ts";
|
|
12
13
|
|
|
@@ -35,6 +36,29 @@ function setWorkerPid(workerId: string, pid: number): void {
|
|
|
35
36
|
).run({ ":pid": pid, ":worker_id": workerId });
|
|
36
37
|
}
|
|
37
38
|
|
|
39
|
+
function insertMilestone(id: string): void {
|
|
40
|
+
const db = _getAdapter()!;
|
|
41
|
+
db.prepare(
|
|
42
|
+
`INSERT INTO milestones (id, title, status, created_at)
|
|
43
|
+
VALUES (:id, :title, 'active', :created_at)`,
|
|
44
|
+
).run({
|
|
45
|
+
":id": id,
|
|
46
|
+
":title": id,
|
|
47
|
+
":created_at": new Date().toISOString(),
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
function writeLegacyLock(base: string, pid: number): void {
|
|
52
|
+
const now = new Date().toISOString();
|
|
53
|
+
writeFileSync(join(base, ".gsd", "auto.lock"), JSON.stringify({
|
|
54
|
+
pid,
|
|
55
|
+
startedAt: now,
|
|
56
|
+
unitType: "execute-task",
|
|
57
|
+
unitId: "M001/S01/T01",
|
|
58
|
+
unitStartedAt: now,
|
|
59
|
+
}));
|
|
60
|
+
}
|
|
61
|
+
|
|
38
62
|
function findDeadPidCandidate(): number {
|
|
39
63
|
const candidates = [99_999, 199_999, 299_999, 399_999];
|
|
40
64
|
for (const pid of candidates) {
|
|
@@ -62,3 +86,41 @@ test("checkRemoteAutoSession clears stale lock state when lock PID is dead", (t)
|
|
|
62
86
|
assert.deepEqual(remote, { running: false });
|
|
63
87
|
assert.equal(readCrashLock(base), null, "stale lock should be cleared by remote session check");
|
|
64
88
|
});
|
|
89
|
+
|
|
90
|
+
test("forceStopAutoRemote escalates a live remote PID and releases worker state", (t) => {
|
|
91
|
+
const base = makeBase();
|
|
92
|
+
t.after(() => cleanup(base));
|
|
93
|
+
|
|
94
|
+
openDatabase(join(base, ".gsd", "gsd.db"));
|
|
95
|
+
const workerId = registerAutoWorker({ projectRootRealpath: normalizeRealPath(base) });
|
|
96
|
+
const pid = 424_242;
|
|
97
|
+
setWorkerPid(workerId, pid);
|
|
98
|
+
insertMilestone("M001");
|
|
99
|
+
writeLegacyLock(base, pid);
|
|
100
|
+
const lease = claimMilestoneLease(workerId, "M001");
|
|
101
|
+
assert.equal(lease.ok, true, "precondition: worker holds a milestone lease");
|
|
102
|
+
|
|
103
|
+
const signals: Array<NodeJS.Signals | 0> = [];
|
|
104
|
+
const originalKill = process.kill;
|
|
105
|
+
process.kill = ((target: number, signal?: NodeJS.Signals | number) => {
|
|
106
|
+
assert.equal(target, pid);
|
|
107
|
+
signals.push((signal ?? 0) as NodeJS.Signals | 0);
|
|
108
|
+
return true;
|
|
109
|
+
}) as typeof process.kill;
|
|
110
|
+
t.after(() => {
|
|
111
|
+
process.kill = originalKill;
|
|
112
|
+
});
|
|
113
|
+
|
|
114
|
+
const result = forceStopAutoRemote(base);
|
|
115
|
+
|
|
116
|
+
assert.deepEqual(result, { found: true, pid });
|
|
117
|
+
assert.ok(signals.includes("SIGTERM"), "force stop should request graceful termination first");
|
|
118
|
+
assert.ok(signals.includes("SIGKILL"), "force stop should escalate when the PID is still alive");
|
|
119
|
+
assert.equal(getAutoWorker(workerId)?.status, "stopping");
|
|
120
|
+
assert.equal(
|
|
121
|
+
getMilestoneLease("M001")?.status,
|
|
122
|
+
"released",
|
|
123
|
+
"force stop should release held milestone leases",
|
|
124
|
+
);
|
|
125
|
+
assert.equal(readCrashLock(base), null, "force stop should remove the visible remote lock");
|
|
126
|
+
});
|
|
@@ -8,7 +8,10 @@ import { join } from "node:path";
|
|
|
8
8
|
|
|
9
9
|
import {
|
|
10
10
|
blockModel,
|
|
11
|
+
blockModelUntil,
|
|
12
|
+
clearTemporaryModelBlocksForTest,
|
|
11
13
|
isModelBlocked,
|
|
14
|
+
isModelTemporarilyUnavailable,
|
|
12
15
|
loadBlockedModels,
|
|
13
16
|
} from "../blocked-models.ts";
|
|
14
17
|
|
|
@@ -96,3 +99,19 @@ test("blocked-models: file created under .gsd/runtime/", () => {
|
|
|
96
99
|
rmSync(base, { recursive: true, force: true });
|
|
97
100
|
}
|
|
98
101
|
});
|
|
102
|
+
|
|
103
|
+
test("blocked-models: temporary rate-limit blocks expire without persisting", () => {
|
|
104
|
+
const base = mkBase();
|
|
105
|
+
try {
|
|
106
|
+
clearTemporaryModelBlocksForTest();
|
|
107
|
+
blockModelUntil(base, "openai-codex", "gpt-5.5", Date.now() + 60_000, "session limit");
|
|
108
|
+
assert.equal(isModelTemporarilyUnavailable(base, "openai-codex", "gpt-5.5"), true);
|
|
109
|
+
assert.equal(loadBlockedModels(base).length, 0, "rate-limit windows must not persist as account blocks");
|
|
110
|
+
|
|
111
|
+
blockModelUntil(base, "openai-codex", "gpt-5.5", Date.now() - 1, "expired");
|
|
112
|
+
assert.equal(isModelTemporarilyUnavailable(base, "openai-codex", "gpt-5.5"), false);
|
|
113
|
+
} finally {
|
|
114
|
+
clearTemporaryModelBlocksForTest();
|
|
115
|
+
rmSync(base, { recursive: true, force: true });
|
|
116
|
+
}
|
|
117
|
+
});
|
|
@@ -238,6 +238,21 @@ test("isAwaitingUserInput does not trigger on thinking-block approval phrases",
|
|
|
238
238
|
assert.equal(shouldPauseForQuestion("discuss-requirements", messages), false);
|
|
239
239
|
});
|
|
240
240
|
|
|
241
|
+
test("isAwaitingUserInput treats plain-text next steps menus as waiting for the user (#454)", () => {
|
|
242
|
+
const messages = [
|
|
243
|
+
{
|
|
244
|
+
role: "assistant",
|
|
245
|
+
content: [
|
|
246
|
+
"Next steps:",
|
|
247
|
+
"1. Walk through the runtime placement check above.",
|
|
248
|
+
"2. Build a release once you're satisfied.",
|
|
249
|
+
"3. Other.",
|
|
250
|
+
].join("\n"),
|
|
251
|
+
},
|
|
252
|
+
];
|
|
253
|
+
assert.equal(isAwaitingUserInput(messages), true);
|
|
254
|
+
});
|
|
255
|
+
|
|
241
256
|
test("isAwaitingUserInput still triggers on text-block question marks when thinking is also present", () => {
|
|
242
257
|
// When thinking + text are both present and the text asks a question, it should still pause.
|
|
243
258
|
const messages = [
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
// Project/App: gsd-pi
|
|
2
|
+
// File Purpose: Doctor git checks treat validation-pass closeout as terminal without SUMMARY.
|
|
3
|
+
|
|
4
|
+
import test from "node:test";
|
|
5
|
+
import assert from "node:assert/strict";
|
|
6
|
+
import { execFileSync } from "node:child_process";
|
|
7
|
+
import { existsSync, mkdirSync, mkdtempSync, rmSync, writeFileSync } from "node:fs";
|
|
8
|
+
import { tmpdir } from "node:os";
|
|
9
|
+
import { join } from "node:path";
|
|
10
|
+
|
|
11
|
+
import { runGSDDoctor } from "../doctor.ts";
|
|
12
|
+
import { openDatabase, insertMilestone, insertSlice, insertAssessment, closeDatabase } from "../gsd-db.js";
|
|
13
|
+
import { createWorktree, worktreePath } from "../worktree-manager.ts";
|
|
14
|
+
|
|
15
|
+
function runGit(args: string[], cwd: string): string {
|
|
16
|
+
return execFileSync("git", args, {
|
|
17
|
+
cwd,
|
|
18
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
19
|
+
encoding: "utf-8",
|
|
20
|
+
}).trim();
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
function makeRepo(): string {
|
|
24
|
+
const base = mkdtempSync(join(tmpdir(), "gsd-doctor-terminal-"));
|
|
25
|
+
runGit(["init", "-b", "main"], base);
|
|
26
|
+
runGit(["config", "user.name", "Test User"], base);
|
|
27
|
+
runGit(["config", "user.email", "test@example.com"], base);
|
|
28
|
+
writeFileSync(join(base, "package.json"), "{\"scripts\":{}}\n", "utf-8");
|
|
29
|
+
runGit(["add", "."], base);
|
|
30
|
+
runGit(["commit", "-m", "chore: init"], base);
|
|
31
|
+
return base;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
test.after(() => {
|
|
35
|
+
closeDatabase();
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
test("doctor flags orphaned worktree for DB-complete milestone without SUMMARY", async (t) => {
|
|
39
|
+
const base = makeRepo();
|
|
40
|
+
t.after(() => rmSync(base, { recursive: true, force: true }));
|
|
41
|
+
|
|
42
|
+
mkdirSync(join(base, ".gsd"), { recursive: true });
|
|
43
|
+
openDatabase(join(base, ".gsd", "gsd.db"));
|
|
44
|
+
insertMilestone({ id: "M008", title: "Done", status: "complete" });
|
|
45
|
+
insertSlice({ id: "S01", milestoneId: "M008", title: "Slice", status: "complete" });
|
|
46
|
+
insertAssessment({
|
|
47
|
+
path: "milestones/M008/M008-VALIDATION.md",
|
|
48
|
+
milestoneId: "M008",
|
|
49
|
+
status: "pass",
|
|
50
|
+
scope: "milestone-validation",
|
|
51
|
+
fullContent: "verdict: pass",
|
|
52
|
+
});
|
|
53
|
+
writeFileSync(
|
|
54
|
+
join(base, ".gsd", "PREFERENCES.md"),
|
|
55
|
+
"---\ngit:\n isolation: worktree\n---\n",
|
|
56
|
+
);
|
|
57
|
+
mkdirSync(join(base, ".gsd", "milestones", "M008"), { recursive: true });
|
|
58
|
+
writeFileSync(
|
|
59
|
+
join(base, ".gsd", "milestones", "M008", "M008-ROADMAP.md"),
|
|
60
|
+
"# M008 Roadmap\n\n- [x] **S01: Slice** `risk:low` `depends:[]`\n",
|
|
61
|
+
);
|
|
62
|
+
|
|
63
|
+
createWorktree(base, "M008", { branch: "milestone/M008" });
|
|
64
|
+
const wtPath = worktreePath(base, "M008");
|
|
65
|
+
assert.ok(existsSync(wtPath), "worktree should exist for the test");
|
|
66
|
+
|
|
67
|
+
const report = await runGSDDoctor(base, { isolationMode: "worktree" });
|
|
68
|
+
|
|
69
|
+
assert.ok(
|
|
70
|
+
report.issues.some((issue) => issue.code === "orphaned_auto_worktree" && issue.unitId === "M008"),
|
|
71
|
+
"doctor should treat DB-complete milestone without SUMMARY as terminal for cleanup",
|
|
72
|
+
);
|
|
73
|
+
});
|
|
@@ -4,10 +4,13 @@
|
|
|
4
4
|
import test from "node:test";
|
|
5
5
|
import assert from "node:assert/strict";
|
|
6
6
|
import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
|
|
7
|
+
import { execFileSync } from "node:child_process";
|
|
7
8
|
import { join } from "node:path";
|
|
8
9
|
import { tmpdir } from "node:os";
|
|
9
10
|
|
|
10
11
|
import { buildGsdHomeModel, showGsdHome } from "../gsd-command-home.ts";
|
|
12
|
+
import { buildIdleMenuSummary, detectIdleMilestoneResidueHint } from "../closeout-wizard.ts";
|
|
13
|
+
import { closeDatabase, insertMilestone, openDatabase } from "../gsd-db.ts";
|
|
11
14
|
import type { GSDState } from "../types.ts";
|
|
12
15
|
|
|
13
16
|
function baseState(overrides: Partial<GSDState> = {}): GSDState {
|
|
@@ -118,6 +121,123 @@ test("/gsd home recommends start or configure after all milestones complete", ()
|
|
|
118
121
|
assert.match(model.summary.join("\n"), /All milestones complete/);
|
|
119
122
|
});
|
|
120
123
|
|
|
124
|
+
test("/gsd home recommends fix or recover when milestone git residue is stranded", () => {
|
|
125
|
+
const model = buildGsdHomeModel(baseState({
|
|
126
|
+
activeMilestone: null,
|
|
127
|
+
activeSlice: null,
|
|
128
|
+
activeTask: null,
|
|
129
|
+
phase: "complete",
|
|
130
|
+
nextAction: "All milestones complete.",
|
|
131
|
+
}), {
|
|
132
|
+
strandedQuick: null,
|
|
133
|
+
unmergedMilestones: [],
|
|
134
|
+
idleResidueHint: {
|
|
135
|
+
milestoneIds: ["M008"],
|
|
136
|
+
message:
|
|
137
|
+
"Stranded milestone git residue detected (M008: worktree dir and/or milestone/* branch). " +
|
|
138
|
+
"Run /gsd dispatch complete-milestone M008 or /gsd status to recover closeout before starting new work.",
|
|
139
|
+
},
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
assert.equal(action(model, "fix_recover").recommended, true);
|
|
143
|
+
assert.equal(action(model, "fix_recover").enabled, true);
|
|
144
|
+
assert.match(model.summary[0], /Stranded milestone git residue/);
|
|
145
|
+
assert.equal(action(model, "continue_step").enabled, false);
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
test("detectIdleMilestoneResidueHint reports missing workflow database in a git repo", () => {
|
|
149
|
+
const base = mkdtempSync(join(tmpdir(), "gsd-home-residue-"));
|
|
150
|
+
try {
|
|
151
|
+
execFileSync("git", ["init", "-b", "main"], { cwd: base, stdio: "ignore" });
|
|
152
|
+
const hint = detectIdleMilestoneResidueHint(base);
|
|
153
|
+
assert.ok(hint);
|
|
154
|
+
assert.match(hint!.message, /\.gsd\/gsd\.db/);
|
|
155
|
+
} finally {
|
|
156
|
+
rmSync(base, { recursive: true, force: true });
|
|
157
|
+
}
|
|
158
|
+
});
|
|
159
|
+
|
|
160
|
+
test("detectIdleMilestoneResidueHint matches unique-format milestone ids (M###-abc123)", () => {
|
|
161
|
+
const base = mkdtempSync(join(tmpdir(), "gsd-home-residue-unique-"));
|
|
162
|
+
try {
|
|
163
|
+
execFileSync("git", ["init", "-b", "main"], { cwd: base, stdio: "ignore" });
|
|
164
|
+
mkdirSync(join(base, ".gsd-worktrees", "M042-abc123"), { recursive: true });
|
|
165
|
+
mkdirSync(join(base, ".gsd"), { recursive: true });
|
|
166
|
+
|
|
167
|
+
openDatabase(join(base, ".gsd", "gsd.db"));
|
|
168
|
+
try {
|
|
169
|
+
// Closed milestone with lingering worktree dir — classic residue.
|
|
170
|
+
insertMilestone({ id: "M042-abc123", title: "Closed Unique", status: "complete" });
|
|
171
|
+
const hint = detectIdleMilestoneResidueHint(base);
|
|
172
|
+
assert.ok(hint, "unique-format closed milestone with worktree should be detected");
|
|
173
|
+
assert.deepEqual(hint!.milestoneIds, ["M042-abc123"]);
|
|
174
|
+
assert.match(hint!.message, /M042-abc123/);
|
|
175
|
+
} finally {
|
|
176
|
+
closeDatabase();
|
|
177
|
+
}
|
|
178
|
+
} finally {
|
|
179
|
+
rmSync(base, { recursive: true, force: true });
|
|
180
|
+
}
|
|
181
|
+
});
|
|
182
|
+
|
|
183
|
+
test("detectIdleMilestoneResidueHint ignores in-flight milestones with worktree artifacts", () => {
|
|
184
|
+
const base = mkdtempSync(join(tmpdir(), "gsd-home-residue-active-"));
|
|
185
|
+
try {
|
|
186
|
+
execFileSync("git", ["init", "-b", "main"], { cwd: base, stdio: "ignore" });
|
|
187
|
+
mkdirSync(join(base, ".gsd-worktrees", "M001"), { recursive: true });
|
|
188
|
+
mkdirSync(join(base, ".gsd-worktrees", "M002-abc123"), { recursive: true });
|
|
189
|
+
mkdirSync(join(base, ".gsd"), { recursive: true });
|
|
190
|
+
|
|
191
|
+
openDatabase(join(base, ".gsd", "gsd.db"));
|
|
192
|
+
try {
|
|
193
|
+
// Active and pending milestones must not be flagged as stranded residue.
|
|
194
|
+
insertMilestone({ id: "M001", title: "Active milestone", status: "active" });
|
|
195
|
+
insertMilestone({ id: "M002-abc123", title: "Pending milestone", status: "pending" });
|
|
196
|
+
const hint = detectIdleMilestoneResidueHint(base);
|
|
197
|
+
assert.equal(hint, null, "active/pending milestone worktrees must not be classified as residue");
|
|
198
|
+
} finally {
|
|
199
|
+
closeDatabase();
|
|
200
|
+
}
|
|
201
|
+
} finally {
|
|
202
|
+
rmSync(base, { recursive: true, force: true });
|
|
203
|
+
}
|
|
204
|
+
});
|
|
205
|
+
|
|
206
|
+
test("buildIdleMenuSummary surfaces idle residue hint even when phase is complete", () => {
|
|
207
|
+
const summary = buildIdleMenuSummary(
|
|
208
|
+
{
|
|
209
|
+
activeMilestone: null,
|
|
210
|
+
activeSlice: null,
|
|
211
|
+
activeTask: null,
|
|
212
|
+
phase: "complete",
|
|
213
|
+
recentDecisions: [],
|
|
214
|
+
blockers: [],
|
|
215
|
+
nextAction: "All milestones complete.",
|
|
216
|
+
lastCompletedMilestone: { id: "M001", title: "Menu Cleanup" },
|
|
217
|
+
registry: [{ id: "M001", title: "Menu Cleanup", status: "complete" }],
|
|
218
|
+
requirements: { active: 0, validated: 0, deferred: 0, outOfScope: 0, blocked: 0, total: 0 },
|
|
219
|
+
progress: {
|
|
220
|
+
milestones: { done: 1, total: 1 },
|
|
221
|
+
slices: { done: 0, total: 0 },
|
|
222
|
+
tasks: { done: 0, total: 0 },
|
|
223
|
+
},
|
|
224
|
+
},
|
|
225
|
+
{
|
|
226
|
+
strandedQuick: null,
|
|
227
|
+
unmergedMilestones: [],
|
|
228
|
+
idleResidueHint: {
|
|
229
|
+
milestoneIds: ["M008"],
|
|
230
|
+
message:
|
|
231
|
+
"Stranded milestone git residue detected (M008: worktree dir and/or milestone/* branch). " +
|
|
232
|
+
"Run /gsd dispatch complete-milestone M008 or /gsd status to recover closeout before starting new work.",
|
|
233
|
+
},
|
|
234
|
+
},
|
|
235
|
+
);
|
|
236
|
+
|
|
237
|
+
assert.match(summary[0] ?? "", /Stranded milestone git residue/);
|
|
238
|
+
assert.doesNotMatch(summary.join("\n"), /All milestones complete/);
|
|
239
|
+
});
|
|
240
|
+
|
|
121
241
|
test("showGsdHome renders the five-slot home text without an interactive TUI", async () => {
|
|
122
242
|
const base = mkdtempSync(join(tmpdir(), "gsd-home-"));
|
|
123
243
|
const notifications: Array<{ message: string; level: string }> = [];
|
|
@@ -77,12 +77,8 @@ test("guided dispatch passes the explicit project root through model and compati
|
|
|
77
77
|
seen.modelRoot = projectRoot;
|
|
78
78
|
return { routing: null, appliedModel: null };
|
|
79
79
|
},
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
_requiredTools: string[],
|
|
83
|
-
options?: { projectRoot?: string },
|
|
84
|
-
) => {
|
|
85
|
-
seen.compatibilityRoot = options?.projectRoot ?? "";
|
|
80
|
+
getDispatchReadinessError: (input: { projectRoot?: string }) => {
|
|
81
|
+
seen.compatibilityRoot = input.projectRoot ?? "";
|
|
86
82
|
return null;
|
|
87
83
|
},
|
|
88
84
|
},
|