gsd-pi 2.74.0-dev.b741afb → 2.74.0-dev.b7c5f96
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/resources/extensions/gsd/activity-log.js +16 -0
- package/dist/resources/extensions/gsd/auto/loop.js +147 -10
- package/dist/resources/extensions/gsd/auto/phases.js +164 -9
- package/dist/resources/extensions/gsd/auto/session.js +10 -0
- package/dist/resources/extensions/gsd/auto-dispatch.js +11 -1
- package/dist/resources/extensions/gsd/auto-model-selection.js +54 -8
- package/dist/resources/extensions/gsd/auto-post-unit.js +215 -8
- package/dist/resources/extensions/gsd/auto-recovery.js +24 -10
- package/dist/resources/extensions/gsd/auto-unit-closeout.js +18 -0
- package/dist/resources/extensions/gsd/auto-verification.js +100 -2
- package/dist/resources/extensions/gsd/auto-worktree.js +2 -0
- package/dist/resources/extensions/gsd/auto.js +28 -2
- package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +5 -3
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +10 -1
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +61 -9
- package/dist/resources/extensions/gsd/cache.js +16 -5
- package/dist/resources/extensions/gsd/commands/catalog.js +6 -1
- package/dist/resources/extensions/gsd/commands/handlers/core.js +5 -1
- package/dist/resources/extensions/gsd/commands/handlers/ops.js +5 -0
- package/dist/resources/extensions/gsd/commands-extract-learnings.js +225 -0
- package/dist/resources/extensions/gsd/commands-prefs-wizard.js +51 -4
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +16 -1
- package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +144 -0
- package/dist/resources/extensions/gsd/ecosystem/loader.js +145 -0
- package/dist/resources/extensions/gsd/git-service.js +49 -1
- package/dist/resources/extensions/gsd/graph-context.js +98 -7
- package/dist/resources/extensions/gsd/gsd-db.js +260 -2
- package/dist/resources/extensions/gsd/guided-flow.js +31 -6
- package/dist/resources/extensions/gsd/init-wizard.js +1 -0
- package/dist/resources/extensions/gsd/journal.js +27 -0
- package/dist/resources/extensions/gsd/metrics.js +19 -0
- package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
- package/dist/resources/extensions/gsd/preferences-models.js +20 -3
- package/dist/resources/extensions/gsd/preferences-types.js +2 -0
- package/dist/resources/extensions/gsd/preferences-validation.js +118 -2
- package/dist/resources/extensions/gsd/preferences.js +31 -0
- package/dist/resources/extensions/gsd/safety/evidence-collector.js +15 -30
- package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
- package/dist/resources/extensions/gsd/templates/PREFERENCES.md +19 -0
- package/dist/resources/extensions/gsd/tools/complete-slice.js +5 -0
- package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
- package/dist/resources/extensions/gsd/unit-ownership.js +1 -1
- package/dist/resources/extensions/gsd/uok/audit-toggle.js +7 -0
- package/dist/resources/extensions/gsd/uok/audit.js +40 -0
- package/dist/resources/extensions/gsd/uok/contracts.js +1 -0
- package/dist/resources/extensions/gsd/uok/execution-graph.js +179 -0
- package/dist/resources/extensions/gsd/uok/flags.js +29 -0
- package/dist/resources/extensions/gsd/uok/gate-runner.js +109 -0
- package/dist/resources/extensions/gsd/uok/gitops.js +53 -0
- package/dist/resources/extensions/gsd/uok/kernel.js +80 -0
- package/dist/resources/extensions/gsd/uok/loop-adapter.js +133 -0
- package/dist/resources/extensions/gsd/uok/model-policy.js +66 -0
- package/dist/resources/extensions/gsd/uok/plan-v2.js +132 -0
- package/dist/resources/extensions/gsd/workflow-logger.js +22 -0
- package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -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 +8 -8
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +8 -8
- package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
- 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/package.json +1 -1
- package/packages/mcp-server/dist/readers/graph.d.ts +1 -1
- package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -1
- package/packages/mcp-server/dist/readers/graph.js +107 -0
- package/packages/mcp-server/dist/readers/graph.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +88 -6
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/src/readers/graph.test.ts +178 -0
- package/packages/mcp-server/src/readers/graph.ts +148 -1
- package/packages/mcp-server/src/workflow-tools.ts +95 -10
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-ai/dist/index.d.ts +1 -9
- package/packages/pi-ai/dist/index.d.ts.map +1 -1
- package/packages/pi-ai/dist/index.js +1 -9
- package/packages/pi-ai/dist/index.js.map +1 -1
- package/packages/pi-ai/dist/models/capability-patches.d.ts +19 -0
- package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/capability-patches.js +36 -0
- package/packages/pi-ai/dist/models/capability-patches.js.map +1 -0
- package/packages/pi-ai/dist/{models.custom.d.ts → models/custom.d.ts} +1 -1
- package/packages/pi-ai/dist/models/custom.d.ts.map +1 -0
- package/packages/pi-ai/dist/{models.custom.js → models/custom.js} +4 -4
- package/packages/pi-ai/dist/models/custom.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts +1482 -0
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.js +1484 -0
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/anthropic.d.ts +377 -0
- package/packages/pi-ai/dist/models/generated/anthropic.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/anthropic.js +379 -0
- package/packages/pi-ai/dist/models/generated/anthropic.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts +700 -0
- package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/azure-openai-responses.js +702 -0
- package/packages/pi-ai/dist/models/generated/azure-openai-responses.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/cerebras.d.ts +71 -0
- package/packages/pi-ai/dist/models/generated/cerebras.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/cerebras.js +73 -0
- package/packages/pi-ai/dist/models/generated/cerebras.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/github-copilot.d.ts +590 -0
- package/packages/pi-ai/dist/models/generated/github-copilot.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/github-copilot.js +444 -0
- package/packages/pi-ai/dist/models/generated/github-copilot.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts +156 -0
- package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-antigravity.js +158 -0
- package/packages/pi-ai/dist/models/generated/google-antigravity.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts +105 -0
- package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-gemini-cli.js +107 -0
- package/packages/pi-ai/dist/models/generated/google-gemini-cli.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-vertex.d.ts +207 -0
- package/packages/pi-ai/dist/models/generated/google-vertex.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-vertex.js +209 -0
- package/packages/pi-ai/dist/models/generated/google-vertex.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/google.d.ts +462 -0
- package/packages/pi-ai/dist/models/generated/google.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/google.js +464 -0
- package/packages/pi-ai/dist/models/generated/google.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/groq.d.ts +309 -0
- package/packages/pi-ai/dist/models/generated/groq.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/groq.js +311 -0
- package/packages/pi-ai/dist/models/generated/groq.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/huggingface.d.ts +383 -0
- package/packages/pi-ai/dist/models/generated/huggingface.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/huggingface.js +347 -0
- package/packages/pi-ai/dist/models/generated/huggingface.js.map +1 -0
- package/packages/pi-ai/dist/{models.generated.d.ts → models/generated/index.d.ts} +1 -1
- package/packages/pi-ai/dist/{models.generated.d.ts.map → models/generated/index.d.ts.map} +1 -1
- package/packages/pi-ai/dist/models/generated/index.js +51 -0
- package/packages/pi-ai/dist/models/generated/index.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts +37 -0
- package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/kimi-coding.js +39 -0
- package/packages/pi-ai/dist/models/generated/kimi-coding.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts +105 -0
- package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/minimax-cn.js +107 -0
- package/packages/pi-ai/dist/models/generated/minimax-cn.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/minimax.d.ts +105 -0
- package/packages/pi-ai/dist/models/generated/minimax.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/minimax.js +107 -0
- package/packages/pi-ai/dist/models/generated/minimax.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/mistral.d.ts +445 -0
- package/packages/pi-ai/dist/models/generated/mistral.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/mistral.js +447 -0
- package/packages/pi-ai/dist/models/generated/mistral.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +139 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.js +141 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/openai.d.ts +700 -0
- package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/openai.js +702 -0
- package/packages/pi-ai/dist/models/generated/openai.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/opencode-go.d.ts +122 -0
- package/packages/pi-ai/dist/models/generated/opencode-go.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/opencode-go.js +124 -0
- package/packages/pi-ai/dist/models/generated/opencode-go.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/opencode.d.ts +530 -0
- package/packages/pi-ai/dist/models/generated/opencode.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/opencode.js +532 -0
- package/packages/pi-ai/dist/models/generated/opencode.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/openrouter.d.ts +4270 -0
- package/packages/pi-ai/dist/models/generated/openrouter.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/openrouter.js +4272 -0
- package/packages/pi-ai/dist/models/generated/openrouter.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts +2604 -0
- package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js +2606 -0
- package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/xai.d.ts +411 -0
- package/packages/pi-ai/dist/models/generated/xai.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/xai.js +413 -0
- package/packages/pi-ai/dist/models/generated/xai.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/zai.d.ts +276 -0
- package/packages/pi-ai/dist/models/generated/zai.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/zai.js +239 -0
- package/packages/pi-ai/dist/models/generated/zai.js.map +1 -0
- package/packages/pi-ai/dist/models/index.d.ts +27 -0
- package/packages/pi-ai/dist/models/index.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/index.js +80 -0
- package/packages/pi-ai/dist/models/index.js.map +1 -0
- package/packages/pi-ai/dist/models.d.ts +1 -36
- package/packages/pi-ai/dist/models.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.test.js +1 -2
- package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
- package/packages/pi-ai/dist/models.js +3 -112
- package/packages/pi-ai/dist/models.js.map +1 -1
- package/packages/pi-ai/dist/models.test.js +6 -5
- package/packages/pi-ai/dist/models.test.js.map +1 -1
- package/packages/pi-ai/scripts/generate-models.ts +74 -40
- package/packages/pi-ai/src/index.ts +1 -9
- package/packages/pi-ai/src/models/capability-patches.ts +40 -0
- package/packages/pi-ai/src/{models.custom.ts → models/custom.ts} +4 -4
- package/packages/pi-ai/src/models/generated/amazon-bedrock.ts +1486 -0
- package/packages/pi-ai/src/models/generated/anthropic.ts +381 -0
- package/packages/pi-ai/src/models/generated/azure-openai-responses.ts +704 -0
- package/packages/pi-ai/src/models/generated/cerebras.ts +75 -0
- package/packages/pi-ai/src/models/generated/github-copilot.ts +446 -0
- package/packages/pi-ai/src/models/generated/google-antigravity.ts +160 -0
- package/packages/pi-ai/src/models/generated/google-gemini-cli.ts +109 -0
- package/packages/pi-ai/src/models/generated/google-vertex.ts +211 -0
- package/packages/pi-ai/src/models/generated/google.ts +466 -0
- package/packages/pi-ai/src/models/generated/groq.ts +313 -0
- package/packages/pi-ai/src/models/generated/huggingface.ts +349 -0
- package/packages/pi-ai/src/models/generated/index.ts +52 -0
- package/packages/pi-ai/src/models/generated/kimi-coding.ts +41 -0
- package/packages/pi-ai/src/models/generated/minimax-cn.ts +109 -0
- package/packages/pi-ai/src/models/generated/minimax.ts +109 -0
- package/packages/pi-ai/src/models/generated/mistral.ts +449 -0
- package/packages/pi-ai/src/models/generated/openai-codex.ts +143 -0
- package/packages/pi-ai/src/models/generated/openai.ts +704 -0
- package/packages/pi-ai/src/models/generated/opencode-go.ts +126 -0
- package/packages/pi-ai/src/models/generated/opencode.ts +534 -0
- package/packages/pi-ai/src/models/generated/openrouter.ts +4274 -0
- package/packages/pi-ai/src/models/generated/vercel-ai-gateway.ts +2608 -0
- package/packages/pi-ai/src/models/generated/xai.ts +415 -0
- package/packages/pi-ai/src/models/generated/zai.ts +241 -0
- package/packages/pi-ai/src/models/index.ts +106 -0
- package/packages/pi-ai/src/models.generated.test.ts +1 -2
- package/packages/pi-ai/src/models.test.ts +6 -5
- package/packages/pi-ai/src/models.ts +3 -153
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.js +8 -2
- package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +472 -0
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js +52 -0
- package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.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 +2 -2
- package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.js +61 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +11 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +23 -9
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +11 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +53 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/compaction-summary-message.d.ts +8 -5
- package/packages/pi-coding-agent/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/compaction-summary-message.js +27 -13
- package/packages/pi-coding-agent/dist/modes/interactive/components/compaction-summary-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +8 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +68 -8
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js +22 -22
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +232 -18
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +38 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +14 -0
- 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 +70 -6
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/src/core/agent-session.ts +12 -6
- package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +612 -0
- package/packages/pi-coding-agent/src/core/model-registry-env-fallback.test.ts +59 -0
- package/packages/pi-coding-agent/src/core/model-registry.ts +2 -1
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.ts +92 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +19 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +25 -10
- package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +75 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/compaction-summary-message.ts +36 -15
- package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +83 -7
- package/packages/pi-coding-agent/src/modes/interactive/components/user-message.ts +23 -26
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +298 -41
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +44 -0
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +92 -6
- package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
- package/packages/pi-coding-agent/tsconfig.json +2 -2
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-tui/dist/tui.d.ts.map +1 -1
- package/packages/pi-tui/dist/tui.js +9 -2
- package/packages/pi-tui/dist/tui.js.map +1 -1
- package/packages/pi-tui/src/tui.ts +9 -1
- package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
- package/src/resources/extensions/gsd/activity-log.ts +21 -0
- package/src/resources/extensions/gsd/auto/loop-deps.ts +4 -0
- package/src/resources/extensions/gsd/auto/loop.ts +159 -10
- package/src/resources/extensions/gsd/auto/phases.ts +193 -9
- package/src/resources/extensions/gsd/auto/session.ts +10 -0
- package/src/resources/extensions/gsd/auto-dispatch.ts +16 -6
- package/src/resources/extensions/gsd/auto-model-selection.ts +69 -8
- package/src/resources/extensions/gsd/auto-post-unit.ts +226 -9
- package/src/resources/extensions/gsd/auto-recovery.ts +29 -9
- package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
- package/src/resources/extensions/gsd/auto-verification.ts +129 -2
- package/src/resources/extensions/gsd/auto-worktree.ts +1 -0
- package/src/resources/extensions/gsd/auto.ts +34 -2
- package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +5 -3
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +15 -1
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +72 -8
- package/src/resources/extensions/gsd/cache.ts +16 -5
- package/src/resources/extensions/gsd/commands/catalog.ts +6 -1
- package/src/resources/extensions/gsd/commands/handlers/core.ts +5 -1
- package/src/resources/extensions/gsd/commands/handlers/ops.ts +5 -0
- package/src/resources/extensions/gsd/commands-extract-learnings.ts +304 -0
- package/src/resources/extensions/gsd/commands-prefs-wizard.ts +58 -4
- package/src/resources/extensions/gsd/docs/preferences-reference.md +16 -1
- package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +228 -0
- package/src/resources/extensions/gsd/ecosystem/loader.ts +201 -0
- package/src/resources/extensions/gsd/git-service.ts +68 -0
- package/src/resources/extensions/gsd/graph-context.ts +139 -12
- package/src/resources/extensions/gsd/gsd-db.ts +321 -3
- package/src/resources/extensions/gsd/guided-flow.ts +36 -2
- package/src/resources/extensions/gsd/init-wizard.ts +3 -2
- package/src/resources/extensions/gsd/journal.ts +30 -0
- package/src/resources/extensions/gsd/metrics.ts +26 -0
- package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
- package/src/resources/extensions/gsd/preferences-models.ts +20 -3
- package/src/resources/extensions/gsd/preferences-types.ts +38 -0
- package/src/resources/extensions/gsd/preferences-validation.ts +117 -2
- package/src/resources/extensions/gsd/preferences.ts +34 -0
- package/src/resources/extensions/gsd/safety/evidence-collector.ts +15 -31
- package/src/resources/extensions/gsd/session-lock.ts +14 -2
- package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +20 -1
- package/src/resources/extensions/gsd/templates/PREFERENCES.md +19 -0
- package/src/resources/extensions/gsd/tests/artifacts-table-preserved-on-cache-invalidate.test.ts +177 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +7 -3
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
- package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
- package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +272 -0
- package/src/resources/extensions/gsd/tests/auto-warning-noise-regression.test.ts +117 -0
- package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
- package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +340 -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/finalize-timeout-guard.test.ts +10 -7
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/health-widget.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/journal-integration.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -2
- package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -3
- package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
- package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +40 -1
- package/src/resources/extensions/gsd/tests/preferences.test.ts +145 -0
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +57 -2
- package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -5
- package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +101 -0
- package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +85 -0
- package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +69 -0
- package/src/resources/extensions/gsd/tests/uok-flags.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +70 -0
- package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +85 -0
- package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +35 -0
- package/src/resources/extensions/gsd/tests/uok-model-policy.test.ts +89 -0
- package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +167 -0
- package/src/resources/extensions/gsd/tests/uok-preferences.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
- package/src/resources/extensions/gsd/tools/complete-slice.ts +9 -2
- package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
- package/src/resources/extensions/gsd/types.ts +14 -1
- package/src/resources/extensions/gsd/unit-ownership.ts +2 -2
- package/src/resources/extensions/gsd/uok/audit-toggle.ts +9 -0
- package/src/resources/extensions/gsd/uok/audit.ts +51 -0
- package/src/resources/extensions/gsd/uok/contracts.ts +135 -0
- package/src/resources/extensions/gsd/uok/execution-graph.ts +241 -0
- package/src/resources/extensions/gsd/uok/flags.ts +45 -0
- package/src/resources/extensions/gsd/uok/gate-runner.ts +146 -0
- package/src/resources/extensions/gsd/uok/gitops.ts +75 -0
- package/src/resources/extensions/gsd/uok/kernel.ts +105 -0
- package/src/resources/extensions/gsd/uok/loop-adapter.ts +162 -0
- package/src/resources/extensions/gsd/uok/model-policy.ts +112 -0
- package/src/resources/extensions/gsd/uok/plan-v2.ts +156 -0
- package/src/resources/extensions/gsd/workflow-logger.ts +27 -1
- package/src/resources/extensions/ttsr/ttsr-manager.ts +10 -5
- package/packages/pi-ai/dist/models.custom.d.ts.map +0 -1
- package/packages/pi-ai/dist/models.custom.js.map +0 -1
- package/packages/pi-ai/dist/models.generated.js +0 -14343
- package/packages/pi-ai/dist/models.generated.js.map +0 -1
- package/packages/pi-ai/src/models.generated.ts +0 -14345
- /package/dist/web/standalone/.next/static/{XnHY5eXUsTCFmNodWHetD → k1VZ0h-SB6uWSZgPhgNwi}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{XnHY5eXUsTCFmNodWHetD → k1VZ0h-SB6uWSZgPhgNwi}/_ssgManifest.js +0 -0
|
@@ -6,6 +6,7 @@ import type {
|
|
|
6
6
|
|
|
7
7
|
import { getAutoDashboardData, startAuto, type AutoDashboardData } from "../auto.js";
|
|
8
8
|
import { resetTransientRetryState } from "./agent-end-recovery.js";
|
|
9
|
+
import { resetSessionTimeoutState } from "../auto/phases.js";
|
|
9
10
|
|
|
10
11
|
type AutoResumeSnapshot = Pick<AutoDashboardData, "active" | "paused" | "stepMode" | "basePath">;
|
|
11
12
|
|
|
@@ -43,10 +44,11 @@ export async function resumeAutoAfterProviderDelay(
|
|
|
43
44
|
return "missing-base";
|
|
44
45
|
}
|
|
45
46
|
|
|
46
|
-
// Reset
|
|
47
|
-
//
|
|
48
|
-
//
|
|
47
|
+
// Reset retry counters before restarting — without this, counters
|
|
48
|
+
// accumulate across pause/resume cycles and permanently lock out
|
|
49
|
+
// auto-resume after their respective MAX thresholds.
|
|
49
50
|
resetTransientRetryState();
|
|
51
|
+
resetSessionTimeoutState();
|
|
50
52
|
|
|
51
53
|
await deps.startAuto(
|
|
52
54
|
ctx as ExtensionCommandContext,
|
|
@@ -4,6 +4,8 @@ import type { ExtensionAPI, ExtensionCommandContext } from "@gsd/pi-coding-agent
|
|
|
4
4
|
|
|
5
5
|
import { registerExitCommand } from "../exit-command.js";
|
|
6
6
|
import { registerWorktreeCommand } from "../worktree-command.js";
|
|
7
|
+
import type { GSDEcosystemBeforeAgentStartHandler } from "../ecosystem/gsd-extension-api.js";
|
|
8
|
+
import { loadEcosystemExtensions } from "../ecosystem/loader.js";
|
|
7
9
|
import { registerDbTools } from "./db-tools.js";
|
|
8
10
|
import { registerDynamicTools } from "./dynamic-tools.js";
|
|
9
11
|
import { registerJournalTools } from "./journal-tools.js";
|
|
@@ -65,6 +67,10 @@ export function registerGsdExtension(pi: ExtensionAPI): void {
|
|
|
65
67
|
|
|
66
68
|
installEpipeGuard();
|
|
67
69
|
|
|
70
|
+
// Ecosystem handlers captured by the GSDExtensionAPI wrapper for the
|
|
71
|
+
// GSD-owned `before_agent_start` dispatch step (#3338).
|
|
72
|
+
const ecosystemHandlers: GSDEcosystemBeforeAgentStartHandler[] = [];
|
|
73
|
+
|
|
68
74
|
pi.registerCommand("kill", {
|
|
69
75
|
description: "Exit GSD immediately (no cleanup)",
|
|
70
76
|
handler: async (_args: string, _ctx: ExtensionCommandContext) => {
|
|
@@ -80,7 +86,15 @@ export function registerGsdExtension(pi: ExtensionAPI): void {
|
|
|
80
86
|
["journal-tools", () => registerJournalTools(pi)],
|
|
81
87
|
["query-tools", () => registerQueryTools(pi)],
|
|
82
88
|
["shortcuts", () => registerShortcuts(pi)],
|
|
83
|
-
["hooks", () => registerHooks(pi)],
|
|
89
|
+
["hooks", () => registerHooks(pi, ecosystemHandlers)],
|
|
90
|
+
["ecosystem", () => {
|
|
91
|
+
void loadEcosystemExtensions(pi, ecosystemHandlers).catch((err) => {
|
|
92
|
+
logWarning(
|
|
93
|
+
"ecosystem",
|
|
94
|
+
`loader failed: ${err instanceof Error ? err.message : String(err)}`,
|
|
95
|
+
);
|
|
96
|
+
});
|
|
97
|
+
}],
|
|
84
98
|
];
|
|
85
99
|
|
|
86
100
|
for (const [name, register] of nonCriticalRegistrations) {
|
|
@@ -3,6 +3,10 @@ import { join } from "node:path";
|
|
|
3
3
|
import type { ExtensionAPI, ExtensionContext } from "@gsd/pi-coding-agent";
|
|
4
4
|
import { isToolCallEventType } from "@gsd/pi-coding-agent";
|
|
5
5
|
|
|
6
|
+
import type { GSDEcosystemBeforeAgentStartHandler } from "../ecosystem/gsd-extension-api.js";
|
|
7
|
+
import { updateSnapshot } from "../ecosystem/gsd-extension-api.js";
|
|
8
|
+
import { getEcosystemReadyPromise } from "../ecosystem/loader.js";
|
|
9
|
+
|
|
6
10
|
import { buildMilestoneFileName, resolveMilestonePath, resolveSliceFile, resolveSlicePath } from "../paths.js";
|
|
7
11
|
import { buildBeforeAgentStartResult } from "./system-context.js";
|
|
8
12
|
import { handleAgentEnd } from "./agent-end-recovery.js";
|
|
@@ -35,7 +39,10 @@ async function syncServiceTierStatus(ctx: ExtensionContext): Promise<void> {
|
|
|
35
39
|
ctx.ui.setStatus("gsd-fast", formatServiceTierFooterStatus(getEffectiveServiceTier(), ctx.model?.id));
|
|
36
40
|
}
|
|
37
41
|
|
|
38
|
-
export function registerHooks(
|
|
42
|
+
export function registerHooks(
|
|
43
|
+
pi: ExtensionAPI,
|
|
44
|
+
ecosystemHandlers: GSDEcosystemBeforeAgentStartHandler[],
|
|
45
|
+
): void {
|
|
39
46
|
pi.on("session_start", async (_event, ctx) => {
|
|
40
47
|
initNotificationStore(process.cwd());
|
|
41
48
|
installNotifyInterceptor(ctx);
|
|
@@ -45,8 +52,12 @@ export function registerHooks(pi: ExtensionAPI): void {
|
|
|
45
52
|
resetToolCallLoopGuard();
|
|
46
53
|
resetAskUserQuestionsCache();
|
|
47
54
|
await syncServiceTierStatus(ctx);
|
|
48
|
-
|
|
49
|
-
|
|
55
|
+
// Skip MCP auto-prep when running inside an auto-worktree (see session_switch below).
|
|
56
|
+
const { isInAutoWorktree } = await import("../auto-worktree.js");
|
|
57
|
+
if (!isInAutoWorktree(process.cwd())) {
|
|
58
|
+
const { prepareWorkflowMcpForProject } = await import("../workflow-mcp-auto-prep.js");
|
|
59
|
+
prepareWorkflowMcpForProject(ctx, process.cwd());
|
|
60
|
+
}
|
|
50
61
|
|
|
51
62
|
// Apply show_token_cost preference (#1515)
|
|
52
63
|
try {
|
|
@@ -87,13 +98,63 @@ export function registerHooks(pi: ExtensionAPI): void {
|
|
|
87
98
|
resetAskUserQuestionsCache();
|
|
88
99
|
clearDiscussionFlowState();
|
|
89
100
|
await syncServiceTierStatus(ctx);
|
|
90
|
-
|
|
91
|
-
|
|
101
|
+
// Skip MCP auto-prep when running inside an auto-worktree. The worktree
|
|
102
|
+
// already has .mcp.json from createAutoWorktree, and re-running the writer
|
|
103
|
+
// post-chdir rewrites the file mid-run (non-idempotent due to cwd-relative
|
|
104
|
+
// CLI path resolution), dirtying the tree and breaking the milestone merge.
|
|
105
|
+
const { isInAutoWorktree } = await import("../auto-worktree.js");
|
|
106
|
+
if (!isInAutoWorktree(process.cwd())) {
|
|
107
|
+
const { prepareWorkflowMcpForProject } = await import("../workflow-mcp-auto-prep.js");
|
|
108
|
+
prepareWorkflowMcpForProject(ctx, process.cwd());
|
|
109
|
+
}
|
|
92
110
|
loadToolApiKeys();
|
|
93
111
|
});
|
|
94
112
|
|
|
95
113
|
pi.on("before_agent_start", async (event, ctx: ExtensionContext) => {
|
|
96
|
-
|
|
114
|
+
// Wait for ecosystem loader to finish (no-op after first turn).
|
|
115
|
+
await getEcosystemReadyPromise();
|
|
116
|
+
|
|
117
|
+
// GSD's own context injection (existing behavior — unchanged).
|
|
118
|
+
const gsdResult = await buildBeforeAgentStartResult(event, ctx);
|
|
119
|
+
|
|
120
|
+
// Refresh the snapshot used by ecosystem getPhase()/getActiveUnit().
|
|
121
|
+
// deriveState has its own ~100ms cache so this is cheap on repeat calls.
|
|
122
|
+
try {
|
|
123
|
+
const state = await deriveState(process.cwd());
|
|
124
|
+
updateSnapshot(state);
|
|
125
|
+
} catch {
|
|
126
|
+
updateSnapshot(null);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
// Chain ecosystem handlers using pi's runner.ts chaining protocol:
|
|
130
|
+
// each handler sees the systemPrompt mutated by prior handlers.
|
|
131
|
+
let currentSystemPrompt = gsdResult?.systemPrompt ?? event.systemPrompt;
|
|
132
|
+
// `any` because pi's BeforeAgentStartEventResult.message uses an internal
|
|
133
|
+
// CustomMessage type that's not re-exported (see ecosystem/gsd-extension-api.ts).
|
|
134
|
+
let lastMessage: any = gsdResult?.message;
|
|
135
|
+
|
|
136
|
+
for (const handler of ecosystemHandlers) {
|
|
137
|
+
try {
|
|
138
|
+
const r = await handler(
|
|
139
|
+
{ ...event, systemPrompt: currentSystemPrompt },
|
|
140
|
+
ctx,
|
|
141
|
+
);
|
|
142
|
+
if (r?.systemPrompt !== undefined) currentSystemPrompt = r.systemPrompt;
|
|
143
|
+
if (r?.message) lastMessage = r.message;
|
|
144
|
+
} catch (err) {
|
|
145
|
+
safetyLogWarning(
|
|
146
|
+
"ecosystem",
|
|
147
|
+
`before_agent_start handler failed: ${err instanceof Error ? err.message : String(err)}`,
|
|
148
|
+
);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
// Compose result. Return undefined if nothing changed (preserves runner contract).
|
|
153
|
+
if (currentSystemPrompt === event.systemPrompt && !lastMessage) return undefined;
|
|
154
|
+
return {
|
|
155
|
+
systemPrompt: currentSystemPrompt !== event.systemPrompt ? currentSystemPrompt : undefined,
|
|
156
|
+
message: lastMessage,
|
|
157
|
+
};
|
|
97
158
|
});
|
|
98
159
|
|
|
99
160
|
pi.on("agent_end", async (event, ctx: ExtensionContext) => {
|
|
@@ -125,7 +186,10 @@ export function registerHooks(pi: ExtensionAPI): void {
|
|
|
125
186
|
await ensureDbOpen();
|
|
126
187
|
const state = await deriveState(basePath);
|
|
127
188
|
if (!state.activeMilestone || !state.activeSlice || !state.activeTask) return;
|
|
128
|
-
|
|
189
|
+
// Write checkpoint for ALL phases, not just "executing" — discuss, research,
|
|
190
|
+
// and planning also carry in-memory state (user answers, gate verification)
|
|
191
|
+
// that would be lost on compaction (#4258).
|
|
192
|
+
// if (state.phase !== "executing") return;
|
|
129
193
|
|
|
130
194
|
const sliceDir = resolveSlicePath(basePath, state.activeMilestone.id, state.activeSlice.id);
|
|
131
195
|
if (!sliceDir) return;
|
|
@@ -261,7 +325,7 @@ export function registerHooks(pi: ExtensionAPI): void {
|
|
|
261
325
|
// ── Safety harness: evidence collection + destructive command warnings ──
|
|
262
326
|
pi.on("tool_call", async (event, ctx) => {
|
|
263
327
|
if (!isAutoActive()) return;
|
|
264
|
-
safetyRecordToolCall(event.toolName, event.input as Record<string, unknown>);
|
|
328
|
+
safetyRecordToolCall(event.toolCallId, event.toolName, event.input as Record<string, unknown>);
|
|
265
329
|
|
|
266
330
|
// Destructive command classification (warn only, never block)
|
|
267
331
|
if (isToolCallEventType("bash", event)) {
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
// GSD Extension — Unified Cache Invalidation
|
|
2
2
|
//
|
|
3
|
-
// Three module-scoped caches exist across the GSD extension:
|
|
3
|
+
// Three module-scoped read caches exist across the GSD extension:
|
|
4
4
|
// 1. State cache (state.ts) — memoized deriveState() result
|
|
5
5
|
// 2. Path cache (paths.ts) — directory listing results (readdirSync)
|
|
6
6
|
// 3. Parse cache (files.ts) — parsed markdown file results
|
|
@@ -8,22 +8,33 @@
|
|
|
8
8
|
// After any file write that changes .gsd/ contents, all three must be
|
|
9
9
|
// invalidated together to prevent stale reads. This module provides a
|
|
10
10
|
// single function that clears all three atomically.
|
|
11
|
+
//
|
|
12
|
+
// NOTE: The DB `artifacts` table is NOT included here. Earlier versions
|
|
13
|
+
// called clearArtifacts() as part of this bundle (#793), intending to
|
|
14
|
+
// force deriveState() to re-parse from disk when files were edited
|
|
15
|
+
// out-of-band. But invalidateAllCaches() fires on every post-unit pass,
|
|
16
|
+
// so bundling a DESTRUCTIVE `DELETE FROM artifacts` with routine cache
|
|
17
|
+
// invalidation meant every row written by saveArtifactToDb / writeAndStore
|
|
18
|
+
// was wiped within seconds — leaving the milestone completed on disk but
|
|
19
|
+
// the `artifacts` table empty and the agent looping on "file exists but
|
|
20
|
+
// DB record missing" recovery calls. If a call site genuinely needs the
|
|
21
|
+
// artifact table cleared after an out-of-band file mutation, it should
|
|
22
|
+
// invoke clearArtifacts() from gsd-db.js explicitly — do not add it back
|
|
23
|
+
// here.
|
|
11
24
|
|
|
12
25
|
import { invalidateStateCache } from './state.js';
|
|
13
26
|
import { clearPathCache } from './paths.js';
|
|
14
27
|
import { clearParseCache } from './files.js';
|
|
15
|
-
import { clearArtifacts } from './gsd-db.js';
|
|
16
28
|
|
|
17
29
|
/**
|
|
18
|
-
* Invalidate all GSD runtime caches in one call.
|
|
30
|
+
* Invalidate all GSD runtime read caches in one call.
|
|
19
31
|
*
|
|
20
32
|
* Call this after file writes, milestone transitions, merge reconciliation,
|
|
21
33
|
* or any operation that changes .gsd/ contents on disk. Forgetting to clear
|
|
22
|
-
* any single cache causes stale reads (see #431
|
|
34
|
+
* any single cache causes stale reads (see #431).
|
|
23
35
|
*/
|
|
24
36
|
export function invalidateAllCaches(): void {
|
|
25
37
|
invalidateStateCache();
|
|
26
38
|
clearPathCache();
|
|
27
39
|
clearParseCache();
|
|
28
|
-
clearArtifacts();
|
|
29
40
|
}
|
|
@@ -15,7 +15,7 @@ export interface GsdCommandDefinition {
|
|
|
15
15
|
type CompletionMap = Record<string, readonly GsdCommandDefinition[]>;
|
|
16
16
|
|
|
17
17
|
export const GSD_COMMAND_DESCRIPTION =
|
|
18
|
-
"GSD — Get Shit Done: /gsd help|start|templates|next|auto|stop|pause|status|widget|visualize|queue|quick|discuss|capture|triage|dispatch|history|undo|undo-task|reset-slice|rate|skip|export|cleanup|model|mode|prefs|config|keys|hooks|run-hook|skill-health|doctor|logs|forensics|changelog|migrate|remote|steer|knowledge|new-milestone|parallel|cmux|park|unpark|init|setup|inspect|extensions|update|fast|mcp|rethink|codebase|notifications|ship|do|session-report|backlog|pr-branch|add-tests";
|
|
18
|
+
"GSD — Get Shit Done: /gsd help|start|templates|next|auto|stop|pause|status|widget|visualize|queue|quick|discuss|capture|triage|dispatch|history|undo|undo-task|reset-slice|rate|skip|export|cleanup|model|mode|prefs|config|keys|hooks|run-hook|skill-health|doctor|logs|forensics|changelog|migrate|remote|steer|knowledge|new-milestone|parallel|cmux|park|unpark|init|setup|inspect|extensions|update|fast|mcp|rethink|codebase|notifications|ship|do|session-report|backlog|pr-branch|add-tests|language";
|
|
19
19
|
|
|
20
20
|
export const TOP_LEVEL_SUBCOMMANDS: readonly GsdCommandDefinition[] = [
|
|
21
21
|
{ cmd: "help", desc: "Categorized command reference with descriptions" },
|
|
@@ -80,6 +80,7 @@ export const TOP_LEVEL_SUBCOMMANDS: readonly GsdCommandDefinition[] = [
|
|
|
80
80
|
{ cmd: "backlog", desc: "Manage backlog items (add, promote, remove, list)" },
|
|
81
81
|
{ cmd: "pr-branch", desc: "Create clean PR branch filtering .gsd/ commits" },
|
|
82
82
|
{ cmd: "add-tests", desc: "Generate tests for completed slices" },
|
|
83
|
+
{ cmd: "language", desc: "Set or clear the global response language (e.g. /gsd language Chinese)" },
|
|
83
84
|
];
|
|
84
85
|
|
|
85
86
|
const NESTED_COMPLETIONS: CompletionMap = {
|
|
@@ -269,6 +270,10 @@ const NESTED_COMPLETIONS: CompletionMap = {
|
|
|
269
270
|
{ cmd: "--dry-run", desc: "Preview what would be filtered" },
|
|
270
271
|
{ cmd: "--name", desc: "Custom branch name" },
|
|
271
272
|
],
|
|
273
|
+
language: [
|
|
274
|
+
{ cmd: "off", desc: "Clear the language preference (revert to default)" },
|
|
275
|
+
{ cmd: "clear", desc: "Alias for off — clear the language preference" },
|
|
276
|
+
],
|
|
272
277
|
};
|
|
273
278
|
|
|
274
279
|
function filterOptions(
|
|
@@ -4,7 +4,7 @@ import type { GSDState } from "../../types.js";
|
|
|
4
4
|
|
|
5
5
|
import { computeProgressScore, formatProgressLine } from "../../progress-score.js";
|
|
6
6
|
import { loadEffectiveGSDPreferences, getGlobalGSDPreferencesPath, getProjectGSDPreferencesPath } from "../../preferences.js";
|
|
7
|
-
import { ensurePreferencesFile, handlePrefs, handlePrefsMode, handlePrefsWizard } from "../../commands-prefs-wizard.js";
|
|
7
|
+
import { ensurePreferencesFile, handlePrefs, handlePrefsMode, handlePrefsWizard, handleLanguage } from "../../commands-prefs-wizard.js";
|
|
8
8
|
import { runEnvironmentChecks } from "../../doctor-environment.js";
|
|
9
9
|
import { deriveState } from "../../state.js";
|
|
10
10
|
import { handleCmux } from "../../commands-cmux.js";
|
|
@@ -395,6 +395,10 @@ export async function handleCoreCommand(
|
|
|
395
395
|
await handlePrefs(trimmed.replace(/^prefs\s*/, "").trim(), ctx);
|
|
396
396
|
return true;
|
|
397
397
|
}
|
|
398
|
+
if (trimmed === "language" || trimmed.startsWith("language ")) {
|
|
399
|
+
await handleLanguage(trimmed.replace(/^language\s*/, "").trim(), ctx);
|
|
400
|
+
return true;
|
|
401
|
+
}
|
|
398
402
|
if (trimmed === "cmux" || trimmed.startsWith("cmux ")) {
|
|
399
403
|
await handleCmux(trimmed.replace(/^cmux\s*/, "").trim(), ctx);
|
|
400
404
|
return true;
|
|
@@ -236,5 +236,10 @@ Examples:
|
|
|
236
236
|
await handleAddTests(trimmed.replace(/^add-tests\s*/, "").trim(), ctx, pi);
|
|
237
237
|
return true;
|
|
238
238
|
}
|
|
239
|
+
if (trimmed === "extract-learnings" || trimmed.startsWith("extract-learnings ")) {
|
|
240
|
+
const { handleExtractLearnings } = await import("../../commands-extract-learnings.js");
|
|
241
|
+
await handleExtractLearnings(trimmed.replace(/^extract-learnings\s*/, "").trim(), ctx, pi);
|
|
242
|
+
return true;
|
|
243
|
+
}
|
|
239
244
|
return false;
|
|
240
245
|
}
|
|
@@ -0,0 +1,304 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* GSD Command — /gsd extract-learnings
|
|
3
|
+
*
|
|
4
|
+
* Analyses completed milestone artefacts and dispatches an LLM turn that
|
|
5
|
+
* extracts structured knowledge into 4 categories:
|
|
6
|
+
* Decisions · Lessons · Patterns · Surprises
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import type { ExtensionAPI, ExtensionCommandContext } from "@gsd/pi-coding-agent";
|
|
10
|
+
|
|
11
|
+
import { existsSync, readFileSync } from "node:fs";
|
|
12
|
+
import { join, basename } from "node:path";
|
|
13
|
+
|
|
14
|
+
import { gsdRoot, resolveMilestonePath } from "./paths.js";
|
|
15
|
+
import { projectRoot } from "./commands/context.js";
|
|
16
|
+
|
|
17
|
+
// ─── Types ────────────────────────────────────────────────────────────────────
|
|
18
|
+
|
|
19
|
+
export interface PhaseArtifacts {
|
|
20
|
+
plan: string | null;
|
|
21
|
+
summary: string | null;
|
|
22
|
+
verification: string | null;
|
|
23
|
+
uat: string | null;
|
|
24
|
+
missingRequired: string[];
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
export interface ExtractLearningsPromptContext {
|
|
28
|
+
milestoneId: string;
|
|
29
|
+
milestoneName: string;
|
|
30
|
+
outputPath: string;
|
|
31
|
+
relativeOutputPath: string;
|
|
32
|
+
planContent: string;
|
|
33
|
+
summaryContent: string;
|
|
34
|
+
verificationContent: string | null;
|
|
35
|
+
uatContent: string | null;
|
|
36
|
+
missingArtifacts: string[];
|
|
37
|
+
projectName: string;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
export interface FrontmatterContext {
|
|
41
|
+
milestoneId: string;
|
|
42
|
+
milestoneName: string;
|
|
43
|
+
projectName: string;
|
|
44
|
+
generatedAt: string;
|
|
45
|
+
counts: {
|
|
46
|
+
decisions: number;
|
|
47
|
+
lessons: number;
|
|
48
|
+
patterns: number;
|
|
49
|
+
surprises: number;
|
|
50
|
+
};
|
|
51
|
+
missingArtifacts: string[];
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
// ─── Pure functions ───────────────────────────────────────────────────────────
|
|
55
|
+
|
|
56
|
+
export function parseExtractLearningsArgs(args: string): { milestoneId: string | null } {
|
|
57
|
+
const trimmed = args.trim();
|
|
58
|
+
return { milestoneId: trimmed || null };
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
export function buildLearningsOutputPath(milestoneDir: string, milestoneId: string): string {
|
|
62
|
+
return join(milestoneDir, `${milestoneId}-LEARNINGS.md`);
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
export function resolvePhaseArtifacts(milestoneDir: string, milestoneId: string): PhaseArtifacts {
|
|
66
|
+
const missingRequired: string[] = [];
|
|
67
|
+
|
|
68
|
+
const planFile = `${milestoneId}-PLAN.md`;
|
|
69
|
+
const summaryFile = `${milestoneId}-SUMMARY.md`;
|
|
70
|
+
const verificationFile = `${milestoneId}-VERIFICATION.md`;
|
|
71
|
+
const uatFile = `${milestoneId}-UAT.md`;
|
|
72
|
+
|
|
73
|
+
const planPath = join(milestoneDir, planFile);
|
|
74
|
+
const summaryPath = join(milestoneDir, summaryFile);
|
|
75
|
+
const verificationPath = join(milestoneDir, verificationFile);
|
|
76
|
+
const uatPath = join(milestoneDir, uatFile);
|
|
77
|
+
|
|
78
|
+
const plan = existsSync(planPath) ? planPath : null;
|
|
79
|
+
const summary = existsSync(summaryPath) ? summaryPath : null;
|
|
80
|
+
const verification = existsSync(verificationPath) ? verificationPath : null;
|
|
81
|
+
const uat = existsSync(uatPath) ? uatPath : null;
|
|
82
|
+
|
|
83
|
+
if (!plan) missingRequired.push(planFile);
|
|
84
|
+
if (!summary) missingRequired.push(summaryFile);
|
|
85
|
+
|
|
86
|
+
return { plan, summary, verification, uat, missingRequired };
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
export function buildExtractLearningsPrompt(ctx: ExtractLearningsPromptContext): string {
|
|
90
|
+
const optionalSections: string[] = [];
|
|
91
|
+
|
|
92
|
+
if (ctx.verificationContent) {
|
|
93
|
+
optionalSections.push(`## Verification Report\n\n${ctx.verificationContent}`);
|
|
94
|
+
}
|
|
95
|
+
if (ctx.uatContent) {
|
|
96
|
+
optionalSections.push(`## UAT Report\n\n${ctx.uatContent}`);
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
const missingNote = ctx.missingArtifacts.length > 0
|
|
100
|
+
? `\nNote: The following optional artefacts were not available: ${ctx.missingArtifacts.join(", ")}\n`
|
|
101
|
+
: "";
|
|
102
|
+
|
|
103
|
+
return `# Extract Learnings — ${ctx.milestoneId}: ${ctx.milestoneName}
|
|
104
|
+
|
|
105
|
+
**Project:** ${ctx.projectName}
|
|
106
|
+
**Output file:** ${ctx.outputPath}
|
|
107
|
+
|
|
108
|
+
## Your Task
|
|
109
|
+
|
|
110
|
+
Analyse the artefacts below and extract structured knowledge from milestone **${ctx.milestoneId}**.
|
|
111
|
+
|
|
112
|
+
Write a LEARNINGS document to \`${ctx.outputPath}\` with the following 4 sections:
|
|
113
|
+
|
|
114
|
+
### Decisions
|
|
115
|
+
Key architectural and design decisions made during this milestone, including the rationale and alternatives considered.
|
|
116
|
+
|
|
117
|
+
### Lessons
|
|
118
|
+
What the team learned — technical discoveries, process insights, and knowledge gaps that were filled.
|
|
119
|
+
|
|
120
|
+
### Patterns
|
|
121
|
+
Reusable patterns, approaches, or solutions that emerged and should be applied in future work.
|
|
122
|
+
|
|
123
|
+
### Surprises
|
|
124
|
+
Unexpected challenges, discoveries, or outcomes — things that deviated from assumptions.
|
|
125
|
+
|
|
126
|
+
### Source Attribution (REQUIRED)
|
|
127
|
+
|
|
128
|
+
Every extracted item MUST include a \`Source:\` line immediately after the item text.
|
|
129
|
+
Format: \`Source: {artifact-filename}/{section}\`
|
|
130
|
+
Example: \`Source: M001-PLAN.md/Architecture Decisions\`
|
|
131
|
+
|
|
132
|
+
Items without a Source attribution are invalid and must not be included in the output.
|
|
133
|
+
|
|
134
|
+
---
|
|
135
|
+
|
|
136
|
+
## Artefacts
|
|
137
|
+
|
|
138
|
+
### Plan
|
|
139
|
+
|
|
140
|
+
${ctx.planContent}
|
|
141
|
+
|
|
142
|
+
---
|
|
143
|
+
|
|
144
|
+
### Summary
|
|
145
|
+
|
|
146
|
+
${ctx.summaryContent}
|
|
147
|
+
|
|
148
|
+
${optionalSections.join("\n\n---\n\n")}
|
|
149
|
+
${missingNote}
|
|
150
|
+
---
|
|
151
|
+
|
|
152
|
+
## Output Format
|
|
153
|
+
|
|
154
|
+
Write the LEARNINGS file to \`${ctx.relativeOutputPath}\` with YAML frontmatter followed by the 4 sections above.
|
|
155
|
+
Each section should contain concise, actionable bullet points.
|
|
156
|
+
Every bullet point MUST be followed by a source line, for example:
|
|
157
|
+
|
|
158
|
+
\`\`\`
|
|
159
|
+
### Decisions
|
|
160
|
+
- Chose PostgreSQL over SQLite for concurrent write support.
|
|
161
|
+
Source: M001-PLAN.md/Architecture Decisions
|
|
162
|
+
\`\`\`
|
|
163
|
+
|
|
164
|
+
Items without a \`Source:\` line are invalid.
|
|
165
|
+
|
|
166
|
+
---
|
|
167
|
+
|
|
168
|
+
## Optional: Capture Individual Learnings
|
|
169
|
+
|
|
170
|
+
If the \`capture_thought\` tool is available, call it once for each extracted item with:
|
|
171
|
+
- category: "decision" | "lesson" | "pattern" | "surprise"
|
|
172
|
+
- phase: "${ctx.milestoneId}"
|
|
173
|
+
- content: {the learning text}
|
|
174
|
+
- source: {artifact filename}
|
|
175
|
+
|
|
176
|
+
If \`capture_thought\` is not available, skip this step silently — do not report an error.
|
|
177
|
+
|
|
178
|
+
---
|
|
179
|
+
|
|
180
|
+
## Rebuild Knowledge Graph
|
|
181
|
+
|
|
182
|
+
After writing LEARNINGS.md, call the \`gsd_graph\` tool with \`{ "mode": "build" }\` to rebuild the knowledge graph so the new learnings are immediately queryable by future milestone prompts.
|
|
183
|
+
|
|
184
|
+
If the \`gsd_graph\` tool is not available, skip this step silently.
|
|
185
|
+
`;
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
export function buildFrontmatter(ctx: FrontmatterContext): string {
|
|
189
|
+
const missingList = ctx.missingArtifacts.length > 0
|
|
190
|
+
? ctx.missingArtifacts.map((a) => ` - ${a}`).join("\n")
|
|
191
|
+
: " []";
|
|
192
|
+
|
|
193
|
+
const missingValue = ctx.missingArtifacts.length > 0
|
|
194
|
+
? `\n${missingList}`
|
|
195
|
+
: " []";
|
|
196
|
+
|
|
197
|
+
return `---
|
|
198
|
+
phase: ${ctx.milestoneId}
|
|
199
|
+
phase_name: ${ctx.milestoneName}
|
|
200
|
+
project: ${ctx.projectName}
|
|
201
|
+
generated: ${ctx.generatedAt}
|
|
202
|
+
counts:
|
|
203
|
+
decisions: ${ctx.counts.decisions}
|
|
204
|
+
lessons: ${ctx.counts.lessons}
|
|
205
|
+
patterns: ${ctx.counts.patterns}
|
|
206
|
+
surprises: ${ctx.counts.surprises}
|
|
207
|
+
missing_artifacts:${missingValue}
|
|
208
|
+
---`;
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
export function extractProjectName(basePath: string): string {
|
|
212
|
+
const projectMdPath = join(gsdRoot(basePath), "PROJECT.md");
|
|
213
|
+
|
|
214
|
+
if (existsSync(projectMdPath)) {
|
|
215
|
+
try {
|
|
216
|
+
const content = readFileSync(projectMdPath, "utf-8");
|
|
217
|
+
const match = content.match(/^name:\s*(.+)$/m);
|
|
218
|
+
if (match) return match[1].trim();
|
|
219
|
+
} catch {
|
|
220
|
+
// non-fatal
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
return basename(basePath);
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
// ─── Handler ──────────────────────────────────────────────────────────────────
|
|
228
|
+
|
|
229
|
+
export async function handleExtractLearnings(
|
|
230
|
+
args: string,
|
|
231
|
+
ctx: ExtensionCommandContext,
|
|
232
|
+
pi: ExtensionAPI,
|
|
233
|
+
): Promise<void> {
|
|
234
|
+
const { milestoneId } = parseExtractLearningsArgs(args);
|
|
235
|
+
|
|
236
|
+
if (!milestoneId) {
|
|
237
|
+
ctx.ui.notify("Usage: /gsd extract-learnings <milestoneId> (e.g. M001)", "warning");
|
|
238
|
+
return;
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
// projectRoot() throws GSDNoProjectError if no project found — intentional, handled by dispatcher
|
|
242
|
+
const basePath = projectRoot();
|
|
243
|
+
const milestoneDir = resolveMilestonePath(basePath, milestoneId);
|
|
244
|
+
|
|
245
|
+
if (!milestoneDir) {
|
|
246
|
+
ctx.ui.notify(`Milestone not found: ${milestoneId}`, "error");
|
|
247
|
+
return;
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
const artifacts = resolvePhaseArtifacts(milestoneDir, milestoneId);
|
|
251
|
+
|
|
252
|
+
if (artifacts.missingRequired.length > 0) {
|
|
253
|
+
ctx.ui.notify(
|
|
254
|
+
`Cannot extract learnings — required artefacts missing: ${artifacts.missingRequired.join(", ")}`,
|
|
255
|
+
"error",
|
|
256
|
+
);
|
|
257
|
+
return;
|
|
258
|
+
}
|
|
259
|
+
|
|
260
|
+
// Read required artefacts
|
|
261
|
+
const planContent = readFileSync(artifacts.plan!, "utf-8");
|
|
262
|
+
const summaryContent = readFileSync(artifacts.summary!, "utf-8");
|
|
263
|
+
|
|
264
|
+
// Read optional artefacts
|
|
265
|
+
const verificationContent = artifacts.verification
|
|
266
|
+
? readFileSync(artifacts.verification, "utf-8")
|
|
267
|
+
: null;
|
|
268
|
+
const uatContent = artifacts.uat
|
|
269
|
+
? readFileSync(artifacts.uat, "utf-8")
|
|
270
|
+
: null;
|
|
271
|
+
|
|
272
|
+
// Determine missing optional artefacts for context
|
|
273
|
+
const missingArtifacts: string[] = [];
|
|
274
|
+
if (!artifacts.verification) missingArtifacts.push(`${milestoneId}-VERIFICATION.md`);
|
|
275
|
+
if (!artifacts.uat) missingArtifacts.push(`${milestoneId}-UAT.md`);
|
|
276
|
+
|
|
277
|
+
// Extract milestone name from Plan H1 or fall back to milestoneId
|
|
278
|
+
const h1Match = planContent.match(/^#\s+(.+)$/m);
|
|
279
|
+
const milestoneName = h1Match?.[1]?.trim() ?? milestoneId;
|
|
280
|
+
|
|
281
|
+
const projectName = extractProjectName(basePath);
|
|
282
|
+
const outputPath = buildLearningsOutputPath(milestoneDir, milestoneId);
|
|
283
|
+
const relativeOutputPath = outputPath.replace(basePath + "/", "");
|
|
284
|
+
|
|
285
|
+
const prompt = buildExtractLearningsPrompt({
|
|
286
|
+
milestoneId,
|
|
287
|
+
milestoneName,
|
|
288
|
+
outputPath,
|
|
289
|
+
relativeOutputPath,
|
|
290
|
+
planContent,
|
|
291
|
+
summaryContent,
|
|
292
|
+
verificationContent,
|
|
293
|
+
uatContent,
|
|
294
|
+
missingArtifacts,
|
|
295
|
+
projectName,
|
|
296
|
+
});
|
|
297
|
+
|
|
298
|
+
ctx.ui.notify(`Extracting learnings for ${milestoneId}: "${milestoneName}"...`, "info");
|
|
299
|
+
|
|
300
|
+
pi.sendMessage(
|
|
301
|
+
{ customType: "gsd-extract-learnings", content: prompt, display: false },
|
|
302
|
+
{ triggerTurn: true },
|
|
303
|
+
);
|
|
304
|
+
}
|
|
@@ -756,8 +756,8 @@ export async function handlePrefsWizard(
|
|
|
756
756
|
/** Wrap a YAML value in double quotes if it contains special characters. */
|
|
757
757
|
export function yamlSafeString(val: unknown): string {
|
|
758
758
|
if (typeof val !== "string") return String(val);
|
|
759
|
-
if (/[:#{\[\]'"
|
|
760
|
-
return `"${val.replace(/\\/g, "\\\\").replace(/"/g, '\\"')}"`;
|
|
759
|
+
if (/[:#{\[\]'"`,|>&*!?@%\r\n]/.test(val) || val.trim() !== val || val === "") {
|
|
760
|
+
return `"${val.replace(/\\/g, "\\\\").replace(/"/g, '\\"').replace(/\r/g, "\\r").replace(/\n/g, "\\n")}"`;
|
|
761
761
|
}
|
|
762
762
|
return val;
|
|
763
763
|
}
|
|
@@ -822,10 +822,10 @@ export function serializePreferencesToFrontmatter(prefs: Record<string, unknown>
|
|
|
822
822
|
"budget_ceiling", "budget_enforcement", "context_pause_threshold",
|
|
823
823
|
"notifications", "cmux", "remote_questions", "git",
|
|
824
824
|
"post_unit_hooks", "pre_dispatch_hooks",
|
|
825
|
-
"dynamic_routing", "token_profile", "phases", "parallel",
|
|
825
|
+
"dynamic_routing", "uok", "token_profile", "phases", "parallel",
|
|
826
826
|
"auto_visualize", "auto_report",
|
|
827
827
|
"verification_commands", "verification_auto_fix", "verification_max_retries",
|
|
828
|
-
"search_provider", "context_selection",
|
|
828
|
+
"search_provider", "context_selection", "language",
|
|
829
829
|
];
|
|
830
830
|
|
|
831
831
|
const seen = new Set<string>();
|
|
@@ -862,3 +862,57 @@ export async function ensurePreferencesFile(
|
|
|
862
862
|
ctx.ui.notify(`Using existing ${scope} GSD skill preferences at ${path}`, "info");
|
|
863
863
|
}
|
|
864
864
|
}
|
|
865
|
+
|
|
866
|
+
/**
|
|
867
|
+
* Handle `/gsd language [code]` — set or clear the global language preference.
|
|
868
|
+
* Without an argument, shows the current setting.
|
|
869
|
+
* Project-level override can be set by editing `.gsd/preferences.md` directly
|
|
870
|
+
* (project language overrides global when both are set).
|
|
871
|
+
*/
|
|
872
|
+
export async function handleLanguage(args: string, ctx: ExtensionCommandContext): Promise<void> {
|
|
873
|
+
const path = getGlobalGSDPreferencesPath();
|
|
874
|
+
const lang = args.trim();
|
|
875
|
+
|
|
876
|
+
// Show current setting when called without argument
|
|
877
|
+
if (!lang) {
|
|
878
|
+
const loaded = loadGlobalGSDPreferences();
|
|
879
|
+
const current = loaded?.preferences.language;
|
|
880
|
+
if (current) {
|
|
881
|
+
ctx.ui.notify(`Current language preference: ${current}\nUse /gsd language <name> to change, or /gsd language off to clear.`, "info");
|
|
882
|
+
} else {
|
|
883
|
+
ctx.ui.notify("No language preference set. Use /gsd language <name> to set one (e.g. /gsd language Chinese).", "info");
|
|
884
|
+
}
|
|
885
|
+
return;
|
|
886
|
+
}
|
|
887
|
+
|
|
888
|
+
// Ensure preferences file exists with the canonical template
|
|
889
|
+
await ensurePreferencesFile(path, ctx, "global");
|
|
890
|
+
|
|
891
|
+
// Read via the same validated path as other handlers
|
|
892
|
+
const existing = loadGlobalGSDPreferences();
|
|
893
|
+
const prefs: Record<string, unknown> = existing?.preferences ? { ...existing.preferences } : { version: 1 };
|
|
894
|
+
|
|
895
|
+
if (lang === "off" || lang === "none" || lang === "clear") {
|
|
896
|
+
delete prefs.language;
|
|
897
|
+
ctx.ui.notify("Language preference cleared. GSD will use the default language.", "info");
|
|
898
|
+
} else {
|
|
899
|
+
// Validate before writing — reject values that would fail on next load
|
|
900
|
+
if (lang.length > 50 || /[\r\n]/.test(lang)) {
|
|
901
|
+
ctx.ui.notify(
|
|
902
|
+
"Language value must be 50 characters or fewer with no newlines (e.g. /gsd language Chinese).",
|
|
903
|
+
"warning",
|
|
904
|
+
);
|
|
905
|
+
return;
|
|
906
|
+
}
|
|
907
|
+
prefs.language = lang;
|
|
908
|
+
ctx.ui.notify(`Language preference set to: ${lang}\nGSD will now respond in ${lang} across all sessions.`, "info");
|
|
909
|
+
}
|
|
910
|
+
|
|
911
|
+
const rawContent = existsSync(path) ? readFileSync(path, "utf-8") : `---\nversion: 1\n---\n`;
|
|
912
|
+
const frontmatter = serializePreferencesToFrontmatter(prefs);
|
|
913
|
+
const body = extractBodyAfterFrontmatter(rawContent)
|
|
914
|
+
?? "\n# GSD Skill Preferences\n\nSee `~/.gsd/agent/extensions/gsd/docs/preferences-reference.md` for full field documentation and examples.\n";
|
|
915
|
+
await saveFile(path, `---\n${frontmatter}---${body}`);
|
|
916
|
+
await ctx.waitForIdle();
|
|
917
|
+
await ctx.reload();
|
|
918
|
+
}
|