@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.
Files changed (349) hide show
  1. package/CHANGELOG.md +2644 -0
  2. package/README.md +1181 -0
  3. package/dist/types/api-registry.d.ts +30 -0
  4. package/dist/types/auth-broker/client.d.ts +66 -0
  5. package/dist/types/auth-broker/index.d.ts +5 -0
  6. package/dist/types/auth-broker/refresher.d.ts +25 -0
  7. package/dist/types/auth-broker/remote-store.d.ts +96 -0
  8. package/dist/types/auth-broker/server.d.ts +32 -0
  9. package/dist/types/auth-broker/types.d.ts +105 -0
  10. package/dist/types/auth-broker/wire-schemas.d.ts +412 -0
  11. package/dist/types/auth-gateway/http.d.ts +39 -0
  12. package/dist/types/auth-gateway/index.d.ts +3 -0
  13. package/dist/types/auth-gateway/server.d.ts +17 -0
  14. package/dist/types/auth-gateway/types.d.ts +115 -0
  15. package/dist/types/auth-storage.d.ts +641 -0
  16. package/dist/types/cli.d.ts +2 -0
  17. package/dist/types/index.d.ts +49 -0
  18. package/dist/types/model-cache.d.ts +17 -0
  19. package/dist/types/model-manager.d.ts +62 -0
  20. package/dist/types/model-thinking.d.ts +71 -0
  21. package/dist/types/models.d.ts +12 -0
  22. package/dist/types/provider-details.d.ts +24 -0
  23. package/dist/types/provider-models/bundled-references.d.ts +4 -0
  24. package/dist/types/provider-models/descriptors.d.ts +48 -0
  25. package/dist/types/provider-models/google.d.ts +20 -0
  26. package/dist/types/provider-models/index.d.ts +5 -0
  27. package/dist/types/provider-models/ollama.d.ts +7 -0
  28. package/dist/types/provider-models/openai-compat.d.ts +237 -0
  29. package/dist/types/provider-models/special.d.ts +16 -0
  30. package/dist/types/providers/amazon-bedrock.d.ts +36 -0
  31. package/dist/types/providers/anthropic-messages-server-schema.d.ts +450 -0
  32. package/dist/types/providers/anthropic-messages-server.d.ts +17 -0
  33. package/dist/types/providers/anthropic.d.ts +188 -0
  34. package/dist/types/providers/aws-credentials.d.ts +43 -0
  35. package/dist/types/providers/aws-eventstream.d.ts +38 -0
  36. package/dist/types/providers/aws-sigv4.d.ts +55 -0
  37. package/dist/types/providers/azure-openai-responses.d.ts +15 -0
  38. package/dist/types/providers/cursor/gen/agent_pb.d.ts +13022 -0
  39. package/dist/types/providers/cursor.d.ts +42 -0
  40. package/dist/types/providers/error-message.d.ts +27 -0
  41. package/dist/types/providers/github-copilot-headers.d.ts +40 -0
  42. package/dist/types/providers/gitlab-duo.d.ts +27 -0
  43. package/dist/types/providers/google-auth.d.ts +24 -0
  44. package/dist/types/providers/google-gemini-cli.d.ts +72 -0
  45. package/dist/types/providers/google-gemini-headers.d.ts +18 -0
  46. package/dist/types/providers/google-shared.d.ts +163 -0
  47. package/dist/types/providers/google-types.d.ts +138 -0
  48. package/dist/types/providers/google-vertex.d.ts +7 -0
  49. package/dist/types/providers/google.d.ts +4 -0
  50. package/dist/types/providers/grammar.d.ts +1 -0
  51. package/dist/types/providers/kimi.d.ts +27 -0
  52. package/dist/types/providers/mock.d.ts +175 -0
  53. package/dist/types/providers/ollama.d.ts +6 -0
  54. package/dist/types/providers/openai-anthropic-shim.d.ts +31 -0
  55. package/dist/types/providers/openai-chat-server-schema.d.ts +814 -0
  56. package/dist/types/providers/openai-chat-server.d.ts +16 -0
  57. package/dist/types/providers/openai-codex/constants.d.ts +26 -0
  58. package/dist/types/providers/openai-codex/request-transformer.d.ts +49 -0
  59. package/dist/types/providers/openai-codex/response-handler.d.ts +17 -0
  60. package/dist/types/providers/openai-codex-responses.d.ts +67 -0
  61. package/dist/types/providers/openai-completions-compat.d.ts +25 -0
  62. package/dist/types/providers/openai-completions.d.ts +33 -0
  63. package/dist/types/providers/openai-responses-server-schema.d.ts +392 -0
  64. package/dist/types/providers/openai-responses-server.d.ts +17 -0
  65. package/dist/types/providers/openai-responses-shared.d.ts +89 -0
  66. package/dist/types/providers/openai-responses.d.ts +32 -0
  67. package/dist/types/providers/pi-native-client.d.ts +13 -0
  68. package/dist/types/providers/pi-native-server.d.ts +68 -0
  69. package/dist/types/providers/register-builtins.d.ts +31 -0
  70. package/dist/types/providers/synthetic.d.ts +26 -0
  71. package/dist/types/providers/transform-messages.d.ts +12 -0
  72. package/dist/types/providers/vision-guard.d.ts +8 -0
  73. package/dist/types/rate-limit-utils.d.ts +19 -0
  74. package/dist/types/stream.d.ts +24 -0
  75. package/dist/types/types.d.ts +746 -0
  76. package/dist/types/usage/claude.d.ts +3 -0
  77. package/dist/types/usage/gemini.d.ts +2 -0
  78. package/dist/types/usage/github-copilot.d.ts +7 -0
  79. package/dist/types/usage/google-antigravity.d.ts +2 -0
  80. package/dist/types/usage/kimi.d.ts +2 -0
  81. package/dist/types/usage/minimax-code.d.ts +2 -0
  82. package/dist/types/usage/openai-codex.d.ts +3 -0
  83. package/dist/types/usage/shared.d.ts +1 -0
  84. package/dist/types/usage/zai.d.ts +2 -0
  85. package/dist/types/usage.d.ts +258 -0
  86. package/dist/types/utils/abort.d.ts +19 -0
  87. package/dist/types/utils/anthropic-auth.d.ts +31 -0
  88. package/dist/types/utils/discovery/antigravity.d.ts +61 -0
  89. package/dist/types/utils/discovery/codex.d.ts +38 -0
  90. package/dist/types/utils/discovery/cursor.d.ts +23 -0
  91. package/dist/types/utils/discovery/gemini.d.ts +25 -0
  92. package/dist/types/utils/discovery/index.d.ts +4 -0
  93. package/dist/types/utils/discovery/openai-compatible.d.ts +72 -0
  94. package/dist/types/utils/event-stream.d.ts +28 -0
  95. package/dist/types/utils/fireworks-model-id.d.ts +10 -0
  96. package/dist/types/utils/foundry.d.ts +1 -0
  97. package/dist/types/utils/h2-fetch.d.ts +22 -0
  98. package/dist/types/utils/http-inspector.d.ts +31 -0
  99. package/dist/types/utils/idle-iterator.d.ts +67 -0
  100. package/dist/types/utils/json-parse.d.ts +10 -0
  101. package/dist/types/utils/oauth/alibaba-coding-plan.d.ts +18 -0
  102. package/dist/types/utils/oauth/anthropic.d.ts +22 -0
  103. package/dist/types/utils/oauth/api-key-login.d.ts +35 -0
  104. package/dist/types/utils/oauth/api-key-validation.d.ts +27 -0
  105. package/dist/types/utils/oauth/callback-server.d.ts +57 -0
  106. package/dist/types/utils/oauth/cerebras.d.ts +1 -0
  107. package/dist/types/utils/oauth/cloudflare-ai-gateway.d.ts +18 -0
  108. package/dist/types/utils/oauth/cursor.d.ts +15 -0
  109. package/dist/types/utils/oauth/deepseek.d.ts +10 -0
  110. package/dist/types/utils/oauth/firepass.d.ts +1 -0
  111. package/dist/types/utils/oauth/fireworks.d.ts +1 -0
  112. package/dist/types/utils/oauth/github-copilot.d.ts +38 -0
  113. package/dist/types/utils/oauth/gitlab-duo.d.ts +3 -0
  114. package/dist/types/utils/oauth/google-antigravity.d.ts +11 -0
  115. package/dist/types/utils/oauth/google-gemini-cli.d.ts +10 -0
  116. package/dist/types/utils/oauth/google-oauth-shared.d.ts +28 -0
  117. package/dist/types/utils/oauth/huggingface.d.ts +19 -0
  118. package/dist/types/utils/oauth/index.d.ts +38 -0
  119. package/dist/types/utils/oauth/kagi.d.ts +17 -0
  120. package/dist/types/utils/oauth/kilo.d.ts +5 -0
  121. package/dist/types/utils/oauth/kimi.d.ts +21 -0
  122. package/dist/types/utils/oauth/litellm.d.ts +18 -0
  123. package/dist/types/utils/oauth/lm-studio.d.ts +17 -0
  124. package/dist/types/utils/oauth/minimax-code.d.ts +28 -0
  125. package/dist/types/utils/oauth/moonshot.d.ts +1 -0
  126. package/dist/types/utils/oauth/nanogpt.d.ts +1 -0
  127. package/dist/types/utils/oauth/nvidia.d.ts +18 -0
  128. package/dist/types/utils/oauth/ollama-cloud.d.ts +2 -0
  129. package/dist/types/utils/oauth/ollama.d.ts +18 -0
  130. package/dist/types/utils/oauth/openai-codex.d.ts +21 -0
  131. package/dist/types/utils/oauth/opencode.d.ts +18 -0
  132. package/dist/types/utils/oauth/parallel.d.ts +17 -0
  133. package/dist/types/utils/oauth/perplexity.d.ts +9 -0
  134. package/dist/types/utils/oauth/pkce.d.ts +8 -0
  135. package/dist/types/utils/oauth/qianfan.d.ts +17 -0
  136. package/dist/types/utils/oauth/qwen-portal.d.ts +19 -0
  137. package/dist/types/utils/oauth/synthetic.d.ts +1 -0
  138. package/dist/types/utils/oauth/tavily.d.ts +17 -0
  139. package/dist/types/utils/oauth/together.d.ts +1 -0
  140. package/dist/types/utils/oauth/types.d.ts +44 -0
  141. package/dist/types/utils/oauth/venice.d.ts +18 -0
  142. package/dist/types/utils/oauth/vercel-ai-gateway.d.ts +18 -0
  143. package/dist/types/utils/oauth/vllm.d.ts +16 -0
  144. package/dist/types/utils/oauth/xiaomi.d.ts +19 -0
  145. package/dist/types/utils/oauth/zai.d.ts +18 -0
  146. package/dist/types/utils/oauth/zenmux.d.ts +1 -0
  147. package/dist/types/utils/overflow.d.ts +54 -0
  148. package/dist/types/utils/parse-bind.d.ts +23 -0
  149. package/dist/types/utils/provider-response.d.ts +3 -0
  150. package/dist/types/utils/retry-after.d.ts +3 -0
  151. package/dist/types/utils/retry.d.ts +26 -0
  152. package/dist/types/utils/schema/adapt.d.ts +24 -0
  153. package/dist/types/utils/schema/compatibility.d.ts +30 -0
  154. package/dist/types/utils/schema/dereference.d.ts +11 -0
  155. package/dist/types/utils/schema/draft.d.ts +10 -0
  156. package/dist/types/utils/schema/equality.d.ts +4 -0
  157. package/dist/types/utils/schema/fields.d.ts +49 -0
  158. package/dist/types/utils/schema/index.d.ts +13 -0
  159. package/dist/types/utils/schema/json-schema-validator.d.ts +12 -0
  160. package/dist/types/utils/schema/meta-validator.d.ts +2 -0
  161. package/dist/types/utils/schema/normalize.d.ts +93 -0
  162. package/dist/types/utils/schema/spill.d.ts +8 -0
  163. package/dist/types/utils/schema/stamps.d.ts +25 -0
  164. package/dist/types/utils/schema/types.d.ts +4 -0
  165. package/dist/types/utils/schema/wire.d.ts +54 -0
  166. package/dist/types/utils/schema/zod-decontaminate.d.ts +31 -0
  167. package/dist/types/utils/sse-debug.d.ts +10 -0
  168. package/dist/types/utils/tool-call-healing.d.ts +71 -0
  169. package/dist/types/utils/tool-choice.d.ts +50 -0
  170. package/dist/types/utils/validation.d.ts +17 -0
  171. package/dist/types/utils.d.ts +28 -0
  172. package/package.json +146 -0
  173. package/src/api-registry.ts +96 -0
  174. package/src/auth-broker/client.ts +358 -0
  175. package/src/auth-broker/index.ts +5 -0
  176. package/src/auth-broker/refresher.ts +127 -0
  177. package/src/auth-broker/remote-store.ts +623 -0
  178. package/src/auth-broker/server.ts +644 -0
  179. package/src/auth-broker/types.ts +127 -0
  180. package/src/auth-broker/wire-schemas.ts +200 -0
  181. package/src/auth-gateway/http.ts +194 -0
  182. package/src/auth-gateway/index.ts +3 -0
  183. package/src/auth-gateway/server.ts +717 -0
  184. package/src/auth-gateway/types.ts +134 -0
  185. package/src/auth-storage.ts +4104 -0
  186. package/src/cli.ts +262 -0
  187. package/src/index.ts +54 -0
  188. package/src/model-cache.ts +129 -0
  189. package/src/model-manager.ts +450 -0
  190. package/src/model-thinking.ts +691 -0
  191. package/src/models.json +73853 -0
  192. package/src/models.json.d.ts +9 -0
  193. package/src/models.ts +56 -0
  194. package/src/prompts/turn-aborted-guidance.md +4 -0
  195. package/src/provider-details.ts +90 -0
  196. package/src/provider-models/bundled-references.ts +38 -0
  197. package/src/provider-models/descriptors.ts +308 -0
  198. package/src/provider-models/google.ts +91 -0
  199. package/src/provider-models/index.ts +5 -0
  200. package/src/provider-models/ollama.ts +153 -0
  201. package/src/provider-models/openai-compat.ts +2275 -0
  202. package/src/provider-models/special.ts +67 -0
  203. package/src/providers/amazon-bedrock.ts +849 -0
  204. package/src/providers/anthropic-messages-server-schema.ts +229 -0
  205. package/src/providers/anthropic-messages-server.ts +677 -0
  206. package/src/providers/anthropic.ts +2696 -0
  207. package/src/providers/aws-credentials.ts +501 -0
  208. package/src/providers/aws-eventstream.ts +185 -0
  209. package/src/providers/aws-sigv4.ts +218 -0
  210. package/src/providers/azure-openai-responses.ts +337 -0
  211. package/src/providers/cursor/gen/agent_pb.ts +15274 -0
  212. package/src/providers/cursor/proto/agent.proto +3526 -0
  213. package/src/providers/cursor/proto/buf.gen.yaml +6 -0
  214. package/src/providers/cursor/proto/buf.yaml +17 -0
  215. package/src/providers/cursor.ts +2561 -0
  216. package/src/providers/error-message.ts +21 -0
  217. package/src/providers/github-copilot-headers.ts +140 -0
  218. package/src/providers/gitlab-duo.ts +372 -0
  219. package/src/providers/google-auth.ts +252 -0
  220. package/src/providers/google-gemini-cli.ts +795 -0
  221. package/src/providers/google-gemini-headers.ts +41 -0
  222. package/src/providers/google-shared.ts +902 -0
  223. package/src/providers/google-types.ts +167 -0
  224. package/src/providers/google-vertex.ts +88 -0
  225. package/src/providers/google.ts +41 -0
  226. package/src/providers/grammar.ts +70 -0
  227. package/src/providers/kimi.ts +52 -0
  228. package/src/providers/mock.ts +500 -0
  229. package/src/providers/ollama.ts +544 -0
  230. package/src/providers/openai-anthropic-shim.ts +138 -0
  231. package/src/providers/openai-chat-server-schema.ts +243 -0
  232. package/src/providers/openai-chat-server.ts +628 -0
  233. package/src/providers/openai-codex/constants.ts +43 -0
  234. package/src/providers/openai-codex/request-transformer.ts +161 -0
  235. package/src/providers/openai-codex/response-handler.ts +81 -0
  236. package/src/providers/openai-codex-responses.ts +2598 -0
  237. package/src/providers/openai-completions-compat.ts +279 -0
  238. package/src/providers/openai-completions.ts +1853 -0
  239. package/src/providers/openai-responses-server-schema.ts +290 -0
  240. package/src/providers/openai-responses-server.ts +1183 -0
  241. package/src/providers/openai-responses-shared.ts +800 -0
  242. package/src/providers/openai-responses.ts +621 -0
  243. package/src/providers/pi-native-client.ts +228 -0
  244. package/src/providers/pi-native-server.ts +210 -0
  245. package/src/providers/register-builtins.ts +412 -0
  246. package/src/providers/synthetic.ts +50 -0
  247. package/src/providers/transform-messages.ts +309 -0
  248. package/src/providers/vision-guard.ts +31 -0
  249. package/src/rate-limit-utils.ts +84 -0
  250. package/src/stream.ts +895 -0
  251. package/src/types.ts +884 -0
  252. package/src/usage/claude.ts +431 -0
  253. package/src/usage/gemini.ts +250 -0
  254. package/src/usage/github-copilot.ts +421 -0
  255. package/src/usage/google-antigravity.ts +201 -0
  256. package/src/usage/kimi.ts +271 -0
  257. package/src/usage/minimax-code.ts +31 -0
  258. package/src/usage/openai-codex.ts +503 -0
  259. package/src/usage/shared.ts +10 -0
  260. package/src/usage/zai.ts +247 -0
  261. package/src/usage.ts +183 -0
  262. package/src/utils/abort.ts +51 -0
  263. package/src/utils/anthropic-auth.ts +87 -0
  264. package/src/utils/discovery/antigravity.ts +261 -0
  265. package/src/utils/discovery/codex.ts +371 -0
  266. package/src/utils/discovery/cursor.ts +306 -0
  267. package/src/utils/discovery/gemini.ts +248 -0
  268. package/src/utils/discovery/index.ts +4 -0
  269. package/src/utils/discovery/openai-compatible.ts +224 -0
  270. package/src/utils/event-stream.ts +142 -0
  271. package/src/utils/fireworks-model-id.ts +30 -0
  272. package/src/utils/foundry.ts +8 -0
  273. package/src/utils/h2-fetch.ts +60 -0
  274. package/src/utils/http-inspector.ts +176 -0
  275. package/src/utils/idle-iterator.ts +250 -0
  276. package/src/utils/json-parse.ts +148 -0
  277. package/src/utils/oauth/alibaba-coding-plan.ts +59 -0
  278. package/src/utils/oauth/anthropic.ts +200 -0
  279. package/src/utils/oauth/api-key-login.ts +87 -0
  280. package/src/utils/oauth/api-key-validation.ts +92 -0
  281. package/src/utils/oauth/callback-server.ts +276 -0
  282. package/src/utils/oauth/cerebras.ts +16 -0
  283. package/src/utils/oauth/cloudflare-ai-gateway.ts +48 -0
  284. package/src/utils/oauth/cursor.ts +157 -0
  285. package/src/utils/oauth/deepseek.ts +53 -0
  286. package/src/utils/oauth/firepass.ts +24 -0
  287. package/src/utils/oauth/fireworks.ts +15 -0
  288. package/src/utils/oauth/github-copilot.ts +362 -0
  289. package/src/utils/oauth/gitlab-duo.ts +123 -0
  290. package/src/utils/oauth/google-antigravity.ts +200 -0
  291. package/src/utils/oauth/google-gemini-cli.ts +256 -0
  292. package/src/utils/oauth/google-oauth-shared.ts +110 -0
  293. package/src/utils/oauth/huggingface.ts +62 -0
  294. package/src/utils/oauth/index.ts +444 -0
  295. package/src/utils/oauth/kagi.ts +47 -0
  296. package/src/utils/oauth/kilo.ts +87 -0
  297. package/src/utils/oauth/kimi.ts +254 -0
  298. package/src/utils/oauth/litellm.ts +47 -0
  299. package/src/utils/oauth/lm-studio.ts +38 -0
  300. package/src/utils/oauth/minimax-code.ts +78 -0
  301. package/src/utils/oauth/moonshot.ts +16 -0
  302. package/src/utils/oauth/nanogpt.ts +15 -0
  303. package/src/utils/oauth/nvidia.ts +70 -0
  304. package/src/utils/oauth/oauth.html +199 -0
  305. package/src/utils/oauth/ollama-cloud.ts +28 -0
  306. package/src/utils/oauth/ollama.ts +47 -0
  307. package/src/utils/oauth/openai-codex.ts +299 -0
  308. package/src/utils/oauth/opencode.ts +49 -0
  309. package/src/utils/oauth/parallel.ts +46 -0
  310. package/src/utils/oauth/perplexity.ts +206 -0
  311. package/src/utils/oauth/pkce.ts +18 -0
  312. package/src/utils/oauth/qianfan.ts +58 -0
  313. package/src/utils/oauth/qwen-portal.ts +60 -0
  314. package/src/utils/oauth/synthetic.ts +16 -0
  315. package/src/utils/oauth/tavily.ts +46 -0
  316. package/src/utils/oauth/together.ts +16 -0
  317. package/src/utils/oauth/types.ts +94 -0
  318. package/src/utils/oauth/venice.ts +59 -0
  319. package/src/utils/oauth/vercel-ai-gateway.ts +47 -0
  320. package/src/utils/oauth/vllm.ts +40 -0
  321. package/src/utils/oauth/xiaomi.ts +137 -0
  322. package/src/utils/oauth/zai.ts +60 -0
  323. package/src/utils/oauth/zenmux.ts +15 -0
  324. package/src/utils/overflow.ts +137 -0
  325. package/src/utils/parse-bind.ts +54 -0
  326. package/src/utils/provider-response.ts +30 -0
  327. package/src/utils/retry-after.ts +110 -0
  328. package/src/utils/retry.ts +54 -0
  329. package/src/utils/schema/CONSTRAINTS.md +164 -0
  330. package/src/utils/schema/adapt.ts +36 -0
  331. package/src/utils/schema/compatibility.ts +435 -0
  332. package/src/utils/schema/dereference.ts +98 -0
  333. package/src/utils/schema/draft.ts +341 -0
  334. package/src/utils/schema/equality.ts +97 -0
  335. package/src/utils/schema/fields.ts +190 -0
  336. package/src/utils/schema/index.ts +13 -0
  337. package/src/utils/schema/json-schema-validator.ts +577 -0
  338. package/src/utils/schema/meta-validator.ts +167 -0
  339. package/src/utils/schema/normalize.ts +1588 -0
  340. package/src/utils/schema/spill.ts +43 -0
  341. package/src/utils/schema/stamps.ts +97 -0
  342. package/src/utils/schema/types.ts +11 -0
  343. package/src/utils/schema/wire.ts +213 -0
  344. package/src/utils/schema/zod-decontaminate.ts +331 -0
  345. package/src/utils/sse-debug.ts +289 -0
  346. package/src/utils/tool-call-healing.ts +271 -0
  347. package/src/utils/tool-choice.ts +99 -0
  348. package/src/utils/validation.ts +1019 -0
  349. 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
+ }