gsd-pi 2.74.0-dev.b741afb → 2.74.0-dev.ffbcc03
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 +113 -3
- 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 +51 -5
- 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.js +28 -2
- 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/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 +1 -1
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +14 -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 +24 -1
- 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 +1 -0
- package/dist/resources/extensions/gsd/preferences-validation.js +108 -2
- package/dist/resources/extensions/gsd/preferences.js +26 -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 +18 -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 +9 -9
- 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 +9 -9
- 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__/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 +47 -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/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__/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 +67 -0
- 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/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 +123 -3
- 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 +66 -5
- 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.ts +34 -2
- 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/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 +1 -1
- package/src/resources/extensions/gsd/docs/preferences-reference.md +14 -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 +33 -1
- 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 +32 -0
- package/src/resources/extensions/gsd/preferences-validation.ts +107 -2
- package/src/resources/extensions/gsd/preferences.ts +28 -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 +18 -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/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/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/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 → kn6xzWKYnogsxp2b6RpDD}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{XnHY5eXUsTCFmNodWHetD → kn6xzWKYnogsxp2b6RpDD}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,201 @@
|
|
|
1
|
+
// GSD2 — Ecosystem extension loader for ./.gsd/extensions/
|
|
2
|
+
// Discovers and registers single-file extensions that consume GSDExtensionAPI.
|
|
3
|
+
// Trust-gated (mirrors pi's `.pi/extensions/` model) and isolated from pi's
|
|
4
|
+
// own loader chain — handlers run in GSD's own dispatch step, not pi's.
|
|
5
|
+
|
|
6
|
+
import * as fs from "node:fs";
|
|
7
|
+
import * as path from "node:path";
|
|
8
|
+
import { pathToFileURL } from "node:url";
|
|
9
|
+
|
|
10
|
+
import type { ExtensionAPI } from "@gsd/pi-coding-agent";
|
|
11
|
+
import { getAgentDir } from "@gsd/pi-coding-agent";
|
|
12
|
+
|
|
13
|
+
import { logWarning } from "../workflow-logger.js";
|
|
14
|
+
import {
|
|
15
|
+
createGSDExtensionAPI,
|
|
16
|
+
type GSDEcosystemBeforeAgentStartHandler,
|
|
17
|
+
type GSDExtensionAPI,
|
|
18
|
+
} from "./gsd-extension-api.js";
|
|
19
|
+
|
|
20
|
+
// ─── Trust check (inlined; pi does not export isProjectTrusted from its
|
|
21
|
+
// package root, and constraint forbids modifying packages/pi-coding-agent/) ─
|
|
22
|
+
|
|
23
|
+
const TRUSTED_PROJECTS_FILE = "trusted-projects.json";
|
|
24
|
+
|
|
25
|
+
function isProjectTrusted(projectPath: string, agentDir: string): boolean {
|
|
26
|
+
const canonical = path.resolve(projectPath);
|
|
27
|
+
const trustedPath = path.join(agentDir, TRUSTED_PROJECTS_FILE);
|
|
28
|
+
try {
|
|
29
|
+
const content = fs.readFileSync(trustedPath, "utf-8");
|
|
30
|
+
const parsed = JSON.parse(content);
|
|
31
|
+
if (Array.isArray(parsed)) {
|
|
32
|
+
return parsed.includes(canonical);
|
|
33
|
+
}
|
|
34
|
+
} catch {
|
|
35
|
+
// missing or malformed — treat as untrusted
|
|
36
|
+
}
|
|
37
|
+
return false;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
// ─── Ready-promise singleton ────────────────────────────────────────────
|
|
41
|
+
|
|
42
|
+
let _readyPromise: Promise<void> | null = null;
|
|
43
|
+
let _untrustedWarned = false;
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Discover and register ecosystem extensions from `./.gsd/extensions/`.
|
|
47
|
+
* Idempotent: subsequent calls with the same arguments return the same
|
|
48
|
+
* pending promise (no double-load).
|
|
49
|
+
*/
|
|
50
|
+
export function loadEcosystemExtensions(
|
|
51
|
+
pi: ExtensionAPI,
|
|
52
|
+
sharedHandlers: GSDEcosystemBeforeAgentStartHandler[],
|
|
53
|
+
cwd: string = process.cwd(),
|
|
54
|
+
): Promise<void> {
|
|
55
|
+
if (_readyPromise) return _readyPromise;
|
|
56
|
+
_readyPromise = _loadEcosystemExtensionsImpl(pi, sharedHandlers, cwd);
|
|
57
|
+
return _readyPromise;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* Returns a promise that resolves when ecosystem loading has completed.
|
|
62
|
+
* If loading was never kicked off this returns a resolved promise so the
|
|
63
|
+
* `before_agent_start` handler can `await` unconditionally.
|
|
64
|
+
*/
|
|
65
|
+
export function getEcosystemReadyPromise(): Promise<void> {
|
|
66
|
+
return _readyPromise ?? Promise.resolve();
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
/** Test-only: clear the singleton so tests can re-run loading. */
|
|
70
|
+
export function _resetEcosystemLoader(): void {
|
|
71
|
+
_readyPromise = null;
|
|
72
|
+
_untrustedWarned = false;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
// ─── Implementation ─────────────────────────────────────────────────────
|
|
76
|
+
|
|
77
|
+
async function _loadEcosystemExtensionsImpl(
|
|
78
|
+
pi: ExtensionAPI,
|
|
79
|
+
sharedHandlers: GSDEcosystemBeforeAgentStartHandler[],
|
|
80
|
+
cwd: string,
|
|
81
|
+
): Promise<void> {
|
|
82
|
+
const extDir = path.join(cwd, ".gsd", "extensions");
|
|
83
|
+
if (!fs.existsSync(extDir)) return;
|
|
84
|
+
|
|
85
|
+
// Trust gate: refuse to load arbitrary code from untrusted project dirs.
|
|
86
|
+
if (!isProjectTrusted(cwd, getAgentDir())) {
|
|
87
|
+
if (!_untrustedWarned) {
|
|
88
|
+
_untrustedWarned = true;
|
|
89
|
+
logWarning(
|
|
90
|
+
"ecosystem",
|
|
91
|
+
".gsd/extensions present but project is not trusted — skipping ecosystem extensions. Run `pi trust` to opt in.",
|
|
92
|
+
);
|
|
93
|
+
}
|
|
94
|
+
return;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
// Resolve realpath ONCE so symlink-escape detection has a stable anchor.
|
|
98
|
+
let realExtDir: string;
|
|
99
|
+
try {
|
|
100
|
+
realExtDir = fs.realpathSync(extDir);
|
|
101
|
+
} catch (err) {
|
|
102
|
+
logWarning(
|
|
103
|
+
"ecosystem",
|
|
104
|
+
`failed to resolve extensions dir: ${err instanceof Error ? err.message : String(err)}`,
|
|
105
|
+
);
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
let entries: string[];
|
|
110
|
+
try {
|
|
111
|
+
entries = fs
|
|
112
|
+
.readdirSync(extDir)
|
|
113
|
+
.filter((f) => f.endsWith(".js") || f.endsWith(".ts"))
|
|
114
|
+
.sort(); // deterministic load order
|
|
115
|
+
} catch (err) {
|
|
116
|
+
logWarning(
|
|
117
|
+
"ecosystem",
|
|
118
|
+
`failed to read extensions dir: ${err instanceof Error ? err.message : String(err)}`,
|
|
119
|
+
);
|
|
120
|
+
return;
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
// The wrapper api is built once per loader run and shared by all extensions
|
|
124
|
+
// so they all read from the same module-level snapshot.
|
|
125
|
+
const api: GSDExtensionAPI = createGSDExtensionAPI(pi, sharedHandlers);
|
|
126
|
+
|
|
127
|
+
for (const entry of entries) {
|
|
128
|
+
await _loadOne(extDir, realExtDir, entry, api);
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
async function _loadOne(
|
|
133
|
+
extDir: string,
|
|
134
|
+
realExtDir: string,
|
|
135
|
+
entry: string,
|
|
136
|
+
api: GSDExtensionAPI,
|
|
137
|
+
): Promise<void> {
|
|
138
|
+
const fullPath = path.join(extDir, entry);
|
|
139
|
+
|
|
140
|
+
// Symlink-escape guard: reject entries whose realpath is not under realExtDir.
|
|
141
|
+
let realFullPath: string;
|
|
142
|
+
try {
|
|
143
|
+
realFullPath = fs.realpathSync(fullPath);
|
|
144
|
+
} catch (err) {
|
|
145
|
+
logWarning(
|
|
146
|
+
"ecosystem",
|
|
147
|
+
`failed to resolve ${entry}: ${err instanceof Error ? err.message : String(err)}`,
|
|
148
|
+
);
|
|
149
|
+
return;
|
|
150
|
+
}
|
|
151
|
+
const realExtDirWithSep = realExtDir.endsWith(path.sep) ? realExtDir : realExtDir + path.sep;
|
|
152
|
+
if (
|
|
153
|
+
realFullPath !== realExtDir &&
|
|
154
|
+
!realFullPath.startsWith(realExtDirWithSep)
|
|
155
|
+
) {
|
|
156
|
+
logWarning("ecosystem", `rejecting ${entry}: realpath escapes extensions dir`);
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
// For .ts files, require a sibling compiled .js — we do not run a TS loader
|
|
161
|
+
// in production. Drop mtime heuristics: if .js exists, prefer it; otherwise warn.
|
|
162
|
+
let importPath = realFullPath;
|
|
163
|
+
if (entry.endsWith(".ts")) {
|
|
164
|
+
const jsSibling = realFullPath.slice(0, -3) + ".js";
|
|
165
|
+
if (fs.existsSync(jsSibling)) {
|
|
166
|
+
importPath = jsSibling;
|
|
167
|
+
} else {
|
|
168
|
+
logWarning(
|
|
169
|
+
"ecosystem",
|
|
170
|
+
`${entry}: TypeScript source has no compiled .js sibling — compile it first`,
|
|
171
|
+
);
|
|
172
|
+
return;
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
let mod: any;
|
|
177
|
+
try {
|
|
178
|
+
mod = await import(pathToFileURL(importPath).href);
|
|
179
|
+
} catch (err) {
|
|
180
|
+
logWarning(
|
|
181
|
+
"ecosystem",
|
|
182
|
+
`failed to import ${entry}: ${err instanceof Error ? err.message : String(err)}`,
|
|
183
|
+
);
|
|
184
|
+
return;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
const factory = mod?.default;
|
|
188
|
+
if (typeof factory !== "function") {
|
|
189
|
+
logWarning("ecosystem", `${entry}: default export is not a function`);
|
|
190
|
+
return;
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
try {
|
|
194
|
+
await factory(api);
|
|
195
|
+
} catch (err) {
|
|
196
|
+
logWarning(
|
|
197
|
+
"ecosystem",
|
|
198
|
+
`factory threw for ${entry}: ${err instanceof Error ? err.message : String(err)}`,
|
|
199
|
+
);
|
|
200
|
+
}
|
|
201
|
+
}
|
|
@@ -34,6 +34,7 @@ import {
|
|
|
34
34
|
nativeAddPaths,
|
|
35
35
|
nativeResetSoft,
|
|
36
36
|
nativeCommitSubject,
|
|
37
|
+
_resetHasChangesCache,
|
|
37
38
|
} from "./native-git-bridge.js";
|
|
38
39
|
import { GSDError, GSD_MERGE_CONFLICT, GSD_GIT_ERROR } from "./errors.js";
|
|
39
40
|
import { getErrorMessage } from "./error-utils.js";
|
|
@@ -93,6 +94,17 @@ export interface CommitOptions {
|
|
|
93
94
|
allowEmpty?: boolean;
|
|
94
95
|
}
|
|
95
96
|
|
|
97
|
+
export type TurnGitActionMode = "commit" | "snapshot" | "status-only";
|
|
98
|
+
|
|
99
|
+
export interface TurnGitActionResult {
|
|
100
|
+
action: TurnGitActionMode;
|
|
101
|
+
status: "ok" | "failed";
|
|
102
|
+
commitMessage?: string;
|
|
103
|
+
snapshotLabel?: string;
|
|
104
|
+
dirty?: boolean;
|
|
105
|
+
error?: string;
|
|
106
|
+
}
|
|
107
|
+
|
|
96
108
|
// ─── Meaningful Commit Message Generation ───────────────────────────────────
|
|
97
109
|
|
|
98
110
|
/** Context for generating a meaningful commit message from task execution results. */
|
|
@@ -822,6 +834,62 @@ export function createGitService(basePath: string): GitServiceImpl {
|
|
|
822
834
|
return new GitServiceImpl(basePath, gitPrefs);
|
|
823
835
|
}
|
|
824
836
|
|
|
837
|
+
function buildTurnSnapshotLabel(unitType: string, unitId: string): string {
|
|
838
|
+
const raw = `${unitType}/${unitId}`.trim();
|
|
839
|
+
if (!raw) return "turn";
|
|
840
|
+
return raw
|
|
841
|
+
.replace(/[^a-zA-Z0-9._/-]/g, "-")
|
|
842
|
+
.replace(/\/{2,}/g, "/")
|
|
843
|
+
.replace(/-{2,}/g, "-")
|
|
844
|
+
.replace(/^[-/]+|[-/]+$/g, "") || "turn";
|
|
845
|
+
}
|
|
846
|
+
|
|
847
|
+
export function runTurnGitAction(args: {
|
|
848
|
+
basePath: string;
|
|
849
|
+
action: TurnGitActionMode;
|
|
850
|
+
unitType: string;
|
|
851
|
+
unitId: string;
|
|
852
|
+
taskContext?: TaskCommitContext;
|
|
853
|
+
}): TurnGitActionResult {
|
|
854
|
+
try {
|
|
855
|
+
// Force fresh working-tree status per turn; nativeHasChanges caches briefly.
|
|
856
|
+
_resetHasChangesCache();
|
|
857
|
+
if (args.action === "status-only") {
|
|
858
|
+
return {
|
|
859
|
+
action: args.action,
|
|
860
|
+
status: "ok",
|
|
861
|
+
dirty: nativeHasChanges(args.basePath),
|
|
862
|
+
};
|
|
863
|
+
}
|
|
864
|
+
|
|
865
|
+
const git = createGitService(args.basePath);
|
|
866
|
+
if (args.action === "snapshot") {
|
|
867
|
+
const label = buildTurnSnapshotLabel(args.unitType, args.unitId);
|
|
868
|
+
git.createSnapshot(label);
|
|
869
|
+
return {
|
|
870
|
+
action: args.action,
|
|
871
|
+
status: "ok",
|
|
872
|
+
snapshotLabel: label,
|
|
873
|
+
dirty: nativeHasChanges(args.basePath),
|
|
874
|
+
};
|
|
875
|
+
}
|
|
876
|
+
|
|
877
|
+
const commitMessage = git.autoCommit(args.unitType, args.unitId, [], args.taskContext) ?? undefined;
|
|
878
|
+
return {
|
|
879
|
+
action: args.action,
|
|
880
|
+
status: "ok",
|
|
881
|
+
commitMessage,
|
|
882
|
+
dirty: nativeHasChanges(args.basePath),
|
|
883
|
+
};
|
|
884
|
+
} catch (err) {
|
|
885
|
+
return {
|
|
886
|
+
action: args.action,
|
|
887
|
+
status: "failed",
|
|
888
|
+
error: getErrorMessage(err),
|
|
889
|
+
};
|
|
890
|
+
}
|
|
891
|
+
}
|
|
892
|
+
|
|
825
893
|
// ─── Commit Type Inference ─────────────────────────────────────────────────
|
|
826
894
|
|
|
827
895
|
/**
|
|
@@ -7,13 +7,144 @@
|
|
|
7
7
|
*/
|
|
8
8
|
|
|
9
9
|
import { logWarning } from "./workflow-logger.js";
|
|
10
|
-
import
|
|
10
|
+
import { readFileSync } from "node:fs";
|
|
11
|
+
import { join } from "node:path";
|
|
12
|
+
|
|
13
|
+
interface GraphNode {
|
|
14
|
+
id: string;
|
|
15
|
+
label: string;
|
|
16
|
+
type: string;
|
|
17
|
+
confidence: string;
|
|
18
|
+
description?: string;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
interface GraphEdge {
|
|
22
|
+
from: string;
|
|
23
|
+
to: string;
|
|
24
|
+
type: string;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
interface GraphQueryResult {
|
|
28
|
+
nodes: GraphNode[];
|
|
29
|
+
edges: GraphEdge[];
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
interface GraphStatusResult {
|
|
33
|
+
exists: boolean;
|
|
34
|
+
stale: boolean;
|
|
35
|
+
ageHours?: number;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
interface GraphApi {
|
|
39
|
+
graphQuery: (projectDir: string, term: string, budget?: number) => Promise<GraphQueryResult>;
|
|
40
|
+
graphStatus: (projectDir: string) => Promise<GraphStatusResult>;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
interface GraphFileShape {
|
|
44
|
+
nodes: GraphNode[];
|
|
45
|
+
edges: GraphEdge[];
|
|
46
|
+
builtAt?: string;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
let cachedGraphApi: GraphApi | null = null;
|
|
50
|
+
let resolvedGraphApi = false;
|
|
11
51
|
|
|
12
52
|
export interface GraphSubgraphOptions {
|
|
13
53
|
/** Budget in tokens passed to graphQuery (1 node ≈ 20 tokens, 1 edge ≈ 10 tokens) */
|
|
14
54
|
budget: number;
|
|
15
55
|
}
|
|
16
56
|
|
|
57
|
+
function readGraphFile(projectDir: string): GraphFileShape | null {
|
|
58
|
+
try {
|
|
59
|
+
const graphPath = join(projectDir, ".gsd", "graphs", "graph.json");
|
|
60
|
+
const raw = readFileSync(graphPath, "utf-8");
|
|
61
|
+
const parsed = JSON.parse(raw) as Partial<GraphFileShape>;
|
|
62
|
+
const nodes = Array.isArray(parsed.nodes) ? parsed.nodes : [];
|
|
63
|
+
const edges = Array.isArray(parsed.edges) ? parsed.edges : [];
|
|
64
|
+
return { nodes, edges, builtAt: typeof parsed.builtAt === "string" ? parsed.builtAt : undefined };
|
|
65
|
+
} catch {
|
|
66
|
+
return null;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
async function fallbackGraphQuery(projectDir: string, term: string, budget = 3000): Promise<GraphQueryResult> {
|
|
71
|
+
const graph = readGraphFile(projectDir);
|
|
72
|
+
if (!graph) return { nodes: [], edges: [] };
|
|
73
|
+
|
|
74
|
+
const needle = term.trim().toLowerCase();
|
|
75
|
+
const matches = graph.nodes.filter((node) => {
|
|
76
|
+
const hay = [node.id, node.label, node.description].filter(Boolean).join(" ").toLowerCase();
|
|
77
|
+
return hay.includes(needle);
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
const maxNodes = Math.max(1, Math.floor(Math.max(1, budget) / 20));
|
|
81
|
+
const selectedIds = new Set(matches.slice(0, maxNodes).map((node) => node.id));
|
|
82
|
+
const nodeById = new Map(graph.nodes.map((node) => [node.id, node] as const));
|
|
83
|
+
|
|
84
|
+
// Pull one-hop neighbors so relation context survives even when the term
|
|
85
|
+
// matches only one side of an edge.
|
|
86
|
+
for (const edge of graph.edges) {
|
|
87
|
+
if (selectedIds.size >= maxNodes) break;
|
|
88
|
+
const touchesSelection = selectedIds.has(edge.from) || selectedIds.has(edge.to);
|
|
89
|
+
if (!touchesSelection) continue;
|
|
90
|
+
if (selectedIds.has(edge.from) && !selectedIds.has(edge.to) && nodeById.has(edge.to)) {
|
|
91
|
+
selectedIds.add(edge.to);
|
|
92
|
+
} else if (selectedIds.has(edge.to) && !selectedIds.has(edge.from) && nodeById.has(edge.from)) {
|
|
93
|
+
selectedIds.add(edge.from);
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
const nodes = graph.nodes.filter((node) => selectedIds.has(node.id));
|
|
98
|
+
|
|
99
|
+
const remainingBudget = Math.max(0, budget - nodes.length * 20);
|
|
100
|
+
const maxEdges = Math.floor(remainingBudget / 10);
|
|
101
|
+
const edges = graph.edges
|
|
102
|
+
.filter((edge) => selectedIds.has(edge.from) && selectedIds.has(edge.to))
|
|
103
|
+
.slice(0, maxEdges);
|
|
104
|
+
|
|
105
|
+
return { nodes, edges };
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
async function fallbackGraphStatus(projectDir: string): Promise<GraphStatusResult> {
|
|
109
|
+
const graph = readGraphFile(projectDir);
|
|
110
|
+
if (!graph) return { exists: false, stale: false };
|
|
111
|
+
if (!graph.builtAt) return { exists: true, stale: false };
|
|
112
|
+
|
|
113
|
+
const builtAtMs = Date.parse(graph.builtAt);
|
|
114
|
+
if (!Number.isFinite(builtAtMs)) return { exists: true, stale: false };
|
|
115
|
+
|
|
116
|
+
const ageHours = (Date.now() - builtAtMs) / (1000 * 60 * 60);
|
|
117
|
+
return { exists: true, stale: ageHours > 24, ageHours };
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
function isGraphApi(mod: unknown): mod is GraphApi {
|
|
121
|
+
if (!mod || typeof mod !== "object") return false;
|
|
122
|
+
const candidate = mod as Record<string, unknown>;
|
|
123
|
+
return typeof candidate.graphQuery === "function" && typeof candidate.graphStatus === "function";
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
async function resolveGraphApi(): Promise<GraphApi> {
|
|
127
|
+
if (resolvedGraphApi && cachedGraphApi) return cachedGraphApi;
|
|
128
|
+
|
|
129
|
+
resolvedGraphApi = true;
|
|
130
|
+
try {
|
|
131
|
+
const imported = await import("@gsd-build/mcp-server");
|
|
132
|
+
if (isGraphApi(imported)) {
|
|
133
|
+
cachedGraphApi = imported;
|
|
134
|
+
return cachedGraphApi;
|
|
135
|
+
}
|
|
136
|
+
logWarning("prompt", "@gsd-build/mcp-server graph exports unavailable; using local graph fallback");
|
|
137
|
+
} catch {
|
|
138
|
+
// Fall back to local reader implementation.
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
cachedGraphApi = {
|
|
142
|
+
graphQuery: fallbackGraphQuery,
|
|
143
|
+
graphStatus: fallbackGraphStatus,
|
|
144
|
+
};
|
|
145
|
+
return cachedGraphApi;
|
|
146
|
+
}
|
|
147
|
+
|
|
17
148
|
/**
|
|
18
149
|
* Query the knowledge graph for nodes related to the given term and format
|
|
19
150
|
* the result as an inlined context block.
|
|
@@ -33,18 +164,14 @@ export async function inlineGraphSubgraph(
|
|
|
33
164
|
if (!term || !term.trim()) return null;
|
|
34
165
|
|
|
35
166
|
try {
|
|
36
|
-
const
|
|
37
|
-
|
|
38
|
-
graphStatus: (projectDir: string) => Promise<GraphStatusResult>;
|
|
39
|
-
};
|
|
40
|
-
|
|
41
|
-
const result = await graphQuery(projectDir, term, opts.budget);
|
|
167
|
+
const graphApi = await resolveGraphApi();
|
|
168
|
+
const result = await graphApi.graphQuery(projectDir, term, opts.budget);
|
|
42
169
|
if (result.nodes.length === 0) return null;
|
|
43
170
|
|
|
44
171
|
// Check staleness for annotation
|
|
45
172
|
let staleAnnotation = "";
|
|
46
173
|
try {
|
|
47
|
-
const status = await graphStatus(projectDir);
|
|
174
|
+
const status = await graphApi.graphStatus(projectDir);
|
|
48
175
|
if (status.exists && status.stale && status.ageHours !== undefined) {
|
|
49
176
|
const hours = Math.round(status.ageHours);
|
|
50
177
|
staleAnnotation = `\n> ⚠ Graph last built ${hours}h ago — context may be outdated`;
|
|
@@ -54,14 +181,14 @@ export async function inlineGraphSubgraph(
|
|
|
54
181
|
}
|
|
55
182
|
|
|
56
183
|
// Format nodes as a compact list
|
|
57
|
-
const nodeLines = result.nodes.map((
|
|
58
|
-
const desc =
|
|
59
|
-
return `- **${
|
|
184
|
+
const nodeLines = result.nodes.map((node) => {
|
|
185
|
+
const desc = node.description ? ` — ${node.description}` : "";
|
|
186
|
+
return `- **${node.label}** (\`${node.type}\`, ${node.confidence})${desc}`;
|
|
60
187
|
});
|
|
61
188
|
|
|
62
189
|
// Format edges as relations (only if present)
|
|
63
190
|
const edgeLines = result.edges.length > 0
|
|
64
|
-
? result.edges.map((
|
|
191
|
+
? result.edges.map((edge) => `- \`${edge.from}\` →[${edge.type}]→ \`${edge.to}\``)
|
|
65
192
|
: [];
|
|
66
193
|
|
|
67
194
|
const sections: string[] = [
|