@opengsd/gsd-pi 1.0.2-dev.5961fbf → 1.0.2-dev.5f7864c
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/README.md +63 -12
- package/dist/onboarding.js +22 -3
- package/dist/resource-loader.d.ts +2 -0
- package/dist/resource-loader.js +18 -1
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/extensions/context7/index.js +12 -2
- package/dist/resources/extensions/get-secrets-from-user.js +16 -16
- package/dist/resources/extensions/google-cli/index.js +30 -0
- package/dist/resources/extensions/google-cli/models.js +55 -0
- package/dist/resources/extensions/google-cli/package.json +11 -0
- package/dist/resources/extensions/google-cli/readiness.js +12 -0
- package/dist/resources/extensions/google-cli/stream-adapter.js +191 -0
- package/dist/resources/extensions/gsd/auto/session.js +3 -0
- package/dist/resources/extensions/gsd/auto-start.js +232 -49
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +4 -3
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +17 -15
- package/dist/resources/extensions/gsd/closeout-recovery.js +7 -1
- package/dist/resources/extensions/gsd/commands/handlers/auto.js +9 -1
- package/dist/resources/extensions/gsd/commands-handlers.js +3 -0
- package/dist/resources/extensions/gsd/commands-usage.js +105 -1
- package/dist/resources/extensions/gsd/config-overlay.js +20 -14
- package/dist/resources/extensions/gsd/context-overlay.js +22 -16
- package/dist/resources/extensions/gsd/dashboard-overlay.js +10 -23
- package/dist/resources/extensions/gsd/doctor-providers.js +54 -24
- package/dist/resources/extensions/gsd/git-conflict-state.js +26 -1
- package/dist/resources/extensions/gsd/guided-flow.js +1 -1
- package/dist/resources/extensions/gsd/key-manager.js +45 -13
- package/dist/resources/extensions/gsd/notification-overlay.js +8 -9
- package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +15 -13
- package/dist/resources/extensions/gsd/prompt-loader.js +2 -0
- package/dist/resources/extensions/gsd/prompts/discuss.md +4 -2
- package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +2 -0
- package/dist/resources/extensions/gsd/queue-reorder-ui.js +28 -18
- package/dist/resources/extensions/gsd/tools/complete-task.js +9 -0
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +40 -1
- package/dist/resources/extensions/gsd/tui/render-kit.js +51 -0
- package/dist/resources/extensions/gsd/vision-ask.js +22 -0
- package/dist/resources/extensions/gsd/visualizer-overlay.js +8 -36
- package/dist/resources/extensions/gsd/worktree-lifecycle.js +24 -3
- package/dist/resources/extensions/search-the-web/native-search.js +57 -8
- package/dist/resources/extensions/shared/confirm-ui.js +9 -6
- package/dist/resources/extensions/shared/dialog-frame.js +42 -0
- package/dist/resources/extensions/shared/interview-ui.js +42 -30
- package/dist/resources/extensions/shared/next-action-ui.js +6 -6
- package/dist/resources/shared/package-manager-detection.js +36 -0
- package/dist/update-check.d.ts +6 -2
- package/dist/update-check.js +7 -3
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +6 -6
- 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/api/update/route.js +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 +6 -6
- package/dist/web/standalone/.next/server/chunks/1834.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/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/dist/modes/interactive/components/dialog-container.d.ts +12 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/dialog-container.d.ts.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/dialog-container.js +45 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/dialog-container.js.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-editor.d.ts +3 -2
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-editor.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-editor.js +11 -11
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-editor.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-input.d.ts +3 -3
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-input.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-input.js +13 -11
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-input.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-selector.d.ts +3 -3
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-selector.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-selector.js +12 -10
- package/packages/gsd-agent-modes/dist/modes/interactive/components/extension-selector.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/index.d.ts +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/index.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/index.js +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/index.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/login-dialog.d.ts +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/login-dialog.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/login-dialog.js +2 -2
- package/packages/gsd-agent-modes/dist/modes/interactive/components/login-dialog.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/oauth-selector.d.ts +6 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/oauth-selector.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/oauth-selector.js +9 -6
- package/packages/gsd-agent-modes/dist/modes/interactive/components/oauth-selector.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/transcript-design.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/transcript-design.js +0 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/transcript-design.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.d.ts +3 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js +144 -2
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-session.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-session.js +2 -14
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-session.js.map +1 -1
- package/packages/gsd-agent-modes/package.json +7 -7
- package/packages/mcp-server/dist/workflow-tools.js +1 -1
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +3 -3
- package/packages/native/package.json +1 -1
- package/packages/pi-agent-core/dist/agent-loop.js +13 -13
- package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-ai/dist/models.generated.d.ts +57 -17
- package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.js +64 -28
- package/packages/pi-ai/dist/models.generated.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.js +50 -0
- package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
- package/packages/pi-ai/dist/types.d.ts +2 -0
- package/packages/pi-ai/dist/types.d.ts.map +1 -1
- package/packages/pi-ai/dist/types.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 +1 -1
- package/packages/rpc-client/package.json +2 -2
- package/pkg/package.json +1 -1
- package/scripts/install/detect-existing.js +17 -3
- package/scripts/install/npm-global.js +103 -33
- package/scripts/install.js +1 -0
- package/src/resources/extensions/context7/index.ts +15 -2
- package/src/resources/extensions/get-secrets-from-user.ts +17 -16
- package/src/resources/extensions/google-cli/index.ts +34 -0
- package/src/resources/extensions/google-cli/models.ts +57 -0
- package/src/resources/extensions/google-cli/package.json +11 -0
- package/src/resources/extensions/google-cli/readiness.ts +15 -0
- package/src/resources/extensions/google-cli/stream-adapter.ts +245 -0
- package/src/resources/extensions/gsd/auto/session.ts +3 -0
- package/src/resources/extensions/gsd/auto-start.ts +307 -56
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +4 -3
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +22 -15
- package/src/resources/extensions/gsd/closeout-recovery.ts +6 -1
- package/src/resources/extensions/gsd/commands/handlers/auto.ts +9 -1
- package/src/resources/extensions/gsd/commands-handlers.ts +2 -0
- package/src/resources/extensions/gsd/commands-usage.ts +110 -5
- package/src/resources/extensions/gsd/config-overlay.ts +19 -16
- package/src/resources/extensions/gsd/context-overlay.ts +24 -19
- package/src/resources/extensions/gsd/dashboard-overlay.ts +14 -27
- package/src/resources/extensions/gsd/doctor-providers.ts +55 -27
- package/src/resources/extensions/gsd/git-conflict-state.ts +25 -1
- package/src/resources/extensions/gsd/guided-flow.ts +1 -1
- package/src/resources/extensions/gsd/key-manager.ts +57 -14
- package/src/resources/extensions/gsd/notification-overlay.ts +12 -11
- package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +16 -12
- package/src/resources/extensions/gsd/prompt-loader.ts +2 -0
- package/src/resources/extensions/gsd/prompts/discuss.md +4 -2
- package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +2 -0
- package/src/resources/extensions/gsd/queue-reorder-ui.ts +29 -20
- package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +436 -0
- package/src/resources/extensions/gsd/tests/closeout-recovery.test.ts +15 -0
- package/src/resources/extensions/gsd/tests/collect-from-manifest.test.ts +31 -0
- package/src/resources/extensions/gsd/tests/commands-context.test.ts +5 -3
- package/src/resources/extensions/gsd/tests/commands-dispatcher-workspace-git.test.ts +15 -2
- package/src/resources/extensions/gsd/tests/commands-usage.test.ts +97 -0
- package/src/resources/extensions/gsd/tests/context-chart.test.ts +9 -0
- package/src/resources/extensions/gsd/tests/dashboard-overlay.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/discuss-prompt.test.ts +4 -2
- package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +105 -0
- package/src/resources/extensions/gsd/tests/guided-discuss-milestone-prompt-rendering.test.ts +6 -0
- package/src/resources/extensions/gsd/tests/key-manager.test.ts +23 -4
- package/src/resources/extensions/gsd/tests/notification-overlay.test.ts +6 -1
- package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +70 -10
- package/src/resources/extensions/gsd/tests/parallel-monitor-overlay.test.ts +7 -1
- package/src/resources/extensions/gsd/tests/queue-reorder-ui.test.ts +46 -0
- package/src/resources/extensions/gsd/tests/show-config-command.test.ts +4 -0
- package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +13 -2
- package/src/resources/extensions/gsd/tests/tool-param-optionality.test.ts +24 -1
- package/src/resources/extensions/gsd/tests/tui-border-assertions.ts +28 -0
- package/src/resources/extensions/gsd/tests/tui-render-kit.test.ts +14 -0
- package/src/resources/extensions/gsd/tests/vision-ask.test.ts +23 -0
- package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +6 -1
- package/src/resources/extensions/gsd/tests/workflow-mcp-auto-prep.test.ts +60 -0
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +54 -0
- package/src/resources/extensions/gsd/tests/workspace-git-preflight.test.ts +16 -1
- package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +28 -0
- package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +45 -1
- package/src/resources/extensions/gsd/tools/complete-task.ts +9 -0
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +56 -4
- package/src/resources/extensions/gsd/tui/render-kit.ts +82 -0
- package/src/resources/extensions/gsd/vision-ask.ts +28 -0
- package/src/resources/extensions/gsd/visualizer-overlay.ts +12 -40
- package/src/resources/extensions/gsd/worktree-lifecycle.ts +37 -2
- package/src/resources/extensions/search-the-web/native-search.ts +60 -8
- package/src/resources/extensions/shared/confirm-ui.ts +8 -12
- package/src/resources/extensions/shared/dialog-frame.ts +71 -0
- package/src/resources/extensions/shared/interview-ui.ts +43 -42
- package/src/resources/extensions/shared/next-action-ui.ts +6 -6
- package/src/resources/extensions/shared/tests/confirm-ui.test.ts +57 -0
- package/src/resources/extensions/shared/tests/interview-ui-border.test.ts +163 -0
- package/src/resources/extensions/shared/tests/next-action-ui-hasui.test.ts +55 -0
- package/src/resources/shared/package-manager-detection.ts +39 -0
- /package/dist/web/standalone/.next/static/{spUYLkQXoHJyxYOMH9VQy → IjxvcC7sl_MHNKXsUZrAy}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{spUYLkQXoHJyxYOMH9VQy → IjxvcC7sl_MHNKXsUZrAy}/_ssgManifest.js +0 -0
|
@@ -8,6 +8,7 @@ import {
|
|
|
8
8
|
scanSessionTokenTotals,
|
|
9
9
|
handleUsage,
|
|
10
10
|
} from "../commands-usage.ts";
|
|
11
|
+
import { assertFullOuterBorder } from "./tui-border-assertions.ts";
|
|
11
12
|
|
|
12
13
|
const TS = 1;
|
|
13
14
|
|
|
@@ -108,3 +109,99 @@ test("handleUsage emits JSON when --json is passed", async () => {
|
|
|
108
109
|
assert.equal(parsed.contextUsage.tokens, 10_000);
|
|
109
110
|
assert.equal(parsed.sessionTotals.input, 0);
|
|
110
111
|
});
|
|
112
|
+
|
|
113
|
+
test("handleUsage renders interactive usage output inside a full border", async () => {
|
|
114
|
+
let renderFn: ((width: number) => string[]) | undefined;
|
|
115
|
+
const messages: string[] = [];
|
|
116
|
+
const ctx = {
|
|
117
|
+
hasUI: true,
|
|
118
|
+
model: { provider: "claude-code", id: "claude-sonnet-4-6", contextWindow: 200_000 },
|
|
119
|
+
getContextUsage: () => ({ tokens: 10_000, contextWindow: 200_000, percent: 5 }),
|
|
120
|
+
sessionManager: { getEntries: () => [] },
|
|
121
|
+
ui: {
|
|
122
|
+
custom: async (factory: any) => {
|
|
123
|
+
const theme = {
|
|
124
|
+
fg: (_color: string, text: string) => text,
|
|
125
|
+
bold: (text: string) => text,
|
|
126
|
+
};
|
|
127
|
+
const component = factory({ requestRender: () => {} }, theme, {}, () => {});
|
|
128
|
+
renderFn = component.render;
|
|
129
|
+
return true;
|
|
130
|
+
},
|
|
131
|
+
notify(message: string) {
|
|
132
|
+
messages.push(message);
|
|
133
|
+
},
|
|
134
|
+
},
|
|
135
|
+
};
|
|
136
|
+
|
|
137
|
+
await handleUsage("", ctx as any);
|
|
138
|
+
|
|
139
|
+
assert.equal(messages.length, 0, "interactive usage should use the dialog instead of notify");
|
|
140
|
+
assert.ok(renderFn, "render function should have been captured");
|
|
141
|
+
const lines = renderFn!(80);
|
|
142
|
+
assertFullOuterBorder(lines, 80);
|
|
143
|
+
assert.match(lines.join("\n"), /claude-code\/claude-sonnet-4-6/);
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
test("handleUsage keeps short-terminal usage dialog scrollable", async () => {
|
|
147
|
+
const originalRowsDescriptor = Object.getOwnPropertyDescriptor(process.stdout, "rows");
|
|
148
|
+
Object.defineProperty(process.stdout, "rows", { value: 10, configurable: true });
|
|
149
|
+
|
|
150
|
+
try {
|
|
151
|
+
let component: { render(width: number): string[]; handleInput(data: string): void } | undefined;
|
|
152
|
+
let closed = false;
|
|
153
|
+
const ctx = {
|
|
154
|
+
hasUI: true,
|
|
155
|
+
model: { provider: "claude-code", id: "claude-sonnet-4-6", contextWindow: 200_000 },
|
|
156
|
+
getContextUsage: () => ({ tokens: 10_000, contextWindow: 200_000, percent: 5 }),
|
|
157
|
+
sessionManager: {
|
|
158
|
+
getEntries: () => sessionEntries({
|
|
159
|
+
role: "assistant",
|
|
160
|
+
usage: {
|
|
161
|
+
input: 1000,
|
|
162
|
+
output: 200,
|
|
163
|
+
cacheRead: 500,
|
|
164
|
+
cacheWrite: 100,
|
|
165
|
+
totalTokens: 1800,
|
|
166
|
+
cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0.05 },
|
|
167
|
+
},
|
|
168
|
+
content: [{ type: "toolCall", id: "tc-1", name: "read", arguments: {} }],
|
|
169
|
+
timestamp: TS,
|
|
170
|
+
}),
|
|
171
|
+
},
|
|
172
|
+
ui: {
|
|
173
|
+
custom: async (factory: any) => {
|
|
174
|
+
const theme = {
|
|
175
|
+
fg: (_color: string, text: string) => text,
|
|
176
|
+
bold: (text: string) => text,
|
|
177
|
+
};
|
|
178
|
+
component = factory({ requestRender: () => {} }, theme, {}, () => {
|
|
179
|
+
closed = true;
|
|
180
|
+
});
|
|
181
|
+
return true;
|
|
182
|
+
},
|
|
183
|
+
notify() {},
|
|
184
|
+
},
|
|
185
|
+
};
|
|
186
|
+
|
|
187
|
+
await handleUsage("", ctx as any);
|
|
188
|
+
|
|
189
|
+
assert.ok(component, "usage dialog should render via custom UI");
|
|
190
|
+
const initialLines = component.render(80);
|
|
191
|
+
assert.ok(initialLines.length <= 8, `usage dialog should fit 80% terminal height, got ${initialLines.length}`);
|
|
192
|
+
|
|
193
|
+
for (let i = 0; i < 10; i++) component.handleInput("\u001b[B");
|
|
194
|
+
|
|
195
|
+
assert.equal(closed, false, "scroll keys should not close a scrollable usage dialog");
|
|
196
|
+
assert.match(component.render(80).join("\n"), /Tool calls: 1/);
|
|
197
|
+
|
|
198
|
+
component.handleInput("x");
|
|
199
|
+
assert.equal(closed, true, "non-scroll keys should close the usage dialog");
|
|
200
|
+
} finally {
|
|
201
|
+
if (originalRowsDescriptor) {
|
|
202
|
+
Object.defineProperty(process.stdout, "rows", originalRowsDescriptor);
|
|
203
|
+
} else {
|
|
204
|
+
delete (process.stdout as { rows?: number }).rows;
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
});
|
|
@@ -10,6 +10,7 @@ import type { ContextBreakdownReport } from "../commands-context.ts";
|
|
|
10
10
|
import { buildContextChartHtml, writeContextChartHtml } from "../context-chart-html.ts";
|
|
11
11
|
import { formatContextChartText, getContextChartTotals } from "../context-overlay.ts";
|
|
12
12
|
import { buildContextBreakdown } from "../commands-context.ts";
|
|
13
|
+
import { assertFullOuterBorder } from "./tui-border-assertions.ts";
|
|
13
14
|
|
|
14
15
|
const TS = 1;
|
|
15
16
|
|
|
@@ -103,3 +104,11 @@ test("formatContextChartText includes chart sections", () => {
|
|
|
103
104
|
assert.match(text, /Conversation/);
|
|
104
105
|
assert.match(text, /█/);
|
|
105
106
|
});
|
|
107
|
+
|
|
108
|
+
test("formatContextChartText renders the context dialog inside a full border", () => {
|
|
109
|
+
const lines = formatContextChartText(SAMPLE_REPORT, 80).split("\n");
|
|
110
|
+
assertFullOuterBorder(lines, 80);
|
|
111
|
+
assert.match(lines[0] ?? "", /^╭─ Context Breakdown /);
|
|
112
|
+
assert.ok(lines.some((line) => line.startsWith("│")), "body rows should have side borders");
|
|
113
|
+
assert.match(lines.at(-1) ?? "", /^╰─+╯$/);
|
|
114
|
+
});
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* GSD dashboard overlay dialog chrome tests.
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import test from "node:test";
|
|
6
|
+
import assert from "node:assert/strict";
|
|
7
|
+
|
|
8
|
+
import { GSDDashboardOverlay } from "../dashboard-overlay.ts";
|
|
9
|
+
import { assertFullOuterBorder } from "./tui-border-assertions.ts";
|
|
10
|
+
|
|
11
|
+
const fakeTheme = {
|
|
12
|
+
fg: (_color: string, text: string) => text,
|
|
13
|
+
bold: (text: string) => text,
|
|
14
|
+
};
|
|
15
|
+
|
|
16
|
+
test("GSDDashboardOverlay renders inside the shared full border", (t) => {
|
|
17
|
+
const overlay = new GSDDashboardOverlay({ requestRender() {} }, fakeTheme as any, () => {});
|
|
18
|
+
t.after(() => overlay.dispose());
|
|
19
|
+
|
|
20
|
+
const lines = overlay.render(100);
|
|
21
|
+
assertFullOuterBorder(lines, 100);
|
|
22
|
+
assert.match(lines[0] ?? "", /^╭─ GSD Dashboard /);
|
|
23
|
+
assert.ok(lines.some((line) => line.startsWith("│")), "body rows should have side borders");
|
|
24
|
+
assert.match(lines.at(-1) ?? "", /^╰─+╯$/);
|
|
25
|
+
});
|
|
@@ -9,7 +9,9 @@ const discussPrompt = readFileSync(promptPath, "utf-8");
|
|
|
9
9
|
test("discuss prompt: resilient vision framing", () => {
|
|
10
10
|
const hardenedPattern = /Say exactly:\s*"What's the vision\?"/;
|
|
11
11
|
assert.ok(!hardenedPattern.test(discussPrompt), "prompt no longer uses exact-verbosity lock");
|
|
12
|
-
assert.ok(discussPrompt.includes('Ask
|
|
12
|
+
assert.ok(discussPrompt.includes('Ask exactly this once: "{{visionAsk}}"'), "prompt asks the injected vision opener exactly once");
|
|
13
|
+
assert.ok(discussPrompt.includes("The opener is intentionally variable"), "prompt documents variable opener voice");
|
|
13
14
|
assert.ok(discussPrompt.includes("Special handling"), "prompt documents special handling");
|
|
14
|
-
assert.ok(discussPrompt.includes(
|
|
15
|
+
assert.ok(discussPrompt.includes("instead of repeating the opener"), "prompt forbids repeating the opener");
|
|
16
|
+
assert.ok(!discussPrompt.includes('"What\'s the vision?"'), "prompt no longer freezes the old opener");
|
|
15
17
|
});
|
|
@@ -702,6 +702,48 @@ test("runProviderChecks reports ok for claude-code without any API key", () => {
|
|
|
702
702
|
rmSync(tmpHome, { recursive: true, force: true });
|
|
703
703
|
});
|
|
704
704
|
|
|
705
|
+
test("runProviderChecks reports errors for required Google CLI providers missing from PATH", () => {
|
|
706
|
+
const scenarios = [
|
|
707
|
+
{ provider: "google-gemini-cli", label: "Google Gemini CLI", model: "gemini-2.5-pro" },
|
|
708
|
+
{ provider: "google-antigravity", label: "Antigravity", model: "default" },
|
|
709
|
+
];
|
|
710
|
+
|
|
711
|
+
for (const { provider, label, model } of scenarios) {
|
|
712
|
+
const repo = realpathSync(mkdtempSync(join(tmpdir(), `gsd-providers-${provider}-repo-`)));
|
|
713
|
+
mkdirSync(join(repo, ".gsd"), { recursive: true });
|
|
714
|
+
writeFileSync(
|
|
715
|
+
join(repo, ".gsd", "PREFERENCES.md"),
|
|
716
|
+
[
|
|
717
|
+
"---",
|
|
718
|
+
"models:",
|
|
719
|
+
" execution:",
|
|
720
|
+
` model: ${model}`,
|
|
721
|
+
` provider: ${provider}`,
|
|
722
|
+
"---",
|
|
723
|
+
"",
|
|
724
|
+
].join("\n"),
|
|
725
|
+
);
|
|
726
|
+
|
|
727
|
+
const tmpHome = realpathSync(mkdtempSync(join(tmpdir(), `gsd-providers-${provider}-home-`)));
|
|
728
|
+
|
|
729
|
+
withEnv({
|
|
730
|
+
HOME: tmpHome,
|
|
731
|
+
PATH: tmpHome,
|
|
732
|
+
}, () => {
|
|
733
|
+
withCwd(repo, () => {
|
|
734
|
+
const results = runProviderChecks();
|
|
735
|
+
const cli = results.find(r => r.name === provider);
|
|
736
|
+
assert.ok(cli, `${provider} result should exist`);
|
|
737
|
+
assert.equal(cli!.status, "error", `${provider} should error when the CLI binary is missing`);
|
|
738
|
+
assert.ok(cli!.detail?.includes(label), "should explain which CLI must be installed");
|
|
739
|
+
});
|
|
740
|
+
});
|
|
741
|
+
|
|
742
|
+
rmSync(repo, { recursive: true, force: true });
|
|
743
|
+
rmSync(tmpHome, { recursive: true, force: true });
|
|
744
|
+
}
|
|
745
|
+
});
|
|
746
|
+
|
|
705
747
|
test("runProviderChecks reports ok for Anthropic via claude-code binary in PATH", () => {
|
|
706
748
|
// Simulate a user who has no Anthropic API key but has the claude CLI installed.
|
|
707
749
|
// Their PREFERENCES use a claude model without an explicit provider, so the doctor
|
|
@@ -736,6 +778,69 @@ test("runProviderChecks reports ok for Anthropic via claude-code binary in PATH"
|
|
|
736
778
|
});
|
|
737
779
|
});
|
|
738
780
|
|
|
781
|
+
test("runProviderChecks ignores external CLI auth sentinels when the CLI is missing", () => {
|
|
782
|
+
const scenarios = [
|
|
783
|
+
{
|
|
784
|
+
requiredProvider: "anthropic",
|
|
785
|
+
routeProvider: "claude-code",
|
|
786
|
+
model: "claude-sonnet-4-6",
|
|
787
|
+
env: {
|
|
788
|
+
ANTHROPIC_API_KEY: undefined,
|
|
789
|
+
ANTHROPIC_OAUTH_TOKEN: undefined,
|
|
790
|
+
COPILOT_GITHUB_TOKEN: undefined,
|
|
791
|
+
GH_TOKEN: undefined,
|
|
792
|
+
GITHUB_TOKEN: undefined,
|
|
793
|
+
},
|
|
794
|
+
},
|
|
795
|
+
{
|
|
796
|
+
requiredProvider: "google",
|
|
797
|
+
routeProvider: "google-gemini-cli",
|
|
798
|
+
model: "gemini-2.5-pro",
|
|
799
|
+
env: {
|
|
800
|
+
GEMINI_API_KEY: undefined,
|
|
801
|
+
GOOGLE_API_KEY: undefined,
|
|
802
|
+
},
|
|
803
|
+
},
|
|
804
|
+
];
|
|
805
|
+
|
|
806
|
+
for (const { requiredProvider, routeProvider, model, env } of scenarios) {
|
|
807
|
+
const repo = realpathSync(mkdtempSync(join(tmpdir(), `gsd-providers-${routeProvider}-sentinel-repo-`)));
|
|
808
|
+
const tmpHome = realpathSync(mkdtempSync(join(tmpdir(), `gsd-providers-${routeProvider}-sentinel-home-`)));
|
|
809
|
+
const agentDir = join(tmpHome, ".gsd", "agent");
|
|
810
|
+
mkdirSync(join(repo, ".gsd"), { recursive: true });
|
|
811
|
+
mkdirSync(agentDir, { recursive: true });
|
|
812
|
+
writeFileSync(
|
|
813
|
+
join(repo, ".gsd", "PREFERENCES.md"),
|
|
814
|
+
[
|
|
815
|
+
"---",
|
|
816
|
+
"models:",
|
|
817
|
+
` execution: ${model}`,
|
|
818
|
+
"---",
|
|
819
|
+
"",
|
|
820
|
+
].join("\n"),
|
|
821
|
+
);
|
|
822
|
+
writeFileSync(join(agentDir, "auth.json"), JSON.stringify({
|
|
823
|
+
[routeProvider]: { type: "api_key", key: "cli" },
|
|
824
|
+
}));
|
|
825
|
+
|
|
826
|
+
withEnv({
|
|
827
|
+
...env,
|
|
828
|
+
HOME: tmpHome,
|
|
829
|
+
PATH: tmpHome,
|
|
830
|
+
}, () => {
|
|
831
|
+
withCwd(repo, () => {
|
|
832
|
+
const results = runProviderChecks();
|
|
833
|
+
const provider = results.find(r => r.name === requiredProvider);
|
|
834
|
+
assert.ok(provider, `${requiredProvider} result should exist`);
|
|
835
|
+
assert.equal(provider!.status, "error", `${routeProvider} sentinel should not satisfy ${requiredProvider}`);
|
|
836
|
+
});
|
|
837
|
+
});
|
|
838
|
+
|
|
839
|
+
rmSync(repo, { recursive: true, force: true });
|
|
840
|
+
rmSync(tmpHome, { recursive: true, force: true });
|
|
841
|
+
}
|
|
842
|
+
});
|
|
843
|
+
|
|
739
844
|
test("runProviderChecks detects claude.cmd in PATH on Windows (#4503)", { skip: process.platform !== "win32" }, () => {
|
|
740
845
|
const tmpHome = realpathSync(mkdtempSync(join(tmpdir(), "gsd-providers-cc-win-route-home-")));
|
|
741
846
|
const binDir = join(tmpHome, "bin");
|
package/src/resources/extensions/gsd/tests/guided-discuss-milestone-prompt-rendering.test.ts
CHANGED
|
@@ -6,6 +6,7 @@ import assert from "node:assert/strict";
|
|
|
6
6
|
import { mkdtempSync, rmSync } from "node:fs";
|
|
7
7
|
import { tmpdir } from "node:os";
|
|
8
8
|
import { join } from "node:path";
|
|
9
|
+
import { VISION_ASK_VARIANTS } from "../vision-ask.ts";
|
|
9
10
|
|
|
10
11
|
test("guided milestone prompt renders compact interview and context guidance", async (t) => {
|
|
11
12
|
const previousGsdHome = process.env.GSD_HOME;
|
|
@@ -31,6 +32,11 @@ test("guided milestone prompt renders compact interview and context guidance", a
|
|
|
31
32
|
|
|
32
33
|
assert.match(prompt, /M001 context written/);
|
|
33
34
|
assert.match(prompt, /Project Shape/);
|
|
35
|
+
assert.ok(
|
|
36
|
+
VISION_ASK_VARIANTS.some((opener) => prompt.includes(opener)),
|
|
37
|
+
"prompt should render a conversational opener variant",
|
|
38
|
+
);
|
|
39
|
+
assert.doesNotMatch(prompt, /\{\{visionAsk\}\}/);
|
|
34
40
|
assert.match(prompt, /default to `complex`/i);
|
|
35
41
|
assert.match(prompt, /3 or 4 concrete, researched options/);
|
|
36
42
|
assert.match(prompt, /"Other — let me discuss"/);
|
|
@@ -10,9 +10,10 @@ import {
|
|
|
10
10
|
formatKeyDashboard,
|
|
11
11
|
formatTestResults,
|
|
12
12
|
runKeyDoctor,
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
13
|
+
formatDoctorFindings,
|
|
14
|
+
PROVIDER_REGISTRY,
|
|
15
|
+
getProviderAuthMode,
|
|
16
|
+
} from "../key-manager.ts";
|
|
16
17
|
|
|
17
18
|
function makeAuth(data: Record<string, any> = {}): AuthStorage {
|
|
18
19
|
return AuthStorage.inMemory(data);
|
|
@@ -76,6 +77,12 @@ test("describeCredential describes an empty API key", () => {
|
|
|
76
77
|
assert.equal(describeCredential({ type: "api_key", key: "" }), "empty key");
|
|
77
78
|
});
|
|
78
79
|
|
|
80
|
+
test("describeCredential describes external CLI sentinels without calling them API keys", () => {
|
|
81
|
+
const provider = PROVIDER_REGISTRY.find((p) => p.id === "claude-code");
|
|
82
|
+
assert.ok(provider);
|
|
83
|
+
assert.equal(describeCredential({ type: "api_key", key: "cli" }, provider), "external CLI");
|
|
84
|
+
});
|
|
85
|
+
|
|
79
86
|
test("describeCredential describes an OAuth token with expiry", () => {
|
|
80
87
|
const result = describeCredential({
|
|
81
88
|
type: "oauth",
|
|
@@ -149,7 +156,19 @@ test("PROVIDER_REGISTRY includes claude-code as a first-class LLM provider (#454
|
|
|
149
156
|
const entry = PROVIDER_REGISTRY.find((p) => p.id === "claude-code");
|
|
150
157
|
assert.ok(entry, "claude-code must be in PROVIDER_REGISTRY");
|
|
151
158
|
assert.equal(entry!.category, "llm");
|
|
152
|
-
assert.
|
|
159
|
+
assert.equal(getProviderAuthMode(entry!), "externalCli");
|
|
160
|
+
});
|
|
161
|
+
|
|
162
|
+
test("PROVIDER_REGISTRY classifies only Copilot and Codex as browser OAuth LLM providers", () => {
|
|
163
|
+
const modes = Object.fromEntries(
|
|
164
|
+
PROVIDER_REGISTRY.filter((p) => p.category === "llm").map((p) => [p.id, getProviderAuthMode(p)]),
|
|
165
|
+
);
|
|
166
|
+
assert.equal(modes.anthropic, "apiKey");
|
|
167
|
+
assert.equal(modes["github-copilot"], "browserOAuth");
|
|
168
|
+
assert.equal(modes["openai-codex"], "browserOAuth");
|
|
169
|
+
assert.equal(modes["claude-code"], "externalCli");
|
|
170
|
+
assert.equal(modes["google-gemini-cli"], "externalCli");
|
|
171
|
+
assert.equal(modes["google-antigravity"], "externalCli");
|
|
153
172
|
});
|
|
154
173
|
|
|
155
174
|
test("PROVIDER_REGISTRY includes all tool/search providers", () => {
|
|
@@ -11,6 +11,7 @@ import { visibleWidth } from "@gsd/pi-tui";
|
|
|
11
11
|
import { appendNotification, initNotificationStore, _resetNotificationStore } from "../notification-store.ts";
|
|
12
12
|
import { GSDNotificationOverlay, notificationOverlayOptions } from "../notification-overlay.ts";
|
|
13
13
|
import { wrapVisibleText } from "../tui/render-kit.ts";
|
|
14
|
+
import { assertFullOuterBorder } from "./tui-border-assertions.ts";
|
|
14
15
|
|
|
15
16
|
const fakeTheme = {
|
|
16
17
|
fg: (_color: string, text: string) => text,
|
|
@@ -92,7 +93,11 @@ describe("notification overlay — wrapText", () => {
|
|
|
92
93
|
t.after(() => overlay.dispose());
|
|
93
94
|
|
|
94
95
|
for (const width of [40, 80, 120]) {
|
|
95
|
-
|
|
96
|
+
const rendered = overlay.render(width);
|
|
97
|
+
assertLinesFit(rendered, width);
|
|
98
|
+
assertFullOuterBorder(rendered, width);
|
|
99
|
+
assert.match(rendered[0] ?? "", /^╭─ Notifications /);
|
|
100
|
+
assert.match(rendered.at(-1) ?? "", /^╰─+╯$/);
|
|
96
101
|
overlay.invalidate();
|
|
97
102
|
}
|
|
98
103
|
});
|
|
@@ -48,20 +48,27 @@ describe("auditOrphanedMilestoneBranches", () => {
|
|
|
48
48
|
const result = auditOrphanedMilestoneBranches(dir, "worktree");
|
|
49
49
|
assert.deepStrictEqual(result.recovered, []);
|
|
50
50
|
assert.deepStrictEqual(result.warnings, []);
|
|
51
|
+
assert.deepStrictEqual(result.actions, []);
|
|
52
|
+
assert.equal(result.blockingStrandedWork, null);
|
|
51
53
|
});
|
|
52
54
|
|
|
53
|
-
test("
|
|
54
|
-
// Create a milestone branch that would otherwise be detected
|
|
55
|
+
test("runs in none isolation mode and cleans safe completed residue", () => {
|
|
55
56
|
run("git branch milestone/M001", dir);
|
|
56
57
|
insertMilestone({ id: "M001", title: "Test", status: "complete" });
|
|
57
58
|
|
|
58
59
|
const result = auditOrphanedMilestoneBranches(dir, "none");
|
|
59
|
-
assert.
|
|
60
|
+
assert.ok(
|
|
61
|
+
result.recovered.some((r) => r.includes("Deleted merged branch milestone/M001")),
|
|
62
|
+
`should clean merged completed residue even in none mode; got: ${JSON.stringify(result.recovered)}`,
|
|
63
|
+
);
|
|
60
64
|
assert.deepStrictEqual(result.warnings, []);
|
|
65
|
+
assert.ok(
|
|
66
|
+
result.actions.some((action) => action.kind === "complete-merged-branch"),
|
|
67
|
+
"should record structured cleanup action",
|
|
68
|
+
);
|
|
61
69
|
|
|
62
|
-
// Branch should still exist
|
|
63
70
|
const branches = run("git branch --list milestone/M001", dir);
|
|
64
|
-
assert.
|
|
71
|
+
assert.equal(branches, "", "safe completed branch should be cleaned in none mode");
|
|
65
72
|
});
|
|
66
73
|
|
|
67
74
|
test("deletes merged branch for completed milestone", () => {
|
|
@@ -149,9 +156,12 @@ describe("auditOrphanedMilestoneBranches", () => {
|
|
|
149
156
|
// Must surface a warning so the user knows the worktree holds uncollapsed work
|
|
150
157
|
assert.ok(result.warnings.length > 0, "should warn about in-progress orphan");
|
|
151
158
|
assert.ok(
|
|
152
|
-
result.warnings.some(w => w.includes("milestone/M001") && w.includes("in-progress")),
|
|
153
|
-
`warning should mention milestone/M001 and in-progress state; got: ${JSON.stringify(result.warnings)}`,
|
|
159
|
+
result.warnings.some(w => w.includes("Stranded work") && w.includes("milestone/M001") && w.includes("in-progress")),
|
|
160
|
+
`warning should mention stranded milestone/M001 and in-progress state; got: ${JSON.stringify(result.warnings)}`,
|
|
154
161
|
);
|
|
162
|
+
assert.equal(result.blockingStrandedWork?.milestoneId, "M001");
|
|
163
|
+
assert.equal(result.blockingStrandedWork?.recoveryMode, "branch");
|
|
164
|
+
assert.equal(result.blockingStrandedWork?.commitsAhead, 1);
|
|
155
165
|
|
|
156
166
|
// Branch must still exist
|
|
157
167
|
const branches = run("git branch --list milestone/M001", dir);
|
|
@@ -184,6 +194,53 @@ describe("auditOrphanedMilestoneBranches", () => {
|
|
|
184
194
|
result.warnings.some(w => w.includes(".gsd/worktrees/M001") || w.includes("worktree")),
|
|
185
195
|
`warning should reference the worktree location; got: ${JSON.stringify(result.warnings)}`,
|
|
186
196
|
);
|
|
197
|
+
assert.equal(result.blockingStrandedWork?.recoveryMode, "worktree");
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
test("detects dirty in-progress worktree even when branch has no commits ahead", () => {
|
|
201
|
+
run("git branch milestone/M001", dir);
|
|
202
|
+
|
|
203
|
+
const wtDir = join(dir, ".gsd", "worktrees", "M001");
|
|
204
|
+
mkdirSync(wtDir, { recursive: true });
|
|
205
|
+
writeFileSync(join(wtDir, ".git"), `gitdir: ${join(dir, ".git", "worktrees", "M001")}\n`);
|
|
206
|
+
writeFileSync(join(wtDir, "dirty.txt"), "uncommitted work\n");
|
|
207
|
+
|
|
208
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
209
|
+
|
|
210
|
+
const result = auditOrphanedMilestoneBranches(dir, "worktree", {
|
|
211
|
+
hasChanges: (basePath) => basePath === wtDir,
|
|
212
|
+
});
|
|
213
|
+
|
|
214
|
+
assert.deepStrictEqual(result.recovered, []);
|
|
215
|
+
assert.ok(
|
|
216
|
+
result.warnings.some((w) => w.includes("uncommitted changes")),
|
|
217
|
+
`dirty worktree should be treated as stranded work; got: ${JSON.stringify(result.warnings)}`,
|
|
218
|
+
);
|
|
219
|
+
assert.equal(result.blockingStrandedWork?.milestoneId, "M001");
|
|
220
|
+
assert.equal(result.blockingStrandedWork?.dirtyWorktree, true);
|
|
221
|
+
assert.equal(result.blockingStrandedWork?.recoveryMode, "worktree");
|
|
222
|
+
});
|
|
223
|
+
|
|
224
|
+
test("detects dirty in-progress worktree even when milestone branch is absent", () => {
|
|
225
|
+
const wtDir = join(dir, ".gsd", "worktrees", "M001");
|
|
226
|
+
mkdirSync(wtDir, { recursive: true });
|
|
227
|
+
writeFileSync(join(wtDir, ".git"), `gitdir: ${join(dir, ".git", "worktrees", "M001")}\n`);
|
|
228
|
+
writeFileSync(join(wtDir, "dirty.txt"), "branchless work\n");
|
|
229
|
+
|
|
230
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
231
|
+
|
|
232
|
+
const result = auditOrphanedMilestoneBranches(dir, "none", {
|
|
233
|
+
hasChanges: (basePath) => basePath === wtDir,
|
|
234
|
+
});
|
|
235
|
+
|
|
236
|
+
assert.deepStrictEqual(result.recovered, []);
|
|
237
|
+
assert.ok(
|
|
238
|
+
result.warnings.some((w) => w.includes("Stranded work") && w.includes("M001")),
|
|
239
|
+
`branchless dirty worktree should block; got: ${JSON.stringify(result.warnings)}`,
|
|
240
|
+
);
|
|
241
|
+
assert.equal(result.blockingStrandedWork?.branch, undefined);
|
|
242
|
+
assert.equal(result.blockingStrandedWork?.dirtyWorktree, true);
|
|
243
|
+
assert.equal(result.blockingStrandedWork?.recoveryMode, "worktree");
|
|
187
244
|
});
|
|
188
245
|
|
|
189
246
|
test("cleans up orphaned worktree directory for merged milestone", () => {
|
|
@@ -359,7 +416,7 @@ describe("auditOrphanedMilestoneBranches", () => {
|
|
|
359
416
|
assert.ok(existsSync(wtDir), "active milestone worktree dir must be preserved");
|
|
360
417
|
});
|
|
361
418
|
|
|
362
|
-
test("#5879 —
|
|
419
|
+
test("#5879 — cleans branch-less complete orphan in 'none' isolation mode", () => {
|
|
363
420
|
insertMilestone({ id: "M001", title: "Test", status: "complete" });
|
|
364
421
|
|
|
365
422
|
const wtDir = join(dir, ".gsd", "worktrees", "M001");
|
|
@@ -368,7 +425,10 @@ describe("auditOrphanedMilestoneBranches", () => {
|
|
|
368
425
|
|
|
369
426
|
const result = auditOrphanedMilestoneBranches(dir, "none");
|
|
370
427
|
|
|
371
|
-
assert.
|
|
372
|
-
|
|
428
|
+
assert.ok(
|
|
429
|
+
result.recovered.some((r) => r.includes("M001") && r.includes("branch already deleted")),
|
|
430
|
+
`none mode should still clean safe completed residue; got: ${JSON.stringify(result.recovered)}`,
|
|
431
|
+
);
|
|
432
|
+
assert.ok(!existsSync(wtDir), "completed orphan worktree dir should be cleaned in none mode");
|
|
373
433
|
});
|
|
374
434
|
});
|
|
@@ -5,6 +5,7 @@ import { describe, it } from "node:test";
|
|
|
5
5
|
import assert from "node:assert/strict";
|
|
6
6
|
|
|
7
7
|
import { visibleWidth } from "@gsd/pi-tui";
|
|
8
|
+
import { assertFullOuterBorder } from "./tui-border-assertions.ts";
|
|
8
9
|
|
|
9
10
|
function assertLinesFit(lines: string[], width: number): void {
|
|
10
11
|
for (const line of lines) {
|
|
@@ -51,6 +52,9 @@ describe("parallel-monitor-overlay", () => {
|
|
|
51
52
|
const joined = lines.join("\n");
|
|
52
53
|
assert.ok(joined.includes("Parallel Monitor"), "should include title");
|
|
53
54
|
assert.ok(joined.includes("No parallel workers found"), "should show empty state");
|
|
55
|
+
assertFullOuterBorder(lines, 80);
|
|
56
|
+
assert.match(lines[0] ?? "", /^╭─ GSD Parallel Monitor /);
|
|
57
|
+
assert.match(lines.at(-1) ?? "", /^╰─+╯$/);
|
|
54
58
|
|
|
55
59
|
// Dispose should not throw
|
|
56
60
|
overlay.dispose();
|
|
@@ -105,7 +109,9 @@ describe("parallel-monitor-overlay", () => {
|
|
|
105
109
|
);
|
|
106
110
|
|
|
107
111
|
for (const width of [40, 80, 120]) {
|
|
108
|
-
|
|
112
|
+
const lines = overlay.render(width);
|
|
113
|
+
assertLinesFit(lines, width);
|
|
114
|
+
assertFullOuterBorder(lines, width);
|
|
109
115
|
overlay.invalidate();
|
|
110
116
|
}
|
|
111
117
|
|
|
@@ -2,6 +2,7 @@ import { describe, test } from "node:test";
|
|
|
2
2
|
import assert from "node:assert/strict";
|
|
3
3
|
|
|
4
4
|
import { showQueueReorder } from "../queue-reorder-ui.ts";
|
|
5
|
+
import { assertFullOuterBorder } from "./tui-border-assertions.ts";
|
|
5
6
|
|
|
6
7
|
const fakeTheme = {
|
|
7
8
|
fg: (_color: string, text: string) => text,
|
|
@@ -43,6 +44,51 @@ describe("queue-reorder-ui", () => {
|
|
|
43
44
|
const joined = lastRender.join("\n");
|
|
44
45
|
assert.ok(joined.includes("M016"), "selected item should stay visible after scrolling");
|
|
45
46
|
assert.ok(lastRender.length <= 16, `overlay should fit terminal max-height, got ${lastRender.length}`);
|
|
47
|
+
assertFullOuterBorder(lastRender, 100);
|
|
48
|
+
assert.match(lastRender[0] ?? "", /^╭─ Queue Reorder /);
|
|
49
|
+
assert.match(lastRender.at(-1) ?? "", /^╰─+╯$/);
|
|
50
|
+
} finally {
|
|
51
|
+
if (originalRowsDescriptor) {
|
|
52
|
+
Object.defineProperty(process.stdout, "rows", originalRowsDescriptor);
|
|
53
|
+
} else {
|
|
54
|
+
delete (process.stdout as { rows?: number }).rows;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
test("draws queue scroll thumb beside queue rows when completed rows are shown", async () => {
|
|
60
|
+
const originalRowsDescriptor = Object.getOwnPropertyDescriptor(process.stdout, "rows");
|
|
61
|
+
Object.defineProperty(process.stdout, "rows", { value: 12, configurable: true });
|
|
62
|
+
|
|
63
|
+
try {
|
|
64
|
+
const completed = [
|
|
65
|
+
{ id: "M000", title: "Already done" },
|
|
66
|
+
];
|
|
67
|
+
const pending = Array.from({ length: 8 }, (_, idx) => ({
|
|
68
|
+
id: `M${String(idx + 1).padStart(3, "0")}`,
|
|
69
|
+
title: `Milestone ${idx + 1}`,
|
|
70
|
+
}));
|
|
71
|
+
let rendered: string[] = [];
|
|
72
|
+
|
|
73
|
+
const ctx = {
|
|
74
|
+
hasUI: true,
|
|
75
|
+
ui: {
|
|
76
|
+
custom: async (factory: any) => {
|
|
77
|
+
const component = factory({ requestRender() {} }, fakeTheme, null, () => {});
|
|
78
|
+
rendered = component.render(80);
|
|
79
|
+
return null;
|
|
80
|
+
},
|
|
81
|
+
},
|
|
82
|
+
} as any;
|
|
83
|
+
|
|
84
|
+
await showQueueReorder(ctx, completed, pending);
|
|
85
|
+
|
|
86
|
+
const completedHeader = rendered.find(line => line.includes("Completed:"));
|
|
87
|
+
const firstQueueRow = rendered.find(line => line.includes("M001"));
|
|
88
|
+
assert.ok(completedHeader, "completed header should be rendered before queue rows");
|
|
89
|
+
assert.ok(firstQueueRow, "first queue row should be rendered");
|
|
90
|
+
assert.match(completedHeader, /│$/);
|
|
91
|
+
assert.match(firstQueueRow, /┃$/);
|
|
46
92
|
} finally {
|
|
47
93
|
if (originalRowsDescriptor) {
|
|
48
94
|
Object.defineProperty(process.stdout, "rows", originalRowsDescriptor);
|
|
@@ -7,6 +7,7 @@ import assert from "node:assert/strict";
|
|
|
7
7
|
|
|
8
8
|
import { GSDConfigOverlay, formatConfigText } from "../config-overlay.ts";
|
|
9
9
|
import { handleCoreCommand } from "../commands/handlers/core.ts";
|
|
10
|
+
import { assertFullOuterBorder } from "./tui-border-assertions.ts";
|
|
10
11
|
|
|
11
12
|
const theme = {
|
|
12
13
|
bold: (s: string) => s,
|
|
@@ -24,6 +25,9 @@ test("GSDConfigOverlay renders and responds to input", () => {
|
|
|
24
25
|
|
|
25
26
|
const lines = overlay.render(60);
|
|
26
27
|
assert.ok(lines.some((line) => line.includes("GSD Configuration")));
|
|
28
|
+
assertFullOuterBorder(lines, 60);
|
|
29
|
+
assert.match(lines[0] ?? "", /^╭─ GSD Configuration /);
|
|
30
|
+
assert.match(lines.at(-1) ?? "", /^╰─+╯$/);
|
|
27
31
|
|
|
28
32
|
overlay.handleInput("j");
|
|
29
33
|
assert.equal(renderRequests, 1);
|
|
@@ -15,6 +15,13 @@ function readGsdFile(relativePath: string): string {
|
|
|
15
15
|
return readFileSync(resolve(gsdDir, relativePath), "utf-8");
|
|
16
16
|
}
|
|
17
17
|
|
|
18
|
+
function firstIndexOfAny(source: string, needles: string[]): number {
|
|
19
|
+
const indexes = needles
|
|
20
|
+
.map((needle) => source.indexOf(needle))
|
|
21
|
+
.filter((index) => index > -1);
|
|
22
|
+
return indexes.length > 0 ? Math.min(...indexes) : -1;
|
|
23
|
+
}
|
|
24
|
+
|
|
18
25
|
test("command entrypoints use startAutoDetached instead of awaiting startAuto (#3733)", () => {
|
|
19
26
|
const autoHandlerSrc = readGsdFile("commands/handlers/auto.ts");
|
|
20
27
|
const workflowHandlerSrc = readGsdFile("commands/handlers/workflow.ts");
|
|
@@ -145,7 +152,11 @@ test("fresh start registers the auto worker before bootstrap enters worktree flo
|
|
|
145
152
|
const resumeEnterMilestoneIdx = resumeBody.indexOf("buildLifecycle().enterMilestone");
|
|
146
153
|
const dbOpenIdx = bootstrapBody.indexOf("await openProjectDbIfPresent(base);");
|
|
147
154
|
const bootstrapRegisterIdx = bootstrapBody.indexOf("registerAutoWorkerForSession(base);");
|
|
148
|
-
const enterMilestoneIdx = bootstrapBody
|
|
155
|
+
const enterMilestoneIdx = firstIndexOfAny(bootstrapBody, [
|
|
156
|
+
"buildLifecycle().enterMilestone",
|
|
157
|
+
"lifecycle.enterMilestone",
|
|
158
|
+
"lifecycle.adoptStrandedMilestone",
|
|
159
|
+
]);
|
|
149
160
|
|
|
150
161
|
assert.ok(startAutoIdx > -1, "startAuto should exist");
|
|
151
162
|
assert.ok(preBootstrapRegisterIdx > -1, "startAuto should register worker before bootstrap");
|
|
@@ -158,7 +169,7 @@ test("fresh start registers the auto worker before bootstrap enters worktree flo
|
|
|
158
169
|
assert.ok(bootstrapIdx > -1, "bootstrapAutoSession should exist");
|
|
159
170
|
assert.ok(dbOpenIdx > -1, "bootstrap should open the project DB");
|
|
160
171
|
assert.ok(bootstrapRegisterIdx > -1, "bootstrap should register worker after DB open");
|
|
161
|
-
assert.ok(enterMilestoneIdx > -1, "bootstrap should enter milestones through lifecycle");
|
|
172
|
+
assert.ok(enterMilestoneIdx > -1, "bootstrap should enter or adopt milestones through lifecycle");
|
|
162
173
|
assert.ok(
|
|
163
174
|
preBootstrapRegisterIdx < bootstrapCallIdx,
|
|
164
175
|
"worker registration must happen before bootstrap so enterMilestone can claim milestone leases on first entry",
|