@contractspec/lib.schema 1.46.2 → 1.48.0

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.
@@ -1 +1 @@
1
- {"version":3,"file":"FieldType.js","names":["obj: Record<string, unknown>"],"sources":["../src/FieldType.ts"],"sourcesContent":["import * as z from 'zod';\nimport { GraphQLScalarType, type GraphQLScalarTypeConfig } from 'graphql';\n\nexport interface FieldTypeConfig<\n TInternal,\n 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":";;;;;;;;;AAmBA,IAAa,YAAb,cAGU,kBAAwC;CAChD,AAAQ;CACR,AAAiB;CAEjB,YAAY,QAA+C;AACzD,QAAM,OAAO;AACb,OAAK,YAAY,OAAO;AACxB,OAAK,gBAAgB,OAAO;;;CAI9B,SAA+B;AAC7B,SAAO,KAAK;;;CAId,YAAqD;AACnD,SAAO;;;CAIT,UAAmB;AACjB,SAAO,OAAO,KAAK,kBAAkB,aAChC,KAAK,eAAiC,GACvC,KAAK;;CAIX,mBAA8C;AAC5C,SAAO,KAAK;;CAId,gBAAyB;EACvB,MAAM,eAAe,MAAwB;GAC3C,MAAM,QAAQ,OAAO,MAAM,aAAc,GAAqB,GAAG;AACjE,OAAI,MAAM,QAAQ,MAAM,CAAE,QAAO,MAAM,KAAK,SAAS,YAAY,KAAK,CAAC;AACvE,OAAI,SAAS,OAAO,UAAU,UAAU;IACtC,MAAMA,MAA+B,EAAE;AACvC,SAAK,MAAM,CAAC,GAAG,QAAQ,OAAO,QAC5B,MACD,CACC,KAAI,KAAK,YAAY,IAAI;AAE3B,WAAO;;AAET,UAAO;;AAET,SAAO,YAAY,KAAK,SAAS,CAAC"}
1
+ {"version":3,"file":"FieldType.js","names":[],"sources":["../src/FieldType.ts"],"sourcesContent":["import * as z from 'zod';\nimport { GraphQLScalarType, type GraphQLScalarTypeConfig } from 'graphql';\n\nexport interface FieldTypeConfig<\n TInternal,\n 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":";;;;;;;;;AAmBA,IAAa,YAAb,cAGU,kBAAwC;CAChD,AAAQ;CACR,AAAiB;CAEjB,YAAY,QAA+C;AACzD,QAAM,OAAO;AACb,OAAK,YAAY,OAAO;AACxB,OAAK,gBAAgB,OAAO;;;CAI9B,SAA+B;AAC7B,SAAO,KAAK;;;CAId,YAAqD;AACnD,SAAO;;;CAIT,UAAmB;AACjB,SAAO,OAAO,KAAK,kBAAkB,aAChC,KAAK,eAAiC,GACvC,KAAK;;CAIX,mBAA8C;AAC5C,SAAO,KAAK;;CAId,gBAAyB;EACvB,MAAM,eAAe,MAAwB;GAC3C,MAAM,QAAQ,OAAO,MAAM,aAAc,GAAqB,GAAG;AACjE,OAAI,MAAM,QAAQ,MAAM,CAAE,QAAO,MAAM,KAAK,SAAS,YAAY,KAAK,CAAC;AACvE,OAAI,SAAS,OAAO,UAAU,UAAU;IACtC,MAAM,MAA+B,EAAE;AACvC,SAAK,MAAM,CAAC,GAAG,QAAQ,OAAO,QAC5B,MACD,CACC,KAAI,KAAK,YAAY,IAAI;AAE3B,WAAO;;AAET,UAAO;;AAET,SAAO,YAAY,KAAK,SAAS,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"GraphQLSchemaType.js","names":["typeDef: string","name: string"],"sources":["../src/GraphQLSchemaType.ts"],"sourcesContent":["import { z } from 'zod';\nimport type { SchemaModelType } from './SchemaModelType';\n\n/**\n * Wrapper for GraphQL Type Definitions to be used in ContractSpec Schemas.\n * Note: Runtime validation is limited to 'unknown' (pass-through) as SDL is not parsed at runtime.\n */\nexport class GraphQLSchemaType implements SchemaModelType {\n readonly _isSchemaType = true;\n\n constructor(\n public readonly typeDef: string,\n public readonly name: string\n ) {}\n\n getZod(): z.ZodType {\n return z.unknown();\n }\n}\n"],"mappings":";;;;;;;AAOA,IAAa,oBAAb,MAA0D;CACxD,AAAS,gBAAgB;CAEzB,YACE,AAAgBA,SAChB,AAAgBC,MAChB;EAFgB;EACA;;CAGlB,SAAoB;AAClB,SAAO,EAAE,SAAS"}
1
+ {"version":3,"file":"GraphQLSchemaType.js","names":[],"sources":["../src/GraphQLSchemaType.ts"],"sourcesContent":["import { z } from 'zod';\nimport type { SchemaModelType } from './SchemaModelType';\n\n/**\n * Wrapper for GraphQL Type Definitions to be used in ContractSpec Schemas.\n * Note: Runtime validation is limited to 'unknown' (pass-through) as SDL is not parsed at runtime.\n */\nexport class GraphQLSchemaType implements SchemaModelType {\n readonly _isSchemaType = true;\n\n constructor(\n public readonly typeDef: string,\n public readonly name: string\n ) {}\n\n getZod(): z.ZodType {\n return z.unknown();\n }\n}\n"],"mappings":";;;;;;;AAOA,IAAa,oBAAb,MAA0D;CACxD,AAAS,gBAAgB;CAEzB,YACE,AAAgB,SAChB,AAAgB,MAChB;EAFgB;EACA;;CAGlB,SAAoB;AAClB,SAAO,EAAE,SAAS"}
@@ -1 +1 @@
1
- {"version":3,"file":"JsonSchemaType.js","names":["z","shape: Record<string, z.ZodType>"],"sources":["../src/JsonSchemaType.ts"],"sourcesContent":["/**\n * JSON Schema wrapper for ContractSpec compatibility.\n *\n * Wraps JSON Schema definitions to implement the SchemaType interface,\n * enabling direct JSON Schema usage in ContractSpec definitions.\n *\n * @module JsonSchemaType\n */\n\nimport * as z from 'zod';\nimport type { SchemaModelType } from './SchemaModelType';\n\n/**\n * JSON Schema definition structure.\n * Supports standard JSON Schema draft-07/2020-12 properties.\n */\nexport interface JsonSchemaDefinition {\n type?: string | string[];\n properties?: Record<string, JsonSchemaDefinition>;\n required?: string[];\n additionalProperties?: boolean | JsonSchemaDefinition;\n items?: JsonSchemaDefinition | JsonSchemaDefinition[];\n enum?: unknown[];\n const?: unknown;\n oneOf?: JsonSchemaDefinition[];\n anyOf?: JsonSchemaDefinition[];\n allOf?: JsonSchemaDefinition[];\n $ref?: string;\n title?: string;\n description?: string;\n default?: unknown;\n format?: string;\n minimum?: number;\n maximum?: number;\n minLength?: number;\n maxLength?: number;\n pattern?: string;\n [key: string]: unknown;\n}\n\n/**\n * Options for JsonSchemaType wrapper.\n */\nexport interface JsonSchemaTypeOptions {\n /** Name for identification */\n name?: string;\n}\n\n/**\n * Wrapper to use JSON Schema as a SchemaType.\n * Converts to Zod at runtime for validation.\n *\n * @example\n * ```typescript\n * const CustomFieldsSchema = fromJsonSchema({\n * type: 'object',\n * additionalProperties: {\n * oneOf: [\n * { type: 'string' },\n * { type: 'number' },\n * { type: 'boolean' },\n * ],\n * },\n * }, { name: 'CustomFields' });\n * ```\n */\nexport class JsonSchemaType implements SchemaModelType<\n Record<string, unknown>\n> {\n private readonly jsonSchema: JsonSchemaDefinition;\n private readonly options: JsonSchemaTypeOptions;\n\n constructor(\n jsonSchema: JsonSchemaDefinition,\n options: JsonSchemaTypeOptions = {}\n ) {\n this.jsonSchema = jsonSchema;\n this.options = options;\n }\n\n /**\n * Convert JSON Schema to Zod schema for runtime validation.\n *\n * Note: This is a simplified conversion. For complex schemas,\n * consider using a dedicated json-schema-to-zod library.\n */\n getZod(): z.ZodType<Record<string, unknown>> {\n // Handle additionalProperties (dictionary/record types)\n if (this.jsonSchema.additionalProperties !== undefined) {\n if (this.jsonSchema.additionalProperties === true) {\n return z.record(z.string(), z.unknown());\n }\n if (typeof this.jsonSchema.additionalProperties === 'object') {\n // For typed additionalProperties, use union or unknown\n return z.record(z.string(), z.unknown());\n }\n if (this.jsonSchema.additionalProperties === false) {\n return z.object({}).strict();\n }\n }\n\n // Handle explicit properties\n if (this.jsonSchema.properties) {\n const shape: Record<string, z.ZodType> = {};\n const required = new Set(this.jsonSchema.required ?? []);\n\n for (const [key, propSchema] of Object.entries(\n this.jsonSchema.properties\n )) {\n const fieldType = this.convertPropertyToZod(propSchema);\n shape[key] = required.has(key) ? fieldType : fieldType.optional();\n }\n\n return z.object(shape).passthrough();\n }\n\n // Default: passthrough object\n return z.object({}).passthrough();\n }\n\n /**\n * Return the original JSON Schema.\n */\n getJsonSchema(): JsonSchemaDefinition {\n return this.jsonSchema;\n }\n\n /**\n * Return GraphQL type info.\n * JSON Schema types map to JSON scalar in GraphQL.\n */\n getPothos(): { type: string; name?: string } {\n return { type: 'JSON', name: this.options.name };\n }\n\n /**\n * Get the configured name for this schema.\n */\n getName(): string | undefined {\n return this.options.name;\n }\n\n /**\n * Convert a single JSON Schema property to Zod.\n */\n private convertPropertyToZod(schema: JsonSchemaDefinition): z.ZodType {\n const type = Array.isArray(schema.type) ? schema.type[0] : schema.type;\n\n switch (type) {\n case 'string':\n return z.string();\n case 'number':\n case 'integer':\n return z.number();\n case 'boolean':\n return z.boolean();\n case 'array':\n if (schema.items && !Array.isArray(schema.items)) {\n return z.array(this.convertPropertyToZod(schema.items));\n }\n return z.array(z.unknown());\n case 'object':\n if (schema.properties) {\n return new JsonSchemaType(schema).getZod();\n }\n return z.record(z.string(), z.unknown());\n case 'null':\n return z.null();\n default:\n return z.unknown();\n }\n }\n}\n\n/**\n * Helper to create a SchemaType from JSON Schema.\n *\n * @param schema - The JSON Schema definition\n * @param options - Optional configuration\n * @returns A SchemaType-compatible wrapper\n *\n * @example\n * ```typescript\n * const MetadataSchema = fromJsonSchema({\n * type: 'object',\n * properties: {\n * createdAt: { type: 'string', format: 'date-time' },\n * updatedAt: { type: 'string', format: 'date-time' },\n * },\n * required: ['createdAt'],\n * });\n * ```\n */\nexport const fromJsonSchema = (\n schema: JsonSchemaDefinition,\n options?: JsonSchemaTypeOptions\n): JsonSchemaType => new JsonSchemaType(schema, options);\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAkEA,IAAa,iBAAb,MAAa,eAEX;CACA,AAAiB;CACjB,AAAiB;CAEjB,YACE,YACA,UAAiC,EAAE,EACnC;AACA,OAAK,aAAa;AAClB,OAAK,UAAU;;;;;;;;CASjB,SAA6C;AAE3C,MAAI,KAAK,WAAW,yBAAyB,QAAW;AACtD,OAAI,KAAK,WAAW,yBAAyB,KAC3C,QAAOA,IAAE,OAAOA,IAAE,QAAQ,EAAEA,IAAE,SAAS,CAAC;AAE1C,OAAI,OAAO,KAAK,WAAW,yBAAyB,SAElD,QAAOA,IAAE,OAAOA,IAAE,QAAQ,EAAEA,IAAE,SAAS,CAAC;AAE1C,OAAI,KAAK,WAAW,yBAAyB,MAC3C,QAAOA,IAAE,OAAO,EAAE,CAAC,CAAC,QAAQ;;AAKhC,MAAI,KAAK,WAAW,YAAY;GAC9B,MAAMC,QAAmC,EAAE;GAC3C,MAAM,WAAW,IAAI,IAAI,KAAK,WAAW,YAAY,EAAE,CAAC;AAExD,QAAK,MAAM,CAAC,KAAK,eAAe,OAAO,QACrC,KAAK,WAAW,WACjB,EAAE;IACD,MAAM,YAAY,KAAK,qBAAqB,WAAW;AACvD,UAAM,OAAO,SAAS,IAAI,IAAI,GAAG,YAAY,UAAU,UAAU;;AAGnE,UAAOD,IAAE,OAAO,MAAM,CAAC,aAAa;;AAItC,SAAOA,IAAE,OAAO,EAAE,CAAC,CAAC,aAAa;;;;;CAMnC,gBAAsC;AACpC,SAAO,KAAK;;;;;;CAOd,YAA6C;AAC3C,SAAO;GAAE,MAAM;GAAQ,MAAM,KAAK,QAAQ;GAAM;;;;;CAMlD,UAA8B;AAC5B,SAAO,KAAK,QAAQ;;;;;CAMtB,AAAQ,qBAAqB,QAAyC;AAGpE,UAFa,MAAM,QAAQ,OAAO,KAAK,GAAG,OAAO,KAAK,KAAK,OAAO,MAElE;GACE,KAAK,SACH,QAAOA,IAAE,QAAQ;GACnB,KAAK;GACL,KAAK,UACH,QAAOA,IAAE,QAAQ;GACnB,KAAK,UACH,QAAOA,IAAE,SAAS;GACpB,KAAK;AACH,QAAI,OAAO,SAAS,CAAC,MAAM,QAAQ,OAAO,MAAM,CAC9C,QAAOA,IAAE,MAAM,KAAK,qBAAqB,OAAO,MAAM,CAAC;AAEzD,WAAOA,IAAE,MAAMA,IAAE,SAAS,CAAC;GAC7B,KAAK;AACH,QAAI,OAAO,WACT,QAAO,IAAI,eAAe,OAAO,CAAC,QAAQ;AAE5C,WAAOA,IAAE,OAAOA,IAAE,QAAQ,EAAEA,IAAE,SAAS,CAAC;GAC1C,KAAK,OACH,QAAOA,IAAE,MAAM;GACjB,QACE,QAAOA,IAAE,SAAS;;;;;;;;;;;;;;;;;;;;;;;AAwB1B,MAAa,kBACX,QACA,YACmB,IAAI,eAAe,QAAQ,QAAQ"}
1
+ {"version":3,"file":"JsonSchemaType.js","names":["z"],"sources":["../src/JsonSchemaType.ts"],"sourcesContent":["/**\n * JSON Schema wrapper for ContractSpec compatibility.\n *\n * Wraps JSON Schema definitions to implement the SchemaType interface,\n * enabling direct JSON Schema usage in ContractSpec definitions.\n *\n * @module JsonSchemaType\n */\n\nimport * as z from 'zod';\nimport type { SchemaModelType } from './SchemaModelType';\n\n/**\n * JSON Schema definition structure.\n * Supports standard JSON Schema draft-07/2020-12 properties.\n */\nexport interface JsonSchemaDefinition {\n type?: string | string[];\n properties?: Record<string, JsonSchemaDefinition>;\n required?: string[];\n additionalProperties?: boolean | JsonSchemaDefinition;\n items?: JsonSchemaDefinition | JsonSchemaDefinition[];\n enum?: unknown[];\n const?: unknown;\n oneOf?: JsonSchemaDefinition[];\n anyOf?: JsonSchemaDefinition[];\n allOf?: JsonSchemaDefinition[];\n $ref?: string;\n title?: string;\n description?: string;\n default?: unknown;\n format?: string;\n minimum?: number;\n maximum?: number;\n minLength?: number;\n maxLength?: number;\n pattern?: string;\n [key: string]: unknown;\n}\n\n/**\n * Options for JsonSchemaType wrapper.\n */\nexport interface JsonSchemaTypeOptions {\n /** Name for identification */\n name?: string;\n}\n\n/**\n * Wrapper to use JSON Schema as a SchemaType.\n * Converts to Zod at runtime for validation.\n *\n * @example\n * ```typescript\n * const CustomFieldsSchema = fromJsonSchema({\n * type: 'object',\n * additionalProperties: {\n * oneOf: [\n * { type: 'string' },\n * { type: 'number' },\n * { type: 'boolean' },\n * ],\n * },\n * }, { name: 'CustomFields' });\n * ```\n */\nexport class JsonSchemaType implements SchemaModelType<\n Record<string, unknown>\n> {\n private readonly jsonSchema: JsonSchemaDefinition;\n private readonly options: JsonSchemaTypeOptions;\n\n constructor(\n jsonSchema: JsonSchemaDefinition,\n options: JsonSchemaTypeOptions = {}\n ) {\n this.jsonSchema = jsonSchema;\n this.options = options;\n }\n\n /**\n * Convert JSON Schema to Zod schema for runtime validation.\n *\n * Note: This is a simplified conversion. For complex schemas,\n * consider using a dedicated json-schema-to-zod library.\n */\n getZod(): z.ZodType<Record<string, unknown>> {\n // Handle additionalProperties (dictionary/record types)\n if (this.jsonSchema.additionalProperties !== undefined) {\n if (this.jsonSchema.additionalProperties === true) {\n return z.record(z.string(), z.unknown());\n }\n if (typeof this.jsonSchema.additionalProperties === 'object') {\n // For typed additionalProperties, use union or unknown\n return z.record(z.string(), z.unknown());\n }\n if (this.jsonSchema.additionalProperties === false) {\n return z.object({}).strict();\n }\n }\n\n // Handle explicit properties\n if (this.jsonSchema.properties) {\n const shape: Record<string, z.ZodType> = {};\n const required = new Set(this.jsonSchema.required ?? []);\n\n for (const [key, propSchema] of Object.entries(\n this.jsonSchema.properties\n )) {\n const fieldType = this.convertPropertyToZod(propSchema);\n shape[key] = required.has(key) ? fieldType : fieldType.optional();\n }\n\n return z.object(shape).passthrough();\n }\n\n // Default: passthrough object\n return z.object({}).passthrough();\n }\n\n /**\n * Return the original JSON Schema.\n */\n getJsonSchema(): JsonSchemaDefinition {\n return this.jsonSchema;\n }\n\n /**\n * Return GraphQL type info.\n * JSON Schema types map to JSON scalar in GraphQL.\n */\n getPothos(): { type: string; name?: string } {\n return { type: 'JSON', name: this.options.name };\n }\n\n /**\n * Get the configured name for this schema.\n */\n getName(): string | undefined {\n return this.options.name;\n }\n\n /**\n * Convert a single JSON Schema property to Zod.\n */\n private convertPropertyToZod(schema: JsonSchemaDefinition): z.ZodType {\n const type = Array.isArray(schema.type) ? schema.type[0] : schema.type;\n\n switch (type) {\n case 'string':\n return z.string();\n case 'number':\n case 'integer':\n return z.number();\n case 'boolean':\n return z.boolean();\n case 'array':\n if (schema.items && !Array.isArray(schema.items)) {\n return z.array(this.convertPropertyToZod(schema.items));\n }\n return z.array(z.unknown());\n case 'object':\n if (schema.properties) {\n return new JsonSchemaType(schema).getZod();\n }\n return z.record(z.string(), z.unknown());\n case 'null':\n return z.null();\n default:\n return z.unknown();\n }\n }\n}\n\n/**\n * Helper to create a SchemaType from JSON Schema.\n *\n * @param schema - The JSON Schema definition\n * @param options - Optional configuration\n * @returns A SchemaType-compatible wrapper\n *\n * @example\n * ```typescript\n * const MetadataSchema = fromJsonSchema({\n * type: 'object',\n * properties: {\n * createdAt: { type: 'string', format: 'date-time' },\n * updatedAt: { type: 'string', format: 'date-time' },\n * },\n * required: ['createdAt'],\n * });\n * ```\n */\nexport const fromJsonSchema = (\n schema: JsonSchemaDefinition,\n options?: JsonSchemaTypeOptions\n): JsonSchemaType => new JsonSchemaType(schema, options);\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAkEA,IAAa,iBAAb,MAAa,eAEX;CACA,AAAiB;CACjB,AAAiB;CAEjB,YACE,YACA,UAAiC,EAAE,EACnC;AACA,OAAK,aAAa;AAClB,OAAK,UAAU;;;;;;;;CASjB,SAA6C;AAE3C,MAAI,KAAK,WAAW,yBAAyB,QAAW;AACtD,OAAI,KAAK,WAAW,yBAAyB,KAC3C,QAAOA,IAAE,OAAOA,IAAE,QAAQ,EAAEA,IAAE,SAAS,CAAC;AAE1C,OAAI,OAAO,KAAK,WAAW,yBAAyB,SAElD,QAAOA,IAAE,OAAOA,IAAE,QAAQ,EAAEA,IAAE,SAAS,CAAC;AAE1C,OAAI,KAAK,WAAW,yBAAyB,MAC3C,QAAOA,IAAE,OAAO,EAAE,CAAC,CAAC,QAAQ;;AAKhC,MAAI,KAAK,WAAW,YAAY;GAC9B,MAAM,QAAmC,EAAE;GAC3C,MAAM,WAAW,IAAI,IAAI,KAAK,WAAW,YAAY,EAAE,CAAC;AAExD,QAAK,MAAM,CAAC,KAAK,eAAe,OAAO,QACrC,KAAK,WAAW,WACjB,EAAE;IACD,MAAM,YAAY,KAAK,qBAAqB,WAAW;AACvD,UAAM,OAAO,SAAS,IAAI,IAAI,GAAG,YAAY,UAAU,UAAU;;AAGnE,UAAOA,IAAE,OAAO,MAAM,CAAC,aAAa;;AAItC,SAAOA,IAAE,OAAO,EAAE,CAAC,CAAC,aAAa;;;;;CAMnC,gBAAsC;AACpC,SAAO,KAAK;;;;;;CAOd,YAA6C;AAC3C,SAAO;GAAE,MAAM;GAAQ,MAAM,KAAK,QAAQ;GAAM;;;;;CAMlD,UAA8B;AAC5B,SAAO,KAAK,QAAQ;;;;;CAMtB,AAAQ,qBAAqB,QAAyC;AAGpE,UAFa,MAAM,QAAQ,OAAO,KAAK,GAAG,OAAO,KAAK,KAAK,OAAO,MAElE;GACE,KAAK,SACH,QAAOA,IAAE,QAAQ;GACnB,KAAK;GACL,KAAK,UACH,QAAOA,IAAE,QAAQ;GACnB,KAAK,UACH,QAAOA,IAAE,SAAS;GACpB,KAAK;AACH,QAAI,OAAO,SAAS,CAAC,MAAM,QAAQ,OAAO,MAAM,CAC9C,QAAOA,IAAE,MAAM,KAAK,qBAAqB,OAAO,MAAM,CAAC;AAEzD,WAAOA,IAAE,MAAMA,IAAE,SAAS,CAAC;GAC7B,KAAK;AACH,QAAI,OAAO,WACT,QAAO,IAAI,eAAe,OAAO,CAAC,QAAQ;AAE5C,WAAOA,IAAE,OAAOA,IAAE,QAAQ,EAAEA,IAAE,SAAS,CAAC;GAC1C,KAAK,OACH,QAAOA,IAAE,MAAM;GACjB,QACE,QAAOA,IAAE,SAAS;;;;;;;;;;;;;;;;;;;;;;;AAwB1B,MAAa,kBACX,QACA,YACmB,IAAI,eAAe,QAAQ,QAAQ"}
@@ -1 +1 @@
1
- {"version":3,"file":"SchemaModel.js","names":["config: SchemaModelConfig<Fields>","base: z.ZodType","z"],"sources":["../src/SchemaModel.ts"],"sourcesContent":["import { type AnyFieldType } from './FieldType';\nimport { type AnyEnumType } from './EnumType';\nimport { type SchemaModelType } from './SchemaModelType';\nimport * as z from 'zod';\nimport type { Maybe } from 'graphql/jsutils/Maybe';\n\n/**\n * All types that can be used as field types in a SchemaModel.\n * Supports FieldType, EnumType, nested SchemaModel, or any SchemaType implementation.\n */\ntype FieldLike = AnyFieldType | AnyEnumType | AnySchemaModel | SchemaModelType;\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<\n Fields extends SchemaModelFieldsAnyConfig,\n> implements SchemaModelType {\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\n/**\n * Union of all types that can serve as a schema model.\n * This is the main type expected by OperationSpec, EventSpec, FormSpec, etc.\n *\n * Supports:\n * - SchemaModel instances (native ContractSpec types)\n * - Any SchemaType implementation (ZodSchemaType, JsonSchemaType, etc.)\n */\nexport type AnySchemaModel =\n | SchemaModel<SchemaModelFieldsAnyConfig>\n | SchemaModelType;\n\n/**\n * Type guard to check if a value is a SchemaModel (not just any SchemaType).\n * Use this when you need to access SchemaModel-specific properties like `config`.\n *\n * @param model - The model to check\n * @returns True if the model is a SchemaModel instance\n *\n * @example\n * ```typescript\n * if (isSchemaModel(model)) {\n * // TypeScript knows model.config is available\n * console.log(model.config.name);\n * }\n * ```\n */\nexport function isSchemaModel(\n model: AnySchemaModel | null | undefined\n): model is SchemaModel<SchemaModelFieldsAnyConfig> {\n return (\n model !== null &&\n model !== undefined &&\n 'config' in model &&\n typeof (model as SchemaModel<SchemaModelFieldsAnyConfig>).config ===\n 'object' &&\n 'name' in (model as SchemaModel<SchemaModelFieldsAnyConfig>).config\n );\n}\n\nexport type ZodSchemaModel<Field extends AnySchemaModel> = z.infer<\n ReturnType<Field['getZod']>\n>;\n\ntype InferZodFromType<T> =\n T extends SchemaModel<SchemaModelFieldsAnyConfig>\n ? z.ZodObject<z.ZodRawShape>\n : T extends AnyFieldType\n ? ReturnType<T['getZod']>\n : T extends AnyEnumType\n ? ReturnType<T['getZod']>\n : T extends SchemaModelType\n ? ReturnType<T['getZod']>\n : z.ZodUnknown;\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\n/**\n * Helper to define a SchemaModel with type inference.\n * Equivalent to `new SchemaModel(config)` but with better ergonomics.\n */\nexport const defineSchemaModel = <Fields extends SchemaModelFieldsAnyConfig>(\n config: SchemaModelConfig<Fields>\n): SchemaModel<Fields> => new SchemaModel(config);\n"],"mappings":";;;;;;;;;AAkCA,IAAa,cAAb,MAE6B;CAC3B,YAAY,AAAgBA,QAAmC;EAAnC;;;;;;CAM5B,SAAuC;EACrC,MAAM,QAAQ,OAAO,QAAQ,KAAK,OAAO,OAAO,CAAC,QAC9C,KAAK,CAAC,KAAK,SAAS;GACnB,MAAMC,OACJ,IAAI,KACJ,QAAQ;GACV,MAAM,YAAY,IAAI,UAAUC,IAAE,MAAM,KAAK,GAAG;AAChD,GAAC,IAAkC,OAAO,IAAI,aAC1C,UAAU,UAAU,GACpB;AACJ,UAAO;KAET,EAAE,CACH;AAED,SAAOA,IAAE,OAAO,MAAM;;;CAIxB,iBAAiB;AACf,SAAO,KAAK,OAAO;;;;;;;;;;;;;;;;;;AA+BvB,SAAgB,cACd,OACkD;AAClD,QACE,UAAU,QACV,UAAU,UACV,YAAY,SACZ,OAAQ,MAAkD,WACxD,YACF,UAAW,MAAkD;;;;;;AAgDjE,MAAa,qBACX,WACwB,IAAI,YAAY,OAAO"}
1
+ {"version":3,"file":"SchemaModel.js","names":["z"],"sources":["../src/SchemaModel.ts"],"sourcesContent":["import { type AnyFieldType } from './FieldType';\nimport { type AnyEnumType } from './EnumType';\nimport { type SchemaModelType } from './SchemaModelType';\nimport * as z from 'zod';\nimport type { Maybe } from 'graphql/jsutils/Maybe';\n\n/**\n * All types that can be used as field types in a SchemaModel.\n * Supports FieldType, EnumType, nested SchemaModel, or any SchemaType implementation.\n */\ntype FieldLike = AnyFieldType | AnyEnumType | AnySchemaModel | SchemaModelType;\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<\n Fields extends SchemaModelFieldsAnyConfig,\n> implements SchemaModelType {\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\n/**\n * Union of all types that can serve as a schema model.\n * This is the main type expected by OperationSpec, EventSpec, FormSpec, etc.\n *\n * Supports:\n * - SchemaModel instances (native ContractSpec types)\n * - Any SchemaType implementation (ZodSchemaType, JsonSchemaType, etc.)\n */\nexport type AnySchemaModel =\n | SchemaModel<SchemaModelFieldsAnyConfig>\n | SchemaModelType;\n\n/**\n * Type guard to check if a value is a SchemaModel (not just any SchemaType).\n * Use this when you need to access SchemaModel-specific properties like `config`.\n *\n * @param model - The model to check\n * @returns True if the model is a SchemaModel instance\n *\n * @example\n * ```typescript\n * if (isSchemaModel(model)) {\n * // TypeScript knows model.config is available\n * console.log(model.config.name);\n * }\n * ```\n */\nexport function isSchemaModel(\n model: AnySchemaModel | null | undefined\n): model is SchemaModel<SchemaModelFieldsAnyConfig> {\n return (\n model !== null &&\n model !== undefined &&\n 'config' in model &&\n typeof (model as SchemaModel<SchemaModelFieldsAnyConfig>).config ===\n 'object' &&\n 'name' in (model as SchemaModel<SchemaModelFieldsAnyConfig>).config\n );\n}\n\nexport type ZodSchemaModel<Field extends AnySchemaModel> = z.infer<\n ReturnType<Field['getZod']>\n>;\n\ntype InferZodFromType<T> =\n T extends SchemaModel<SchemaModelFieldsAnyConfig>\n ? z.ZodObject<z.ZodRawShape>\n : T extends AnyFieldType\n ? ReturnType<T['getZod']>\n : T extends AnyEnumType\n ? ReturnType<T['getZod']>\n : T extends SchemaModelType\n ? ReturnType<T['getZod']>\n : z.ZodUnknown;\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\n/**\n * Helper to define a SchemaModel with type inference.\n * Equivalent to `new SchemaModel(config)` but with better ergonomics.\n */\nexport const defineSchemaModel = <Fields extends SchemaModelFieldsAnyConfig>(\n config: SchemaModelConfig<Fields>\n): SchemaModel<Fields> => new SchemaModel(config);\n"],"mappings":";;;;;;;;;AAkCA,IAAa,cAAb,MAE6B;CAC3B,YAAY,AAAgB,QAAmC;EAAnC;;;;;;CAM5B,SAAuC;EACrC,MAAM,QAAQ,OAAO,QAAQ,KAAK,OAAO,OAAO,CAAC,QAC9C,KAAK,CAAC,KAAK,SAAS;GACnB,MAAM,OACJ,IAAI,KACJ,QAAQ;GACV,MAAM,YAAY,IAAI,UAAUA,IAAE,MAAM,KAAK,GAAG;AAChD,GAAC,IAAkC,OAAO,IAAI,aAC1C,UAAU,UAAU,GACpB;AACJ,UAAO;KAET,EAAE,CACH;AAED,SAAOA,IAAE,OAAO,MAAM;;;CAIxB,iBAAiB;AACf,SAAO,KAAK,OAAO;;;;;;;;;;;;;;;;;;AA+BvB,SAAgB,cACd,OACkD;AAClD,QACE,UAAU,QACV,UAAU,UACV,YAAY,SACZ,OAAQ,MAAkD,WACxD,YACF,UAAW,MAAkD;;;;;;AAgDjE,MAAa,qBACX,WACwB,IAAI,YAAY,OAAO"}
@@ -1 +1 @@
1
- {"version":3,"file":"generator.js","names":["lines: string[]","parts: string[]","attrs: string[]","relationParts: string[]","options: string[]","allEntities: EntitySpec[]"],"sources":["../../src/entity/generator.ts"],"sourcesContent":["import type {\n EntityEnumDef,\n EntityEnumField,\n EntityField,\n EntityIndex,\n EntityRelationField,\n EntityScalarField,\n EntitySpec,\n ModuleSchemaContribution,\n} from './types';\n\n/**\n * Options for Prisma schema generation.\n */\nexport interface PrismaGeneratorOptions {\n /** Output file path for the generated schema */\n outputPath?: string;\n /** Prisma datasource provider (default: 'postgresql') */\n provider?: 'postgresql' | 'mysql' | 'sqlite' | 'mongodb' | 'sqlserver';\n /** Prisma generator output path */\n clientOutput?: string;\n /** Include Pothos generator */\n includePothos?: boolean;\n /** Pothos output path */\n pothosOutput?: string;\n}\n\n/**\n * Generate Prisma schema content from entity specifications.\n */\nexport function generatePrismaSchema(\n entities: EntitySpec[],\n options: PrismaGeneratorOptions = {}\n): string {\n const {\n provider = 'postgresql',\n clientOutput = '../../src/generated/prisma',\n includePothos = true,\n pothosOutput = '../../src/generated/pothos-types.ts',\n } = options;\n\n const lines: string[] = [];\n\n // Collect all schemas used\n const schemas = new Set<string>();\n entities.forEach((entity) => {\n if (entity.schema) {\n schemas.add(entity.schema);\n }\n });\n const schemaList = schemas.size > 0 ? Array.from(schemas) : ['public'];\n\n // Datasource block\n lines.push('datasource db {');\n lines.push(` provider = \"${provider}\"`);\n if (schemas.size > 0) {\n lines.push(` schemas = [${schemaList.map((s) => `\"${s}\"`).join(', ')}]`);\n }\n lines.push('}');\n lines.push('');\n\n // Generator block\n lines.push('generator client {');\n lines.push(' provider = \"prisma-client\"');\n lines.push(` output = \"${clientOutput}\"`);\n lines.push('');\n lines.push(' engineType = \"client\"');\n lines.push(' runtime = \"bun\"');\n lines.push(' moduleFormat = \"esm\"');\n lines.push(' generatedFileExtension = \"ts\"');\n lines.push(' importFileExtension = \"ts\"');\n lines.push('}');\n lines.push('');\n\n // Pothos generator (optional)\n if (includePothos) {\n lines.push('generator pothos {');\n lines.push(' provider = \"prisma-pothos-types\"');\n lines.push(' clientOutput = \"./prisma\"');\n lines.push(` output = \"${pothosOutput}\"`);\n lines.push(' generateDatamodel = true');\n lines.push(' documentation = false');\n lines.push('}');\n lines.push('');\n }\n\n // Collect all enums across entities\n const enumMap = new Map<string, EntityEnumDef>();\n entities.forEach((entity) => {\n entity.enums?.forEach((enumDef) => {\n if (!enumMap.has(enumDef.name)) {\n enumMap.set(enumDef.name, enumDef);\n }\n });\n // Also collect inline enums from fields\n Object.values(entity.fields).forEach((field) => {\n if (\n field.kind === 'enum' &&\n field.values &&\n !enumMap.has(field.enumName)\n ) {\n enumMap.set(field.enumName, {\n name: field.enumName,\n values: field.values,\n schema: entity.schema,\n });\n }\n });\n });\n\n // Generate enums\n enumMap.forEach((enumDef) => {\n lines.push(...generateEnumBlock(enumDef));\n lines.push('');\n });\n\n // Generate models\n entities.forEach((entity) => {\n lines.push(...generateModelBlock(entity));\n lines.push('');\n });\n\n return lines.join('\\n');\n}\n\n/**\n * Generate Prisma enum block.\n */\nfunction generateEnumBlock(enumDef: EntityEnumDef): string[] {\n const lines: string[] = [];\n\n if (enumDef.description) {\n lines.push(`/// ${enumDef.description}`);\n }\n\n lines.push(`enum ${enumDef.name} {`);\n enumDef.values.forEach((value) => {\n lines.push(` ${value}`);\n });\n lines.push('');\n if (enumDef.schema) {\n lines.push(` @@schema(\"${enumDef.schema}\")`);\n }\n lines.push('}');\n\n return lines;\n}\n\n/**\n * Generate Prisma model block from entity spec.\n */\nfunction generateModelBlock(entity: EntitySpec): string[] {\n const lines: string[] = [];\n\n // Model description\n if (entity.description) {\n lines.push(`/// ${entity.description}`);\n }\n\n lines.push(`model ${entity.name} {`);\n\n // Generate fields\n Object.entries(entity.fields).forEach(([fieldName, field]) => {\n const fieldLine = generateFieldLine(fieldName, field);\n if (field.description) {\n lines.push(` /// ${field.description}`);\n }\n lines.push(` ${fieldLine}`);\n });\n\n // Generate indexes\n if (entity.indexes && entity.indexes.length > 0) {\n lines.push('');\n entity.indexes.forEach((idx) => {\n lines.push(` ${generateIndexLine(idx)}`);\n });\n }\n\n // Table mapping\n if (entity.map) {\n lines.push(` @@map(\"${entity.map}\")`);\n }\n\n // Schema\n if (entity.schema) {\n lines.push(` @@schema(\"${entity.schema}\")`);\n }\n\n lines.push('}');\n\n return lines;\n}\n\n/**\n * Generate a single field line.\n */\nfunction generateFieldLine(fieldName: string, field: EntityField): string {\n if (field.kind === 'scalar') {\n return generateScalarFieldLine(fieldName, field);\n } else if (field.kind === 'enum') {\n return generateEnumFieldLine(fieldName, field);\n } else if (field.kind === 'relation') {\n return generateRelationFieldLine(fieldName, field);\n }\n throw new Error(`Unknown field kind: ${(field as EntityField).kind}`);\n}\n\n/**\n * Generate scalar field line.\n */\nfunction generateScalarFieldLine(\n fieldName: string,\n field: EntityScalarField\n): string {\n const parts: string[] = [fieldName];\n\n // Type with optional/array modifiers\n let typeStr = field.type;\n if (field.isArray) {\n typeStr += '[]';\n }\n if (field.isOptional) {\n typeStr += '?';\n }\n parts.push(typeStr);\n\n // Attributes\n const attrs: string[] = [];\n\n if (field.isId) {\n attrs.push('@id');\n }\n\n if (field.default !== undefined) {\n if (typeof field.default === 'string' && field.default.includes('(')) {\n // Function default like cuid(), now(), autoincrement()\n attrs.push(`@default(${field.default})`);\n } else if (typeof field.default === 'boolean') {\n attrs.push(`@default(${field.default})`);\n } else if (typeof field.default === 'number') {\n attrs.push(`@default(${field.default})`);\n } else {\n attrs.push(`@default(\"${field.default}\")`);\n }\n }\n\n if (field.updatedAt) {\n attrs.push('@updatedAt');\n }\n\n if (field.isUnique) {\n attrs.push('@unique');\n }\n\n if (field.map) {\n attrs.push(`@map(\"${field.map}\")`);\n }\n\n if (field.dbType) {\n attrs.push(`@db.${field.dbType}`);\n }\n\n if (attrs.length > 0) {\n parts.push(attrs.join(' '));\n }\n\n return parts.join(' ');\n}\n\n/**\n * Generate enum field line.\n */\nfunction generateEnumFieldLine(\n fieldName: string,\n field: EntityEnumField\n): string {\n const parts: string[] = [fieldName];\n\n // Type with optional/array modifiers\n let typeStr = field.enumName;\n if (field.isArray) {\n typeStr += '[]';\n }\n if (field.isOptional) {\n typeStr += '?';\n }\n parts.push(typeStr);\n\n // Attributes\n const attrs: string[] = [];\n\n if (field.default !== undefined) {\n attrs.push(`@default(${field.default})`);\n }\n\n if (field.isUnique) {\n attrs.push('@unique');\n }\n\n if (field.map) {\n attrs.push(`@map(\"${field.map}\")`);\n }\n\n if (attrs.length > 0) {\n parts.push(attrs.join(' '));\n }\n\n return parts.join(' ');\n}\n\n/**\n * Generate relation field line.\n */\nfunction generateRelationFieldLine(\n fieldName: string,\n field: EntityRelationField\n): string {\n const parts: string[] = [fieldName];\n\n // Type with array modifier for hasMany\n let typeStr = field.target;\n if (field.type === 'hasMany') {\n typeStr += '[]';\n }\n if (field.type === 'hasOne') {\n typeStr += '?';\n }\n parts.push(typeStr);\n\n // @relation attribute\n const relationParts: string[] = [];\n\n if (field.name) {\n relationParts.push(`name: \"${field.name}\"`);\n }\n\n if (field.fields && field.fields.length > 0) {\n relationParts.push(`fields: [${field.fields.join(', ')}]`);\n }\n\n if (field.references && field.references.length > 0) {\n relationParts.push(`references: [${field.references.join(', ')}]`);\n }\n\n if (field.onDelete) {\n relationParts.push(`onDelete: ${field.onDelete}`);\n }\n\n if (field.onUpdate) {\n relationParts.push(`onUpdate: ${field.onUpdate}`);\n }\n\n if (relationParts.length > 0) {\n parts.push(`@relation(${relationParts.join(', ')})`);\n }\n\n return parts.join(' ');\n}\n\n/**\n * Generate index line.\n */\nfunction generateIndexLine(idx: EntityIndex): string {\n const fieldList = idx.fields.join(', ');\n const parts: string[] = [];\n\n if (idx.unique) {\n parts.push(`@@unique([${fieldList}]`);\n } else {\n parts.push(`@@index([${fieldList}]`);\n }\n\n const options: string[] = [];\n\n if (idx.name) {\n options.push(`name: \"${idx.name}\"`);\n }\n\n if (idx.type) {\n options.push(`type: ${idx.type}`);\n }\n\n if (options.length > 0) {\n parts[0] += `, ${options.join(', ')}`;\n }\n\n parts[0] += ')';\n\n return parts.join('');\n}\n\n/**\n * Compose multiple module schema contributions into a single schema.\n */\nexport function composeModuleSchemas(\n contributions: ModuleSchemaContribution[],\n options: PrismaGeneratorOptions = {}\n): string {\n // Merge all entities\n const allEntities: EntitySpec[] = [];\n const allEnums = new Map<string, EntityEnumDef>();\n\n contributions.forEach((contrib) => {\n // Add entities\n contrib.entities.forEach((entity) => {\n allEntities.push({\n ...entity,\n module: contrib.moduleId,\n });\n });\n\n // Merge enums\n contrib.enums?.forEach((enumDef) => {\n if (!allEnums.has(enumDef.name)) {\n allEnums.set(enumDef.name, enumDef);\n }\n });\n });\n\n // Add collected enums to first entity or create a holder\n if (allEntities[0] && allEnums.size > 0) {\n allEntities[0] = {\n ...allEntities[0],\n enums: [...(allEntities[0].enums ?? []), ...allEnums.values()],\n };\n }\n\n return generatePrismaSchema(allEntities, options);\n}\n\n/**\n * Generate a single entity's Prisma schema fragment (for modular output).\n */\nexport function generateEntityFragment(entity: EntitySpec): string {\n return generateModelBlock(entity).join('\\n');\n}\n\n/**\n * Generate a single enum's Prisma schema fragment (for modular output).\n */\nexport function generateEnumFragment(enumDef: EntityEnumDef): string {\n return generateEnumBlock(enumDef).join('\\n');\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"generator.js","names":[],"sources":["../../src/entity/generator.ts"],"sourcesContent":["import type {\n EntityEnumDef,\n EntityEnumField,\n EntityField,\n EntityIndex,\n EntityRelationField,\n EntityScalarField,\n EntitySpec,\n ModuleSchemaContribution,\n} from './types';\n\n/**\n * Options for Prisma schema generation.\n */\nexport interface PrismaGeneratorOptions {\n /** Output file path for the generated schema */\n outputPath?: string;\n /** Prisma datasource provider (default: 'postgresql') */\n provider?: 'postgresql' | 'mysql' | 'sqlite' | 'mongodb' | 'sqlserver';\n /** Prisma generator output path */\n clientOutput?: string;\n /** Include Pothos generator */\n includePothos?: boolean;\n /** Pothos output path */\n pothosOutput?: string;\n}\n\n/**\n * Generate Prisma schema content from entity specifications.\n */\nexport function generatePrismaSchema(\n entities: EntitySpec[],\n options: PrismaGeneratorOptions = {}\n): string {\n const {\n provider = 'postgresql',\n clientOutput = '../../src/generated/prisma',\n includePothos = true,\n pothosOutput = '../../src/generated/pothos-types.ts',\n } = options;\n\n const lines: string[] = [];\n\n // Collect all schemas used\n const schemas = new Set<string>();\n entities.forEach((entity) => {\n if (entity.schema) {\n schemas.add(entity.schema);\n }\n });\n const schemaList = schemas.size > 0 ? Array.from(schemas) : ['public'];\n\n // Datasource block\n lines.push('datasource db {');\n lines.push(` provider = \"${provider}\"`);\n if (schemas.size > 0) {\n lines.push(` schemas = [${schemaList.map((s) => `\"${s}\"`).join(', ')}]`);\n }\n lines.push('}');\n lines.push('');\n\n // Generator block\n lines.push('generator client {');\n lines.push(' provider = \"prisma-client\"');\n lines.push(` output = \"${clientOutput}\"`);\n lines.push('');\n lines.push(' engineType = \"client\"');\n lines.push(' runtime = \"bun\"');\n lines.push(' moduleFormat = \"esm\"');\n lines.push(' generatedFileExtension = \"ts\"');\n lines.push(' importFileExtension = \"ts\"');\n lines.push('}');\n lines.push('');\n\n // Pothos generator (optional)\n if (includePothos) {\n lines.push('generator pothos {');\n lines.push(' provider = \"prisma-pothos-types\"');\n lines.push(' clientOutput = \"./prisma\"');\n lines.push(` output = \"${pothosOutput}\"`);\n lines.push(' generateDatamodel = true');\n lines.push(' documentation = false');\n lines.push('}');\n lines.push('');\n }\n\n // Collect all enums across entities\n const enumMap = new Map<string, EntityEnumDef>();\n entities.forEach((entity) => {\n entity.enums?.forEach((enumDef) => {\n if (!enumMap.has(enumDef.name)) {\n enumMap.set(enumDef.name, enumDef);\n }\n });\n // Also collect inline enums from fields\n Object.values(entity.fields).forEach((field) => {\n if (\n field.kind === 'enum' &&\n field.values &&\n !enumMap.has(field.enumName)\n ) {\n enumMap.set(field.enumName, {\n name: field.enumName,\n values: field.values,\n schema: entity.schema,\n });\n }\n });\n });\n\n // Generate enums\n enumMap.forEach((enumDef) => {\n lines.push(...generateEnumBlock(enumDef));\n lines.push('');\n });\n\n // Generate models\n entities.forEach((entity) => {\n lines.push(...generateModelBlock(entity));\n lines.push('');\n });\n\n return lines.join('\\n');\n}\n\n/**\n * Generate Prisma enum block.\n */\nfunction generateEnumBlock(enumDef: EntityEnumDef): string[] {\n const lines: string[] = [];\n\n if (enumDef.description) {\n lines.push(`/// ${enumDef.description}`);\n }\n\n lines.push(`enum ${enumDef.name} {`);\n enumDef.values.forEach((value) => {\n lines.push(` ${value}`);\n });\n lines.push('');\n if (enumDef.schema) {\n lines.push(` @@schema(\"${enumDef.schema}\")`);\n }\n lines.push('}');\n\n return lines;\n}\n\n/**\n * Generate Prisma model block from entity spec.\n */\nfunction generateModelBlock(entity: EntitySpec): string[] {\n const lines: string[] = [];\n\n // Model description\n if (entity.description) {\n lines.push(`/// ${entity.description}`);\n }\n\n lines.push(`model ${entity.name} {`);\n\n // Generate fields\n Object.entries(entity.fields).forEach(([fieldName, field]) => {\n const fieldLine = generateFieldLine(fieldName, field);\n if (field.description) {\n lines.push(` /// ${field.description}`);\n }\n lines.push(` ${fieldLine}`);\n });\n\n // Generate indexes\n if (entity.indexes && entity.indexes.length > 0) {\n lines.push('');\n entity.indexes.forEach((idx) => {\n lines.push(` ${generateIndexLine(idx)}`);\n });\n }\n\n // Table mapping\n if (entity.map) {\n lines.push(` @@map(\"${entity.map}\")`);\n }\n\n // Schema\n if (entity.schema) {\n lines.push(` @@schema(\"${entity.schema}\")`);\n }\n\n lines.push('}');\n\n return lines;\n}\n\n/**\n * Generate a single field line.\n */\nfunction generateFieldLine(fieldName: string, field: EntityField): string {\n if (field.kind === 'scalar') {\n return generateScalarFieldLine(fieldName, field);\n } else if (field.kind === 'enum') {\n return generateEnumFieldLine(fieldName, field);\n } else if (field.kind === 'relation') {\n return generateRelationFieldLine(fieldName, field);\n }\n throw new Error(`Unknown field kind: ${(field as EntityField).kind}`);\n}\n\n/**\n * Generate scalar field line.\n */\nfunction generateScalarFieldLine(\n fieldName: string,\n field: EntityScalarField\n): string {\n const parts: string[] = [fieldName];\n\n // Type with optional/array modifiers\n let typeStr = field.type;\n if (field.isArray) {\n typeStr += '[]';\n }\n if (field.isOptional) {\n typeStr += '?';\n }\n parts.push(typeStr);\n\n // Attributes\n const attrs: string[] = [];\n\n if (field.isId) {\n attrs.push('@id');\n }\n\n if (field.default !== undefined) {\n if (typeof field.default === 'string' && field.default.includes('(')) {\n // Function default like cuid(), now(), autoincrement()\n attrs.push(`@default(${field.default})`);\n } else if (typeof field.default === 'boolean') {\n attrs.push(`@default(${field.default})`);\n } else if (typeof field.default === 'number') {\n attrs.push(`@default(${field.default})`);\n } else {\n attrs.push(`@default(\"${field.default}\")`);\n }\n }\n\n if (field.updatedAt) {\n attrs.push('@updatedAt');\n }\n\n if (field.isUnique) {\n attrs.push('@unique');\n }\n\n if (field.map) {\n attrs.push(`@map(\"${field.map}\")`);\n }\n\n if (field.dbType) {\n attrs.push(`@db.${field.dbType}`);\n }\n\n if (attrs.length > 0) {\n parts.push(attrs.join(' '));\n }\n\n return parts.join(' ');\n}\n\n/**\n * Generate enum field line.\n */\nfunction generateEnumFieldLine(\n fieldName: string,\n field: EntityEnumField\n): string {\n const parts: string[] = [fieldName];\n\n // Type with optional/array modifiers\n let typeStr = field.enumName;\n if (field.isArray) {\n typeStr += '[]';\n }\n if (field.isOptional) {\n typeStr += '?';\n }\n parts.push(typeStr);\n\n // Attributes\n const attrs: string[] = [];\n\n if (field.default !== undefined) {\n attrs.push(`@default(${field.default})`);\n }\n\n if (field.isUnique) {\n attrs.push('@unique');\n }\n\n if (field.map) {\n attrs.push(`@map(\"${field.map}\")`);\n }\n\n if (attrs.length > 0) {\n parts.push(attrs.join(' '));\n }\n\n return parts.join(' ');\n}\n\n/**\n * Generate relation field line.\n */\nfunction generateRelationFieldLine(\n fieldName: string,\n field: EntityRelationField\n): string {\n const parts: string[] = [fieldName];\n\n // Type with array modifier for hasMany\n let typeStr = field.target;\n if (field.type === 'hasMany') {\n typeStr += '[]';\n }\n if (field.type === 'hasOne') {\n typeStr += '?';\n }\n parts.push(typeStr);\n\n // @relation attribute\n const relationParts: string[] = [];\n\n if (field.name) {\n relationParts.push(`name: \"${field.name}\"`);\n }\n\n if (field.fields && field.fields.length > 0) {\n relationParts.push(`fields: [${field.fields.join(', ')}]`);\n }\n\n if (field.references && field.references.length > 0) {\n relationParts.push(`references: [${field.references.join(', ')}]`);\n }\n\n if (field.onDelete) {\n relationParts.push(`onDelete: ${field.onDelete}`);\n }\n\n if (field.onUpdate) {\n relationParts.push(`onUpdate: ${field.onUpdate}`);\n }\n\n if (relationParts.length > 0) {\n parts.push(`@relation(${relationParts.join(', ')})`);\n }\n\n return parts.join(' ');\n}\n\n/**\n * Generate index line.\n */\nfunction generateIndexLine(idx: EntityIndex): string {\n const fieldList = idx.fields.join(', ');\n const parts: string[] = [];\n\n if (idx.unique) {\n parts.push(`@@unique([${fieldList}]`);\n } else {\n parts.push(`@@index([${fieldList}]`);\n }\n\n const options: string[] = [];\n\n if (idx.name) {\n options.push(`name: \"${idx.name}\"`);\n }\n\n if (idx.type) {\n options.push(`type: ${idx.type}`);\n }\n\n if (options.length > 0) {\n parts[0] += `, ${options.join(', ')}`;\n }\n\n parts[0] += ')';\n\n return parts.join('');\n}\n\n/**\n * Compose multiple module schema contributions into a single schema.\n */\nexport function composeModuleSchemas(\n contributions: ModuleSchemaContribution[],\n options: PrismaGeneratorOptions = {}\n): string {\n // Merge all entities\n const allEntities: EntitySpec[] = [];\n const allEnums = new Map<string, EntityEnumDef>();\n\n contributions.forEach((contrib) => {\n // Add entities\n contrib.entities.forEach((entity) => {\n allEntities.push({\n ...entity,\n module: contrib.moduleId,\n });\n });\n\n // Merge enums\n contrib.enums?.forEach((enumDef) => {\n if (!allEnums.has(enumDef.name)) {\n allEnums.set(enumDef.name, enumDef);\n }\n });\n });\n\n // Add collected enums to first entity or create a holder\n if (allEntities[0] && allEnums.size > 0) {\n allEntities[0] = {\n ...allEntities[0],\n enums: [...(allEntities[0].enums ?? []), ...allEnums.values()],\n };\n }\n\n return generatePrismaSchema(allEntities, options);\n}\n\n/**\n * Generate a single entity's Prisma schema fragment (for modular output).\n */\nexport function generateEntityFragment(entity: EntitySpec): string {\n return generateModelBlock(entity).join('\\n');\n}\n\n/**\n * Generate a single enum's Prisma schema fragment (for modular output).\n */\nexport function generateEnumFragment(enumDef: EntityEnumDef): string {\n return generateEnumBlock(enumDef).join('\\n');\n}\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@contractspec/lib.schema",
3
- "version": "1.46.2",
3
+ "version": "1.48.0",
4
4
  "description": "Schema utilities for Zod, JSON Schema, and GraphQL",
5
5
  "keywords": [
6
6
  "contractspec",
@@ -24,13 +24,13 @@
24
24
  "lint:check": "eslint src"
25
25
  },
26
26
  "devDependencies": {
27
- "@contractspec/tool.tsdown": "1.46.2",
28
- "@contractspec/tool.typescript": "1.46.2",
29
- "tsdown": "^0.18.3",
27
+ "@contractspec/tool.tsdown": "1.48.0",
28
+ "@contractspec/tool.typescript": "1.48.0",
29
+ "tsdown": "^0.19.0",
30
30
  "typescript": "^5.9.3"
31
31
  },
32
32
  "dependencies": {
33
- "zod": "^4.1.13",
33
+ "zod": "^4.3.5",
34
34
  "graphql": "^16.8.1"
35
35
  },
36
36
  "files": [
@@ -38,8 +38,6 @@
38
38
  "README.md"
39
39
  ],
40
40
  "type": "module",
41
- "main": "./dist/index.js",
42
- "module": "./dist/index.js",
43
41
  "types": "./dist/index.d.ts",
44
42
  "exports": {
45
43
  ".": "./dist/index.js",