on-zero 0.1.21 → 0.1.23

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.
Files changed (68) hide show
  1. package/dist/cjs/cli.cjs +17 -420
  2. package/dist/cjs/cli.js +7 -398
  3. package/dist/cjs/cli.js.map +2 -2
  4. package/dist/cjs/cli.native.js +15 -514
  5. package/dist/cjs/cli.native.js.map +1 -1
  6. package/dist/cjs/generate.cjs +370 -0
  7. package/dist/cjs/generate.js +339 -0
  8. package/dist/cjs/generate.js.map +6 -0
  9. package/dist/cjs/generate.native.js +464 -0
  10. package/dist/cjs/generate.native.js.map +1 -0
  11. package/dist/cjs/helpers/createMutators.cjs +4 -3
  12. package/dist/cjs/helpers/createMutators.js +12 -9
  13. package/dist/cjs/helpers/createMutators.js.map +1 -1
  14. package/dist/cjs/helpers/createMutators.native.js +25 -21
  15. package/dist/cjs/helpers/createMutators.native.js.map +1 -1
  16. package/dist/cjs/mutations.cjs +34 -4
  17. package/dist/cjs/mutations.js +29 -4
  18. package/dist/cjs/mutations.js.map +1 -1
  19. package/dist/cjs/mutations.native.js +36 -4
  20. package/dist/cjs/mutations.native.js.map +1 -1
  21. package/dist/cjs/vite-plugin.cjs +84 -0
  22. package/dist/cjs/vite-plugin.js +86 -0
  23. package/dist/cjs/vite-plugin.js.map +6 -0
  24. package/dist/cjs/vite-plugin.native.js +99 -0
  25. package/dist/cjs/vite-plugin.native.js.map +1 -0
  26. package/dist/esm/cli.js +8 -384
  27. package/dist/esm/cli.js.map +2 -2
  28. package/dist/esm/cli.mjs +17 -398
  29. package/dist/esm/cli.mjs.map +1 -1
  30. package/dist/esm/cli.native.js +15 -492
  31. package/dist/esm/cli.native.js.map +1 -1
  32. package/dist/esm/generate.js +317 -0
  33. package/dist/esm/generate.js.map +6 -0
  34. package/dist/esm/generate.mjs +335 -0
  35. package/dist/esm/generate.mjs.map +1 -0
  36. package/dist/esm/generate.native.js +426 -0
  37. package/dist/esm/generate.native.js.map +1 -0
  38. package/dist/esm/helpers/createMutators.js +12 -9
  39. package/dist/esm/helpers/createMutators.js.map +1 -1
  40. package/dist/esm/helpers/createMutators.mjs +4 -3
  41. package/dist/esm/helpers/createMutators.mjs.map +1 -1
  42. package/dist/esm/helpers/createMutators.native.js +25 -21
  43. package/dist/esm/helpers/createMutators.native.js.map +1 -1
  44. package/dist/esm/mutations.js +29 -4
  45. package/dist/esm/mutations.js.map +1 -1
  46. package/dist/esm/mutations.mjs +34 -4
  47. package/dist/esm/mutations.mjs.map +1 -1
  48. package/dist/esm/mutations.native.js +35 -3
  49. package/dist/esm/mutations.native.js.map +1 -1
  50. package/dist/esm/vite-plugin.js +71 -0
  51. package/dist/esm/vite-plugin.js.map +6 -0
  52. package/dist/esm/vite-plugin.mjs +59 -0
  53. package/dist/esm/vite-plugin.mjs.map +1 -0
  54. package/dist/esm/vite-plugin.native.js +71 -0
  55. package/dist/esm/vite-plugin.native.js.map +1 -0
  56. package/package.json +7 -2
  57. package/readme.md +42 -32
  58. package/src/cli.ts +9 -638
  59. package/src/generate.ts +490 -0
  60. package/src/helpers/createMutators.ts +14 -8
  61. package/src/mutations.ts +57 -4
  62. package/src/vite-plugin.ts +110 -0
  63. package/types/generate.d.ts +21 -0
  64. package/types/generate.d.ts.map +1 -0
  65. package/types/helpers/createMutators.d.ts.map +1 -1
  66. package/types/mutations.d.ts.map +1 -1
  67. package/types/vite-plugin.d.ts +16 -0
  68. package/types/vite-plugin.d.ts.map +1 -0
@@ -0,0 +1,317 @@
1
+ import { createHash } from "node:crypto";
2
+ import { existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from "node:fs";
3
+ import { basename, 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
+ if (import.meta.hot) {
51
+ import.meta.hot.accept()
52
+ }
53
+ `;
54
+ }
55
+ function generateTypesFile(modelFiles) {
56
+ const modelNames = modelFiles.map((f) => basename(f, ".ts")).sort(), getSchemaName = (name) => name === "user" ? "userPublic" : name;
57
+ return `import type { TableInsertRow, TableUpdateRow } from 'on-zero'
58
+ import type * as schema from './tables'
59
+
60
+ ${modelNames.map((name) => {
61
+ const pascalName = name.charAt(0).toUpperCase() + name.slice(1), schemaName = getSchemaName(name);
62
+ return `export type ${pascalName} = TableInsertRow<typeof schema.${schemaName}>
63
+ export type ${pascalName}Update = TableUpdateRow<typeof schema.${schemaName}>`;
64
+ }).join(`
65
+
66
+ `)}
67
+ `;
68
+ }
69
+ function generateTablesFile(modelFiles) {
70
+ const modelNames = modelFiles.map((f) => basename(f, ".ts")).sort(), getExportName = (name) => name === "user" ? "userPublic" : name;
71
+ return `// auto-generated by: on-zero generate
72
+ // this is separate from models as otherwise you end up with circular types :/
73
+
74
+ ${modelNames.map((name) => `export { schema as ${getExportName(name)} } from '../models/${name}'`).join(`
75
+ `)}
76
+ `;
77
+ }
78
+ function generateReadmeFile() {
79
+ return `# generated
80
+
81
+ this folder is auto-generated by on-zero. do not edit files here directly.
82
+
83
+ ## what's generated
84
+
85
+ - \`models.ts\` - exports all models from ../models
86
+ - \`types.ts\` - typescript types derived from table schemas
87
+ - \`tables.ts\` - exports table schemas for type inference
88
+ - \`groupedQueries.ts\` - namespaced query re-exports for client setup
89
+ - \`syncedQueries.ts\` - namespaced syncedQuery wrappers for server setup
90
+
91
+ ## usage guidelines
92
+
93
+ **do not import generated files outside of the data folder.**
94
+
95
+ ### queries
96
+
97
+ write your queries as plain functions in \`../queries/\` and import them directly:
98
+
99
+ \`\`\`ts
100
+ // \u2705 good - import from queries
101
+ import { channelMessages } from '~/data/queries/message'
102
+ \`\`\`
103
+
104
+ the generated query files are only used internally by zero client/server setup.
105
+
106
+ ### types
107
+
108
+ you can import types from this folder, but prefer re-exporting from \`../types.ts\`:
109
+
110
+ \`\`\`ts
111
+ // \u274C okay but not preferred
112
+ import type { Message } from '~/data/generated/types'
113
+
114
+ // \u2705 better - re-export from types.ts
115
+ import type { Message } from '~/data/types'
116
+ \`\`\`
117
+
118
+ ## regeneration
119
+
120
+ files are regenerated when you run:
121
+
122
+ \`\`\`bash
123
+ bun on-zero generate
124
+ \`\`\`
125
+
126
+ or in watch mode:
127
+
128
+ \`\`\`bash
129
+ bun on-zero generate --watch
130
+ \`\`\`
131
+
132
+ ## more info
133
+
134
+ see the [on-zero readme](./node_modules/on-zero/README.md) for full documentation.
135
+ `;
136
+ }
137
+ function generateGroupedQueriesFile(queries) {
138
+ return `/**
139
+ * auto-generated by: on-zero generate
140
+ *
141
+ * grouped query re-exports for minification-safe query identity.
142
+ * this file re-exports all query modules - while this breaks tree-shaking,
143
+ * queries are typically small and few in number even in larger apps.
144
+ */
145
+ ${[...new Set(queries.map((q) => q.sourceFile))].sort().map((file) => `export * as ${file} from '../queries/${file}'`).join(`
146
+ `)}
147
+ `;
148
+ }
149
+ function generateSyncedQueriesFile(queries) {
150
+ const queryByFile = /* @__PURE__ */ new Map();
151
+ for (const q of queries)
152
+ queryByFile.has(q.sourceFile) || queryByFile.set(q.sourceFile, []), queryByFile.get(q.sourceFile).push(q);
153
+ const sortedFiles = Array.from(queryByFile.keys()).sort(), imports = `// auto-generated by: on-zero generate
154
+ // server-side query definitions with validators
155
+ import { defineQuery, defineQueries } from '@rocicorp/zero'
156
+ import * as v from 'valibot'
157
+ import * as Queries from './groupedQueries'
158
+ `, namespaceDefs = sortedFiles.map((file) => {
159
+ const queryDefs = queryByFile.get(file).sort((a, b) => a.name.localeCompare(b.name)).map((q) => {
160
+ const lines = q.valibotCode.split(`
161
+ `).filter((l) => l.trim()), schemaLineIndex = lines.findIndex(
162
+ (l) => l.startsWith("export const QueryParams")
163
+ );
164
+ let validatorDef = "";
165
+ if (schemaLineIndex !== -1) {
166
+ const schemaLines = [];
167
+ let openBraces = 0, started = !1;
168
+ for (let i = schemaLineIndex; i < lines.length; i++) {
169
+ const line = lines[i], cleaned = started ? line : line.replace("export const QueryParams = ", "");
170
+ if (schemaLines.push(cleaned), started = !0, openBraces += (cleaned.match(/\{/g) || []).length, openBraces -= (cleaned.match(/\}/g) || []).length, openBraces += (cleaned.match(/\(/g) || []).length, openBraces -= (cleaned.match(/\)/g) || []).length, openBraces === 0 && schemaLines.length > 0)
171
+ break;
172
+ }
173
+ validatorDef = schemaLines.join(`
174
+ `);
175
+ }
176
+ if (q.params === "void" || !validatorDef)
177
+ return ` ${q.name}: defineQuery(() => Queries.${file}.${q.name}()),`;
178
+ const indentedValidator = validatorDef.split(`
179
+ `).map((line, i) => i === 0 ? line : ` ${line}`).join(`
180
+ `);
181
+ return ` ${q.name}: defineQuery(
182
+ ${indentedValidator},
183
+ ({ args }) => Queries.${file}.${q.name}(args)
184
+ ),`;
185
+ }).join(`
186
+ `);
187
+ return `const ${file} = {
188
+ ${queryDefs}
189
+ }`;
190
+ }).join(`
191
+
192
+ `), queriesObject = sortedFiles.map((file) => ` ${file},`).join(`
193
+ `);
194
+ return `${imports}
195
+ ${namespaceDefs}
196
+
197
+ export const queries = defineQueries({
198
+ ${queriesObject}
199
+ })
200
+ `;
201
+ }
202
+ async function generate(options) {
203
+ const { dir, after, silent } = options, baseDir = resolve(dir), modelsDir = resolve(baseDir, "models"), generatedDir = resolve(baseDir, "generated"), queriesDir = resolve(baseDir, "queries");
204
+ existsSync(generatedDir) || mkdirSync(generatedDir, { recursive: !0 }), loadCache();
205
+ const allModelFiles = readdirSync(modelsDir).filter((f) => f.endsWith(".ts")).sort(), filesWithSchema = allModelFiles.filter(
206
+ (f) => readFileSync(resolve(modelsDir, f), "utf-8").includes("export const schema = table(")
207
+ );
208
+ let filesChanged = [
209
+ writeFileIfChanged(
210
+ resolve(generatedDir, "models.ts"),
211
+ generateModelsFile(allModelFiles)
212
+ ),
213
+ writeFileIfChanged(
214
+ resolve(generatedDir, "types.ts"),
215
+ generateTypesFile(filesWithSchema)
216
+ ),
217
+ writeFileIfChanged(
218
+ resolve(generatedDir, "tables.ts"),
219
+ generateTablesFile(filesWithSchema)
220
+ ),
221
+ writeFileIfChanged(resolve(generatedDir, "README.md"), generateReadmeFile())
222
+ ].filter(Boolean).length, queryCount = 0;
223
+ if (existsSync(queriesDir)) {
224
+ const ts = await import("typescript"), { ModelToValibot } = await import("@sinclair/typebox-codegen/model"), { TypeScriptToModel } = await import("@sinclair/typebox-codegen/typescript"), queryFiles = readdirSync(queriesDir).filter((f) => f.endsWith(".ts")), allQueries = [];
225
+ for (const file of queryFiles) {
226
+ const filePath = resolve(queriesDir, file), fileBaseName = basename(file, ".ts");
227
+ try {
228
+ const content = readFileSync(filePath, "utf-8"), sourceFile = ts.createSourceFile(
229
+ filePath,
230
+ content,
231
+ ts.ScriptTarget.Latest,
232
+ !0
233
+ );
234
+ ts.forEachChild(sourceFile, (node) => {
235
+ if (ts.isVariableStatement(node)) {
236
+ if (!node.modifiers?.find(
237
+ (m) => m.kind === ts.SyntaxKind.ExportKeyword
238
+ )) return;
239
+ const declaration = node.declarationList.declarations[0];
240
+ if (!declaration || !ts.isVariableDeclaration(declaration)) return;
241
+ const name = declaration.name.getText(sourceFile);
242
+ if (name === "permission") return;
243
+ if (declaration.initializer && ts.isArrowFunction(declaration.initializer)) {
244
+ const params = declaration.initializer.parameters;
245
+ let paramType = "void";
246
+ params.length > 0 && (paramType = params[0].type?.getText(sourceFile) || "unknown");
247
+ try {
248
+ const typeString = `type QueryParams = ${paramType}`, model = TypeScriptToModel.Generate(typeString), valibotCode = ModelToValibot.Generate(model);
249
+ allQueries.push({
250
+ name,
251
+ params: paramType,
252
+ valibotCode,
253
+ sourceFile: fileBaseName
254
+ });
255
+ } catch (err) {
256
+ silent || console.error(`\u2717 ${name}: ${err}`);
257
+ }
258
+ }
259
+ }
260
+ });
261
+ } catch (err) {
262
+ silent || console.error(`Error processing ${file}:`, err);
263
+ }
264
+ }
265
+ queryCount = allQueries.length;
266
+ const groupedChanged = writeFileIfChanged(
267
+ resolve(generatedDir, "groupedQueries.ts"),
268
+ generateGroupedQueriesFile(allQueries)
269
+ ), syncedChanged = writeFileIfChanged(
270
+ resolve(generatedDir, "syncedQueries.ts"),
271
+ generateSyncedQueriesFile(allQueries)
272
+ );
273
+ groupedChanged && filesChanged++, syncedChanged && filesChanged++;
274
+ }
275
+ if (filesChanged > 0 && !silent && console.info(
276
+ `\u2713 ${allModelFiles.length} models (${filesWithSchema.length} schemas)${queryCount ? `, ${queryCount} queries` : ""}`
277
+ ), filesChanged > 0 && after) {
278
+ const { execSync } = await import("node:child_process");
279
+ try {
280
+ execSync(after, {
281
+ stdio: "inherit",
282
+ env: { ...process.env, ON_ZERO_GENERATED_DIR: generatedDir }
283
+ });
284
+ } catch (err) {
285
+ silent || console.error(`Error running after command: ${err}`);
286
+ }
287
+ }
288
+ return saveCache(), {
289
+ filesChanged,
290
+ modelCount: allModelFiles.length,
291
+ schemaCount: filesWithSchema.length,
292
+ queryCount
293
+ };
294
+ }
295
+ async function watch(options) {
296
+ const { dir, debounce = 1e3 } = options, baseDir = resolve(dir), modelsDir = resolve(baseDir, "models"), queriesDir = resolve(baseDir, "queries"), generatedDir = resolve(baseDir, "generated");
297
+ await generate({ ...options, silent: !0 }), console.info(`\u{1F440} watching...
298
+ `);
299
+ const chokidar = await import("chokidar");
300
+ let debounceTimer = null;
301
+ const debouncedRegenerate = (path, event) => {
302
+ debounceTimer && clearTimeout(debounceTimer), console.info(`
303
+ ${event} ${path}`), debounceTimer = setTimeout(() => {
304
+ generate({ ...options, silent: !1 });
305
+ }, debounce);
306
+ }, watcher = chokidar.watch([modelsDir, queriesDir], {
307
+ persistent: !0,
308
+ ignoreInitial: !0,
309
+ ignored: [generatedDir]
310
+ });
311
+ 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;
312
+ }
313
+ export {
314
+ generate,
315
+ watch
316
+ };
317
+ //# sourceMappingURL=generate.js.map
@@ -0,0 +1,6 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/generate.ts"],
4
+ "mappings": "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",
5
+ "names": []
6
+ }
@@ -0,0 +1,335 @@
1
+ import { createHash } from "node:crypto";
2
+ import { existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from "node:fs";
3
+ import { basename, resolve } from "node:path";
4
+ const hash = s => createHash("sha256").update(s).digest("hex");
5
+ let generateCache = {},
6
+ generateCachePath = "";
7
+ function getCacheDir() {
8
+ let dir = process.cwd();
9
+ for (; dir !== "/";) {
10
+ const nm = resolve(dir, "node_modules");
11
+ if (existsSync(nm)) {
12
+ const cacheDir = resolve(nm, ".on-zero");
13
+ return existsSync(cacheDir) || mkdirSync(cacheDir, {
14
+ recursive: !0
15
+ }), cacheDir;
16
+ }
17
+ dir = resolve(dir, "..");
18
+ }
19
+ return null;
20
+ }
21
+ function loadCache() {
22
+ const cacheDir = getCacheDir();
23
+ if (cacheDir) {
24
+ generateCachePath = resolve(cacheDir, "generate-cache.json");
25
+ try {
26
+ generateCache = JSON.parse(readFileSync(generateCachePath, "utf-8"));
27
+ } catch {
28
+ generateCache = {};
29
+ }
30
+ }
31
+ }
32
+ function saveCache() {
33
+ generateCachePath && writeFileSync(generateCachePath, JSON.stringify(generateCache) + `
34
+ `, "utf-8");
35
+ }
36
+ function writeFileIfChanged(filePath, content) {
37
+ const contentHash = hash(content);
38
+ return generateCache[filePath] === contentHash && existsSync(filePath) ? !1 : (writeFileSync(filePath, content, "utf-8"), generateCache[filePath] = contentHash, !0);
39
+ }
40
+ function generateModelsFile(modelFiles) {
41
+ const modelNames = modelFiles.map(f => basename(f, ".ts")).sort(),
42
+ getImportName = name => name === "user" ? "userPublic" : name,
43
+ imports = modelNames.map(name => `import * as ${getImportName(name)} from '../models/${name}'`).join(`
44
+ `),
45
+ modelsObj = `export const models = {
46
+ ${[...modelNames].sort((a, b) => getImportName(a).localeCompare(getImportName(b))).map(name => ` ${getImportName(name)},`).join(`
47
+ `)}
48
+ }`;
49
+ return `// auto-generated by: on-zero generate
50
+ ${imports}
51
+
52
+ ${modelsObj}
53
+
54
+ if (import.meta.hot) {
55
+ import.meta.hot.accept()
56
+ }
57
+ `;
58
+ }
59
+ function generateTypesFile(modelFiles) {
60
+ const modelNames = modelFiles.map(f => basename(f, ".ts")).sort(),
61
+ getSchemaName = name => name === "user" ? "userPublic" : name;
62
+ return `import type { TableInsertRow, TableUpdateRow } from 'on-zero'
63
+ import type * as schema from './tables'
64
+
65
+ ${modelNames.map(name => {
66
+ const pascalName = name.charAt(0).toUpperCase() + name.slice(1),
67
+ schemaName = getSchemaName(name);
68
+ return `export type ${pascalName} = TableInsertRow<typeof schema.${schemaName}>
69
+ export type ${pascalName}Update = TableUpdateRow<typeof schema.${schemaName}>`;
70
+ }).join(`
71
+
72
+ `)}
73
+ `;
74
+ }
75
+ function generateTablesFile(modelFiles) {
76
+ const modelNames = modelFiles.map(f => basename(f, ".ts")).sort(),
77
+ getExportName = name => name === "user" ? "userPublic" : name;
78
+ return `// auto-generated by: on-zero generate
79
+ // this is separate from models as otherwise you end up with circular types :/
80
+
81
+ ${modelNames.map(name => `export { schema as ${getExportName(name)} } from '../models/${name}'`).join(`
82
+ `)}
83
+ `;
84
+ }
85
+ function generateReadmeFile() {
86
+ return `# generated
87
+
88
+ this folder is auto-generated by on-zero. do not edit files here directly.
89
+
90
+ ## what's generated
91
+
92
+ - \`models.ts\` - exports all models from ../models
93
+ - \`types.ts\` - typescript types derived from table schemas
94
+ - \`tables.ts\` - exports table schemas for type inference
95
+ - \`groupedQueries.ts\` - namespaced query re-exports for client setup
96
+ - \`syncedQueries.ts\` - namespaced syncedQuery wrappers for server setup
97
+
98
+ ## usage guidelines
99
+
100
+ **do not import generated files outside of the data folder.**
101
+
102
+ ### queries
103
+
104
+ write your queries as plain functions in \`../queries/\` and import them directly:
105
+
106
+ \`\`\`ts
107
+ // \u2705 good - import from queries
108
+ import { channelMessages } from '~/data/queries/message'
109
+ \`\`\`
110
+
111
+ the generated query files are only used internally by zero client/server setup.
112
+
113
+ ### types
114
+
115
+ you can import types from this folder, but prefer re-exporting from \`../types.ts\`:
116
+
117
+ \`\`\`ts
118
+ // \u274C okay but not preferred
119
+ import type { Message } from '~/data/generated/types'
120
+
121
+ // \u2705 better - re-export from types.ts
122
+ import type { Message } from '~/data/types'
123
+ \`\`\`
124
+
125
+ ## regeneration
126
+
127
+ files are regenerated when you run:
128
+
129
+ \`\`\`bash
130
+ bun on-zero generate
131
+ \`\`\`
132
+
133
+ or in watch mode:
134
+
135
+ \`\`\`bash
136
+ bun on-zero generate --watch
137
+ \`\`\`
138
+
139
+ ## more info
140
+
141
+ see the [on-zero readme](./node_modules/on-zero/README.md) for full documentation.
142
+ `;
143
+ }
144
+ function generateGroupedQueriesFile(queries) {
145
+ return `/**
146
+ * auto-generated by: on-zero generate
147
+ *
148
+ * grouped query re-exports for minification-safe query identity.
149
+ * this file re-exports all query modules - while this breaks tree-shaking,
150
+ * queries are typically small and few in number even in larger apps.
151
+ */
152
+ ${[...new Set(queries.map(q => q.sourceFile))].sort().map(file => `export * as ${file} from '../queries/${file}'`).join(`
153
+ `)}
154
+ `;
155
+ }
156
+ function generateSyncedQueriesFile(queries) {
157
+ const queryByFile = /* @__PURE__ */new Map();
158
+ for (const q of queries) queryByFile.has(q.sourceFile) || queryByFile.set(q.sourceFile, []), queryByFile.get(q.sourceFile).push(q);
159
+ const sortedFiles = Array.from(queryByFile.keys()).sort(),
160
+ imports = `// auto-generated by: on-zero generate
161
+ // server-side query definitions with validators
162
+ import { defineQuery, defineQueries } from '@rocicorp/zero'
163
+ import * as v from 'valibot'
164
+ import * as Queries from './groupedQueries'
165
+ `,
166
+ namespaceDefs = sortedFiles.map(file => {
167
+ const queryDefs = queryByFile.get(file).sort((a, b) => a.name.localeCompare(b.name)).map(q => {
168
+ const lines = q.valibotCode.split(`
169
+ `).filter(l => l.trim()),
170
+ schemaLineIndex = lines.findIndex(l => l.startsWith("export const QueryParams"));
171
+ let validatorDef = "";
172
+ if (schemaLineIndex !== -1) {
173
+ const schemaLines = [];
174
+ let openBraces = 0,
175
+ started = !1;
176
+ for (let i = schemaLineIndex; i < lines.length; i++) {
177
+ const line = lines[i],
178
+ cleaned = started ? line : line.replace("export const QueryParams = ", "");
179
+ if (schemaLines.push(cleaned), started = !0, openBraces += (cleaned.match(/\{/g) || []).length, openBraces -= (cleaned.match(/\}/g) || []).length, openBraces += (cleaned.match(/\(/g) || []).length, openBraces -= (cleaned.match(/\)/g) || []).length, openBraces === 0 && schemaLines.length > 0) break;
180
+ }
181
+ validatorDef = schemaLines.join(`
182
+ `);
183
+ }
184
+ if (q.params === "void" || !validatorDef) return ` ${q.name}: defineQuery(() => Queries.${file}.${q.name}()),`;
185
+ const indentedValidator = validatorDef.split(`
186
+ `).map((line, i) => i === 0 ? line : ` ${line}`).join(`
187
+ `);
188
+ return ` ${q.name}: defineQuery(
189
+ ${indentedValidator},
190
+ ({ args }) => Queries.${file}.${q.name}(args)
191
+ ),`;
192
+ }).join(`
193
+ `);
194
+ return `const ${file} = {
195
+ ${queryDefs}
196
+ }`;
197
+ }).join(`
198
+
199
+ `),
200
+ queriesObject = sortedFiles.map(file => ` ${file},`).join(`
201
+ `);
202
+ return `${imports}
203
+ ${namespaceDefs}
204
+
205
+ export const queries = defineQueries({
206
+ ${queriesObject}
207
+ })
208
+ `;
209
+ }
210
+ async function generate(options) {
211
+ const {
212
+ dir,
213
+ after,
214
+ silent
215
+ } = options,
216
+ baseDir = resolve(dir),
217
+ modelsDir = resolve(baseDir, "models"),
218
+ generatedDir = resolve(baseDir, "generated"),
219
+ queriesDir = resolve(baseDir, "queries");
220
+ existsSync(generatedDir) || mkdirSync(generatedDir, {
221
+ recursive: !0
222
+ }), loadCache();
223
+ const allModelFiles = readdirSync(modelsDir).filter(f => f.endsWith(".ts")).sort(),
224
+ filesWithSchema = allModelFiles.filter(f => readFileSync(resolve(modelsDir, f), "utf-8").includes("export const schema = table("));
225
+ let filesChanged = [writeFileIfChanged(resolve(generatedDir, "models.ts"), generateModelsFile(allModelFiles)), writeFileIfChanged(resolve(generatedDir, "types.ts"), generateTypesFile(filesWithSchema)), writeFileIfChanged(resolve(generatedDir, "tables.ts"), generateTablesFile(filesWithSchema)), writeFileIfChanged(resolve(generatedDir, "README.md"), generateReadmeFile())].filter(Boolean).length,
226
+ queryCount = 0;
227
+ if (existsSync(queriesDir)) {
228
+ const ts = await import("typescript"),
229
+ {
230
+ ModelToValibot
231
+ } = await import("@sinclair/typebox-codegen/model"),
232
+ {
233
+ TypeScriptToModel
234
+ } = await import("@sinclair/typebox-codegen/typescript"),
235
+ queryFiles = readdirSync(queriesDir).filter(f => f.endsWith(".ts")),
236
+ allQueries = [];
237
+ for (const file of queryFiles) {
238
+ const filePath = resolve(queriesDir, file),
239
+ fileBaseName = basename(file, ".ts");
240
+ try {
241
+ const content = readFileSync(filePath, "utf-8"),
242
+ sourceFile = ts.createSourceFile(filePath, content, ts.ScriptTarget.Latest, !0);
243
+ ts.forEachChild(sourceFile, node => {
244
+ if (ts.isVariableStatement(node)) {
245
+ if (!node.modifiers?.find(m => m.kind === ts.SyntaxKind.ExportKeyword)) return;
246
+ const declaration = node.declarationList.declarations[0];
247
+ if (!declaration || !ts.isVariableDeclaration(declaration)) return;
248
+ const name = declaration.name.getText(sourceFile);
249
+ if (name === "permission") return;
250
+ if (declaration.initializer && ts.isArrowFunction(declaration.initializer)) {
251
+ const params = declaration.initializer.parameters;
252
+ let paramType = "void";
253
+ params.length > 0 && (paramType = params[0].type?.getText(sourceFile) || "unknown");
254
+ try {
255
+ const typeString = `type QueryParams = ${paramType}`,
256
+ model = TypeScriptToModel.Generate(typeString),
257
+ valibotCode = ModelToValibot.Generate(model);
258
+ allQueries.push({
259
+ name,
260
+ params: paramType,
261
+ valibotCode,
262
+ sourceFile: fileBaseName
263
+ });
264
+ } catch (err) {
265
+ silent || console.error(`\u2717 ${name}: ${err}`);
266
+ }
267
+ }
268
+ }
269
+ });
270
+ } catch (err) {
271
+ silent || console.error(`Error processing ${file}:`, err);
272
+ }
273
+ }
274
+ queryCount = allQueries.length;
275
+ const groupedChanged = writeFileIfChanged(resolve(generatedDir, "groupedQueries.ts"), generateGroupedQueriesFile(allQueries)),
276
+ syncedChanged = writeFileIfChanged(resolve(generatedDir, "syncedQueries.ts"), generateSyncedQueriesFile(allQueries));
277
+ groupedChanged && filesChanged++, syncedChanged && filesChanged++;
278
+ }
279
+ if (filesChanged > 0 && !silent && console.info(`\u2713 ${allModelFiles.length} models (${filesWithSchema.length} schemas)${queryCount ? `, ${queryCount} queries` : ""}`), filesChanged > 0 && after) {
280
+ const {
281
+ execSync
282
+ } = await import("node:child_process");
283
+ try {
284
+ execSync(after, {
285
+ stdio: "inherit",
286
+ env: {
287
+ ...process.env,
288
+ ON_ZERO_GENERATED_DIR: generatedDir
289
+ }
290
+ });
291
+ } catch (err) {
292
+ silent || console.error(`Error running after command: ${err}`);
293
+ }
294
+ }
295
+ return saveCache(), {
296
+ filesChanged,
297
+ modelCount: allModelFiles.length,
298
+ schemaCount: filesWithSchema.length,
299
+ queryCount
300
+ };
301
+ }
302
+ async function watch(options) {
303
+ const {
304
+ dir,
305
+ debounce = 1e3
306
+ } = options,
307
+ baseDir = resolve(dir),
308
+ modelsDir = resolve(baseDir, "models"),
309
+ queriesDir = resolve(baseDir, "queries"),
310
+ generatedDir = resolve(baseDir, "generated");
311
+ await generate({
312
+ ...options,
313
+ silent: !0
314
+ }), console.info(`\u{1F440} watching...
315
+ `);
316
+ const chokidar = await import("chokidar");
317
+ let debounceTimer = null;
318
+ const debouncedRegenerate = (path, event) => {
319
+ debounceTimer && clearTimeout(debounceTimer), console.info(`
320
+ ${event} ${path}`), debounceTimer = setTimeout(() => {
321
+ generate({
322
+ ...options,
323
+ silent: !1
324
+ });
325
+ }, debounce);
326
+ },
327
+ watcher = chokidar.watch([modelsDir, queriesDir], {
328
+ persistent: !0,
329
+ ignoreInitial: !0,
330
+ ignored: [generatedDir]
331
+ });
332
+ 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;
333
+ }
334
+ export { generate, watch };
335
+ //# sourceMappingURL=generate.mjs.map