gsd-pi 2.76.0-dev.82e249f7b → 2.76.0-dev.97807402
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/claude-cli-check.js +32 -3
- package/dist/mcp-server.d.ts +7 -0
- package/dist/mcp-server.js +35 -1
- package/dist/resource-loader.d.ts +1 -1
- package/dist/resource-loader.js +2 -8
- package/dist/resources/extensions/claude-code-cli/readiness.js +4 -3
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +77 -17
- package/dist/resources/extensions/gsd/auto/phases.js +14 -0
- package/dist/resources/extensions/gsd/auto/run-unit.js +27 -0
- package/dist/resources/extensions/gsd/auto-model-selection.js +1 -1
- package/dist/resources/extensions/gsd/auto-post-unit.js +1 -1
- package/dist/resources/extensions/gsd/auto-recovery.js +13 -0
- package/dist/resources/extensions/gsd/auto-start.js +27 -18
- package/dist/resources/extensions/gsd/auto-worktree.js +30 -48
- package/dist/resources/extensions/gsd/auto.js +13 -17
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +17 -1
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +39 -9
- package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +93 -0
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +2 -0
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +40 -4
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +12 -1
- package/dist/resources/extensions/gsd/commands-prefs-wizard.js +968 -23
- package/dist/resources/extensions/gsd/compaction-snapshot.js +121 -0
- package/dist/resources/extensions/gsd/error-classifier.js +10 -3
- package/dist/resources/extensions/gsd/exec-history.js +120 -0
- package/dist/resources/extensions/gsd/exec-sandbox.js +258 -0
- package/dist/resources/extensions/gsd/gsd-db.js +115 -7
- package/dist/resources/extensions/gsd/guided-flow.js +189 -0
- package/dist/resources/extensions/gsd/health-widget.js +4 -1
- package/dist/resources/extensions/gsd/init-wizard.js +15 -1
- package/dist/resources/extensions/gsd/key-manager.js +6 -0
- package/dist/resources/extensions/gsd/model-router.js +36 -3
- package/dist/resources/extensions/gsd/pre-execution-checks.js +35 -9
- package/dist/resources/extensions/gsd/preferences-types.js +9 -0
- package/dist/resources/extensions/gsd/preferences-validation.js +83 -0
- package/dist/resources/extensions/gsd/preferences.js +17 -17
- package/dist/resources/extensions/gsd/prompts/discuss-headless.md +8 -0
- package/dist/resources/extensions/gsd/prompts/discuss.md +29 -2
- package/dist/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
- package/dist/resources/extensions/gsd/safety/file-change-validator.js +10 -4
- package/dist/resources/extensions/gsd/safety/safety-harness.js +4 -0
- package/dist/resources/extensions/gsd/token-counter.js +22 -5
- package/dist/resources/extensions/gsd/tools/exec-search-tool.js +59 -0
- package/dist/resources/extensions/gsd/tools/exec-tool.js +126 -0
- package/dist/resources/extensions/gsd/tools/resume-tool.js +23 -0
- package/dist/resources/extensions/gsd/workflow-mcp.js +3 -0
- package/dist/resources/skills/verify-before-complete/SKILL.md +2 -1
- package/dist/resources/skills/write-docs/SKILL.md +2 -1
- package/dist/tsconfig.extensions.tsbuildinfo +1 -1
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +7 -7
- 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/required-server-files.json +1 -1
- 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 +7 -7
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
- 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/server.js +1 -1
- package/package.json +1 -1
- package/packages/mcp-server/dist/remote-questions.d.ts +45 -0
- package/packages/mcp-server/dist/remote-questions.d.ts.map +1 -0
- package/packages/mcp-server/dist/remote-questions.js +732 -0
- package/packages/mcp-server/dist/remote-questions.js.map +1 -0
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +18 -1
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +64 -25
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +2 -1
- package/packages/mcp-server/src/remote-questions.test.ts +294 -0
- package/packages/mcp-server/src/remote-questions.ts +916 -0
- package/packages/mcp-server/src/server.ts +19 -1
- package/packages/mcp-server/src/workflow-tools.test.ts +146 -1
- package/packages/mcp-server/src/workflow-tools.ts +84 -43
- package/packages/mcp-server/tsconfig.test.json +19 -0
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.js +2 -0
- package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
- package/packages/pi-ai/dist/providers/simple-options.d.ts +10 -0
- package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/simple-options.js +16 -1
- package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
- package/packages/pi-ai/src/providers/anthropic-shared.ts +3 -1
- package/packages/pi-ai/src/providers/simple-options.ts +17 -1
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.js +203 -0
- package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.js +14 -0
- package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/redact-secrets.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.js +49 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.test.js +67 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/session-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/session-manager.js +9 -5
- package/packages/pi-coding-agent/dist/core/session-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/session-manager.test.js +25 -1
- package/packages/pi-coding-agent/dist/core/session-manager.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +5 -4
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.d.ts +7 -6
- package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.js +29 -21
- package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +13 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/src/core/model-registry-custom-caps.test.ts +245 -0
- package/packages/pi-coding-agent/src/core/model-registry.ts +16 -0
- package/packages/pi-coding-agent/src/core/redact-secrets.test.ts +86 -0
- package/packages/pi-coding-agent/src/core/redact-secrets.ts +58 -0
- package/packages/pi-coding-agent/src/core/session-manager.test.ts +36 -1
- package/packages/pi-coding-agent/src/core/session-manager.ts +9 -5
- package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +6 -6
- package/packages/pi-coding-agent/src/modes/interactive/components/skill-invocation-message.ts +36 -22
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +13 -1
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
- package/src/resources/extensions/claude-code-cli/readiness.ts +4 -3
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +78 -17
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +149 -5
- package/src/resources/extensions/gsd/auto/phases.ts +14 -0
- package/src/resources/extensions/gsd/auto/run-unit.ts +29 -0
- package/src/resources/extensions/gsd/auto-model-selection.ts +1 -1
- package/src/resources/extensions/gsd/auto-post-unit.ts +1 -2
- package/src/resources/extensions/gsd/auto-recovery.ts +15 -0
- package/src/resources/extensions/gsd/auto-start.ts +29 -19
- package/src/resources/extensions/gsd/auto-worktree.ts +34 -52
- package/src/resources/extensions/gsd/auto.ts +12 -17
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +23 -1
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +40 -9
- package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +109 -0
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +2 -0
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +42 -4
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +13 -1
- package/src/resources/extensions/gsd/commands-prefs-wizard.ts +898 -32
- package/src/resources/extensions/gsd/compaction-snapshot.ts +165 -0
- package/src/resources/extensions/gsd/error-classifier.ts +10 -3
- package/src/resources/extensions/gsd/exec-history.ts +153 -0
- package/src/resources/extensions/gsd/exec-sandbox.ts +326 -0
- package/src/resources/extensions/gsd/gsd-db.ts +122 -7
- package/src/resources/extensions/gsd/guided-flow.ts +221 -0
- package/src/resources/extensions/gsd/health-widget.ts +3 -1
- package/src/resources/extensions/gsd/init-wizard.ts +15 -1
- package/src/resources/extensions/gsd/journal.ts +2 -1
- package/src/resources/extensions/gsd/key-manager.ts +6 -0
- package/src/resources/extensions/gsd/model-router.ts +42 -1
- package/src/resources/extensions/gsd/pre-execution-checks.ts +36 -10
- package/src/resources/extensions/gsd/preferences-types.ts +46 -0
- package/src/resources/extensions/gsd/preferences-validation.ts +79 -0
- package/src/resources/extensions/gsd/preferences.ts +17 -17
- package/src/resources/extensions/gsd/prompts/discuss-headless.md +8 -0
- package/src/resources/extensions/gsd/prompts/discuss.md +29 -2
- package/src/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
- package/src/resources/extensions/gsd/safety/file-change-validator.ts +14 -3
- package/src/resources/extensions/gsd/safety/safety-harness.ts +6 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +116 -0
- package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +49 -0
- package/src/resources/extensions/gsd/tests/compaction-snapshot.test.ts +123 -0
- package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +31 -0
- package/src/resources/extensions/gsd/tests/ensure-db-open.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/escalation.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/exec-history.test.ts +124 -0
- package/src/resources/extensions/gsd/tests/exec-sandbox.test.ts +210 -0
- package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +58 -0
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +152 -1
- package/src/resources/extensions/gsd/tests/init-wizard.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/isolation-none-branch-guard.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/issue-4540-regressions.test.ts +288 -0
- package/src/resources/extensions/gsd/tests/key-manager.test.ts +7 -0
- package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +19 -0
- package/src/resources/extensions/gsd/tests/pre-exec-backtick-strip.test.ts +14 -0
- package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +234 -0
- package/src/resources/extensions/gsd/tests/preferences.test.ts +110 -0
- package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +44 -0
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +48 -0
- package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +388 -0
- package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +9 -3
- package/src/resources/extensions/gsd/tests/save-gate-result-render.test.ts +95 -0
- package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +32 -40
- package/src/resources/extensions/gsd/tests/stash-queued-context-files.test.ts +56 -0
- package/src/resources/extensions/gsd/tests/token-counter.test.ts +105 -1
- package/src/resources/extensions/gsd/tests/tool-compatibility.test.ts +107 -0
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +65 -2
- package/src/resources/extensions/gsd/tests/write-gate.test.ts +64 -0
- package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +3 -1
- package/src/resources/extensions/gsd/token-counter.ts +22 -5
- package/src/resources/extensions/gsd/tools/exec-search-tool.ts +81 -0
- package/src/resources/extensions/gsd/tools/exec-tool.ts +183 -0
- package/src/resources/extensions/gsd/tools/resume-tool.ts +40 -0
- package/src/resources/extensions/gsd/workflow-logger.ts +2 -1
- package/src/resources/extensions/gsd/workflow-mcp.ts +3 -0
- package/src/resources/skills/verify-before-complete/SKILL.md +2 -1
- package/src/resources/skills/write-docs/SKILL.md +2 -1
- /package/dist/web/standalone/.next/static/{ecSsu49rxxcpbNmVP4mLD → pI48IF3dgfs0CBrYi2bh_}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{ecSsu49rxxcpbNmVP4mLD → pI48IF3dgfs0CBrYi2bh_}/_ssgManifest.js +0 -0
|
@@ -3,12 +3,14 @@ import assert from 'node:assert/strict';
|
|
|
3
3
|
import * as fs from 'node:fs';
|
|
4
4
|
import * as path from 'node:path';
|
|
5
5
|
import * as os from 'node:os';
|
|
6
|
+
import { createRequire } from 'node:module';
|
|
6
7
|
import {
|
|
7
8
|
openDatabase,
|
|
8
9
|
closeDatabase,
|
|
9
10
|
isDbAvailable,
|
|
10
11
|
wasDbOpenAttempted,
|
|
11
12
|
getDbProvider,
|
|
13
|
+
getDbStatus,
|
|
12
14
|
insertDecision,
|
|
13
15
|
getDecisionById,
|
|
14
16
|
insertRequirement,
|
|
@@ -26,6 +28,8 @@ import {
|
|
|
26
28
|
checkpointDatabase,
|
|
27
29
|
} from '../gsd-db.ts';
|
|
28
30
|
|
|
31
|
+
const _require = createRequire(import.meta.url);
|
|
32
|
+
|
|
29
33
|
// ═══════════════════════════════════════════════════════════════════════════
|
|
30
34
|
// Helper: create a temp file path for file-backed DB tests
|
|
31
35
|
// ═══════════════════════════════════════════════════════════════════════════
|
|
@@ -59,6 +63,20 @@ function withPlatform<T>(platform: NodeJS.Platform, fn: () => T): T {
|
|
|
59
63
|
}
|
|
60
64
|
}
|
|
61
65
|
|
|
66
|
+
function openRawSqliteForTest(dbPath: string): { exec(sql: string): void; close(): void } {
|
|
67
|
+
try {
|
|
68
|
+
const mod = _require('node:sqlite') as { DatabaseSync: new (path: string) => { exec(sql: string): void; close(): void } };
|
|
69
|
+
return new mod.DatabaseSync(dbPath);
|
|
70
|
+
} catch {
|
|
71
|
+
type SqliteCtor = new (path: string) => { exec(sql: string): void; close(): void };
|
|
72
|
+
const mod = _require('better-sqlite3') as
|
|
73
|
+
| SqliteCtor
|
|
74
|
+
| { default: SqliteCtor };
|
|
75
|
+
const DatabaseCtor: SqliteCtor = typeof mod === 'function' ? mod : mod.default;
|
|
76
|
+
return new DatabaseCtor(dbPath);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
|
|
62
80
|
// ═══════════════════════════════════════════════════════════════════════════
|
|
63
81
|
// gsd-db tests
|
|
64
82
|
// ═══════════════════════════════════════════════════════════════════════════
|
|
@@ -81,7 +99,7 @@ describe('gsd-db', () => {
|
|
|
81
99
|
// Check schema_version table
|
|
82
100
|
const adapter = _getAdapter()!;
|
|
83
101
|
const version = adapter.prepare('SELECT MAX(version) as version FROM schema_version').get();
|
|
84
|
-
assert.deepStrictEqual(version?.['version'],
|
|
102
|
+
assert.deepStrictEqual(version?.['version'], 22, 'schema version should be 22');
|
|
85
103
|
|
|
86
104
|
// Check tables exist by querying them
|
|
87
105
|
const dRows = adapter.prepare('SELECT count(*) as cnt FROM decisions').get();
|
|
@@ -404,6 +422,53 @@ describe('gsd-db', () => {
|
|
|
404
422
|
cleanup(dbPath);
|
|
405
423
|
});
|
|
406
424
|
|
|
425
|
+
test('gsd-db: legacy DB missing memories.scope opens and bootstraps index columns', () => {
|
|
426
|
+
const dbPath = tempDbPath();
|
|
427
|
+
const legacyDb = openRawSqliteForTest(dbPath);
|
|
428
|
+
legacyDb.exec(`
|
|
429
|
+
CREATE TABLE schema_version (
|
|
430
|
+
version INTEGER NOT NULL,
|
|
431
|
+
applied_at TEXT NOT NULL
|
|
432
|
+
);
|
|
433
|
+
INSERT INTO schema_version(version, applied_at) VALUES (17, '2026-04-20T00:00:00.000Z');
|
|
434
|
+
CREATE TABLE memories (
|
|
435
|
+
seq INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
436
|
+
id TEXT NOT NULL UNIQUE,
|
|
437
|
+
category TEXT NOT NULL,
|
|
438
|
+
content TEXT NOT NULL,
|
|
439
|
+
confidence REAL NOT NULL DEFAULT 0.8,
|
|
440
|
+
source_unit_type TEXT,
|
|
441
|
+
source_unit_id TEXT,
|
|
442
|
+
created_at TEXT NOT NULL,
|
|
443
|
+
updated_at TEXT NOT NULL,
|
|
444
|
+
superseded_by TEXT DEFAULT NULL,
|
|
445
|
+
hit_count INTEGER NOT NULL DEFAULT 0
|
|
446
|
+
);
|
|
447
|
+
INSERT INTO memories(id, category, content, created_at, updated_at)
|
|
448
|
+
VALUES ('legacy-memory', 'note', 'legacy row', '2026-04-20T00:00:00.000Z', '2026-04-20T00:00:00.000Z');
|
|
449
|
+
`);
|
|
450
|
+
legacyDb.close();
|
|
451
|
+
|
|
452
|
+
assert.equal(openDatabase(dbPath), true, 'openDatabase should succeed for legacy DB missing memories.scope');
|
|
453
|
+
|
|
454
|
+
const adapter = _getAdapter()!;
|
|
455
|
+
const columns = adapter.prepare('PRAGMA table_info(memories)').all();
|
|
456
|
+
const names = columns.map((row) => row['name']);
|
|
457
|
+
assert.ok(names.includes('scope'), 'memories.scope should be added during bootstrap');
|
|
458
|
+
assert.ok(names.includes('tags'), 'memories.tags should be added during bootstrap');
|
|
459
|
+
|
|
460
|
+
const row = adapter.prepare(`SELECT scope, tags FROM memories WHERE id = 'legacy-memory'`).get();
|
|
461
|
+
assert.equal(row?.['scope'], 'project', 'legacy rows should receive default scope');
|
|
462
|
+
assert.equal(row?.['tags'], '[]', 'legacy rows should receive default tags');
|
|
463
|
+
|
|
464
|
+
const index = adapter.prepare(
|
|
465
|
+
"SELECT name FROM sqlite_master WHERE type = 'index' AND name = 'idx_memories_scope'",
|
|
466
|
+
).get();
|
|
467
|
+
assert.equal(index?.['name'], 'idx_memories_scope', 'scope index should be created after bootstrap columns are present');
|
|
468
|
+
|
|
469
|
+
cleanup(dbPath);
|
|
470
|
+
});
|
|
471
|
+
|
|
407
472
|
test('gsd-db: rowToTask tolerates legacy comma-separated task arrays', () => {
|
|
408
473
|
openDatabase(':memory:');
|
|
409
474
|
|
|
@@ -561,6 +626,92 @@ describe('gsd-db', () => {
|
|
|
561
626
|
});
|
|
562
627
|
});
|
|
563
628
|
|
|
629
|
+
// ─── getDbStatus ───────────────────────────────────────────────────────────
|
|
630
|
+
|
|
631
|
+
describe('getDbStatus', () => {
|
|
632
|
+
test('getDbStatus: initial state before any open', () => {
|
|
633
|
+
closeDatabase();
|
|
634
|
+
const status = getDbStatus();
|
|
635
|
+
assert.strictEqual(status.available, false, 'available false before open');
|
|
636
|
+
assert.strictEqual(status.attempted, false, 'attempted false before open');
|
|
637
|
+
assert.strictEqual(status.lastError, null, 'lastError null before open');
|
|
638
|
+
assert.strictEqual(status.lastPhase, null, 'lastPhase null before open');
|
|
639
|
+
});
|
|
640
|
+
|
|
641
|
+
test('getDbStatus: available after successful open', () => {
|
|
642
|
+
openDatabase(':memory:');
|
|
643
|
+
const status = getDbStatus();
|
|
644
|
+
assert.strictEqual(status.available, true, 'available true after open');
|
|
645
|
+
assert.strictEqual(status.attempted, true, 'attempted true after open');
|
|
646
|
+
assert.ok(status.provider !== null, 'provider set after open');
|
|
647
|
+
assert.strictEqual(status.lastError, null, 'lastError null on success');
|
|
648
|
+
assert.strictEqual(status.lastPhase, null, 'lastPhase null on success');
|
|
649
|
+
closeDatabase();
|
|
650
|
+
});
|
|
651
|
+
|
|
652
|
+
test('getDbStatus: resets lastError/lastPhase after closeDatabase', () => {
|
|
653
|
+
// Simulate a failed open to set error state
|
|
654
|
+
const corruptPath = path.join(os.tmpdir(), `gsd-corrupt-${Date.now()}.db`);
|
|
655
|
+
fs.writeFileSync(corruptPath, Buffer.from('not a sqlite file at all!!!!!'));
|
|
656
|
+
try {
|
|
657
|
+
openDatabase(corruptPath);
|
|
658
|
+
} catch {
|
|
659
|
+
// expected
|
|
660
|
+
}
|
|
661
|
+
assert.ok(getDbStatus().lastError !== null, 'lastError set after failed open');
|
|
662
|
+
|
|
663
|
+
// closeDatabase should clear it even though no DB was opened
|
|
664
|
+
closeDatabase();
|
|
665
|
+
const status = getDbStatus();
|
|
666
|
+
assert.strictEqual(status.lastError, null, 'lastError cleared by closeDatabase');
|
|
667
|
+
assert.strictEqual(status.lastPhase, null, 'lastPhase cleared by closeDatabase');
|
|
668
|
+
assert.strictEqual(status.attempted, false, 'attempted reset by closeDatabase');
|
|
669
|
+
fs.unlinkSync(corruptPath);
|
|
670
|
+
});
|
|
671
|
+
|
|
672
|
+
test('getDbStatus: captures open-phase error on corrupt file', () => {
|
|
673
|
+
closeDatabase();
|
|
674
|
+
const corruptPath = path.join(os.tmpdir(), `gsd-corrupt-${Date.now()}.db`);
|
|
675
|
+
fs.writeFileSync(corruptPath, Buffer.from('not a sqlite file at all!!!!!'));
|
|
676
|
+
try {
|
|
677
|
+
openDatabase(corruptPath);
|
|
678
|
+
} catch {
|
|
679
|
+
// expected — both providers should reject a non-SQLite file
|
|
680
|
+
}
|
|
681
|
+
const status = getDbStatus();
|
|
682
|
+
if (!status.available) {
|
|
683
|
+
// open failed (expected in most environments)
|
|
684
|
+
assert.strictEqual(status.attempted, true, 'attempted true after failed open');
|
|
685
|
+
// provider may reject at raw-open level ("open") or at SQL init level ("initSchema")
|
|
686
|
+
assert.ok(
|
|
687
|
+
status.lastPhase === 'open' || status.lastPhase === 'initSchema',
|
|
688
|
+
`lastPhase should be "open" or "initSchema", got: ${status.lastPhase}`,
|
|
689
|
+
);
|
|
690
|
+
assert.ok(status.lastError instanceof Error, 'lastError is an Error');
|
|
691
|
+
}
|
|
692
|
+
// If somehow it succeeded (unlikely with garbage content), that's also fine
|
|
693
|
+
closeDatabase();
|
|
694
|
+
try { fs.unlinkSync(corruptPath); } catch { /* best effort */ }
|
|
695
|
+
});
|
|
696
|
+
|
|
697
|
+
test('getDbStatus: error state resets on next successful open', () => {
|
|
698
|
+
closeDatabase();
|
|
699
|
+
const corruptPath = path.join(os.tmpdir(), `gsd-corrupt-${Date.now()}.db`);
|
|
700
|
+
fs.writeFileSync(corruptPath, Buffer.from('not a sqlite file at all!!!!!'));
|
|
701
|
+
try { openDatabase(corruptPath); } catch { /* expected */ }
|
|
702
|
+
assert.ok(!getDbStatus().available, 'DB unavailable after corrupt open');
|
|
703
|
+
|
|
704
|
+
// Now open a valid in-memory DB — error state should clear
|
|
705
|
+
openDatabase(':memory:');
|
|
706
|
+
const status = getDbStatus();
|
|
707
|
+
assert.strictEqual(status.available, true, 'available after valid open');
|
|
708
|
+
assert.strictEqual(status.lastError, null, 'lastError cleared on successful open');
|
|
709
|
+
assert.strictEqual(status.lastPhase, null, 'lastPhase cleared on successful open');
|
|
710
|
+
closeDatabase();
|
|
711
|
+
try { fs.unlinkSync(corruptPath); } catch { /* best effort */ }
|
|
712
|
+
});
|
|
713
|
+
});
|
|
714
|
+
|
|
564
715
|
// ─── Final Report ──────────────────────────────────────────────────────────
|
|
565
716
|
|
|
566
717
|
});
|
|
@@ -178,6 +178,33 @@ test("init-wizard: multiple project files detected together", (t) => {
|
|
|
178
178
|
}
|
|
179
179
|
});
|
|
180
180
|
|
|
181
|
+
// ─── Git init + initial commit regression (#4530) ───────────────────────────
|
|
182
|
+
|
|
183
|
+
import { execFileSync } from "node:child_process";
|
|
184
|
+
import { nativeInit, nativeAddAll, nativeCommit } from "../native-git-bridge.ts";
|
|
185
|
+
|
|
186
|
+
test("init-wizard: nativeInit + nativeAddAll + nativeCommit produces a reachable HEAD (#4530)", (t) => {
|
|
187
|
+
// Regression: showProjectInit called nativeInit but never committed, leaving
|
|
188
|
+
// the branch unborn. git log and git worktree add both fail on zero-commit repos.
|
|
189
|
+
const dir = makeTempDir("git-init-commit");
|
|
190
|
+
t.after(() => { cleanup(dir); });
|
|
191
|
+
|
|
192
|
+
nativeInit(dir, "main");
|
|
193
|
+
execFileSync("git", ["config", "user.email", "test@test.com"], { cwd: dir });
|
|
194
|
+
execFileSync("git", ["config", "user.name", "Test"], { cwd: dir });
|
|
195
|
+
writeFileSync(join(dir, ".gitignore"), "*.log\n", "utf-8");
|
|
196
|
+
|
|
197
|
+
nativeAddAll(dir);
|
|
198
|
+
nativeCommit(dir, "chore: init project");
|
|
199
|
+
|
|
200
|
+
// git log must succeed (was: fatal: your current branch 'main' does not have any commits yet)
|
|
201
|
+
const subject = execFileSync("git", ["log", "-1", "--format=%s"], {
|
|
202
|
+
cwd: dir,
|
|
203
|
+
encoding: "utf-8",
|
|
204
|
+
}).trim();
|
|
205
|
+
assert.equal(subject, "chore: init project");
|
|
206
|
+
});
|
|
207
|
+
|
|
181
208
|
test("init-wizard: v1 with both .planning/ and .gsd/ prioritizes v2", (t) => {
|
|
182
209
|
const dir = makeTempDir("both-v1-v2");
|
|
183
210
|
try {
|
|
@@ -42,7 +42,7 @@ describe('isolation:none stale branch guard (#3675)', () => {
|
|
|
42
42
|
});
|
|
43
43
|
|
|
44
44
|
test('guard is conditional on isolation mode "none"', () => {
|
|
45
|
-
assert.match(source, /getIsolationMode\(
|
|
45
|
+
assert.match(source, /getIsolationMode\([^)]*\)\s*===\s*["']none["']/,
|
|
46
46
|
'guard should only activate when isolation mode is "none"');
|
|
47
47
|
});
|
|
48
48
|
|
|
@@ -0,0 +1,288 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Regression tests for issue #4540:
|
|
3
|
+
* Bug 1 — Invalid quality_gates migration bricks gsd.db
|
|
4
|
+
* Bug 2 — Artifact retries emit no journal event, look like stuck loops
|
|
5
|
+
*/
|
|
6
|
+
import { describe, test, beforeEach, afterEach } from "node:test";
|
|
7
|
+
import assert from "node:assert/strict";
|
|
8
|
+
import { mkdtempSync, rmSync, mkdirSync } from "node:fs";
|
|
9
|
+
import { join } from "node:path";
|
|
10
|
+
import { tmpdir } from "node:os";
|
|
11
|
+
import { createRequire } from "node:module";
|
|
12
|
+
|
|
13
|
+
import {
|
|
14
|
+
openDatabase,
|
|
15
|
+
closeDatabase,
|
|
16
|
+
insertGateRow,
|
|
17
|
+
getPendingGates,
|
|
18
|
+
_getAdapter,
|
|
19
|
+
} from "../gsd-db.ts";
|
|
20
|
+
|
|
21
|
+
import { emitJournalEvent, queryJournal } from "../journal.ts";
|
|
22
|
+
|
|
23
|
+
const _require = createRequire(import.meta.url);
|
|
24
|
+
|
|
25
|
+
// ─── helpers ─────────────────────────────────────────────────────────────────
|
|
26
|
+
|
|
27
|
+
function tmpDb(): { dir: string; dbPath: string } {
|
|
28
|
+
const dir = mkdtempSync(join(tmpdir(), "gsd-4540-"));
|
|
29
|
+
return { dir, dbPath: join(dir, "gsd.db") };
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
function cleanup(dir: string): void {
|
|
33
|
+
try { rmSync(dir, { recursive: true, force: true }); } catch { /* ignore */ }
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Builds a v12 database with the broken quality_gates DDL:
|
|
38
|
+
* task_id is nullable and there is no proper multi-column PK.
|
|
39
|
+
* This simulates a DB that was created before the v12 fix was applied.
|
|
40
|
+
*/
|
|
41
|
+
function createBrokenV12Db(dbPath: string): void {
|
|
42
|
+
const sqlite = _require("node:sqlite");
|
|
43
|
+
const db = new sqlite.DatabaseSync(dbPath);
|
|
44
|
+
db.exec("PRAGMA journal_mode=WAL");
|
|
45
|
+
|
|
46
|
+
db.exec(`CREATE TABLE schema_version (version INTEGER NOT NULL, applied_at TEXT NOT NULL)`);
|
|
47
|
+
db.prepare("INSERT INTO schema_version (version, applied_at) VALUES (?, ?)").run(12, "2025-01-01T00:00:00.000Z");
|
|
48
|
+
|
|
49
|
+
db.exec(`
|
|
50
|
+
CREATE TABLE decisions (
|
|
51
|
+
seq INTEGER PRIMARY KEY AUTOINCREMENT, id TEXT NOT NULL UNIQUE,
|
|
52
|
+
when_context TEXT NOT NULL DEFAULT '', scope TEXT NOT NULL DEFAULT '',
|
|
53
|
+
decision TEXT NOT NULL DEFAULT '', choice TEXT NOT NULL DEFAULT '',
|
|
54
|
+
rationale TEXT NOT NULL DEFAULT '', revisable TEXT NOT NULL DEFAULT '',
|
|
55
|
+
made_by TEXT NOT NULL DEFAULT 'agent', superseded_by TEXT DEFAULT NULL
|
|
56
|
+
);
|
|
57
|
+
CREATE VIEW active_decisions AS SELECT * FROM decisions WHERE superseded_by IS NULL;
|
|
58
|
+
CREATE TABLE requirements (
|
|
59
|
+
id TEXT PRIMARY KEY, class TEXT NOT NULL DEFAULT '', status TEXT NOT NULL DEFAULT '',
|
|
60
|
+
description TEXT NOT NULL DEFAULT '', why TEXT NOT NULL DEFAULT '',
|
|
61
|
+
source TEXT NOT NULL DEFAULT '', primary_owner TEXT NOT NULL DEFAULT '',
|
|
62
|
+
supporting_slices TEXT NOT NULL DEFAULT '', validation TEXT NOT NULL DEFAULT '',
|
|
63
|
+
notes TEXT NOT NULL DEFAULT '', full_content TEXT NOT NULL DEFAULT '',
|
|
64
|
+
superseded_by TEXT DEFAULT NULL
|
|
65
|
+
);
|
|
66
|
+
CREATE TABLE artifacts (
|
|
67
|
+
path TEXT PRIMARY KEY, artifact_type TEXT NOT NULL DEFAULT '',
|
|
68
|
+
milestone_id TEXT DEFAULT NULL, slice_id TEXT DEFAULT NULL, task_id TEXT DEFAULT NULL,
|
|
69
|
+
full_content TEXT NOT NULL DEFAULT '', imported_at TEXT NOT NULL DEFAULT ''
|
|
70
|
+
);
|
|
71
|
+
CREATE TABLE memories (
|
|
72
|
+
seq INTEGER PRIMARY KEY AUTOINCREMENT, id TEXT NOT NULL UNIQUE,
|
|
73
|
+
category TEXT NOT NULL, content TEXT NOT NULL, confidence REAL NOT NULL DEFAULT 0.8,
|
|
74
|
+
source_unit_type TEXT, source_unit_id TEXT, created_at TEXT NOT NULL,
|
|
75
|
+
updated_at TEXT NOT NULL, superseded_by TEXT DEFAULT NULL,
|
|
76
|
+
hit_count INTEGER NOT NULL DEFAULT 0,
|
|
77
|
+
scope TEXT NOT NULL DEFAULT 'project', tags TEXT NOT NULL DEFAULT '[]',
|
|
78
|
+
structured_fields TEXT DEFAULT NULL
|
|
79
|
+
);
|
|
80
|
+
CREATE TABLE memory_sources (
|
|
81
|
+
id TEXT PRIMARY KEY, kind TEXT NOT NULL DEFAULT '', path TEXT DEFAULT NULL,
|
|
82
|
+
imported_at TEXT NOT NULL DEFAULT '',
|
|
83
|
+
scope TEXT NOT NULL DEFAULT 'project', tags TEXT NOT NULL DEFAULT '[]'
|
|
84
|
+
);
|
|
85
|
+
CREATE TABLE memory_relations (
|
|
86
|
+
from_id TEXT NOT NULL, to_id TEXT NOT NULL, relation TEXT NOT NULL DEFAULT '',
|
|
87
|
+
PRIMARY KEY (from_id, to_id)
|
|
88
|
+
);
|
|
89
|
+
CREATE TABLE memory_processed_units (
|
|
90
|
+
unit_key TEXT PRIMARY KEY, activity_file TEXT, processed_at TEXT NOT NULL
|
|
91
|
+
);
|
|
92
|
+
CREATE TABLE milestones (
|
|
93
|
+
id TEXT PRIMARY KEY, title TEXT NOT NULL DEFAULT '', status TEXT NOT NULL DEFAULT 'active',
|
|
94
|
+
depends_on TEXT NOT NULL DEFAULT '[]', created_at TEXT NOT NULL DEFAULT '',
|
|
95
|
+
completed_at TEXT DEFAULT NULL, vision TEXT NOT NULL DEFAULT '',
|
|
96
|
+
success_criteria TEXT NOT NULL DEFAULT '[]', key_risks TEXT NOT NULL DEFAULT '[]',
|
|
97
|
+
proof_strategy TEXT NOT NULL DEFAULT '[]', verification_contract TEXT NOT NULL DEFAULT '',
|
|
98
|
+
verification_integration TEXT NOT NULL DEFAULT '',
|
|
99
|
+
verification_operational TEXT NOT NULL DEFAULT '', verification_uat TEXT NOT NULL DEFAULT '',
|
|
100
|
+
definition_of_done TEXT NOT NULL DEFAULT '[]', requirement_coverage TEXT NOT NULL DEFAULT '',
|
|
101
|
+
boundary_map_markdown TEXT NOT NULL DEFAULT ''
|
|
102
|
+
);
|
|
103
|
+
CREATE TABLE slices (
|
|
104
|
+
milestone_id TEXT NOT NULL, id TEXT NOT NULL, title TEXT NOT NULL DEFAULT '',
|
|
105
|
+
status TEXT NOT NULL DEFAULT 'pending', risk TEXT NOT NULL DEFAULT 'medium',
|
|
106
|
+
depends TEXT NOT NULL DEFAULT '[]', demo TEXT NOT NULL DEFAULT '',
|
|
107
|
+
created_at TEXT NOT NULL DEFAULT '', completed_at TEXT DEFAULT NULL,
|
|
108
|
+
full_summary_md TEXT NOT NULL DEFAULT '', full_uat_md TEXT NOT NULL DEFAULT '',
|
|
109
|
+
goal TEXT NOT NULL DEFAULT '', success_criteria TEXT NOT NULL DEFAULT '',
|
|
110
|
+
proof_level TEXT NOT NULL DEFAULT '', integration_closure TEXT NOT NULL DEFAULT '',
|
|
111
|
+
observability_impact TEXT NOT NULL DEFAULT '', sequence INTEGER DEFAULT 0,
|
|
112
|
+
replan_triggered_at TEXT DEFAULT NULL,
|
|
113
|
+
is_sketch INTEGER NOT NULL DEFAULT 0, sketch_scope TEXT NOT NULL DEFAULT '',
|
|
114
|
+
PRIMARY KEY (milestone_id, id), FOREIGN KEY (milestone_id) REFERENCES milestones(id)
|
|
115
|
+
);
|
|
116
|
+
CREATE TABLE tasks (
|
|
117
|
+
milestone_id TEXT NOT NULL, slice_id TEXT NOT NULL, id TEXT NOT NULL,
|
|
118
|
+
title TEXT NOT NULL DEFAULT '', status TEXT NOT NULL DEFAULT 'pending',
|
|
119
|
+
one_liner TEXT NOT NULL DEFAULT '', narrative TEXT NOT NULL DEFAULT '',
|
|
120
|
+
verification_result TEXT NOT NULL DEFAULT '',
|
|
121
|
+
escalation_pending INTEGER NOT NULL DEFAULT 0,
|
|
122
|
+
PRIMARY KEY (milestone_id, slice_id, id),
|
|
123
|
+
FOREIGN KEY (milestone_id, slice_id) REFERENCES slices(milestone_id, id)
|
|
124
|
+
);
|
|
125
|
+
CREATE TABLE assessments (
|
|
126
|
+
path TEXT PRIMARY KEY, milestone_id TEXT NOT NULL DEFAULT '',
|
|
127
|
+
slice_id TEXT DEFAULT NULL, task_id TEXT DEFAULT NULL,
|
|
128
|
+
status TEXT NOT NULL DEFAULT '', scope TEXT NOT NULL DEFAULT '',
|
|
129
|
+
full_content TEXT NOT NULL DEFAULT '', created_at TEXT NOT NULL DEFAULT '',
|
|
130
|
+
FOREIGN KEY (milestone_id) REFERENCES milestones(id)
|
|
131
|
+
);
|
|
132
|
+
CREATE TABLE replan_history (
|
|
133
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT, milestone_id TEXT NOT NULL,
|
|
134
|
+
slice_id TEXT DEFAULT NULL, task_id TEXT DEFAULT NULL,
|
|
135
|
+
reason TEXT NOT NULL DEFAULT '', created_at TEXT NOT NULL DEFAULT ''
|
|
136
|
+
);
|
|
137
|
+
CREATE TABLE verification_evidence (
|
|
138
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT, milestone_id TEXT NOT NULL DEFAULT '',
|
|
139
|
+
slice_id TEXT NOT NULL DEFAULT '', task_id TEXT NOT NULL DEFAULT '',
|
|
140
|
+
unit_type TEXT NOT NULL DEFAULT '', unit_id TEXT NOT NULL DEFAULT '',
|
|
141
|
+
evidence_type TEXT NOT NULL DEFAULT '', content TEXT NOT NULL DEFAULT '',
|
|
142
|
+
created_at TEXT NOT NULL DEFAULT '',
|
|
143
|
+
command TEXT NOT NULL DEFAULT '', verdict TEXT NOT NULL DEFAULT ''
|
|
144
|
+
);
|
|
145
|
+
`);
|
|
146
|
+
|
|
147
|
+
// Broken quality_gates: task_id nullable, no multi-column PK
|
|
148
|
+
db.exec(`
|
|
149
|
+
CREATE TABLE quality_gates (
|
|
150
|
+
milestone_id TEXT NOT NULL, slice_id TEXT NOT NULL, gate_id TEXT NOT NULL,
|
|
151
|
+
scope TEXT NOT NULL DEFAULT 'slice', task_id TEXT DEFAULT NULL,
|
|
152
|
+
status TEXT NOT NULL DEFAULT 'pending', verdict TEXT NOT NULL DEFAULT '',
|
|
153
|
+
rationale TEXT NOT NULL DEFAULT '', findings TEXT NOT NULL DEFAULT '',
|
|
154
|
+
evaluated_at TEXT DEFAULT NULL,
|
|
155
|
+
FOREIGN KEY (milestone_id, slice_id) REFERENCES slices(milestone_id, id)
|
|
156
|
+
)
|
|
157
|
+
`);
|
|
158
|
+
|
|
159
|
+
// Parent rows + gate row with NULL task_id
|
|
160
|
+
db.prepare("INSERT INTO milestones (id, title, status) VALUES (?, ?, ?)").run("M001", "Milestone 1", "active");
|
|
161
|
+
db.prepare("INSERT INTO slices (milestone_id, id, title, status, risk, depends) VALUES (?, ?, ?, ?, ?, ?)").run("M001", "S01", "Slice 1", "pending", "medium", "[]");
|
|
162
|
+
db.prepare("INSERT INTO quality_gates (milestone_id, slice_id, gate_id, scope, task_id, status) VALUES (?, ?, ?, ?, ?, ?)").run("M001", "S01", "Q3", "slice", null, "pending");
|
|
163
|
+
|
|
164
|
+
db.close();
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
// ─── Bug 1 tests ─────────────────────────────────────────────────────────────
|
|
168
|
+
|
|
169
|
+
describe("Bug 1 — quality_gates migration repair (#4540)", () => {
|
|
170
|
+
let dir: string;
|
|
171
|
+
let dbPath: string;
|
|
172
|
+
|
|
173
|
+
beforeEach(() => {
|
|
174
|
+
({ dir, dbPath } = tmpDb());
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
afterEach(() => {
|
|
178
|
+
closeDatabase();
|
|
179
|
+
cleanup(dir);
|
|
180
|
+
});
|
|
181
|
+
|
|
182
|
+
test("fresh DB: quality_gates task_id is NOT NULL with empty-string default", () => {
|
|
183
|
+
openDatabase(dbPath);
|
|
184
|
+
const adapter = _getAdapter()!;
|
|
185
|
+
const cols = adapter.prepare("PRAGMA table_info(quality_gates)").all() as Array<Record<string, unknown>>;
|
|
186
|
+
const col = cols.find((c) => c["name"] === "task_id");
|
|
187
|
+
assert.ok(col, "task_id column must exist");
|
|
188
|
+
assert.equal(col["notnull"], 1, "task_id must be NOT NULL");
|
|
189
|
+
assert.equal(col["dflt_value"], "''", "task_id default must be ''");
|
|
190
|
+
});
|
|
191
|
+
|
|
192
|
+
test("fresh DB: insertGateRow with no taskId stores '' and is idempotent", () => {
|
|
193
|
+
openDatabase(dbPath);
|
|
194
|
+
const adapter = _getAdapter()!;
|
|
195
|
+
adapter.prepare("INSERT OR IGNORE INTO milestones (id, title, status) VALUES (?, ?, ?)").run("M001", "Test", "active");
|
|
196
|
+
adapter.prepare("INSERT OR IGNORE INTO slices (milestone_id, id, title, status, risk, depends) VALUES (?, ?, ?, ?, ?, ?)").run("M001", "S01", "Slice", "pending", "medium", "[]");
|
|
197
|
+
|
|
198
|
+
insertGateRow({ milestoneId: "M001", sliceId: "S01", gateId: "Q3", scope: "slice" });
|
|
199
|
+
insertGateRow({ milestoneId: "M001", sliceId: "S01", gateId: "Q3", scope: "slice" });
|
|
200
|
+
const rows = getPendingGates("M001", "S01");
|
|
201
|
+
assert.equal(rows.length, 1);
|
|
202
|
+
assert.equal(rows[0].task_id, "");
|
|
203
|
+
});
|
|
204
|
+
|
|
205
|
+
test("v22 repair: broken v12 DB opens without error", () => {
|
|
206
|
+
createBrokenV12Db(dbPath);
|
|
207
|
+
assert.doesNotThrow(() => openDatabase(dbPath));
|
|
208
|
+
});
|
|
209
|
+
|
|
210
|
+
test("v22 repair: task_id becomes NOT NULL after healing broken v12 DB", () => {
|
|
211
|
+
createBrokenV12Db(dbPath);
|
|
212
|
+
openDatabase(dbPath);
|
|
213
|
+
const adapter = _getAdapter()!;
|
|
214
|
+
const cols = adapter.prepare("PRAGMA table_info(quality_gates)").all() as Array<Record<string, unknown>>;
|
|
215
|
+
const col = cols.find((c) => c["name"] === "task_id");
|
|
216
|
+
assert.ok(col, "task_id column must exist after repair");
|
|
217
|
+
assert.equal(col["notnull"], 1, "task_id must be NOT NULL after repair");
|
|
218
|
+
});
|
|
219
|
+
|
|
220
|
+
test("v22 repair: existing NULL task_id row is COALESCE'd to '' during repair", () => {
|
|
221
|
+
createBrokenV12Db(dbPath);
|
|
222
|
+
openDatabase(dbPath);
|
|
223
|
+
const adapter = _getAdapter()!;
|
|
224
|
+
const row = adapter.prepare(
|
|
225
|
+
"SELECT task_id FROM quality_gates WHERE milestone_id = 'M001' AND slice_id = 'S01' AND gate_id = 'Q3'"
|
|
226
|
+
).get() as Record<string, unknown> | undefined;
|
|
227
|
+
assert.ok(row, "original gate row must survive the repair migration");
|
|
228
|
+
assert.equal(row["task_id"], "", "NULL task_id must be repaired to ''");
|
|
229
|
+
});
|
|
230
|
+
|
|
231
|
+
test("v22 repair: scope column present on quality_gates after open", () => {
|
|
232
|
+
openDatabase(dbPath);
|
|
233
|
+
const adapter = _getAdapter()!;
|
|
234
|
+
const cols = adapter.prepare("PRAGMA table_info(quality_gates)").all() as Array<Record<string, unknown>>;
|
|
235
|
+
assert.ok(cols.some((c) => c["name"] === "scope"), "quality_gates.scope must exist");
|
|
236
|
+
});
|
|
237
|
+
|
|
238
|
+
test("v22 repair: scope column present on assessments after open", () => {
|
|
239
|
+
openDatabase(dbPath);
|
|
240
|
+
const adapter = _getAdapter()!;
|
|
241
|
+
const cols = adapter.prepare("PRAGMA table_info(assessments)").all() as Array<Record<string, unknown>>;
|
|
242
|
+
assert.ok(cols.some((c) => c["name"] === "scope"), "assessments.scope must exist");
|
|
243
|
+
});
|
|
244
|
+
});
|
|
245
|
+
|
|
246
|
+
// ─── Bug 2 tests ─────────────────────────────────────────────────────────────
|
|
247
|
+
|
|
248
|
+
describe("Bug 2 — artifact-verification-retry journal event (#4540)", () => {
|
|
249
|
+
test("emitJournalEvent accepts artifact-verification-retry event type", () => {
|
|
250
|
+
const basePath = mkdtempSync(join(tmpdir(), "gsd-journal-4540-"));
|
|
251
|
+
try {
|
|
252
|
+
mkdirSync(join(basePath, ".gsd"), { recursive: true });
|
|
253
|
+
emitJournalEvent(basePath, {
|
|
254
|
+
ts: new Date().toISOString(),
|
|
255
|
+
flowId: "flow-4540",
|
|
256
|
+
seq: 1,
|
|
257
|
+
eventType: "artifact-verification-retry",
|
|
258
|
+
data: { unitType: "plan-slice", unitId: "M001/S01", attempt: 1 },
|
|
259
|
+
});
|
|
260
|
+
const entries = queryJournal(basePath, { flowId: "flow-4540" });
|
|
261
|
+
assert.equal(entries.length, 1);
|
|
262
|
+
assert.equal(entries[0].eventType, "artifact-verification-retry");
|
|
263
|
+
} finally {
|
|
264
|
+
rmSync(basePath, { recursive: true, force: true });
|
|
265
|
+
}
|
|
266
|
+
});
|
|
267
|
+
|
|
268
|
+
test("artifact-verification-retry event carries attempt count", () => {
|
|
269
|
+
const basePath = mkdtempSync(join(tmpdir(), "gsd-journal-4540b-"));
|
|
270
|
+
try {
|
|
271
|
+
mkdirSync(join(basePath, ".gsd"), { recursive: true });
|
|
272
|
+
emitJournalEvent(basePath, {
|
|
273
|
+
ts: new Date().toISOString(),
|
|
274
|
+
flowId: "flow-4540b",
|
|
275
|
+
seq: 1,
|
|
276
|
+
eventType: "artifact-verification-retry",
|
|
277
|
+
data: { unitType: "exec-slice", unitId: "M002/S02", attempt: 2 },
|
|
278
|
+
});
|
|
279
|
+
const entries = queryJournal(basePath, { flowId: "flow-4540b", eventType: "artifact-verification-retry" });
|
|
280
|
+
assert.equal(entries.length, 1);
|
|
281
|
+
const payload = entries[0].data as Record<string, unknown>;
|
|
282
|
+
assert.equal(payload["attempt"], 2);
|
|
283
|
+
assert.equal(payload["unitId"], "M002/S02");
|
|
284
|
+
} finally {
|
|
285
|
+
rmSync(basePath, { recursive: true, force: true });
|
|
286
|
+
}
|
|
287
|
+
});
|
|
288
|
+
});
|
|
@@ -143,6 +143,13 @@ test("PROVIDER_REGISTRY includes all major LLM providers", () => {
|
|
|
143
143
|
assert.ok(ids.includes("groq"));
|
|
144
144
|
});
|
|
145
145
|
|
|
146
|
+
test("PROVIDER_REGISTRY includes claude-code as a first-class LLM provider (#4541)", () => {
|
|
147
|
+
const entry = PROVIDER_REGISTRY.find((p) => p.id === "claude-code");
|
|
148
|
+
assert.ok(entry, "claude-code must be in PROVIDER_REGISTRY");
|
|
149
|
+
assert.equal(entry!.category, "llm");
|
|
150
|
+
assert.ok(entry!.hasOAuth, "claude-code uses OAuth (CLI auth)");
|
|
151
|
+
});
|
|
152
|
+
|
|
146
153
|
test("PROVIDER_REGISTRY includes all tool/search providers", () => {
|
|
147
154
|
const ids = PROVIDER_REGISTRY.map((p) => p.id);
|
|
148
155
|
assert.ok(ids.includes("tavily"));
|
|
@@ -363,7 +363,7 @@ test('md-importer: schema v1→v2 migration', () => {
|
|
|
363
363
|
openDatabase(':memory:');
|
|
364
364
|
const adapter = _getAdapter();
|
|
365
365
|
const version = adapter?.prepare('SELECT MAX(version) as v FROM schema_version').get();
|
|
366
|
-
assert.deepStrictEqual(version?.v,
|
|
366
|
+
assert.deepStrictEqual(version?.v, 22, 'new DB should be at schema version 22');
|
|
367
367
|
|
|
368
368
|
// Artifacts table should exist
|
|
369
369
|
const tableCheck = adapter?.prepare("SELECT count(*) as c FROM sqlite_master WHERE type='table' AND name='artifacts'").get();
|
|
@@ -323,9 +323,9 @@ test('memory-store: schema includes memories table', () => {
|
|
|
323
323
|
const viewCount = adapter.prepare('SELECT count(*) as cnt FROM active_memories').get();
|
|
324
324
|
assert.deepStrictEqual(viewCount?.['cnt'], 0, 'active_memories view should exist');
|
|
325
325
|
|
|
326
|
-
// Verify schema version is
|
|
326
|
+
// Verify schema version is 22 (v22 quality_gates DDL fix included)
|
|
327
327
|
const version = adapter.prepare('SELECT MAX(version) as v FROM schema_version').get();
|
|
328
|
-
assert.deepStrictEqual(version?.["v"],
|
|
328
|
+
assert.deepStrictEqual(version?.["v"], 22, 'schema version should be 22');
|
|
329
329
|
|
|
330
330
|
closeDatabase();
|
|
331
331
|
});
|
|
@@ -110,6 +110,25 @@ test("template: parallel-research-slices.md has required variables", () => {
|
|
|
110
110
|
assert.ok(templateSrc.includes("{{subagentPrompts}}"), "template should use subagentPrompts");
|
|
111
111
|
});
|
|
112
112
|
|
|
113
|
+
test("#4068: template: parallel-research-slices retry cap prevents infinite subagent loop", () => {
|
|
114
|
+
// The template must cap retries at 1 ("retry it once") and instruct the
|
|
115
|
+
// agent to write a BLOCKER note on the second failure rather than looping.
|
|
116
|
+
// Without this, a timing-out subagent causes the orchestrating agent to
|
|
117
|
+
// retry indefinitely (issue #4068 / #4355).
|
|
118
|
+
assert.ok(
|
|
119
|
+
templateSrc.includes("once") || templateSrc.includes("one retry") || templateSrc.match(/retry.{0,20}once/),
|
|
120
|
+
"template should cap subagent retries at one",
|
|
121
|
+
);
|
|
122
|
+
assert.ok(
|
|
123
|
+
templateSrc.toLowerCase().includes("blocker"),
|
|
124
|
+
"template should instruct writing a BLOCKER note instead of infinite retries",
|
|
125
|
+
);
|
|
126
|
+
assert.ok(
|
|
127
|
+
!templateSrc.match(/re-run it individually\s*\n/),
|
|
128
|
+
"template must not have unbounded re-run instruction without a retry cap",
|
|
129
|
+
);
|
|
130
|
+
});
|
|
131
|
+
|
|
113
132
|
// ─── Validate milestone prompt ────────────────────────────────────────────
|
|
114
133
|
|
|
115
134
|
test("template: validate-milestone uses parallel reviewers", () => {
|
|
@@ -30,6 +30,20 @@ describe('normalizeFilePath backtick stripping (#3649)', () => {
|
|
|
30
30
|
assert.equal(normalizeFilePath('``src/foo.ts`` (current state)'), 'src/foo.ts')
|
|
31
31
|
})
|
|
32
32
|
|
|
33
|
+
it('strips stray backticks from dash-annotated bare paths (#4550)', () => {
|
|
34
|
+
assert.equal(
|
|
35
|
+
normalizeFilePath('.gsd/KNOWLEDGE.md` — append-only S05 lessons section'),
|
|
36
|
+
'.gsd/KNOWLEDGE.md',
|
|
37
|
+
)
|
|
38
|
+
})
|
|
39
|
+
|
|
40
|
+
it('prefers a backticked path inside a dash-annotated prefix (#4550)', () => {
|
|
41
|
+
assert.equal(
|
|
42
|
+
normalizeFilePath('Input `src/foo.ts` — current state'),
|
|
43
|
+
'src/foo.ts',
|
|
44
|
+
)
|
|
45
|
+
})
|
|
46
|
+
|
|
33
47
|
it('strips backticks even when mixed with other normalization', () => {
|
|
34
48
|
assert.equal(normalizeFilePath('`./src//bar.ts`'), 'src/bar.ts')
|
|
35
49
|
})
|