@schemashift/zod-valibot 0.7.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.
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/index.ts","../src/transformer.ts","../src/mappings.ts","../src/handler.ts"],"sourcesContent":["export { createZodToValibotHandler } from './handler.js';\nexport {\n FACTORY_MAPPINGS,\n MODIFIER_METHODS,\n NUMBER_VALIDATION_MAPPINGS,\n OBJECT_METHODS,\n VALIDATION_MAPPINGS,\n} from './mappings.js';\nexport { ZodToValibotTransformer } from './transformer.js';\n","import {\n isInsideStringLiteral,\n type MethodCallInfo,\n parseCallChain,\n type TransformError,\n type TransformResult,\n} from '@schemashift/core';\nimport { type CallExpression, Node, type SourceFile } from 'ts-morph';\nimport {\n FACTORY_MAPPINGS,\n MODIFIER_METHODS,\n NUMBER_VALIDATION_MAPPINGS,\n OBJECT_METHODS,\n VALIDATION_MAPPINGS,\n} from './mappings.js';\n\n/**\n * Zod to Valibot transformer.\n *\n * Converts Zod's fluent method chain API to Valibot's pipe-based API.\n *\n * Zod: z.string().email().min(5).max(100)\n * Valibot: v.pipe(v.string(), v.email(), v.minLength(5), v.maxLength(100))\n *\n * Zod: z.string().optional()\n * Valibot: v.optional(v.string())\n *\n * Zod: z.object({ name: z.string() }).partial()\n * Valibot: v.partial(v.object({ name: v.string() }))\n */\nexport class ZodToValibotTransformer {\n private errors: TransformError[] = [];\n private warnings: string[] = [];\n\n transform(sourceFile: SourceFile): TransformResult {\n this.errors = [];\n this.warnings = [];\n\n const filePath = sourceFile.getFilePath();\n const originalCode = sourceFile.getFullText();\n\n try {\n this.updateImports(sourceFile);\n this.transformTypeHelpers(sourceFile);\n this.transformSchemaExpressions(sourceFile);\n\n return {\n success: this.errors.length === 0,\n filePath,\n originalCode,\n transformedCode: sourceFile.getFullText(),\n errors: this.errors,\n warnings: this.warnings,\n };\n } catch (error) {\n this.errors.push({\n message: error instanceof Error ? error.message : 'Unknown error',\n });\n return {\n success: false,\n filePath,\n originalCode,\n errors: this.errors,\n warnings: this.warnings,\n };\n }\n }\n\n private updateImports(sourceFile: SourceFile): void {\n const zodImports = sourceFile\n .getImportDeclarations()\n .filter((imp) => imp.getModuleSpecifierValue() === 'zod');\n\n for (const imp of zodImports) {\n imp.setModuleSpecifier('valibot');\n\n const namespaceImport = imp.getNamespaceImport();\n const defaultImport = imp.getDefaultImport();\n const namedImports = imp.getNamedImports();\n\n if (namespaceImport) imp.removeNamespaceImport();\n if (defaultImport) imp.removeDefaultImport();\n if (namedImports.length > 0) imp.removeNamedImports();\n\n imp.setNamespaceImport('v');\n }\n }\n\n private transformTypeHelpers(sourceFile: SourceFile): void {\n const fullText = sourceFile.getFullText();\n let newText = fullText;\n newText = newText.replace(/\\bz\\.infer</g, 'v.InferOutput<');\n newText = newText.replace(/\\bz\\.input</g, 'v.InferInput<');\n newText = newText.replace(/\\bz\\.output</g, 'v.InferOutput<');\n if (newText !== fullText) {\n sourceFile.replaceWithText(newText);\n }\n }\n\n private transformSchemaExpressions(sourceFile: SourceFile): void {\n const nodesToTransform: CallExpression[] = [];\n\n sourceFile.forEachDescendant((node) => {\n if (Node.isCallExpression(node)) {\n if (isInsideStringLiteral(node)) return;\n\n const text = node.getText();\n if (text.startsWith('z.')) {\n nodesToTransform.push(node);\n }\n }\n });\n\n const outermost = this.filterOutermostNodes(nodesToTransform);\n outermost.sort((a, b) => b.getStart() - a.getStart());\n\n for (const node of outermost) {\n this.transformCallChain(node);\n }\n }\n\n private filterOutermostNodes(nodes: CallExpression[]): CallExpression[] {\n const result: CallExpression[] = [];\n for (const node of nodes) {\n let isChild = false;\n for (const other of nodes) {\n if (other === node) continue;\n if (other.getStart() <= node.getStart() && other.getEnd() >= node.getEnd()) {\n isChild = true;\n break;\n }\n }\n if (!isChild) result.push(node);\n }\n return result;\n }\n\n private transformCallChain(node: CallExpression): void {\n const chain = parseCallChain(node);\n\n if (!chain) {\n this.transformCallChainFallback(node);\n return;\n }\n\n if (chain.base !== 'z') return;\n\n const filePath = node.getSourceFile().getFilePath();\n const lineNumber = node.getStartLineNumber();\n\n // Map the factory method\n const factoryMapping = FACTORY_MAPPINGS[chain.factoryMethod];\n if (!factoryMapping) {\n this.warnings.push(\n `${filePath}:${lineNumber}: Unknown Zod factory z.${chain.factoryMethod}() — kept as-is`,\n );\n return;\n }\n\n // Determine the factory context for contextual mappings (string vs number vs array)\n const factoryType = chain.factoryMethod;\n\n // Categorize methods\n const pipeValidators: string[] = [];\n const modifiers: Array<{ name: string; args: string[] }> = [];\n const objectMethods: Array<{ name: string; args: string[] }> = [];\n\n for (const method of chain.methods) {\n const result = this.categorizeMethod(method, factoryType, filePath, lineNumber);\n if (result.type === 'pipe') {\n pipeValidators.push(result.code);\n } else if (result.type === 'modifier') {\n modifiers.push({ name: result.name, args: result.args });\n } else if (result.type === 'object') {\n objectMethods.push({ name: result.name, args: result.args });\n }\n // 'skip' type means the method is removed\n }\n\n // Build the Valibot expression\n let result: string;\n\n // Start with the factory call\n const factoryCall = `${factoryMapping}(${chain.factoryArgs.join(', ')})`;\n\n // If there are pipe validators, wrap in v.pipe()\n if (pipeValidators.length > 0) {\n result = `v.pipe(${factoryCall}, ${pipeValidators.join(', ')})`;\n } else {\n result = factoryCall;\n }\n\n // Apply object methods (wrapping)\n for (const objMethod of objectMethods) {\n result = this.applyObjectMethod(objMethod.name, objMethod.args, result, filePath, lineNumber);\n }\n\n // Apply modifiers (wrapping from inside out)\n for (const modifier of modifiers) {\n result = this.applyModifier(modifier.name, modifier.args, result, filePath, lineNumber);\n }\n\n // Post-process: transform any remaining nested z. references in the result\n result = this.transformNestedZodReferences(result);\n\n node.replaceWithText(result);\n }\n\n private transformNestedZodReferences(text: string): string {\n // Replace z.factoryName( with v.factoryName( for all known factories\n let result = text;\n for (const [zodName, valibotName] of Object.entries(FACTORY_MAPPINGS)) {\n const regex = new RegExp(`\\\\bz\\\\.${zodName}\\\\(`, 'g');\n result = result.replace(regex, `${valibotName}(`);\n }\n return result;\n }\n\n private categorizeMethod(\n method: MethodCallInfo,\n factoryType: string,\n filePath: string,\n lineNumber: number,\n ):\n | { type: 'pipe'; code: string }\n | { type: 'modifier'; name: string; args: string[] }\n | { type: 'object'; name: string; args: string[] }\n | { type: 'skip' } {\n const { name, args } = method;\n\n // Check if it's a modifier (wrapping)\n if (MODIFIER_METHODS.has(name)) {\n return { type: 'modifier', name, args };\n }\n\n // Check if it's an object method\n if (OBJECT_METHODS.has(name)) {\n return { type: 'object', name, args };\n }\n\n // Check number-specific validations\n if (\n (factoryType === 'number' || factoryType === 'bigint') &&\n NUMBER_VALIDATION_MAPPINGS[name]\n ) {\n const mapping = NUMBER_VALIDATION_MAPPINGS[name];\n if (!mapping) return { type: 'skip' as const };\n const mappedArgs = mapping.args(args);\n return { type: 'pipe', code: `${mapping.name}(${mappedArgs.join(', ')})` };\n }\n\n // Check string-specific min/max -> minLength/maxLength\n if (factoryType === 'string' || factoryType === 'array') {\n if (name === 'min') {\n const vName = factoryType === 'string' ? 'v.minLength' : 'v.minLength';\n return { type: 'pipe', code: `${vName}(${args.join(', ')})` };\n }\n if (name === 'max') {\n const vName = factoryType === 'string' ? 'v.maxLength' : 'v.maxLength';\n return { type: 'pipe', code: `${vName}(${args.join(', ')})` };\n }\n }\n\n // Check general validation mappings\n const validationMapping = VALIDATION_MAPPINGS[name];\n if (validationMapping === null) {\n // Should be handled by modifier/object checks above, but just in case\n return { type: 'skip' };\n }\n if (validationMapping) {\n return { type: 'pipe', code: `${validationMapping}(${args.join(', ')})` };\n }\n\n // Unknown method\n this.warnings.push(\n `${filePath}:${lineNumber}: Unknown Zod method .${name}() — kept as-is in pipe`,\n );\n return { type: 'pipe', code: `/* TODO: .${name}(${args.join(', ')}) */` };\n }\n\n private applyModifier(\n name: string,\n args: string[],\n inner: string,\n filePath: string,\n lineNumber: number,\n ): string {\n switch (name) {\n case 'optional':\n return `v.optional(${inner})`;\n case 'nullable':\n return `v.nullable(${inner})`;\n case 'nullish':\n return `v.nullish(${inner})`;\n case 'default':\n return `v.optional(${inner}, ${args.join(', ')})`;\n case 'catch':\n return `v.fallback(${inner}, ${args.join(', ')})`;\n case 'readonly':\n return `v.readonly(${inner})`;\n case 'transform':\n return `v.pipe(${inner}, v.transform(${args.join(', ')}))`;\n case 'refine':\n if (args.length >= 2) {\n return `v.pipe(${inner}, v.check(${args[0]}, ${args.slice(1).join(', ')}))`;\n }\n return `v.pipe(${inner}, v.check(${args.join(', ')}))`;\n case 'superRefine':\n this.warnings.push(\n `${filePath}:${lineNumber}: .superRefine() requires manual conversion to Valibot custom validation`,\n );\n return `v.pipe(${inner}, /* TODO: superRefine -> custom validation */ v.check(${args.join(', ')}))`;\n case 'brand':\n return `v.pipe(${inner}, v.brand(${args.join(', ')}))`;\n case 'describe':\n // Valibot doesn't have describe, skip it\n return inner;\n case 'pipe':\n return `v.pipe(${inner}, ${args.join(', ')})`;\n default:\n return inner;\n }\n }\n\n private applyObjectMethod(\n name: string,\n args: string[],\n inner: string,\n filePath: string,\n lineNumber: number,\n ): string {\n switch (name) {\n case 'partial':\n return `v.partial(${inner})`;\n case 'required':\n return `v.required(${inner})`;\n case 'pick':\n return `v.pick(${inner}, ${args.join(', ')})`;\n case 'omit':\n return `v.omit(${inner}, ${args.join(', ')})`;\n case 'extend':\n case 'merge':\n return `v.merge([${inner}, ${args.join(', ')}])`;\n case 'passthrough':\n this.warnings.push(\n `${filePath}:${lineNumber}: .passthrough() — Valibot objects are loose by default, no action needed`,\n );\n return inner;\n case 'strict':\n return `v.strict(${inner})`;\n case 'strip':\n return inner; // Valibot strips unknown by default when using v.object\n case 'catchall':\n this.warnings.push(\n `${filePath}:${lineNumber}: .catchall() requires manual conversion to Valibot record or custom validation`,\n );\n return inner;\n case 'keyof':\n this.warnings.push(\n `${filePath}:${lineNumber}: .keyof() requires manual conversion to v.picklist(Object.keys(...))`,\n );\n return inner;\n default:\n return inner;\n }\n }\n\n private transformCallChainFallback(node: CallExpression): void {\n const text = node.getText();\n const filePath = node.getSourceFile().getFilePath();\n const lineNumber = node.getStartLineNumber();\n\n // Simple regex-based fallback for chains we can't parse\n let transformed = text;\n\n // Replace z. with v.\n transformed = transformed.replace(/^z\\./, 'v.');\n\n // Basic type mappings\n for (const [zodName, valibotName] of Object.entries(FACTORY_MAPPINGS)) {\n const regex = new RegExp(`\\\\bv\\\\.${zodName}\\\\(`, 'g');\n transformed = transformed.replace(regex, `${valibotName}(`);\n }\n\n this.warnings.push(\n `${filePath}:${lineNumber}: Complex Zod expression used fallback conversion — manual review recommended`,\n );\n\n node.replaceWithText(transformed);\n }\n}\n","/**\n * Zod -> Valibot method name mappings.\n *\n * Valibot uses a pipe-based API: v.pipe(v.string(), v.email(), v.minLength(5))\n * instead of Zod's fluent chain: z.string().email().min(5)\n *\n * Factory methods map directly. Validation methods become pipe arguments.\n */\n\n/** Zod factory -> Valibot factory */\nexport const FACTORY_MAPPINGS: Record<string, string> = {\n string: 'v.string',\n number: 'v.number',\n boolean: 'v.boolean',\n date: 'v.date',\n bigint: 'v.bigint',\n symbol: 'v.symbol',\n undefined: 'v.undefined_',\n null: 'v.null_',\n void: 'v.void_',\n any: 'v.any',\n unknown: 'v.unknown',\n never: 'v.never',\n nan: 'v.nan',\n literal: 'v.literal',\n enum: 'v.picklist',\n nativeEnum: 'v.enum_',\n array: 'v.array',\n object: 'v.object',\n record: 'v.record',\n tuple: 'v.tuple',\n union: 'v.union',\n discriminatedUnion: 'v.variant',\n intersection: 'v.intersect',\n lazy: 'v.lazy',\n promise: 'v.promise',\n instanceof: 'v.instance',\n};\n\n/** Zod chain methods -> Valibot pipe validators */\nexport const VALIDATION_MAPPINGS: Record<string, string | null> = {\n // String validations\n email: 'v.email',\n url: 'v.url',\n uuid: 'v.uuid',\n cuid: 'v.cuid2',\n cuid2: 'v.cuid2',\n ulid: 'v.ulid',\n regex: 'v.regex',\n ip: 'v.ip',\n datetime: 'v.isoDateTime',\n date: 'v.isoDate',\n time: 'v.isoTime',\n emoji: 'v.emoji',\n includes: 'v.includes',\n startsWith: 'v.startsWith',\n endsWith: 'v.endsWith',\n trim: 'v.trim',\n toLowerCase: 'v.toLowerCase',\n toUpperCase: 'v.toUpperCase',\n\n // Shared validations (string/number/array/date)\n min: 'v.minLength', // Will be context-sensitive\n max: 'v.maxLength', // Will be context-sensitive\n length: 'v.length',\n\n // Number validations\n gt: 'v.minValue',\n gte: 'v.minValue',\n lt: 'v.maxValue',\n lte: 'v.maxValue',\n int: 'v.integer',\n positive: 'v.minValue',\n negative: 'v.maxValue',\n nonnegative: 'v.minValue',\n nonpositive: 'v.maxValue',\n multipleOf: 'v.multipleOf',\n finite: 'v.finite',\n safe: 'v.safeInteger',\n\n // Array validations\n nonempty: 'v.nonEmpty',\n\n // Schema modifiers (not pipe validators)\n optional: null, // Handled specially\n nullable: null, // Handled specially\n nullish: null, // Handled specially\n default: null, // Handled specially\n catch: null, // Handled specially\n transform: null, // Handled specially\n refine: null, // Handled specially\n superRefine: null, // Handled specially\n brand: null, // Handled specially\n readonly: null, // Handled specially\n pipe: null, // Handled specially\n describe: null, // Handled specially\n\n // Object methods\n partial: null, // Handled specially\n required: null, // Handled specially\n pick: null, // Handled specially\n omit: null, // Handled specially\n extend: null, // Handled specially\n merge: null, // Handled specially\n passthrough: null, // Handled specially\n strict: null, // Handled specially\n strip: null, // Handled specially\n catchall: null, // Handled specially\n keyof: null, // Handled specially\n\n // Array methods\n element: null, // Part of factory\n};\n\n/** Zod number methods that need specific arg mapping to Valibot */\nexport const NUMBER_VALIDATION_MAPPINGS: Record<\n string,\n { name: string; args: (args: string[]) => string[] }\n> = {\n min: { name: 'v.minValue', args: (a) => a },\n max: { name: 'v.maxValue', args: (a) => a },\n gt: { name: 'v.minValue', args: (a) => (a.length > 0 ? [String(Number(a[0]) + 1)] : a) },\n lt: { name: 'v.maxValue', args: (a) => (a.length > 0 ? [String(Number(a[0]) - 1)] : a) },\n gte: { name: 'v.minValue', args: (a) => a },\n lte: { name: 'v.maxValue', args: (a) => a },\n positive: { name: 'v.minValue', args: () => ['1'] },\n negative: { name: 'v.maxValue', args: () => ['-1'] },\n nonnegative: { name: 'v.minValue', args: () => ['0'] },\n nonpositive: { name: 'v.maxValue', args: () => ['0'] },\n};\n\n/** Methods that wrap schemas (modifiers) */\nexport const MODIFIER_METHODS = new Set([\n 'optional',\n 'nullable',\n 'nullish',\n 'default',\n 'catch',\n 'readonly',\n 'brand',\n 'describe',\n 'transform',\n 'refine',\n 'superRefine',\n 'pipe',\n]);\n\n/** Object-specific methods */\nexport const OBJECT_METHODS = new Set([\n 'partial',\n 'required',\n 'pick',\n 'omit',\n 'extend',\n 'merge',\n 'passthrough',\n 'strict',\n 'strip',\n 'catchall',\n 'keyof',\n]);\n","import type { TransformHandler, TransformOptions, TransformResult } from '@schemashift/core';\nimport type { SourceFile } from 'ts-morph';\nimport { ZodToValibotTransformer } from './transformer.js';\n\nexport function createZodToValibotHandler(): TransformHandler {\n const transformer = new ZodToValibotTransformer();\n return {\n transform(sourceFile: SourceFile, _options: TransformOptions): TransformResult {\n return transformer.transform(sourceFile);\n },\n };\n}\n"],"mappings":"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","names":["result"]}
@@ -0,0 +1,58 @@
1
+ import { TransformHandler, TransformResult } from '@schemashift/core';
2
+ import { SourceFile } from 'ts-morph';
3
+
4
+ declare function createZodToValibotHandler(): TransformHandler;
5
+
6
+ /**
7
+ * Zod -> Valibot method name mappings.
8
+ *
9
+ * Valibot uses a pipe-based API: v.pipe(v.string(), v.email(), v.minLength(5))
10
+ * instead of Zod's fluent chain: z.string().email().min(5)
11
+ *
12
+ * Factory methods map directly. Validation methods become pipe arguments.
13
+ */
14
+ /** Zod factory -> Valibot factory */
15
+ declare const FACTORY_MAPPINGS: Record<string, string>;
16
+ /** Zod chain methods -> Valibot pipe validators */
17
+ declare const VALIDATION_MAPPINGS: Record<string, string | null>;
18
+ /** Zod number methods that need specific arg mapping to Valibot */
19
+ declare const NUMBER_VALIDATION_MAPPINGS: Record<string, {
20
+ name: string;
21
+ args: (args: string[]) => string[];
22
+ }>;
23
+ /** Methods that wrap schemas (modifiers) */
24
+ declare const MODIFIER_METHODS: Set<string>;
25
+ /** Object-specific methods */
26
+ declare const OBJECT_METHODS: Set<string>;
27
+
28
+ /**
29
+ * Zod to Valibot transformer.
30
+ *
31
+ * Converts Zod's fluent method chain API to Valibot's pipe-based API.
32
+ *
33
+ * Zod: z.string().email().min(5).max(100)
34
+ * Valibot: v.pipe(v.string(), v.email(), v.minLength(5), v.maxLength(100))
35
+ *
36
+ * Zod: z.string().optional()
37
+ * Valibot: v.optional(v.string())
38
+ *
39
+ * Zod: z.object({ name: z.string() }).partial()
40
+ * Valibot: v.partial(v.object({ name: v.string() }))
41
+ */
42
+ declare class ZodToValibotTransformer {
43
+ private errors;
44
+ private warnings;
45
+ transform(sourceFile: SourceFile): TransformResult;
46
+ private updateImports;
47
+ private transformTypeHelpers;
48
+ private transformSchemaExpressions;
49
+ private filterOutermostNodes;
50
+ private transformCallChain;
51
+ private transformNestedZodReferences;
52
+ private categorizeMethod;
53
+ private applyModifier;
54
+ private applyObjectMethod;
55
+ private transformCallChainFallback;
56
+ }
57
+
58
+ export { FACTORY_MAPPINGS, MODIFIER_METHODS, NUMBER_VALIDATION_MAPPINGS, OBJECT_METHODS, VALIDATION_MAPPINGS, ZodToValibotTransformer, createZodToValibotHandler };
@@ -0,0 +1,58 @@
1
+ import { TransformHandler, TransformResult } from '@schemashift/core';
2
+ import { SourceFile } from 'ts-morph';
3
+
4
+ declare function createZodToValibotHandler(): TransformHandler;
5
+
6
+ /**
7
+ * Zod -> Valibot method name mappings.
8
+ *
9
+ * Valibot uses a pipe-based API: v.pipe(v.string(), v.email(), v.minLength(5))
10
+ * instead of Zod's fluent chain: z.string().email().min(5)
11
+ *
12
+ * Factory methods map directly. Validation methods become pipe arguments.
13
+ */
14
+ /** Zod factory -> Valibot factory */
15
+ declare const FACTORY_MAPPINGS: Record<string, string>;
16
+ /** Zod chain methods -> Valibot pipe validators */
17
+ declare const VALIDATION_MAPPINGS: Record<string, string | null>;
18
+ /** Zod number methods that need specific arg mapping to Valibot */
19
+ declare const NUMBER_VALIDATION_MAPPINGS: Record<string, {
20
+ name: string;
21
+ args: (args: string[]) => string[];
22
+ }>;
23
+ /** Methods that wrap schemas (modifiers) */
24
+ declare const MODIFIER_METHODS: Set<string>;
25
+ /** Object-specific methods */
26
+ declare const OBJECT_METHODS: Set<string>;
27
+
28
+ /**
29
+ * Zod to Valibot transformer.
30
+ *
31
+ * Converts Zod's fluent method chain API to Valibot's pipe-based API.
32
+ *
33
+ * Zod: z.string().email().min(5).max(100)
34
+ * Valibot: v.pipe(v.string(), v.email(), v.minLength(5), v.maxLength(100))
35
+ *
36
+ * Zod: z.string().optional()
37
+ * Valibot: v.optional(v.string())
38
+ *
39
+ * Zod: z.object({ name: z.string() }).partial()
40
+ * Valibot: v.partial(v.object({ name: v.string() }))
41
+ */
42
+ declare class ZodToValibotTransformer {
43
+ private errors;
44
+ private warnings;
45
+ transform(sourceFile: SourceFile): TransformResult;
46
+ private updateImports;
47
+ private transformTypeHelpers;
48
+ private transformSchemaExpressions;
49
+ private filterOutermostNodes;
50
+ private transformCallChain;
51
+ private transformNestedZodReferences;
52
+ private categorizeMethod;
53
+ private applyModifier;
54
+ private applyObjectMethod;
55
+ private transformCallChainFallback;
56
+ }
57
+
58
+ export { FACTORY_MAPPINGS, MODIFIER_METHODS, NUMBER_VALIDATION_MAPPINGS, OBJECT_METHODS, VALIDATION_MAPPINGS, ZodToValibotTransformer, createZodToValibotHandler };
package/dist/index.js ADDED
@@ -0,0 +1,457 @@
1
+ // src/transformer.ts
2
+ import {
3
+ isInsideStringLiteral,
4
+ parseCallChain
5
+ } from "@schemashift/core";
6
+ import { Node } from "ts-morph";
7
+
8
+ // src/mappings.ts
9
+ var FACTORY_MAPPINGS = {
10
+ string: "v.string",
11
+ number: "v.number",
12
+ boolean: "v.boolean",
13
+ date: "v.date",
14
+ bigint: "v.bigint",
15
+ symbol: "v.symbol",
16
+ undefined: "v.undefined_",
17
+ null: "v.null_",
18
+ void: "v.void_",
19
+ any: "v.any",
20
+ unknown: "v.unknown",
21
+ never: "v.never",
22
+ nan: "v.nan",
23
+ literal: "v.literal",
24
+ enum: "v.picklist",
25
+ nativeEnum: "v.enum_",
26
+ array: "v.array",
27
+ object: "v.object",
28
+ record: "v.record",
29
+ tuple: "v.tuple",
30
+ union: "v.union",
31
+ discriminatedUnion: "v.variant",
32
+ intersection: "v.intersect",
33
+ lazy: "v.lazy",
34
+ promise: "v.promise",
35
+ instanceof: "v.instance"
36
+ };
37
+ var VALIDATION_MAPPINGS = {
38
+ // String validations
39
+ email: "v.email",
40
+ url: "v.url",
41
+ uuid: "v.uuid",
42
+ cuid: "v.cuid2",
43
+ cuid2: "v.cuid2",
44
+ ulid: "v.ulid",
45
+ regex: "v.regex",
46
+ ip: "v.ip",
47
+ datetime: "v.isoDateTime",
48
+ date: "v.isoDate",
49
+ time: "v.isoTime",
50
+ emoji: "v.emoji",
51
+ includes: "v.includes",
52
+ startsWith: "v.startsWith",
53
+ endsWith: "v.endsWith",
54
+ trim: "v.trim",
55
+ toLowerCase: "v.toLowerCase",
56
+ toUpperCase: "v.toUpperCase",
57
+ // Shared validations (string/number/array/date)
58
+ min: "v.minLength",
59
+ // Will be context-sensitive
60
+ max: "v.maxLength",
61
+ // Will be context-sensitive
62
+ length: "v.length",
63
+ // Number validations
64
+ gt: "v.minValue",
65
+ gte: "v.minValue",
66
+ lt: "v.maxValue",
67
+ lte: "v.maxValue",
68
+ int: "v.integer",
69
+ positive: "v.minValue",
70
+ negative: "v.maxValue",
71
+ nonnegative: "v.minValue",
72
+ nonpositive: "v.maxValue",
73
+ multipleOf: "v.multipleOf",
74
+ finite: "v.finite",
75
+ safe: "v.safeInteger",
76
+ // Array validations
77
+ nonempty: "v.nonEmpty",
78
+ // Schema modifiers (not pipe validators)
79
+ optional: null,
80
+ // Handled specially
81
+ nullable: null,
82
+ // Handled specially
83
+ nullish: null,
84
+ // Handled specially
85
+ default: null,
86
+ // Handled specially
87
+ catch: null,
88
+ // Handled specially
89
+ transform: null,
90
+ // Handled specially
91
+ refine: null,
92
+ // Handled specially
93
+ superRefine: null,
94
+ // Handled specially
95
+ brand: null,
96
+ // Handled specially
97
+ readonly: null,
98
+ // Handled specially
99
+ pipe: null,
100
+ // Handled specially
101
+ describe: null,
102
+ // Handled specially
103
+ // Object methods
104
+ partial: null,
105
+ // Handled specially
106
+ required: null,
107
+ // Handled specially
108
+ pick: null,
109
+ // Handled specially
110
+ omit: null,
111
+ // Handled specially
112
+ extend: null,
113
+ // Handled specially
114
+ merge: null,
115
+ // Handled specially
116
+ passthrough: null,
117
+ // Handled specially
118
+ strict: null,
119
+ // Handled specially
120
+ strip: null,
121
+ // Handled specially
122
+ catchall: null,
123
+ // Handled specially
124
+ keyof: null,
125
+ // Handled specially
126
+ // Array methods
127
+ element: null
128
+ // Part of factory
129
+ };
130
+ var NUMBER_VALIDATION_MAPPINGS = {
131
+ min: { name: "v.minValue", args: (a) => a },
132
+ max: { name: "v.maxValue", args: (a) => a },
133
+ gt: { name: "v.minValue", args: (a) => a.length > 0 ? [String(Number(a[0]) + 1)] : a },
134
+ lt: { name: "v.maxValue", args: (a) => a.length > 0 ? [String(Number(a[0]) - 1)] : a },
135
+ gte: { name: "v.minValue", args: (a) => a },
136
+ lte: { name: "v.maxValue", args: (a) => a },
137
+ positive: { name: "v.minValue", args: () => ["1"] },
138
+ negative: { name: "v.maxValue", args: () => ["-1"] },
139
+ nonnegative: { name: "v.minValue", args: () => ["0"] },
140
+ nonpositive: { name: "v.maxValue", args: () => ["0"] }
141
+ };
142
+ var MODIFIER_METHODS = /* @__PURE__ */ new Set([
143
+ "optional",
144
+ "nullable",
145
+ "nullish",
146
+ "default",
147
+ "catch",
148
+ "readonly",
149
+ "brand",
150
+ "describe",
151
+ "transform",
152
+ "refine",
153
+ "superRefine",
154
+ "pipe"
155
+ ]);
156
+ var OBJECT_METHODS = /* @__PURE__ */ new Set([
157
+ "partial",
158
+ "required",
159
+ "pick",
160
+ "omit",
161
+ "extend",
162
+ "merge",
163
+ "passthrough",
164
+ "strict",
165
+ "strip",
166
+ "catchall",
167
+ "keyof"
168
+ ]);
169
+
170
+ // src/transformer.ts
171
+ var ZodToValibotTransformer = class {
172
+ errors = [];
173
+ warnings = [];
174
+ transform(sourceFile) {
175
+ this.errors = [];
176
+ this.warnings = [];
177
+ const filePath = sourceFile.getFilePath();
178
+ const originalCode = sourceFile.getFullText();
179
+ try {
180
+ this.updateImports(sourceFile);
181
+ this.transformTypeHelpers(sourceFile);
182
+ this.transformSchemaExpressions(sourceFile);
183
+ return {
184
+ success: this.errors.length === 0,
185
+ filePath,
186
+ originalCode,
187
+ transformedCode: sourceFile.getFullText(),
188
+ errors: this.errors,
189
+ warnings: this.warnings
190
+ };
191
+ } catch (error) {
192
+ this.errors.push({
193
+ message: error instanceof Error ? error.message : "Unknown error"
194
+ });
195
+ return {
196
+ success: false,
197
+ filePath,
198
+ originalCode,
199
+ errors: this.errors,
200
+ warnings: this.warnings
201
+ };
202
+ }
203
+ }
204
+ updateImports(sourceFile) {
205
+ const zodImports = sourceFile.getImportDeclarations().filter((imp) => imp.getModuleSpecifierValue() === "zod");
206
+ for (const imp of zodImports) {
207
+ imp.setModuleSpecifier("valibot");
208
+ const namespaceImport = imp.getNamespaceImport();
209
+ const defaultImport = imp.getDefaultImport();
210
+ const namedImports = imp.getNamedImports();
211
+ if (namespaceImport) imp.removeNamespaceImport();
212
+ if (defaultImport) imp.removeDefaultImport();
213
+ if (namedImports.length > 0) imp.removeNamedImports();
214
+ imp.setNamespaceImport("v");
215
+ }
216
+ }
217
+ transformTypeHelpers(sourceFile) {
218
+ const fullText = sourceFile.getFullText();
219
+ let newText = fullText;
220
+ newText = newText.replace(/\bz\.infer</g, "v.InferOutput<");
221
+ newText = newText.replace(/\bz\.input</g, "v.InferInput<");
222
+ newText = newText.replace(/\bz\.output</g, "v.InferOutput<");
223
+ if (newText !== fullText) {
224
+ sourceFile.replaceWithText(newText);
225
+ }
226
+ }
227
+ transformSchemaExpressions(sourceFile) {
228
+ const nodesToTransform = [];
229
+ sourceFile.forEachDescendant((node) => {
230
+ if (Node.isCallExpression(node)) {
231
+ if (isInsideStringLiteral(node)) return;
232
+ const text = node.getText();
233
+ if (text.startsWith("z.")) {
234
+ nodesToTransform.push(node);
235
+ }
236
+ }
237
+ });
238
+ const outermost = this.filterOutermostNodes(nodesToTransform);
239
+ outermost.sort((a, b) => b.getStart() - a.getStart());
240
+ for (const node of outermost) {
241
+ this.transformCallChain(node);
242
+ }
243
+ }
244
+ filterOutermostNodes(nodes) {
245
+ const result = [];
246
+ for (const node of nodes) {
247
+ let isChild = false;
248
+ for (const other of nodes) {
249
+ if (other === node) continue;
250
+ if (other.getStart() <= node.getStart() && other.getEnd() >= node.getEnd()) {
251
+ isChild = true;
252
+ break;
253
+ }
254
+ }
255
+ if (!isChild) result.push(node);
256
+ }
257
+ return result;
258
+ }
259
+ transformCallChain(node) {
260
+ const chain = parseCallChain(node);
261
+ if (!chain) {
262
+ this.transformCallChainFallback(node);
263
+ return;
264
+ }
265
+ if (chain.base !== "z") return;
266
+ const filePath = node.getSourceFile().getFilePath();
267
+ const lineNumber = node.getStartLineNumber();
268
+ const factoryMapping = FACTORY_MAPPINGS[chain.factoryMethod];
269
+ if (!factoryMapping) {
270
+ this.warnings.push(
271
+ `${filePath}:${lineNumber}: Unknown Zod factory z.${chain.factoryMethod}() \u2014 kept as-is`
272
+ );
273
+ return;
274
+ }
275
+ const factoryType = chain.factoryMethod;
276
+ const pipeValidators = [];
277
+ const modifiers = [];
278
+ const objectMethods = [];
279
+ for (const method of chain.methods) {
280
+ const result2 = this.categorizeMethod(method, factoryType, filePath, lineNumber);
281
+ if (result2.type === "pipe") {
282
+ pipeValidators.push(result2.code);
283
+ } else if (result2.type === "modifier") {
284
+ modifiers.push({ name: result2.name, args: result2.args });
285
+ } else if (result2.type === "object") {
286
+ objectMethods.push({ name: result2.name, args: result2.args });
287
+ }
288
+ }
289
+ let result;
290
+ const factoryCall = `${factoryMapping}(${chain.factoryArgs.join(", ")})`;
291
+ if (pipeValidators.length > 0) {
292
+ result = `v.pipe(${factoryCall}, ${pipeValidators.join(", ")})`;
293
+ } else {
294
+ result = factoryCall;
295
+ }
296
+ for (const objMethod of objectMethods) {
297
+ result = this.applyObjectMethod(objMethod.name, objMethod.args, result, filePath, lineNumber);
298
+ }
299
+ for (const modifier of modifiers) {
300
+ result = this.applyModifier(modifier.name, modifier.args, result, filePath, lineNumber);
301
+ }
302
+ result = this.transformNestedZodReferences(result);
303
+ node.replaceWithText(result);
304
+ }
305
+ transformNestedZodReferences(text) {
306
+ let result = text;
307
+ for (const [zodName, valibotName] of Object.entries(FACTORY_MAPPINGS)) {
308
+ const regex = new RegExp(`\\bz\\.${zodName}\\(`, "g");
309
+ result = result.replace(regex, `${valibotName}(`);
310
+ }
311
+ return result;
312
+ }
313
+ categorizeMethod(method, factoryType, filePath, lineNumber) {
314
+ const { name, args } = method;
315
+ if (MODIFIER_METHODS.has(name)) {
316
+ return { type: "modifier", name, args };
317
+ }
318
+ if (OBJECT_METHODS.has(name)) {
319
+ return { type: "object", name, args };
320
+ }
321
+ if ((factoryType === "number" || factoryType === "bigint") && NUMBER_VALIDATION_MAPPINGS[name]) {
322
+ const mapping = NUMBER_VALIDATION_MAPPINGS[name];
323
+ if (!mapping) return { type: "skip" };
324
+ const mappedArgs = mapping.args(args);
325
+ return { type: "pipe", code: `${mapping.name}(${mappedArgs.join(", ")})` };
326
+ }
327
+ if (factoryType === "string" || factoryType === "array") {
328
+ if (name === "min") {
329
+ const vName = factoryType === "string" ? "v.minLength" : "v.minLength";
330
+ return { type: "pipe", code: `${vName}(${args.join(", ")})` };
331
+ }
332
+ if (name === "max") {
333
+ const vName = factoryType === "string" ? "v.maxLength" : "v.maxLength";
334
+ return { type: "pipe", code: `${vName}(${args.join(", ")})` };
335
+ }
336
+ }
337
+ const validationMapping = VALIDATION_MAPPINGS[name];
338
+ if (validationMapping === null) {
339
+ return { type: "skip" };
340
+ }
341
+ if (validationMapping) {
342
+ return { type: "pipe", code: `${validationMapping}(${args.join(", ")})` };
343
+ }
344
+ this.warnings.push(
345
+ `${filePath}:${lineNumber}: Unknown Zod method .${name}() \u2014 kept as-is in pipe`
346
+ );
347
+ return { type: "pipe", code: `/* TODO: .${name}(${args.join(", ")}) */` };
348
+ }
349
+ applyModifier(name, args, inner, filePath, lineNumber) {
350
+ switch (name) {
351
+ case "optional":
352
+ return `v.optional(${inner})`;
353
+ case "nullable":
354
+ return `v.nullable(${inner})`;
355
+ case "nullish":
356
+ return `v.nullish(${inner})`;
357
+ case "default":
358
+ return `v.optional(${inner}, ${args.join(", ")})`;
359
+ case "catch":
360
+ return `v.fallback(${inner}, ${args.join(", ")})`;
361
+ case "readonly":
362
+ return `v.readonly(${inner})`;
363
+ case "transform":
364
+ return `v.pipe(${inner}, v.transform(${args.join(", ")}))`;
365
+ case "refine":
366
+ if (args.length >= 2) {
367
+ return `v.pipe(${inner}, v.check(${args[0]}, ${args.slice(1).join(", ")}))`;
368
+ }
369
+ return `v.pipe(${inner}, v.check(${args.join(", ")}))`;
370
+ case "superRefine":
371
+ this.warnings.push(
372
+ `${filePath}:${lineNumber}: .superRefine() requires manual conversion to Valibot custom validation`
373
+ );
374
+ return `v.pipe(${inner}, /* TODO: superRefine -> custom validation */ v.check(${args.join(", ")}))`;
375
+ case "brand":
376
+ return `v.pipe(${inner}, v.brand(${args.join(", ")}))`;
377
+ case "describe":
378
+ return inner;
379
+ case "pipe":
380
+ return `v.pipe(${inner}, ${args.join(", ")})`;
381
+ default:
382
+ return inner;
383
+ }
384
+ }
385
+ applyObjectMethod(name, args, inner, filePath, lineNumber) {
386
+ switch (name) {
387
+ case "partial":
388
+ return `v.partial(${inner})`;
389
+ case "required":
390
+ return `v.required(${inner})`;
391
+ case "pick":
392
+ return `v.pick(${inner}, ${args.join(", ")})`;
393
+ case "omit":
394
+ return `v.omit(${inner}, ${args.join(", ")})`;
395
+ case "extend":
396
+ case "merge":
397
+ return `v.merge([${inner}, ${args.join(", ")}])`;
398
+ case "passthrough":
399
+ this.warnings.push(
400
+ `${filePath}:${lineNumber}: .passthrough() \u2014 Valibot objects are loose by default, no action needed`
401
+ );
402
+ return inner;
403
+ case "strict":
404
+ return `v.strict(${inner})`;
405
+ case "strip":
406
+ return inner;
407
+ // Valibot strips unknown by default when using v.object
408
+ case "catchall":
409
+ this.warnings.push(
410
+ `${filePath}:${lineNumber}: .catchall() requires manual conversion to Valibot record or custom validation`
411
+ );
412
+ return inner;
413
+ case "keyof":
414
+ this.warnings.push(
415
+ `${filePath}:${lineNumber}: .keyof() requires manual conversion to v.picklist(Object.keys(...))`
416
+ );
417
+ return inner;
418
+ default:
419
+ return inner;
420
+ }
421
+ }
422
+ transformCallChainFallback(node) {
423
+ const text = node.getText();
424
+ const filePath = node.getSourceFile().getFilePath();
425
+ const lineNumber = node.getStartLineNumber();
426
+ let transformed = text;
427
+ transformed = transformed.replace(/^z\./, "v.");
428
+ for (const [zodName, valibotName] of Object.entries(FACTORY_MAPPINGS)) {
429
+ const regex = new RegExp(`\\bv\\.${zodName}\\(`, "g");
430
+ transformed = transformed.replace(regex, `${valibotName}(`);
431
+ }
432
+ this.warnings.push(
433
+ `${filePath}:${lineNumber}: Complex Zod expression used fallback conversion \u2014 manual review recommended`
434
+ );
435
+ node.replaceWithText(transformed);
436
+ }
437
+ };
438
+
439
+ // src/handler.ts
440
+ function createZodToValibotHandler() {
441
+ const transformer = new ZodToValibotTransformer();
442
+ return {
443
+ transform(sourceFile, _options) {
444
+ return transformer.transform(sourceFile);
445
+ }
446
+ };
447
+ }
448
+ export {
449
+ FACTORY_MAPPINGS,
450
+ MODIFIER_METHODS,
451
+ NUMBER_VALIDATION_MAPPINGS,
452
+ OBJECT_METHODS,
453
+ VALIDATION_MAPPINGS,
454
+ ZodToValibotTransformer,
455
+ createZodToValibotHandler
456
+ };
457
+ //# sourceMappingURL=index.js.map