appflare 0.2.24 → 0.2.26

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 (138) hide show
  1. package/Documentation.md +758 -758
  2. package/cli/commands/index.ts +238 -238
  3. package/cli/generate.ts +178 -178
  4. package/cli/index.ts +120 -120
  5. package/cli/load-config.ts +184 -184
  6. package/cli/schema-compiler.ts +1183 -1183
  7. package/cli/templates/auth/README.md +156 -156
  8. package/cli/templates/auth/config.ts +61 -61
  9. package/cli/templates/auth/route-config.ts +1 -1
  10. package/cli/templates/auth/route-handler.ts +1 -1
  11. package/cli/templates/auth/route-request-utils.ts +5 -5
  12. package/cli/templates/auth/route.config.ts +18 -18
  13. package/cli/templates/auth/route.handler.ts +18 -18
  14. package/cli/templates/auth/route.request-utils.ts +55 -55
  15. package/cli/templates/auth/route.ts +14 -14
  16. package/cli/templates/core/README.md +266 -266
  17. package/cli/templates/core/app-creation.ts +19 -19
  18. package/cli/templates/core/client/appflare.ts +112 -112
  19. package/cli/templates/core/client/handlers/index.ts +748 -749
  20. package/cli/templates/core/client/handlers.ts +1 -1
  21. package/cli/templates/core/client/index.ts +7 -7
  22. package/cli/templates/core/client/storage.ts +180 -180
  23. package/cli/templates/core/client/types.ts +184 -184
  24. package/cli/templates/core/client-modules/appflare.ts +1 -1
  25. package/cli/templates/core/client-modules/handlers.ts +1 -1
  26. package/cli/templates/core/client-modules/index.ts +1 -1
  27. package/cli/templates/core/client-modules/storage.ts +1 -1
  28. package/cli/templates/core/client-modules/types.ts +1 -1
  29. package/cli/templates/core/client.artifacts.ts +39 -39
  30. package/cli/templates/core/client.ts +4 -4
  31. package/cli/templates/core/drizzle.ts +15 -15
  32. package/cli/templates/core/export.ts +14 -14
  33. package/cli/templates/core/handlers.route.ts +24 -24
  34. package/cli/templates/core/handlers.ts +1 -1
  35. package/cli/templates/core/imports.ts +9 -9
  36. package/cli/templates/core/server.ts +38 -38
  37. package/cli/templates/core/types.ts +6 -6
  38. package/cli/templates/core/wrangler.ts +109 -109
  39. package/cli/templates/dashboard/builders/functions/index.ts +17 -17
  40. package/cli/templates/dashboard/builders/functions/render-page/header.ts +20 -20
  41. package/cli/templates/dashboard/builders/functions/render-page/index.ts +33 -33
  42. package/cli/templates/dashboard/builders/functions/render-page/request-panel.ts +171 -171
  43. package/cli/templates/dashboard/builders/functions/render-page/result-panel.ts +85 -85
  44. package/cli/templates/dashboard/builders/functions/render-page/scripts.ts +554 -554
  45. package/cli/templates/dashboard/builders/navigation.ts +122 -122
  46. package/cli/templates/dashboard/builders/storage/index.ts +13 -13
  47. package/cli/templates/dashboard/builders/storage/routes/create-directory-route.ts +29 -29
  48. package/cli/templates/dashboard/builders/storage/routes/delete-route.ts +18 -18
  49. package/cli/templates/dashboard/builders/storage/routes/download-route.ts +23 -23
  50. package/cli/templates/dashboard/builders/storage/routes/index.ts +22 -22
  51. package/cli/templates/dashboard/builders/storage/routes/list-route.ts +25 -25
  52. package/cli/templates/dashboard/builders/storage/routes/preview-route.ts +21 -21
  53. package/cli/templates/dashboard/builders/storage/routes/upload-route.ts +21 -21
  54. package/cli/templates/dashboard/builders/storage/runtime/helpers.ts +72 -72
  55. package/cli/templates/dashboard/builders/storage/runtime/storage-page.ts +130 -130
  56. package/cli/templates/dashboard/builders/table-routes/common/drawer-panel.ts +27 -27
  57. package/cli/templates/dashboard/builders/table-routes/common/pagination.ts +30 -30
  58. package/cli/templates/dashboard/builders/table-routes/common/search-bar.ts +23 -23
  59. package/cli/templates/dashboard/builders/table-routes/fragments.ts +217 -217
  60. package/cli/templates/dashboard/builders/table-routes/helpers.ts +45 -45
  61. package/cli/templates/dashboard/builders/table-routes/index.ts +8 -8
  62. package/cli/templates/dashboard/builders/table-routes/table/actions-cell.ts +71 -71
  63. package/cli/templates/dashboard/builders/table-routes/table/get-route.ts +291 -291
  64. package/cli/templates/dashboard/builders/table-routes/table/index.ts +80 -80
  65. package/cli/templates/dashboard/builders/table-routes/table/post-routes.ts +163 -163
  66. package/cli/templates/dashboard/builders/table-routes/table-route.ts +7 -7
  67. package/cli/templates/dashboard/builders/table-routes/users/get-route.ts +69 -69
  68. package/cli/templates/dashboard/builders/table-routes/users/html/modals.ts +57 -57
  69. package/cli/templates/dashboard/builders/table-routes/users/html/page.ts +27 -27
  70. package/cli/templates/dashboard/builders/table-routes/users/html/table.ts +128 -128
  71. package/cli/templates/dashboard/builders/table-routes/users/index.ts +32 -32
  72. package/cli/templates/dashboard/builders/table-routes/users/post-routes.ts +150 -150
  73. package/cli/templates/dashboard/builders/table-routes/users/redirect.ts +14 -14
  74. package/cli/templates/dashboard/builders/table-routes/users-route.ts +10 -10
  75. package/cli/templates/dashboard/components/dashboard-home.ts +23 -23
  76. package/cli/templates/dashboard/components/layout.ts +388 -388
  77. package/cli/templates/dashboard/components/login-page.ts +65 -65
  78. package/cli/templates/dashboard/index.ts +61 -61
  79. package/cli/templates/dashboard/types.ts +9 -9
  80. package/cli/templates/handlers/README.md +353 -353
  81. package/cli/templates/handlers/auth.ts +37 -37
  82. package/cli/templates/handlers/execution.ts +42 -42
  83. package/cli/templates/handlers/generators/context/context-creation.ts +101 -101
  84. package/cli/templates/handlers/generators/context/error-helpers.ts +11 -11
  85. package/cli/templates/handlers/generators/context/scheduler.ts +24 -24
  86. package/cli/templates/handlers/generators/context/storage-api.ts +134 -112
  87. package/cli/templates/handlers/generators/context/storage-helpers.ts +59 -59
  88. package/cli/templates/handlers/generators/context/types.ts +18 -18
  89. package/cli/templates/handlers/generators/context.ts +43 -43
  90. package/cli/templates/handlers/generators/execution.ts +15 -15
  91. package/cli/templates/handlers/generators/handlers.ts +13 -13
  92. package/cli/templates/handlers/generators/registration/modules/cron.ts +26 -26
  93. package/cli/templates/handlers/generators/registration/modules/realtime/auth.ts +75 -75
  94. package/cli/templates/handlers/generators/registration/modules/realtime/durable-object.ts +144 -144
  95. package/cli/templates/handlers/generators/registration/modules/realtime/index.ts +14 -14
  96. package/cli/templates/handlers/generators/registration/modules/realtime/publisher.ts +102 -102
  97. package/cli/templates/handlers/generators/registration/modules/realtime/routes.ts +164 -164
  98. package/cli/templates/handlers/generators/registration/modules/realtime/types.ts +30 -30
  99. package/cli/templates/handlers/generators/registration/modules/realtime/utils.ts +516 -516
  100. package/cli/templates/handlers/generators/registration/modules/scheduler.ts +56 -56
  101. package/cli/templates/handlers/generators/registration/modules/storage.ts +196 -194
  102. package/cli/templates/handlers/generators/registration/sections.ts +210 -210
  103. package/cli/templates/handlers/generators/types/context.ts +68 -66
  104. package/cli/templates/handlers/generators/types/core.ts +106 -106
  105. package/cli/templates/handlers/generators/types/operations.ts +135 -135
  106. package/cli/templates/handlers/generators/types/query-definitions/filter-and-where-types.ts +259 -259
  107. package/cli/templates/handlers/generators/types/query-definitions/query-api-types.ts +135 -135
  108. package/cli/templates/handlers/generators/types/query-definitions/query-helper-functions.ts +1031 -1031
  109. package/cli/templates/handlers/generators/types/query-definitions/schema-and-table-types.ts +246 -246
  110. package/cli/templates/handlers/generators/types/query-definitions.ts +13 -13
  111. package/cli/templates/handlers/generators/types/query-runtime/handled-error.ts +13 -13
  112. package/cli/templates/handlers/generators/types/query-runtime/runtime-aggregate-and-footer.ts +174 -174
  113. package/cli/templates/handlers/generators/types/query-runtime/runtime-read.ts +121 -121
  114. package/cli/templates/handlers/generators/types/query-runtime/runtime-setup.ts +45 -45
  115. package/cli/templates/handlers/generators/types/query-runtime/runtime-write.ts +676 -676
  116. package/cli/templates/handlers/generators/types/query-runtime.ts +15 -15
  117. package/cli/templates/handlers/index.ts +43 -43
  118. package/cli/templates/handlers/operations.ts +116 -116
  119. package/cli/templates/handlers/registration.ts +91 -83
  120. package/cli/templates/handlers/types.ts +15 -15
  121. package/cli/templates/handlers/utils.ts +48 -48
  122. package/cli/types.ts +110 -110
  123. package/cli/utils/handler-discovery.ts +466 -466
  124. package/cli/utils/json-utils.ts +24 -24
  125. package/cli/utils/path-utils.ts +19 -19
  126. package/cli/utils/schema-discovery.ts +399 -399
  127. package/dist/cli/index.js +61 -28
  128. package/dist/cli/index.mjs +61 -28
  129. package/index.ts +18 -18
  130. package/package.json +58 -58
  131. package/react/index.ts +5 -5
  132. package/react/use-infinite-query.ts +252 -252
  133. package/react/use-mutation.ts +89 -89
  134. package/react/use-query.ts +207 -207
  135. package/schema.ts +415 -415
  136. package/test-better-auth-hash.ts +2 -2
  137. package/tsconfig.json +6 -6
  138. package/tsup.config.ts +82 -82
package/schema.ts CHANGED
@@ -1,415 +1,415 @@
1
- export type ColumnType = "int" | "string" | "boolean" | "date";
2
-
3
- export type ColumnBuilderOptions = {
4
- sqlName?: string;
5
- length?: number;
6
- };
7
-
8
- export type PrimaryKeyOptions = {
9
- autoIncrement?: boolean;
10
- };
11
-
12
- export type ForeignKeyAction =
13
- | "cascade"
14
- | "set null"
15
- | "set default"
16
- | "restrict"
17
- | "no action";
18
-
19
- export type RelationOneOptions = {
20
- referenceField?: string;
21
- field?: string;
22
- fkType?: ColumnType;
23
- sqlName?: string;
24
- notNull?: boolean;
25
- nullable?: boolean;
26
- onDelete?: ForeignKeyAction;
27
- onUpdate?: ForeignKeyAction;
28
- };
29
-
30
- export type RelationManyOptions = {
31
- referenceField?: string;
32
- field?: string;
33
- fkType?: ColumnType;
34
- sqlName?: string;
35
- notNull?: boolean;
36
- nullable?: boolean;
37
- onDelete?: ForeignKeyAction;
38
- onUpdate?: ForeignKeyAction;
39
- };
40
-
41
- export type RelationManyToManyOptions = {
42
- referenceField?: string;
43
- targetReferenceField?: string;
44
- junctionTable?: string;
45
- sourceField?: string;
46
- targetField?: string;
47
- sourceSqlName?: string;
48
- targetSqlName?: string;
49
- onDelete?: ForeignKeyAction;
50
- onUpdate?: ForeignKeyAction;
51
- };
52
-
53
- export type ColumnReference = {
54
- table: string;
55
- column: string;
56
- onDelete?: ForeignKeyAction;
57
- onUpdate?: ForeignKeyAction;
58
- };
59
-
60
- export type ColumnDefinition = {
61
- kind: "column";
62
- type: ColumnType;
63
- sqlName?: string;
64
- length?: number;
65
- notNull?: boolean;
66
- nullable?: boolean;
67
- primaryKey?: boolean;
68
- autoIncrement?: boolean;
69
- /** When true the column stores a UUID generated at runtime (string type). */
70
- uuidPrimaryKey?: boolean;
71
- unique?: true | { name?: string };
72
- index?: true | { name?: string };
73
- sqlDefault?: unknown;
74
- /** When true, the SQL column gets DEFAULT CURRENT_TIMESTAMP and runtime inserts use `new Date()`. */
75
- nowDefault?: boolean;
76
- runtimeDefaultFn?: () => unknown;
77
- references?: ColumnReference;
78
- };
79
-
80
- export type OneRelationDefinition = {
81
- kind: "relation";
82
- relation: "one";
83
- targetTable: string;
84
- field?: string;
85
- referenceField?: string;
86
- fkType?: ColumnType;
87
- sqlName?: string;
88
- notNull?: boolean;
89
- nullable?: boolean;
90
- onDelete?: ForeignKeyAction;
91
- onUpdate?: ForeignKeyAction;
92
- };
93
-
94
- export type ManyRelationDefinition = {
95
- kind: "relation";
96
- relation: "many";
97
- targetTable: string;
98
- field?: string;
99
- referenceField?: string;
100
- fkType?: ColumnType;
101
- sqlName?: string;
102
- notNull?: boolean;
103
- nullable?: boolean;
104
- onDelete?: ForeignKeyAction;
105
- onUpdate?: ForeignKeyAction;
106
- };
107
-
108
- export type ManyToManyRelationDefinition = {
109
- kind: "relation";
110
- relation: "manyToMany";
111
- targetTable: string;
112
- referenceField?: string;
113
- targetReferenceField?: string;
114
- junctionTable?: string;
115
- sourceField?: string;
116
- targetField?: string;
117
- sourceSqlName?: string;
118
- targetSqlName?: string;
119
- onDelete?: ForeignKeyAction;
120
- onUpdate?: ForeignKeyAction;
121
- };
122
-
123
- export type RelationDefinition =
124
- | OneRelationDefinition
125
- | ManyRelationDefinition
126
- | ManyToManyRelationDefinition;
127
-
128
- export type TableOptions = {
129
- sqlName?: string;
130
- };
131
-
132
- export type TableShape = Record<string, ColumnBuilder | RelationDefinition>;
133
-
134
- export type TableDefinition = {
135
- kind: "table";
136
- sqlName?: string;
137
- columns: Record<string, ColumnDefinition>;
138
- relations: Record<string, RelationDefinition>;
139
- };
140
-
141
- export type SchemaDefinition = {
142
- kind: "schema";
143
- tables: Record<string, TableDefinition>;
144
- };
145
-
146
- export class ColumnBuilder {
147
- private definition: ColumnDefinition;
148
-
149
- public constructor(
150
- typeOrDefinition: ColumnType | ColumnDefinition,
151
- options: ColumnBuilderOptions = {},
152
- ) {
153
- if (typeof typeOrDefinition === "string") {
154
- this.definition = {
155
- kind: "column",
156
- type: typeOrDefinition,
157
- sqlName: options.sqlName,
158
- length: options.length,
159
- };
160
- return;
161
- }
162
-
163
- this.definition = {
164
- ...typeOrDefinition,
165
- };
166
- }
167
-
168
- private with(patch: Partial<ColumnDefinition>): ColumnBuilder {
169
- return new ColumnBuilder({
170
- ...this.definition,
171
- ...patch,
172
- });
173
- }
174
-
175
- public sql(name: string): ColumnBuilder {
176
- return this.with({ sqlName: name });
177
- }
178
-
179
- public notNull(): ColumnBuilder {
180
- return this.with({ notNull: true, nullable: false });
181
- }
182
-
183
- public nullable(): ColumnBuilder {
184
- return this.with({ nullable: true, notNull: false });
185
- }
186
-
187
- public primaryKey(options: PrimaryKeyOptions = {}): ColumnBuilder {
188
- return this.with({
189
- primaryKey: true,
190
- autoIncrement: options.autoIncrement ?? this.definition.autoIncrement,
191
- });
192
- }
193
-
194
- public unique(name?: string): ColumnBuilder {
195
- return this.with({ unique: name ? { name } : true });
196
- }
197
-
198
- public index(name?: string): ColumnBuilder {
199
- return this.with({ index: name ? { name } : true });
200
- }
201
-
202
- public default(value: unknown): ColumnBuilder {
203
- return this.with({ sqlDefault: value });
204
- }
205
-
206
- public defaultFn(fn: () => unknown): ColumnBuilder {
207
- return this.with({ runtimeDefaultFn: fn });
208
- }
209
-
210
- /**
211
- * Shorthand for date columns that should default to the current timestamp.
212
- * Sets SQL `DEFAULT CURRENT_TIMESTAMP` (via `nowDefault`) and
213
- * a runtime default of `new Date()`.
214
- */
215
- public defaultNow(): ColumnBuilder {
216
- return this.with({
217
- nowDefault: true,
218
- runtimeDefaultFn: () => new Date(),
219
- });
220
- }
221
-
222
- public references(
223
- table: string,
224
- column = "id",
225
- actions: Pick<ColumnReference, "onDelete" | "onUpdate"> = {},
226
- ): ColumnBuilder {
227
- return this.with({
228
- references: {
229
- table,
230
- column,
231
- onDelete: actions.onDelete,
232
- onUpdate: actions.onUpdate,
233
- },
234
- });
235
- }
236
-
237
- public toDefinition(): ColumnDefinition {
238
- return { ...this.definition };
239
- }
240
- }
241
-
242
- export function table(
243
- shape: TableShape,
244
- options: TableOptions = {},
245
- ): TableDefinition {
246
- const columns: Record<string, ColumnDefinition> = {};
247
- const relations: Record<string, RelationDefinition> = {};
248
-
249
- for (const [fieldName, value] of Object.entries(shape)) {
250
- if (value instanceof ColumnBuilder) {
251
- columns[fieldName] = value.toDefinition();
252
- continue;
253
- }
254
-
255
- if (value.kind === "relation") {
256
- relations[fieldName] = value;
257
- continue;
258
- }
259
-
260
- throw new Error(
261
- `Invalid table field '${fieldName}'. Use column builders or relation helpers.`,
262
- );
263
- }
264
-
265
- return {
266
- kind: "table",
267
- sqlName: options.sqlName,
268
- columns,
269
- relations,
270
- };
271
- }
272
-
273
- export function schema(
274
- tables: Record<string, TableDefinition>,
275
- ): SchemaDefinition {
276
- return {
277
- kind: "schema",
278
- tables,
279
- };
280
- }
281
-
282
- export function isSchemaDefinition(value: unknown): value is SchemaDefinition {
283
- if (typeof value !== "object" || value === null) {
284
- return false;
285
- }
286
- const candidate = value as { kind?: string; tables?: unknown };
287
- return candidate.kind === "schema" && typeof candidate.tables === "object";
288
- }
289
-
290
- // ---------------------------------------------------------------------------
291
- // Relation builder helpers
292
- // ---------------------------------------------------------------------------
293
-
294
- function buildOneRelation(
295
- targetTable: string,
296
- fieldOrOptions?: string | RelationOneOptions,
297
- options: RelationOneOptions = {},
298
- ): OneRelationDefinition {
299
- const field = typeof fieldOrOptions === "string" ? fieldOrOptions : undefined;
300
- const mergedOptions =
301
- typeof fieldOrOptions === "string" ? options : (fieldOrOptions ?? options);
302
-
303
- return {
304
- kind: "relation",
305
- relation: "one",
306
- targetTable,
307
- field: mergedOptions.field ?? field,
308
- referenceField: mergedOptions.referenceField,
309
- fkType: mergedOptions.fkType,
310
- sqlName: mergedOptions.sqlName,
311
- notNull: mergedOptions.notNull,
312
- nullable: mergedOptions.nullable,
313
- onDelete: mergedOptions.onDelete,
314
- onUpdate: mergedOptions.onUpdate,
315
- };
316
- }
317
-
318
- function buildManyRelation(
319
- targetTable: string,
320
- fieldOrOptions?: string | RelationManyOptions,
321
- options: RelationManyOptions = {},
322
- ): ManyRelationDefinition {
323
- const field = typeof fieldOrOptions === "string" ? fieldOrOptions : undefined;
324
- const mergedOptions =
325
- typeof fieldOrOptions === "string" ? options : (fieldOrOptions ?? options);
326
-
327
- return {
328
- kind: "relation",
329
- relation: "many",
330
- targetTable,
331
- field: mergedOptions.field ?? field,
332
- referenceField: mergedOptions.referenceField,
333
- fkType: mergedOptions.fkType,
334
- sqlName: mergedOptions.sqlName,
335
- notNull: mergedOptions.notNull,
336
- nullable: mergedOptions.nullable,
337
- onDelete: mergedOptions.onDelete,
338
- onUpdate: mergedOptions.onUpdate,
339
- };
340
- }
341
-
342
- function buildManyToManyRelation(
343
- targetTable: string,
344
- options: RelationManyToManyOptions = {},
345
- ): ManyToManyRelationDefinition {
346
- return {
347
- kind: "relation",
348
- relation: "manyToMany",
349
- targetTable,
350
- referenceField: options.referenceField,
351
- targetReferenceField: options.targetReferenceField,
352
- junctionTable: options.junctionTable,
353
- sourceField: options.sourceField,
354
- targetField: options.targetField,
355
- sourceSqlName: options.sourceSqlName,
356
- targetSqlName: options.targetSqlName,
357
- onDelete: options.onDelete,
358
- onUpdate: options.onUpdate,
359
- };
360
- }
361
-
362
- // ---------------------------------------------------------------------------
363
- // Public API
364
- // ---------------------------------------------------------------------------
365
-
366
- export const v = {
367
- int: (options: ColumnBuilderOptions = {}) =>
368
- new ColumnBuilder("int", options),
369
- number: (options: ColumnBuilderOptions = {}) =>
370
- new ColumnBuilder("int", options),
371
- string: (options: ColumnBuilderOptions = {}) =>
372
- new ColumnBuilder("string", options),
373
- boolean: (options: ColumnBuilderOptions = {}) =>
374
- new ColumnBuilder("boolean", options),
375
- date: (options: ColumnBuilderOptions = {}) =>
376
- new ColumnBuilder("date", options),
377
-
378
- /**
379
- * UUID primary key column.
380
- * Stored as a string, marked as primary key and auto-populated with
381
- * `crypto.randomUUID()` at runtime (no SQL sequence).
382
- */
383
- uuid: (options: ColumnBuilderOptions = {}) =>
384
- new ColumnBuilder({
385
- kind: "column",
386
- type: "string",
387
- sqlName: options.sqlName,
388
- length: options.length ?? 36,
389
- primaryKey: true,
390
- notNull: true,
391
- nullable: false,
392
- uuidPrimaryKey: true,
393
- runtimeDefaultFn: () => crypto.randomUUID(),
394
- }),
395
-
396
- one: (
397
- targetTable: string,
398
- fieldOrOptions?: string | RelationOneOptions,
399
- options: RelationOneOptions = {},
400
- ): OneRelationDefinition =>
401
- buildOneRelation(targetTable, fieldOrOptions, options),
402
-
403
- many: (
404
- targetTable: string,
405
- fieldOrOptions?: string | RelationManyOptions,
406
- options: RelationManyOptions = {},
407
- ): ManyRelationDefinition =>
408
- buildManyRelation(targetTable, fieldOrOptions, options),
409
-
410
- manyToMany: (
411
- targetTable: string,
412
- options: RelationManyToManyOptions = {},
413
- ): ManyToManyRelationDefinition =>
414
- buildManyToManyRelation(targetTable, options),
415
- };
1
+ export type ColumnType = "int" | "string" | "boolean" | "date";
2
+
3
+ export type ColumnBuilderOptions = {
4
+ sqlName?: string;
5
+ length?: number;
6
+ };
7
+
8
+ export type PrimaryKeyOptions = {
9
+ autoIncrement?: boolean;
10
+ };
11
+
12
+ export type ForeignKeyAction =
13
+ | "cascade"
14
+ | "set null"
15
+ | "set default"
16
+ | "restrict"
17
+ | "no action";
18
+
19
+ export type RelationOneOptions = {
20
+ referenceField?: string;
21
+ field?: string;
22
+ fkType?: ColumnType;
23
+ sqlName?: string;
24
+ notNull?: boolean;
25
+ nullable?: boolean;
26
+ onDelete?: ForeignKeyAction;
27
+ onUpdate?: ForeignKeyAction;
28
+ };
29
+
30
+ export type RelationManyOptions = {
31
+ referenceField?: string;
32
+ field?: string;
33
+ fkType?: ColumnType;
34
+ sqlName?: string;
35
+ notNull?: boolean;
36
+ nullable?: boolean;
37
+ onDelete?: ForeignKeyAction;
38
+ onUpdate?: ForeignKeyAction;
39
+ };
40
+
41
+ export type RelationManyToManyOptions = {
42
+ referenceField?: string;
43
+ targetReferenceField?: string;
44
+ junctionTable?: string;
45
+ sourceField?: string;
46
+ targetField?: string;
47
+ sourceSqlName?: string;
48
+ targetSqlName?: string;
49
+ onDelete?: ForeignKeyAction;
50
+ onUpdate?: ForeignKeyAction;
51
+ };
52
+
53
+ export type ColumnReference = {
54
+ table: string;
55
+ column: string;
56
+ onDelete?: ForeignKeyAction;
57
+ onUpdate?: ForeignKeyAction;
58
+ };
59
+
60
+ export type ColumnDefinition = {
61
+ kind: "column";
62
+ type: ColumnType;
63
+ sqlName?: string;
64
+ length?: number;
65
+ notNull?: boolean;
66
+ nullable?: boolean;
67
+ primaryKey?: boolean;
68
+ autoIncrement?: boolean;
69
+ /** When true the column stores a UUID generated at runtime (string type). */
70
+ uuidPrimaryKey?: boolean;
71
+ unique?: true | { name?: string };
72
+ index?: true | { name?: string };
73
+ sqlDefault?: unknown;
74
+ /** When true, the SQL column gets DEFAULT CURRENT_TIMESTAMP and runtime inserts use `new Date()`. */
75
+ nowDefault?: boolean;
76
+ runtimeDefaultFn?: () => unknown;
77
+ references?: ColumnReference;
78
+ };
79
+
80
+ export type OneRelationDefinition = {
81
+ kind: "relation";
82
+ relation: "one";
83
+ targetTable: string;
84
+ field?: string;
85
+ referenceField?: string;
86
+ fkType?: ColumnType;
87
+ sqlName?: string;
88
+ notNull?: boolean;
89
+ nullable?: boolean;
90
+ onDelete?: ForeignKeyAction;
91
+ onUpdate?: ForeignKeyAction;
92
+ };
93
+
94
+ export type ManyRelationDefinition = {
95
+ kind: "relation";
96
+ relation: "many";
97
+ targetTable: string;
98
+ field?: string;
99
+ referenceField?: string;
100
+ fkType?: ColumnType;
101
+ sqlName?: string;
102
+ notNull?: boolean;
103
+ nullable?: boolean;
104
+ onDelete?: ForeignKeyAction;
105
+ onUpdate?: ForeignKeyAction;
106
+ };
107
+
108
+ export type ManyToManyRelationDefinition = {
109
+ kind: "relation";
110
+ relation: "manyToMany";
111
+ targetTable: string;
112
+ referenceField?: string;
113
+ targetReferenceField?: string;
114
+ junctionTable?: string;
115
+ sourceField?: string;
116
+ targetField?: string;
117
+ sourceSqlName?: string;
118
+ targetSqlName?: string;
119
+ onDelete?: ForeignKeyAction;
120
+ onUpdate?: ForeignKeyAction;
121
+ };
122
+
123
+ export type RelationDefinition =
124
+ | OneRelationDefinition
125
+ | ManyRelationDefinition
126
+ | ManyToManyRelationDefinition;
127
+
128
+ export type TableOptions = {
129
+ sqlName?: string;
130
+ };
131
+
132
+ export type TableShape = Record<string, ColumnBuilder | RelationDefinition>;
133
+
134
+ export type TableDefinition = {
135
+ kind: "table";
136
+ sqlName?: string;
137
+ columns: Record<string, ColumnDefinition>;
138
+ relations: Record<string, RelationDefinition>;
139
+ };
140
+
141
+ export type SchemaDefinition = {
142
+ kind: "schema";
143
+ tables: Record<string, TableDefinition>;
144
+ };
145
+
146
+ export class ColumnBuilder {
147
+ private definition: ColumnDefinition;
148
+
149
+ public constructor(
150
+ typeOrDefinition: ColumnType | ColumnDefinition,
151
+ options: ColumnBuilderOptions = {},
152
+ ) {
153
+ if (typeof typeOrDefinition === "string") {
154
+ this.definition = {
155
+ kind: "column",
156
+ type: typeOrDefinition,
157
+ sqlName: options.sqlName,
158
+ length: options.length,
159
+ };
160
+ return;
161
+ }
162
+
163
+ this.definition = {
164
+ ...typeOrDefinition,
165
+ };
166
+ }
167
+
168
+ private with(patch: Partial<ColumnDefinition>): ColumnBuilder {
169
+ return new ColumnBuilder({
170
+ ...this.definition,
171
+ ...patch,
172
+ });
173
+ }
174
+
175
+ public sql(name: string): ColumnBuilder {
176
+ return this.with({ sqlName: name });
177
+ }
178
+
179
+ public notNull(): ColumnBuilder {
180
+ return this.with({ notNull: true, nullable: false });
181
+ }
182
+
183
+ public nullable(): ColumnBuilder {
184
+ return this.with({ nullable: true, notNull: false });
185
+ }
186
+
187
+ public primaryKey(options: PrimaryKeyOptions = {}): ColumnBuilder {
188
+ return this.with({
189
+ primaryKey: true,
190
+ autoIncrement: options.autoIncrement ?? this.definition.autoIncrement,
191
+ });
192
+ }
193
+
194
+ public unique(name?: string): ColumnBuilder {
195
+ return this.with({ unique: name ? { name } : true });
196
+ }
197
+
198
+ public index(name?: string): ColumnBuilder {
199
+ return this.with({ index: name ? { name } : true });
200
+ }
201
+
202
+ public default(value: unknown): ColumnBuilder {
203
+ return this.with({ sqlDefault: value });
204
+ }
205
+
206
+ public defaultFn(fn: () => unknown): ColumnBuilder {
207
+ return this.with({ runtimeDefaultFn: fn });
208
+ }
209
+
210
+ /**
211
+ * Shorthand for date columns that should default to the current timestamp.
212
+ * Sets SQL `DEFAULT CURRENT_TIMESTAMP` (via `nowDefault`) and
213
+ * a runtime default of `new Date()`.
214
+ */
215
+ public defaultNow(): ColumnBuilder {
216
+ return this.with({
217
+ nowDefault: true,
218
+ runtimeDefaultFn: () => new Date(),
219
+ });
220
+ }
221
+
222
+ public references(
223
+ table: string,
224
+ column = "id",
225
+ actions: Pick<ColumnReference, "onDelete" | "onUpdate"> = {},
226
+ ): ColumnBuilder {
227
+ return this.with({
228
+ references: {
229
+ table,
230
+ column,
231
+ onDelete: actions.onDelete,
232
+ onUpdate: actions.onUpdate,
233
+ },
234
+ });
235
+ }
236
+
237
+ public toDefinition(): ColumnDefinition {
238
+ return { ...this.definition };
239
+ }
240
+ }
241
+
242
+ export function table(
243
+ shape: TableShape,
244
+ options: TableOptions = {},
245
+ ): TableDefinition {
246
+ const columns: Record<string, ColumnDefinition> = {};
247
+ const relations: Record<string, RelationDefinition> = {};
248
+
249
+ for (const [fieldName, value] of Object.entries(shape)) {
250
+ if (value instanceof ColumnBuilder) {
251
+ columns[fieldName] = value.toDefinition();
252
+ continue;
253
+ }
254
+
255
+ if (value.kind === "relation") {
256
+ relations[fieldName] = value;
257
+ continue;
258
+ }
259
+
260
+ throw new Error(
261
+ `Invalid table field '${fieldName}'. Use column builders or relation helpers.`,
262
+ );
263
+ }
264
+
265
+ return {
266
+ kind: "table",
267
+ sqlName: options.sqlName,
268
+ columns,
269
+ relations,
270
+ };
271
+ }
272
+
273
+ export function schema(
274
+ tables: Record<string, TableDefinition>,
275
+ ): SchemaDefinition {
276
+ return {
277
+ kind: "schema",
278
+ tables,
279
+ };
280
+ }
281
+
282
+ export function isSchemaDefinition(value: unknown): value is SchemaDefinition {
283
+ if (typeof value !== "object" || value === null) {
284
+ return false;
285
+ }
286
+ const candidate = value as { kind?: string; tables?: unknown };
287
+ return candidate.kind === "schema" && typeof candidate.tables === "object";
288
+ }
289
+
290
+ // ---------------------------------------------------------------------------
291
+ // Relation builder helpers
292
+ // ---------------------------------------------------------------------------
293
+
294
+ function buildOneRelation(
295
+ targetTable: string,
296
+ fieldOrOptions?: string | RelationOneOptions,
297
+ options: RelationOneOptions = {},
298
+ ): OneRelationDefinition {
299
+ const field = typeof fieldOrOptions === "string" ? fieldOrOptions : undefined;
300
+ const mergedOptions =
301
+ typeof fieldOrOptions === "string" ? options : (fieldOrOptions ?? options);
302
+
303
+ return {
304
+ kind: "relation",
305
+ relation: "one",
306
+ targetTable,
307
+ field: mergedOptions.field ?? field,
308
+ referenceField: mergedOptions.referenceField,
309
+ fkType: mergedOptions.fkType,
310
+ sqlName: mergedOptions.sqlName,
311
+ notNull: mergedOptions.notNull,
312
+ nullable: mergedOptions.nullable,
313
+ onDelete: mergedOptions.onDelete,
314
+ onUpdate: mergedOptions.onUpdate,
315
+ };
316
+ }
317
+
318
+ function buildManyRelation(
319
+ targetTable: string,
320
+ fieldOrOptions?: string | RelationManyOptions,
321
+ options: RelationManyOptions = {},
322
+ ): ManyRelationDefinition {
323
+ const field = typeof fieldOrOptions === "string" ? fieldOrOptions : undefined;
324
+ const mergedOptions =
325
+ typeof fieldOrOptions === "string" ? options : (fieldOrOptions ?? options);
326
+
327
+ return {
328
+ kind: "relation",
329
+ relation: "many",
330
+ targetTable,
331
+ field: mergedOptions.field ?? field,
332
+ referenceField: mergedOptions.referenceField,
333
+ fkType: mergedOptions.fkType,
334
+ sqlName: mergedOptions.sqlName,
335
+ notNull: mergedOptions.notNull,
336
+ nullable: mergedOptions.nullable,
337
+ onDelete: mergedOptions.onDelete,
338
+ onUpdate: mergedOptions.onUpdate,
339
+ };
340
+ }
341
+
342
+ function buildManyToManyRelation(
343
+ targetTable: string,
344
+ options: RelationManyToManyOptions = {},
345
+ ): ManyToManyRelationDefinition {
346
+ return {
347
+ kind: "relation",
348
+ relation: "manyToMany",
349
+ targetTable,
350
+ referenceField: options.referenceField,
351
+ targetReferenceField: options.targetReferenceField,
352
+ junctionTable: options.junctionTable,
353
+ sourceField: options.sourceField,
354
+ targetField: options.targetField,
355
+ sourceSqlName: options.sourceSqlName,
356
+ targetSqlName: options.targetSqlName,
357
+ onDelete: options.onDelete,
358
+ onUpdate: options.onUpdate,
359
+ };
360
+ }
361
+
362
+ // ---------------------------------------------------------------------------
363
+ // Public API
364
+ // ---------------------------------------------------------------------------
365
+
366
+ export const v = {
367
+ int: (options: ColumnBuilderOptions = {}) =>
368
+ new ColumnBuilder("int", options),
369
+ number: (options: ColumnBuilderOptions = {}) =>
370
+ new ColumnBuilder("int", options),
371
+ string: (options: ColumnBuilderOptions = {}) =>
372
+ new ColumnBuilder("string", options),
373
+ boolean: (options: ColumnBuilderOptions = {}) =>
374
+ new ColumnBuilder("boolean", options),
375
+ date: (options: ColumnBuilderOptions = {}) =>
376
+ new ColumnBuilder("date", options),
377
+
378
+ /**
379
+ * UUID primary key column.
380
+ * Stored as a string, marked as primary key and auto-populated with
381
+ * `crypto.randomUUID()` at runtime (no SQL sequence).
382
+ */
383
+ uuid: (options: ColumnBuilderOptions = {}) =>
384
+ new ColumnBuilder({
385
+ kind: "column",
386
+ type: "string",
387
+ sqlName: options.sqlName,
388
+ length: options.length ?? 36,
389
+ primaryKey: true,
390
+ notNull: true,
391
+ nullable: false,
392
+ uuidPrimaryKey: true,
393
+ runtimeDefaultFn: () => crypto.randomUUID(),
394
+ }),
395
+
396
+ one: (
397
+ targetTable: string,
398
+ fieldOrOptions?: string | RelationOneOptions,
399
+ options: RelationOneOptions = {},
400
+ ): OneRelationDefinition =>
401
+ buildOneRelation(targetTable, fieldOrOptions, options),
402
+
403
+ many: (
404
+ targetTable: string,
405
+ fieldOrOptions?: string | RelationManyOptions,
406
+ options: RelationManyOptions = {},
407
+ ): ManyRelationDefinition =>
408
+ buildManyRelation(targetTable, fieldOrOptions, options),
409
+
410
+ manyToMany: (
411
+ targetTable: string,
412
+ options: RelationManyToManyOptions = {},
413
+ ): ManyToManyRelationDefinition =>
414
+ buildManyToManyRelation(targetTable, options),
415
+ };