gsd-pi 2.76.0-dev.b072ebb73 → 2.76.0-dev.fe143342a
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/mcp-server.d.ts +7 -0
- package/dist/mcp-server.js +35 -1
- package/dist/resource-loader.d.ts +1 -1
- package/dist/resource-loader.js +2 -8
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +66 -4
- package/dist/resources/extensions/gsd/auto/phases.js +4 -1
- package/dist/resources/extensions/gsd/auto/session.js +4 -0
- package/dist/resources/extensions/gsd/auto-model-selection.js +39 -13
- package/dist/resources/extensions/gsd/auto-start.js +39 -21
- package/dist/resources/extensions/gsd/auto.js +15 -12
- package/dist/resources/extensions/gsd/blocked-models.js +68 -0
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +76 -0
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +39 -9
- package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +93 -0
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +2 -0
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +35 -0
- package/dist/resources/extensions/gsd/compaction-snapshot.js +121 -0
- package/dist/resources/extensions/gsd/complexity-classifier.js +5 -3
- package/dist/resources/extensions/gsd/error-classifier.js +31 -3
- package/dist/resources/extensions/gsd/exec-history.js +120 -0
- package/dist/resources/extensions/gsd/exec-sandbox.js +258 -0
- package/dist/resources/extensions/gsd/gsd-db.js +62 -4
- package/dist/resources/extensions/gsd/init-wizard.js +15 -1
- package/dist/resources/extensions/gsd/key-manager.js +6 -0
- package/dist/resources/extensions/gsd/pre-execution-checks.js +13 -3
- package/dist/resources/extensions/gsd/preferences-types.js +9 -0
- package/dist/resources/extensions/gsd/preferences-validation.js +83 -0
- package/dist/resources/extensions/gsd/preferences.js +17 -17
- package/dist/resources/extensions/gsd/prompt-loader.js +22 -7
- package/dist/resources/extensions/gsd/safety/file-change-validator.js +1 -1
- package/dist/resources/extensions/gsd/tools/exec-search-tool.js +59 -0
- package/dist/resources/extensions/gsd/tools/exec-tool.js +126 -0
- package/dist/resources/extensions/gsd/tools/resume-tool.js +23 -0
- package/dist/resources/extensions/gsd/workflow-mcp.js +3 -0
- package/dist/resources/extensions/search-the-web/command-search-provider.js +5 -4
- package/dist/resources/extensions/search-the-web/native-search.js +45 -13
- package/dist/tsconfig.extensions.tsbuildinfo +1 -1
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +8 -8
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/required-server-files.json +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +8 -8
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/dist/web/standalone/server.js +1 -1
- package/package.json +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +64 -25
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/src/workflow-tools.test.ts +146 -1
- package/packages/mcp-server/src/workflow-tools.ts +84 -43
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-ai/dist/providers/openai-completions.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/openai-completions.js +60 -15
- package/packages/pi-ai/dist/providers/openai-completions.js.map +1 -1
- package/packages/pi-ai/dist/providers/think-tag-parser.d.ts +17 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.js +75 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.js.map +1 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.test.d.ts +2 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.test.js +41 -0
- package/packages/pi-ai/dist/providers/think-tag-parser.test.js.map +1 -0
- package/packages/pi-ai/src/providers/openai-completions.ts +57 -16
- package/packages/pi-ai/src/providers/think-tag-parser.test.ts +44 -0
- package/packages/pi-ai/src/providers/think-tag-parser.ts +94 -0
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-coding-agent/dist/core/model-discovery.d.ts +3 -1
- package/packages/pi-coding-agent/dist/core/model-discovery.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-discovery.js +92 -12
- package/packages/pi-coding-agent/dist/core/model-discovery.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-discovery.test.js +16 -1
- package/packages/pi-coding-agent/dist/core/model-discovery.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry-discovery.test.js +61 -1
- package/packages/pi-coding-agent/dist/core/model-registry-discovery.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts +5 -0
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.js +76 -10
- package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/redact-secrets.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.js +49 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.test.js +67 -0
- package/packages/pi-coding-agent/dist/core/redact-secrets.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/session-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/session-manager.js +9 -5
- package/packages/pi-coding-agent/dist/core/session-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/session-manager.test.js +25 -1
- package/packages/pi-coding-agent/dist/core/session-manager.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +5 -4
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.js +13 -7
- package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.d.ts +7 -6
- package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.js +29 -21
- package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.js.map +1 -1
- package/packages/pi-coding-agent/src/core/model-discovery.test.ts +19 -0
- package/packages/pi-coding-agent/src/core/model-discovery.ts +99 -12
- package/packages/pi-coding-agent/src/core/model-registry-discovery.test.ts +75 -0
- package/packages/pi-coding-agent/src/core/model-registry.ts +86 -10
- package/packages/pi-coding-agent/src/core/redact-secrets.test.ts +86 -0
- package/packages/pi-coding-agent/src/core/redact-secrets.ts +58 -0
- package/packages/pi-coding-agent/src/core/session-manager.test.ts +36 -1
- package/packages/pi-coding-agent/src/core/session-manager.ts +9 -5
- package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +6 -6
- package/packages/pi-coding-agent/src/modes/interactive/components/provider-manager.ts +16 -7
- package/packages/pi-coding-agent/src/modes/interactive/components/skill-invocation-message.ts +36 -22
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
- package/scripts/link-workspace-packages.cjs +1 -0
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +67 -4
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +137 -2
- package/src/resources/extensions/gsd/auto/loop-deps.ts +1 -0
- package/src/resources/extensions/gsd/auto/phases.ts +4 -0
- package/src/resources/extensions/gsd/auto/session.ts +7 -1
- package/src/resources/extensions/gsd/auto-model-selection.ts +50 -12
- package/src/resources/extensions/gsd/auto-start.ts +40 -22
- package/src/resources/extensions/gsd/auto.ts +15 -12
- package/src/resources/extensions/gsd/blocked-models.ts +98 -0
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +97 -0
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +40 -9
- package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +109 -0
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +2 -0
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +36 -0
- package/src/resources/extensions/gsd/compaction-snapshot.ts +165 -0
- package/src/resources/extensions/gsd/complexity-classifier.ts +5 -3
- package/src/resources/extensions/gsd/error-classifier.ts +36 -3
- package/src/resources/extensions/gsd/exec-history.ts +153 -0
- package/src/resources/extensions/gsd/exec-sandbox.ts +326 -0
- package/src/resources/extensions/gsd/gsd-db.ts +68 -4
- package/src/resources/extensions/gsd/init-wizard.ts +15 -1
- package/src/resources/extensions/gsd/key-manager.ts +6 -0
- package/src/resources/extensions/gsd/pre-execution-checks.ts +13 -3
- package/src/resources/extensions/gsd/preferences-types.ts +38 -0
- package/src/resources/extensions/gsd/preferences-validation.ts +79 -0
- package/src/resources/extensions/gsd/preferences.ts +17 -17
- package/src/resources/extensions/gsd/prompt-loader.ts +30 -7
- package/src/resources/extensions/gsd/safety/file-change-validator.ts +1 -1
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +12 -0
- package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +33 -3
- package/src/resources/extensions/gsd/tests/auto-thinking-restore.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/blocked-models.test.ts +98 -0
- package/src/resources/extensions/gsd/tests/compaction-snapshot.test.ts +123 -0
- package/src/resources/extensions/gsd/tests/complexity-classifier.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/exec-history.test.ts +124 -0
- package/src/resources/extensions/gsd/tests/exec-sandbox.test.ts +210 -0
- package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +20 -0
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +151 -0
- package/src/resources/extensions/gsd/tests/init-wizard.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/isolation-none-branch-guard.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/key-manager.test.ts +7 -0
- package/src/resources/extensions/gsd/tests/pre-exec-backtick-strip.test.ts +14 -0
- package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +19 -0
- package/src/resources/extensions/gsd/tests/preferences.test.ts +110 -0
- package/src/resources/extensions/gsd/tests/prompt-loader-extension-dir.test.ts +49 -0
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +91 -0
- package/src/resources/extensions/gsd/tests/save-gate-result-render.test.ts +95 -0
- package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +3 -1
- package/src/resources/extensions/gsd/tools/exec-search-tool.ts +81 -0
- package/src/resources/extensions/gsd/tools/exec-tool.ts +183 -0
- package/src/resources/extensions/gsd/tools/resume-tool.ts +40 -0
- package/src/resources/extensions/gsd/workflow-logger.ts +2 -1
- package/src/resources/extensions/gsd/workflow-mcp.ts +3 -0
- package/src/resources/extensions/search-the-web/command-search-provider.ts +5 -4
- package/src/resources/extensions/search-the-web/native-search.ts +48 -12
- /package/dist/web/standalone/.next/static/{pBwmOoye64ZrRp-_rf0v1 → n21VtX2hZlkpdEUO_nU4z}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{pBwmOoye64ZrRp-_rf0v1 → n21VtX2hZlkpdEUO_nU4z}/_ssgManifest.js +0 -0
|
@@ -660,6 +660,55 @@ export function validatePreferences(preferences) {
|
|
|
660
660
|
errors.push("context_management must be an object");
|
|
661
661
|
}
|
|
662
662
|
}
|
|
663
|
+
// ─── Context Mode (gsd_exec sandbox) ────────────────────────────────────
|
|
664
|
+
if (preferences.context_mode !== undefined) {
|
|
665
|
+
if (typeof preferences.context_mode === "object" && preferences.context_mode !== null) {
|
|
666
|
+
const cmode = preferences.context_mode;
|
|
667
|
+
const validCmode = {};
|
|
668
|
+
if (cmode.enabled !== undefined) {
|
|
669
|
+
if (typeof cmode.enabled === "boolean")
|
|
670
|
+
validCmode.enabled = cmode.enabled;
|
|
671
|
+
else
|
|
672
|
+
errors.push("context_mode.enabled must be a boolean");
|
|
673
|
+
}
|
|
674
|
+
if (cmode.exec_timeout_ms !== undefined) {
|
|
675
|
+
const t = cmode.exec_timeout_ms;
|
|
676
|
+
if (typeof t === "number" && t >= 1000 && t <= 600_000)
|
|
677
|
+
validCmode.exec_timeout_ms = Math.floor(t);
|
|
678
|
+
else
|
|
679
|
+
errors.push("context_mode.exec_timeout_ms must be a number between 1000 and 600000");
|
|
680
|
+
}
|
|
681
|
+
if (cmode.exec_stdout_cap_bytes !== undefined) {
|
|
682
|
+
const b = cmode.exec_stdout_cap_bytes;
|
|
683
|
+
if (typeof b === "number" && b >= 4096 && b <= 16_777_216)
|
|
684
|
+
validCmode.exec_stdout_cap_bytes = Math.floor(b);
|
|
685
|
+
else
|
|
686
|
+
errors.push("context_mode.exec_stdout_cap_bytes must be a number between 4096 and 16777216");
|
|
687
|
+
}
|
|
688
|
+
if (cmode.exec_digest_chars !== undefined) {
|
|
689
|
+
const c = cmode.exec_digest_chars;
|
|
690
|
+
if (typeof c === "number" && c >= 0 && c <= 4000)
|
|
691
|
+
validCmode.exec_digest_chars = Math.floor(c);
|
|
692
|
+
else
|
|
693
|
+
errors.push("context_mode.exec_digest_chars must be a number between 0 and 4000");
|
|
694
|
+
}
|
|
695
|
+
if (cmode.exec_env_allowlist !== undefined) {
|
|
696
|
+
if (Array.isArray(cmode.exec_env_allowlist) &&
|
|
697
|
+
cmode.exec_env_allowlist.every((v) => typeof v === "string" && /^[A-Z_][A-Z0-9_]*$/i.test(v))) {
|
|
698
|
+
validCmode.exec_env_allowlist = cmode.exec_env_allowlist;
|
|
699
|
+
}
|
|
700
|
+
else {
|
|
701
|
+
errors.push("context_mode.exec_env_allowlist must be an array of valid env var names");
|
|
702
|
+
}
|
|
703
|
+
}
|
|
704
|
+
if (Object.keys(validCmode).length > 0) {
|
|
705
|
+
validated.context_mode = validCmode;
|
|
706
|
+
}
|
|
707
|
+
}
|
|
708
|
+
else {
|
|
709
|
+
errors.push("context_mode must be an object");
|
|
710
|
+
}
|
|
711
|
+
}
|
|
663
712
|
// ─── Parallel Config ────────────────────────────────────────────────────
|
|
664
713
|
if (preferences.parallel && typeof preferences.parallel === "object") {
|
|
665
714
|
const p = preferences.parallel;
|
|
@@ -716,6 +765,40 @@ export function validatePreferences(preferences) {
|
|
|
716
765
|
validated.parallel = parallel;
|
|
717
766
|
}
|
|
718
767
|
}
|
|
768
|
+
// ─── Slice Parallel Config ───────────────────────────────────────────────
|
|
769
|
+
if (preferences.slice_parallel !== undefined) {
|
|
770
|
+
if (typeof preferences.slice_parallel === "object" && preferences.slice_parallel !== null) {
|
|
771
|
+
const sp = preferences.slice_parallel;
|
|
772
|
+
const validSp = {};
|
|
773
|
+
if (sp.enabled !== undefined) {
|
|
774
|
+
if (typeof sp.enabled === "boolean")
|
|
775
|
+
validSp.enabled = sp.enabled;
|
|
776
|
+
else
|
|
777
|
+
errors.push("slice_parallel.enabled must be a boolean");
|
|
778
|
+
}
|
|
779
|
+
if (sp.max_workers !== undefined) {
|
|
780
|
+
const maxWorkers = typeof sp.max_workers === "number" ? sp.max_workers : Number(sp.max_workers);
|
|
781
|
+
if (Number.isFinite(maxWorkers) && maxWorkers >= 1 && maxWorkers <= 8) {
|
|
782
|
+
validSp.max_workers = Math.floor(maxWorkers);
|
|
783
|
+
}
|
|
784
|
+
else {
|
|
785
|
+
errors.push("slice_parallel.max_workers must be a number between 1 and 8");
|
|
786
|
+
}
|
|
787
|
+
}
|
|
788
|
+
const knownSliceParallelKeys = new Set(["enabled", "max_workers"]);
|
|
789
|
+
for (const key of Object.keys(sp)) {
|
|
790
|
+
if (!knownSliceParallelKeys.has(key)) {
|
|
791
|
+
warnings.push(`unknown slice_parallel key "${key}" — ignored`);
|
|
792
|
+
}
|
|
793
|
+
}
|
|
794
|
+
if (Object.keys(validSp).length > 0) {
|
|
795
|
+
validated.slice_parallel = validSp;
|
|
796
|
+
}
|
|
797
|
+
}
|
|
798
|
+
else {
|
|
799
|
+
errors.push("slice_parallel must be an object");
|
|
800
|
+
}
|
|
801
|
+
}
|
|
719
802
|
// ─── Reactive Execution ─────────────────────────────────────────────────
|
|
720
803
|
if (preferences.reactive_execution !== undefined) {
|
|
721
804
|
if (typeof preferences.reactive_execution === "object" && preferences.reactive_execution !== null) {
|
|
@@ -26,12 +26,12 @@ export { resolveAllSkillReferences } from "./preferences-skills.js";
|
|
|
26
26
|
// These lived in preferences-skills.ts but imported loadEffectiveGSDPreferences
|
|
27
27
|
// back from this file, creating a circular dependency. Moved here since they
|
|
28
28
|
// are trivial wrappers over loadEffectiveGSDPreferences.
|
|
29
|
-
export function resolveSkillDiscoveryMode() {
|
|
30
|
-
const prefs = loadEffectiveGSDPreferences();
|
|
29
|
+
export function resolveSkillDiscoveryMode(basePath) {
|
|
30
|
+
const prefs = loadEffectiveGSDPreferences(basePath);
|
|
31
31
|
return prefs?.preferences.skill_discovery ?? "suggest";
|
|
32
32
|
}
|
|
33
|
-
export function resolveSkillStalenessDays() {
|
|
34
|
-
const prefs = loadEffectiveGSDPreferences();
|
|
33
|
+
export function resolveSkillStalenessDays(basePath) {
|
|
34
|
+
const prefs = loadEffectiveGSDPreferences(basePath);
|
|
35
35
|
return prefs?.preferences.skill_staleness_days ?? 60;
|
|
36
36
|
}
|
|
37
37
|
// ─── Re-exports: models ─────────────────────────────────────────────────────
|
|
@@ -46,16 +46,16 @@ function globalPreferencesPath() {
|
|
|
46
46
|
function legacyGlobalPreferencesPath() {
|
|
47
47
|
return join(homedir(), ".pi", "agent", "gsd-preferences.md");
|
|
48
48
|
}
|
|
49
|
-
function projectPreferencesPath() {
|
|
50
|
-
return join(gsdRoot(
|
|
49
|
+
function projectPreferencesPath(basePath = process.cwd()) {
|
|
50
|
+
return join(gsdRoot(basePath), "PREFERENCES.md");
|
|
51
51
|
}
|
|
52
52
|
// Legacy lowercase files can still exist in older projects. Keep them as a
|
|
53
53
|
// compatibility-only fallback, but route new reads/writes through PREFERENCES.md.
|
|
54
54
|
function legacyGlobalPreferencesPathLowercase() {
|
|
55
55
|
return join(gsdHome(), "preferences.md");
|
|
56
56
|
}
|
|
57
|
-
function legacyProjectPreferencesPathLowercase() {
|
|
58
|
-
return join(gsdRoot(
|
|
57
|
+
function legacyProjectPreferencesPathLowercase(basePath = process.cwd()) {
|
|
58
|
+
return join(gsdRoot(basePath), "preferences.md");
|
|
59
59
|
}
|
|
60
60
|
export function getGlobalGSDPreferencesPath() {
|
|
61
61
|
return globalPreferencesPath();
|
|
@@ -63,8 +63,8 @@ export function getGlobalGSDPreferencesPath() {
|
|
|
63
63
|
export function getLegacyGlobalGSDPreferencesPath() {
|
|
64
64
|
return legacyGlobalPreferencesPath();
|
|
65
65
|
}
|
|
66
|
-
export function getProjectGSDPreferencesPath() {
|
|
67
|
-
return projectPreferencesPath();
|
|
66
|
+
export function getProjectGSDPreferencesPath(basePath) {
|
|
67
|
+
return projectPreferencesPath(basePath);
|
|
68
68
|
}
|
|
69
69
|
// ─── Loading ────────────────────────────────────────────────────────────────
|
|
70
70
|
export function loadGlobalGSDPreferences() {
|
|
@@ -72,13 +72,13 @@ export function loadGlobalGSDPreferences() {
|
|
|
72
72
|
?? loadPreferencesFile(legacyGlobalPreferencesPathLowercase(), "global")
|
|
73
73
|
?? loadPreferencesFile(legacyGlobalPreferencesPath(), "global");
|
|
74
74
|
}
|
|
75
|
-
export function loadProjectGSDPreferences() {
|
|
76
|
-
return loadPreferencesFile(projectPreferencesPath(), "project")
|
|
77
|
-
?? loadPreferencesFile(legacyProjectPreferencesPathLowercase(), "project");
|
|
75
|
+
export function loadProjectGSDPreferences(basePath) {
|
|
76
|
+
return loadPreferencesFile(projectPreferencesPath(basePath), "project")
|
|
77
|
+
?? loadPreferencesFile(legacyProjectPreferencesPathLowercase(basePath), "project");
|
|
78
78
|
}
|
|
79
|
-
export function loadEffectiveGSDPreferences() {
|
|
79
|
+
export function loadEffectiveGSDPreferences(basePath) {
|
|
80
80
|
const globalPreferences = loadGlobalGSDPreferences();
|
|
81
|
-
const projectPreferences = loadProjectGSDPreferences();
|
|
81
|
+
const projectPreferences = loadProjectGSDPreferences(basePath);
|
|
82
82
|
if (!globalPreferences && !projectPreferences)
|
|
83
83
|
return null;
|
|
84
84
|
let result;
|
|
@@ -489,8 +489,8 @@ export function resolvePreDispatchHooks() {
|
|
|
489
489
|
* Worktree isolation requires explicit opt-in because it depends on git
|
|
490
490
|
* branch infrastructure that must be set up before use.
|
|
491
491
|
*/
|
|
492
|
-
export function getIsolationMode() {
|
|
493
|
-
const prefs = loadEffectiveGSDPreferences()?.preferences?.git;
|
|
492
|
+
export function getIsolationMode(basePath) {
|
|
493
|
+
const prefs = loadEffectiveGSDPreferences(basePath)?.preferences?.git;
|
|
494
494
|
if (prefs?.isolation === "worktree")
|
|
495
495
|
return "worktree";
|
|
496
496
|
if (prefs?.isolation === "branch")
|
|
@@ -22,6 +22,27 @@ import { join, dirname } from "node:path";
|
|
|
22
22
|
import { fileURLToPath } from "node:url";
|
|
23
23
|
import { homedir } from "node:os";
|
|
24
24
|
import { logWarning } from "./workflow-logger.js";
|
|
25
|
+
function hasRequiredExtensionAssets(rootDir, exists = existsSync) {
|
|
26
|
+
return (exists(join(rootDir, "prompts")) &&
|
|
27
|
+
exists(join(rootDir, "templates", "task-summary.md")));
|
|
28
|
+
}
|
|
29
|
+
export function resolveExtensionDirFromCandidates(moduleDir, agentGsdDir, exists = existsSync) {
|
|
30
|
+
const moduleUsable = hasRequiredExtensionAssets(moduleDir, exists);
|
|
31
|
+
const agentUsable = hasRequiredExtensionAssets(agentGsdDir, exists);
|
|
32
|
+
// Prefer the user-local extension tree when both are valid. This avoids
|
|
33
|
+
// leaking npm/global-install paths into prompts on Windows.
|
|
34
|
+
if (agentUsable)
|
|
35
|
+
return agentGsdDir;
|
|
36
|
+
if (moduleUsable)
|
|
37
|
+
return moduleDir;
|
|
38
|
+
// Degraded fallback: if required template is missing in both locations,
|
|
39
|
+
// keep previous behavior and prefer whichever still has prompts/.
|
|
40
|
+
if (exists(join(moduleDir, "prompts")))
|
|
41
|
+
return moduleDir;
|
|
42
|
+
if (exists(join(agentGsdDir, "prompts")))
|
|
43
|
+
return agentGsdDir;
|
|
44
|
+
return moduleDir;
|
|
45
|
+
}
|
|
25
46
|
/**
|
|
26
47
|
* Resolve the GSD extension directory.
|
|
27
48
|
*
|
|
@@ -34,15 +55,9 @@ import { logWarning } from "./workflow-logger.js";
|
|
|
34
55
|
*/
|
|
35
56
|
function resolveExtensionDir() {
|
|
36
57
|
const moduleDir = dirname(fileURLToPath(import.meta.url));
|
|
37
|
-
if (existsSync(join(moduleDir, "prompts")))
|
|
38
|
-
return moduleDir;
|
|
39
|
-
// Fallback: user-local agent directory
|
|
40
58
|
const gsdHome = process.env.GSD_HOME || join(homedir(), ".gsd");
|
|
41
59
|
const agentGsdDir = join(gsdHome, "agent", "extensions", "gsd");
|
|
42
|
-
|
|
43
|
-
return agentGsdDir;
|
|
44
|
-
// Last resort: return the module dir (warmCache will silently handle the miss)
|
|
45
|
-
return moduleDir;
|
|
60
|
+
return resolveExtensionDirFromCandidates(moduleDir, agentGsdDir);
|
|
46
61
|
}
|
|
47
62
|
const __extensionDir = resolveExtensionDir();
|
|
48
63
|
const promptsDir = join(__extensionDir, "prompts");
|
|
@@ -62,7 +62,7 @@ export function validateFileChanges(basePath, expectedOutput, plannedFiles) {
|
|
|
62
62
|
// ─── Internals ──────────────────────────────────────────────────────────────
|
|
63
63
|
function getChangedFilesFromLastCommit(basePath) {
|
|
64
64
|
try {
|
|
65
|
-
const result = execFileSync("git", ["diff", "--
|
|
65
|
+
const result = execFileSync("git", ["diff-tree", "--root", "--no-commit-id", "-r", "--name-only", "HEAD"], { cwd: basePath, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8" }).trim();
|
|
66
66
|
return result ? result.split("\n").filter(Boolean) : [];
|
|
67
67
|
}
|
|
68
68
|
catch (e) {
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
// GSD Exec Search Tool — lists and filters prior gsd_exec runs.
|
|
2
|
+
//
|
|
3
|
+
// Scans .gsd/exec/*.meta.json and returns a ranked summary so agents can
|
|
4
|
+
// re-discover past runs without re-executing. Read-only; no DB writes.
|
|
5
|
+
import { searchExecHistory } from "../exec-history.js";
|
|
6
|
+
export function executeExecSearch(params, opts) {
|
|
7
|
+
const searchOpts = {
|
|
8
|
+
query: typeof params.query === "string" ? params.query : undefined,
|
|
9
|
+
runtime: params.runtime,
|
|
10
|
+
failing_only: params.failing_only === true,
|
|
11
|
+
limit: typeof params.limit === "number" ? params.limit : undefined,
|
|
12
|
+
};
|
|
13
|
+
const hits = searchExecHistory(opts.baseDir, searchOpts);
|
|
14
|
+
if (hits.length === 0) {
|
|
15
|
+
return {
|
|
16
|
+
content: [{ type: "text", text: "No prior gsd_exec runs match those filters." }],
|
|
17
|
+
details: { operation: "gsd_exec_search", matches: 0 },
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
const lines = [`Found ${hits.length} exec run(s), most recent first:`];
|
|
21
|
+
for (const hit of hits) {
|
|
22
|
+
const e = hit.entry;
|
|
23
|
+
const status = formatStatus(e);
|
|
24
|
+
const purpose = e.purpose ? ` — ${e.purpose}` : "";
|
|
25
|
+
const truncated = e.stdout_truncated ? " (stdout truncated)" : "";
|
|
26
|
+
lines.push(`- [${e.id}] ${e.runtime} ${status} ${e.duration_ms}ms${truncated}${purpose}`, ` stdout: ${e.stdout_path}`);
|
|
27
|
+
if (hit.digest_preview) {
|
|
28
|
+
const preview = hit.digest_preview.replace(/\n/g, "\n ");
|
|
29
|
+
lines.push(` preview:\n ${preview}`);
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
return {
|
|
33
|
+
content: [{ type: "text", text: lines.join("\n") }],
|
|
34
|
+
details: {
|
|
35
|
+
operation: "gsd_exec_search",
|
|
36
|
+
matches: hits.length,
|
|
37
|
+
results: hits.map((hit) => ({
|
|
38
|
+
id: hit.entry.id,
|
|
39
|
+
runtime: hit.entry.runtime,
|
|
40
|
+
exit_code: hit.entry.exit_code,
|
|
41
|
+
timed_out: hit.entry.timed_out,
|
|
42
|
+
duration_ms: hit.entry.duration_ms,
|
|
43
|
+
purpose: hit.entry.purpose,
|
|
44
|
+
stdout_path: hit.entry.stdout_path,
|
|
45
|
+
stderr_path: hit.entry.stderr_path,
|
|
46
|
+
meta_path: hit.entry.meta_path,
|
|
47
|
+
})),
|
|
48
|
+
},
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
function formatStatus(entry) {
|
|
52
|
+
if (entry.timed_out)
|
|
53
|
+
return "timeout";
|
|
54
|
+
if (entry.signal)
|
|
55
|
+
return `signal:${entry.signal}`;
|
|
56
|
+
if (entry.exit_code === null)
|
|
57
|
+
return "exit:null";
|
|
58
|
+
return `exit:${entry.exit_code}`;
|
|
59
|
+
}
|
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
// GSD Exec Tool — executor for the gsd_exec MCP tool.
|
|
2
|
+
//
|
|
3
|
+
// Thin wrapper around exec-sandbox.ts that reads effective options from
|
|
4
|
+
// the project preferences (context_mode block) and formats the result
|
|
5
|
+
// for MCP return.
|
|
6
|
+
import { EXEC_DEFAULTS, runExecSandbox, } from "../exec-sandbox.js";
|
|
7
|
+
import { isContextModeEnabled } from "../preferences-types.js";
|
|
8
|
+
export function buildExecOptions(baseDir, cfg, extras) {
|
|
9
|
+
const allowlist = Array.isArray(cfg?.exec_env_allowlist) ? cfg.exec_env_allowlist : EXEC_DEFAULTS.envAllowlist;
|
|
10
|
+
const stdoutCap = clampNumber(cfg?.exec_stdout_cap_bytes, EXEC_DEFAULTS.stdoutCapBytes, 4_096, 16_777_216);
|
|
11
|
+
const defaultTimeout = clampNumber(cfg?.exec_timeout_ms, EXEC_DEFAULTS.defaultTimeoutMs, 1_000, EXEC_DEFAULTS.clampTimeoutMs);
|
|
12
|
+
const digestChars = clampNumber(cfg?.exec_digest_chars, EXEC_DEFAULTS.digestChars, 0, 4_000);
|
|
13
|
+
return {
|
|
14
|
+
baseDir,
|
|
15
|
+
clamp_timeout_ms: EXEC_DEFAULTS.clampTimeoutMs,
|
|
16
|
+
default_timeout_ms: defaultTimeout,
|
|
17
|
+
stdout_cap_bytes: stdoutCap,
|
|
18
|
+
stderr_cap_bytes: EXEC_DEFAULTS.stderrCapBytes,
|
|
19
|
+
digest_chars: digestChars,
|
|
20
|
+
env_allowlist: allowlist,
|
|
21
|
+
...extras,
|
|
22
|
+
};
|
|
23
|
+
}
|
|
24
|
+
function clampNumber(value, fallback, min, max) {
|
|
25
|
+
if (typeof value !== "number" || !Number.isFinite(value))
|
|
26
|
+
return fallback;
|
|
27
|
+
if (value < min)
|
|
28
|
+
return min;
|
|
29
|
+
if (value > max)
|
|
30
|
+
return max;
|
|
31
|
+
return Math.floor(value);
|
|
32
|
+
}
|
|
33
|
+
function isEnabled(prefs) {
|
|
34
|
+
return isContextModeEnabled(prefs);
|
|
35
|
+
}
|
|
36
|
+
function disabledResult() {
|
|
37
|
+
return {
|
|
38
|
+
content: [
|
|
39
|
+
{
|
|
40
|
+
type: "text",
|
|
41
|
+
text: "gsd_exec is disabled by `context_mode.enabled: false` in preferences. Remove that " +
|
|
42
|
+
"override (or set it to true) to re-enable sandboxed tool-output execution.",
|
|
43
|
+
},
|
|
44
|
+
],
|
|
45
|
+
details: { operation: "gsd_exec", error: "context_mode_disabled" },
|
|
46
|
+
isError: true,
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
function paramError(message) {
|
|
50
|
+
return {
|
|
51
|
+
content: [{ type: "text", text: `Error: ${message}` }],
|
|
52
|
+
details: { operation: "gsd_exec", error: "invalid_params", detail: message },
|
|
53
|
+
isError: true,
|
|
54
|
+
};
|
|
55
|
+
}
|
|
56
|
+
export async function executeGsdExec(params, deps) {
|
|
57
|
+
if (!isEnabled(deps.preferences))
|
|
58
|
+
return disabledResult();
|
|
59
|
+
const runtime = params.runtime;
|
|
60
|
+
if (runtime !== "bash" && runtime !== "node" && runtime !== "python") {
|
|
61
|
+
return paramError(`invalid runtime "${String(runtime)}" — must be bash | node | python`);
|
|
62
|
+
}
|
|
63
|
+
const script = typeof params.script === "string" ? params.script : "";
|
|
64
|
+
if (script.trim().length === 0) {
|
|
65
|
+
return paramError("script is required and must be a non-empty string");
|
|
66
|
+
}
|
|
67
|
+
if (Buffer.byteLength(script, "utf8") > 200_000) {
|
|
68
|
+
return paramError("script exceeds the 200 KB length limit");
|
|
69
|
+
}
|
|
70
|
+
const opts = buildExecOptions(deps.baseDir, deps.preferences?.context_mode, { now: deps.now, generateId: deps.generateId });
|
|
71
|
+
const run = deps.run ?? runExecSandbox;
|
|
72
|
+
try {
|
|
73
|
+
const result = await run({
|
|
74
|
+
runtime,
|
|
75
|
+
script,
|
|
76
|
+
...(typeof params.purpose === "string" ? { purpose: params.purpose } : {}),
|
|
77
|
+
...(typeof params.timeout_ms === "number" ? { timeout_ms: params.timeout_ms } : {}),
|
|
78
|
+
}, opts);
|
|
79
|
+
return formatResult(result);
|
|
80
|
+
}
|
|
81
|
+
catch (err) {
|
|
82
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
83
|
+
return {
|
|
84
|
+
content: [{ type: "text", text: `Error: gsd_exec failed — ${message}` }],
|
|
85
|
+
details: { operation: "gsd_exec", error: message },
|
|
86
|
+
isError: true,
|
|
87
|
+
};
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
function formatResult(result) {
|
|
91
|
+
const headerLines = [
|
|
92
|
+
`gsd_exec[${result.id}] runtime=${result.runtime} exit=${formatExit(result)} duration=${result.duration_ms}ms`,
|
|
93
|
+
` stdout: ${result.stdout_bytes}B${result.stdout_truncated ? " (truncated)" : ""} → ${result.stdout_path}`,
|
|
94
|
+
` stderr: ${result.stderr_bytes}B${result.stderr_truncated ? " (truncated)" : ""} → ${result.stderr_path}`,
|
|
95
|
+
];
|
|
96
|
+
const summary = `${headerLines.join("\n")}\n--- digest ---\n${result.digest}`.trimEnd();
|
|
97
|
+
return {
|
|
98
|
+
content: [{ type: "text", text: summary }],
|
|
99
|
+
details: {
|
|
100
|
+
operation: "gsd_exec",
|
|
101
|
+
id: result.id,
|
|
102
|
+
runtime: result.runtime,
|
|
103
|
+
exit_code: result.exit_code,
|
|
104
|
+
signal: result.signal,
|
|
105
|
+
timed_out: result.timed_out,
|
|
106
|
+
duration_ms: result.duration_ms,
|
|
107
|
+
stdout_bytes: result.stdout_bytes,
|
|
108
|
+
stderr_bytes: result.stderr_bytes,
|
|
109
|
+
stdout_truncated: result.stdout_truncated,
|
|
110
|
+
stderr_truncated: result.stderr_truncated,
|
|
111
|
+
stdout_path: result.stdout_path,
|
|
112
|
+
stderr_path: result.stderr_path,
|
|
113
|
+
meta_path: result.meta_path,
|
|
114
|
+
},
|
|
115
|
+
isError: result.timed_out || result.signal !== null || result.exit_code !== 0,
|
|
116
|
+
};
|
|
117
|
+
}
|
|
118
|
+
function formatExit(result) {
|
|
119
|
+
if (result.timed_out)
|
|
120
|
+
return "timeout";
|
|
121
|
+
if (result.signal)
|
|
122
|
+
return `signal:${result.signal}`;
|
|
123
|
+
if (result.exit_code === null)
|
|
124
|
+
return "null";
|
|
125
|
+
return String(result.exit_code);
|
|
126
|
+
}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
// GSD Resume Tool — returns the contents of .gsd/last-snapshot.md so
|
|
2
|
+
// agents can re-orient after compaction or session resume without
|
|
3
|
+
// re-deriving project memory state.
|
|
4
|
+
import { readCompactionSnapshot } from "../compaction-snapshot.js";
|
|
5
|
+
export function executeResume(_params, opts) {
|
|
6
|
+
const snapshot = readCompactionSnapshot(opts.baseDir);
|
|
7
|
+
if (snapshot == null) {
|
|
8
|
+
return {
|
|
9
|
+
content: [
|
|
10
|
+
{
|
|
11
|
+
type: "text",
|
|
12
|
+
text: "No snapshot found at .gsd/last-snapshot.md. The snapshot is written automatically " +
|
|
13
|
+
"on session_before_compact (enabled by default; set context_mode.enabled=false to opt out).",
|
|
14
|
+
},
|
|
15
|
+
],
|
|
16
|
+
details: { operation: "gsd_resume", found: false },
|
|
17
|
+
};
|
|
18
|
+
}
|
|
19
|
+
return {
|
|
20
|
+
content: [{ type: "text", text: snapshot }],
|
|
21
|
+
details: { operation: "gsd_resume", found: true, bytes: Buffer.byteLength(snapshot, "utf-8") },
|
|
22
|
+
};
|
|
23
|
+
}
|
|
@@ -5,6 +5,9 @@ import { fileURLToPath, pathToFileURL } from "node:url";
|
|
|
5
5
|
const MCP_WORKFLOW_TOOL_SURFACE = new Set([
|
|
6
6
|
"ask_user_questions",
|
|
7
7
|
"gsd_decision_save",
|
|
8
|
+
"gsd_exec",
|
|
9
|
+
"gsd_exec_search",
|
|
10
|
+
"gsd_resume",
|
|
8
11
|
"gsd_complete_milestone",
|
|
9
12
|
"gsd_complete_task",
|
|
10
13
|
"gsd_complete_slice",
|
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
*
|
|
8
8
|
* All provider logic lives in provider.ts (S01) — this is pure UI wiring.
|
|
9
9
|
*/
|
|
10
|
-
import {
|
|
10
|
+
import { supportsNativeWebSearch } from './native-search.js';
|
|
11
11
|
import { getTavilyApiKey, getBraveApiKey, getOllamaApiKey, getSearchProviderPreference, setSearchProviderPreference, resolveSearchProvider, } from './provider.js';
|
|
12
12
|
const VALID_PREFERENCES = ['tavily', 'brave', 'ollama', 'auto'];
|
|
13
13
|
function keyStatus(provider) {
|
|
@@ -72,9 +72,10 @@ export function registerSearchProviderCommand(pi) {
|
|
|
72
72
|
}
|
|
73
73
|
setSearchProviderPreference(chosen);
|
|
74
74
|
const effective = resolveSearchProvider();
|
|
75
|
-
// Gate on api
|
|
76
|
-
//
|
|
77
|
-
|
|
75
|
+
// Gate on api shape + provider allowlist: the info note must match the
|
|
76
|
+
// actual runtime behavior in native-search.ts. Claude served via copilot
|
|
77
|
+
// / minimax / kimi is anthropic-shaped but does NOT run native search.
|
|
78
|
+
const isAnthropic = supportsNativeWebSearch(ctx.model);
|
|
78
79
|
const nativeNote = isAnthropic ? '\nNote: Native Anthropic web search is also active (automatic, no API key needed).' : '';
|
|
79
80
|
ctx.ui.notify(`Search provider set to ${chosen}. Effective provider: ${effective ?? 'none (no API keys)'}${nativeNote}`, 'info');
|
|
80
81
|
},
|
|
@@ -12,6 +12,38 @@ export const BRAVE_TOOL_NAMES = ["search-the-web", "search_and_read"];
|
|
|
12
12
|
export const CUSTOM_SEARCH_TOOL_NAMES = ["search-the-web", "search_and_read", "google_search"];
|
|
13
13
|
/** Thinking block types that require signature validation by the API */
|
|
14
14
|
const THINKING_TYPES = new Set(["thinking", "redacted_thinking"]);
|
|
15
|
+
/**
|
|
16
|
+
* Providers whose Anthropic-Messages endpoint is known to accept the native
|
|
17
|
+
* `web_search_20250305` server tool. Anthropic-shaped transports NOT in this
|
|
18
|
+
* set (github-copilot, minimax, kimi-coding, opencode, vercel-ai-gateway,
|
|
19
|
+
* etc.) route Claude or Claude-compatible models through the Messages API
|
|
20
|
+
* but do NOT expose the server-side search tool — injecting it yields a
|
|
21
|
+
* 400 "unsupported_value" from their endpoints (regression from #4492).
|
|
22
|
+
*
|
|
23
|
+
* Keep this allowlist tight — err on the side of custom/Brave search rather
|
|
24
|
+
* than a runtime 400. Add a provider here only after confirming its endpoint
|
|
25
|
+
* accepts the tool type.
|
|
26
|
+
*/
|
|
27
|
+
const NATIVE_WEB_SEARCH_PROVIDERS = new Set([
|
|
28
|
+
"anthropic",
|
|
29
|
+
"claude-code",
|
|
30
|
+
"anthropic-vertex",
|
|
31
|
+
"vercel-ai-gateway",
|
|
32
|
+
]);
|
|
33
|
+
/**
|
|
34
|
+
* True when the model is an Anthropic-shaped transport AND the provider is
|
|
35
|
+
* known to accept the native `web_search_20250305` tool. Gate both on api
|
|
36
|
+
* shape (#4478 / ADR-012) and on provider identity (#444 regression guard
|
|
37
|
+
* and #4492 scope correction) — provider-level discrimination is legitimate
|
|
38
|
+
* per ADR-012 for credential/behavior differences that api shape can't
|
|
39
|
+
* express.
|
|
40
|
+
*/
|
|
41
|
+
export function supportsNativeWebSearch(model) {
|
|
42
|
+
if (!isAnthropicApi(model))
|
|
43
|
+
return false;
|
|
44
|
+
const provider = model?.provider;
|
|
45
|
+
return typeof provider === "string" && NATIVE_WEB_SEARCH_PROVIDERS.has(provider);
|
|
46
|
+
}
|
|
15
47
|
/**
|
|
16
48
|
* Maximum number of native web searches allowed per session (agent unit).
|
|
17
49
|
* The Anthropic API's `max_uses` is per-request — it resets on each API call.
|
|
@@ -76,10 +108,11 @@ export function registerNativeSearchHooks(pi) {
|
|
|
76
108
|
pi.on("model_select", async (event, ctx) => {
|
|
77
109
|
modelSelectFired = true;
|
|
78
110
|
const wasAnthropic = isAnthropicProvider;
|
|
79
|
-
// Gate on
|
|
80
|
-
//
|
|
81
|
-
//
|
|
82
|
-
|
|
111
|
+
// Gate on api shape AND provider allowlist: direct Anthropic, claude-code
|
|
112
|
+
// OAuth, and anthropic-vertex accept `web_search_20250305`; copilot /
|
|
113
|
+
// minimax / kimi / opencode route Claude-compat models through the same
|
|
114
|
+
// wire protocol but reject the server-side tool (#4492 regression).
|
|
115
|
+
isAnthropicProvider = supportsNativeWebSearch(event.model);
|
|
83
116
|
const hasBrave = !!process.env.BRAVE_API_KEY;
|
|
84
117
|
// When Anthropic (and not preferring Brave): disable custom search tools —
|
|
85
118
|
// native web_search is server-side and more reliable.
|
|
@@ -120,20 +153,19 @@ export function registerNativeSearchHooks(pi) {
|
|
|
120
153
|
// modelsAreEqual suppresses model_select AND the SDK doesn't pass model.
|
|
121
154
|
const eventModel = event.model;
|
|
122
155
|
let isAnthropic;
|
|
123
|
-
if (eventModel?.api) {
|
|
124
|
-
// Preferred path: gate on
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
// `anthropic` maps unambiguously without the api field. Other Anthropic
|
|
130
|
-
// transports will arrive via the modelSelectFired or model-name branch.
|
|
131
|
-
isAnthropic = eventModel.provider === "anthropic";
|
|
156
|
+
if (eventModel?.api || eventModel?.provider) {
|
|
157
|
+
// Preferred path: gate on api shape + provider allowlist. Both fields
|
|
158
|
+
// are authoritative when present — do NOT fall back to the model-name
|
|
159
|
+
// heuristic, which would misclassify copilot-served Claude as Anthropic
|
|
160
|
+
// (#444 regression) or minimax-served Claude-compat as Anthropic (#4492).
|
|
161
|
+
isAnthropic = supportsNativeWebSearch(eventModel);
|
|
132
162
|
}
|
|
133
163
|
else if (modelSelectFired) {
|
|
134
164
|
isAnthropic = isAnthropicProvider;
|
|
135
165
|
}
|
|
136
166
|
else {
|
|
167
|
+
// Last resort: session-restore paths where the SDK doesn't pass model.
|
|
168
|
+
// The model-name prefix is best-effort and assumes direct Anthropic.
|
|
137
169
|
const modelName = typeof payload.model === "string" ? payload.model : "";
|
|
138
170
|
isAnthropic = modelName.startsWith("claude-");
|
|
139
171
|
}
|