@xata.io/pgroll
Advanced tools
Comparing version 0.0.0-alpha.v92f15f9 to 0.0.0-alpha.v93d50b0007583f9e6521066511ad160b7dba571b
# @xata.io/pgroll | ||
## 0.0.0-alpha.v92f15f9 | ||
## 0.0.0-alpha.v93d50b0007583f9e6521066511ad160b7dba571b | ||
### Patch Changes | ||
- Force canary build | ||
## 0.6.0 | ||
### Minor Changes | ||
- [#1250](https://github.com/xataio/client-ts/pull/1250) [`5c7f7ec8`](https://github.com/xataio/client-ts/commit/5c7f7ec8f85428c8c5de3cc6e49d95d261b340f0) Thanks [@SferaDev](https://github.com/SferaDev)! - Initial release | ||
- [#1478](https://github.com/xataio/client-ts/pull/1478) [`81a06ca`](https://github.com/xataio/client-ts/commit/81a06ca98f4b46bad8e32bf284cc63ada1ab0dc3) Thanks [@SferaDev](https://github.com/SferaDev)! - Update pgroll package to version 0.6.0 | ||
## 0.5.0 | ||
### Minor Changes | ||
- [#1372](https://github.com/xataio/client-ts/pull/1372) [`db01bea`](https://github.com/xataio/client-ts/commit/db01bead243bee3910895d391abe21a80b252a8f) Thanks [@SferaDev](https://github.com/SferaDev)! - Bump pgroll version to match v0.5.0 | ||
## 0.4.4 | ||
### Patch Changes | ||
- [#1250](https://github.com/xataio/client-ts/pull/1250) [`9a6af72`](https://github.com/xataio/client-ts/commit/9a6af72ba4dd7880e8196a0a57d4133930957add) Thanks [@SferaDev](https://github.com/SferaDev)! - Add new package for pgroll migrations |
{ | ||
"name": "@xata.io/pgroll", | ||
"version": "0.0.0-alpha.v92f15f9", | ||
"version": "0.0.0-alpha.v93d50b0007583f9e6521066511ad160b7dba571b", | ||
"description": "Migration tool for PostgreSQL", | ||
@@ -28,7 +28,8 @@ "type": "module", | ||
"dependencies": { | ||
"zod": "^3.22.4", | ||
"zod-to-json-schema": "^3.21.4" | ||
"zod": "^3.23.8", | ||
"zod-to-json-schema": "^3.23.3" | ||
}, | ||
"devDependencies": { | ||
"tsx": "^4.1.2" | ||
"ts-morph": "^23.0.0", | ||
"tsx": "^4.19.1" | ||
}, | ||
@@ -35,0 +36,0 @@ "scripts": { |
@@ -1,5 +0,212 @@ | ||
import { generateJSONSchema } from '../src'; | ||
import fs from 'fs'; | ||
import fs from 'fs/promises'; | ||
import { Project, ScriptTarget, VariableDeclarationKind } from 'ts-morph'; | ||
import { z } from 'zod'; | ||
import { PGROLL_JSON_SCHEMA_URL } from '../src'; | ||
import prettier from 'prettier'; | ||
const jsonSchema = generateJSONSchema(); | ||
fs.writeFileSync('pgroll.schema.json', JSON.stringify(jsonSchema, null, 2)); | ||
type DefintionType = 'string' | 'boolean' | 'number' | 'null'; | ||
type Definition = | ||
| { type: DefintionType | DefintionType[]; description?: string } | ||
| { $ref: string; description?: string } | ||
| { | ||
type: 'object'; | ||
properties: Record<string, Definition>; | ||
oneOf?: unknown[]; | ||
required?: string[]; | ||
description?: string; | ||
additionalProperties?: boolean; | ||
} | ||
| { type: 'array'; items: Definition | Definition[]; description?: string } | ||
| { anyOf: Definition[] }; | ||
const DefinitionTypeSchema = z.enum(['string', 'boolean', 'number', 'null']); | ||
const DefinitionSchema: z.ZodSchema<Definition> = z.lazy(() => | ||
z.union([ | ||
z.object({ | ||
type: z.union([DefinitionTypeSchema, z.array(DefinitionTypeSchema)]), | ||
description: z.string().optional() | ||
}), | ||
z.object({ | ||
$ref: z.string(), | ||
description: z.string().optional() | ||
}), | ||
z.object({ | ||
type: z.literal('object'), | ||
properties: z.record(DefinitionSchema), | ||
// TODO: Add full support for oneOf | ||
oneOf: z.array(z.any()).optional(), | ||
required: z.array(z.string()).optional(), | ||
description: z.string().optional(), | ||
additionalProperties: z.boolean().optional() | ||
}), | ||
z.object({ | ||
type: z.literal('array'), | ||
items: z.union([DefinitionSchema, z.array(DefinitionSchema)]), | ||
description: z.string().optional() | ||
}), | ||
z.object({ | ||
anyOf: z.array(DefinitionSchema) | ||
}) | ||
]) | ||
); | ||
const JSONSchema = z.object({ | ||
$id: z.string(), | ||
$schema: z.string(), | ||
title: z.string(), | ||
description: z.string(), | ||
$defs: z.record(DefinitionSchema) | ||
}); | ||
function buildZodSchema(definition: Definition): string { | ||
if ('$ref' in definition) { | ||
return definition.$ref.replace(/^#\/\$defs\//, '') + 'Definition'; | ||
} | ||
if ('anyOf' in definition) { | ||
const schemas = definition.anyOf.map(buildZodSchema).join(', '); | ||
return `z.union([${schemas}])`; | ||
} | ||
if (definition.type === 'array') { | ||
const itemsSchema = Array.isArray(definition.items) | ||
? buildZodSchema(definition.items[0]) | ||
: buildZodSchema(definition.items); | ||
return `z.array(${itemsSchema})`; | ||
} | ||
if (definition.type === 'object') { | ||
const properties: string[] = []; | ||
for (const [name, property] of Object.entries(definition.properties)) { | ||
const optional = definition.required?.includes(name) ? '' : '.optional()'; | ||
properties.push(`${name}: ${buildZodSchema(property)}${optional}`); | ||
} | ||
return `z.object({ ${properties.join(', ')} })`; | ||
} | ||
const types = typeof definition.type === 'string' ? [definition.type] : definition.type; | ||
const zodTypes = types.map((type) => { | ||
if (type === 'string') return 'z.string()'; | ||
if (type === 'boolean') return 'z.boolean()'; | ||
if (type === 'number') return 'z.number()'; | ||
if (type === 'null') return 'z.null()'; | ||
throw new Error(`Unknown type: ${type}`); | ||
}); | ||
if (zodTypes.length === 1) return zodTypes[0]; | ||
return `z.union([${zodTypes.join(', ')}])`; | ||
} | ||
function getDependencies(definition: Definition): string[] { | ||
if ('$ref' in definition) { | ||
return [definition.$ref.replace(/^#\/\$defs\//, '')]; | ||
} | ||
if ('anyOf' in definition) { | ||
return definition.anyOf.flatMap(getDependencies); | ||
} | ||
if (definition.type === 'array') { | ||
return Array.isArray(definition.items) | ||
? definition.items.flatMap(getDependencies) | ||
: getDependencies(definition.items); | ||
} | ||
if (definition.type === 'object') { | ||
return Object.values(definition.properties).flatMap(getDependencies); | ||
} | ||
return []; | ||
} | ||
function topologicalSort(nodes: [string, Definition][]): [string, Definition][] { | ||
const sorted: [string, Definition][] = []; | ||
const visited = new Set<string>(); | ||
// Recursive function to visit nodes in a topological order | ||
function visit(name: string) { | ||
if (visited.has(name)) return; | ||
visited.add(name); | ||
const node = nodes.find(([n]) => n === name); | ||
if (!node) throw new Error(`Unknown node: ${name}`); | ||
// Visit dependencies before adding the current node | ||
for (const dep of getDependencies(node[1])) { | ||
visit(dep); | ||
} | ||
sorted.push(node); | ||
} | ||
// Visit all nodes in the graph | ||
for (const [name] of nodes) { | ||
visit(name); | ||
} | ||
return sorted; | ||
} | ||
async function main() { | ||
const url = process.env.PGROLL_JSON_SCHEMA_URL ?? PGROLL_JSON_SCHEMA_URL; | ||
const response = await fetch(url).then((response) => response.json()); | ||
const schema = JSONSchema.parse(response); | ||
// Create a TypeScript project | ||
const project = new Project({ compilerOptions: { target: ScriptTarget.ESNext } }); | ||
const schemaFile = project.createSourceFile('schema.ts', '', { overwrite: true }); | ||
const typesFile = project.createSourceFile('types.ts', '', { overwrite: true }); | ||
// Write the JSON schema to a file | ||
schemaFile.addStatements(`export const schema = ${JSON.stringify(response, null, 2)} as const;`); | ||
// Add import statements | ||
typesFile.addImportDeclaration({ moduleSpecifier: 'zod', namedImports: ['z'] }); | ||
// Topologically sort the schema definitions | ||
const statements = topologicalSort(Object.entries(schema.$defs)).map(([name, definition]) => [ | ||
name, | ||
buildZodSchema(definition) | ||
]); | ||
// Generate TypeScript code for each definition | ||
for (const [name, statement] of statements) { | ||
// Add a type alias for the Zod type | ||
typesFile.addTypeAlias({ name, type: `z.infer<typeof ${name}Definition>`, isExported: true }); | ||
// Add a variable statement for the Zod schema | ||
typesFile.addVariableStatement({ | ||
declarationKind: VariableDeclarationKind.Const, | ||
declarations: [{ name: `${name}Definition`, initializer: statement }], | ||
isExported: true | ||
}); | ||
} | ||
// Add a type alias for the OperationType | ||
typesFile.addTypeAlias({ | ||
name: 'OperationType', | ||
type: `(typeof operationTypes)[number]`, | ||
isExported: true | ||
}); | ||
// Extract operation types from the schema and add a variable statement | ||
const operationTypes = (schema.$defs['PgRollOperation'] as any).anyOf.flatMap((def) => Object.keys(def.properties)); | ||
typesFile.addVariableStatement({ | ||
declarationKind: VariableDeclarationKind.Const, | ||
declarations: [ | ||
{ | ||
name: 'operationTypes', | ||
initializer: `[${operationTypes.map((name) => `'${name}'`).join(', ')}] as const` | ||
} | ||
], | ||
isExported: true | ||
}); | ||
// Write the generated TypeScript code to a file | ||
await fs.writeFile('src/schema.ts', prettier.format(schemaFile.getFullText(), { parser: 'typescript' })); | ||
await fs.writeFile('src/types.ts', prettier.format(typesFile.getFullText(), { parser: 'typescript' })); | ||
} | ||
main(); |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
259331
5336
2
11
Updatedzod@^3.23.8
Updatedzod-to-json-schema@^3.23.3