@lssm/lib.schema 0.0.0-canary-20251120170226

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 ADDED
@@ -0,0 +1,134 @@
1
+ # @lssm/lib.schema
2
+
3
+ A small schema dictionary to describe operation I/O once and export to:
4
+
5
+ - zod (runtime validation)
6
+ - Pothos (GraphQL type refs)
7
+ - JSON Schema (via `zod-to-json-schema` or custom `getJsonSchema()` on `FieldType`)
8
+
9
+ ## Installation
10
+
11
+ ```bash
12
+ npm install @lssm/lib.schema
13
+ # or
14
+ bun add @lssm/lib.schema
15
+ ```
16
+
17
+ The package ships only pre-built `dist/` artifacts and type declarations. Import helpers directly, e.g.:
18
+
19
+ ```ts
20
+ import { SchemaModel, ScalarTypeEnum } from '@lssm/lib.schema';
21
+ ```
22
+
23
+ ## Primitives
24
+
25
+ - `FieldType<T>` wraps a GraphQLScalarType and carries a zod schema, plus optional JSON Schema definition.
26
+ - `SchemaModel` composes fields into named object models with helpers:
27
+ - `getZod()` → typed `z.ZodObject` preserving each field's schema and optionality
28
+ - `getPothosInput()` → builder name for input object
29
+ - `getJsonSchema()` → resolved JSON Schema
30
+
31
+ ### Enums
32
+
33
+ - `EnumType<T>` represents string enums with a single source of truth:
34
+ - `getZod()` → `z.enum([...])`
35
+ - `getPothos()` → GraphQL `GraphQLEnumType`
36
+ - `getJsonSchema()` → `{ type: 'string', enum: [...] }`
37
+ - Create with `defineEnum('Name', ['A','B'] as const)`
38
+
39
+ Example
40
+
41
+ ```ts
42
+ import { defineEnum, SchemaModel, ScalarTypeEnum } from '@lssm/lib.schema';
43
+
44
+ const Weekday = defineEnum('Weekday', [
45
+ 'MO',
46
+ 'TU',
47
+ 'WE',
48
+ 'TH',
49
+ 'FR',
50
+ 'SA',
51
+ 'SU',
52
+ ] as const);
53
+
54
+ const Rule = new SchemaModel({
55
+ name: 'Rule',
56
+ fields: {
57
+ timezone: { type: ScalarTypeEnum.TimeZone(), isOptional: false },
58
+ frequency: {
59
+ type: defineEnum('RecurrenceFrequency', [
60
+ 'DAILY',
61
+ 'WEEKLY',
62
+ 'MONTHLY',
63
+ 'YEARLY',
64
+ ] as const),
65
+ isOptional: false,
66
+ },
67
+ byWeekday: { type: Weekday, isOptional: true, isArray: true },
68
+ },
69
+ });
70
+ ```
71
+
72
+ ## Usage
73
+
74
+ Define fields with `FieldType` and group them into a `SchemaModel`. Adapters (REST/MCP/GraphQL) can consume these to generate appropriate I/O types while preserving a single source of truth.
75
+
76
+ ## API Overview
77
+
78
+ - **FieldType<TInternal, TExternal = TInternal>**
79
+ - `getZod()` → zod schema
80
+ - `getPothos()` → GraphQL scalar type
81
+ - `getJsonSchema()` → deeply-resolved JSON Schema
82
+ - **EnumType<T>**
83
+ - Construct via `defineEnum('Name', ['A','B'] as const)`
84
+ - `getZod()`, `getPothos()`, `getJsonSchema()`
85
+ - **SchemaModel<Fields>**
86
+ - `getZod()` → typed `z.ZodObject`
87
+ - `getPothosInput()` → input object name for GraphQL builders
88
+ - Each field supports `{ isOptional: boolean, isArray?: true }`
89
+
90
+ ## Patterns & Conventions
91
+
92
+ - Name models with PascalCase and suffix with `Input`/`Result` when ambiguous.
93
+ - Use `ScalarTypeEnum` for common scalars: `NonEmptyString`, `Date`, `DateTime`, `Locale`, `TimeZone`, `Latitude`, `Longitude`, `PhoneNumber`, etc.
94
+ - Prefer enums for constrained strings. Reuse enums across input/output models to ensure consistency.
95
+ - Top-level arrays: define a nested model and set `isArray: true` on the field that holds the list, or create a dedicated list model if needed by adapters.
96
+
97
+ ## Example: Nested models + arrays
98
+
99
+ ```ts
100
+ import { ScalarTypeEnum, SchemaModel, defineEnum } from '@lssm/lib.schema';
101
+
102
+ const Weekday = defineEnum('Weekday', [
103
+ 'MO',
104
+ 'TU',
105
+ 'WE',
106
+ 'TH',
107
+ 'FR',
108
+ 'SA',
109
+ 'SU',
110
+ ] as const);
111
+
112
+ const Address = new SchemaModel({
113
+ name: 'Address',
114
+ fields: {
115
+ city: { type: ScalarTypeEnum.NonEmptyString(), isOptional: false },
116
+ country: { type: ScalarTypeEnum.CountryCode(), isOptional: false },
117
+ },
118
+ });
119
+
120
+ export const CreateSpotInput = new SchemaModel({
121
+ name: 'CreateSpotInput',
122
+ fields: {
123
+ name: { type: ScalarTypeEnum.NonEmptyString(), isOptional: false },
124
+ latitude: { type: ScalarTypeEnum.Latitude(), isOptional: false },
125
+ longitude: { type: ScalarTypeEnum.Longitude(), isOptional: false },
126
+ availabilityWeekdays: { type: Weekday, isOptional: true, isArray: true },
127
+ address: { type: Address, isOptional: true },
128
+ },
129
+ });
130
+ ```
131
+
132
+ ## JSON Schema export
133
+
134
+ Prefer `getZod()` + `zod-to-json-schema` for consistency. For advanced cases, expose a custom `getJsonSchema()` from `FieldType` or a specialized wrapper.
@@ -0,0 +1,42 @@
1
+ import { z } from "zod";
2
+ import { GraphQLEnumType } from "graphql";
3
+
4
+ //#region src/EnumType.d.ts
5
+
6
+ /**
7
+ * Strongly-typed string enum wrapper with one source of truth for zod, GraphQL, and JSON Schema.
8
+ */
9
+ declare class EnumType<T extends [string, ...string[]]> {
10
+ private readonly name;
11
+ private readonly values;
12
+ private readonly gqlEnum;
13
+ constructor(name: string, values: T);
14
+ /** Enum type name (used by GraphQL and JSON Schema). */
15
+ getName(): string;
16
+ /** Returns the literal tuple of allowed values. */
17
+ getEnumValues(): T;
18
+ /** GraphQL enum instance suitable for Pothos or vanilla GraphQL schemas. */
19
+ getPothos(): GraphQLEnumType;
20
+ /** zod schema representing this enum. */
21
+ getZod(): z.ZodEnum<{ [K in T[number]]: K }>;
22
+ /** Minimal JSON representation (alias of getJsonSchema). */
23
+ getJson(): {
24
+ type: 'string';
25
+ enum: T;
26
+ };
27
+ /** JSON Schema for this enum. */
28
+ getJsonSchema(): {
29
+ type: 'string';
30
+ enum: T;
31
+ };
32
+ }
33
+ type AnyEnumType = EnumType<[string, ...string[]]>;
34
+ /**
35
+ * Helper to define an EnumType.
36
+ * @param name Display/type name used across GraphQL and JSON Schema
37
+ * @param values Literal tuple of allowed string values (at least one)
38
+ */
39
+ declare const defineEnum: <T extends [string, ...string[]]>(name: string, values: T) => EnumType<T>;
40
+ //#endregion
41
+ export { AnyEnumType, EnumType, defineEnum };
42
+ //# sourceMappingURL=EnumType.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"EnumType.d.ts","names":[],"sources":["../src/EnumType.ts"],"sourcesContent":[],"mappings":";;;;;;;AAMA;AAKoC,cALvB,QAKuB,CAAA,UAAA,CAAA,MAAA,EAAA,GAAA,MAAA,EAAA,CAAA,CAAA,CAAA;EAejB,iBAAA,IAAA;EAMJ,iBAAA,MAAA;EAKe,iBAAA,OAAA;EAAY,WAAA,CAAA,IAAA,EAAA,MAAA,EAAA,MAAA,EA1BN,CA0BM;EAA5B;EAKuB,OAAA,CAAA,CAAA,EAAA,MAAA;EAKM;EAAC,aAAA,CAAA,CAAA,EArBzB,CAqByB;EAKhC;EAOC,SAAA,CAAA,CAAA,EA3BE,eA8BgB;EADrB;EAAC,MAAA,CAAA,CAAA,EAxBC,CAAA,CAAE,OAwBH,CAAA,QAxBmB,CAwBnB,CAAA,MAAA,CAAA,GAxB+B,CAwB/B,EAAA,CAAA;;;;UAnB0B;;;;;UAKM;;;KAK/B,WAAA,GAAc;;;;;;cAOb,oEAEH,MAAC,SAAA"}
@@ -0,0 +1,2 @@
1
+ import{z as e}from"zod";import{GraphQLEnumType as t}from"graphql";var n=class{name;values;gqlEnum;constructor(e,n){this.name=e,this.values=n,this.gqlEnum=new t({name:this.name,values:Object.fromEntries(n.map(e=>[e,{value:e}]))})}getName(){return this.name}getEnumValues(){return this.values}getPothos(){return this.gqlEnum}getZod(){return e.enum(this.values)}getJson(){return{type:`string`,enum:this.values}}getJsonSchema(){return this.getJson()}};const r=(e,t)=>new n(e,t);export{n as EnumType,r as defineEnum};
2
+ //# sourceMappingURL=EnumType.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"EnumType.js","names":[],"sources":["../src/EnumType.ts"],"sourcesContent":["import { z } from 'zod';\nimport { GraphQLEnumType } from 'graphql';\n\n/**\n * Strongly-typed string enum wrapper with one source of truth for zod, GraphQL, and JSON Schema.\n */\nexport class EnumType<T extends [string, ...string[]]> {\n private readonly name: string;\n private readonly values: T;\n private readonly gqlEnum: GraphQLEnumType;\n\n constructor(name: string, values: T) {\n this.name = name;\n this.values = values;\n this.gqlEnum = new GraphQLEnumType({\n name: this.name,\n values: Object.fromEntries(values.map((v) => [v, { value: v }] as const)),\n });\n }\n\n /** Enum type name (used by GraphQL and JSON Schema). */\n getName(): string {\n return this.name;\n }\n\n /** Returns the literal tuple of allowed values. */\n getEnumValues(): T {\n return this.values;\n }\n\n // For SchemaModel and adapters expecting a Pothos/GraphQL type-like object\n /** GraphQL enum instance suitable for Pothos or vanilla GraphQL schemas. */\n getPothos(): GraphQLEnumType {\n return this.gqlEnum;\n }\n\n /** zod schema representing this enum. */\n getZod(): z.ZodEnum<{ [K in T[number]]: K }> {\n return z.enum(this.values);\n }\n\n /** Minimal JSON representation (alias of getJsonSchema). */\n getJson(): { type: 'string'; enum: T } {\n return { type: 'string', enum: this.values } as const;\n }\n\n /** JSON Schema for this enum. */\n getJsonSchema(): { type: 'string'; enum: T } {\n return this.getJson();\n }\n}\n\nexport type AnyEnumType = EnumType<[string, ...string[]]>;\n\n/**\n * Helper to define an EnumType.\n * @param name Display/type name used across GraphQL and JSON Schema\n * @param values Literal tuple of allowed string values (at least one)\n */\nexport const defineEnum = <T extends [string, ...string[]]>(\n name: string,\n values: T\n) => new EnumType(name, values);\n"],"mappings":"kEAMA,IAAa,EAAb,KAAuD,CACrD,KACA,OACA,QAEA,YAAY,EAAc,EAAW,CACnC,KAAK,KAAO,EACZ,KAAK,OAAS,EACd,KAAK,QAAU,IAAI,EAAgB,CACjC,KAAM,KAAK,KACX,OAAQ,OAAO,YAAY,EAAO,IAAK,GAAM,CAAC,EAAG,CAAE,MAAO,EAAG,CAAC,CAAU,CAAC,CAC1E,CAAC,CAIJ,SAAkB,CAChB,OAAO,KAAK,KAId,eAAmB,CACjB,OAAO,KAAK,OAKd,WAA6B,CAC3B,OAAO,KAAK,QAId,QAA6C,CAC3C,OAAO,EAAE,KAAK,KAAK,OAAO,CAI5B,SAAuC,CACrC,MAAO,CAAE,KAAM,SAAU,KAAM,KAAK,OAAQ,CAI9C,eAA6C,CAC3C,OAAO,KAAK,SAAS,GAWzB,MAAa,GACX,EACA,IACG,IAAI,EAAS,EAAM,EAAO"}
@@ -0,0 +1,31 @@
1
+ import { z } from "zod";
2
+ import { GraphQLScalarType, GraphQLScalarTypeConfig } from "graphql";
3
+
4
+ //#region src/FieldType.d.ts
5
+ interface FieldTypeConfig<TInternal, TExternal = TInternal> extends GraphQLScalarTypeConfig<TInternal, TExternal> {
6
+ zod: z.ZodType<TInternal>;
7
+ jsonSchema: unknown | (() => unknown);
8
+ }
9
+ type AnyFieldType = FieldType<any, any>;
10
+ /**
11
+ * GraphQL scalar wrapper that carries zod and JSON Schema metadata.
12
+ *
13
+ * TInternal is the runtime representation; TExternal is the GraphQL output.
14
+ */
15
+ declare class FieldType<TInternal, TExternal = TInternal> extends GraphQLScalarType<TInternal, TExternal> {
16
+ private zodSchema;
17
+ private readonly jsonSchemaDef?;
18
+ constructor(config: FieldTypeConfig<TInternal, TExternal>);
19
+ /** Return the attached zod schema for validation. */
20
+ getZod(): z.ZodType<TInternal>;
21
+ /** GraphQL scalar instance usable by Pothos or vanilla GraphQL. */
22
+ getPothos(): GraphQLScalarType<TInternal, TExternal>;
23
+ /** Return the JSON Schema (evaluates factory if provided). */
24
+ getJson(): unknown;
25
+ getJsonSchemaDef(): unknown | (() => unknown);
26
+ getJsonSchema(): unknown;
27
+ }
28
+ type ZodFieldType<Field extends AnyFieldType> = z.infer<ReturnType<Field['getZod']>>;
29
+ //#endregion
30
+ export { AnyFieldType, FieldType, FieldTypeConfig, ZodFieldType };
31
+ //# sourceMappingURL=FieldType.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"FieldType.d.ts","names":[],"sources":["../src/FieldType.ts"],"sourcesContent":[],"mappings":";;;;UAQiB,uCAAuC,mBAC9C,wBAAwB,WAAW;OACtC,CAAA,CAAE,QAAQ;EAFA,UAAA,EAAA,OAAe,GAAA,CAAA,GAAA,GAAA,OAAA,CAAA;;AACE,KAMtB,YAAA,GAAe,SANO,CAAA,GAAA,EAAA,GAAA,CAAA;;;;;;AAMtB,cAOC,SAPc,CAAA,SAAS,EAAA,YAStB,SATsB,CAAA,SAU1B,iBAV0B,CAUR,SAVQ,EAUG,SAVH,CAAA,CAAA;EAOvB,QAAA,SAAS;EAER,iBAAA,aAAA;EACc,WAAA,CAAA,MAAA,EAIN,eAJM,CAIU,SAJV,EAIqB,SAJrB,CAAA;EAAW;EAID,MAAA,CAAA,CAAA,EAO1B,CAAA,CAAE,OAPwB,CAOhB,SAPgB,CAAA;EAAW;EAA3B,SAAA,CAAA,CAAA,EAYP,iBAZO,CAYW,SAZX,EAYsB,SAZtB,CAAA;EAOA;EAAR,OAAA,CAAA,CAAA,EAAA,OAAA;EAKmB,gBAAA,CAAA,CAAA,EAAA,OAAA,GAAA,CAAA,GAAA,GAAA,OAAA,CAAA;EAAW,aAAA,CAAA,CAAA,EAAA,OAAA;;AAhBlC,KAoDE,YApDF,CAAA,cAoD6B,YApD7B,CAAA,GAoD6C,CAAA,CAAE,KApD/C,CAqDR,UArDQ,CAqDG,KArDH,CAAA,QAAA,CAAA,CAAA,CAAA"}
@@ -0,0 +1,2 @@
1
+ import{z as e}from"zod";import{GraphQLScalarType as t,Kind as n}from"graphql";var r=class extends t{zodSchema;jsonSchemaDef;constructor(e){super(e),this.zodSchema=e.zod,this.jsonSchemaDef=e.jsonSchema}getZod(){return this.zodSchema}getPothos(){return this}getJson(){return typeof this.jsonSchemaDef==`function`?this.jsonSchemaDef():this.jsonSchemaDef}getJsonSchemaDef(){return this.jsonSchemaDef}getJsonSchema(){let e=t=>{let n=typeof t==`function`?t():t;if(Array.isArray(n))return n.map(t=>e(t));if(n&&typeof n==`object`){let t={};for(let[r,i]of Object.entries(n))t[r]=e(i);return t}return n};return e(this.getJson())}};export{r as FieldType};
2
+ //# sourceMappingURL=FieldType.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"FieldType.js","names":["obj: Record<string, unknown>"],"sources":["../src/FieldType.ts"],"sourcesContent":["import { z } from 'zod';\nimport {\n GraphQLScalarType,\n type GraphQLScalarTypeConfig,\n Kind,\n type ValueNode,\n} from 'graphql';\n\nexport interface FieldTypeConfig<TInternal, TExternal = TInternal>\n extends GraphQLScalarTypeConfig<TInternal, TExternal> {\n zod: z.ZodType<TInternal>;\n jsonSchema: unknown | (() => unknown);\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport type AnyFieldType = FieldType<any, any>;\n\n/**\n * GraphQL scalar wrapper that carries zod and JSON Schema metadata.\n *\n * TInternal is the runtime representation; TExternal is the GraphQL output.\n */\nexport class FieldType<\n TInternal,\n TExternal = TInternal,\n> extends GraphQLScalarType<TInternal, TExternal> {\n private zodSchema: z.ZodType<TInternal>;\n private readonly jsonSchemaDef?: unknown | (() => unknown);\n\n constructor(config: FieldTypeConfig<TInternal, TExternal>) {\n super(config);\n this.zodSchema = config.zod;\n this.jsonSchemaDef = config.jsonSchema;\n }\n\n /** Return the attached zod schema for validation. */\n getZod(): z.ZodType<TInternal> {\n return this.zodSchema;\n }\n\n /** GraphQL scalar instance usable by Pothos or vanilla GraphQL. */\n getPothos(): GraphQLScalarType<TInternal, TExternal> {\n return this;\n }\n\n /** Return the JSON Schema (evaluates factory if provided). */\n getJson(): unknown {\n return typeof this.jsonSchemaDef === 'function'\n ? (this.jsonSchemaDef as () => unknown)()\n : this.jsonSchemaDef;\n }\n\n // Return the raw jsonSchema config (value or factory)\n getJsonSchemaDef(): unknown | (() => unknown) {\n return this.jsonSchemaDef;\n }\n\n // Return a deep-resolved JSON Schema (evaluates factory and nested factories)\n getJsonSchema(): unknown {\n const deepResolve = (v: unknown): unknown => {\n const value = typeof v === 'function' ? (v as () => unknown)() : v;\n if (Array.isArray(value)) return value.map((item) => deepResolve(item));\n if (value && typeof value === 'object') {\n const obj: Record<string, unknown> = {};\n for (const [k, val] of Object.entries(\n value as Record<string, unknown>\n )) {\n obj[k] = deepResolve(val);\n }\n return obj;\n }\n return value;\n };\n return deepResolve(this.getJson());\n }\n}\n\nexport type ZodFieldType<Field extends AnyFieldType> = z.infer<\n ReturnType<Field['getZod']>\n>;\n"],"mappings":"8EAsBA,IAAa,EAAb,cAGU,CAAwC,CAChD,UACA,cAEA,YAAY,EAA+C,CACzD,MAAM,EAAO,CACb,KAAK,UAAY,EAAO,IACxB,KAAK,cAAgB,EAAO,WAI9B,QAA+B,CAC7B,OAAO,KAAK,UAId,WAAqD,CACnD,OAAO,KAIT,SAAmB,CACjB,OAAO,OAAO,KAAK,eAAkB,WAChC,KAAK,eAAiC,CACvC,KAAK,cAIX,kBAA8C,CAC5C,OAAO,KAAK,cAId,eAAyB,CACvB,IAAM,EAAe,GAAwB,CAC3C,IAAM,EAAQ,OAAO,GAAM,WAAc,GAAqB,CAAG,EACjE,GAAI,MAAM,QAAQ,EAAM,CAAE,OAAO,EAAM,IAAK,GAAS,EAAY,EAAK,CAAC,CACvE,GAAI,GAAS,OAAO,GAAU,SAAU,CACtC,IAAMA,EAA+B,EAAE,CACvC,IAAK,GAAM,CAAC,EAAG,KAAQ,OAAO,QAC5B,EACD,CACC,EAAI,GAAK,EAAY,EAAI,CAE3B,OAAO,EAET,OAAO,GAET,OAAO,EAAY,KAAK,SAAS,CAAC"}
@@ -0,0 +1,33 @@
1
+ import { FieldType } from "./FieldType.js";
2
+
3
+ //#region src/ScalarTypeEnum.d.ts
4
+
5
+ /**
6
+ * Factory functions for common scalar FieldTypes with zod/GraphQL/JSON Schema.
7
+ */
8
+ declare const ScalarTypeEnum: {
9
+ readonly String_unsecure: () => FieldType<string>;
10
+ readonly Int_unsecure: () => FieldType<number>;
11
+ readonly Float_unsecure: () => FieldType<number>;
12
+ readonly Boolean: () => FieldType<boolean>;
13
+ readonly ID: () => FieldType<string>;
14
+ readonly JSON: () => FieldType<unknown>;
15
+ readonly JSONObject: () => FieldType<Record<string, unknown>>;
16
+ readonly Date: () => FieldType<Date, string>;
17
+ readonly DateTime: () => FieldType<Date, string>;
18
+ readonly Time: () => FieldType<string>;
19
+ readonly EmailAddress: () => FieldType<string>;
20
+ readonly URL: () => FieldType<string>;
21
+ readonly PhoneNumber: () => FieldType<string>;
22
+ readonly NonEmptyString: () => FieldType<string>;
23
+ readonly Locale: () => FieldType<string>;
24
+ readonly TimeZone: () => FieldType<string>;
25
+ readonly Latitude: () => FieldType<number>;
26
+ readonly Longitude: () => FieldType<number>;
27
+ readonly Currency: () => FieldType<string>;
28
+ readonly CountryCode: () => FieldType<string>;
29
+ };
30
+ type ScalarTypeEnum = [keyof typeof ScalarTypeEnum];
31
+ //#endregion
32
+ export { ScalarTypeEnum };
33
+ //# sourceMappingURL=ScalarTypeEnum.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ScalarTypeEnum.d.ts","names":[],"sources":["../src/ScalarTypeEnum.ts"],"sourcesContent":[],"mappings":";;;;;;AAkBA;AAEuB,cAFV,cAEU,EAAA;EAaH,SAAA,eAAA,EAAA,GAAA,GAbG,SAaH,CAAA,MAAA,CAAA;EAgBE,SAAA,YAAA,EAAA,GAAA,GAhBF,SAgBE,CAAA,MAAA,CAAA;EAiBP,SAAA,cAAA,EAAA,GAAA,GAjBO,SAiBP,CAAA,MAAA,CAAA;EAaL,SAAA,OAAA,EAAA,GAAA,GAbK,SAaL,CAAA,OAAA,CAAA;EAeE,SAAA,EAAA,EAAA,GAAA,GAfF,SAeE,CAAA,MAAA,CAAA;EAQgB,SAAA,IAAA,EAAA,GAAA,GARhB,SAQgB,CAAA,OAAA,CAAA;EAAV,SAAA,UAAA,EAAA,GAAA,GAAA,SAAA,CAAU,MAAV,CAAA,MAAA,EAAA,OAAA,CAAA,CAAA;EAQI,SAAA,IAAA,EAAA,GAAA,GAAV,SAAU,CAAA,IAAA,EAAA,MAAA,CAAA;EAAV,SAAA,QAAA,EAAA,GAAA,GAUI,SAVJ,CAUc,IAVd,EAAA,MAAA,CAAA;EAUc,SAAA,IAAA,EAAA,GAAA,GAUd,SAVc,CAAA,MAAA,CAAA;EAAV,SAAA,YAAA,EAAA,GAAA,GAsBI,SAtBJ,CAAA,MAAA,CAAA;EAUJ,SAAA,GAAA,EAAA,GAAA,GAoBD,SApBC,CAAA,MAAA,CAAA;EAYQ,SAAA,WAAA,EAAA,GAAA,GAgBD,SAhBC,CAAA,MAAA,CAAA;EAQT,SAAA,cAAA,EAAA,GAAA,GAgBW,SAhBX,CAAA,MAAA,CAAA;EAQQ,SAAA,MAAA,EAAA,GAAA,GAgBL,SAhBK,CAAA,MAAA,CAAA;EAQG,SAAA,QAAA,EAAA,GAAA,GAgBN,SAhBM,CAAA,MAAA,CAAA;EAQR,SAAA,QAAA,EAAA,GAAA,GAgBE,SAhBF,CAAA,MAAA,CAAA;EAQE,SAAA,SAAA,EAAA,GAAA,GAgBC,SAhBD,CAAA,MAAA,CAAA;EAQA,SAAA,QAAA,EAAA,GAAA,GAgBA,SAhBA,CAAA,MAAA,CAAA;EAQC,SAAA,WAAA,EAAA,GAAA,GAgBE,SAhBF,CAAA,MAAA,CAAA;CAQD;AAQG,KAqBP,cAAA,GArBO,CAAA,MAAA,OAqBwB,cArBxB,CAAA"}
@@ -0,0 +1,2 @@
1
+ import{FieldType as e}from"./FieldType.js";import{z as t}from"zod";import{Kind as n}from"graphql";const r=/^[A-Za-z]{2}(?:-[A-Za-z0-9]{2,8})*$/,i=/^(?:UTC|[A-Za-z_]+\/[A-Za-z_]+)$/,a=/^[+]?\d[\d\s().-]{3,}$/,o=/^[A-Z]{3}$/,s=/^[A-Z]{2}$/,c=-180,l={String_unsecure:()=>new e({name:`String_unsecure`,description:`Unvalidated string scalar`,zod:t.string(),parseValue:e=>t.string().parse(e),serialize:e=>String(e),parseLiteral:e=>{if(e.kind!==n.STRING)throw TypeError(`Invalid literal`);return e.value},jsonSchema:{type:`string`}}),Int_unsecure:()=>new e({name:`Int_unsecure`,description:`Unvalidated integer scalar`,zod:t.number().int(),parseValue:e=>{let n=typeof e==`number`?e:Number(e);return t.number().int().parse(n)},serialize:e=>Math.trunc(typeof e==`number`?e:Number(e)),parseLiteral:e=>{if(e.kind!==n.INT)throw TypeError(`Invalid literal`);return Number(e.value)},jsonSchema:{type:`integer`}}),Float_unsecure:()=>new e({name:`Float_unsecure`,description:`Unvalidated float scalar`,zod:t.number(),parseValue:e=>{let n=typeof e==`number`?e:Number(e);return t.number().parse(n)},serialize:e=>Number(e),parseLiteral:e=>{if(e.kind!==n.FLOAT&&e.kind!==n.INT)throw TypeError(`Invalid literal`);return Number(e.value)},jsonSchema:{type:`number`}}),Boolean:()=>new e({name:`Boolean`,description:`Unvalidated boolean scalar`,zod:t.boolean(),parseValue:e=>t.coerce.boolean().parse(e),serialize:e=>!!e,parseLiteral:e=>{if(e.kind!==n.BOOLEAN)throw TypeError(`Invalid literal`);return e.value},jsonSchema:{type:`boolean`}}),ID:()=>new e({name:`ID`,description:`Unvalidated id scalar`,zod:t.string(),parseValue:e=>t.string().parse(e),serialize:e=>String(e),parseLiteral:e=>{if(e.kind!==n.STRING)throw TypeError(`Invalid literal`);return e.value},jsonSchema:{type:`string`}}),JSON:()=>new e({name:`JSON`,zod:t.any(),parseValue:e=>e,serialize:e=>e,jsonSchema:{}}),JSONObject:()=>new e({name:`JSONObject`,zod:t.record(t.string(),t.any()),parseValue:e=>t.record(t.string(),t.any()).parse(e),serialize:e=>e??{},jsonSchema:{type:`object`}}),Date:()=>new e({name:`Date`,zod:t.date(),parseValue:e=>e instanceof Date?e:new Date(String(e)),serialize:e=>e instanceof Date?e.toISOString().split(`T`)[0]:String(e),jsonSchema:{type:`string`,format:`date`}}),DateTime:()=>new e({name:`DateTime`,zod:t.date(),parseValue:e=>e instanceof Date?e:new Date(String(e)),serialize:e=>e instanceof Date?e.toISOString():String(e),jsonSchema:{type:`string`,format:`date-time`}}),Time:()=>new e({name:`Time`,zod:t.string().regex(/^\d{2}:\d{2}(:\d{2})?$/),parseValue:e=>t.string().regex(/^\d{2}:\d{2}(:\d{2})?$/).parse(e),serialize:e=>String(e),jsonSchema:{type:`string`,pattern:`^\\d{2}:\\d{2}(:\\d{2})?$`}}),EmailAddress:()=>new e({name:`EmailAddress`,zod:t.string().email(),parseValue:e=>t.string().email().parse(e),serialize:e=>String(e),jsonSchema:{type:`string`,format:`email`}}),URL:()=>new e({name:`URL`,zod:t.string().url(),parseValue:e=>t.string().url().parse(e),serialize:e=>String(e),jsonSchema:{type:`string`,format:`uri`}}),PhoneNumber:()=>new e({name:`PhoneNumber`,zod:t.string().regex(a),parseValue:e=>t.string().regex(a).parse(e),serialize:e=>String(e),jsonSchema:{type:`string`,pattern:a.source}}),NonEmptyString:()=>new e({name:`NonEmptyString`,zod:t.string().min(1),parseValue:e=>t.string().min(1).parse(e),serialize:e=>String(e),jsonSchema:{type:`string`,minLength:1}}),Locale:()=>new e({name:`Locale`,zod:t.string().regex(r),parseValue:e=>t.string().regex(r).parse(e),serialize:e=>String(e),jsonSchema:{type:`string`,pattern:r.source}}),TimeZone:()=>new e({name:`TimeZone`,zod:t.string().regex(i),parseValue:e=>t.string().regex(i).parse(e),serialize:e=>String(e),jsonSchema:{type:`string`,pattern:i.source}}),Latitude:()=>new e({name:`Latitude`,zod:t.number().min(-90).max(90),parseValue:e=>t.coerce.number().min(-90).max(90).parse(e),serialize:e=>Number(e),jsonSchema:{type:`number`,minimum:-90,maximum:90}}),Longitude:()=>new e({name:`Longitude`,zod:t.number().min(c).max(180),parseValue:e=>t.coerce.number().min(c).max(180).parse(e),serialize:e=>Number(e),jsonSchema:{type:`number`,minimum:c,maximum:180}}),Currency:()=>new e({name:`Currency`,zod:t.string().regex(o),parseValue:e=>t.string().regex(o).parse(e),serialize:e=>String(e),jsonSchema:{type:`string`,pattern:o.source}}),CountryCode:()=>new e({name:`CountryCode`,zod:t.string().regex(s),parseValue:e=>t.string().regex(s).parse(e),serialize:e=>String(e),jsonSchema:{type:`string`,pattern:s.source}})};export{l as ScalarTypeEnum};
2
+ //# sourceMappingURL=ScalarTypeEnum.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ScalarTypeEnum.js","names":[],"sources":["../src/ScalarTypeEnum.ts"],"sourcesContent":["import { z } from 'zod';\nimport { type GraphQLScalarTypeConfig, Kind, type ValueNode } from 'graphql';\nimport { FieldType } from './FieldType';\n\n// Helpers to build standard scalars\nconst localeRegex = /^[A-Za-z]{2}(?:-[A-Za-z0-9]{2,8})*$/;\nconst timezoneRegex = /^(?:UTC|[A-Za-z_]+\\/[A-Za-z_]+)$/;\nconst phoneRegex = /^[+]?\\d[\\d\\s().-]{3,}$/;\nconst currencyRegex = /^[A-Z]{3}$/;\nconst countryRegex = /^[A-Z]{2}$/;\nconst latMin = -90;\nconst latMax = 90;\nconst lonMin = -180;\nconst lonMax = 180;\n\n/**\n * Factory functions for common scalar FieldTypes with zod/GraphQL/JSON Schema.\n */\nexport const ScalarTypeEnum = {\n // primitives (_unsecure)\n String_unsecure: (): FieldType<string> =>\n new FieldType<string>({\n name: 'String_unsecure',\n description: 'Unvalidated string scalar',\n zod: z.string(),\n parseValue: (v) => z.string().parse(v),\n serialize: (v) => String(v),\n parseLiteral: (ast: ValueNode) => {\n if (ast.kind !== Kind.STRING) throw new TypeError('Invalid literal');\n return ast.value;\n },\n jsonSchema: { type: 'string' },\n }),\n Int_unsecure: (): FieldType<number> =>\n new FieldType<number>({\n name: 'Int_unsecure',\n description: 'Unvalidated integer scalar',\n zod: z.number().int(),\n parseValue: (v) => {\n const num = typeof v === 'number' ? v : Number(v as unknown);\n return z.number().int().parse(num);\n },\n serialize: (v) => Math.trunc(typeof v === 'number' ? v : Number(v)),\n parseLiteral: (ast: ValueNode) => {\n if (ast.kind !== Kind.INT) throw new TypeError('Invalid literal');\n return Number(ast.value);\n },\n jsonSchema: { type: 'integer' },\n }),\n Float_unsecure: (): FieldType<number> =>\n new FieldType<number>({\n name: 'Float_unsecure',\n description: 'Unvalidated float scalar',\n zod: z.number(),\n parseValue: (v) => {\n const num = typeof v === 'number' ? v : Number(v as unknown);\n return z.number().parse(num);\n },\n serialize: (v) => Number(v),\n parseLiteral: (ast: ValueNode) => {\n if (ast.kind !== Kind.FLOAT && ast.kind !== Kind.INT)\n throw new TypeError('Invalid literal');\n return Number(ast.value);\n },\n jsonSchema: { type: 'number' },\n }),\n Boolean: (): FieldType<boolean> =>\n new FieldType<boolean>({\n name: 'Boolean',\n description: 'Unvalidated boolean scalar',\n zod: z.boolean(),\n parseValue: (v) => z.coerce.boolean().parse(v),\n serialize: (v) => Boolean(v),\n parseLiteral: (ast: ValueNode) => {\n if (ast.kind !== Kind.BOOLEAN) throw new TypeError('Invalid literal');\n return ast.value;\n },\n jsonSchema: { type: 'boolean' },\n }),\n ID: (): FieldType<string> =>\n new FieldType<string>({\n name: 'ID',\n description: 'Unvalidated id scalar',\n zod: z.string(),\n parseValue: (v) => z.string().parse(v),\n serialize: (v) => String(v),\n parseLiteral: (ast: ValueNode) => {\n if (ast.kind !== Kind.STRING) throw new TypeError('Invalid literal');\n return ast.value;\n },\n jsonSchema: { type: 'string' },\n }),\n\n // Validated custom scalars\n JSON: (): FieldType<unknown> =>\n new FieldType<unknown>({\n name: 'JSON',\n zod: z.any(),\n parseValue: (v) => v,\n serialize: (v) => v,\n jsonSchema: {},\n }),\n JSONObject: (): FieldType<Record<string, unknown>> =>\n new FieldType<Record<string, unknown>>({\n name: 'JSONObject',\n zod: z.record(z.string(), z.any()),\n parseValue: (v) => z.record(z.string(), z.any()).parse(v),\n serialize: (v) => (v ?? {}) as Record<string, unknown>,\n jsonSchema: { type: 'object' },\n }),\n Date: (): FieldType<Date, string> =>\n new FieldType<Date, string>({\n name: 'Date',\n zod: z.date(),\n parseValue: (v) => (v instanceof Date ? v : new Date(String(v))),\n serialize: (v) =>\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n v instanceof Date ? v.toISOString().split('T')[0]! : String(v),\n jsonSchema: { type: 'string', format: 'date' },\n }),\n DateTime: (): FieldType<Date, string> =>\n new FieldType<Date, string>({\n name: 'DateTime',\n zod: z.date(),\n parseValue: (v) => (v instanceof Date ? v : new Date(String(v))),\n serialize: (v) => {\n return v instanceof Date ? v.toISOString() : String(v);\n },\n jsonSchema: { type: 'string', format: 'date-time' },\n }),\n Time: (): FieldType<string> =>\n new FieldType<string>({\n name: 'Time',\n zod: z.string().regex(/^\\d{2}:\\d{2}(:\\d{2})?$/),\n parseValue: (v) =>\n z\n .string()\n .regex(/^\\d{2}:\\d{2}(:\\d{2})?$/)\n .parse(v),\n serialize: (v) => String(v),\n jsonSchema: { type: 'string', pattern: '^\\\\d{2}:\\\\d{2}(:\\\\d{2})?$' },\n }),\n EmailAddress: (): FieldType<string> =>\n new FieldType<string>({\n name: 'EmailAddress',\n zod: z.string().email(),\n parseValue: (v) => z.string().email().parse(v),\n serialize: (v) => String(v),\n jsonSchema: { type: 'string', format: 'email' },\n }),\n URL: (): FieldType<string> =>\n new FieldType<string>({\n name: 'URL',\n zod: z.string().url(),\n parseValue: (v) => z.string().url().parse(v),\n serialize: (v) => String(v),\n jsonSchema: { type: 'string', format: 'uri' },\n }),\n PhoneNumber: (): FieldType<string> =>\n new FieldType<string>({\n name: 'PhoneNumber',\n zod: z.string().regex(phoneRegex),\n parseValue: (v) => z.string().regex(phoneRegex).parse(v),\n serialize: (v) => String(v),\n jsonSchema: { type: 'string', pattern: phoneRegex.source },\n }),\n NonEmptyString: (): FieldType<string> =>\n new FieldType<string>({\n name: 'NonEmptyString',\n zod: z.string().min(1),\n parseValue: (v) => z.string().min(1).parse(v),\n serialize: (v) => String(v),\n jsonSchema: { type: 'string', minLength: 1 },\n }),\n Locale: (): FieldType<string> =>\n new FieldType<string>({\n name: 'Locale',\n zod: z.string().regex(localeRegex),\n parseValue: (v) => z.string().regex(localeRegex).parse(v),\n serialize: (v) => String(v),\n jsonSchema: { type: 'string', pattern: localeRegex.source },\n }),\n TimeZone: (): FieldType<string> =>\n new FieldType<string>({\n name: 'TimeZone',\n zod: z.string().regex(timezoneRegex),\n parseValue: (v) => z.string().regex(timezoneRegex).parse(v),\n serialize: (v) => String(v),\n jsonSchema: { type: 'string', pattern: timezoneRegex.source },\n }),\n Latitude: (): FieldType<number> =>\n new FieldType<number>({\n name: 'Latitude',\n zod: z.number().min(latMin).max(latMax),\n parseValue: (v) => z.coerce.number().min(latMin).max(latMax).parse(v),\n serialize: (v) => Number(v),\n jsonSchema: { type: 'number', minimum: latMin, maximum: latMax },\n }),\n Longitude: (): FieldType<number> =>\n new FieldType<number>({\n name: 'Longitude',\n zod: z.number().min(lonMin).max(lonMax),\n parseValue: (v) => z.coerce.number().min(lonMin).max(lonMax).parse(v),\n serialize: (v) => Number(v),\n jsonSchema: { type: 'number', minimum: lonMin, maximum: lonMax },\n }),\n Currency: (): FieldType<string> =>\n new FieldType<string>({\n name: 'Currency',\n zod: z.string().regex(currencyRegex),\n parseValue: (v) => z.string().regex(currencyRegex).parse(v),\n serialize: (v) => String(v),\n jsonSchema: { type: 'string', pattern: currencyRegex.source },\n }),\n CountryCode: (): FieldType<string> =>\n new FieldType<string>({\n name: 'CountryCode',\n zod: z.string().regex(countryRegex),\n parseValue: (v) => z.string().regex(countryRegex).parse(v),\n serialize: (v) => String(v),\n jsonSchema: { type: 'string', pattern: countryRegex.source },\n }),\n // GeoJSON: (): FieldType<Record<string, unknown>, Record<string, unknown>> =>\n // new FieldType<Record<string, unknown>, Record<string, unknown>>({\n // name: 'GeoJSON',\n // zod: z.object({ type: z.string(), coordinates: z.any() }).passthrough(),\n // parseValue: (v) =>\n // z\n // .object({ type: z.string(), coordinates: z.any() })\n // .passthrough()\n // .parse(v),\n // serialize: (v) => v,\n // jsonSchema: { type: 'object' },\n // }),\n} as const;\nexport type ScalarTypeEnum = [keyof typeof ScalarTypeEnum];\n"],"mappings":"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"}
@@ -0,0 +1,57 @@
1
+ import { AnyEnumType } from "./EnumType.js";
2
+ import { AnyFieldType } from "./FieldType.js";
3
+ import { z } from "zod";
4
+ import { Maybe } from "graphql/jsutils/Maybe";
5
+
6
+ //#region src/SchemaModel.d.ts
7
+ type FieldLike = AnyFieldType | AnyEnumType | AnySchemaModel;
8
+ /** Field configuration for a SchemaModel property. */
9
+ interface SchemaFieldConfig<Type extends FieldLike = FieldLike> {
10
+ type: Type;
11
+ isOptional: boolean;
12
+ /** When present and true, the field is an array */
13
+ isArray?: true;
14
+ }
15
+ type SchemaModelFieldsAnyConfig<Type extends FieldLike = FieldLike> = Record<string, SchemaFieldConfig<Type>>;
16
+ /** Model definition: name and fields. */
17
+ interface SchemaModelConfig<Fields extends SchemaModelFieldsAnyConfig> {
18
+ name: string;
19
+ description?: Maybe<string>;
20
+ fields: Fields;
21
+ }
22
+ /**
23
+ * Named object model built from FieldType/EnumType/SchemaModel fields.
24
+ * Provides zod and GraphQL input helpers, and supports arrays/optional fields.
25
+ */
26
+ declare class SchemaModel<Fields extends SchemaModelFieldsAnyConfig> {
27
+ readonly config: SchemaModelConfig<Fields>;
28
+ constructor(config: SchemaModelConfig<Fields>);
29
+ /**
30
+ * Build a typed ZodObject from the model fields, preserving each field's
31
+ * Zod schema and optionality at the type level when possible.
32
+ */
33
+ getZod(): TopLevelZodFromModel<Fields>;
34
+ /** Input object name for GraphQL builder adapters. */
35
+ getPothosInput(): string;
36
+ }
37
+ type AnySchemaModel = SchemaModel<SchemaModelFieldsAnyConfig>;
38
+ type ZodSchemaModel<Field extends AnySchemaModel> = z.infer<ReturnType<Field['getZod']>>;
39
+ type InferZodFromType<T> = T extends SchemaModel<any> ? z.ZodObject<any> : T extends AnyFieldType ? ReturnType<T['getZod']> : T extends AnyEnumType ? ReturnType<T['getZod']> : never;
40
+ type MaybeArray<Z extends z.ZodType, A> = A extends true ? z.ZodArray<Z> : Z;
41
+ type MaybeOptional<Z extends z.ZodType, O> = O extends true ? z.ZodOptional<Z> : Z;
42
+ /**
43
+ * Helper type: derive the Zod shape from the field config.
44
+ * Supports nested SchemaModel and arrays, preserving optionality and list-ness.
45
+ */
46
+ type FieldIsArray<FC> = FC extends {
47
+ isArray: true;
48
+ } ? true : false;
49
+ type ZodShapeFromFields<F extends SchemaModelFieldsAnyConfig> = { [K in keyof F]: MaybeOptional<MaybeArray<InferZodFromType<F[K]['type']>, FieldIsArray<F[K]>>, F[K]['isOptional']> };
50
+ /**
51
+ * The top-level Zod schema returned by getZod():
52
+ * either ZodObject<...> or ZodArray<ZodObject<...>> when config.isArray.
53
+ */
54
+ type TopLevelZodFromModel<F extends SchemaModelFieldsAnyConfig> = z.ZodObject<ZodShapeFromFields<F>>;
55
+ //#endregion
56
+ export { AnySchemaModel, SchemaFieldConfig, SchemaModel, SchemaModelConfig, SchemaModelFieldsAnyConfig, TopLevelZodFromModel, ZodSchemaModel, ZodShapeFromFields };
57
+ //# sourceMappingURL=SchemaModel.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SchemaModel.d.ts","names":[],"sources":["../src/SchemaModel.ts"],"sourcesContent":[],"mappings":";;;;;;KAKK,SAAA,GAAY,eAAe,cAAc;;AAAzC,UAGY,iBAHH,CAAA,aAGkC,SAHlC,GAG8C,SAH9C,CAAA,CAAA;EAAG,IAAA,EAIT,IAJS;EAAe,UAAA,EAAA,OAAA;EAAc;EAAc,OAAA,CAAA,EAAA,IAAA;AAG5D;AAAgD,KAOpC,0BAPoC,CAAA,aAOI,SAPJ,GAOgB,SAPhB,CAAA,GAQ9C,MAR8C,CAAA,MAAA,EAQ/B,iBAR+B,CAQb,IARa,CAAA,CAAA;;AACxC,UAUS,iBAVT,CAAA,eAU0C,0BAV1C,CAAA,CAAA;EAAI,IAAA,EAAA,MAAA;EAMA,WAAA,CAAA,EAMI,KANJ,CAAA,MAAA,CAAA;EAAwC,MAAA,EAO1C,MAP0C;;;;;;AAInC,cAUJ,WAVqB,CAAA,eAUM,0BAVN,CAAA,CAAA;EAAgB,SAAA,MAAA,EAWZ,iBAXY,CAWM,MAXN,CAAA;EAElC,WAAA,CAAA,MAAA,EASsB,iBATtB,CASwC,MATxC,CAAA;EACN;;AAOV;;EACwD,MAAA,CAAA,CAAA,EAM5C,oBAN4C,CAMvB,MANuB,CAAA;EAAlB;EAAkB,cAAA,CAAA,CAAA,EAAA,MAAA;;AAMvB,KAwBrB,cAAA,GAAiB,WAxBI,CAwBQ,0BAxBR,CAAA;AAArB,KA0BA,cA1BA,CAAA,cA0B6B,cA1B7B,CAAA,GA0B+C,CAAA,CAAE,KA1BjD,CA2BV,UA3BU,CA2BC,KA3BD,CAAA,QAAA,CAAA,CAAA,CAAA;KA8BP,gBA9B2B,CAAA,CAAA,CAAA,GA+B9B,CA/B8B,SA+BpB,WA/BoB,CAAA,GAAA,CAAA,GAgC1B,CAAA,CAAE,SAhCwB,CAAA,GAAA,CAAA,GAiC1B,CAjC0B,SAiChB,YAjCgB,GAkCxB,UAlCwB,CAkCb,CAlCa,CAAA,QAAA,CAAA,CAAA,GAmCxB,CAnCwB,SAmCd,WAnCc,GAoCtB,UApCsB,CAoCX,CApCW,CAAA,QAAA,CAAA,CAAA,GAAA,KAAA;AAwBhC,KAeK,UAfO,CAAA,UAec,CAAA,CAAE,OAfa,EAAA,CAAA,CAAA,GAeC,CAfD,SAAA,IAAA,GAekB,CAAA,CAAE,QAfhC,CAeyC,CAfzC,CAAA,GAe8C,CAfnC;AAExC,KAcK,aAdO,CAAA,UAciB,CAAA,CAAE,OAdL,EAAA,CAAA,CAAA,GAcmB,CAdnB,SAAA,IAAA,GAetB,CAAA,CAAE,WAfoB,CAeR,CAfQ,CAAA,GAgBtB,CAhBsB;;;;;KAsBrB,YAtB6D,CAAA,EAAA,CAAA,GAsB1C,EAtB0C,SAAA;EAI7D,OAAA,EAAA,IAAA;CACH,GAAA,IAAA,GAAA,KAAA;AAAU,KAmBA,kBAnBA,CAAA,UAmB6B,0BAnB7B,CAAA,GAAA,QACJ,MAmBM,CAnBN,GAmBU,aAnBV,CAoBJ,UApBI,CAoBO,gBApBP,CAoBwB,CApBxB,CAoB0B,CApB1B,CAAA,CAAA,MAAA,CAAA,CAAA,EAoBuC,YApBvC,CAoBoD,CApBpD,CAoBsD,CApBtD,CAAA,CAAA,CAAA,EAqBJ,CArBI,CAqBF,CArBE,CAAA,CAAA,YAAA,CAAA,CAAA,EACF;;;;;AAEY,KA0BN,oBA1BM,CAAA,UA0ByB,0BA1BzB,CAAA,GA2BhB,CAAA,CAAE,SA3Bc,CA2BJ,kBA3BI,CA2Be,CA3Bf,CAAA,CAAA"}
@@ -0,0 +1,2 @@
1
+ import"./EnumType.js";import"./FieldType.js";import{z as e}from"zod";var t=class{constructor(e){this.config=e}getZod(){let t=Object.entries(this.config.fields).reduce((t,[n,r])=>{let i=r.type.getZod(),a=r.isArray?e.array(i):i;return t[n]=r.isOptional?a.optional():a,t},{});return e.object(t)}getPothosInput(){return this.config.name}};export{t as SchemaModel};
2
+ //# sourceMappingURL=SchemaModel.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SchemaModel.js","names":["config: SchemaModelConfig<Fields>","base: z.ZodType"],"sources":["../src/SchemaModel.ts"],"sourcesContent":["import { type AnyFieldType } from './FieldType';\nimport { type AnyEnumType } from './EnumType';\nimport { z } from 'zod';\nimport type { Maybe } from 'graphql/jsutils/Maybe';\n\ntype FieldLike = AnyFieldType | AnyEnumType | AnySchemaModel;\n\n/** Field configuration for a SchemaModel property. */\nexport interface SchemaFieldConfig<Type extends FieldLike = FieldLike> {\n type: Type;\n isOptional: boolean;\n /** When present and true, the field is an array */\n isArray?: true;\n}\n\nexport type SchemaModelFieldsAnyConfig<Type extends FieldLike = FieldLike> =\n Record<string, SchemaFieldConfig<Type>>;\n\n/** Model definition: name and fields. */\nexport interface SchemaModelConfig<Fields extends SchemaModelFieldsAnyConfig> {\n name: string;\n description?: Maybe<string>;\n fields: Fields;\n}\n\n/**\n * Named object model built from FieldType/EnumType/SchemaModel fields.\n * Provides zod and GraphQL input helpers, and supports arrays/optional fields.\n */\nexport class SchemaModel<Fields extends SchemaModelFieldsAnyConfig> {\n constructor(public readonly config: SchemaModelConfig<Fields>) {}\n\n /**\n * Build a typed ZodObject from the model fields, preserving each field's\n * Zod schema and optionality at the type level when possible.\n */\n getZod(): TopLevelZodFromModel<Fields> {\n const shape = Object.entries(this.config.fields).reduce(\n (acc, [key, def]) => {\n const base: z.ZodType = (\n def.type as unknown as { getZod: () => z.ZodType }\n ).getZod();\n const withArray = def.isArray ? z.array(base) : base;\n (acc as Record<string, z.ZodType>)[key] = def.isOptional\n ? withArray.optional()\n : withArray;\n return acc;\n },\n {} as Record<string, z.ZodType>\n ) as unknown as ZodShapeFromFields<Fields>;\n\n return z.object(shape) as z.ZodObject<ZodShapeFromFields<Fields>>;\n }\n\n /** Input object name for GraphQL builder adapters. */\n getPothosInput() {\n return this.config.name;\n }\n}\n\nexport type AnySchemaModel = SchemaModel<SchemaModelFieldsAnyConfig>;\n\nexport type ZodSchemaModel<Field extends AnySchemaModel> = z.infer<\n ReturnType<Field['getZod']>\n>;\n\ntype InferZodFromType<T> =\n T extends SchemaModel<any>\n ? z.ZodObject<any>\n : T extends AnyFieldType\n ? ReturnType<T['getZod']>\n : T extends AnyEnumType\n ? ReturnType<T['getZod']>\n : never;\n\ntype MaybeArray<Z extends z.ZodType, A> = A extends true ? z.ZodArray<Z> : Z;\ntype MaybeOptional<Z extends z.ZodType, O> = O extends true\n ? z.ZodOptional<Z>\n : Z;\n\n/**\n * Helper type: derive the Zod shape from the field config.\n * Supports nested SchemaModel and arrays, preserving optionality and list-ness.\n */\ntype FieldIsArray<FC> = FC extends { isArray: true } ? true : false;\n\nexport type ZodShapeFromFields<F extends SchemaModelFieldsAnyConfig> = {\n [K in keyof F]: MaybeOptional<\n MaybeArray<InferZodFromType<F[K]['type']>, FieldIsArray<F[K]>>,\n F[K]['isOptional']\n >;\n};\n\n/**\n * The top-level Zod schema returned by getZod():\n * either ZodObject<...> or ZodArray<ZodObject<...>> when config.isArray.\n */\nexport type TopLevelZodFromModel<F extends SchemaModelFieldsAnyConfig> =\n z.ZodObject<ZodShapeFromFields<F>>;\n"],"mappings":"qEA6BA,IAAa,EAAb,KAAoE,CAClE,YAAY,EAAmD,CAAnC,KAAA,OAAA,EAM5B,QAAuC,CACrC,IAAM,EAAQ,OAAO,QAAQ,KAAK,OAAO,OAAO,CAAC,QAC9C,EAAK,CAAC,EAAK,KAAS,CACnB,IAAMC,EACJ,EAAI,KACJ,QAAQ,CACJ,EAAY,EAAI,QAAU,EAAE,MAAM,EAAK,CAAG,EAIhD,MAHC,GAAkC,GAAO,EAAI,WAC1C,EAAU,UAAU,CACpB,EACG,GAET,EAAE,CACH,CAED,OAAO,EAAE,OAAO,EAAM,CAIxB,gBAAiB,CACf,OAAO,KAAK,OAAO"}
@@ -0,0 +1,5 @@
1
+ import { AnyEnumType, EnumType, defineEnum } from "./EnumType.js";
2
+ import { AnyFieldType, FieldType, FieldTypeConfig, ZodFieldType } from "./FieldType.js";
3
+ import { ScalarTypeEnum } from "./ScalarTypeEnum.js";
4
+ import { AnySchemaModel, SchemaFieldConfig, SchemaModel, SchemaModelConfig, SchemaModelFieldsAnyConfig, TopLevelZodFromModel, ZodSchemaModel, ZodShapeFromFields } from "./SchemaModel.js";
5
+ export { AnyEnumType, AnyFieldType, AnySchemaModel, EnumType, FieldType, FieldTypeConfig, ScalarTypeEnum, SchemaFieldConfig, SchemaModel, SchemaModelConfig, SchemaModelFieldsAnyConfig, TopLevelZodFromModel, ZodFieldType, ZodSchemaModel, ZodShapeFromFields, defineEnum };
package/dist/index.js ADDED
@@ -0,0 +1 @@
1
+ import{EnumType as e,defineEnum as t}from"./EnumType.js";import{FieldType as n}from"./FieldType.js";import{ScalarTypeEnum as r}from"./ScalarTypeEnum.js";import{SchemaModel as i}from"./SchemaModel.js";export{e as EnumType,n as FieldType,r as ScalarTypeEnum,i as SchemaModel,t as defineEnum};
package/package.json ADDED
@@ -0,0 +1,50 @@
1
+ {
2
+ "name": "@lssm/lib.schema",
3
+ "version": "0.0.0-canary-20251120170226",
4
+ "scripts": {
5
+ "build": "bun build:bundle && bun build:types",
6
+ "build:bundle": "tsdown",
7
+ "build:types": "tsc --noEmit",
8
+ "dev": "bun build:bundle --watch",
9
+ "clean": "rimraf dist .turbo",
10
+ "lint": "bun lint:fix",
11
+ "lint:fix": "eslint src --fix",
12
+ "lint:check": "eslint src"
13
+ },
14
+ "devDependencies": {
15
+ "@lssm/tool.tsdown": "0.0.0-canary-20251120170226",
16
+ "@lssm/tool.typescript": "0.0.0-canary-20251120170226",
17
+ "tsdown": "^0.15.9",
18
+ "typescript": "^5.9.3"
19
+ },
20
+ "dependencies": {
21
+ "zod": "^4.1.5",
22
+ "graphql": "^16.8.1"
23
+ },
24
+ "files": [
25
+ "dist",
26
+ "README.md"
27
+ ],
28
+ "type": "module",
29
+ "main": "./dist/index.js",
30
+ "module": "./dist/index.js",
31
+ "types": "./dist/index.d.ts",
32
+ "exports": {
33
+ ".": "./dist/index.js",
34
+ "./EnumType": "./dist/EnumType.js",
35
+ "./FieldType": "./dist/FieldType.js",
36
+ "./ScalarTypeEnum": "./dist/ScalarTypeEnum.js",
37
+ "./SchemaModel": "./dist/SchemaModel.js",
38
+ "./*": "./*"
39
+ },
40
+ "publishConfig": {
41
+ "exports": {
42
+ ".": "./dist/index.js",
43
+ "./EnumType": "./dist/EnumType.js",
44
+ "./FieldType": "./dist/FieldType.js",
45
+ "./ScalarTypeEnum": "./dist/ScalarTypeEnum.js",
46
+ "./SchemaModel": "./dist/SchemaModel.js",
47
+ "./*": "./*"
48
+ }
49
+ }
50
+ }