on-zero 0.2.9 → 0.3.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.
- package/dist/cjs/cli.cjs +0 -0
- package/dist/cjs/cli.native.js +0 -0
- package/dist/cjs/createPermissions.native.js.map +1 -1
- package/dist/cjs/createSchemaFromDrizzle.cjs +30 -0
- package/dist/cjs/createSchemaFromDrizzle.native.js +33 -0
- package/dist/cjs/createSchemaFromDrizzle.native.js.map +1 -0
- package/dist/cjs/createUseQuery.cjs +1 -1
- package/dist/cjs/createUseQuery.native.js +1 -1
- package/dist/cjs/createZeroClient.cjs +1 -1
- package/dist/cjs/createZeroClient.native.js +1 -1
- package/dist/cjs/createZeroClient.native.js.map +1 -1
- package/dist/cjs/createZeroServer.native.js.map +1 -1
- package/dist/cjs/generate.cjs +64 -12
- package/dist/cjs/generate.native.js +114 -12
- package/dist/cjs/generate.native.js.map +1 -1
- package/dist/cjs/helpers/createMutators.native.js.map +1 -1
- package/dist/cjs/index.cjs +2 -1
- package/dist/cjs/index.native.js +1 -0
- package/dist/cjs/index.native.js.map +1 -1
- package/dist/cjs/mutations.cjs +1 -1
- package/dist/cjs/mutations.native.js +1 -1
- package/dist/cjs/mutations.native.js.map +1 -1
- package/dist/cjs/server.native.js +1 -18
- package/dist/cjs/server.native.js.map +1 -1
- package/dist/cjs/vite-plugin.cjs +1 -1
- package/dist/cjs/vite-plugin.native.js +1 -1
- package/dist/cjs/vite-plugin.native.js.map +1 -1
- package/dist/esm/cli.mjs +0 -0
- package/dist/esm/cli.native.js +0 -0
- package/dist/esm/createPermissions.mjs.map +1 -1
- package/dist/esm/createPermissions.native.js.map +1 -1
- package/dist/esm/createSchemaFromDrizzle.mjs +5 -0
- package/dist/esm/createSchemaFromDrizzle.mjs.map +1 -0
- package/dist/esm/createSchemaFromDrizzle.native.js +5 -0
- package/dist/esm/createSchemaFromDrizzle.native.js.map +1 -0
- package/dist/esm/createUseQuery.mjs +2 -2
- package/dist/esm/createUseQuery.mjs.map +1 -1
- package/dist/esm/createUseQuery.native.js +2 -2
- package/dist/esm/createUseQuery.native.js.map +1 -1
- package/dist/esm/createZeroClient.mjs +2 -2
- package/dist/esm/createZeroClient.mjs.map +1 -1
- package/dist/esm/createZeroClient.native.js +2 -2
- package/dist/esm/createZeroClient.native.js.map +1 -1
- package/dist/esm/createZeroServer.mjs.map +1 -1
- package/dist/esm/createZeroServer.native.js.map +1 -1
- package/dist/esm/generate.mjs +64 -13
- package/dist/esm/generate.mjs.map +1 -1
- package/dist/esm/generate.native.js +114 -13
- package/dist/esm/generate.native.js.map +1 -1
- package/dist/esm/helpers/createMutators.mjs.map +1 -1
- package/dist/esm/helpers/createMutators.native.js.map +1 -1
- package/dist/esm/index.js +19 -24
- package/dist/esm/index.js.map +1 -6
- package/dist/esm/index.mjs +1 -0
- package/dist/esm/index.mjs.map +1 -1
- package/dist/esm/index.native.js +1 -0
- package/dist/esm/index.native.js.map +1 -1
- package/dist/esm/mutations.mjs +1 -1
- package/dist/esm/mutations.mjs.map +1 -1
- package/dist/esm/mutations.native.js +1 -1
- package/dist/esm/mutations.native.js.map +1 -1
- package/dist/esm/server.native.js +1 -1
- package/dist/esm/server.native.js.map +1 -1
- package/dist/esm/vite-plugin.mjs +1 -1
- package/dist/esm/vite-plugin.mjs.map +1 -1
- package/dist/esm/vite-plugin.native.js +1 -1
- package/dist/esm/vite-plugin.native.js.map +1 -1
- package/package.json +10 -4
- package/readme.md +80 -54
- package/src/createPermissions.ts +4 -9
- package/src/createSchemaFromDrizzle.ts +3 -0
- package/src/createUseQuery.tsx +4 -4
- package/src/createZeroClient.tsx +14 -14
- package/src/createZeroServer.ts +12 -12
- package/src/generate.test.ts +32 -32
- package/src/generate.ts +208 -55
- package/src/helpers/batchQuery.ts +1 -1
- package/src/helpers/createMutators.ts +7 -6
- package/src/helpers/didRunPermissionCheck.ts +1 -1
- package/src/helpers/mutatorContext.ts +2 -2
- package/src/helpers/prettyFormatZeroQuery.ts +1 -1
- package/src/helpers/queryContext.ts +1 -1
- package/src/helpers/useZeroDebug.ts +2 -2
- package/src/index.ts +2 -0
- package/src/mutations.ts +33 -7
- package/src/queryRegistry.ts +1 -1
- package/src/resolveQuery.ts +3 -3
- package/src/run.ts +5 -5
- package/src/server.native.ts +3 -0
- package/src/serverWhere.ts +2 -2
- package/src/types.ts +4 -2
- package/src/vite-plugin.ts +2 -2
- package/src/where.ts +5 -5
- package/src/zeroRunner.ts +2 -2
- package/types/createPermissions.d.ts +1 -1
- package/types/createPermissions.d.ts.map +1 -1
- package/types/createSchemaFromDrizzle.d.ts +4 -0
- package/types/createSchemaFromDrizzle.d.ts.map +1 -0
- package/types/createUseQuery.d.ts.map +1 -1
- package/types/createZeroServer.d.ts +11 -11
- package/types/generate.d.ts +29 -0
- package/types/generate.d.ts.map +1 -1
- package/types/helpers/createMutators.d.ts.map +1 -1
- package/types/index.d.ts +1 -0
- package/types/index.d.ts.map +1 -1
- package/types/mutations.d.ts +4 -1
- package/types/mutations.d.ts.map +1 -1
- package/types/resolveQuery.d.ts +1 -1
- package/types/resolveQuery.d.ts.map +1 -1
- package/types/server.native.d.ts +1 -0
- package/types/server.native.d.ts.map +1 -0
- package/types/types.d.ts.map +1 -1
- package/dist/cjs/cli.js +0 -42
- package/dist/cjs/cli.js.map +0 -6
- package/dist/cjs/constants.js +0 -23
- package/dist/cjs/constants.js.map +0 -6
- package/dist/cjs/createPermissions.js +0 -92
- package/dist/cjs/createPermissions.js.map +0 -6
- package/dist/cjs/createUseQuery.js +0 -35
- package/dist/cjs/createUseQuery.js.map +0 -6
- package/dist/cjs/createZeroClient.js +0 -178
- package/dist/cjs/createZeroClient.js.map +0 -6
- package/dist/cjs/createZeroServer.js +0 -189
- package/dist/cjs/createZeroServer.js.map +0 -6
- package/dist/cjs/generate.js +0 -811
- package/dist/cjs/generate.js.map +0 -6
- package/dist/cjs/generate.test.js +0 -404
- package/dist/cjs/generate.test.js.map +0 -6
- package/dist/cjs/helpers/batchQuery.js +0 -41
- package/dist/cjs/helpers/batchQuery.js.map +0 -6
- package/dist/cjs/helpers/createMutators.js +0 -118
- package/dist/cjs/helpers/createMutators.js.map +0 -6
- package/dist/cjs/helpers/didRunPermissionCheck.js +0 -26
- package/dist/cjs/helpers/didRunPermissionCheck.js.map +0 -6
- package/dist/cjs/helpers/ensureLoggedIn.js +0 -25
- package/dist/cjs/helpers/ensureLoggedIn.js.map +0 -6
- package/dist/cjs/helpers/getAuth.js +0 -31
- package/dist/cjs/helpers/getAuth.js.map +0 -6
- package/dist/cjs/helpers/mutatorContext.js +0 -44
- package/dist/cjs/helpers/mutatorContext.js.map +0 -6
- package/dist/cjs/helpers/prettyFormatZeroQuery.js +0 -92
- package/dist/cjs/helpers/prettyFormatZeroQuery.js.map +0 -6
- package/dist/cjs/helpers/queryContext.js +0 -33
- package/dist/cjs/helpers/queryContext.js.map +0 -6
- package/dist/cjs/helpers/useZeroDebug.js +0 -49
- package/dist/cjs/helpers/useZeroDebug.js.map +0 -6
- package/dist/cjs/index.js +0 -40
- package/dist/cjs/index.js.map +0 -6
- package/dist/cjs/modelRegistry.js +0 -32
- package/dist/cjs/modelRegistry.js.map +0 -6
- package/dist/cjs/mutations.js +0 -70
- package/dist/cjs/mutations.js.map +0 -6
- package/dist/cjs/queryRegistry.js +0 -32
- package/dist/cjs/queryRegistry.js.map +0 -6
- package/dist/cjs/resolveQuery.js +0 -40
- package/dist/cjs/resolveQuery.js.map +0 -6
- package/dist/cjs/run.js +0 -42
- package/dist/cjs/run.js.map +0 -6
- package/dist/cjs/server.js +0 -15
- package/dist/cjs/server.js.map +0 -6
- package/dist/cjs/serverWhere.js +0 -24
- package/dist/cjs/serverWhere.js.map +0 -6
- package/dist/cjs/serverWhere.test.js +0 -52
- package/dist/cjs/serverWhere.test.js.map +0 -6
- package/dist/cjs/state.js +0 -48
- package/dist/cjs/state.js.map +0 -6
- package/dist/cjs/types.js +0 -14
- package/dist/cjs/types.js.map +0 -6
- package/dist/cjs/usePermission.test.js +0 -29
- package/dist/cjs/usePermission.test.js.map +0 -6
- package/dist/cjs/vite-plugin.js +0 -89
- package/dist/cjs/vite-plugin.js.map +0 -6
- package/dist/cjs/where.js +0 -50
- package/dist/cjs/where.js.map +0 -6
- package/dist/cjs/zeroRunner.js +0 -35
- package/dist/cjs/zeroRunner.js.map +0 -6
- package/dist/cjs/zql.js +0 -26
- package/dist/cjs/zql.js.map +0 -6
- package/dist/esm/cli.js +0 -44
- package/dist/esm/cli.js.map +0 -6
- package/dist/esm/constants.js +0 -7
- package/dist/esm/constants.js.map +0 -6
- package/dist/esm/createPermissions.js +0 -81
- package/dist/esm/createPermissions.js.map +0 -6
- package/dist/esm/createUseQuery.js +0 -22
- package/dist/esm/createUseQuery.js.map +0 -6
- package/dist/esm/createZeroClient.js +0 -185
- package/dist/esm/createZeroClient.js.map +0 -6
- package/dist/esm/createZeroServer.js +0 -192
- package/dist/esm/createZeroServer.js.map +0 -6
- package/dist/esm/generate.js +0 -789
- package/dist/esm/generate.js.map +0 -6
- package/dist/esm/generate.test.js +0 -408
- package/dist/esm/generate.test.js.map +0 -6
- package/dist/esm/helpers/batchQuery.js +0 -25
- package/dist/esm/helpers/batchQuery.js.map +0 -6
- package/dist/esm/helpers/createMutators.js +0 -96
- package/dist/esm/helpers/createMutators.js.map +0 -6
- package/dist/esm/helpers/didRunPermissionCheck.js +0 -10
- package/dist/esm/helpers/didRunPermissionCheck.js.map +0 -6
- package/dist/esm/helpers/ensureLoggedIn.js +0 -10
- package/dist/esm/helpers/ensureLoggedIn.js.map +0 -6
- package/dist/esm/helpers/getAuth.js +0 -17
- package/dist/esm/helpers/getAuth.js.map +0 -6
- package/dist/esm/helpers/mutatorContext.js +0 -28
- package/dist/esm/helpers/mutatorContext.js.map +0 -6
- package/dist/esm/helpers/prettyFormatZeroQuery.js +0 -76
- package/dist/esm/helpers/prettyFormatZeroQuery.js.map +0 -6
- package/dist/esm/helpers/queryContext.js +0 -17
- package/dist/esm/helpers/queryContext.js.map +0 -6
- package/dist/esm/helpers/useZeroDebug.js +0 -35
- package/dist/esm/helpers/useZeroDebug.js.map +0 -6
- package/dist/esm/modelRegistry.js +0 -16
- package/dist/esm/modelRegistry.js.map +0 -6
- package/dist/esm/mutations.js +0 -56
- package/dist/esm/mutations.js.map +0 -6
- package/dist/esm/queryRegistry.js +0 -16
- package/dist/esm/queryRegistry.js.map +0 -6
- package/dist/esm/resolveQuery.js +0 -24
- package/dist/esm/resolveQuery.js.map +0 -6
- package/dist/esm/run.js +0 -27
- package/dist/esm/run.js.map +0 -6
- package/dist/esm/server.js +0 -2
- package/dist/esm/server.js.map +0 -6
- package/dist/esm/serverWhere.js +0 -8
- package/dist/esm/serverWhere.js.map +0 -6
- package/dist/esm/serverWhere.test.js +0 -55
- package/dist/esm/serverWhere.test.js.map +0 -6
- package/dist/esm/state.js +0 -33
- package/dist/esm/state.js.map +0 -6
- package/dist/esm/types.js +0 -1
- package/dist/esm/types.js.map +0 -6
- package/dist/esm/usePermission.test.js +0 -33
- package/dist/esm/usePermission.test.js.map +0 -6
- package/dist/esm/vite-plugin.js +0 -74
- package/dist/esm/vite-plugin.js.map +0 -6
- package/dist/esm/where.js +0 -36
- package/dist/esm/where.js.map +0 -6
- package/dist/esm/zeroRunner.js +0 -19
- package/dist/esm/zeroRunner.js.map +0 -6
- package/dist/esm/zql.js +0 -10
- package/dist/esm/zql.js.map +0 -6
package/dist/esm/generate.js
DELETED
|
@@ -1,789 +0,0 @@
|
|
|
1
|
-
import { createHash } from "node:crypto";
|
|
2
|
-
import { existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from "node:fs";
|
|
3
|
-
import { basename, dirname, resolve } from "node:path";
|
|
4
|
-
const hash = (s) => createHash("sha256").update(s).digest("hex");
|
|
5
|
-
let generateCache = {}, generateCachePath = "";
|
|
6
|
-
function getCacheDir() {
|
|
7
|
-
let dir = process.cwd();
|
|
8
|
-
for (; dir !== "/"; ) {
|
|
9
|
-
const nm = resolve(dir, "node_modules");
|
|
10
|
-
if (existsSync(nm)) {
|
|
11
|
-
const cacheDir = resolve(nm, ".on-zero");
|
|
12
|
-
return existsSync(cacheDir) || mkdirSync(cacheDir, { recursive: !0 }), cacheDir;
|
|
13
|
-
}
|
|
14
|
-
dir = resolve(dir, "..");
|
|
15
|
-
}
|
|
16
|
-
return null;
|
|
17
|
-
}
|
|
18
|
-
function loadCache() {
|
|
19
|
-
const cacheDir = getCacheDir();
|
|
20
|
-
if (cacheDir) {
|
|
21
|
-
generateCachePath = resolve(cacheDir, "generate-cache.json");
|
|
22
|
-
try {
|
|
23
|
-
generateCache = JSON.parse(readFileSync(generateCachePath, "utf-8"));
|
|
24
|
-
} catch {
|
|
25
|
-
generateCache = {};
|
|
26
|
-
}
|
|
27
|
-
}
|
|
28
|
-
}
|
|
29
|
-
function saveCache() {
|
|
30
|
-
generateCachePath && writeFileSync(generateCachePath, JSON.stringify(generateCache) + `
|
|
31
|
-
`, "utf-8");
|
|
32
|
-
}
|
|
33
|
-
function writeFileIfChanged(filePath, content) {
|
|
34
|
-
const contentHash = hash(content);
|
|
35
|
-
return generateCache[filePath] === contentHash && existsSync(filePath) ? !1 : (writeFileSync(filePath, content, "utf-8"), generateCache[filePath] = contentHash, !0);
|
|
36
|
-
}
|
|
37
|
-
function generateModelsFile(modelFiles) {
|
|
38
|
-
const modelNames = modelFiles.map((f) => basename(f, ".ts")).sort(), getImportName = (name) => name === "user" ? "userPublic" : name, imports = modelNames.map((name) => `import * as ${getImportName(name)} from '../models/${name}'`).join(`
|
|
39
|
-
`), modelsObj = `export const models = {
|
|
40
|
-
${[...modelNames].sort(
|
|
41
|
-
(a, b) => getImportName(a).localeCompare(getImportName(b))
|
|
42
|
-
).map((name) => ` ${getImportName(name)},`).join(`
|
|
43
|
-
`)}
|
|
44
|
-
}`;
|
|
45
|
-
return `// auto-generated by: on-zero generate
|
|
46
|
-
${imports}
|
|
47
|
-
|
|
48
|
-
${modelsObj}
|
|
49
|
-
`;
|
|
50
|
-
}
|
|
51
|
-
function generateTypesFile(modelFiles) {
|
|
52
|
-
const modelNames = modelFiles.map((f) => basename(f, ".ts")).sort(), getSchemaName = (name) => name === "user" ? "userPublic" : name;
|
|
53
|
-
return `import type { TableInsertRow, TableUpdateRow } from 'on-zero'
|
|
54
|
-
import type * as schema from './tables'
|
|
55
|
-
|
|
56
|
-
${modelNames.map((name) => {
|
|
57
|
-
const pascalName = name.charAt(0).toUpperCase() + name.slice(1), schemaName = getSchemaName(name);
|
|
58
|
-
return `export type ${pascalName} = TableInsertRow<typeof schema.${schemaName}>
|
|
59
|
-
export type ${pascalName}Update = TableUpdateRow<typeof schema.${schemaName}>`;
|
|
60
|
-
}).join(`
|
|
61
|
-
|
|
62
|
-
`)}
|
|
63
|
-
`;
|
|
64
|
-
}
|
|
65
|
-
function generateTablesFile(modelFiles) {
|
|
66
|
-
const modelNames = modelFiles.map((f) => basename(f, ".ts")).sort(), getExportName = (name) => name === "user" ? "userPublic" : name;
|
|
67
|
-
return `// auto-generated by: on-zero generate
|
|
68
|
-
// this is separate from models as otherwise you end up with circular types :/
|
|
69
|
-
|
|
70
|
-
${modelNames.map((name) => `export { schema as ${getExportName(name)} } from '../models/${name}'`).join(`
|
|
71
|
-
`)}
|
|
72
|
-
`;
|
|
73
|
-
}
|
|
74
|
-
function generateReadmeFile() {
|
|
75
|
-
return `# generated
|
|
76
|
-
|
|
77
|
-
this folder is auto-generated by on-zero. do not edit files here directly.
|
|
78
|
-
|
|
79
|
-
## what's generated
|
|
80
|
-
|
|
81
|
-
- \`models.ts\` - exports all models from ../models
|
|
82
|
-
- \`types.ts\` - typescript types derived from table schemas
|
|
83
|
-
- \`tables.ts\` - exports table schemas for type inference
|
|
84
|
-
- \`groupedQueries.ts\` - namespaced query re-exports for client setup
|
|
85
|
-
- \`syncedQueries.ts\` - namespaced syncedQuery wrappers for server setup
|
|
86
|
-
- \`syncedMutations.ts\` - valibot validators for mutation args (server auto-validation)
|
|
87
|
-
|
|
88
|
-
## usage guidelines
|
|
89
|
-
|
|
90
|
-
**do not import generated files outside of the data folder.**
|
|
91
|
-
|
|
92
|
-
### queries
|
|
93
|
-
|
|
94
|
-
write your queries as plain functions in \`../queries/\` and import them directly:
|
|
95
|
-
|
|
96
|
-
\`\`\`ts
|
|
97
|
-
// \u2705 good - import from queries
|
|
98
|
-
import { channelMessages } from '~/data/queries/message'
|
|
99
|
-
\`\`\`
|
|
100
|
-
|
|
101
|
-
the generated query files are only used internally by zero client/server setup.
|
|
102
|
-
|
|
103
|
-
### types
|
|
104
|
-
|
|
105
|
-
you can import types from this folder, but prefer re-exporting from \`../types.ts\`:
|
|
106
|
-
|
|
107
|
-
\`\`\`ts
|
|
108
|
-
// \u274C okay but not preferred
|
|
109
|
-
import type { Message } from '~/data/generated/types'
|
|
110
|
-
|
|
111
|
-
// \u2705 better - re-export from types.ts
|
|
112
|
-
import type { Message } from '~/data/types'
|
|
113
|
-
\`\`\`
|
|
114
|
-
|
|
115
|
-
## regeneration
|
|
116
|
-
|
|
117
|
-
files are regenerated when you run:
|
|
118
|
-
|
|
119
|
-
\`\`\`bash
|
|
120
|
-
bun on-zero generate
|
|
121
|
-
\`\`\`
|
|
122
|
-
|
|
123
|
-
or in watch mode:
|
|
124
|
-
|
|
125
|
-
\`\`\`bash
|
|
126
|
-
bun on-zero generate --watch
|
|
127
|
-
\`\`\`
|
|
128
|
-
|
|
129
|
-
## more info
|
|
130
|
-
|
|
131
|
-
see the [on-zero readme](./node_modules/on-zero/README.md) for full documentation.
|
|
132
|
-
`;
|
|
133
|
-
}
|
|
134
|
-
function generateGroupedQueriesFile(queries) {
|
|
135
|
-
return `/**
|
|
136
|
-
* auto-generated by: on-zero generate
|
|
137
|
-
*
|
|
138
|
-
* grouped query re-exports for minification-safe query identity.
|
|
139
|
-
* this file re-exports all query modules - while this breaks tree-shaking,
|
|
140
|
-
* queries are typically small and few in number even in larger apps.
|
|
141
|
-
*/
|
|
142
|
-
${[...new Set(queries.map((q) => q.sourceFile))].sort().map((file) => `export * as ${file} from '../queries/${file}'`).join(`
|
|
143
|
-
`)}
|
|
144
|
-
`;
|
|
145
|
-
}
|
|
146
|
-
function generateSyncedQueriesFile(queries) {
|
|
147
|
-
const queryByFile = /* @__PURE__ */ new Map();
|
|
148
|
-
for (const q of queries)
|
|
149
|
-
queryByFile.has(q.sourceFile) || queryByFile.set(q.sourceFile, []), queryByFile.get(q.sourceFile).push(q);
|
|
150
|
-
const sortedFiles = Array.from(queryByFile.keys()).sort(), imports = `// auto-generated by: on-zero generate
|
|
151
|
-
// server-side query definitions with validators
|
|
152
|
-
import { defineQuery, defineQueries } from '@rocicorp/zero'
|
|
153
|
-
import * as v from 'valibot'
|
|
154
|
-
import * as Queries from './groupedQueries'
|
|
155
|
-
`, namespaceDefs = sortedFiles.map((file) => {
|
|
156
|
-
const queryDefs = queryByFile.get(file).sort((a, b) => a.name.localeCompare(b.name)).map((q) => {
|
|
157
|
-
const validatorDef = q.valibotCode.trim();
|
|
158
|
-
if (q.params === "void" || !validatorDef)
|
|
159
|
-
return ` ${q.name}: defineQuery(() => Queries.${file}.${q.name}()),`;
|
|
160
|
-
const indentedValidator = validatorDef.split(`
|
|
161
|
-
`).map((line, i) => i === 0 ? line : ` ${line}`).join(`
|
|
162
|
-
`);
|
|
163
|
-
return ` ${q.name}: defineQuery(
|
|
164
|
-
${indentedValidator},
|
|
165
|
-
({ args }) => Queries.${file}.${q.name}(args)
|
|
166
|
-
),`;
|
|
167
|
-
}).join(`
|
|
168
|
-
`);
|
|
169
|
-
return `const ${file} = {
|
|
170
|
-
${queryDefs}
|
|
171
|
-
}`;
|
|
172
|
-
}).join(`
|
|
173
|
-
|
|
174
|
-
`), queriesObject = sortedFiles.map((file) => ` ${file},`).join(`
|
|
175
|
-
`);
|
|
176
|
-
return `${imports}
|
|
177
|
-
${namespaceDefs}
|
|
178
|
-
|
|
179
|
-
export const queries = defineQueries({
|
|
180
|
-
${queriesObject}
|
|
181
|
-
})
|
|
182
|
-
`;
|
|
183
|
-
}
|
|
184
|
-
function createTypeResolver(ts, files, dir) {
|
|
185
|
-
const configPath = ts.findConfigFile(dir, ts.sys.fileExists, "tsconfig.json");
|
|
186
|
-
let compilerOptions = {
|
|
187
|
-
target: ts.ScriptTarget.Latest,
|
|
188
|
-
module: ts.ModuleKind.ESNext,
|
|
189
|
-
moduleResolution: ts.ModuleResolutionKind.Bundler,
|
|
190
|
-
strict: !1,
|
|
191
|
-
skipLibCheck: !0,
|
|
192
|
-
noEmit: !0
|
|
193
|
-
};
|
|
194
|
-
if (configPath) {
|
|
195
|
-
const configFile = ts.readConfigFile(configPath, ts.sys.readFile);
|
|
196
|
-
if (configFile.config) {
|
|
197
|
-
const parsed = ts.parseJsonConfigFileContent(
|
|
198
|
-
configFile.config,
|
|
199
|
-
ts.sys,
|
|
200
|
-
dirname(configPath)
|
|
201
|
-
);
|
|
202
|
-
compilerOptions = { ...compilerOptions, ...parsed.options };
|
|
203
|
-
}
|
|
204
|
-
}
|
|
205
|
-
const fileMap = /* @__PURE__ */ new Map();
|
|
206
|
-
for (const f of files)
|
|
207
|
-
fileMap.set(f.path, f.content);
|
|
208
|
-
const host = ts.createCompilerHost(compilerOptions), originalGetSourceFile = host.getSourceFile.bind(host);
|
|
209
|
-
host.getSourceFile = (fileName, languageVersion, onError) => {
|
|
210
|
-
const content = fileMap.get(fileName);
|
|
211
|
-
return content !== void 0 ? ts.createSourceFile(fileName, content, languageVersion, !0) : originalGetSourceFile(fileName, languageVersion, onError);
|
|
212
|
-
}, host.fileExists = (fileName) => fileMap.has(fileName) || ts.sys.fileExists(fileName), host.readFile = (fileName) => fileMap.get(fileName) ?? ts.sys.readFile(fileName);
|
|
213
|
-
const program = ts.createProgram(
|
|
214
|
-
files.map((f) => f.path),
|
|
215
|
-
compilerOptions,
|
|
216
|
-
host
|
|
217
|
-
), checker = program.getTypeChecker();
|
|
218
|
-
return {
|
|
219
|
-
program,
|
|
220
|
-
checker,
|
|
221
|
-
// resolve a type annotation node to a ts.Type
|
|
222
|
-
resolveType(node) {
|
|
223
|
-
try {
|
|
224
|
-
return checker.getTypeFromTypeNode(node);
|
|
225
|
-
} catch {
|
|
226
|
-
return null;
|
|
227
|
-
}
|
|
228
|
-
},
|
|
229
|
-
// convert a resolved type to valibot code
|
|
230
|
-
typeToValibot(type) {
|
|
231
|
-
return tsTypeToValibot(ts, checker, type);
|
|
232
|
-
}
|
|
233
|
-
};
|
|
234
|
-
}
|
|
235
|
-
function resolveParamType(ts, resolver, sourceFile, exportName, paramIndex) {
|
|
236
|
-
let result = null;
|
|
237
|
-
return ts.forEachChild(sourceFile, (node) => {
|
|
238
|
-
if (result || !ts.isVariableStatement(node) || !node.modifiers?.some((m) => m.kind === ts.SyntaxKind.ExportKeyword)) return;
|
|
239
|
-
const decl = node.declarationList.declarations[0];
|
|
240
|
-
if (!(!decl || !ts.isVariableDeclaration(decl)) && decl.name.getText(sourceFile) === exportName && decl.initializer && ts.isArrowFunction(decl.initializer)) {
|
|
241
|
-
const param = decl.initializer.parameters[paramIndex];
|
|
242
|
-
param?.type && (result = resolver.resolveType(param.type));
|
|
243
|
-
}
|
|
244
|
-
}), result;
|
|
245
|
-
}
|
|
246
|
-
function resolveMutationParamTypes(ts, resolver, sourceFile) {
|
|
247
|
-
const resolved = /* @__PURE__ */ new Map();
|
|
248
|
-
return ts.forEachChild(sourceFile, (node) => {
|
|
249
|
-
if (!ts.isVariableStatement(node) || !node.modifiers?.some((m) => m.kind === ts.SyntaxKind.ExportKeyword)) return;
|
|
250
|
-
const decl = node.declarationList.declarations[0];
|
|
251
|
-
if (!decl || !ts.isVariableDeclaration(decl) || decl.name.getText(sourceFile) !== "mutate" || !decl.initializer || !ts.isCallExpression(decl.initializer)) return;
|
|
252
|
-
const args = decl.initializer.arguments;
|
|
253
|
-
let handlersArg = null;
|
|
254
|
-
for (let i = args.length - 1; i >= 0; i--)
|
|
255
|
-
if (ts.isObjectLiteralExpression(args[i])) {
|
|
256
|
-
handlersArg = args[i];
|
|
257
|
-
break;
|
|
258
|
-
}
|
|
259
|
-
if (handlersArg)
|
|
260
|
-
for (const prop of handlersArg.properties) {
|
|
261
|
-
if (!ts.isPropertyAssignment(prop) && !ts.isMethodDeclaration(prop)) continue;
|
|
262
|
-
const name = prop.name?.getText(sourceFile);
|
|
263
|
-
if (!name) continue;
|
|
264
|
-
let params = null;
|
|
265
|
-
if (ts.isPropertyAssignment(prop)) {
|
|
266
|
-
const init = prop.initializer;
|
|
267
|
-
(ts.isArrowFunction(init) || ts.isFunctionExpression(init)) && (params = init.parameters);
|
|
268
|
-
} else ts.isMethodDeclaration(prop) && (params = prop.parameters);
|
|
269
|
-
if (!params || params.length < 2) continue;
|
|
270
|
-
const typeNode = params[1].type;
|
|
271
|
-
if (!typeNode) continue;
|
|
272
|
-
const expanded = resolver.resolveType(typeNode);
|
|
273
|
-
expanded && resolved.set(name, expanded);
|
|
274
|
-
}
|
|
275
|
-
}), resolved;
|
|
276
|
-
}
|
|
277
|
-
function extractMutationsFromModel(ts, sourceFile, content, fileName, silent, typeToValibot, resolvedTypes, resolvedTypeToValibot) {
|
|
278
|
-
let mutateNode = null;
|
|
279
|
-
if (ts.forEachChild(sourceFile, (node) => {
|
|
280
|
-
if (!ts.isVariableStatement(node) || !node.modifiers?.some((m) => m.kind === ts.SyntaxKind.ExportKeyword)) return;
|
|
281
|
-
const decl = node.declarationList.declarations[0];
|
|
282
|
-
!decl || !ts.isVariableDeclaration(decl) || decl.name.getText(sourceFile) === "mutate" && decl.initializer && ts.isCallExpression(decl.initializer) && (mutateNode = decl.initializer);
|
|
283
|
-
}), !mutateNode)
|
|
284
|
-
return {
|
|
285
|
-
modelName: basename(fileName, ".ts"),
|
|
286
|
-
hasCRUD: !1,
|
|
287
|
-
columns: {},
|
|
288
|
-
primaryKeys: [],
|
|
289
|
-
custom: []
|
|
290
|
-
};
|
|
291
|
-
const args = mutateNode.arguments, hasCRUD = args.length >= 2;
|
|
292
|
-
let handlersArg = null;
|
|
293
|
-
args.length === 1 && ts.isObjectLiteralExpression(args[0]) ? handlersArg = args[0] : args.length === 3 && ts.isObjectLiteralExpression(args[2]) && (handlersArg = args[2]);
|
|
294
|
-
const columns = {}, primaryKeys = [];
|
|
295
|
-
hasCRUD && extractSchemaColumns(ts, sourceFile, columns, primaryKeys);
|
|
296
|
-
const custom = [];
|
|
297
|
-
if (handlersArg)
|
|
298
|
-
for (const prop of handlersArg.properties) {
|
|
299
|
-
if (!ts.isPropertyAssignment(prop) && !ts.isMethodDeclaration(prop)) continue;
|
|
300
|
-
const name = prop.name?.getText(sourceFile);
|
|
301
|
-
if (!name) continue;
|
|
302
|
-
let params = null;
|
|
303
|
-
if (ts.isPropertyAssignment(prop)) {
|
|
304
|
-
const init = prop.initializer;
|
|
305
|
-
(ts.isArrowFunction(init) || ts.isFunctionExpression(init)) && (params = init.parameters);
|
|
306
|
-
} else ts.isMethodDeclaration(prop) && (params = prop.parameters);
|
|
307
|
-
if (!params) continue;
|
|
308
|
-
if (params.length < 2) {
|
|
309
|
-
custom.push({ name, paramType: "void", valibotCode: "" });
|
|
310
|
-
continue;
|
|
311
|
-
}
|
|
312
|
-
const paramType = params[1].type?.getText(sourceFile) || "unknown";
|
|
313
|
-
if (paramType === "unknown") {
|
|
314
|
-
custom.push({ name, paramType: "unknown", valibotCode: "" });
|
|
315
|
-
continue;
|
|
316
|
-
}
|
|
317
|
-
let valibotCode = typeToValibot(paramType);
|
|
318
|
-
if (!valibotCode && resolvedTypes && resolvedTypeToValibot) {
|
|
319
|
-
const resolvedType = resolvedTypes.get(name);
|
|
320
|
-
resolvedType && (valibotCode = resolvedTypeToValibot(resolvedType));
|
|
321
|
-
}
|
|
322
|
-
custom.push({
|
|
323
|
-
name,
|
|
324
|
-
paramType,
|
|
325
|
-
valibotCode: valibotCode || ""
|
|
326
|
-
});
|
|
327
|
-
}
|
|
328
|
-
return {
|
|
329
|
-
modelName: "",
|
|
330
|
-
hasCRUD,
|
|
331
|
-
columns,
|
|
332
|
-
primaryKeys,
|
|
333
|
-
custom
|
|
334
|
-
};
|
|
335
|
-
}
|
|
336
|
-
function extractSchemaColumns(ts, sourceFile, columns, primaryKeys) {
|
|
337
|
-
function visit(node) {
|
|
338
|
-
if (ts.isCallExpression(node)) {
|
|
339
|
-
const text = node.expression.getText(sourceFile);
|
|
340
|
-
if (text.endsWith(".primaryKey"))
|
|
341
|
-
for (const arg of node.arguments)
|
|
342
|
-
ts.isStringLiteral(arg) && primaryKeys.push(arg.text);
|
|
343
|
-
if (text.endsWith(".columns") && node.arguments.length === 1) {
|
|
344
|
-
const obj = node.arguments[0];
|
|
345
|
-
if (ts.isObjectLiteralExpression(obj))
|
|
346
|
-
for (const prop of obj.properties) {
|
|
347
|
-
if (!ts.isPropertyAssignment(prop)) continue;
|
|
348
|
-
const colName = prop.name?.getText(sourceFile);
|
|
349
|
-
if (!colName) continue;
|
|
350
|
-
const initText = prop.initializer.getText(sourceFile), colType = parseColumnType(initText);
|
|
351
|
-
columns[colName] = colType;
|
|
352
|
-
}
|
|
353
|
-
}
|
|
354
|
-
}
|
|
355
|
-
ts.forEachChild(node, visit);
|
|
356
|
-
}
|
|
357
|
-
visit(sourceFile);
|
|
358
|
-
}
|
|
359
|
-
function parseColumnType(initText) {
|
|
360
|
-
const optional = initText.includes(".optional()");
|
|
361
|
-
let type = "string";
|
|
362
|
-
return initText.startsWith("number(") ? type = "number" : initText.startsWith("boolean(") ? type = "boolean" : initText.startsWith("json(") || initText.startsWith("json<") ? type = "json" : initText.startsWith("enumeration(") && (type = "enum"), { type, optional };
|
|
363
|
-
}
|
|
364
|
-
function columnTypeToValibot(col) {
|
|
365
|
-
let base;
|
|
366
|
-
switch (col.type) {
|
|
367
|
-
case "string":
|
|
368
|
-
base = "v.string()";
|
|
369
|
-
break;
|
|
370
|
-
case "number":
|
|
371
|
-
base = "v.number()";
|
|
372
|
-
break;
|
|
373
|
-
case "boolean":
|
|
374
|
-
base = "v.boolean()";
|
|
375
|
-
break;
|
|
376
|
-
case "json":
|
|
377
|
-
base = "v.unknown()";
|
|
378
|
-
break;
|
|
379
|
-
case "enum":
|
|
380
|
-
base = "v.string()";
|
|
381
|
-
break;
|
|
382
|
-
default:
|
|
383
|
-
base = "v.unknown()";
|
|
384
|
-
}
|
|
385
|
-
return col.optional ? `v.optional(v.nullable(${base}))` : base;
|
|
386
|
-
}
|
|
387
|
-
function shouldSkipObjectKey(name) {
|
|
388
|
-
return name.startsWith("__@");
|
|
389
|
-
}
|
|
390
|
-
function formatObjectKey(name) {
|
|
391
|
-
return /^[$A-Z_a-z][$\w]*$/.test(name) ? name : JSON.stringify(name);
|
|
392
|
-
}
|
|
393
|
-
function schemaColumnsToValibot(columns, primaryKeys, mode) {
|
|
394
|
-
const entries = [];
|
|
395
|
-
if (mode === "delete")
|
|
396
|
-
for (const pk of primaryKeys) {
|
|
397
|
-
const col = columns[pk];
|
|
398
|
-
col && entries.push(
|
|
399
|
-
`${formatObjectKey(pk)}: ${columnTypeToValibot({ ...col, optional: !1 })}`
|
|
400
|
-
);
|
|
401
|
-
}
|
|
402
|
-
else if (mode === "update")
|
|
403
|
-
for (const [name, col] of Object.entries(columns))
|
|
404
|
-
primaryKeys.includes(name) ? entries.push(
|
|
405
|
-
`${formatObjectKey(name)}: ${columnTypeToValibot({ ...col, optional: !1 })}`
|
|
406
|
-
) : entries.push(
|
|
407
|
-
`${formatObjectKey(name)}: ${columnTypeToValibot({ ...col, optional: !0 })}`
|
|
408
|
-
);
|
|
409
|
-
else
|
|
410
|
-
for (const [name, col] of Object.entries(columns))
|
|
411
|
-
entries.push(`${formatObjectKey(name)}: ${columnTypeToValibot(col)}`);
|
|
412
|
-
return `v.object({
|
|
413
|
-
${entries.join(`,
|
|
414
|
-
`)},
|
|
415
|
-
})`;
|
|
416
|
-
}
|
|
417
|
-
function generateSyncedMutationsFile(modelMutations) {
|
|
418
|
-
return `// auto-generated by: on-zero generate
|
|
419
|
-
// mutation validators derived from model schemas and handler types
|
|
420
|
-
import * as v from 'valibot'
|
|
421
|
-
|
|
422
|
-
export const mutationValidators = {
|
|
423
|
-
${[...modelMutations].sort(
|
|
424
|
-
(a, b) => a.modelName.localeCompare(b.modelName)
|
|
425
|
-
).map((model) => {
|
|
426
|
-
const entries = [];
|
|
427
|
-
if (model.hasCRUD && Object.keys(model.columns).length > 0)
|
|
428
|
-
for (const mode of ["insert", "update", "delete"])
|
|
429
|
-
if (model.custom.some((m) => m.name === mode)) {
|
|
430
|
-
const customMut = model.custom.find((m) => m.name === mode);
|
|
431
|
-
customMut.valibotCode ? entries.push(
|
|
432
|
-
` ${mode}: ${extractValibotExpression(customMut.valibotCode)},`
|
|
433
|
-
) : entries.push(
|
|
434
|
-
` ${mode}: ${schemaColumnsToValibot(model.columns, model.primaryKeys, mode)},`
|
|
435
|
-
);
|
|
436
|
-
} else
|
|
437
|
-
entries.push(
|
|
438
|
-
` ${mode}: ${schemaColumnsToValibot(model.columns, model.primaryKeys, mode)},`
|
|
439
|
-
);
|
|
440
|
-
for (const mut of model.custom)
|
|
441
|
-
if (!(model.hasCRUD && ["insert", "update", "delete", "upsert"].includes(mut.name))) {
|
|
442
|
-
if (mut.paramType === "void" || !mut.valibotCode) {
|
|
443
|
-
entries.push(` ${mut.name}: v.void_(),`);
|
|
444
|
-
continue;
|
|
445
|
-
}
|
|
446
|
-
entries.push(` ${mut.name}: ${extractValibotExpression(mut.valibotCode)},`);
|
|
447
|
-
}
|
|
448
|
-
return ` ${model.modelName}: {
|
|
449
|
-
${entries.join(`
|
|
450
|
-
`)}
|
|
451
|
-
},`;
|
|
452
|
-
}).join(`
|
|
453
|
-
`)}
|
|
454
|
-
}
|
|
455
|
-
`;
|
|
456
|
-
}
|
|
457
|
-
function extractValibotExpression(valibotCode) {
|
|
458
|
-
return valibotCode.trim() || "v.unknown()";
|
|
459
|
-
}
|
|
460
|
-
function parseTypeString(type) {
|
|
461
|
-
if (type = type.trim(), type === "string") return "v.string()";
|
|
462
|
-
if (type === "number") return "v.number()";
|
|
463
|
-
if (type === "boolean") return "v.boolean()";
|
|
464
|
-
if (type === "void" || type === "undefined") return "v.void_()";
|
|
465
|
-
if (type === "null") return "v.null_()";
|
|
466
|
-
if (type === "any" || type === "unknown") return "v.unknown()";
|
|
467
|
-
if (type.startsWith("{") && type.endsWith("}")) {
|
|
468
|
-
const inner = type.slice(1, -1).trim();
|
|
469
|
-
if (!inner) return "v.object({})";
|
|
470
|
-
const normalized = inner.replace(/\n/g, "; ").replace(/;\s*;/g, ";"), entries = [];
|
|
471
|
-
for (const part of normalized.split(";")) {
|
|
472
|
-
const trimmed = part.trim().replace(/,\s*$/, "");
|
|
473
|
-
if (!trimmed) continue;
|
|
474
|
-
const match = trimmed.match(/^(?:readonly\s+)?(\w+)(\?)?:\s*(.+)$/);
|
|
475
|
-
if (!match) continue;
|
|
476
|
-
const [, name, opt, typeStr] = match, parsed = parseTypeString(typeStr.trim());
|
|
477
|
-
if (!parsed) return null;
|
|
478
|
-
let val = parsed;
|
|
479
|
-
opt && (val = `v.optional(${val})`), entries.push(`${formatObjectKey(name)}: ${val}`);
|
|
480
|
-
}
|
|
481
|
-
return entries.length === 0 ? "v.object({})" : `v.object({
|
|
482
|
-
${entries.join(`,
|
|
483
|
-
`)},
|
|
484
|
-
})`;
|
|
485
|
-
}
|
|
486
|
-
if (type.endsWith("[]")) {
|
|
487
|
-
const inner = parseTypeString(type.slice(0, -2).trim());
|
|
488
|
-
return inner ? `v.array(${inner})` : null;
|
|
489
|
-
}
|
|
490
|
-
return null;
|
|
491
|
-
}
|
|
492
|
-
function tsTypeToValibot(ts, checker, type, seen) {
|
|
493
|
-
seen || (seen = /* @__PURE__ */ new Set());
|
|
494
|
-
const flags = type.getFlags();
|
|
495
|
-
if (flags & (ts.TypeFlags.Object | ts.TypeFlags.Intersection)) {
|
|
496
|
-
if (seen.has(type)) return "v.unknown()";
|
|
497
|
-
seen.add(type);
|
|
498
|
-
}
|
|
499
|
-
const recurse = (t) => tsTypeToValibot(ts, checker, t, seen);
|
|
500
|
-
if (flags & ts.TypeFlags.String) return "v.string()";
|
|
501
|
-
if (flags & ts.TypeFlags.Number) return "v.number()";
|
|
502
|
-
if (flags & ts.TypeFlags.Boolean) return "v.boolean()";
|
|
503
|
-
if (flags & ts.TypeFlags.Void || flags & ts.TypeFlags.Undefined) return "v.void_()";
|
|
504
|
-
if (flags & ts.TypeFlags.Null) return "v.null_()";
|
|
505
|
-
if (flags & ts.TypeFlags.Any || flags & ts.TypeFlags.Unknown) return "v.unknown()";
|
|
506
|
-
if (flags & ts.TypeFlags.Never) return "v.never()";
|
|
507
|
-
if (flags & ts.TypeFlags.StringLiteral)
|
|
508
|
-
return `v.literal(${JSON.stringify(type.value)})`;
|
|
509
|
-
if (flags & ts.TypeFlags.NumberLiteral)
|
|
510
|
-
return `v.literal(${type.value})`;
|
|
511
|
-
if (flags & ts.TypeFlags.BooleanLiteral)
|
|
512
|
-
return `v.literal(${type.intrinsicName === "true"})`;
|
|
513
|
-
if (type.isUnion()) {
|
|
514
|
-
const members = type.types, hasNull = members.some((t) => t.getFlags() & ts.TypeFlags.Null), hasUndefined = members.some(
|
|
515
|
-
(t) => t.getFlags() & (ts.TypeFlags.Undefined | ts.TypeFlags.Void)
|
|
516
|
-
), rest = members.filter(
|
|
517
|
-
(t) => !(t.getFlags() & (ts.TypeFlags.Null | ts.TypeFlags.Undefined | ts.TypeFlags.Void))
|
|
518
|
-
);
|
|
519
|
-
if (rest.length === 2 && rest.every((t) => t.getFlags() & ts.TypeFlags.BooleanLiteral)) {
|
|
520
|
-
let inner2 = "v.boolean()";
|
|
521
|
-
return hasNull && (inner2 = `v.nullable(${inner2})`), hasUndefined && (inner2 = `v.optional(${inner2})`), inner2;
|
|
522
|
-
}
|
|
523
|
-
if (rest.length === 0) return "v.unknown()";
|
|
524
|
-
let inner = rest.length === 1 ? recurse(rest[0]) : `v.union([${rest.map((t) => recurse(t)).join(", ")}])`;
|
|
525
|
-
return hasNull && (inner = `v.nullable(${inner})`), hasUndefined && (inner = `v.optional(${inner})`), inner;
|
|
526
|
-
}
|
|
527
|
-
const resolveSymbolType = (prop) => prop.valueDeclaration ? checker.getTypeOfSymbolAtLocation(prop, prop.valueDeclaration) : prop.declarations?.[0] ? checker.getTypeOfSymbolAtLocation(prop, prop.declarations[0]) : checker.getDeclaredTypeOfSymbol(prop);
|
|
528
|
-
if (type.isIntersection()) {
|
|
529
|
-
const props2 = type.getProperties();
|
|
530
|
-
if (props2.length === 0) return "v.object({})";
|
|
531
|
-
const entries = [];
|
|
532
|
-
for (const prop of props2) {
|
|
533
|
-
const name = prop.getName();
|
|
534
|
-
if (shouldSkipObjectKey(name)) continue;
|
|
535
|
-
const propType = resolveSymbolType(prop), isOptional = !!(prop.getFlags() & ts.SymbolFlags.Optional);
|
|
536
|
-
let val = recurse(propType);
|
|
537
|
-
isOptional && !val.startsWith("v.optional(") && (val = `v.optional(${val})`), entries.push(`${formatObjectKey(name)}: ${val}`);
|
|
538
|
-
}
|
|
539
|
-
return entries.length === 0 ? "v.object({})" : `v.object({
|
|
540
|
-
${entries.join(`,
|
|
541
|
-
`)},
|
|
542
|
-
})`;
|
|
543
|
-
}
|
|
544
|
-
const props = type.getProperties();
|
|
545
|
-
if (props.length > 0 && (type.getFlags() & ts.TypeFlags.Object || type.objectFlags)) {
|
|
546
|
-
const objectFlags = type.objectFlags ?? 0;
|
|
547
|
-
if (objectFlags & ts.ObjectFlags.Reference) {
|
|
548
|
-
const typeRef = type, name = type.getSymbol()?.getName();
|
|
549
|
-
if ((name === "Array" || name === "ReadonlyArray") && typeRef.typeArguments?.length === 1)
|
|
550
|
-
return `v.array(${recurse(typeRef.typeArguments[0])})`;
|
|
551
|
-
}
|
|
552
|
-
if (objectFlags & ts.ObjectFlags.Tuple)
|
|
553
|
-
return `v.tuple([${(type.typeArguments || []).map((t) => recurse(t)).join(", ")}])`;
|
|
554
|
-
const entries = [];
|
|
555
|
-
for (const prop of props) {
|
|
556
|
-
const name = prop.getName();
|
|
557
|
-
if (shouldSkipObjectKey(name)) continue;
|
|
558
|
-
const propType = resolveSymbolType(prop), isOptional = !!(prop.getFlags() & ts.SymbolFlags.Optional);
|
|
559
|
-
let val = recurse(propType);
|
|
560
|
-
isOptional && !val.startsWith("v.optional(") && (val = `v.optional(${val})`), entries.push(`${formatObjectKey(name)}: ${val}`);
|
|
561
|
-
}
|
|
562
|
-
return entries.length === 0 ? "v.object({})" : `v.object({
|
|
563
|
-
${entries.join(`,
|
|
564
|
-
`)},
|
|
565
|
-
})`;
|
|
566
|
-
}
|
|
567
|
-
const stringIndex = type.getStringIndexType();
|
|
568
|
-
if (stringIndex)
|
|
569
|
-
return `v.record(v.string(), ${recurse(stringIndex)})`;
|
|
570
|
-
const numberIndex = type.getNumberIndexType();
|
|
571
|
-
return numberIndex ? `v.record(v.number(), ${recurse(numberIndex)})` : "v.unknown()";
|
|
572
|
-
}
|
|
573
|
-
async function generate(options) {
|
|
574
|
-
const { dir, after, silent } = options, baseDir = resolve(dir), modelsDir = resolve(baseDir, "models"), generatedDir = resolve(baseDir, "generated"), queriesDir = resolve(baseDir, "queries");
|
|
575
|
-
existsSync(generatedDir) || mkdirSync(generatedDir, { recursive: !0 }), loadCache();
|
|
576
|
-
const allModelFiles = readdirSync(modelsDir).filter((f) => f.endsWith(".ts")).sort(), filesWithSchema = allModelFiles.filter(
|
|
577
|
-
(f) => readFileSync(resolve(modelsDir, f), "utf-8").includes("export const schema = table(")
|
|
578
|
-
);
|
|
579
|
-
let filesChanged = [
|
|
580
|
-
writeFileIfChanged(
|
|
581
|
-
resolve(generatedDir, "models.ts"),
|
|
582
|
-
generateModelsFile(allModelFiles)
|
|
583
|
-
),
|
|
584
|
-
writeFileIfChanged(
|
|
585
|
-
resolve(generatedDir, "types.ts"),
|
|
586
|
-
generateTypesFile(filesWithSchema)
|
|
587
|
-
),
|
|
588
|
-
writeFileIfChanged(
|
|
589
|
-
resolve(generatedDir, "tables.ts"),
|
|
590
|
-
generateTablesFile(filesWithSchema)
|
|
591
|
-
),
|
|
592
|
-
writeFileIfChanged(resolve(generatedDir, "README.md"), generateReadmeFile())
|
|
593
|
-
].filter(Boolean).length, queryCount = 0, mutationCount = 0;
|
|
594
|
-
const ts = await import("typescript"), typeToValibot = (paramType) => {
|
|
595
|
-
try {
|
|
596
|
-
return parseTypeString(paramType.trim());
|
|
597
|
-
} catch {
|
|
598
|
-
return null;
|
|
599
|
-
}
|
|
600
|
-
};
|
|
601
|
-
if (existsSync(queriesDir)) {
|
|
602
|
-
const queryFiles = readdirSync(queriesDir).filter((f) => f.endsWith(".ts")), allQueries = [];
|
|
603
|
-
let queryResolver = null;
|
|
604
|
-
const getQueryResolver = () => {
|
|
605
|
-
if (!queryResolver) {
|
|
606
|
-
const allFiles = readdirSync(queriesDir).filter((f) => f.endsWith(".ts")).map((f) => ({
|
|
607
|
-
path: resolve(queriesDir, f),
|
|
608
|
-
content: readFileSync(resolve(queriesDir, f), "utf-8")
|
|
609
|
-
}));
|
|
610
|
-
queryResolver = createTypeResolver(ts, allFiles, queriesDir);
|
|
611
|
-
}
|
|
612
|
-
return queryResolver;
|
|
613
|
-
};
|
|
614
|
-
for (const file of queryFiles) {
|
|
615
|
-
const filePath = resolve(queriesDir, file), fileBaseName = basename(file, ".ts");
|
|
616
|
-
try {
|
|
617
|
-
const content = readFileSync(filePath, "utf-8"), sourceFile = ts.createSourceFile(
|
|
618
|
-
filePath,
|
|
619
|
-
content,
|
|
620
|
-
ts.ScriptTarget.Latest,
|
|
621
|
-
!0
|
|
622
|
-
);
|
|
623
|
-
ts.forEachChild(sourceFile, (node) => {
|
|
624
|
-
if (ts.isVariableStatement(node)) {
|
|
625
|
-
if (!node.modifiers?.find(
|
|
626
|
-
(m) => m.kind === ts.SyntaxKind.ExportKeyword
|
|
627
|
-
)) return;
|
|
628
|
-
const declaration = node.declarationList.declarations[0];
|
|
629
|
-
if (!declaration || !ts.isVariableDeclaration(declaration)) return;
|
|
630
|
-
const name = declaration.name.getText(sourceFile);
|
|
631
|
-
if (name === "permission") return;
|
|
632
|
-
if (declaration.initializer && ts.isArrowFunction(declaration.initializer)) {
|
|
633
|
-
const params = declaration.initializer.parameters;
|
|
634
|
-
let paramType = "void";
|
|
635
|
-
params.length > 0 && (paramType = params[0].type?.getText(sourceFile) || "unknown");
|
|
636
|
-
let valibotCode = typeToValibot(paramType);
|
|
637
|
-
if (!valibotCode && params.length > 0 && params[0].type) {
|
|
638
|
-
const resolver = getQueryResolver(), resolverSourceFile = resolver.program.getSourceFile(filePath);
|
|
639
|
-
if (resolverSourceFile) {
|
|
640
|
-
const resolvedType = resolveParamType(
|
|
641
|
-
ts,
|
|
642
|
-
resolver,
|
|
643
|
-
resolverSourceFile,
|
|
644
|
-
name,
|
|
645
|
-
0
|
|
646
|
-
);
|
|
647
|
-
resolvedType && (valibotCode = resolver.typeToValibot(resolvedType));
|
|
648
|
-
}
|
|
649
|
-
}
|
|
650
|
-
valibotCode ? allQueries.push({
|
|
651
|
-
name,
|
|
652
|
-
params: paramType,
|
|
653
|
-
valibotCode,
|
|
654
|
-
sourceFile: fileBaseName
|
|
655
|
-
}) : !silent && paramType !== "void" && console.error(`\u2717 ${name}: could not resolve type "${paramType}"`);
|
|
656
|
-
}
|
|
657
|
-
}
|
|
658
|
-
});
|
|
659
|
-
} catch (err) {
|
|
660
|
-
silent || console.error(`Error processing ${file}:`, err);
|
|
661
|
-
}
|
|
662
|
-
}
|
|
663
|
-
queryCount = allQueries.length;
|
|
664
|
-
const groupedChanged = writeFileIfChanged(
|
|
665
|
-
resolve(generatedDir, "groupedQueries.ts"),
|
|
666
|
-
generateGroupedQueriesFile(allQueries)
|
|
667
|
-
), syncedChanged = writeFileIfChanged(
|
|
668
|
-
resolve(generatedDir, "syncedQueries.ts"),
|
|
669
|
-
generateSyncedQueriesFile(allQueries)
|
|
670
|
-
);
|
|
671
|
-
groupedChanged && filesChanged++, syncedChanged && filesChanged++;
|
|
672
|
-
}
|
|
673
|
-
const allModelMutations = [], mutationFiles = [], unresolvedModels = [];
|
|
674
|
-
for (const file of allModelFiles) {
|
|
675
|
-
const filePath = resolve(modelsDir, file), fileBaseName = basename(file, ".ts");
|
|
676
|
-
try {
|
|
677
|
-
const content = readFileSync(filePath, "utf-8");
|
|
678
|
-
mutationFiles.push({ path: filePath, content, baseName: fileBaseName });
|
|
679
|
-
const sourceFile = ts.createSourceFile(
|
|
680
|
-
filePath,
|
|
681
|
-
content,
|
|
682
|
-
ts.ScriptTarget.Latest,
|
|
683
|
-
!0
|
|
684
|
-
), result = extractMutationsFromModel(
|
|
685
|
-
ts,
|
|
686
|
-
sourceFile,
|
|
687
|
-
content,
|
|
688
|
-
filePath,
|
|
689
|
-
!!silent,
|
|
690
|
-
typeToValibot
|
|
691
|
-
);
|
|
692
|
-
result && (result.modelName = fileBaseName, allModelMutations.push(result), result.custom.some(
|
|
693
|
-
(m) => m.paramType !== "void" && m.paramType !== "unknown" && !m.valibotCode
|
|
694
|
-
) && unresolvedModels.push({ baseName: fileBaseName, filePath }));
|
|
695
|
-
} catch (err) {
|
|
696
|
-
silent || console.error(`Error extracting mutations from ${file}:`, err);
|
|
697
|
-
}
|
|
698
|
-
}
|
|
699
|
-
if (unresolvedModels.length > 0) {
|
|
700
|
-
const collectTsFiles = (dir2) => {
|
|
701
|
-
const results = [];
|
|
702
|
-
for (const entry of readdirSync(dir2, { withFileTypes: !0 })) {
|
|
703
|
-
const fullPath = resolve(dir2, entry.name);
|
|
704
|
-
entry.isDirectory() && entry.name !== "node_modules" ? results.push(...collectTsFiles(fullPath)) : entry.isFile() && entry.name.endsWith(".ts") && !entry.name.endsWith(".d.ts") && results.push({ path: fullPath, content: readFileSync(fullPath, "utf-8") });
|
|
705
|
-
}
|
|
706
|
-
return results;
|
|
707
|
-
}, allFiles = collectTsFiles(baseDir), modelResolver = createTypeResolver(ts, allFiles, baseDir);
|
|
708
|
-
for (const { baseName, filePath } of unresolvedModels) {
|
|
709
|
-
const resolverSourceFile = modelResolver.program.getSourceFile(filePath);
|
|
710
|
-
if (!resolverSourceFile) continue;
|
|
711
|
-
const resolvedTypes = resolveMutationParamTypes(
|
|
712
|
-
ts,
|
|
713
|
-
modelResolver,
|
|
714
|
-
resolverSourceFile
|
|
715
|
-
);
|
|
716
|
-
if (resolvedTypes.size === 0) continue;
|
|
717
|
-
const content = readFileSync(filePath, "utf-8"), sourceFile = ts.createSourceFile(
|
|
718
|
-
filePath,
|
|
719
|
-
content,
|
|
720
|
-
ts.ScriptTarget.Latest,
|
|
721
|
-
!0
|
|
722
|
-
), result = extractMutationsFromModel(
|
|
723
|
-
ts,
|
|
724
|
-
sourceFile,
|
|
725
|
-
content,
|
|
726
|
-
filePath,
|
|
727
|
-
!!silent,
|
|
728
|
-
typeToValibot,
|
|
729
|
-
resolvedTypes,
|
|
730
|
-
modelResolver.typeToValibot
|
|
731
|
-
);
|
|
732
|
-
if (result) {
|
|
733
|
-
result.modelName = baseName;
|
|
734
|
-
const idx = allModelMutations.findIndex((m) => m.modelName === baseName);
|
|
735
|
-
idx >= 0 && (allModelMutations[idx] = result);
|
|
736
|
-
}
|
|
737
|
-
}
|
|
738
|
-
}
|
|
739
|
-
for (const model of allModelMutations)
|
|
740
|
-
model.hasCRUD && (mutationCount += 3), mutationCount += model.custom.filter(
|
|
741
|
-
(m) => !model.hasCRUD || !["insert", "update", "delete", "upsert"].includes(m.name)
|
|
742
|
-
).length;
|
|
743
|
-
if (allModelMutations.length > 0 && writeFileIfChanged(
|
|
744
|
-
resolve(generatedDir, "syncedMutations.ts"),
|
|
745
|
-
generateSyncedMutationsFile(allModelMutations)
|
|
746
|
-
) && filesChanged++, filesChanged > 0 && !silent && console.info(
|
|
747
|
-
`\u2713 ${allModelFiles.length} models (${filesWithSchema.length} schemas)${queryCount ? `, ${queryCount} queries` : ""}${mutationCount ? `, ${mutationCount} mutations` : ""}`
|
|
748
|
-
), filesChanged > 0 && after) {
|
|
749
|
-
const { execSync } = await import("node:child_process");
|
|
750
|
-
try {
|
|
751
|
-
execSync(after, {
|
|
752
|
-
stdio: "inherit",
|
|
753
|
-
env: { ...process.env, ON_ZERO_GENERATED_DIR: generatedDir }
|
|
754
|
-
});
|
|
755
|
-
} catch (err) {
|
|
756
|
-
silent || console.error(`Error running after command: ${err}`);
|
|
757
|
-
}
|
|
758
|
-
}
|
|
759
|
-
return saveCache(), {
|
|
760
|
-
filesChanged,
|
|
761
|
-
modelCount: allModelFiles.length,
|
|
762
|
-
schemaCount: filesWithSchema.length,
|
|
763
|
-
queryCount,
|
|
764
|
-
mutationCount
|
|
765
|
-
};
|
|
766
|
-
}
|
|
767
|
-
async function watch(options) {
|
|
768
|
-
const { dir, debounce = 1e3 } = options, baseDir = resolve(dir), modelsDir = resolve(baseDir, "models"), queriesDir = resolve(baseDir, "queries"), generatedDir = resolve(baseDir, "generated");
|
|
769
|
-
await generate({ ...options, silent: !0 }), console.info(`\u{1F440} watching...
|
|
770
|
-
`);
|
|
771
|
-
const chokidar = await import("chokidar");
|
|
772
|
-
let debounceTimer = null;
|
|
773
|
-
const debouncedRegenerate = (path, event) => {
|
|
774
|
-
debounceTimer && clearTimeout(debounceTimer), console.info(`
|
|
775
|
-
${event} ${path}`), debounceTimer = setTimeout(() => {
|
|
776
|
-
generate({ ...options, silent: !1 });
|
|
777
|
-
}, debounce);
|
|
778
|
-
}, watcher = chokidar.watch([modelsDir, queriesDir], {
|
|
779
|
-
persistent: !0,
|
|
780
|
-
ignoreInitial: !0,
|
|
781
|
-
ignored: [generatedDir]
|
|
782
|
-
});
|
|
783
|
-
return watcher.on("change", (path) => debouncedRegenerate(path, "\u{1F4DD}")), watcher.on("add", (path) => debouncedRegenerate(path, "\u2795")), watcher.on("unlink", (path) => debouncedRegenerate(path, "\u{1F5D1}\uFE0F ")), watcher;
|
|
784
|
-
}
|
|
785
|
-
export {
|
|
786
|
-
generate,
|
|
787
|
-
watch
|
|
788
|
-
};
|
|
789
|
-
//# sourceMappingURL=generate.js.map
|