@gajae-code/ai 0.1.1
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/CHANGELOG.md +2644 -0
- package/README.md +1181 -0
- package/dist/types/api-registry.d.ts +30 -0
- package/dist/types/auth-broker/client.d.ts +66 -0
- package/dist/types/auth-broker/index.d.ts +5 -0
- package/dist/types/auth-broker/refresher.d.ts +25 -0
- package/dist/types/auth-broker/remote-store.d.ts +96 -0
- package/dist/types/auth-broker/server.d.ts +32 -0
- package/dist/types/auth-broker/types.d.ts +105 -0
- package/dist/types/auth-broker/wire-schemas.d.ts +412 -0
- package/dist/types/auth-gateway/http.d.ts +39 -0
- package/dist/types/auth-gateway/index.d.ts +3 -0
- package/dist/types/auth-gateway/server.d.ts +17 -0
- package/dist/types/auth-gateway/types.d.ts +115 -0
- package/dist/types/auth-storage.d.ts +641 -0
- package/dist/types/cli.d.ts +2 -0
- package/dist/types/index.d.ts +49 -0
- package/dist/types/model-cache.d.ts +17 -0
- package/dist/types/model-manager.d.ts +62 -0
- package/dist/types/model-thinking.d.ts +71 -0
- package/dist/types/models.d.ts +12 -0
- package/dist/types/provider-details.d.ts +24 -0
- package/dist/types/provider-models/bundled-references.d.ts +4 -0
- package/dist/types/provider-models/descriptors.d.ts +48 -0
- package/dist/types/provider-models/google.d.ts +20 -0
- package/dist/types/provider-models/index.d.ts +5 -0
- package/dist/types/provider-models/ollama.d.ts +7 -0
- package/dist/types/provider-models/openai-compat.d.ts +237 -0
- package/dist/types/provider-models/special.d.ts +16 -0
- package/dist/types/providers/amazon-bedrock.d.ts +36 -0
- package/dist/types/providers/anthropic-messages-server-schema.d.ts +450 -0
- package/dist/types/providers/anthropic-messages-server.d.ts +17 -0
- package/dist/types/providers/anthropic.d.ts +188 -0
- package/dist/types/providers/aws-credentials.d.ts +43 -0
- package/dist/types/providers/aws-eventstream.d.ts +38 -0
- package/dist/types/providers/aws-sigv4.d.ts +55 -0
- package/dist/types/providers/azure-openai-responses.d.ts +15 -0
- package/dist/types/providers/cursor/gen/agent_pb.d.ts +13022 -0
- package/dist/types/providers/cursor.d.ts +42 -0
- package/dist/types/providers/error-message.d.ts +27 -0
- package/dist/types/providers/github-copilot-headers.d.ts +40 -0
- package/dist/types/providers/gitlab-duo.d.ts +27 -0
- package/dist/types/providers/google-auth.d.ts +24 -0
- package/dist/types/providers/google-gemini-cli.d.ts +72 -0
- package/dist/types/providers/google-gemini-headers.d.ts +18 -0
- package/dist/types/providers/google-shared.d.ts +163 -0
- package/dist/types/providers/google-types.d.ts +138 -0
- package/dist/types/providers/google-vertex.d.ts +7 -0
- package/dist/types/providers/google.d.ts +4 -0
- package/dist/types/providers/grammar.d.ts +1 -0
- package/dist/types/providers/kimi.d.ts +27 -0
- package/dist/types/providers/mock.d.ts +175 -0
- package/dist/types/providers/ollama.d.ts +6 -0
- package/dist/types/providers/openai-anthropic-shim.d.ts +31 -0
- package/dist/types/providers/openai-chat-server-schema.d.ts +814 -0
- package/dist/types/providers/openai-chat-server.d.ts +16 -0
- package/dist/types/providers/openai-codex/constants.d.ts +26 -0
- package/dist/types/providers/openai-codex/request-transformer.d.ts +49 -0
- package/dist/types/providers/openai-codex/response-handler.d.ts +17 -0
- package/dist/types/providers/openai-codex-responses.d.ts +67 -0
- package/dist/types/providers/openai-completions-compat.d.ts +25 -0
- package/dist/types/providers/openai-completions.d.ts +33 -0
- package/dist/types/providers/openai-responses-server-schema.d.ts +392 -0
- package/dist/types/providers/openai-responses-server.d.ts +17 -0
- package/dist/types/providers/openai-responses-shared.d.ts +89 -0
- package/dist/types/providers/openai-responses.d.ts +32 -0
- package/dist/types/providers/pi-native-client.d.ts +13 -0
- package/dist/types/providers/pi-native-server.d.ts +68 -0
- package/dist/types/providers/register-builtins.d.ts +31 -0
- package/dist/types/providers/synthetic.d.ts +26 -0
- package/dist/types/providers/transform-messages.d.ts +12 -0
- package/dist/types/providers/vision-guard.d.ts +8 -0
- package/dist/types/rate-limit-utils.d.ts +19 -0
- package/dist/types/stream.d.ts +24 -0
- package/dist/types/types.d.ts +746 -0
- package/dist/types/usage/claude.d.ts +3 -0
- package/dist/types/usage/gemini.d.ts +2 -0
- package/dist/types/usage/github-copilot.d.ts +7 -0
- package/dist/types/usage/google-antigravity.d.ts +2 -0
- package/dist/types/usage/kimi.d.ts +2 -0
- package/dist/types/usage/minimax-code.d.ts +2 -0
- package/dist/types/usage/openai-codex.d.ts +3 -0
- package/dist/types/usage/shared.d.ts +1 -0
- package/dist/types/usage/zai.d.ts +2 -0
- package/dist/types/usage.d.ts +258 -0
- package/dist/types/utils/abort.d.ts +19 -0
- package/dist/types/utils/anthropic-auth.d.ts +31 -0
- package/dist/types/utils/discovery/antigravity.d.ts +61 -0
- package/dist/types/utils/discovery/codex.d.ts +38 -0
- package/dist/types/utils/discovery/cursor.d.ts +23 -0
- package/dist/types/utils/discovery/gemini.d.ts +25 -0
- package/dist/types/utils/discovery/index.d.ts +4 -0
- package/dist/types/utils/discovery/openai-compatible.d.ts +72 -0
- package/dist/types/utils/event-stream.d.ts +28 -0
- package/dist/types/utils/fireworks-model-id.d.ts +10 -0
- package/dist/types/utils/foundry.d.ts +1 -0
- package/dist/types/utils/h2-fetch.d.ts +22 -0
- package/dist/types/utils/http-inspector.d.ts +31 -0
- package/dist/types/utils/idle-iterator.d.ts +67 -0
- package/dist/types/utils/json-parse.d.ts +10 -0
- package/dist/types/utils/oauth/alibaba-coding-plan.d.ts +18 -0
- package/dist/types/utils/oauth/anthropic.d.ts +22 -0
- package/dist/types/utils/oauth/api-key-login.d.ts +35 -0
- package/dist/types/utils/oauth/api-key-validation.d.ts +27 -0
- package/dist/types/utils/oauth/callback-server.d.ts +57 -0
- package/dist/types/utils/oauth/cerebras.d.ts +1 -0
- package/dist/types/utils/oauth/cloudflare-ai-gateway.d.ts +18 -0
- package/dist/types/utils/oauth/cursor.d.ts +15 -0
- package/dist/types/utils/oauth/deepseek.d.ts +10 -0
- package/dist/types/utils/oauth/firepass.d.ts +1 -0
- package/dist/types/utils/oauth/fireworks.d.ts +1 -0
- package/dist/types/utils/oauth/github-copilot.d.ts +38 -0
- package/dist/types/utils/oauth/gitlab-duo.d.ts +3 -0
- package/dist/types/utils/oauth/google-antigravity.d.ts +11 -0
- package/dist/types/utils/oauth/google-gemini-cli.d.ts +10 -0
- package/dist/types/utils/oauth/google-oauth-shared.d.ts +28 -0
- package/dist/types/utils/oauth/huggingface.d.ts +19 -0
- package/dist/types/utils/oauth/index.d.ts +38 -0
- package/dist/types/utils/oauth/kagi.d.ts +17 -0
- package/dist/types/utils/oauth/kilo.d.ts +5 -0
- package/dist/types/utils/oauth/kimi.d.ts +21 -0
- package/dist/types/utils/oauth/litellm.d.ts +18 -0
- package/dist/types/utils/oauth/lm-studio.d.ts +17 -0
- package/dist/types/utils/oauth/minimax-code.d.ts +28 -0
- package/dist/types/utils/oauth/moonshot.d.ts +1 -0
- package/dist/types/utils/oauth/nanogpt.d.ts +1 -0
- package/dist/types/utils/oauth/nvidia.d.ts +18 -0
- package/dist/types/utils/oauth/ollama-cloud.d.ts +2 -0
- package/dist/types/utils/oauth/ollama.d.ts +18 -0
- package/dist/types/utils/oauth/openai-codex.d.ts +21 -0
- package/dist/types/utils/oauth/opencode.d.ts +18 -0
- package/dist/types/utils/oauth/parallel.d.ts +17 -0
- package/dist/types/utils/oauth/perplexity.d.ts +9 -0
- package/dist/types/utils/oauth/pkce.d.ts +8 -0
- package/dist/types/utils/oauth/qianfan.d.ts +17 -0
- package/dist/types/utils/oauth/qwen-portal.d.ts +19 -0
- package/dist/types/utils/oauth/synthetic.d.ts +1 -0
- package/dist/types/utils/oauth/tavily.d.ts +17 -0
- package/dist/types/utils/oauth/together.d.ts +1 -0
- package/dist/types/utils/oauth/types.d.ts +44 -0
- package/dist/types/utils/oauth/venice.d.ts +18 -0
- package/dist/types/utils/oauth/vercel-ai-gateway.d.ts +18 -0
- package/dist/types/utils/oauth/vllm.d.ts +16 -0
- package/dist/types/utils/oauth/xiaomi.d.ts +19 -0
- package/dist/types/utils/oauth/zai.d.ts +18 -0
- package/dist/types/utils/oauth/zenmux.d.ts +1 -0
- package/dist/types/utils/overflow.d.ts +54 -0
- package/dist/types/utils/parse-bind.d.ts +23 -0
- package/dist/types/utils/provider-response.d.ts +3 -0
- package/dist/types/utils/retry-after.d.ts +3 -0
- package/dist/types/utils/retry.d.ts +26 -0
- package/dist/types/utils/schema/adapt.d.ts +24 -0
- package/dist/types/utils/schema/compatibility.d.ts +30 -0
- package/dist/types/utils/schema/dereference.d.ts +11 -0
- package/dist/types/utils/schema/draft.d.ts +10 -0
- package/dist/types/utils/schema/equality.d.ts +4 -0
- package/dist/types/utils/schema/fields.d.ts +49 -0
- package/dist/types/utils/schema/index.d.ts +13 -0
- package/dist/types/utils/schema/json-schema-validator.d.ts +12 -0
- package/dist/types/utils/schema/meta-validator.d.ts +2 -0
- package/dist/types/utils/schema/normalize.d.ts +93 -0
- package/dist/types/utils/schema/spill.d.ts +8 -0
- package/dist/types/utils/schema/stamps.d.ts +25 -0
- package/dist/types/utils/schema/types.d.ts +4 -0
- package/dist/types/utils/schema/wire.d.ts +54 -0
- package/dist/types/utils/schema/zod-decontaminate.d.ts +31 -0
- package/dist/types/utils/sse-debug.d.ts +10 -0
- package/dist/types/utils/tool-call-healing.d.ts +71 -0
- package/dist/types/utils/tool-choice.d.ts +50 -0
- package/dist/types/utils/validation.d.ts +17 -0
- package/dist/types/utils.d.ts +28 -0
- package/package.json +146 -0
- package/src/api-registry.ts +96 -0
- package/src/auth-broker/client.ts +358 -0
- package/src/auth-broker/index.ts +5 -0
- package/src/auth-broker/refresher.ts +127 -0
- package/src/auth-broker/remote-store.ts +623 -0
- package/src/auth-broker/server.ts +644 -0
- package/src/auth-broker/types.ts +127 -0
- package/src/auth-broker/wire-schemas.ts +200 -0
- package/src/auth-gateway/http.ts +194 -0
- package/src/auth-gateway/index.ts +3 -0
- package/src/auth-gateway/server.ts +717 -0
- package/src/auth-gateway/types.ts +134 -0
- package/src/auth-storage.ts +4104 -0
- package/src/cli.ts +262 -0
- package/src/index.ts +54 -0
- package/src/model-cache.ts +129 -0
- package/src/model-manager.ts +450 -0
- package/src/model-thinking.ts +691 -0
- package/src/models.json +73853 -0
- package/src/models.json.d.ts +9 -0
- package/src/models.ts +56 -0
- package/src/prompts/turn-aborted-guidance.md +4 -0
- package/src/provider-details.ts +90 -0
- package/src/provider-models/bundled-references.ts +38 -0
- package/src/provider-models/descriptors.ts +308 -0
- package/src/provider-models/google.ts +91 -0
- package/src/provider-models/index.ts +5 -0
- package/src/provider-models/ollama.ts +153 -0
- package/src/provider-models/openai-compat.ts +2275 -0
- package/src/provider-models/special.ts +67 -0
- package/src/providers/amazon-bedrock.ts +849 -0
- package/src/providers/anthropic-messages-server-schema.ts +229 -0
- package/src/providers/anthropic-messages-server.ts +677 -0
- package/src/providers/anthropic.ts +2696 -0
- package/src/providers/aws-credentials.ts +501 -0
- package/src/providers/aws-eventstream.ts +185 -0
- package/src/providers/aws-sigv4.ts +218 -0
- package/src/providers/azure-openai-responses.ts +337 -0
- package/src/providers/cursor/gen/agent_pb.ts +15274 -0
- package/src/providers/cursor/proto/agent.proto +3526 -0
- package/src/providers/cursor/proto/buf.gen.yaml +6 -0
- package/src/providers/cursor/proto/buf.yaml +17 -0
- package/src/providers/cursor.ts +2561 -0
- package/src/providers/error-message.ts +21 -0
- package/src/providers/github-copilot-headers.ts +140 -0
- package/src/providers/gitlab-duo.ts +372 -0
- package/src/providers/google-auth.ts +252 -0
- package/src/providers/google-gemini-cli.ts +795 -0
- package/src/providers/google-gemini-headers.ts +41 -0
- package/src/providers/google-shared.ts +902 -0
- package/src/providers/google-types.ts +167 -0
- package/src/providers/google-vertex.ts +88 -0
- package/src/providers/google.ts +41 -0
- package/src/providers/grammar.ts +70 -0
- package/src/providers/kimi.ts +52 -0
- package/src/providers/mock.ts +500 -0
- package/src/providers/ollama.ts +544 -0
- package/src/providers/openai-anthropic-shim.ts +138 -0
- package/src/providers/openai-chat-server-schema.ts +243 -0
- package/src/providers/openai-chat-server.ts +628 -0
- package/src/providers/openai-codex/constants.ts +43 -0
- package/src/providers/openai-codex/request-transformer.ts +161 -0
- package/src/providers/openai-codex/response-handler.ts +81 -0
- package/src/providers/openai-codex-responses.ts +2598 -0
- package/src/providers/openai-completions-compat.ts +279 -0
- package/src/providers/openai-completions.ts +1853 -0
- package/src/providers/openai-responses-server-schema.ts +290 -0
- package/src/providers/openai-responses-server.ts +1183 -0
- package/src/providers/openai-responses-shared.ts +800 -0
- package/src/providers/openai-responses.ts +621 -0
- package/src/providers/pi-native-client.ts +228 -0
- package/src/providers/pi-native-server.ts +210 -0
- package/src/providers/register-builtins.ts +412 -0
- package/src/providers/synthetic.ts +50 -0
- package/src/providers/transform-messages.ts +309 -0
- package/src/providers/vision-guard.ts +31 -0
- package/src/rate-limit-utils.ts +84 -0
- package/src/stream.ts +895 -0
- package/src/types.ts +884 -0
- package/src/usage/claude.ts +431 -0
- package/src/usage/gemini.ts +250 -0
- package/src/usage/github-copilot.ts +421 -0
- package/src/usage/google-antigravity.ts +201 -0
- package/src/usage/kimi.ts +271 -0
- package/src/usage/minimax-code.ts +31 -0
- package/src/usage/openai-codex.ts +503 -0
- package/src/usage/shared.ts +10 -0
- package/src/usage/zai.ts +247 -0
- package/src/usage.ts +183 -0
- package/src/utils/abort.ts +51 -0
- package/src/utils/anthropic-auth.ts +87 -0
- package/src/utils/discovery/antigravity.ts +261 -0
- package/src/utils/discovery/codex.ts +371 -0
- package/src/utils/discovery/cursor.ts +306 -0
- package/src/utils/discovery/gemini.ts +248 -0
- package/src/utils/discovery/index.ts +4 -0
- package/src/utils/discovery/openai-compatible.ts +224 -0
- package/src/utils/event-stream.ts +142 -0
- package/src/utils/fireworks-model-id.ts +30 -0
- package/src/utils/foundry.ts +8 -0
- package/src/utils/h2-fetch.ts +60 -0
- package/src/utils/http-inspector.ts +176 -0
- package/src/utils/idle-iterator.ts +250 -0
- package/src/utils/json-parse.ts +148 -0
- package/src/utils/oauth/alibaba-coding-plan.ts +59 -0
- package/src/utils/oauth/anthropic.ts +200 -0
- package/src/utils/oauth/api-key-login.ts +87 -0
- package/src/utils/oauth/api-key-validation.ts +92 -0
- package/src/utils/oauth/callback-server.ts +276 -0
- package/src/utils/oauth/cerebras.ts +16 -0
- package/src/utils/oauth/cloudflare-ai-gateway.ts +48 -0
- package/src/utils/oauth/cursor.ts +157 -0
- package/src/utils/oauth/deepseek.ts +53 -0
- package/src/utils/oauth/firepass.ts +24 -0
- package/src/utils/oauth/fireworks.ts +15 -0
- package/src/utils/oauth/github-copilot.ts +362 -0
- package/src/utils/oauth/gitlab-duo.ts +123 -0
- package/src/utils/oauth/google-antigravity.ts +200 -0
- package/src/utils/oauth/google-gemini-cli.ts +256 -0
- package/src/utils/oauth/google-oauth-shared.ts +110 -0
- package/src/utils/oauth/huggingface.ts +62 -0
- package/src/utils/oauth/index.ts +444 -0
- package/src/utils/oauth/kagi.ts +47 -0
- package/src/utils/oauth/kilo.ts +87 -0
- package/src/utils/oauth/kimi.ts +254 -0
- package/src/utils/oauth/litellm.ts +47 -0
- package/src/utils/oauth/lm-studio.ts +38 -0
- package/src/utils/oauth/minimax-code.ts +78 -0
- package/src/utils/oauth/moonshot.ts +16 -0
- package/src/utils/oauth/nanogpt.ts +15 -0
- package/src/utils/oauth/nvidia.ts +70 -0
- package/src/utils/oauth/oauth.html +199 -0
- package/src/utils/oauth/ollama-cloud.ts +28 -0
- package/src/utils/oauth/ollama.ts +47 -0
- package/src/utils/oauth/openai-codex.ts +299 -0
- package/src/utils/oauth/opencode.ts +49 -0
- package/src/utils/oauth/parallel.ts +46 -0
- package/src/utils/oauth/perplexity.ts +206 -0
- package/src/utils/oauth/pkce.ts +18 -0
- package/src/utils/oauth/qianfan.ts +58 -0
- package/src/utils/oauth/qwen-portal.ts +60 -0
- package/src/utils/oauth/synthetic.ts +16 -0
- package/src/utils/oauth/tavily.ts +46 -0
- package/src/utils/oauth/together.ts +16 -0
- package/src/utils/oauth/types.ts +94 -0
- package/src/utils/oauth/venice.ts +59 -0
- package/src/utils/oauth/vercel-ai-gateway.ts +47 -0
- package/src/utils/oauth/vllm.ts +40 -0
- package/src/utils/oauth/xiaomi.ts +137 -0
- package/src/utils/oauth/zai.ts +60 -0
- package/src/utils/oauth/zenmux.ts +15 -0
- package/src/utils/overflow.ts +137 -0
- package/src/utils/parse-bind.ts +54 -0
- package/src/utils/provider-response.ts +30 -0
- package/src/utils/retry-after.ts +110 -0
- package/src/utils/retry.ts +54 -0
- package/src/utils/schema/CONSTRAINTS.md +164 -0
- package/src/utils/schema/adapt.ts +36 -0
- package/src/utils/schema/compatibility.ts +435 -0
- package/src/utils/schema/dereference.ts +98 -0
- package/src/utils/schema/draft.ts +341 -0
- package/src/utils/schema/equality.ts +97 -0
- package/src/utils/schema/fields.ts +190 -0
- package/src/utils/schema/index.ts +13 -0
- package/src/utils/schema/json-schema-validator.ts +577 -0
- package/src/utils/schema/meta-validator.ts +167 -0
- package/src/utils/schema/normalize.ts +1588 -0
- package/src/utils/schema/spill.ts +43 -0
- package/src/utils/schema/stamps.ts +97 -0
- package/src/utils/schema/types.ts +11 -0
- package/src/utils/schema/wire.ts +213 -0
- package/src/utils/schema/zod-decontaminate.ts +331 -0
- package/src/utils/sse-debug.ts +289 -0
- package/src/utils/tool-call-healing.ts +271 -0
- package/src/utils/tool-choice.ts +99 -0
- package/src/utils/validation.ts +1019 -0
- package/src/utils.ts +166 -0
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import type { JsonObject } from "./types";
|
|
2
|
+
|
|
3
|
+
export type DescriptionSpillFormat = "spill" | "paren";
|
|
4
|
+
|
|
5
|
+
function formatSpillValue(value: unknown): string {
|
|
6
|
+
return JSON.stringify(value);
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
function formatParenValue(value: unknown): string {
|
|
10
|
+
return typeof value === "string" ? value : JSON.stringify(value);
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Demote stripped JSON Schema keywords into a node's `description` so the model
|
|
15
|
+
* still receives the constraint as natural-language context after the wire
|
|
16
|
+
* schema drops it.
|
|
17
|
+
*/
|
|
18
|
+
export function spillToDescription(
|
|
19
|
+
node: JsonObject,
|
|
20
|
+
entries: ReadonlyArray<readonly [string, unknown]>,
|
|
21
|
+
format: DescriptionSpillFormat = "spill",
|
|
22
|
+
): void {
|
|
23
|
+
let spilled: Array<readonly [string, unknown]> | undefined;
|
|
24
|
+
for (const entry of entries) {
|
|
25
|
+
if (entry[1] === undefined) continue;
|
|
26
|
+
if (spilled === undefined) spilled = [];
|
|
27
|
+
spilled.push(entry);
|
|
28
|
+
}
|
|
29
|
+
if (spilled === undefined || spilled.length === 0) return;
|
|
30
|
+
|
|
31
|
+
const existing = typeof node.description === "string" ? node.description : "";
|
|
32
|
+
if (format === "paren") {
|
|
33
|
+
let suffix = "";
|
|
34
|
+
for (const [key, value] of spilled) {
|
|
35
|
+
suffix += ` (${key}: ${formatParenValue(value)})`;
|
|
36
|
+
}
|
|
37
|
+
node.description = `${existing}${suffix}`;
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
const formatted = `{${spilled.map(([key, value]) => `${key}: ${formatSpillValue(value)}`).join(", ")}}`;
|
|
42
|
+
node.description = existing ? `${existing}\n\n${formatted}` : formatted;
|
|
43
|
+
}
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Symbol-keyed lazy memoization stamped directly onto the host object.
|
|
3
|
+
*
|
|
4
|
+
* Faster than a module-level `WeakMap` in V8/JSC because the symbol slot is
|
|
5
|
+
* resolved through the object's hidden class instead of a side-table hash
|
|
6
|
+
* lookup. The slot is defined as a non-enumerable property so the stamp
|
|
7
|
+
* does not leak through `{...spread}`, `Object.keys`, `JSON.stringify`, or
|
|
8
|
+
* `toEqual`-style deep equality.
|
|
9
|
+
*
|
|
10
|
+
* Caveats: the stamp lives as long as the host object, even after callers
|
|
11
|
+
* release their references to the cached value — only use this for caches
|
|
12
|
+
* whose lifetime should match the host. Frozen hosts will throw on write in
|
|
13
|
+
* strict mode; callers that may receive frozen input must handle that.
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
function define<T extends object>(target: T, key: symbol, value: unknown): void {
|
|
17
|
+
Object.defineProperty(target, key, { value, writable: true, configurable: true });
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export function stamp<T extends object, V>(target: T, key: symbol, compute: (target: T) => V): V {
|
|
21
|
+
const slot = target as Record<symbol, V | undefined>;
|
|
22
|
+
const existing = slot[key];
|
|
23
|
+
if (existing !== undefined) return existing;
|
|
24
|
+
const value = compute(target);
|
|
25
|
+
define(target, key, value);
|
|
26
|
+
return value;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Epoch-keyed cycle guard. Cheaper than `WeakSet` for recursive traversal
|
|
31
|
+
* because the marker is a single property slot on the host object, written
|
|
32
|
+
* once and overwritten in place on every subsequent traversal — the hidden
|
|
33
|
+
* class transitions once per object lifetime, not per traversal.
|
|
34
|
+
*
|
|
35
|
+
* Usage:
|
|
36
|
+
* function walk(node, epoch = epochNext()) {
|
|
37
|
+
* if (!once(node, epoch)) return; // cycle
|
|
38
|
+
* for (const child of node.children) walk(child, epoch);
|
|
39
|
+
* }
|
|
40
|
+
*/
|
|
41
|
+
const kEpoch = Symbol("pi.schema.epoch");
|
|
42
|
+
let __epoch = 0;
|
|
43
|
+
|
|
44
|
+
export function epochNext(): number {
|
|
45
|
+
return ++__epoch;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
/**
|
|
49
|
+
* Marks `target` as visited for this `epoch`. Returns `true` the first time
|
|
50
|
+
* it is called for a given (target, epoch) pair and `false` on every
|
|
51
|
+
* subsequent call within the same epoch.
|
|
52
|
+
*/
|
|
53
|
+
export function once<T extends object>(target: T, epoch: number): boolean {
|
|
54
|
+
const slot = target as Record<symbol, number | undefined>;
|
|
55
|
+
const cur = slot[kEpoch];
|
|
56
|
+
if (cur !== undefined && cur >= epoch) return false;
|
|
57
|
+
if (cur === undefined) define(target, kEpoch, epoch);
|
|
58
|
+
else slot[kEpoch] = epoch;
|
|
59
|
+
return true;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
/**
|
|
63
|
+
* Counter-based path tracker. Use when a traversal needs to distinguish
|
|
64
|
+
* "currently on the recursion path" from "previously visited" — i.e. cycle
|
|
65
|
+
* detection that throws while still allowing DAG sharing. Increment on
|
|
66
|
+
* entry, decrement on exit; the slot returns to 0 after a balanced walk so
|
|
67
|
+
* subsequent top-level calls see a fresh state without any reset.
|
|
68
|
+
*
|
|
69
|
+
* Unlike a `WeakSet` with `seen.delete(...)`, the property is never deleted
|
|
70
|
+
* — only incremented and decremented — so the host object's hidden class
|
|
71
|
+
* is never invalidated.
|
|
72
|
+
*
|
|
73
|
+
* Usage:
|
|
74
|
+
* function walk(node) {
|
|
75
|
+
* if (!enter(node)) throw new Error("cycle");
|
|
76
|
+
* try { for (const c of node.children) walk(c); }
|
|
77
|
+
* finally { exit(node); }
|
|
78
|
+
* }
|
|
79
|
+
*/
|
|
80
|
+
const kDepth = Symbol("pi.schema.depth");
|
|
81
|
+
|
|
82
|
+
/** Returns `true` on first entry, `false` if `target` is already on the current path. */
|
|
83
|
+
export function enter<T extends object>(target: T): boolean {
|
|
84
|
+
const slot = target as Record<symbol, number | undefined>;
|
|
85
|
+
const cur = slot[kDepth];
|
|
86
|
+
if (cur === undefined) {
|
|
87
|
+
define(target, kDepth, 1);
|
|
88
|
+
return true;
|
|
89
|
+
}
|
|
90
|
+
slot[kDepth] = cur + 1;
|
|
91
|
+
return cur === 0;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
export function exit<T extends object>(target: T): void {
|
|
95
|
+
const slot = target as Record<symbol, number>;
|
|
96
|
+
slot[kDepth]--;
|
|
97
|
+
}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
export type JsonObject = Record<string, unknown>;
|
|
2
|
+
|
|
3
|
+
export function isJsonObject(value: unknown): value is JsonObject {
|
|
4
|
+
return !!value && typeof value === "object" && !Array.isArray(value);
|
|
5
|
+
}
|
|
6
|
+
|
|
7
|
+
/** True when `value` is a plain JSON object with no own enumerable keys. */
|
|
8
|
+
export function isJsonObjectEmpty(value: JsonObject): boolean {
|
|
9
|
+
for (const _ in value) return false;
|
|
10
|
+
return true;
|
|
11
|
+
}
|
|
@@ -0,0 +1,213 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Compute the wire (JSON Schema) representation of a tool's parameters and
|
|
3
|
+
* convert TypeBox-style schemas into Zod for internal validation.
|
|
4
|
+
*
|
|
5
|
+
* Tools may author parameters in two shapes:
|
|
6
|
+
* 1. Zod (canonical going forward) — converted to JSON Schema on demand.
|
|
7
|
+
* 2. TypeBox / plain JSON Schema (legacy + extension compat) — upgraded to
|
|
8
|
+
* draft 2020-12 without converting through Zod.
|
|
9
|
+
*
|
|
10
|
+
* Both are normalized at the boundary so providers and validators see the same
|
|
11
|
+
* JSON Schema dialect.
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
// We import the Zod *value* (z) for runtime APIs. Marker checks rely on the
|
|
15
|
+
// `_zod` symbol that every Zod v4 schema instance carries.
|
|
16
|
+
import { type ZodType, z } from "zod/v4";
|
|
17
|
+
import type { Tool, TSchema } from "../../types";
|
|
18
|
+
import { upgradeJsonSchemaTo202012 } from "./draft";
|
|
19
|
+
import { stamp } from "./stamps";
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* True when `value` is a live Zod schema instance.
|
|
23
|
+
*
|
|
24
|
+
* The check is stricter than "has a `_zod` property" because a JSON
|
|
25
|
+
* round-trip preserves the `_zod` key as a plain object and would otherwise
|
|
26
|
+
* fool the predicate — see issue #1101, where MCP servers ship
|
|
27
|
+
* `JSON.stringify(zodSchemaInstance)` as a tool's `inputSchema` and the
|
|
28
|
+
* resulting plain object then explodes `z.toJSONSchema` because the prototype
|
|
29
|
+
* (and every Zod parsing method) is gone.
|
|
30
|
+
*
|
|
31
|
+
* Live Zod instances always carry a `.parse` function on the prototype;
|
|
32
|
+
* impostors do not.
|
|
33
|
+
*/
|
|
34
|
+
export function isZodSchema(value: unknown): value is ZodType {
|
|
35
|
+
return (
|
|
36
|
+
typeof value === "object" &&
|
|
37
|
+
value !== null &&
|
|
38
|
+
// Zod v4 instances expose a `_zod` internal property with a `def` object.
|
|
39
|
+
// Tagging on this marker keeps the check stable across Zod minor versions.
|
|
40
|
+
// (`_zod` is part of Zod's documented internal contract used by introspection.)
|
|
41
|
+
// We avoid checking constructor name because Zod ships multiple variants
|
|
42
|
+
// (`ZodObject`, `ZodOptional`, etc.) and a tagged-union style check would
|
|
43
|
+
// have to enumerate them all.
|
|
44
|
+
"_zod" in value &&
|
|
45
|
+
typeof (value as { _zod?: { def?: unknown } })._zod === "object" &&
|
|
46
|
+
// Reject JSON-roundtripped objects that kept the `_zod` key but lost the
|
|
47
|
+
// prototype. Real instances have `.parse` on the prototype chain.
|
|
48
|
+
typeof (value as { parse?: unknown }).parse === "function"
|
|
49
|
+
);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/** Symbol-stamped caches keyed by schema object identity. */
|
|
53
|
+
const kZodWireSchema = Symbol("pi.schema.zod.wire");
|
|
54
|
+
const kJsonWireSchema = Symbol("pi.schema.json.wire");
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* Post-process Zod-emitted JSON Schema so it matches the wire shape providers
|
|
58
|
+
* already expect from TypeBox-authored tools:
|
|
59
|
+
*
|
|
60
|
+
* - Drop the `$schema` URL (providers parse the body, not the metadata).
|
|
61
|
+
* - Make fields with a `default` non-required (TypeBox/JSON-Schema semantics
|
|
62
|
+
* treat defaulted fields as optional; Zod inverts this and keeps them
|
|
63
|
+
* required at the input boundary, then materializes the default).
|
|
64
|
+
* - Strip the noisy safe-integer bounds Zod injects for `z.number().int()`.
|
|
65
|
+
*
|
|
66
|
+
* The empty-schema normalization (`{}` → `true`, see `normalizeEmptySchemas`)
|
|
67
|
+
* runs separately from `toolWireSchema` so both Zod and TypeBox tools get it.
|
|
68
|
+
*/
|
|
69
|
+
function postProcess(schema: Record<string, unknown>): Record<string, unknown> {
|
|
70
|
+
delete schema.$schema;
|
|
71
|
+
walk(schema);
|
|
72
|
+
normalizeEmptySchemas(schema);
|
|
73
|
+
return schema;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
const SAFE_INTEGER_MAX = Number.MAX_SAFE_INTEGER;
|
|
77
|
+
const SAFE_INTEGER_MIN = Number.MIN_SAFE_INTEGER;
|
|
78
|
+
|
|
79
|
+
/** Keys whose values are a single JSON Schema (not an array or map). */
|
|
80
|
+
const SCHEMA_VALUE_KEYS = [
|
|
81
|
+
"additionalProperties",
|
|
82
|
+
"unevaluatedProperties",
|
|
83
|
+
"unevaluatedItems",
|
|
84
|
+
"items",
|
|
85
|
+
"contains",
|
|
86
|
+
"propertyNames",
|
|
87
|
+
"if",
|
|
88
|
+
"then",
|
|
89
|
+
"else",
|
|
90
|
+
"not",
|
|
91
|
+
] as const;
|
|
92
|
+
|
|
93
|
+
/** Keys whose values are a map of `{ key: Schema }` entries. */
|
|
94
|
+
const SCHEMA_MAP_KEYS = ["properties", "patternProperties", "$defs", "definitions"] as const;
|
|
95
|
+
|
|
96
|
+
/** Keys whose values are an array of schemas. */
|
|
97
|
+
const SCHEMA_ARRAY_KEYS = ["anyOf", "oneOf", "allOf", "prefixItems"] as const;
|
|
98
|
+
|
|
99
|
+
/** True when `val` is a plain empty object `{}`. */
|
|
100
|
+
function isEmptyObject(val: unknown): val is Record<string, never> {
|
|
101
|
+
if (val === null || typeof val !== "object" || Array.isArray(val)) return false;
|
|
102
|
+
for (const _ in val as object) return false;
|
|
103
|
+
return true;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
function walk(node: unknown): void {
|
|
107
|
+
if (Array.isArray(node)) {
|
|
108
|
+
for (const child of node) walk(child);
|
|
109
|
+
return;
|
|
110
|
+
}
|
|
111
|
+
if (!node || typeof node !== "object") return;
|
|
112
|
+
const obj = node as Record<string, unknown>;
|
|
113
|
+
|
|
114
|
+
// Drop noise injected for `z.number().int()`.
|
|
115
|
+
if (obj.type === "integer") {
|
|
116
|
+
if (obj.minimum === SAFE_INTEGER_MIN) delete obj.minimum;
|
|
117
|
+
if (obj.maximum === SAFE_INTEGER_MAX) delete obj.maximum;
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
// Make defaulted properties non-required.
|
|
121
|
+
if (Array.isArray(obj.required) && obj.properties && typeof obj.properties === "object") {
|
|
122
|
+
const properties = obj.properties as Record<string, unknown>;
|
|
123
|
+
const required = obj.required as string[];
|
|
124
|
+
const filtered = required.filter(name => {
|
|
125
|
+
const propertySchema = properties[name];
|
|
126
|
+
if (!propertySchema || typeof propertySchema !== "object") return true;
|
|
127
|
+
return !("default" in (propertySchema as Record<string, unknown>));
|
|
128
|
+
});
|
|
129
|
+
if (filtered.length !== required.length) {
|
|
130
|
+
if (filtered.length === 0) {
|
|
131
|
+
delete obj.required;
|
|
132
|
+
} else {
|
|
133
|
+
obj.required = filtered;
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
for (const k in obj) walk(obj[k]);
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
/**
|
|
142
|
+
* Normalize `{}` (empty JSON Schema = `z.unknown()` / unconstrained value) to
|
|
143
|
+
* boolean `true` in every schema-valued position. JSON Schema draft 2020-12
|
|
144
|
+
* §4.3.1: `{}` and `true` are semantically equivalent ("any JSON value").
|
|
145
|
+
* Grammar-constrained samplers (llama.cpp, etc.) treat the object form as
|
|
146
|
+
* "generate an empty object" rather than "any JSON value", causing open-typed
|
|
147
|
+
* fields like `extra.title` (from `z.record(z.string(), z.unknown())`) to
|
|
148
|
+
* always emit `{}` instead of the intended string/number/etc. (issue #1179).
|
|
149
|
+
*
|
|
150
|
+
* Mutates in place. Provider-agnostic — applied to every tool wire schema so
|
|
151
|
+
* Anthropic, Google, OpenAI, Ollama, Bedrock, and Cursor all see the
|
|
152
|
+
* normalized form, regardless of whether the source was Zod or TypeBox.
|
|
153
|
+
*/
|
|
154
|
+
export function normalizeEmptySchemas(node: unknown): void {
|
|
155
|
+
if (Array.isArray(node)) {
|
|
156
|
+
for (const child of node) normalizeEmptySchemas(child);
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
if (!node || typeof node !== "object") return;
|
|
160
|
+
const obj = node as Record<string, unknown>;
|
|
161
|
+
|
|
162
|
+
for (const key of SCHEMA_VALUE_KEYS) {
|
|
163
|
+
if (Object.hasOwn(obj, key) && isEmptyObject(obj[key])) obj[key] = true;
|
|
164
|
+
}
|
|
165
|
+
for (const mapKey of SCHEMA_MAP_KEYS) {
|
|
166
|
+
const map = obj[mapKey];
|
|
167
|
+
if (map !== null && typeof map === "object" && !Array.isArray(map)) {
|
|
168
|
+
for (const k in map as Record<string, unknown>) {
|
|
169
|
+
if (isEmptyObject((map as Record<string, unknown>)[k])) (map as Record<string, unknown>)[k] = true;
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
for (const arrKey of SCHEMA_ARRAY_KEYS) {
|
|
174
|
+
const arr = obj[arrKey];
|
|
175
|
+
if (Array.isArray(arr)) {
|
|
176
|
+
for (let i = 0; i < arr.length; i++) {
|
|
177
|
+
if (isEmptyObject(arr[i])) arr[i] = true;
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
for (const k in obj) normalizeEmptySchemas(obj[k]);
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
/** Convert a Zod schema into the JSON Schema shape providers consume. */
|
|
186
|
+
export function zodToWireSchema(schema: ZodType): Record<string, unknown> {
|
|
187
|
+
return stamp(schema, kZodWireSchema, s => {
|
|
188
|
+
// `target: "draft-2020-12"` matches what Anthropic's `input_schema` validator
|
|
189
|
+
// requires out of the box; our other provider sanitizers (OpenAI strict,
|
|
190
|
+
// Google, Anthropic CCA) already handle the superset structurally.
|
|
191
|
+
const raw = z.toJSONSchema(s, { target: "draft-2020-12" }) as Record<string, unknown>;
|
|
192
|
+
return postProcess(raw);
|
|
193
|
+
});
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
/**
|
|
197
|
+
* Resolve a tool's parameters to a JSON Schema object suitable for sending
|
|
198
|
+
* over the wire. Zod schemas are converted (and cached); legacy TypeBox / raw
|
|
199
|
+
* JSON Schema parameters are upgraded to draft 2020-12 (and cached).
|
|
200
|
+
*
|
|
201
|
+
* Both branches finish with `normalizeEmptySchemas` so every provider —
|
|
202
|
+
* OpenAI, Anthropic, Google, Ollama, Bedrock, Cursor — sees `{}` normalized
|
|
203
|
+
* to `true` in schema-valued positions (issue #1179).
|
|
204
|
+
*/
|
|
205
|
+
export function toolWireSchema(tool: Tool): Record<string, unknown> {
|
|
206
|
+
const params: TSchema = tool.parameters;
|
|
207
|
+
if (isZodSchema(params)) return zodToWireSchema(params);
|
|
208
|
+
return stamp(params as Record<string, unknown>, kJsonWireSchema, p => {
|
|
209
|
+
const upgraded = upgradeJsonSchemaTo202012(p) as Record<string, unknown>;
|
|
210
|
+
normalizeEmptySchemas(upgraded);
|
|
211
|
+
return upgraded;
|
|
212
|
+
});
|
|
213
|
+
}
|
|
@@ -0,0 +1,331 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Defensive rewrite for nodes that look like `JSON.stringify(zodSchemaInstance)`
|
|
3
|
+
* output rather than JSON Schema. MCP servers using Zod 4 sometimes ship a
|
|
4
|
+
* serialised schema instance directly as a tool's `inputSchema`, because the
|
|
5
|
+
* fields Zod surfaces on its instances (`type`, `enum`, `options`, `def`) shadow
|
|
6
|
+
* (and clash with) JSON Schema keywords. The resulting payload is neither valid
|
|
7
|
+
* Zod nor valid JSON Schema 2020-12 and Anthropic's strict validator rejects
|
|
8
|
+
* the whole tool list.
|
|
9
|
+
*
|
|
10
|
+
* Symptoms we've observed (gitnexus_impact.direction):
|
|
11
|
+
* {
|
|
12
|
+
* def: { type: "enum", entries: { upstream: "upstream", ... } },
|
|
13
|
+
* type: "enum", // <- invalid `type` value
|
|
14
|
+
* enum: { upstream: "upstream", ... }, // <- `enum` MUST be an array
|
|
15
|
+
* options: ["upstream", "downstream"],
|
|
16
|
+
* }
|
|
17
|
+
*
|
|
18
|
+
* This module recognises the shape (`def.type === node.type` and `def.type` is
|
|
19
|
+
* a known Zod kind) and rewrites it to clean JSON Schema where deterministic.
|
|
20
|
+
* For Zod kinds we don't fully model, we strip the toxic siblings (`def`,
|
|
21
|
+
* `options`, object-shaped `enum`) and drop an invalid `type` so the remainder
|
|
22
|
+
* passes meta-schema validation as a permissive node.
|
|
23
|
+
*
|
|
24
|
+
* Pure / identity-preserving: returns the input reference when nothing changes.
|
|
25
|
+
*/
|
|
26
|
+
|
|
27
|
+
import { isJsonObject, type JsonObject } from "./types";
|
|
28
|
+
|
|
29
|
+
const VALID_JSON_SCHEMA_TYPES: Record<string, true> = {
|
|
30
|
+
string: true,
|
|
31
|
+
number: true,
|
|
32
|
+
integer: true,
|
|
33
|
+
boolean: true,
|
|
34
|
+
object: true,
|
|
35
|
+
array: true,
|
|
36
|
+
null: true,
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Known Zod 4 schema kinds as surfaced on `_def.type` / `.type`. Matching this
|
|
41
|
+
* set (rather than just "has `def`") is what keeps us from rewriting legitimate
|
|
42
|
+
* JSON Schemas that happen to use `def` as a property name.
|
|
43
|
+
*/
|
|
44
|
+
const ZOD_KINDS: Record<string, true> = {
|
|
45
|
+
string: true,
|
|
46
|
+
number: true,
|
|
47
|
+
int: true,
|
|
48
|
+
boolean: true,
|
|
49
|
+
bigint: true,
|
|
50
|
+
null: true,
|
|
51
|
+
undefined: true,
|
|
52
|
+
void: true,
|
|
53
|
+
any: true,
|
|
54
|
+
unknown: true,
|
|
55
|
+
never: true,
|
|
56
|
+
date: true,
|
|
57
|
+
symbol: true,
|
|
58
|
+
nan: true,
|
|
59
|
+
enum: true,
|
|
60
|
+
literal: true,
|
|
61
|
+
object: true,
|
|
62
|
+
array: true,
|
|
63
|
+
tuple: true,
|
|
64
|
+
record: true,
|
|
65
|
+
map: true,
|
|
66
|
+
set: true,
|
|
67
|
+
union: true,
|
|
68
|
+
discriminatedUnion: true,
|
|
69
|
+
intersection: true,
|
|
70
|
+
lazy: true,
|
|
71
|
+
promise: true,
|
|
72
|
+
function: true,
|
|
73
|
+
file: true,
|
|
74
|
+
custom: true,
|
|
75
|
+
template_literal: true,
|
|
76
|
+
optional: true,
|
|
77
|
+
nullable: true,
|
|
78
|
+
default: true,
|
|
79
|
+
prefault: true,
|
|
80
|
+
catch: true,
|
|
81
|
+
pipe: true,
|
|
82
|
+
transform: true,
|
|
83
|
+
brand: true,
|
|
84
|
+
readonly: true,
|
|
85
|
+
success: true,
|
|
86
|
+
nonoptional: true,
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
const ZOD_SCALAR_TO_JSON_TYPE: Record<string, string> = {
|
|
90
|
+
string: "string",
|
|
91
|
+
number: "number",
|
|
92
|
+
int: "integer",
|
|
93
|
+
boolean: "boolean",
|
|
94
|
+
null: "null",
|
|
95
|
+
bigint: "string",
|
|
96
|
+
date: "string",
|
|
97
|
+
nan: "number",
|
|
98
|
+
};
|
|
99
|
+
|
|
100
|
+
const ZOD_NOISE_KEYS: Record<string, true> = {
|
|
101
|
+
def: true,
|
|
102
|
+
options: true,
|
|
103
|
+
_zod: true,
|
|
104
|
+
checks: true,
|
|
105
|
+
};
|
|
106
|
+
|
|
107
|
+
/**
|
|
108
|
+
* JSON Schema keywords where `null` is a legal value (literal payload positions).
|
|
109
|
+
* Anywhere else, a `null`-valued key is a meta-schema violation — Zod scalars
|
|
110
|
+
* leak `format: null`, `minLength: null`, etc. that we have to scrub.
|
|
111
|
+
*/
|
|
112
|
+
const KEYS_THAT_ACCEPT_NULL: Record<string, true> = {
|
|
113
|
+
default: true,
|
|
114
|
+
const: true,
|
|
115
|
+
examples: true,
|
|
116
|
+
};
|
|
117
|
+
|
|
118
|
+
function isZodLeak(node: JsonObject): boolean {
|
|
119
|
+
const def = node.def;
|
|
120
|
+
if (!isJsonObject(def)) return false;
|
|
121
|
+
const defType = def.type;
|
|
122
|
+
if (typeof defType !== "string" || !ZOD_KINDS[defType]) return false;
|
|
123
|
+
// Both surface and inner `.type` must agree — Zod always mirrors `_def.type`
|
|
124
|
+
// onto the instance, so this is a near-zero false-positive guard.
|
|
125
|
+
return node.type === defType;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
function inferTypeFromValues(values: readonly unknown[]): string {
|
|
129
|
+
if (values.length === 0) return "string";
|
|
130
|
+
const first = values[0];
|
|
131
|
+
if (typeof first === "number") return Number.isInteger(first) ? "integer" : "number";
|
|
132
|
+
if (typeof first === "boolean") return "boolean";
|
|
133
|
+
if (first === null) return "null";
|
|
134
|
+
return "string";
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
function unwrapInnerSchema(def: JsonObject): unknown {
|
|
138
|
+
// Zod uses different fields depending on the wrapper:
|
|
139
|
+
// optional/nullable/readonly/brand/default → `innerType`
|
|
140
|
+
// pipe → `in` (or `out`)
|
|
141
|
+
// lazy → `getter` (a function — gone after JSON.stringify); fall back to {}
|
|
142
|
+
return def.innerType ?? def.in ?? def.out ?? def.schema ?? def.element ?? {};
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
function copyWithoutNoise(node: JsonObject): JsonObject {
|
|
146
|
+
const out: JsonObject = {};
|
|
147
|
+
for (const key in node) {
|
|
148
|
+
if (ZOD_NOISE_KEYS[key]) continue;
|
|
149
|
+
const value = node[key];
|
|
150
|
+
if (value === null && !KEYS_THAT_ACCEPT_NULL[key]) continue;
|
|
151
|
+
out[key] = value;
|
|
152
|
+
}
|
|
153
|
+
return out;
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
function rewriteZodNode(node: JsonObject, seen: WeakSet<object>): unknown {
|
|
157
|
+
const def = node.def as JsonObject;
|
|
158
|
+
const kind = def.type as string;
|
|
159
|
+
|
|
160
|
+
switch (kind) {
|
|
161
|
+
case "enum": {
|
|
162
|
+
// Prefer node.options (array form Zod exposes) → def.entries values →
|
|
163
|
+
// object-shaped node.enum values. All three carry the same data.
|
|
164
|
+
const optionsArray = Array.isArray(node.options) ? (node.options as unknown[]) : null;
|
|
165
|
+
const entries = isJsonObject(def.entries) ? Object.values(def.entries) : null;
|
|
166
|
+
const enumObj = isJsonObject(node.enum) ? Object.values(node.enum) : null;
|
|
167
|
+
const values = optionsArray ?? entries ?? enumObj ?? [];
|
|
168
|
+
return { type: inferTypeFromValues(values), enum: values };
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
case "literal": {
|
|
172
|
+
const values = Array.isArray(def.values) ? (def.values as unknown[]) : [];
|
|
173
|
+
if (values.length === 1) {
|
|
174
|
+
return { const: values[0] };
|
|
175
|
+
}
|
|
176
|
+
if (values.length > 1) {
|
|
177
|
+
return { type: inferTypeFromValues(values), enum: values };
|
|
178
|
+
}
|
|
179
|
+
return {};
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
case "union":
|
|
183
|
+
case "discriminatedUnion": {
|
|
184
|
+
const arms = Array.isArray(def.options)
|
|
185
|
+
? (def.options as unknown[])
|
|
186
|
+
: Array.isArray(node.options)
|
|
187
|
+
? (node.options as unknown[])
|
|
188
|
+
: [];
|
|
189
|
+
return { anyOf: arms.map(x => walk(x, seen)) };
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
case "intersection": {
|
|
193
|
+
return {
|
|
194
|
+
allOf: [walk(def.left, seen), walk(def.right, seen)],
|
|
195
|
+
};
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
case "array": {
|
|
199
|
+
return { type: "array", items: walk(def.element, seen) };
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
case "set": {
|
|
203
|
+
const element = def.valueType ?? def.element;
|
|
204
|
+
return { type: "array", uniqueItems: true, items: walk(element, seen) };
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
case "tuple": {
|
|
208
|
+
const items = Array.isArray(def.items) ? (def.items as unknown[]) : [];
|
|
209
|
+
const out: JsonObject = { type: "array", prefixItems: items.map(x => walk(x, seen)) };
|
|
210
|
+
const rest = def.rest;
|
|
211
|
+
if (rest != null) out.items = walk(rest, seen);
|
|
212
|
+
return out;
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
case "record":
|
|
216
|
+
case "map": {
|
|
217
|
+
return { type: "object", additionalProperties: walk(def.valueType, seen) };
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
case "object": {
|
|
221
|
+
const shape = isJsonObject(def.shape) ? def.shape : ({} as JsonObject);
|
|
222
|
+
const properties: JsonObject = {};
|
|
223
|
+
const required: string[] = [];
|
|
224
|
+
for (const key in shape) {
|
|
225
|
+
const inner = walk(shape[key], seen);
|
|
226
|
+
properties[key] = inner;
|
|
227
|
+
if (!isOptionalEntry(shape[key])) required.push(key);
|
|
228
|
+
}
|
|
229
|
+
const out: JsonObject = { type: "object", properties };
|
|
230
|
+
if (required.length > 0) out.required = required;
|
|
231
|
+
return out;
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
case "nonoptional":
|
|
235
|
+
case "optional":
|
|
236
|
+
case "nullable":
|
|
237
|
+
case "default":
|
|
238
|
+
case "prefault":
|
|
239
|
+
case "catch":
|
|
240
|
+
case "readonly":
|
|
241
|
+
case "brand":
|
|
242
|
+
case "lazy":
|
|
243
|
+
case "pipe":
|
|
244
|
+
case "transform": {
|
|
245
|
+
const inner = walk(unwrapInnerSchema(def), seen);
|
|
246
|
+
if (kind === "nullable" && isJsonObject(inner)) {
|
|
247
|
+
if (typeof inner.type === "string") {
|
|
248
|
+
return { ...inner, type: [inner.type, "null"] };
|
|
249
|
+
}
|
|
250
|
+
if (Array.isArray(inner.type)) {
|
|
251
|
+
return (inner.type as string[]).includes("null")
|
|
252
|
+
? inner
|
|
253
|
+
: { ...inner, type: [...(inner.type as string[]), "null"] };
|
|
254
|
+
}
|
|
255
|
+
// anyOf / allOf / $ref shapes — no scalar `type` field
|
|
256
|
+
return { anyOf: [inner, { type: "null" }] };
|
|
257
|
+
}
|
|
258
|
+
return inner;
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
default: {
|
|
262
|
+
// Best-effort: drop the noise, map the kind to a JSON Schema type if
|
|
263
|
+
// we know one, otherwise drop `type` so the node validates as
|
|
264
|
+
// permissive.
|
|
265
|
+
const cleaned = copyWithoutNoise(node);
|
|
266
|
+
const mapped = ZOD_SCALAR_TO_JSON_TYPE[kind];
|
|
267
|
+
if (mapped) {
|
|
268
|
+
cleaned.type = mapped;
|
|
269
|
+
} else if (typeof cleaned.type === "string" && !VALID_JSON_SCHEMA_TYPES[cleaned.type]) {
|
|
270
|
+
delete cleaned.type;
|
|
271
|
+
}
|
|
272
|
+
// Object-shaped `enum` survives as a noise field — remove if present.
|
|
273
|
+
if (cleaned.enum !== undefined && !Array.isArray(cleaned.enum)) {
|
|
274
|
+
delete cleaned.enum;
|
|
275
|
+
}
|
|
276
|
+
return cleaned;
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
function isOptionalEntry(value: unknown): boolean {
|
|
282
|
+
if (!isJsonObject(value)) return false;
|
|
283
|
+
if (!isZodLeak(value)) return false;
|
|
284
|
+
const kind = (value.def as JsonObject).type;
|
|
285
|
+
return kind === "optional" || kind === "default" || kind === "prefault";
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
/**
|
|
289
|
+
* Walks a JSON value and rewrites every Zod-instance-shaped node into clean
|
|
290
|
+
* JSON Schema 2020-12. Identity-preserving when no rewrite fires. Tolerates
|
|
291
|
+
* self-referential graphs — a revisited node returns as-is.
|
|
292
|
+
*/
|
|
293
|
+
export function decontaminateZodInstance(value: unknown): unknown {
|
|
294
|
+
return walk(value, new WeakSet());
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
function walk(value: unknown, seen: WeakSet<object>): unknown {
|
|
298
|
+
if (Array.isArray(value)) {
|
|
299
|
+
if (seen.has(value)) return value;
|
|
300
|
+
seen.add(value);
|
|
301
|
+
let changed = false;
|
|
302
|
+
const out = value.map(entry => {
|
|
303
|
+
const rewritten = walk(entry, seen);
|
|
304
|
+
if (rewritten !== entry) changed = true;
|
|
305
|
+
return rewritten;
|
|
306
|
+
});
|
|
307
|
+
return changed ? out : value;
|
|
308
|
+
}
|
|
309
|
+
if (!isJsonObject(value)) return value;
|
|
310
|
+
if (seen.has(value)) return value;
|
|
311
|
+
seen.add(value);
|
|
312
|
+
|
|
313
|
+
if (isZodLeak(value)) {
|
|
314
|
+
// Rewrite the node itself, then recurse into the rewrite so any nested
|
|
315
|
+
// Zod-instance children get cleaned in the same pass.
|
|
316
|
+
const rewritten = rewriteZodNode(value, seen);
|
|
317
|
+
return rewritten === value ? value : walk(rewritten, seen);
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
// Plain JSON Schema node: recurse into children, preserving identity when
|
|
321
|
+
// nothing under us changed.
|
|
322
|
+
let changed = false;
|
|
323
|
+
const out: JsonObject = {};
|
|
324
|
+
for (const key in value) {
|
|
325
|
+
const child = value[key];
|
|
326
|
+
const rewritten = walk(child, seen);
|
|
327
|
+
if (rewritten !== child) changed = true;
|
|
328
|
+
out[key] = rewritten;
|
|
329
|
+
}
|
|
330
|
+
return changed ? out : value;
|
|
331
|
+
}
|