@orpc/zod 0.0.0-next.9adcd05 → 0.0.0-next.9af314c
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/README.md +31 -23
- package/dist/index.d.mts +20 -11
- package/dist/index.d.ts +20 -11
- package/dist/index.mjs +79 -43
- package/dist/zod4/index.d.mts +314 -0
- package/dist/zod4/index.d.ts +314 -0
- package/dist/zod4/index.mjs +680 -0
- package/package.json +13 -7
package/README.md
CHANGED
@@ -21,28 +21,24 @@
|
|
21
21
|
|
22
22
|
<h3 align="center">Typesafe APIs Made Simple 🪄</h3>
|
23
23
|
|
24
|
-
**oRPC is a powerful combination of RPC and OpenAPI**, makes it easy to build APIs that are end-to-end type-safe and adhere to OpenAPI standards
|
24
|
+
**oRPC is a powerful combination of RPC and OpenAPI**, makes it easy to build APIs that are end-to-end type-safe and adhere to OpenAPI standards
|
25
25
|
|
26
26
|
---
|
27
27
|
|
28
28
|
## Highlights
|
29
29
|
|
30
|
-
-
|
31
|
-
-
|
32
|
-
-
|
33
|
-
-
|
34
|
-
-
|
35
|
-
-
|
36
|
-
-
|
37
|
-
-
|
38
|
-
-
|
39
|
-
-
|
40
|
-
-
|
41
|
-
-
|
42
|
-
- **Reusability 🔄**: Write once and reuse your code across multiple purposes effortlessly.
|
43
|
-
- **Extendability 🔌**: Easily enhance oRPC with plugins, middleware, and interceptors.
|
44
|
-
- **Reliability 🛡️**: Well-tested, fully TypeScript, production-ready, and MIT licensed for peace of mind.
|
45
|
-
- **Simplicity 💡**: Enjoy straightforward, clean code with no hidden magic.
|
30
|
+
- **🔗 End-to-End Type Safety**: Ensure type-safe inputs, outputs, and errors from client to server.
|
31
|
+
- **📘 First-Class OpenAPI**: Built-in support that fully adheres to the OpenAPI standard.
|
32
|
+
- **📝 Contract-First Development**: Optionally define your API contract before implementation.
|
33
|
+
- **⚙️ Framework Integrations**: Seamlessly integrate with TanStack Query (React, Vue, Solid, Svelte, Angular), Pinia Colada, and more.
|
34
|
+
- **🚀 Server Actions**: Fully compatible with React Server Actions on Next.js, TanStack Start, and other platforms.
|
35
|
+
- **🔠 Standard Schema Support**: Works out of the box with Zod, Valibot, ArkType, and other schema validators.
|
36
|
+
- **🗃️ Native Types**: Supports native types like Date, File, Blob, BigInt, URL, and more.
|
37
|
+
- **⏱️ Lazy Router**: Enhance cold start times with our lazy routing feature.
|
38
|
+
- **📡 SSE & Streaming**: Enjoy full type-safe support for SSE and streaming.
|
39
|
+
- **🌍 Multi-Runtime Support**: Fast and lightweight on Cloudflare, Deno, Bun, Node.js, and beyond.
|
40
|
+
- **🔌 Extendability**: Easily extend functionality with plugins, middleware, and interceptors.
|
41
|
+
- **🛡️ Reliability**: Well-tested, TypeScript-based, production-ready, and MIT licensed.
|
46
42
|
|
47
43
|
## Documentation
|
48
44
|
|
@@ -53,11 +49,15 @@ You can find the full documentation [here](https://orpc.unnoq.com).
|
|
53
49
|
- [@orpc/contract](https://www.npmjs.com/package/@orpc/contract): Build your API contract.
|
54
50
|
- [@orpc/server](https://www.npmjs.com/package/@orpc/server): Build your API or implement API contract.
|
55
51
|
- [@orpc/client](https://www.npmjs.com/package/@orpc/client): Consume your API on the client with type-safety.
|
56
|
-
- [@orpc/react-query](https://www.npmjs.com/package/@orpc/react-query): Integration with [React Query](https://tanstack.com/query/latest/docs/framework/react/overview).
|
57
|
-
- [@orpc/vue-query](https://www.npmjs.com/package/@orpc/vue-query): Integration with [Vue Query](https://tanstack.com/query/latest/docs/framework/vue/overview).
|
58
|
-
- [@orpc/vue-colada](https://www.npmjs.com/package/@orpc/vue-colada): Integration with [Pinia Colada](https://pinia-colada.esm.dev/).
|
59
52
|
- [@orpc/openapi](https://www.npmjs.com/package/@orpc/openapi): Generate OpenAPI specs and handle OpenAPI requests.
|
53
|
+
- [@orpc/nest](https://www.npmjs.com/package/@orpc/nest): Deeply integrate oRPC with [NestJS](https://nestjs.com/).
|
54
|
+
- [@orpc/react](https://www.npmjs.com/package/@orpc/react): Utilities for integrating oRPC with React and React Server Actions.
|
55
|
+
- [@orpc/tanstack-query](https://www.npmjs.com/package/@orpc/tanstack-query): [TanStack Query](https://tanstack.com/query/latest) integration.
|
56
|
+
- [@orpc/vue-colada](https://www.npmjs.com/package/@orpc/vue-colada): Integration with [Pinia Colada](https://pinia-colada.esm.dev/).
|
57
|
+
- [@orpc/hey-api](https://www.npmjs.com/package/@orpc/hey-api): [Hey API](https://heyapi.dev/) integration.
|
60
58
|
- [@orpc/zod](https://www.npmjs.com/package/@orpc/zod): More schemas that [Zod](https://zod.dev/) doesn't support yet.
|
59
|
+
- [@orpc/valibot](https://www.npmjs.com/package/@orpc/valibot): OpenAPI spec generation from [Valibot](https://valibot.dev/).
|
60
|
+
- [@orpc/arktype](https://www.npmjs.com/package/@orpc/arktype): OpenAPI spec generation from [ArkType](https://arktype.io/).
|
61
61
|
|
62
62
|
## `@orpc/zod`
|
63
63
|
|
@@ -72,7 +72,7 @@ More schemas that [Zod](https://zod.dev/) doesn't support yet, and provides `Zod
|
|
72
72
|
|
73
73
|
```ts
|
74
74
|
import { oz } from '@orpc/zod'
|
75
|
-
import { z } from 'zod'
|
75
|
+
import { z } from 'zod/v3'
|
76
76
|
|
77
77
|
const Example = z.object({
|
78
78
|
url: oz.url(),
|
@@ -86,7 +86,7 @@ const Example = z.object({
|
|
86
86
|
|
87
87
|
```ts
|
88
88
|
import { OpenAPIGenerator } from '@orpc/openapi'
|
89
|
-
import { ZodToJsonSchemaConverter } from '@orpc/zod'
|
89
|
+
import { ZodToJsonSchemaConverter } from '@orpc/zod/zod4'
|
90
90
|
|
91
91
|
const openAPIGenerator = new OpenAPIGenerator({
|
92
92
|
schemaConverters: [new ZodToJsonSchemaConverter()],
|
@@ -111,7 +111,7 @@ const specFromRouter = await openAPIGenerator.generate(router, {
|
|
111
111
|
|
112
112
|
```ts
|
113
113
|
import { oz } from '@orpc/zod'
|
114
|
-
import
|
114
|
+
import * as z from 'zod'
|
115
115
|
|
116
116
|
const InputSchema = oz.openapi(
|
117
117
|
z.object({
|
@@ -127,6 +127,14 @@ const InputSchema = oz.openapi(
|
|
127
127
|
)
|
128
128
|
```
|
129
129
|
|
130
|
+
## Sponsors
|
131
|
+
|
132
|
+
<p align="center">
|
133
|
+
<a href="https://cdn.jsdelivr.net/gh/unnoq/unnoq/sponsors.svg">
|
134
|
+
<img src='https://cdn.jsdelivr.net/gh/unnoq/unnoq/sponsors.svg'/>
|
135
|
+
</a>
|
136
|
+
</p>
|
137
|
+
|
130
138
|
## License
|
131
139
|
|
132
140
|
Distributed under the MIT License. See [LICENSE](https://github.com/unnoq/orpc/blob/main/LICENSE) for more information.
|
package/dist/index.d.mts
CHANGED
@@ -1,8 +1,7 @@
|
|
1
1
|
import { JSONSchema, ConditionalSchemaConverter, SchemaConvertOptions } from '@orpc/openapi';
|
2
|
-
import { ZodTypeAny, input, output, ZodTypeDef, CustomErrorParams, ZodType, ZodEffects } from 'zod';
|
2
|
+
import { ZodTypeAny, input, output, ZodTypeDef, CustomErrorParams, ZodType, ZodEffects } from 'zod/v3';
|
3
3
|
import { Context } from '@orpc/server';
|
4
|
-
import {
|
5
|
-
import { StandardHandlerOptions } from '@orpc/server/standard';
|
4
|
+
import { StandardHandlerPlugin, StandardHandlerOptions } from '@orpc/server/standard';
|
6
5
|
import { AnySchema } from '@orpc/contract';
|
7
6
|
|
8
7
|
declare function getCustomJsonSchema(def: ZodTypeDef, options: {
|
@@ -35,40 +34,49 @@ declare function regexp(params?: string | CustomParams | ((input: unknown) => Cu
|
|
35
34
|
|
36
35
|
declare function url(params?: string | CustomParams | ((input: unknown) => CustomParams)): ZodType<URL, ZodTypeDef, URL>;
|
37
36
|
|
38
|
-
declare class ZodSmartCoercionPlugin<TContext extends Context> implements
|
37
|
+
declare class ZodSmartCoercionPlugin<TContext extends Context> implements StandardHandlerPlugin<TContext> {
|
39
38
|
init(options: StandardHandlerOptions<TContext>): void;
|
40
39
|
}
|
41
40
|
|
42
41
|
interface ZodToJsonSchemaOptions {
|
43
42
|
/**
|
44
|
-
* Max depth of lazy type
|
43
|
+
* Max depth of lazy type
|
45
44
|
*
|
46
|
-
* Used `{}` when
|
45
|
+
* Used `{}` when exceed max depth
|
47
46
|
*
|
48
47
|
* @default 3
|
49
48
|
*/
|
50
49
|
maxLazyDepth?: number;
|
51
50
|
/**
|
52
|
-
*
|
51
|
+
* Max depth of nested types
|
53
52
|
*
|
54
|
-
*
|
53
|
+
* Used anyJsonSchema (`{}`) when exceed max depth
|
54
|
+
*
|
55
|
+
* @default 10
|
55
56
|
*/
|
56
|
-
|
57
|
+
maxStructureDepth?: number;
|
57
58
|
/**
|
58
59
|
* The schema to be used to represent the any | unknown type.
|
59
60
|
*
|
60
61
|
* @default { }
|
61
62
|
*/
|
62
63
|
anyJsonSchema?: Exclude<JSONSchema, boolean>;
|
64
|
+
/**
|
65
|
+
* The schema to be used when the Zod schema is unsupported.
|
66
|
+
*
|
67
|
+
* @default { not: {} }
|
68
|
+
*/
|
69
|
+
unsupportedJsonSchema?: Exclude<JSONSchema, boolean>;
|
63
70
|
}
|
64
71
|
declare class ZodToJsonSchemaConverter implements ConditionalSchemaConverter {
|
65
72
|
#private;
|
66
73
|
private readonly maxLazyDepth;
|
74
|
+
private readonly maxStructureDepth;
|
67
75
|
private readonly unsupportedJsonSchema;
|
68
76
|
private readonly anyJsonSchema;
|
69
77
|
constructor(options?: ZodToJsonSchemaOptions);
|
70
78
|
condition(schema: AnySchema | undefined): boolean;
|
71
|
-
convert(schema: AnySchema | undefined, options: SchemaConvertOptions, lazyDepth?: number, isHandledCustomJSONSchema?: boolean, isHandledZodDescription?: boolean): [required: boolean, jsonSchema: Exclude<JSONSchema, boolean>];
|
79
|
+
convert(schema: AnySchema | undefined, options: SchemaConvertOptions, lazyDepth?: number, isHandledCustomJSONSchema?: boolean, isHandledZodDescription?: boolean, structureDepth?: number): [required: boolean, jsonSchema: Exclude<JSONSchema, boolean>];
|
72
80
|
}
|
73
81
|
|
74
82
|
declare const oz: {
|
@@ -79,4 +87,5 @@ declare const oz: {
|
|
79
87
|
openapi: typeof customJsonSchema;
|
80
88
|
};
|
81
89
|
|
82
|
-
export {
|
90
|
+
export { ZodSmartCoercionPlugin, ZodToJsonSchemaConverter, blob, composeParams, customJsonSchema, file, getCustomJsonSchema, getCustomZodDef, oz, regexp, setCustomZodDef, url };
|
91
|
+
export type { CustomParams, CustomZodDef, ZodToJsonSchemaOptions };
|
package/dist/index.d.ts
CHANGED
@@ -1,8 +1,7 @@
|
|
1
1
|
import { JSONSchema, ConditionalSchemaConverter, SchemaConvertOptions } from '@orpc/openapi';
|
2
|
-
import { ZodTypeAny, input, output, ZodTypeDef, CustomErrorParams, ZodType, ZodEffects } from 'zod';
|
2
|
+
import { ZodTypeAny, input, output, ZodTypeDef, CustomErrorParams, ZodType, ZodEffects } from 'zod/v3';
|
3
3
|
import { Context } from '@orpc/server';
|
4
|
-
import {
|
5
|
-
import { StandardHandlerOptions } from '@orpc/server/standard';
|
4
|
+
import { StandardHandlerPlugin, StandardHandlerOptions } from '@orpc/server/standard';
|
6
5
|
import { AnySchema } from '@orpc/contract';
|
7
6
|
|
8
7
|
declare function getCustomJsonSchema(def: ZodTypeDef, options: {
|
@@ -35,40 +34,49 @@ declare function regexp(params?: string | CustomParams | ((input: unknown) => Cu
|
|
35
34
|
|
36
35
|
declare function url(params?: string | CustomParams | ((input: unknown) => CustomParams)): ZodType<URL, ZodTypeDef, URL>;
|
37
36
|
|
38
|
-
declare class ZodSmartCoercionPlugin<TContext extends Context> implements
|
37
|
+
declare class ZodSmartCoercionPlugin<TContext extends Context> implements StandardHandlerPlugin<TContext> {
|
39
38
|
init(options: StandardHandlerOptions<TContext>): void;
|
40
39
|
}
|
41
40
|
|
42
41
|
interface ZodToJsonSchemaOptions {
|
43
42
|
/**
|
44
|
-
* Max depth of lazy type
|
43
|
+
* Max depth of lazy type
|
45
44
|
*
|
46
|
-
* Used `{}` when
|
45
|
+
* Used `{}` when exceed max depth
|
47
46
|
*
|
48
47
|
* @default 3
|
49
48
|
*/
|
50
49
|
maxLazyDepth?: number;
|
51
50
|
/**
|
52
|
-
*
|
51
|
+
* Max depth of nested types
|
53
52
|
*
|
54
|
-
*
|
53
|
+
* Used anyJsonSchema (`{}`) when exceed max depth
|
54
|
+
*
|
55
|
+
* @default 10
|
55
56
|
*/
|
56
|
-
|
57
|
+
maxStructureDepth?: number;
|
57
58
|
/**
|
58
59
|
* The schema to be used to represent the any | unknown type.
|
59
60
|
*
|
60
61
|
* @default { }
|
61
62
|
*/
|
62
63
|
anyJsonSchema?: Exclude<JSONSchema, boolean>;
|
64
|
+
/**
|
65
|
+
* The schema to be used when the Zod schema is unsupported.
|
66
|
+
*
|
67
|
+
* @default { not: {} }
|
68
|
+
*/
|
69
|
+
unsupportedJsonSchema?: Exclude<JSONSchema, boolean>;
|
63
70
|
}
|
64
71
|
declare class ZodToJsonSchemaConverter implements ConditionalSchemaConverter {
|
65
72
|
#private;
|
66
73
|
private readonly maxLazyDepth;
|
74
|
+
private readonly maxStructureDepth;
|
67
75
|
private readonly unsupportedJsonSchema;
|
68
76
|
private readonly anyJsonSchema;
|
69
77
|
constructor(options?: ZodToJsonSchemaOptions);
|
70
78
|
condition(schema: AnySchema | undefined): boolean;
|
71
|
-
convert(schema: AnySchema | undefined, options: SchemaConvertOptions, lazyDepth?: number, isHandledCustomJSONSchema?: boolean, isHandledZodDescription?: boolean): [required: boolean, jsonSchema: Exclude<JSONSchema, boolean>];
|
79
|
+
convert(schema: AnySchema | undefined, options: SchemaConvertOptions, lazyDepth?: number, isHandledCustomJSONSchema?: boolean, isHandledZodDescription?: boolean, structureDepth?: number): [required: boolean, jsonSchema: Exclude<JSONSchema, boolean>];
|
72
80
|
}
|
73
81
|
|
74
82
|
declare const oz: {
|
@@ -79,4 +87,5 @@ declare const oz: {
|
|
79
87
|
openapi: typeof customJsonSchema;
|
80
88
|
};
|
81
89
|
|
82
|
-
export {
|
90
|
+
export { ZodSmartCoercionPlugin, ZodToJsonSchemaConverter, blob, composeParams, customJsonSchema, file, getCustomJsonSchema, getCustomZodDef, oz, regexp, setCustomZodDef, url };
|
91
|
+
export type { CustomParams, CustomZodDef, ZodToJsonSchemaOptions };
|
package/dist/index.mjs
CHANGED
@@ -1,6 +1,7 @@
|
|
1
|
-
import { custom, ZodFirstPartyTypeKind } from 'zod';
|
1
|
+
import { custom, ZodFirstPartyTypeKind } from 'zod/v3';
|
2
2
|
import wcmatch from 'wildcard-match';
|
3
|
-
import { isObject, guard } from '@orpc/shared';
|
3
|
+
import { isObject, guard, toArray } from '@orpc/shared';
|
4
|
+
import { JsonSchemaXNativeType } from '@orpc/json-schema';
|
4
5
|
import { JSONSchemaFormat } from '@orpc/openapi';
|
5
6
|
import escapeStringRegexp from 'escape-string-regexp';
|
6
7
|
|
@@ -128,7 +129,7 @@ class ZodSmartCoercionPlugin {
|
|
128
129
|
options.clientInterceptors ??= [];
|
129
130
|
options.clientInterceptors.unshift((options2) => {
|
130
131
|
const inputSchema = options2.procedure["~orpc"].inputSchema;
|
131
|
-
if (!inputSchema || inputSchema["~standard"].vendor !== "zod") {
|
132
|
+
if (!inputSchema || inputSchema["~standard"].vendor !== "zod" || "_zod" in inputSchema) {
|
132
133
|
return options2.next();
|
133
134
|
}
|
134
135
|
const coercedInput = zodCoerceInternal(inputSchema, options2.input);
|
@@ -390,25 +391,39 @@ function safeToDate(value) {
|
|
390
391
|
|
391
392
|
class ZodToJsonSchemaConverter {
|
392
393
|
maxLazyDepth;
|
394
|
+
maxStructureDepth;
|
393
395
|
unsupportedJsonSchema;
|
394
396
|
anyJsonSchema;
|
395
397
|
constructor(options = {}) {
|
396
398
|
this.maxLazyDepth = options.maxLazyDepth ?? 3;
|
399
|
+
this.maxStructureDepth = options.maxStructureDepth ?? 10;
|
397
400
|
this.unsupportedJsonSchema = options.unsupportedJsonSchema ?? { not: {} };
|
398
401
|
this.anyJsonSchema = options.anyJsonSchema ?? {};
|
399
402
|
}
|
400
403
|
condition(schema) {
|
401
|
-
return schema !== void 0 && schema["~standard"].vendor === "zod";
|
404
|
+
return schema !== void 0 && schema["~standard"].vendor === "zod" && !("_zod" in schema);
|
402
405
|
}
|
403
|
-
convert(schema, options, lazyDepth = 0, isHandledCustomJSONSchema = false, isHandledZodDescription = false) {
|
406
|
+
convert(schema, options, lazyDepth = 0, isHandledCustomJSONSchema = false, isHandledZodDescription = false, structureDepth = 0) {
|
404
407
|
const def = schema._def;
|
408
|
+
if (structureDepth > this.maxStructureDepth) {
|
409
|
+
return [false, this.anyJsonSchema];
|
410
|
+
}
|
411
|
+
if (!options.minStructureDepthForRef || options.minStructureDepthForRef <= structureDepth) {
|
412
|
+
const components = toArray(options.components);
|
413
|
+
for (const component of components) {
|
414
|
+
if (component.schema === schema && component.allowedStrategies.includes(options.strategy)) {
|
415
|
+
return [component.required, { $ref: component.ref }];
|
416
|
+
}
|
417
|
+
}
|
418
|
+
}
|
405
419
|
if (!isHandledZodDescription && "description" in def && typeof def.description === "string") {
|
406
420
|
const [required, json] = this.convert(
|
407
421
|
schema,
|
408
422
|
options,
|
409
423
|
lazyDepth,
|
410
424
|
isHandledCustomJSONSchema,
|
411
|
-
true
|
425
|
+
true,
|
426
|
+
structureDepth
|
412
427
|
);
|
413
428
|
return [required, { ...json, description: def.description }];
|
414
429
|
}
|
@@ -420,7 +435,8 @@ class ZodToJsonSchemaConverter {
|
|
420
435
|
options,
|
421
436
|
lazyDepth,
|
422
437
|
true,
|
423
|
-
isHandledZodDescription
|
438
|
+
isHandledZodDescription,
|
439
|
+
structureDepth
|
424
440
|
);
|
425
441
|
return [required, { ...json, ...customJSONSchema }];
|
426
442
|
}
|
@@ -548,7 +564,11 @@ class ZodToJsonSchemaConverter {
|
|
548
564
|
return [true, json];
|
549
565
|
}
|
550
566
|
case ZodFirstPartyTypeKind.ZodBigInt: {
|
551
|
-
const json = {
|
567
|
+
const json = {
|
568
|
+
"type": "string",
|
569
|
+
"pattern": "^-?[0-9]+$",
|
570
|
+
"x-native-type": JsonSchemaXNativeType.BigInt
|
571
|
+
};
|
552
572
|
return [true, json];
|
553
573
|
}
|
554
574
|
case ZodFirstPartyTypeKind.ZodNaN: {
|
@@ -558,7 +578,11 @@ class ZodToJsonSchemaConverter {
|
|
558
578
|
return [true, { type: "boolean" }];
|
559
579
|
}
|
560
580
|
case ZodFirstPartyTypeKind.ZodDate: {
|
561
|
-
const schema2 = {
|
581
|
+
const schema2 = {
|
582
|
+
"type": "string",
|
583
|
+
"format": JSONSchemaFormat.DateTime,
|
584
|
+
"x-native-type": JsonSchemaXNativeType.Date
|
585
|
+
};
|
562
586
|
return [true, schema2];
|
563
587
|
}
|
564
588
|
case ZodFirstPartyTypeKind.ZodNull: {
|
@@ -591,7 +615,7 @@ class ZodToJsonSchemaConverter {
|
|
591
615
|
const schema_ = schema;
|
592
616
|
const def2 = schema_._def;
|
593
617
|
const json = { type: "array" };
|
594
|
-
const [itemRequired, itemJson] = this.convert(def2.type, options, lazyDepth, false, false);
|
618
|
+
const [itemRequired, itemJson] = this.convert(def2.type, options, lazyDepth, false, false, structureDepth + 1);
|
595
619
|
json.items = this.#toArrayItemJsonSchema(itemRequired, itemJson, options.strategy);
|
596
620
|
if (def2.exactLength) {
|
597
621
|
json.maxItems = def2.exactLength.value;
|
@@ -610,7 +634,7 @@ class ZodToJsonSchemaConverter {
|
|
610
634
|
const prefixItems = [];
|
611
635
|
const json = { type: "array" };
|
612
636
|
for (const item of schema_._def.items) {
|
613
|
-
const [itemRequired, itemJson] = this.convert(item, options, lazyDepth, false, false);
|
637
|
+
const [itemRequired, itemJson] = this.convert(item, options, lazyDepth, false, false, structureDepth + 1);
|
614
638
|
prefixItems.push(
|
615
639
|
this.#toArrayItemJsonSchema(itemRequired, itemJson, options.strategy)
|
616
640
|
);
|
@@ -619,7 +643,7 @@ class ZodToJsonSchemaConverter {
|
|
619
643
|
json.prefixItems = prefixItems;
|
620
644
|
}
|
621
645
|
if (schema_._def.rest) {
|
622
|
-
const [itemRequired, itemJson] = this.convert(schema_._def.rest, options, lazyDepth, false, false);
|
646
|
+
const [itemRequired, itemJson] = this.convert(schema_._def.rest, options, lazyDepth, false, false, structureDepth + 1);
|
623
647
|
json.items = this.#toArrayItemJsonSchema(itemRequired, itemJson, options.strategy);
|
624
648
|
}
|
625
649
|
return [true, json];
|
@@ -630,7 +654,7 @@ class ZodToJsonSchemaConverter {
|
|
630
654
|
const properties = {};
|
631
655
|
const required = [];
|
632
656
|
for (const [key, value] of Object.entries(schema_.shape)) {
|
633
|
-
const [itemRequired, itemJson] = this.convert(value, options, lazyDepth, false, false);
|
657
|
+
const [itemRequired, itemJson] = this.convert(value, options, lazyDepth, false, false, structureDepth + 1);
|
634
658
|
properties[key] = itemJson;
|
635
659
|
if (itemRequired) {
|
636
660
|
required.push(key);
|
@@ -648,7 +672,7 @@ class ZodToJsonSchemaConverter {
|
|
648
672
|
json.additionalProperties = false;
|
649
673
|
}
|
650
674
|
} else {
|
651
|
-
const [_, addJson] = this.convert(schema_._def.catchall, options, lazyDepth, false, false);
|
675
|
+
const [_, addJson] = this.convert(schema_._def.catchall, options, lazyDepth, false, false, structureDepth + 1);
|
652
676
|
json.additionalProperties = addJson;
|
653
677
|
}
|
654
678
|
return [true, json];
|
@@ -656,29 +680,32 @@ class ZodToJsonSchemaConverter {
|
|
656
680
|
case ZodFirstPartyTypeKind.ZodRecord: {
|
657
681
|
const schema_ = schema;
|
658
682
|
const json = { type: "object" };
|
659
|
-
const
|
660
|
-
if (
|
661
|
-
const [_2, keyJson] = this.convert(schema_._def.keyType, options, lazyDepth, false, false);
|
683
|
+
const [__, keyJson] = this.convert(schema_._def.keyType, options, lazyDepth, false, false, structureDepth + 1);
|
684
|
+
if (Object.entries(keyJson).some(([k, v]) => k !== "type" || v !== "string")) {
|
662
685
|
json.propertyNames = keyJson;
|
663
686
|
}
|
664
|
-
const [_, itemJson] = this.convert(schema_._def.valueType, options, lazyDepth, false, false);
|
687
|
+
const [_, itemJson] = this.convert(schema_._def.valueType, options, lazyDepth, false, false, structureDepth + 1);
|
665
688
|
json.additionalProperties = itemJson;
|
666
689
|
return [true, json];
|
667
690
|
}
|
668
691
|
case ZodFirstPartyTypeKind.ZodSet: {
|
669
692
|
const schema_ = schema;
|
670
|
-
const json = {
|
671
|
-
|
693
|
+
const json = {
|
694
|
+
"type": "array",
|
695
|
+
"uniqueItems": true,
|
696
|
+
"x-native-type": JsonSchemaXNativeType.Set
|
697
|
+
};
|
698
|
+
const [itemRequired, itemJson] = this.convert(schema_._def.valueType, options, lazyDepth, false, false, structureDepth + 1);
|
672
699
|
json.items = this.#toArrayItemJsonSchema(itemRequired, itemJson, options.strategy);
|
673
700
|
return [true, json];
|
674
701
|
}
|
675
702
|
case ZodFirstPartyTypeKind.ZodMap: {
|
676
703
|
const schema_ = schema;
|
677
|
-
const [keyRequired, keyJson] = this.convert(schema_._def.keyType, options, lazyDepth, false, false);
|
678
|
-
const [valueRequired, valueJson] = this.convert(schema_._def.valueType, options, lazyDepth, false, false);
|
679
|
-
|
680
|
-
type: "array",
|
681
|
-
items: {
|
704
|
+
const [keyRequired, keyJson] = this.convert(schema_._def.keyType, options, lazyDepth, false, false, structureDepth + 1);
|
705
|
+
const [valueRequired, valueJson] = this.convert(schema_._def.valueType, options, lazyDepth, false, false, structureDepth + 1);
|
706
|
+
const json = {
|
707
|
+
"type": "array",
|
708
|
+
"items": {
|
682
709
|
type: "array",
|
683
710
|
prefixItems: [
|
684
711
|
this.#toArrayItemJsonSchema(keyRequired, keyJson, options.strategy),
|
@@ -686,8 +713,10 @@ class ZodToJsonSchemaConverter {
|
|
686
713
|
],
|
687
714
|
maxItems: 2,
|
688
715
|
minItems: 2
|
689
|
-
}
|
690
|
-
|
716
|
+
},
|
717
|
+
"x-native-type": JsonSchemaXNativeType.Map
|
718
|
+
};
|
719
|
+
return [true, json];
|
691
720
|
}
|
692
721
|
case ZodFirstPartyTypeKind.ZodUnion:
|
693
722
|
case ZodFirstPartyTypeKind.ZodDiscriminatedUnion: {
|
@@ -695,7 +724,7 @@ class ZodToJsonSchemaConverter {
|
|
695
724
|
const anyOf = [];
|
696
725
|
let required = true;
|
697
726
|
for (const item of schema_._def.options) {
|
698
|
-
const [itemRequired, itemJson] = this.convert(item, options, lazyDepth, false, false);
|
727
|
+
const [itemRequired, itemJson] = this.convert(item, options, lazyDepth, false, false, structureDepth);
|
699
728
|
if (!itemRequired) {
|
700
729
|
required = false;
|
701
730
|
if (itemJson !== this.unsupportedJsonSchema) {
|
@@ -715,7 +744,7 @@ class ZodToJsonSchemaConverter {
|
|
715
744
|
const allOf = [];
|
716
745
|
let required = false;
|
717
746
|
for (const item of [schema_._def.left, schema_._def.right]) {
|
718
|
-
const [itemRequired, itemJson] = this.convert(item, options, lazyDepth, false, false);
|
747
|
+
const [itemRequired, itemJson] = this.convert(item, options, lazyDepth, false, false, structureDepth);
|
719
748
|
allOf.push(itemJson);
|
720
749
|
if (itemRequired) {
|
721
750
|
required = true;
|
@@ -724,25 +753,26 @@ class ZodToJsonSchemaConverter {
|
|
724
753
|
return [required, { allOf }];
|
725
754
|
}
|
726
755
|
case ZodFirstPartyTypeKind.ZodLazy: {
|
727
|
-
|
756
|
+
const currentLazyDepth = lazyDepth + 1;
|
757
|
+
if (currentLazyDepth > this.maxLazyDepth) {
|
728
758
|
return [false, this.anyJsonSchema];
|
729
759
|
}
|
730
760
|
const schema_ = schema;
|
731
|
-
return this.convert(schema_._def.getter(), options,
|
761
|
+
return this.convert(schema_._def.getter(), options, currentLazyDepth, false, false, structureDepth);
|
732
762
|
}
|
733
763
|
case ZodFirstPartyTypeKind.ZodOptional: {
|
734
764
|
const schema_ = schema;
|
735
|
-
const [_, inner] = this.convert(schema_._def.innerType, options, lazyDepth, false, false);
|
765
|
+
const [_, inner] = this.convert(schema_._def.innerType, options, lazyDepth, false, false, structureDepth);
|
736
766
|
return [false, inner];
|
737
767
|
}
|
738
768
|
case ZodFirstPartyTypeKind.ZodReadonly: {
|
739
769
|
const schema_ = schema;
|
740
|
-
const [required, json] = this.convert(schema_._def.innerType, options, lazyDepth, false, false);
|
770
|
+
const [required, json] = this.convert(schema_._def.innerType, options, lazyDepth, false, false, structureDepth);
|
741
771
|
return [required, { ...json, readOnly: true }];
|
742
772
|
}
|
743
773
|
case ZodFirstPartyTypeKind.ZodDefault: {
|
744
774
|
const schema_ = schema;
|
745
|
-
const [_, json] = this.convert(schema_._def.innerType, options, lazyDepth, false, false);
|
775
|
+
const [_, json] = this.convert(schema_._def.innerType, options, lazyDepth, false, false, structureDepth);
|
746
776
|
return [false, { default: schema_._def.defaultValue(), ...json }];
|
747
777
|
}
|
748
778
|
case ZodFirstPartyTypeKind.ZodEffects: {
|
@@ -750,15 +780,15 @@ class ZodToJsonSchemaConverter {
|
|
750
780
|
if (schema_._def.effect.type === "transform" && options.strategy === "output") {
|
751
781
|
return [false, this.anyJsonSchema];
|
752
782
|
}
|
753
|
-
return this.convert(schema_._def.schema, options, lazyDepth, false, false);
|
783
|
+
return this.convert(schema_._def.schema, options, lazyDepth, false, false, structureDepth);
|
754
784
|
}
|
755
785
|
case ZodFirstPartyTypeKind.ZodCatch: {
|
756
786
|
const schema_ = schema;
|
757
|
-
return this.convert(schema_._def.innerType, options, lazyDepth, false, false);
|
787
|
+
return this.convert(schema_._def.innerType, options, lazyDepth, false, false, structureDepth);
|
758
788
|
}
|
759
789
|
case ZodFirstPartyTypeKind.ZodBranded: {
|
760
790
|
const schema_ = schema;
|
761
|
-
return this.convert(schema_._def.type, options, lazyDepth, false, false);
|
791
|
+
return this.convert(schema_._def.type, options, lazyDepth, false, false, structureDepth);
|
762
792
|
}
|
763
793
|
case ZodFirstPartyTypeKind.ZodPipeline: {
|
764
794
|
const schema_ = schema;
|
@@ -767,13 +797,14 @@ class ZodToJsonSchemaConverter {
|
|
767
797
|
options,
|
768
798
|
lazyDepth,
|
769
799
|
false,
|
770
|
-
false
|
800
|
+
false,
|
801
|
+
structureDepth
|
771
802
|
);
|
772
803
|
}
|
773
804
|
case ZodFirstPartyTypeKind.ZodNullable: {
|
774
805
|
const schema_ = schema;
|
775
|
-
const [required, json] = this.convert(schema_._def.innerType, options, lazyDepth, false, false);
|
776
|
-
return [required, { anyOf: [{ type: "null" }
|
806
|
+
const [required, json] = this.convert(schema_._def.innerType, options, lazyDepth, false, false, structureDepth);
|
807
|
+
return [required, { anyOf: [json, { type: "null" }] }];
|
777
808
|
}
|
778
809
|
}
|
779
810
|
return [true, this.unsupportedJsonSchema];
|
@@ -792,12 +823,17 @@ class ZodToJsonSchemaConverter {
|
|
792
823
|
}
|
793
824
|
case "regexp": {
|
794
825
|
return {
|
795
|
-
type: "string",
|
796
|
-
pattern: "^\\/(.*)\\/([a-z]*)$"
|
826
|
+
"type": "string",
|
827
|
+
"pattern": "^\\/(.*)\\/([a-z]*)$",
|
828
|
+
"x-native-type": JsonSchemaXNativeType.RegExp
|
797
829
|
};
|
798
830
|
}
|
799
831
|
case "url": {
|
800
|
-
return {
|
832
|
+
return {
|
833
|
+
"type": "string",
|
834
|
+
"format": JSONSchemaFormat.URI,
|
835
|
+
"x-native-type": JsonSchemaXNativeType.Url
|
836
|
+
};
|
801
837
|
}
|
802
838
|
}
|
803
839
|
}
|