gsd-pi 2.73.1-dev.d987996 → 2.74.0-dev.0306a2e
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/cli-web-branch.d.ts +4 -3
- package/dist/cli-web-branch.js +10 -7
- package/dist/cli.js +184 -206
- package/dist/headless-query.js +4 -1
- package/dist/help-text.js +23 -0
- package/dist/logo.d.ts +1 -1
- package/dist/logo.js +1 -1
- package/dist/onboarding.js +59 -53
- package/dist/resource-loader.js +2 -2
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +68 -4
- package/dist/resources/extensions/gsd/activity-log.js +16 -0
- package/dist/resources/extensions/gsd/auto/detect-stuck.js +11 -4
- package/dist/resources/extensions/gsd/auto/loop.js +147 -10
- package/dist/resources/extensions/gsd/auto/phases.js +173 -13
- package/dist/resources/extensions/gsd/auto/session.js +10 -0
- package/dist/resources/extensions/gsd/auto-dispatch.js +22 -4
- package/dist/resources/extensions/gsd/auto-model-selection.js +105 -16
- package/dist/resources/extensions/gsd/auto-post-unit.js +254 -15
- package/dist/resources/extensions/gsd/auto-prompts.js +12 -0
- package/dist/resources/extensions/gsd/auto-start.js +23 -6
- package/dist/resources/extensions/gsd/auto-timeout-recovery.js +13 -0
- package/dist/resources/extensions/gsd/auto-unit-closeout.js +18 -0
- package/dist/resources/extensions/gsd/auto-verification.js +186 -3
- package/dist/resources/extensions/gsd/auto.js +65 -12
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +30 -8
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +41 -2
- package/dist/resources/extensions/gsd/commands/catalog.js +26 -1
- package/dist/resources/extensions/gsd/commands/handlers/ops.js +25 -0
- package/dist/resources/extensions/gsd/commands/handlers/workflow.js +68 -9
- package/dist/resources/extensions/gsd/commands-add-tests.js +111 -0
- package/dist/resources/extensions/gsd/commands-backlog.js +140 -0
- package/dist/resources/extensions/gsd/commands-do.js +79 -0
- package/dist/resources/extensions/gsd/commands-extract-learnings.js +225 -0
- package/dist/resources/extensions/gsd/commands-handlers.js +8 -2
- package/dist/resources/extensions/gsd/commands-maintenance.js +6 -6
- package/dist/resources/extensions/gsd/commands-pr-branch.js +180 -0
- package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
- package/dist/resources/extensions/gsd/commands-session-report.js +82 -0
- package/dist/resources/extensions/gsd/commands-ship.js +187 -0
- package/dist/resources/extensions/gsd/db-writer.js +3 -5
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +15 -2
- 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 +157 -0
- package/dist/resources/extensions/gsd/gsd-db.js +581 -2
- package/dist/resources/extensions/gsd/guided-flow.js +23 -0
- package/dist/resources/extensions/gsd/index.js +15 -2
- package/dist/resources/extensions/gsd/init-wizard.js +1 -0
- package/dist/resources/extensions/gsd/journal.js +27 -0
- package/dist/resources/extensions/gsd/md-importer.js +3 -4
- package/dist/resources/extensions/gsd/memory-store.js +19 -51
- package/dist/resources/extensions/gsd/metrics.js +19 -0
- package/dist/resources/extensions/gsd/milestone-validation-gates.js +13 -12
- package/dist/resources/extensions/gsd/native-git-bridge.js +7 -4
- package/dist/resources/extensions/gsd/notification-widget.js +2 -2
- package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
- package/dist/resources/extensions/gsd/preferences-models.js +63 -3
- package/dist/resources/extensions/gsd/preferences-types.js +2 -0
- package/dist/resources/extensions/gsd/preferences-validation.js +130 -2
- package/dist/resources/extensions/gsd/preferences.js +26 -0
- package/dist/resources/extensions/gsd/prompts/add-tests.md +35 -0
- package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
- package/dist/resources/extensions/gsd/state.js +66 -15
- package/dist/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
- package/dist/resources/extensions/gsd/tools/complete-slice.js +20 -0
- package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +3 -14
- package/dist/resources/extensions/gsd/triage-resolution.js +2 -5
- 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/gsd/workflow-manifest.js +8 -69
- package/dist/resources/extensions/gsd/workflow-migration.js +21 -22
- package/dist/resources/extensions/gsd/workflow-projections.js +4 -1
- package/dist/resources/extensions/gsd/workflow-reconcile.js +14 -11
- package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
- package/dist/tsconfig.extensions.tsbuildinfo +1 -0
- package/dist/update-check.d.ts +1 -0
- package/dist/update-check.js +13 -5
- package/dist/update-cmd.js +4 -3
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +13 -13
- 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 +13 -13
- 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 +3 -3
- package/packages/daemon/package.json +2 -2
- package/packages/mcp-server/dist/index.d.ts +3 -0
- package/packages/mcp-server/dist/index.d.ts.map +1 -1
- package/packages/mcp-server/dist/index.js +3 -0
- package/packages/mcp-server/dist/index.js.map +1 -1
- package/packages/mcp-server/dist/readers/graph.d.ts +87 -0
- package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -0
- package/packages/mcp-server/dist/readers/graph.js +655 -0
- package/packages/mcp-server/dist/readers/graph.js.map +1 -0
- package/packages/mcp-server/dist/readers/index.d.ts +2 -0
- package/packages/mcp-server/dist/readers/index.d.ts.map +1 -1
- package/packages/mcp-server/dist/readers/index.js +1 -0
- package/packages/mcp-server/dist/readers/index.js.map +1 -1
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +65 -0
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/package.json +2 -2
- package/packages/mcp-server/src/index.ts +15 -0
- package/packages/mcp-server/src/readers/graph.test.ts +604 -0
- package/packages/mcp-server/src/readers/graph.ts +855 -0
- package/packages/mcp-server/src/readers/index.ts +12 -0
- package/packages/mcp-server/src/server.ts +83 -0
- package/packages/mcp-server/tsconfig.json +1 -0
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -0
- package/packages/native/package.json +2 -2
- package/packages/native/tsconfig.tsbuildinfo +1 -0
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-agent-core/tsconfig.json +1 -0
- package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -0
- package/packages/pi-ai/dist/index.d.ts +2 -9
- package/packages/pi-ai/dist/index.d.ts.map +1 -1
- package/packages/pi-ai/dist/index.js +2 -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/dist/utils/overflow.d.ts.map +1 -1
- package/packages/pi-ai/dist/utils/overflow.js +12 -0
- package/packages/pi-ai/dist/utils/overflow.js.map +1 -1
- package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts +2 -0
- package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/utils/tests/overflow.test.js +50 -0
- package/packages/pi-ai/dist/utils/tests/overflow.test.js.map +1 -0
- package/packages/pi-ai/package.json +1 -1
- package/packages/pi-ai/scripts/generate-models.ts +74 -40
- package/packages/pi-ai/src/index.ts +5 -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/src/utils/overflow.ts +14 -1
- package/packages/pi-ai/src/utils/tests/overflow.test.ts +58 -0
- package/packages/pi-ai/tsconfig.json +1 -0
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -0
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +721 -8
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction/utils.js +5 -5
- package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +45 -0
- package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -0
- 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/assistant-message.d.ts +12 -2
- 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 +65 -28
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts +2 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js +9 -3
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +52 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -0
- 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 +305 -20
- 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 +13 -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 +59 -6
- 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/chat-controller-ordering.test.ts +884 -8
- package/packages/pi-coding-agent/src/core/compaction/utils.ts +5 -5
- package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +50 -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/assistant-message.ts +78 -32
- package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +73 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.ts +9 -3
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +381 -39
- 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 +79 -6
- package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
- package/packages/pi-coding-agent/tsconfig.json +3 -2
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -0
- package/packages/pi-tui/dist/__tests__/tui.test.js +60 -1
- package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
- package/packages/pi-tui/dist/tui.d.ts +8 -0
- package/packages/pi-tui/dist/tui.d.ts.map +1 -1
- package/packages/pi-tui/dist/tui.js +32 -3
- package/packages/pi-tui/dist/tui.js.map +1 -1
- package/packages/pi-tui/package.json +1 -1
- package/packages/pi-tui/src/__tests__/tui.test.ts +76 -1
- package/packages/pi-tui/src/tui.ts +31 -3
- package/packages/pi-tui/tsconfig.json +1 -0
- package/packages/pi-tui/tsconfig.tsbuildinfo +1 -0
- package/packages/rpc-client/package.json +1 -1
- package/packages/rpc-client/tsconfig.json +1 -0
- package/packages/rpc-client/tsconfig.tsbuildinfo +1 -0
- package/pkg/package.json +1 -1
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +107 -5
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +111 -2
- package/src/resources/extensions/gsd/activity-log.ts +21 -0
- package/src/resources/extensions/gsd/auto/detect-stuck.ts +12 -4
- package/src/resources/extensions/gsd/auto/loop-deps.ts +10 -0
- package/src/resources/extensions/gsd/auto/loop.ts +159 -10
- package/src/resources/extensions/gsd/auto/phases.ts +213 -13
- package/src/resources/extensions/gsd/auto/session.ts +10 -0
- package/src/resources/extensions/gsd/auto-dispatch.ts +26 -10
- package/src/resources/extensions/gsd/auto-model-selection.ts +151 -16
- package/src/resources/extensions/gsd/auto-post-unit.ts +278 -16
- package/src/resources/extensions/gsd/auto-prompts.ts +13 -0
- package/src/resources/extensions/gsd/auto-start.ts +30 -6
- package/src/resources/extensions/gsd/auto-timeout-recovery.ts +17 -0
- package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
- package/src/resources/extensions/gsd/auto-verification.ts +225 -3
- package/src/resources/extensions/gsd/auto.ts +72 -16
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +38 -8
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +52 -2
- package/src/resources/extensions/gsd/commands/catalog.ts +26 -1
- package/src/resources/extensions/gsd/commands/handlers/ops.ts +25 -0
- package/src/resources/extensions/gsd/commands/handlers/workflow.ts +74 -9
- package/src/resources/extensions/gsd/commands-add-tests.ts +137 -0
- package/src/resources/extensions/gsd/commands-backlog.ts +182 -0
- package/src/resources/extensions/gsd/commands-do.ts +109 -0
- package/src/resources/extensions/gsd/commands-extract-learnings.ts +304 -0
- package/src/resources/extensions/gsd/commands-handlers.ts +8 -2
- package/src/resources/extensions/gsd/commands-maintenance.ts +6 -6
- package/src/resources/extensions/gsd/commands-pr-branch.ts +234 -0
- package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
- package/src/resources/extensions/gsd/commands-session-report.ts +101 -0
- package/src/resources/extensions/gsd/commands-ship.ts +219 -0
- package/src/resources/extensions/gsd/db-writer.ts +3 -5
- package/src/resources/extensions/gsd/docs/preferences-reference.md +15 -2
- 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 +212 -0
- package/src/resources/extensions/gsd/gsd-db.ts +788 -3
- package/src/resources/extensions/gsd/guided-flow.ts +32 -0
- package/src/resources/extensions/gsd/index.ts +18 -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/md-importer.ts +3 -5
- package/src/resources/extensions/gsd/memory-store.ts +31 -62
- package/src/resources/extensions/gsd/metrics.ts +26 -0
- package/src/resources/extensions/gsd/milestone-validation-gates.ts +13 -14
- package/src/resources/extensions/gsd/native-git-bridge.ts +11 -12
- package/src/resources/extensions/gsd/notification-widget.ts +2 -2
- package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
- package/src/resources/extensions/gsd/preferences-models.ts +61 -3
- package/src/resources/extensions/gsd/preferences-types.ts +44 -0
- package/src/resources/extensions/gsd/preferences-validation.ts +130 -2
- package/src/resources/extensions/gsd/preferences.ts +28 -0
- package/src/resources/extensions/gsd/prompts/add-tests.md +35 -0
- 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/state.ts +80 -17
- package/src/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +9 -5
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
- package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +53 -0
- package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
- package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +51 -2
- package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
- package/src/resources/extensions/gsd/tests/commands-backlog.test.ts +158 -0
- package/src/resources/extensions/gsd/tests/commands-do.test.ts +127 -0
- package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +340 -0
- package/src/resources/extensions/gsd/tests/commands-pr-branch.test.ts +68 -0
- package/src/resources/extensions/gsd/tests/commands-session-report.test.ts +82 -0
- package/src/resources/extensions/gsd/tests/commands-ship.test.ts +71 -0
- package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +14 -0
- package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +142 -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/completed-at-reconcile.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +68 -8
- package/src/resources/extensions/gsd/tests/derive-state.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +154 -0
- package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +10 -7
- package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +137 -1
- package/src/resources/extensions/gsd/tests/graph-context.test.ts +337 -0
- 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/integration/state-machine-edge-cases.test.ts +4 -2
- package/src/resources/extensions/gsd/tests/journal-integration.test.ts +68 -1
- 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/model-isolation.test.ts +91 -2
- package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +140 -0
- 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 +47 -0
- package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +180 -0
- package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +5 -7
- package/src/resources/extensions/gsd/tests/token-profile.test.ts +9 -6
- 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-stuck-guard.test.ts +179 -0
- package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +223 -0
- package/src/resources/extensions/gsd/tools/complete-slice.ts +26 -0
- package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +3 -11
- package/src/resources/extensions/gsd/triage-resolution.ts +2 -7
- 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/gsd/workflow-manifest.ts +9 -104
- package/src/resources/extensions/gsd/workflow-migration.ts +21 -29
- package/src/resources/extensions/gsd/workflow-projections.ts +8 -1
- package/src/resources/extensions/gsd/workflow-reconcile.ts +15 -15
- 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/{cGmbVq2su4f9tMpgIkG8u → tqdo0yKKYz6fJXQnIgbdx}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{cGmbVq2su4f9tMpgIkG8u → tqdo0yKKYz6fJXQnIgbdx}/_ssgManifest.js +0 -0
|
@@ -2,9 +2,10 @@
|
|
|
2
2
|
|
|
3
3
|
import type { ExtensionAPI, ExtensionCommandContext } from "@gsd/pi-coding-agent";
|
|
4
4
|
|
|
5
|
-
import { registerGSDCommand } from "../commands.js";
|
|
6
5
|
import { registerExitCommand } from "../exit-command.js";
|
|
7
6
|
import { registerWorktreeCommand } from "../worktree-command.js";
|
|
7
|
+
import type { GSDEcosystemBeforeAgentStartHandler } from "../ecosystem/gsd-extension-api.js";
|
|
8
|
+
import { loadEcosystemExtensions } from "../ecosystem/loader.js";
|
|
8
9
|
import { registerDbTools } from "./db-tools.js";
|
|
9
10
|
import { registerDynamicTools } from "./dynamic-tools.js";
|
|
10
11
|
import { registerJournalTools } from "./journal-tools.js";
|
|
@@ -12,6 +13,7 @@ import { registerQueryTools } from "./query-tools.js";
|
|
|
12
13
|
import { registerHooks } from "./register-hooks.js";
|
|
13
14
|
import { registerShortcuts } from "./register-shortcuts.js";
|
|
14
15
|
import { writeCrashLog } from "./crash-log.js";
|
|
16
|
+
import { logWarning } from "../workflow-logger.js";
|
|
15
17
|
|
|
16
18
|
export { writeCrashLog } from "./crash-log.js";
|
|
17
19
|
|
|
@@ -58,12 +60,17 @@ function installEpipeGuard(): void {
|
|
|
58
60
|
}
|
|
59
61
|
|
|
60
62
|
export function registerGsdExtension(pi: ExtensionAPI): void {
|
|
61
|
-
registerGSDCommand
|
|
63
|
+
// Note: registerGSDCommand is called by index.ts before this function,
|
|
64
|
+
// so we intentionally skip it here to avoid double-registration.
|
|
62
65
|
registerWorktreeCommand(pi);
|
|
63
66
|
registerExitCommand(pi);
|
|
64
67
|
|
|
65
68
|
installEpipeGuard();
|
|
66
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
|
+
|
|
67
74
|
pi.registerCommand("kill", {
|
|
68
75
|
description: "Exit GSD immediately (no cleanup)",
|
|
69
76
|
handler: async (_args: string, _ctx: ExtensionCommandContext) => {
|
|
@@ -71,10 +78,33 @@ export function registerGsdExtension(pi: ExtensionAPI): void {
|
|
|
71
78
|
},
|
|
72
79
|
});
|
|
73
80
|
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
81
|
+
// Wrap non-critical registrations individually so one failure
|
|
82
|
+
// doesn't prevent the others from loading.
|
|
83
|
+
const nonCriticalRegistrations: Array<[string, () => void]> = [
|
|
84
|
+
["dynamic-tools", () => registerDynamicTools(pi)],
|
|
85
|
+
["db-tools", () => registerDbTools(pi)],
|
|
86
|
+
["journal-tools", () => registerJournalTools(pi)],
|
|
87
|
+
["query-tools", () => registerQueryTools(pi)],
|
|
88
|
+
["shortcuts", () => registerShortcuts(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
|
+
}],
|
|
98
|
+
];
|
|
99
|
+
|
|
100
|
+
for (const [name, register] of nonCriticalRegistrations) {
|
|
101
|
+
try {
|
|
102
|
+
register();
|
|
103
|
+
} catch (err) {
|
|
104
|
+
logWarning(
|
|
105
|
+
"bootstrap",
|
|
106
|
+
`Failed to register ${name}: ${err instanceof Error ? err.message : String(err)}`,
|
|
107
|
+
);
|
|
108
|
+
}
|
|
109
|
+
}
|
|
80
110
|
}
|
|
@@ -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);
|
|
@@ -93,7 +100,50 @@ export function registerHooks(pi: ExtensionAPI): void {
|
|
|
93
100
|
});
|
|
94
101
|
|
|
95
102
|
pi.on("before_agent_start", async (event, ctx: ExtensionContext) => {
|
|
96
|
-
|
|
103
|
+
// Wait for ecosystem loader to finish (no-op after first turn).
|
|
104
|
+
await getEcosystemReadyPromise();
|
|
105
|
+
|
|
106
|
+
// GSD's own context injection (existing behavior — unchanged).
|
|
107
|
+
const gsdResult = await buildBeforeAgentStartResult(event, ctx);
|
|
108
|
+
|
|
109
|
+
// Refresh the snapshot used by ecosystem getPhase()/getActiveUnit().
|
|
110
|
+
// deriveState has its own ~100ms cache so this is cheap on repeat calls.
|
|
111
|
+
try {
|
|
112
|
+
const state = await deriveState(process.cwd());
|
|
113
|
+
updateSnapshot(state);
|
|
114
|
+
} catch {
|
|
115
|
+
updateSnapshot(null);
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
// Chain ecosystem handlers using pi's runner.ts chaining protocol:
|
|
119
|
+
// each handler sees the systemPrompt mutated by prior handlers.
|
|
120
|
+
let currentSystemPrompt = gsdResult?.systemPrompt ?? event.systemPrompt;
|
|
121
|
+
// `any` because pi's BeforeAgentStartEventResult.message uses an internal
|
|
122
|
+
// CustomMessage type that's not re-exported (see ecosystem/gsd-extension-api.ts).
|
|
123
|
+
let lastMessage: any = gsdResult?.message;
|
|
124
|
+
|
|
125
|
+
for (const handler of ecosystemHandlers) {
|
|
126
|
+
try {
|
|
127
|
+
const r = await handler(
|
|
128
|
+
{ ...event, systemPrompt: currentSystemPrompt },
|
|
129
|
+
ctx,
|
|
130
|
+
);
|
|
131
|
+
if (r?.systemPrompt !== undefined) currentSystemPrompt = r.systemPrompt;
|
|
132
|
+
if (r?.message) lastMessage = r.message;
|
|
133
|
+
} catch (err) {
|
|
134
|
+
safetyLogWarning(
|
|
135
|
+
"ecosystem",
|
|
136
|
+
`before_agent_start handler failed: ${err instanceof Error ? err.message : String(err)}`,
|
|
137
|
+
);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
// Compose result. Return undefined if nothing changed (preserves runner contract).
|
|
142
|
+
if (currentSystemPrompt === event.systemPrompt && !lastMessage) return undefined;
|
|
143
|
+
return {
|
|
144
|
+
systemPrompt: currentSystemPrompt !== event.systemPrompt ? currentSystemPrompt : undefined,
|
|
145
|
+
message: lastMessage,
|
|
146
|
+
};
|
|
97
147
|
});
|
|
98
148
|
|
|
99
149
|
pi.on("agent_end", async (event, ctx: ExtensionContext) => {
|
|
@@ -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";
|
|
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";
|
|
19
19
|
|
|
20
20
|
export const TOP_LEVEL_SUBCOMMANDS: readonly GsdCommandDefinition[] = [
|
|
21
21
|
{ cmd: "help", desc: "Categorized command reference with descriptions" },
|
|
@@ -74,6 +74,12 @@ export const TOP_LEVEL_SUBCOMMANDS: readonly GsdCommandDefinition[] = [
|
|
|
74
74
|
{ cmd: "rethink", desc: "Conversational project reorganization — reorder, park, discard, add milestones" },
|
|
75
75
|
{ cmd: "workflow", desc: "Custom workflow lifecycle (new, run, list, validate, pause, resume)" },
|
|
76
76
|
{ cmd: "codebase", desc: "Generate, refresh, and inspect the codebase map cache (.gsd/CODEBASE.md)" },
|
|
77
|
+
{ cmd: "ship", desc: "Create PR from milestone artifacts and open for review" },
|
|
78
|
+
{ cmd: "do", desc: "Route freeform text to the right GSD command" },
|
|
79
|
+
{ cmd: "session-report", desc: "Session cost, tokens, and work summary" },
|
|
80
|
+
{ cmd: "backlog", desc: "Manage backlog items (add, promote, remove, list)" },
|
|
81
|
+
{ cmd: "pr-branch", desc: "Create clean PR branch filtering .gsd/ commits" },
|
|
82
|
+
{ cmd: "add-tests", desc: "Generate tests for completed slices" },
|
|
77
83
|
];
|
|
78
84
|
|
|
79
85
|
const NESTED_COMPLETIONS: CompletionMap = {
|
|
@@ -244,6 +250,25 @@ const NESTED_COMPLETIONS: CompletionMap = {
|
|
|
244
250
|
{ cmd: "stats", desc: "Show file count, description coverage, and generation time" },
|
|
245
251
|
{ cmd: "help", desc: "Show usage and available subcommands" },
|
|
246
252
|
],
|
|
253
|
+
ship: [
|
|
254
|
+
{ cmd: "--dry-run", desc: "Preview PR without creating" },
|
|
255
|
+
{ cmd: "--draft", desc: "Open as draft PR" },
|
|
256
|
+
{ cmd: "--base", desc: "Override target branch (default: main)" },
|
|
257
|
+
{ cmd: "--force", desc: "Ship even with pending tasks" },
|
|
258
|
+
],
|
|
259
|
+
"session-report": [
|
|
260
|
+
{ cmd: "--json", desc: "Machine-readable JSON output" },
|
|
261
|
+
{ cmd: "--save", desc: "Save report to .gsd/reports/" },
|
|
262
|
+
],
|
|
263
|
+
backlog: [
|
|
264
|
+
{ cmd: "add", desc: "Add item to backlog" },
|
|
265
|
+
{ cmd: "promote", desc: "Promote backlog item to active slice" },
|
|
266
|
+
{ cmd: "remove", desc: "Remove backlog item" },
|
|
267
|
+
],
|
|
268
|
+
"pr-branch": [
|
|
269
|
+
{ cmd: "--dry-run", desc: "Preview what would be filtered" },
|
|
270
|
+
{ cmd: "--name", desc: "Custom branch name" },
|
|
271
|
+
],
|
|
247
272
|
};
|
|
248
273
|
|
|
249
274
|
function filterOptions(
|
|
@@ -11,6 +11,9 @@ import { handleExport } from "../../export.js";
|
|
|
11
11
|
import { handleHistory } from "../../history.js";
|
|
12
12
|
import { handleUndo } from "../../undo.js";
|
|
13
13
|
import { handleRemote } from "../../../remote-questions/mod.js";
|
|
14
|
+
import { handleShip } from "../../commands-ship.js";
|
|
15
|
+
import { handleSessionReport } from "../../commands-session-report.js";
|
|
16
|
+
import { handlePrBranch } from "../../commands-pr-branch.js";
|
|
14
17
|
import { projectRoot } from "../context.js";
|
|
15
18
|
|
|
16
19
|
export async function handleOpsCommand(trimmed: string, ctx: ExtensionCommandContext, pi: ExtensionAPI): Promise<boolean> {
|
|
@@ -216,5 +219,27 @@ Examples:
|
|
|
216
219
|
await handleCodebase(trimmed.replace(/^codebase\s*/, "").trim(), ctx, pi);
|
|
217
220
|
return true;
|
|
218
221
|
}
|
|
222
|
+
if (trimmed === "ship" || trimmed.startsWith("ship ")) {
|
|
223
|
+
await handleShip(trimmed.replace(/^ship\s*/, "").trim(), ctx, pi);
|
|
224
|
+
return true;
|
|
225
|
+
}
|
|
226
|
+
if (trimmed === "session-report" || trimmed.startsWith("session-report ")) {
|
|
227
|
+
await handleSessionReport(trimmed.replace(/^session-report\s*/, "").trim(), ctx);
|
|
228
|
+
return true;
|
|
229
|
+
}
|
|
230
|
+
if (trimmed === "pr-branch" || trimmed.startsWith("pr-branch ")) {
|
|
231
|
+
await handlePrBranch(trimmed.replace(/^pr-branch\s*/, "").trim(), ctx);
|
|
232
|
+
return true;
|
|
233
|
+
}
|
|
234
|
+
if (trimmed === "add-tests" || trimmed.startsWith("add-tests ")) {
|
|
235
|
+
const { handleAddTests } = await import("../../commands-add-tests.js");
|
|
236
|
+
await handleAddTests(trimmed.replace(/^add-tests\s*/, "").trim(), ctx, pi);
|
|
237
|
+
return true;
|
|
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
|
+
}
|
|
219
244
|
return false;
|
|
220
245
|
}
|
|
@@ -38,6 +38,67 @@ const WORKFLOW_USAGE = [
|
|
|
38
38
|
" resume — Resume paused custom workflow auto-mode",
|
|
39
39
|
].join("\n");
|
|
40
40
|
|
|
41
|
+
function splitWorkflowRunArgs(input: string): string[] {
|
|
42
|
+
const tokens: string[] = [];
|
|
43
|
+
let current = "";
|
|
44
|
+
let quote: '"' | "'" | null = null;
|
|
45
|
+
let escapeNext = false;
|
|
46
|
+
|
|
47
|
+
for (const ch of input) {
|
|
48
|
+
if (escapeNext) {
|
|
49
|
+
current += ch;
|
|
50
|
+
escapeNext = false;
|
|
51
|
+
continue;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
if (ch === "\\") {
|
|
55
|
+
escapeNext = true;
|
|
56
|
+
continue;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
if (quote) {
|
|
60
|
+
if (ch === quote) {
|
|
61
|
+
quote = null;
|
|
62
|
+
} else {
|
|
63
|
+
current += ch;
|
|
64
|
+
}
|
|
65
|
+
continue;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
if (ch === '"' || ch === "'") {
|
|
69
|
+
quote = ch;
|
|
70
|
+
continue;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
if (/\s/.test(ch)) {
|
|
74
|
+
if (current) {
|
|
75
|
+
tokens.push(current);
|
|
76
|
+
current = "";
|
|
77
|
+
}
|
|
78
|
+
continue;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
current += ch;
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
if (escapeNext) current += "\\";
|
|
85
|
+
if (current) tokens.push(current);
|
|
86
|
+
return tokens;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
export function parseWorkflowRunArgs(args: string): { defName: string; overrides: Record<string, string> } {
|
|
90
|
+
const parts = splitWorkflowRunArgs(args);
|
|
91
|
+
const defName = parts[0] ?? "";
|
|
92
|
+
const overrides: Record<string, string> = {};
|
|
93
|
+
for (let i = 1; i < parts.length; i++) {
|
|
94
|
+
const eqIdx = parts[i].indexOf("=");
|
|
95
|
+
if (eqIdx > 0) {
|
|
96
|
+
overrides[parts[i].slice(0, eqIdx)] = parts[i].slice(eqIdx + 1);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
return { defName, overrides };
|
|
100
|
+
}
|
|
101
|
+
|
|
41
102
|
async function handleCustomWorkflow(
|
|
42
103
|
sub: string,
|
|
43
104
|
ctx: ExtensionCommandContext,
|
|
@@ -62,15 +123,7 @@ async function handleCustomWorkflow(
|
|
|
62
123
|
ctx.ui.notify("Usage: /gsd workflow run <name> [param=value ...]", "warning");
|
|
63
124
|
return true;
|
|
64
125
|
}
|
|
65
|
-
const
|
|
66
|
-
const defName = parts[0];
|
|
67
|
-
const overrides: Record<string, string> = {};
|
|
68
|
-
for (let i = 1; i < parts.length; i++) {
|
|
69
|
-
const eqIdx = parts[i].indexOf("=");
|
|
70
|
-
if (eqIdx > 0) {
|
|
71
|
-
overrides[parts[i].slice(0, eqIdx)] = parts[i].slice(eqIdx + 1);
|
|
72
|
-
}
|
|
73
|
-
}
|
|
126
|
+
const { defName, overrides } = parseWorkflowRunArgs(args);
|
|
74
127
|
try {
|
|
75
128
|
const base = projectRoot();
|
|
76
129
|
const runDir = createRun(base, defName, Object.keys(overrides).length > 0 ? overrides : undefined);
|
|
@@ -168,6 +221,18 @@ async function handleCustomWorkflow(
|
|
|
168
221
|
}
|
|
169
222
|
|
|
170
223
|
export async function handleWorkflowCommand(trimmed: string, ctx: ExtensionCommandContext, pi: ExtensionAPI): Promise<boolean> {
|
|
224
|
+
// ── /gsd do — natural language routing (must be early to route to other commands) ──
|
|
225
|
+
if (trimmed === "do" || trimmed.startsWith("do ")) {
|
|
226
|
+
const { handleDo } = await import("../../commands-do.js");
|
|
227
|
+
await handleDo(trimmed.replace(/^do\s*/, "").trim(), ctx, pi);
|
|
228
|
+
return true;
|
|
229
|
+
}
|
|
230
|
+
// ── Backlog management ──
|
|
231
|
+
if (trimmed === "backlog" || trimmed.startsWith("backlog ")) {
|
|
232
|
+
const { handleBacklog } = await import("../../commands-backlog.js");
|
|
233
|
+
await handleBacklog(trimmed.replace(/^backlog\s*/, "").trim(), ctx, pi);
|
|
234
|
+
return true;
|
|
235
|
+
}
|
|
171
236
|
// ── Custom workflow commands (`/gsd workflow ...`) ──
|
|
172
237
|
if (trimmed === "workflow" || trimmed.startsWith("workflow ")) {
|
|
173
238
|
const sub = trimmed.slice("workflow".length).trim();
|
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* GSD Command — /gsd add-tests
|
|
3
|
+
*
|
|
4
|
+
* Generates tests for a completed slice by dispatching an LLM prompt
|
|
5
|
+
* with implementation context (summaries, changed files, test patterns).
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import type { ExtensionAPI, ExtensionCommandContext } from "@gsd/pi-coding-agent";
|
|
9
|
+
|
|
10
|
+
import { existsSync, readFileSync, readdirSync } from "node:fs";
|
|
11
|
+
import { join } from "node:path";
|
|
12
|
+
|
|
13
|
+
import { deriveState } from "./state.js";
|
|
14
|
+
import { gsdRoot, resolveSliceFile } from "./paths.js";
|
|
15
|
+
import { loadPrompt } from "./prompt-loader.js";
|
|
16
|
+
|
|
17
|
+
function findLastCompletedSlice(basePath: string, milestoneId: string): string | null {
|
|
18
|
+
// Scan disk for slices that have a SUMMARY.md (indicating completion)
|
|
19
|
+
const slicesDir = join(gsdRoot(basePath), "milestones", milestoneId, "slices");
|
|
20
|
+
if (!existsSync(slicesDir)) return null;
|
|
21
|
+
|
|
22
|
+
try {
|
|
23
|
+
const entries = readdirSync(slicesDir, { withFileTypes: true })
|
|
24
|
+
.filter((e) => e.isDirectory() && /^S\d+$/.test(e.name))
|
|
25
|
+
.sort((a, b) => b.name.localeCompare(a.name)); // reverse order — latest first
|
|
26
|
+
|
|
27
|
+
for (const entry of entries) {
|
|
28
|
+
const summaryPath = join(slicesDir, entry.name, `${entry.name}-SUMMARY.md`);
|
|
29
|
+
if (existsSync(summaryPath)) return entry.name;
|
|
30
|
+
}
|
|
31
|
+
} catch {
|
|
32
|
+
// non-fatal
|
|
33
|
+
}
|
|
34
|
+
return null;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
function readSliceSummary(basePath: string, milestoneId: string, sliceId: string): { title: string; content: string } {
|
|
38
|
+
const summaryPath = resolveSliceFile(basePath, milestoneId, sliceId, "SUMMARY");
|
|
39
|
+
if (summaryPath && existsSync(summaryPath)) {
|
|
40
|
+
const content = readFileSync(summaryPath, "utf-8");
|
|
41
|
+
const titleMatch = content.match(/^#\s+(.+)/m);
|
|
42
|
+
return { title: titleMatch?.[1] ?? sliceId, content };
|
|
43
|
+
}
|
|
44
|
+
return { title: sliceId, content: "(no summary available)" };
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
function detectTestPatterns(basePath: string): string {
|
|
48
|
+
const patterns: string[] = [];
|
|
49
|
+
|
|
50
|
+
// Check for common test configs
|
|
51
|
+
const checks = [
|
|
52
|
+
{ file: "jest.config.ts", name: "Jest" },
|
|
53
|
+
{ file: "jest.config.js", name: "Jest" },
|
|
54
|
+
{ file: "vitest.config.ts", name: "Vitest" },
|
|
55
|
+
{ file: "vitest.config.js", name: "Vitest" },
|
|
56
|
+
{ file: ".mocharc.yml", name: "Mocha" },
|
|
57
|
+
];
|
|
58
|
+
|
|
59
|
+
for (const check of checks) {
|
|
60
|
+
if (existsSync(join(basePath, check.file))) {
|
|
61
|
+
patterns.push(`Framework: ${check.name} (${check.file})`);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
// Look for existing test files to infer patterns
|
|
66
|
+
const testDirs = ["tests", "test", "src/__tests__", "__tests__"];
|
|
67
|
+
for (const dir of testDirs) {
|
|
68
|
+
const fullDir = join(basePath, dir);
|
|
69
|
+
if (existsSync(fullDir)) {
|
|
70
|
+
try {
|
|
71
|
+
const files = readdirSync(fullDir).filter((f) => f.endsWith(".test.ts") || f.endsWith(".spec.ts") || f.endsWith(".test.js"));
|
|
72
|
+
if (files.length > 0) {
|
|
73
|
+
patterns.push(`Test directory: ${dir}/ (${files.length} test files)`);
|
|
74
|
+
// Read first test file for patterns
|
|
75
|
+
const samplePath = join(fullDir, files[0]);
|
|
76
|
+
const sample = readFileSync(samplePath, "utf-8").slice(0, 500);
|
|
77
|
+
patterns.push(`Sample pattern from ${files[0]}:\n${sample}`);
|
|
78
|
+
break;
|
|
79
|
+
}
|
|
80
|
+
} catch {
|
|
81
|
+
// non-fatal
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
return patterns.length > 0 ? patterns.join("\n") : "No test framework detected. Use Node.js built-in test runner.";
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
export async function handleAddTests(
|
|
90
|
+
args: string,
|
|
91
|
+
ctx: ExtensionCommandContext,
|
|
92
|
+
pi: ExtensionAPI,
|
|
93
|
+
): Promise<void> {
|
|
94
|
+
const basePath = process.cwd();
|
|
95
|
+
const state = await deriveState(basePath);
|
|
96
|
+
|
|
97
|
+
if (!state.activeMilestone) {
|
|
98
|
+
ctx.ui.notify("No active milestone.", "warning");
|
|
99
|
+
return;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
const milestoneId = state.activeMilestone.id;
|
|
103
|
+
|
|
104
|
+
// Determine target
|
|
105
|
+
const targetId = args.trim() || findLastCompletedSlice(basePath, milestoneId);
|
|
106
|
+
if (!targetId) {
|
|
107
|
+
ctx.ui.notify(
|
|
108
|
+
"No completed slices found. Specify a slice ID: /gsd add-tests S03",
|
|
109
|
+
"warning",
|
|
110
|
+
);
|
|
111
|
+
return;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
// Gather context
|
|
115
|
+
const summary = readSliceSummary(basePath, milestoneId, targetId);
|
|
116
|
+
const testPatterns = detectTestPatterns(basePath);
|
|
117
|
+
|
|
118
|
+
ctx.ui.notify(`Generating tests for ${targetId}: "${summary.title}"...`, "info");
|
|
119
|
+
|
|
120
|
+
try {
|
|
121
|
+
const prompt = loadPrompt("add-tests", {
|
|
122
|
+
sliceId: targetId,
|
|
123
|
+
sliceTitle: summary.title,
|
|
124
|
+
sliceSummary: summary.content,
|
|
125
|
+
existingTestPatterns: testPatterns,
|
|
126
|
+
workingDirectory: basePath,
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
pi.sendMessage(
|
|
130
|
+
{ customType: "gsd-add-tests", content: prompt, display: false },
|
|
131
|
+
{ triggerTurn: true },
|
|
132
|
+
);
|
|
133
|
+
} catch (err) {
|
|
134
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
135
|
+
ctx.ui.notify(`Failed to dispatch test generation: ${msg}`, "error");
|
|
136
|
+
}
|
|
137
|
+
}
|
|
@@ -0,0 +1,182 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* GSD Command — /gsd backlog
|
|
3
|
+
*
|
|
4
|
+
* Structured backlog management with 999.x numbering.
|
|
5
|
+
* Items stored in .gsd/BACKLOG.md as markdown checklist.
|
|
6
|
+
* Items can be promoted to active slices via add-slice.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import type { ExtensionAPI, ExtensionCommandContext } from "@gsd/pi-coding-agent";
|
|
10
|
+
|
|
11
|
+
import { existsSync, readFileSync, writeFileSync, mkdirSync } from "node:fs";
|
|
12
|
+
import { join, dirname } from "node:path";
|
|
13
|
+
|
|
14
|
+
import { gsdRoot } from "./paths.js";
|
|
15
|
+
|
|
16
|
+
interface BacklogItem {
|
|
17
|
+
id: string;
|
|
18
|
+
title: string;
|
|
19
|
+
done: boolean;
|
|
20
|
+
note: string;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
function backlogPath(basePath: string): string {
|
|
24
|
+
return join(gsdRoot(basePath), "BACKLOG.md");
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
function parseBacklog(basePath: string): BacklogItem[] {
|
|
28
|
+
const filePath = backlogPath(basePath);
|
|
29
|
+
if (!existsSync(filePath)) return [];
|
|
30
|
+
|
|
31
|
+
const content = readFileSync(filePath, "utf-8");
|
|
32
|
+
const items: BacklogItem[] = [];
|
|
33
|
+
|
|
34
|
+
for (const line of content.split("\n")) {
|
|
35
|
+
const match = line.match(/^- \[([ x])\] (999\.\d+) — (.+?)(?:\s*\((.+)\))?$/);
|
|
36
|
+
if (match) {
|
|
37
|
+
items.push({
|
|
38
|
+
id: match[2],
|
|
39
|
+
title: match[3].trim(),
|
|
40
|
+
done: match[1] === "x",
|
|
41
|
+
note: match[4] ?? "",
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
return items;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
function writeBacklog(basePath: string, items: BacklogItem[]): void {
|
|
50
|
+
const filePath = backlogPath(basePath);
|
|
51
|
+
mkdirSync(dirname(filePath), { recursive: true });
|
|
52
|
+
const lines = ["# Backlog\n"];
|
|
53
|
+
for (const item of items) {
|
|
54
|
+
const check = item.done ? "x" : " ";
|
|
55
|
+
const note = item.note ? ` (${item.note})` : "";
|
|
56
|
+
lines.push(`- [${check}] ${item.id} — ${item.title}${note}`);
|
|
57
|
+
}
|
|
58
|
+
lines.push(""); // trailing newline
|
|
59
|
+
writeFileSync(filePath, lines.join("\n"), "utf-8");
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
function nextBacklogId(items: BacklogItem[]): string {
|
|
63
|
+
let maxNum = 0;
|
|
64
|
+
for (const item of items) {
|
|
65
|
+
const match = item.id.match(/^999\.(\d+)$/);
|
|
66
|
+
if (match) {
|
|
67
|
+
const num = parseInt(match[1], 10);
|
|
68
|
+
if (num > maxNum) maxNum = num;
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
return `999.${maxNum + 1}`;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
async function listBacklog(basePath: string, ctx: ExtensionCommandContext): Promise<void> {
|
|
75
|
+
const items = parseBacklog(basePath);
|
|
76
|
+
if (items.length === 0) {
|
|
77
|
+
ctx.ui.notify("Backlog is empty. Add items with /gsd backlog add <title>", "info");
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
const lines = ["Backlog:\n"];
|
|
82
|
+
for (const item of items) {
|
|
83
|
+
const status = item.done ? "✓" : "○";
|
|
84
|
+
const note = item.note ? ` (${item.note})` : "";
|
|
85
|
+
lines.push(` ${status} ${item.id} — ${item.title}${note}`);
|
|
86
|
+
}
|
|
87
|
+
const pending = items.filter((i) => !i.done).length;
|
|
88
|
+
lines.push(`\n${pending} pending, ${items.length - pending} promoted/done`);
|
|
89
|
+
ctx.ui.notify(lines.join("\n"), "info");
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
async function addBacklogItem(basePath: string, title: string, ctx: ExtensionCommandContext): Promise<void> {
|
|
93
|
+
if (!title) {
|
|
94
|
+
ctx.ui.notify("Usage: /gsd backlog add <title>", "warning");
|
|
95
|
+
return;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
const items = parseBacklog(basePath);
|
|
99
|
+
const id = nextBacklogId(items);
|
|
100
|
+
const date = new Date().toISOString().slice(0, 10);
|
|
101
|
+
|
|
102
|
+
items.push({ id, title: title.replace(/^['"]|['"]$/g, ""), done: false, note: `added ${date}` });
|
|
103
|
+
writeBacklog(basePath, items);
|
|
104
|
+
|
|
105
|
+
ctx.ui.notify(`Added ${id}: "${title}"`, "success");
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
async function promoteBacklogItem(
|
|
109
|
+
basePath: string,
|
|
110
|
+
itemId: string,
|
|
111
|
+
ctx: ExtensionCommandContext,
|
|
112
|
+
pi: ExtensionAPI,
|
|
113
|
+
): Promise<void> {
|
|
114
|
+
if (!itemId) {
|
|
115
|
+
ctx.ui.notify("Usage: /gsd backlog promote <id>\nExample: /gsd backlog promote 999.1", "warning");
|
|
116
|
+
return;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
const items = parseBacklog(basePath);
|
|
120
|
+
const item = items.find((i) => i.id === itemId);
|
|
121
|
+
|
|
122
|
+
if (!item) {
|
|
123
|
+
ctx.ui.notify(`Backlog item ${itemId} not found.`, "warning");
|
|
124
|
+
return;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
if (item.done) {
|
|
128
|
+
ctx.ui.notify(`${itemId} is already promoted/done.`, "info");
|
|
129
|
+
return;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
// Promote — currently requires single-writer engine (not yet available)
|
|
133
|
+
// Mark as promoted in backlog for now; slice creation will be available with the engine.
|
|
134
|
+
item.done = true;
|
|
135
|
+
item.note = `promoted ${new Date().toISOString().slice(0, 10)}`;
|
|
136
|
+
writeBacklog(basePath, items);
|
|
137
|
+
ctx.ui.notify(`Promoted ${itemId}: "${item.title}" — add it to the roadmap manually or wait for engine slice commands.`, "info");
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
async function removeBacklogItem(basePath: string, itemId: string, ctx: ExtensionCommandContext): Promise<void> {
|
|
141
|
+
if (!itemId) {
|
|
142
|
+
ctx.ui.notify("Usage: /gsd backlog remove <id>", "warning");
|
|
143
|
+
return;
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
const items = parseBacklog(basePath);
|
|
147
|
+
const idx = items.findIndex((i) => i.id === itemId);
|
|
148
|
+
|
|
149
|
+
if (idx === -1) {
|
|
150
|
+
ctx.ui.notify(`Backlog item ${itemId} not found.`, "warning");
|
|
151
|
+
return;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
const removed = items.splice(idx, 1)[0];
|
|
155
|
+
writeBacklog(basePath, items);
|
|
156
|
+
ctx.ui.notify(`Removed ${removed.id}: "${removed.title}"`, "success");
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
export async function handleBacklog(
|
|
160
|
+
args: string,
|
|
161
|
+
ctx: ExtensionCommandContext,
|
|
162
|
+
pi: ExtensionAPI,
|
|
163
|
+
): Promise<void> {
|
|
164
|
+
const basePath = process.cwd();
|
|
165
|
+
const parts = args.trim().split(/\s+/);
|
|
166
|
+
const sub = parts[0] ?? "";
|
|
167
|
+
const rest = parts.slice(1).join(" ");
|
|
168
|
+
|
|
169
|
+
switch (sub) {
|
|
170
|
+
case "":
|
|
171
|
+
return listBacklog(basePath, ctx);
|
|
172
|
+
case "add":
|
|
173
|
+
return addBacklogItem(basePath, rest, ctx);
|
|
174
|
+
case "promote":
|
|
175
|
+
return promoteBacklogItem(basePath, rest.trim(), ctx, pi);
|
|
176
|
+
case "remove":
|
|
177
|
+
return removeBacklogItem(basePath, rest.trim(), ctx);
|
|
178
|
+
default:
|
|
179
|
+
// Treat as implicit add
|
|
180
|
+
return addBacklogItem(basePath, args, ctx);
|
|
181
|
+
}
|
|
182
|
+
}
|