gsd-pi 2.76.0 → 2.77.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +45 -25
- 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/onboarding.js +45 -0
- package/dist/resource-loader.d.ts +1 -1
- package/dist/resource-loader.js +2 -8
- package/dist/resources/agents/researcher.md +1 -1
- package/dist/resources/extensions/claude-code-cli/readiness.js +31 -8
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +77 -17
- package/dist/resources/extensions/gsd/auto/loop.js +9 -0
- package/dist/resources/extensions/gsd/auto/phases.js +104 -11
- package/dist/resources/extensions/gsd/auto/run-unit.js +38 -2
- package/dist/resources/extensions/gsd/auto/session.js +22 -1
- package/dist/resources/extensions/gsd/auto-dispatch.js +16 -3
- package/dist/resources/extensions/gsd/auto-model-selection.js +53 -16
- package/dist/resources/extensions/gsd/auto-post-unit.js +25 -2
- package/dist/resources/extensions/gsd/auto-prompts.js +14 -0
- package/dist/resources/extensions/gsd/auto-recovery.js +32 -1
- package/dist/resources/extensions/gsd/auto-start.js +58 -57
- package/dist/resources/extensions/gsd/auto-verification.js +33 -0
- package/dist/resources/extensions/gsd/auto-worktree.js +51 -53
- package/dist/resources/extensions/gsd/auto.js +70 -28
- package/dist/resources/extensions/gsd/blocked-models.js +68 -0
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +93 -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/memory-tools.js +3 -0
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +12 -0
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +52 -6
- package/dist/resources/extensions/gsd/bootstrap/system-context.js +84 -23
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +34 -2
- package/dist/resources/extensions/gsd/clean-root-preflight.js +93 -0
- package/dist/resources/extensions/gsd/commands-extract-learnings.js +54 -89
- 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/complexity-classifier.js +5 -3
- package/dist/resources/extensions/gsd/db-writer.js +88 -16
- package/dist/resources/extensions/gsd/doctor-git-checks.js +23 -29
- package/dist/resources/extensions/gsd/doctor-providers.js +51 -5
- package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +1 -0
- package/dist/resources/extensions/gsd/error-classifier.js +31 -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/gitignore.js +1 -0
- package/dist/resources/extensions/gsd/gsd-db.js +168 -23
- package/dist/resources/extensions/gsd/guided-flow.js +190 -1
- package/dist/resources/extensions/gsd/health-widget.js +4 -1
- package/dist/resources/extensions/gsd/hook-emitter.js +108 -0
- package/dist/resources/extensions/gsd/init-wizard.js +15 -1
- package/dist/resources/extensions/gsd/key-manager.js +28 -0
- package/dist/resources/extensions/gsd/memory-backfill.js +126 -0
- package/dist/resources/extensions/gsd/memory-store.js +19 -0
- package/dist/resources/extensions/gsd/model-router.js +36 -3
- package/dist/resources/extensions/gsd/pre-execution-checks.js +44 -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/prompt-loader.js +22 -7
- package/dist/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -2
- package/dist/resources/extensions/gsd/prompts/debug-diagnose.md +2 -0
- 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/execute-task.md +3 -2
- package/dist/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -0
- package/dist/resources/extensions/gsd/prompts/research-slice.md +1 -0
- package/dist/resources/extensions/gsd/safety/evidence-collector.js +96 -0
- package/dist/resources/extensions/gsd/safety/file-change-validator.js +13 -5
- package/dist/resources/extensions/gsd/safety/safety-harness.js +5 -1
- package/dist/resources/extensions/gsd/state.js +43 -4
- package/dist/resources/extensions/gsd/token-counter.js +22 -5
- package/dist/resources/extensions/gsd/tools/complete-milestone.js +16 -10
- 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/memory-tools.js +26 -1
- package/dist/resources/extensions/gsd/tools/resume-tool.js +23 -0
- package/dist/resources/extensions/gsd/uok/plan-v2.js +20 -3
- package/dist/resources/extensions/gsd/workflow-mcp.js +3 -0
- package/dist/resources/extensions/gsd/workflow-templates/spike.md +6 -0
- package/dist/resources/extensions/gsd/worktree-resolver.js +50 -10
- package/dist/resources/extensions/search-the-web/command-search-provider.js +5 -4
- package/dist/resources/extensions/search-the-web/native-search.js +45 -13
- package/dist/resources/skills/api-design/SKILL.md +190 -0
- package/dist/resources/skills/create-mcp-server/SKILL.md +121 -0
- package/dist/resources/skills/decompose-into-slices/SKILL.md +139 -0
- package/dist/resources/skills/dependency-upgrade/SKILL.md +158 -0
- package/dist/resources/skills/design-an-interface/SKILL.md +102 -0
- package/dist/resources/skills/forensics/SKILL.md +153 -0
- package/dist/resources/skills/grill-me/SKILL.md +93 -0
- package/dist/resources/skills/handoff/SKILL.md +121 -0
- package/dist/resources/skills/observability/SKILL.md +174 -0
- package/dist/resources/skills/security-review/SKILL.md +181 -0
- package/dist/resources/skills/spike-wrap-up/SKILL.md +138 -0
- package/dist/resources/skills/tdd/SKILL.md +112 -0
- package/dist/resources/skills/verify-before-complete/SKILL.md +98 -0
- package/dist/resources/skills/write-docs/SKILL.md +82 -0
- package/dist/resources/skills/write-milestone-brief/SKILL.md +135 -0
- 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 +10 -10
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- 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 +10 -10
- package/dist/web/standalone/.next/server/chunks/6897.js +2 -2
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-manifest.json +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/server.js +1 -1
- package/dist/welcome-screen.js +6 -1
- package/dist/wizard.js +2 -0
- package/package.json +1 -1
- package/packages/daemon/package.json +2 -2
- 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 +7 -0
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +70 -8
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/session-manager.d.ts +14 -0
- package/packages/mcp-server/dist/session-manager.d.ts.map +1 -1
- package/packages/mcp-server/dist/session-manager.js +49 -1
- package/packages/mcp-server/dist/session-manager.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +163 -25
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +4 -3
- package/packages/mcp-server/src/mcp-server.test.ts +67 -0
- 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 +89 -14
- package/packages/mcp-server/src/session-manager.ts +43 -1
- package/packages/mcp-server/src/workflow-tools.test.ts +146 -1
- package/packages/mcp-server/src/workflow-tools.ts +215 -43
- package/packages/mcp-server/tsconfig.test.json +19 -0
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
- package/packages/native/package.json +1 -1
- package/packages/pi-agent-core/dist/agent-loop.js +12 -0
- package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
- package/packages/pi-agent-core/dist/types.d.ts +30 -0
- package/packages/pi-agent-core/dist/types.d.ts.map +1 -1
- package/packages/pi-agent-core/dist/types.js.map +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-agent-core/src/agent-loop.ts +14 -0
- package/packages/pi-agent-core/src/types.ts +34 -0
- package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-ai/dist/models/custom.d.ts +38 -0
- package/packages/pi-ai/dist/models/custom.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/custom.js +41 -0
- package/packages/pi-ai/dist/models/custom.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-auth.test.js +1 -1
- package/packages/pi-ai/dist/providers/anthropic-auth.test.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.d.ts +2 -0
- package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.js +13 -0
- package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.js.map +1 -0
- package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.js +27 -4
- package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.js +13 -4
- package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
- package/packages/pi-ai/dist/providers/minimax-tool-name.test.d.ts +2 -0
- package/packages/pi-ai/dist/providers/minimax-tool-name.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/minimax-tool-name.test.js +80 -0
- package/packages/pi-ai/dist/providers/minimax-tool-name.test.js.map +1 -0
- package/packages/pi-ai/dist/providers/openai-completions.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/openai-completions.js +60 -15
- package/packages/pi-ai/dist/providers/openai-completions.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/dist/providers/think-tag-parser.d.ts +17 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.js +75 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.js.map +1 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.test.d.ts +2 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.test.js +41 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.test.js.map +1 -0
- package/packages/pi-ai/dist/utils/oauth/github-copilot.d.ts.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/github-copilot.js +12 -2
- package/packages/pi-ai/dist/utils/oauth/github-copilot.js.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/github-copilot.test.js +164 -14
- package/packages/pi-ai/dist/utils/oauth/github-copilot.test.js.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/google-antigravity.d.ts.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/google-antigravity.js +15 -3
- package/packages/pi-ai/dist/utils/oauth/google-antigravity.js.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.d.ts +2 -0
- package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.js +67 -0
- package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.js.map +1 -0
- package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.d.ts.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.js +16 -3
- package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.js.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.d.ts +2 -0
- package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.js +67 -0
- package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.js.map +1 -0
- package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts +2 -0
- package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js +289 -0
- package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js.map +1 -0
- package/packages/pi-ai/package.json +1 -1
- package/packages/pi-ai/src/models/custom.ts +42 -0
- package/packages/pi-ai/src/providers/anthropic-auth.test.ts +1 -1
- package/packages/pi-ai/src/providers/anthropic-bearer-auth.test.ts +26 -0
- package/packages/pi-ai/src/providers/anthropic-shared.ts +26 -5
- package/packages/pi-ai/src/providers/anthropic.ts +15 -4
- package/packages/pi-ai/src/providers/minimax-tool-name.test.ts +98 -0
- package/packages/pi-ai/src/providers/openai-completions.ts +57 -16
- package/packages/pi-ai/src/providers/simple-options.ts +17 -1
- package/packages/pi-ai/src/providers/think-tag-parser.test.ts +44 -0
- package/packages/pi-ai/src/providers/think-tag-parser.ts +94 -0
- package/packages/pi-ai/src/utils/oauth/github-copilot.test.ts +200 -23
- package/packages/pi-ai/src/utils/oauth/github-copilot.ts +12 -2
- package/packages/pi-ai/src/utils/oauth/google-antigravity.test.ts +84 -0
- package/packages/pi-ai/src/utils/oauth/google-antigravity.ts +15 -5
- package/packages/pi-ai/src/utils/oauth/google-gemini-cli.test.ts +84 -0
- package/packages/pi-ai/src/utils/oauth/google-gemini-cli.ts +16 -5
- package/packages/pi-ai/src/utils/oauth/oauth-providers.test.ts +363 -0
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +3 -2
- package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.js +32 -2
- package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/index.d.ts +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/index.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/index.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/loader.js +4 -0
- package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +35 -2
- package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.js +233 -0
- package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +205 -2
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/hooks-runner.d.ts +53 -0
- package/packages/pi-coding-agent/dist/core/hooks-runner.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/hooks-runner.js +337 -0
- package/packages/pi-coding-agent/dist/core/hooks-runner.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/hooks-runner.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/hooks-runner.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/hooks-runner.test.js +234 -0
- package/packages/pi-coding-agent/dist/core/hooks-runner.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/index.d.ts +1 -0
- package/packages/pi-coding-agent/dist/core/index.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/index.js +1 -0
- package/packages/pi-coding-agent/dist/core/index.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-discovery.d.ts +3 -1
- package/packages/pi-coding-agent/dist/core/model-discovery.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-discovery.js +92 -12
- package/packages/pi-coding-agent/dist/core/model-discovery.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-discovery.test.js +16 -1
- package/packages/pi-coding-agent/dist/core/model-discovery.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.js +40 -0
- package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.js.map +1 -0
- 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-discovery.test.js +61 -1
- package/packages/pi-coding-agent/dist/core/model-registry-discovery.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts +5 -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 +90 -10
- 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 +10 -6
- package/packages/pi-coding-agent/dist/core/session-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/session-manager.test.js +45 -1
- package/packages/pi-coding-agent/dist/core/session-manager.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +55 -0
- package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/index.d.ts +1 -1
- package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/index.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/provider-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.js +13 -7
- package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.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/package.json +1 -1
- package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +3 -2
- package/packages/pi-coding-agent/src/core/agent-session.ts +38 -2
- package/packages/pi-coding-agent/src/core/extensions/index.ts +16 -0
- package/packages/pi-coding-agent/src/core/extensions/loader.ts +5 -0
- package/packages/pi-coding-agent/src/core/extensions/runner.ts +351 -0
- package/packages/pi-coding-agent/src/core/extensions/types.ts +258 -0
- package/packages/pi-coding-agent/src/core/hooks-runner.test.ts +269 -0
- package/packages/pi-coding-agent/src/core/hooks-runner.ts +460 -0
- package/packages/pi-coding-agent/src/core/index.ts +10 -0
- package/packages/pi-coding-agent/src/core/model-discovery.test.ts +19 -0
- package/packages/pi-coding-agent/src/core/model-discovery.ts +99 -12
- package/packages/pi-coding-agent/src/core/model-registry-auth-header.test.ts +44 -0
- package/packages/pi-coding-agent/src/core/model-registry-custom-caps.test.ts +245 -0
- package/packages/pi-coding-agent/src/core/model-registry-discovery.test.ts +75 -0
- package/packages/pi-coding-agent/src/core/model-registry.ts +102 -10
- 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 +65 -1
- package/packages/pi-coding-agent/src/core/session-manager.ts +10 -6
- package/packages/pi-coding-agent/src/core/settings-manager.ts +57 -0
- package/packages/pi-coding-agent/src/index.ts +16 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +6 -6
- package/packages/pi-coding-agent/src/modes/interactive/components/provider-manager.ts +16 -7
- 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/packages/pi-tui/package.json +1 -1
- package/packages/rpc-client/package.json +1 -1
- package/pkg/package.json +1 -1
- package/scripts/link-workspace-packages.cjs +1 -0
- package/src/resources/agents/researcher.md +1 -1
- package/src/resources/extensions/claude-code-cli/readiness.ts +32 -8
- 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/loop-deps.ts +14 -0
- package/src/resources/extensions/gsd/auto/loop.ts +9 -0
- package/src/resources/extensions/gsd/auto/phases.ts +131 -10
- package/src/resources/extensions/gsd/auto/run-unit.ts +40 -2
- package/src/resources/extensions/gsd/auto/session.ts +35 -2
- package/src/resources/extensions/gsd/auto-dispatch.ts +16 -3
- package/src/resources/extensions/gsd/auto-model-selection.ts +71 -15
- package/src/resources/extensions/gsd/auto-post-unit.ts +29 -3
- package/src/resources/extensions/gsd/auto-prompts.ts +28 -1
- package/src/resources/extensions/gsd/auto-recovery.ts +26 -1
- package/src/resources/extensions/gsd/auto-start.ts +60 -68
- package/src/resources/extensions/gsd/auto-verification.ts +33 -0
- package/src/resources/extensions/gsd/auto-worktree.ts +62 -63
- package/src/resources/extensions/gsd/auto.ts +73 -28
- package/src/resources/extensions/gsd/blocked-models.ts +98 -0
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +120 -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/memory-tools.ts +5 -0
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +15 -0
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +54 -6
- package/src/resources/extensions/gsd/bootstrap/system-context.ts +89 -26
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +35 -2
- package/src/resources/extensions/gsd/clean-root-preflight.ts +111 -0
- package/src/resources/extensions/gsd/commands-extract-learnings.ts +55 -90
- 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/complexity-classifier.ts +5 -3
- package/src/resources/extensions/gsd/db-writer.ts +88 -17
- package/src/resources/extensions/gsd/doctor-git-checks.ts +23 -27
- package/src/resources/extensions/gsd/doctor-providers.ts +59 -6
- package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +2 -0
- package/src/resources/extensions/gsd/error-classifier.ts +36 -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/gitignore.ts +1 -1
- package/src/resources/extensions/gsd/gsd-db.ts +186 -23
- package/src/resources/extensions/gsd/guided-flow.ts +222 -1
- package/src/resources/extensions/gsd/health-widget.ts +3 -1
- package/src/resources/extensions/gsd/hook-emitter.ts +188 -0
- 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 +28 -0
- package/src/resources/extensions/gsd/memory-backfill.ts +140 -0
- package/src/resources/extensions/gsd/memory-store.ts +26 -0
- package/src/resources/extensions/gsd/model-router.ts +42 -1
- package/src/resources/extensions/gsd/pre-execution-checks.ts +46 -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/prompt-loader.ts +30 -7
- package/src/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
- package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -2
- package/src/resources/extensions/gsd/prompts/debug-diagnose.md +2 -0
- 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/execute-task.md +3 -2
- package/src/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
- package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -0
- package/src/resources/extensions/gsd/prompts/research-slice.md +1 -0
- package/src/resources/extensions/gsd/safety/evidence-collector.ts +119 -0
- package/src/resources/extensions/gsd/safety/file-change-validator.ts +17 -4
- package/src/resources/extensions/gsd/safety/safety-harness.ts +9 -0
- package/src/resources/extensions/gsd/state.ts +45 -4
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +188 -2
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +95 -1
- package/src/resources/extensions/gsd/tests/auto-paused-session-validation.test.ts +12 -0
- package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +49 -0
- package/src/resources/extensions/gsd/tests/auto-start-bootstrap-await-3420.test.ts +141 -0
- package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +33 -3
- package/src/resources/extensions/gsd/tests/auto-thinking-restore.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/auto-wrapup-inflight-guard.test.ts +23 -0
- package/src/resources/extensions/gsd/tests/blocked-models.test.ts +98 -0
- package/src/resources/extensions/gsd/tests/bundled-skill-triggers.test.ts +54 -0
- package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +186 -0
- package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +68 -66
- package/src/resources/extensions/gsd/tests/compaction-snapshot.test.ts +123 -0
- package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +61 -1
- 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/complexity-classifier.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +2 -0
- package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +8 -4
- package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +148 -3
- package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/ensure-db-open.test.ts +306 -1
- package/src/resources/extensions/gsd/tests/escalation.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/exec-history.test.ts +237 -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/flat-rate-routing-guard.test.ts +40 -9
- package/src/resources/extensions/gsd/tests/freeform-decisions.test.ts +62 -0
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +447 -1
- package/src/resources/extensions/gsd/tests/init-wizard.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/integration/doctor-git-symlink-cwd.test.ts +11 -0
- package/src/resources/extensions/gsd/tests/integration/doctor-git.test.ts +78 -0
- package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/integration/gitignore-tracked-gsd.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/integration/idle-recovery.test.ts +30 -0
- package/src/resources/extensions/gsd/tests/interactive-routing-bypass.test.ts +1 -1
- 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/journal-integration.test.ts +37 -0
- package/src/resources/extensions/gsd/tests/key-manager.test.ts +9 -0
- package/src/resources/extensions/gsd/tests/load-memory-block.test.ts +36 -0
- package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/memory-pressure-stuck-state.test.ts +12 -0
- 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/plan-gate-failed-doctor-heal-hint.test.ts +37 -0
- package/src/resources/extensions/gsd/tests/pre-exec-backtick-strip.test.ts +14 -0
- package/src/resources/extensions/gsd/tests/pre-exec-gate-loop.test.ts +272 -0
- package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +356 -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/prompt-loader-extension-dir.test.ts +49 -0
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +103 -4
- 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/resume-dispatch-worktree.test.ts +230 -0
- package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +205 -0
- package/src/resources/extensions/gsd/tests/save-gate-result-render.test.ts +95 -0
- package/src/resources/extensions/gsd/tests/schema-v21-sequence.test.ts +413 -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/uok-plan-v2-wiring.test.ts +23 -0
- package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +9 -3
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +65 -2
- package/src/resources/extensions/gsd/tests/worktree-db.test.ts +35 -0
- package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +6 -1
- package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +78 -5
- 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/complete-milestone.ts +15 -9
- 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/memory-tools.ts +31 -1
- package/src/resources/extensions/gsd/tools/resume-tool.ts +40 -0
- package/src/resources/extensions/gsd/uok/plan-v2.ts +26 -3
- package/src/resources/extensions/gsd/workflow-logger.ts +4 -1
- package/src/resources/extensions/gsd/workflow-mcp.ts +3 -0
- package/src/resources/extensions/gsd/workflow-templates/spike.md +6 -0
- package/src/resources/extensions/gsd/worktree-resolver.ts +54 -9
- package/src/resources/extensions/search-the-web/command-search-provider.ts +5 -4
- package/src/resources/extensions/search-the-web/native-search.ts +48 -12
- package/src/resources/skills/api-design/SKILL.md +190 -0
- package/src/resources/skills/create-mcp-server/SKILL.md +121 -0
- package/src/resources/skills/decompose-into-slices/SKILL.md +139 -0
- package/src/resources/skills/dependency-upgrade/SKILL.md +158 -0
- package/src/resources/skills/design-an-interface/SKILL.md +102 -0
- package/src/resources/skills/forensics/SKILL.md +153 -0
- package/src/resources/skills/grill-me/SKILL.md +93 -0
- package/src/resources/skills/handoff/SKILL.md +121 -0
- package/src/resources/skills/observability/SKILL.md +174 -0
- package/src/resources/skills/security-review/SKILL.md +181 -0
- package/src/resources/skills/spike-wrap-up/SKILL.md +138 -0
- package/src/resources/skills/tdd/SKILL.md +112 -0
- package/src/resources/skills/verify-before-complete/SKILL.md +98 -0
- package/src/resources/skills/write-docs/SKILL.md +82 -0
- package/src/resources/skills/write-milestone-brief/SKILL.md +135 -0
- /package/dist/web/standalone/.next/static/{ssX7BLv3Dw9Fb4CtrCGeR → pV-mPo7rYGb5JBC09C8GG}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{ssX7BLv3Dw9Fb4CtrCGeR → pV-mPo7rYGb5JBC09C8GG}/_ssgManifest.js +0 -0
|
@@ -442,3 +442,33 @@ test('writeBlockerPlaceholder: updates DB slice status for complete-slice (#2653
|
|
|
442
442
|
cleanup(base);
|
|
443
443
|
}
|
|
444
444
|
});
|
|
445
|
+
|
|
446
|
+
test('writeBlockerPlaceholder: inserts placeholder slice for plan-milestone so deriveState exits pre-planning (#4378)', async () => {
|
|
447
|
+
const base = createFixtureBase();
|
|
448
|
+
try {
|
|
449
|
+
const { openDatabase, closeDatabase, insertMilestone, getMilestoneSlices, isDbAvailable } =
|
|
450
|
+
await import("../../gsd-db.ts");
|
|
451
|
+
|
|
452
|
+
const dbPath = join(base, ".gsd", "gsd.db");
|
|
453
|
+
mkdirSync(join(base, ".gsd", "milestones", "M001"), { recursive: true });
|
|
454
|
+
|
|
455
|
+
openDatabase(dbPath);
|
|
456
|
+
try {
|
|
457
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
458
|
+
|
|
459
|
+
// Before fix: writeBlockerPlaceholder wrote the placeholder ROADMAP.md but
|
|
460
|
+
// never updated the DB, so activeMilestoneSlices.length === 0 on next deriveState
|
|
461
|
+
// call → state.phase stays 'pre-planning' → plan-milestone dispatches again → infinite loop
|
|
462
|
+
writeBlockerPlaceholder("plan-milestone", "M001", base, "idle recovery exhausted");
|
|
463
|
+
|
|
464
|
+
const slices = getMilestoneSlices("M001");
|
|
465
|
+
assert.ok(slices.length > 0,
|
|
466
|
+
"writeBlockerPlaceholder must insert a placeholder slice for plan-milestone so " +
|
|
467
|
+
"deriveState sees activeMilestoneSlices.length > 0 and exits pre-planning phase (#4378)");
|
|
468
|
+
} finally {
|
|
469
|
+
if (isDbAvailable()) closeDatabase();
|
|
470
|
+
}
|
|
471
|
+
} finally {
|
|
472
|
+
cleanup(base);
|
|
473
|
+
}
|
|
474
|
+
});
|
|
@@ -62,7 +62,7 @@ describe("interactive routing bypass (#3962)", () => {
|
|
|
62
62
|
);
|
|
63
63
|
// The function should check isAutoMode before routing synthesis
|
|
64
64
|
const fnIdx = modelSelectionSrc.indexOf("function resolvePreferredModelConfig");
|
|
65
|
-
const fnBody = modelSelectionSrc.slice(fnIdx, fnIdx +
|
|
65
|
+
const fnBody = modelSelectionSrc.slice(fnIdx, fnIdx + 900);
|
|
66
66
|
assert.ok(
|
|
67
67
|
fnBody.includes("isAutoMode"),
|
|
68
68
|
"resolvePreferredModelConfig should accept isAutoMode parameter",
|
|
@@ -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
|
+
});
|
|
@@ -77,6 +77,8 @@ function makeMockDeps(
|
|
|
77
77
|
autoWorktreeBranch: () => "auto/M001",
|
|
78
78
|
resolveMilestoneFile: () => null,
|
|
79
79
|
reconcileMergeState: () => "clean",
|
|
80
|
+
preflightCleanRoot: () => ({ stashPushed: false, summary: "" }),
|
|
81
|
+
postflightPopStash: () => {},
|
|
80
82
|
getLedger: () => ({ units: [] }),
|
|
81
83
|
getProjectTotals: () => ({ cost: 0 }),
|
|
82
84
|
formatCost: (c: number) => `$${c.toFixed(2)}`,
|
|
@@ -837,3 +839,38 @@ test("runFinalize pauses and emits unit-end when pre-verification times out", as
|
|
|
837
839
|
assert.equal((endEvents[0].data as any).artifactVerified, false);
|
|
838
840
|
assert.equal((endEvents[0].data as any).finalizeStage, "pre");
|
|
839
841
|
});
|
|
842
|
+
|
|
843
|
+
test("transient session-failed cancellations pause instead of hard-stopping", async () => {
|
|
844
|
+
const capture = createEventCapture();
|
|
845
|
+
const { resolveAgentEndCancelled, _resetPendingResolve } = await import("../auto-loop.js");
|
|
846
|
+
_resetPendingResolve();
|
|
847
|
+
|
|
848
|
+
const deps = makeMockDeps(capture);
|
|
849
|
+
const ic = makeIC(deps);
|
|
850
|
+
const iterData: IterationData = {
|
|
851
|
+
unitType: "execute-task",
|
|
852
|
+
unitId: "M001/S01/T02",
|
|
853
|
+
prompt: "do more stuff",
|
|
854
|
+
finalPrompt: "do more stuff",
|
|
855
|
+
pauseAfterUatDispatch: false,
|
|
856
|
+
state: { phase: "executing", activeMilestone: { id: "M001" }, activeSlice: { id: "S01" }, registry: [], blockers: [] } as any,
|
|
857
|
+
mid: "M001",
|
|
858
|
+
midTitle: "Test",
|
|
859
|
+
isRetry: false,
|
|
860
|
+
previousTier: undefined,
|
|
861
|
+
};
|
|
862
|
+
const loopState: LoopState = { recentUnits: [{ key: "execute-task/M001/S01/T02" }], stuckRecoveryAttempts: 0, consecutiveFinalizeTimeouts: 0 };
|
|
863
|
+
|
|
864
|
+
const unitPromise = runUnitPhase(ic, iterData, loopState);
|
|
865
|
+
await new Promise(r => setTimeout(r, 50));
|
|
866
|
+
|
|
867
|
+
resolveAgentEndCancelled({ message: "Session creation failed: temporary bootstrap overload", category: "session-failed", isTransient: true });
|
|
868
|
+
|
|
869
|
+
const result = await unitPromise;
|
|
870
|
+
assert.equal(result.action, "break");
|
|
871
|
+
assert.equal((result as any).reason, "session-timeout");
|
|
872
|
+
|
|
873
|
+
const entry = loopState.recentUnits[loopState.recentUnits.length - 1];
|
|
874
|
+
assert.ok(entry.error, "window entry must have error set");
|
|
875
|
+
assert.ok(entry.error!.startsWith("session-failed:"), "error must preserve the session-failed category");
|
|
876
|
+
});
|
|
@@ -141,6 +141,15 @@ test("PROVIDER_REGISTRY includes all major LLM providers", () => {
|
|
|
141
141
|
assert.ok(ids.includes("openai"));
|
|
142
142
|
assert.ok(ids.includes("google"));
|
|
143
143
|
assert.ok(ids.includes("groq"));
|
|
144
|
+
assert.ok(ids.includes("minimax"));
|
|
145
|
+
assert.ok(ids.includes("minimax-cn"));
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
test("PROVIDER_REGISTRY includes claude-code as a first-class LLM provider (#4541)", () => {
|
|
149
|
+
const entry = PROVIDER_REGISTRY.find((p) => p.id === "claude-code");
|
|
150
|
+
assert.ok(entry, "claude-code must be in PROVIDER_REGISTRY");
|
|
151
|
+
assert.equal(entry!.category, "llm");
|
|
152
|
+
assert.ok(entry!.hasOAuth, "claude-code uses OAuth (CLI auth)");
|
|
144
153
|
});
|
|
145
154
|
|
|
146
155
|
test("PROVIDER_REGISTRY includes all tool/search providers", () => {
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
// GSD-2 — loadMemoryBlock tests (ADR-013 step 4 auto-injection parity)
|
|
2
|
+
import { test } from 'node:test';
|
|
3
|
+
import assert from 'node:assert/strict';
|
|
4
|
+
|
|
5
|
+
import { closeDatabase, openDatabase } from '../gsd-db.ts';
|
|
6
|
+
import { createMemory } from '../memory-store.ts';
|
|
7
|
+
import { loadMemoryBlock } from '../bootstrap/system-context.ts';
|
|
8
|
+
|
|
9
|
+
// ─── Success path: critical memories surface in the labeled block ──────────
|
|
10
|
+
|
|
11
|
+
test('loadMemoryBlock: renders MEMORY block when critical memories exist', async () => {
|
|
12
|
+
openDatabase(':memory:');
|
|
13
|
+
try {
|
|
14
|
+
const id = createMemory({
|
|
15
|
+
category: 'architecture',
|
|
16
|
+
content: 'Use the memories table as the single source of truth for decisions.',
|
|
17
|
+
confidence: 0.95,
|
|
18
|
+
});
|
|
19
|
+
assert.ok(id, 'createMemory should seed a memory');
|
|
20
|
+
|
|
21
|
+
const block = await loadMemoryBlock('');
|
|
22
|
+
assert.ok(block.length > 0, 'block should be non-empty when critical memories exist');
|
|
23
|
+
assert.match(block, /\[MEMORY — Critical and prompt-relevant memories/);
|
|
24
|
+
assert.match(block, /memories table as the single source of truth/);
|
|
25
|
+
} finally {
|
|
26
|
+
closeDatabase();
|
|
27
|
+
}
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
// ─── Failure / degraded path: no DB → returns "" without throwing ───────────
|
|
31
|
+
|
|
32
|
+
test('loadMemoryBlock: returns empty string when no DB is available', async () => {
|
|
33
|
+
closeDatabase();
|
|
34
|
+
const block = await loadMemoryBlock('anything');
|
|
35
|
+
assert.equal(block, '', 'no DB → empty block (graceful degradation)');
|
|
36
|
+
});
|
|
@@ -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();
|
|
@@ -51,4 +51,16 @@ describe("stuck detection persistence (#3704)", () => {
|
|
|
51
51
|
test("stuck state file path uses runtime directory", () => {
|
|
52
52
|
assert.match(loopSource, /stuck-state\.json/);
|
|
53
53
|
});
|
|
54
|
+
|
|
55
|
+
test("saveStuckState called in standard dev path as well as custom engine path (#4382)", () => {
|
|
56
|
+
// Count all call-sites of saveStuckState (excluding the function definition itself).
|
|
57
|
+
// After the fix, both the custom-engine path and the standard dev path must each
|
|
58
|
+
// call saveStuckState so stuckRecoveryAttempts survives session restarts.
|
|
59
|
+
const callMatches = loopSource.match(/saveStuckState\(s\.basePath,\s*loopState\)/g) ?? [];
|
|
60
|
+
assert.ok(
|
|
61
|
+
callMatches.length >= 2,
|
|
62
|
+
`saveStuckState must be called in both the custom-engine path and the standard dev path ` +
|
|
63
|
+
`(found ${callMatches.length} call(s) — standard path is missing its call, #4382)`,
|
|
64
|
+
);
|
|
65
|
+
});
|
|
54
66
|
});
|
|
@@ -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?.[
|
|
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", () => {
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Regression test for #4620.
|
|
3
|
+
*
|
|
4
|
+
* Ensures plan gate failed-closed errors include a self-heal hint
|
|
5
|
+
* directing users to /gsd doctor heal.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import test from "node:test";
|
|
9
|
+
import assert from "node:assert/strict";
|
|
10
|
+
import { readFileSync } from "node:fs";
|
|
11
|
+
import { dirname, join } from "node:path";
|
|
12
|
+
import { fileURLToPath } from "node:url";
|
|
13
|
+
|
|
14
|
+
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
15
|
+
const gsdDir = join(__dirname, "..");
|
|
16
|
+
|
|
17
|
+
function readSrc(file: string): string {
|
|
18
|
+
return readFileSync(join(gsdDir, file), "utf-8");
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
test("#4620: auto/phases plan gate failed message includes doctor heal hint", () => {
|
|
22
|
+
const src = readSrc("auto/phases.ts");
|
|
23
|
+
assert.match(
|
|
24
|
+
src,
|
|
25
|
+
/Plan gate failed-closed:[\s\S]*\/gsd doctor heal/,
|
|
26
|
+
"auto/phases.ts should include /gsd doctor heal in plan gate failed notification",
|
|
27
|
+
);
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
test("#4620: guided-flow plan gate failed message includes doctor heal hint", () => {
|
|
31
|
+
const src = readSrc("guided-flow.ts");
|
|
32
|
+
assert.match(
|
|
33
|
+
src,
|
|
34
|
+
/Plan gate failed-closed:[\s\S]*\/gsd doctor heal/,
|
|
35
|
+
"guided-flow.ts should include /gsd doctor heal in plan gate failed notification",
|
|
36
|
+
);
|
|
37
|
+
});
|
|
@@ -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
|
})
|