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