@prisma-next/sql-contract 0.3.0-dev.13 → 0.3.0-dev.131

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 (60) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +84 -10
  3. package/dist/factories.d.mts +48 -0
  4. package/dist/factories.d.mts.map +1 -0
  5. package/dist/factories.mjs +84 -0
  6. package/dist/factories.mjs.map +1 -0
  7. package/dist/pack-types.d.mts +13 -0
  8. package/dist/pack-types.d.mts.map +1 -0
  9. package/dist/pack-types.mjs +1 -0
  10. package/dist/types-CB821Pqa.d.mts +197 -0
  11. package/dist/types-CB821Pqa.d.mts.map +1 -0
  12. package/dist/types-DRR5stkj.mjs +13 -0
  13. package/dist/types-DRR5stkj.mjs.map +1 -0
  14. package/dist/types.d.mts +2 -0
  15. package/dist/types.mjs +3 -0
  16. package/dist/validate.d.mts +11 -0
  17. package/dist/validate.d.mts.map +1 -0
  18. package/dist/validate.mjs +437 -0
  19. package/dist/validate.mjs.map +1 -0
  20. package/dist/validators-CNxeypbZ.mjs +234 -0
  21. package/dist/validators-CNxeypbZ.mjs.map +1 -0
  22. package/dist/validators.d.mts +71 -0
  23. package/dist/validators.d.mts.map +1 -0
  24. package/dist/validators.mjs +3 -0
  25. package/package.json +20 -24
  26. package/src/construct.ts +181 -0
  27. package/src/exports/types.ts +21 -0
  28. package/src/exports/validate.ts +6 -0
  29. package/src/exports/validators.ts +1 -1
  30. package/src/factories.ts +41 -8
  31. package/src/index.ts +1 -0
  32. package/src/types.ts +176 -9
  33. package/src/validate.ts +560 -0
  34. package/src/validators.ts +184 -18
  35. package/dist/exports/factories.d.ts +0 -2
  36. package/dist/exports/factories.d.ts.map +0 -1
  37. package/dist/exports/factories.js +0 -83
  38. package/dist/exports/factories.js.map +0 -1
  39. package/dist/exports/pack-types.d.ts +0 -2
  40. package/dist/exports/pack-types.d.ts.map +0 -1
  41. package/dist/exports/pack-types.js +0 -1
  42. package/dist/exports/pack-types.js.map +0 -1
  43. package/dist/exports/types.d.ts +0 -2
  44. package/dist/exports/types.d.ts.map +0 -1
  45. package/dist/exports/types.js +0 -1
  46. package/dist/exports/types.js.map +0 -1
  47. package/dist/exports/validators.d.ts +0 -2
  48. package/dist/exports/validators.d.ts.map +0 -1
  49. package/dist/exports/validators.js +0 -96
  50. package/dist/exports/validators.js.map +0 -1
  51. package/dist/factories.d.ts +0 -38
  52. package/dist/factories.d.ts.map +0 -1
  53. package/dist/index.d.ts +0 -4
  54. package/dist/index.d.ts.map +0 -1
  55. package/dist/pack-types.d.ts +0 -10
  56. package/dist/pack-types.d.ts.map +0 -1
  57. package/dist/types.d.ts +0 -68
  58. package/dist/types.d.ts.map +0 -1
  59. package/dist/validators.d.ts +0 -35
  60. package/dist/validators.d.ts.map +0 -1
@@ -0,0 +1,437 @@
1
+ import { r as applyFkDefaults } from "./types-DRR5stkj.mjs";
2
+ import { d as validateStorageSemantics, l as validateSqlContract } from "./validators-CNxeypbZ.mjs";
3
+ import { isTaggedBigInt, isTaggedRaw } from "@prisma-next/contract/types";
4
+ import { validateContractDomain } from "@prisma-next/contract/validate-domain";
5
+
6
+ //#region src/construct.ts
7
+ function computeDefaultMappings(models) {
8
+ const modelToTable = {};
9
+ const tableToModel = {};
10
+ const fieldToColumn = {};
11
+ const columnToField = {};
12
+ for (const [modelName, model] of Object.entries(models)) {
13
+ const tableName = model.storage.table;
14
+ modelToTable[modelName] = tableName;
15
+ tableToModel[tableName] = modelName;
16
+ const modelFieldToColumn = {};
17
+ for (const [fieldName, field] of Object.entries(model.fields)) {
18
+ const columnName = field.column;
19
+ modelFieldToColumn[fieldName] = columnName;
20
+ if (!columnToField[tableName]) columnToField[tableName] = {};
21
+ columnToField[tableName][columnName] = fieldName;
22
+ }
23
+ fieldToColumn[modelName] = modelFieldToColumn;
24
+ }
25
+ return {
26
+ modelToTable,
27
+ tableToModel,
28
+ fieldToColumn,
29
+ columnToField
30
+ };
31
+ }
32
+ function assertInverseModelMappings(modelToTable, tableToModel) {
33
+ for (const [model, table] of Object.entries(modelToTable)) if (tableToModel[table] !== model) throw new Error(`Mappings override mismatch: modelToTable.${model}="${table}" is not mirrored in tableToModel`);
34
+ for (const [table, model] of Object.entries(tableToModel)) if (modelToTable[model] !== table) throw new Error(`Mappings override mismatch: tableToModel.${table}="${model}" is not mirrored in modelToTable`);
35
+ }
36
+ function assertInverseFieldMappings(fieldToColumn, columnToField, modelToTable, tableToModel) {
37
+ for (const [model, fields] of Object.entries(fieldToColumn)) {
38
+ const table = modelToTable[model];
39
+ if (!table) throw new Error(`Mappings override mismatch: fieldToColumn references unknown model "${model}"`);
40
+ const reverseFields = columnToField[table];
41
+ if (!reverseFields) throw new Error(`Mappings override mismatch: columnToField is missing table "${table}" for model "${model}"`);
42
+ for (const [field, column] of Object.entries(fields)) if (reverseFields[column] !== field) throw new Error(`Mappings override mismatch: fieldToColumn.${model}.${field}="${column}" is not mirrored in columnToField.${table}`);
43
+ }
44
+ for (const [table, columns] of Object.entries(columnToField)) {
45
+ const model = tableToModel[table];
46
+ if (!model) throw new Error(`Mappings override mismatch: columnToField references unknown table "${table}"`);
47
+ const forwardFields = fieldToColumn[model];
48
+ if (!forwardFields) throw new Error(`Mappings override mismatch: fieldToColumn is missing model "${model}" for table "${table}"`);
49
+ for (const [column, field] of Object.entries(columns)) if (forwardFields[field] !== column) throw new Error(`Mappings override mismatch: columnToField.${table}.${column}="${field}" is not mirrored in fieldToColumn.${model}`);
50
+ }
51
+ }
52
+ function mergeMappings(defaults, existingMappings) {
53
+ const hasModelToTable = existingMappings?.modelToTable !== void 0;
54
+ const hasTableToModel = existingMappings?.tableToModel !== void 0;
55
+ if (hasModelToTable !== hasTableToModel) throw new Error("Mappings override mismatch: modelToTable and tableToModel must be provided together");
56
+ const hasFieldToColumn = existingMappings?.fieldToColumn !== void 0;
57
+ const hasColumnToField = existingMappings?.columnToField !== void 0;
58
+ if (hasFieldToColumn !== hasColumnToField) throw new Error("Mappings override mismatch: fieldToColumn and columnToField must be provided together");
59
+ const modelToTable = hasModelToTable ? existingMappings?.modelToTable ?? {} : defaults.modelToTable;
60
+ const tableToModel = hasTableToModel ? existingMappings?.tableToModel ?? {} : defaults.tableToModel;
61
+ assertInverseModelMappings(modelToTable, tableToModel);
62
+ const fieldToColumn = hasFieldToColumn ? existingMappings?.fieldToColumn ?? {} : defaults.fieldToColumn;
63
+ const columnToField = hasColumnToField ? existingMappings?.columnToField ?? {} : defaults.columnToField;
64
+ assertInverseFieldMappings(fieldToColumn, columnToField, modelToTable, tableToModel);
65
+ return {
66
+ modelToTable,
67
+ tableToModel,
68
+ fieldToColumn,
69
+ columnToField
70
+ };
71
+ }
72
+ function stripGenerated(obj) {
73
+ const { _generated: _, ...rest } = obj;
74
+ return rest;
75
+ }
76
+ function constructContract(input) {
77
+ const existingMappings = input.mappings;
78
+ const mappings = mergeMappings(computeDefaultMappings(input.models), existingMappings);
79
+ const stripped = stripGenerated(input);
80
+ return {
81
+ ...stripped,
82
+ mappings,
83
+ roots: stripped.roots
84
+ };
85
+ }
86
+
87
+ //#endregion
88
+ //#region src/validate.ts
89
+ function extractDomainShape(contract) {
90
+ return {
91
+ roots: contract.roots,
92
+ models: contract.models
93
+ };
94
+ }
95
+ function validateContractLogic(contract) {
96
+ const tableNames = new Set(Object.keys(contract.storage.tables));
97
+ for (const [tableName, table] of Object.entries(contract.storage.tables)) {
98
+ const columnNames = new Set(Object.keys(table.columns));
99
+ if (table.primaryKey) {
100
+ for (const colName of table.primaryKey.columns) if (!columnNames.has(colName)) throw new Error(`Table "${tableName}" primaryKey references non-existent column "${colName}"`);
101
+ }
102
+ for (const unique of table.uniques) for (const colName of unique.columns) if (!columnNames.has(colName)) throw new Error(`Table "${tableName}" unique constraint references non-existent column "${colName}"`);
103
+ for (const index of table.indexes) for (const colName of index.columns) if (!columnNames.has(colName)) throw new Error(`Table "${tableName}" index references non-existent column "${colName}"`);
104
+ for (const [colName, column] of Object.entries(table.columns)) if (!column.nullable && column.default?.kind === "literal" && column.default.value === null) throw new Error(`Table "${tableName}" column "${colName}" is NOT NULL but has a literal null default`);
105
+ for (const fk of table.foreignKeys) {
106
+ for (const colName of fk.columns) if (!columnNames.has(colName)) throw new Error(`Table "${tableName}" foreignKey references non-existent column "${colName}"`);
107
+ if (!tableNames.has(fk.references.table)) throw new Error(`Table "${tableName}" foreignKey references non-existent table "${fk.references.table}"`);
108
+ const referencedTable = contract.storage.tables[fk.references.table];
109
+ const referencedColumnNames = new Set(Object.keys(referencedTable.columns));
110
+ for (const colName of fk.references.columns) if (!referencedColumnNames.has(colName)) throw new Error(`Table "${tableName}" foreignKey references non-existent column "${colName}" in table "${fk.references.table}"`);
111
+ if (fk.columns.length !== fk.references.columns.length) throw new Error(`Table "${tableName}" foreignKey column count (${fk.columns.length}) does not match referenced column count (${fk.references.columns.length})`);
112
+ }
113
+ }
114
+ }
115
+ const BIGINT_NATIVE_TYPES = new Set(["bigint", "int8"]);
116
+ function isBigIntColumn(column) {
117
+ const nativeType = column.nativeType?.toLowerCase() ?? "";
118
+ if (BIGINT_NATIVE_TYPES.has(nativeType)) return true;
119
+ const codecId = column.codecId?.toLowerCase() ?? "";
120
+ return codecId.includes("int8") || codecId.includes("bigint");
121
+ }
122
+ function decodeDefaultLiteralValue(value, column, tableName, columnName) {
123
+ if (value instanceof Date) return value;
124
+ if (isTaggedRaw(value)) return value.value;
125
+ if (isTaggedBigInt(value)) {
126
+ if (!isBigIntColumn(column)) return value;
127
+ try {
128
+ return BigInt(value.value);
129
+ } catch {
130
+ throw new Error(`Invalid tagged bigint for default value on "${tableName}.${columnName}": "${value.value}" is not a valid integer`);
131
+ }
132
+ }
133
+ return value;
134
+ }
135
+ function decodeContractDefaults(contract) {
136
+ const tables = contract.storage.tables;
137
+ let tablesChanged = false;
138
+ const decodedTables = {};
139
+ for (const [tableName, table] of Object.entries(tables)) {
140
+ let columnsChanged = false;
141
+ const decodedColumns = {};
142
+ for (const [columnName, column] of Object.entries(table.columns)) {
143
+ if (column.default?.kind === "literal") {
144
+ const decodedValue = decodeDefaultLiteralValue(column.default.value, column, tableName, columnName);
145
+ if (decodedValue !== column.default.value) {
146
+ columnsChanged = true;
147
+ decodedColumns[columnName] = {
148
+ ...column,
149
+ default: {
150
+ kind: "literal",
151
+ value: decodedValue
152
+ }
153
+ };
154
+ continue;
155
+ }
156
+ }
157
+ decodedColumns[columnName] = column;
158
+ }
159
+ if (columnsChanged) {
160
+ tablesChanged = true;
161
+ decodedTables[tableName] = {
162
+ ...table,
163
+ columns: decodedColumns
164
+ };
165
+ } else decodedTables[tableName] = table;
166
+ }
167
+ if (!tablesChanged) return contract;
168
+ return {
169
+ ...contract,
170
+ storage: {
171
+ ...contract.storage,
172
+ tables: decodedTables
173
+ }
174
+ };
175
+ }
176
+ function normalizeStorage(contractObj) {
177
+ const normalizedStorage = contractObj["storage"];
178
+ if (!normalizedStorage || typeof normalizedStorage !== "object") return normalizedStorage;
179
+ const storage = normalizedStorage;
180
+ const tables = storage["tables"];
181
+ if (!tables) return storage;
182
+ const normalizedTables = {};
183
+ for (const [tableName, table] of Object.entries(tables)) {
184
+ const tableObj = table;
185
+ const columns = tableObj["columns"];
186
+ if (columns) {
187
+ const normalizedColumns = {};
188
+ for (const [columnName, column] of Object.entries(columns)) {
189
+ const columnObj = column;
190
+ normalizedColumns[columnName] = {
191
+ ...columnObj,
192
+ nullable: columnObj["nullable"] ?? false
193
+ };
194
+ }
195
+ const normalizedForeignKeys = (tableObj["foreignKeys"] ?? []).map((fk) => ({
196
+ ...fk,
197
+ ...applyFkDefaults({
198
+ constraint: typeof fk["constraint"] === "boolean" ? fk["constraint"] : void 0,
199
+ index: typeof fk["index"] === "boolean" ? fk["index"] : void 0
200
+ })
201
+ }));
202
+ normalizedTables[tableName] = {
203
+ ...tableObj,
204
+ columns: normalizedColumns,
205
+ uniques: tableObj["uniques"] ?? [],
206
+ indexes: tableObj["indexes"] ?? [],
207
+ foreignKeys: normalizedForeignKeys
208
+ };
209
+ } else normalizedTables[tableName] = tableObj;
210
+ }
211
+ return {
212
+ ...storage,
213
+ tables: normalizedTables
214
+ };
215
+ }
216
+ function detectFormat(models) {
217
+ for (const model of Object.values(models)) {
218
+ const fields = model["fields"];
219
+ if (!fields) continue;
220
+ for (const field of Object.values(fields)) {
221
+ if ("column" in field) return "old";
222
+ if ("codecId" in field) return "new";
223
+ }
224
+ }
225
+ return "old";
226
+ }
227
+ function buildColumnToFieldMap(fields, modelName) {
228
+ const map = {};
229
+ for (const [fieldName, field] of Object.entries(fields)) {
230
+ const col = field["column"];
231
+ if (!col) continue;
232
+ if (Object.hasOwn(map, col)) throw new Error(`Model "${modelName}" has duplicate column mapping: fields "${map[col]}" and "${fieldName}" both map to column "${col}"`);
233
+ map[col] = fieldName;
234
+ }
235
+ return map;
236
+ }
237
+ function enrichOldFormatModels(models, storageObj, topRelations) {
238
+ const roots = {};
239
+ const tableToModel = {};
240
+ for (const [modelName, model] of Object.entries(models)) {
241
+ const tableName = model["storage"]?.["table"];
242
+ if (tableName) {
243
+ if (!model["owner"]) roots[modelName] = modelName;
244
+ tableToModel[tableName] = modelName;
245
+ }
246
+ }
247
+ const enrichedModels = {};
248
+ for (const [modelName, model] of Object.entries(models)) {
249
+ const fields = model["fields"] ?? {};
250
+ const modelStorage = model["storage"];
251
+ const tableName = modelStorage?.["table"];
252
+ const storageColumns = (tableName ? storageObj.tables[tableName] : void 0)?.["columns"] ?? {};
253
+ const enrichedFields = {};
254
+ const modelStorageFields = {};
255
+ const hasStorageColumns = Object.keys(storageColumns).length > 0;
256
+ for (const [fieldName, field] of Object.entries(fields)) {
257
+ const colName = field["column"];
258
+ const storageCol = storageColumns[colName];
259
+ if (!storageCol && hasStorageColumns && colName) throw new Error(`Model "${modelName}" field "${fieldName}" references non-existent column "${colName}" in table "${tableName}"`);
260
+ enrichedFields[fieldName] = {
261
+ ...field,
262
+ nullable: storageCol?.["nullable"] ?? false,
263
+ codecId: storageCol?.["codecId"] ?? ""
264
+ };
265
+ modelStorageFields[fieldName] = { column: colName };
266
+ }
267
+ const enrichedStorage = {
268
+ ...modelStorage ?? {},
269
+ fields: modelStorageFields
270
+ };
271
+ enrichedModels[modelName] = {
272
+ ...model,
273
+ fields: enrichedFields,
274
+ storage: enrichedStorage,
275
+ relations: model["relations"] ?? {}
276
+ };
277
+ }
278
+ for (const [tableName, tableRels] of Object.entries(topRelations)) {
279
+ const modelName = tableToModel[tableName];
280
+ if (!modelName) continue;
281
+ const existingModel = enrichedModels[modelName];
282
+ if (!existingModel) continue;
283
+ const existingRels = existingModel["relations"] ?? {};
284
+ const targetColumnToField = {};
285
+ const modelRelations = { ...existingRels };
286
+ for (const [relName, rel] of Object.entries(tableRels)) {
287
+ const on = rel["on"];
288
+ const parentCols = on?.["parentCols"] ?? [];
289
+ const childCols = on?.["childCols"] ?? [];
290
+ const toModel = rel["to"];
291
+ const sourceColToField = buildColumnToFieldMap(existingModel["fields"] ?? {}, modelName);
292
+ if (!targetColumnToField[toModel]) {
293
+ const targetModelObj = enrichedModels[toModel];
294
+ if (targetModelObj) targetColumnToField[toModel] = buildColumnToFieldMap(targetModelObj["fields"] ?? {}, toModel);
295
+ else targetColumnToField[toModel] = {};
296
+ }
297
+ const targetColToField = targetColumnToField[toModel] ?? {};
298
+ const localFields = parentCols.map((c) => sourceColToField[c] ?? c);
299
+ const targetFields = childCols.map((c) => targetColToField[c] ?? c);
300
+ modelRelations[relName] = {
301
+ to: toModel,
302
+ cardinality: rel["cardinality"],
303
+ on: {
304
+ localFields,
305
+ targetFields
306
+ }
307
+ };
308
+ }
309
+ enrichedModels[modelName] = {
310
+ ...existingModel,
311
+ relations: modelRelations
312
+ };
313
+ }
314
+ return {
315
+ enrichedModels,
316
+ roots
317
+ };
318
+ }
319
+ function enrichNewFormatModels(models) {
320
+ const enrichedModels = {};
321
+ const topRelations = {};
322
+ const modelToTable = {};
323
+ for (const [modelName, model] of Object.entries(models)) {
324
+ const tableName = model["storage"]?.["table"];
325
+ if (tableName) modelToTable[modelName] = tableName;
326
+ }
327
+ for (const [modelName, model] of Object.entries(models)) {
328
+ const fields = model["fields"] ?? {};
329
+ const storageFields = model["storage"]?.["fields"] ?? {};
330
+ const enrichedFields = {};
331
+ for (const [fieldName, field] of Object.entries(fields)) {
332
+ const column = storageFields[fieldName]?.["column"];
333
+ enrichedFields[fieldName] = column ? {
334
+ ...field,
335
+ column
336
+ } : { ...field };
337
+ }
338
+ enrichedModels[modelName] = {
339
+ ...model,
340
+ fields: enrichedFields,
341
+ relations: model["relations"] ?? {}
342
+ };
343
+ const modelRels = model["relations"] ?? {};
344
+ const tableName = modelToTable[modelName];
345
+ if (!tableName) continue;
346
+ for (const [relName, rel] of Object.entries(modelRels)) {
347
+ const on = rel["on"];
348
+ if (!on) continue;
349
+ const toModel = rel["to"];
350
+ if (!modelToTable[toModel]) continue;
351
+ const sourceFields = enrichedFields;
352
+ const targetModelObj = models[toModel];
353
+ const targetFields = targetModelObj?.["fields"] ?? {};
354
+ const targetStorageFields = (targetModelObj?.["storage"])?.["fields"] ?? {};
355
+ const parentCols = (on.localFields ?? []).map((f) => {
356
+ return storageFields[f]?.["column"] ?? sourceFields[f]?.["column"] ?? f;
357
+ });
358
+ const childCols = (on.targetFields ?? []).map((f) => {
359
+ return targetStorageFields[f]?.["column"] ?? targetFields[f]?.["column"] ?? f;
360
+ });
361
+ if (!topRelations[tableName]) topRelations[tableName] = {};
362
+ topRelations[tableName][relName] = {
363
+ to: toModel,
364
+ cardinality: rel["cardinality"],
365
+ on: {
366
+ parentCols,
367
+ childCols
368
+ }
369
+ };
370
+ }
371
+ }
372
+ return {
373
+ enrichedModels,
374
+ topRelations
375
+ };
376
+ }
377
+ function normalizeContract(contract) {
378
+ if (typeof contract !== "object" || contract === null) return contract;
379
+ const contractObj = contract;
380
+ const normalizedStorage = normalizeStorage(contractObj);
381
+ const rawModels = contractObj["models"];
382
+ if (!rawModels || typeof rawModels !== "object" || rawModels === null) return {
383
+ ...contractObj,
384
+ roots: contractObj["roots"] ?? {},
385
+ models: rawModels ?? {},
386
+ relations: contractObj["relations"] ?? {},
387
+ storage: normalizedStorage,
388
+ extensionPacks: contractObj["extensionPacks"] ?? {},
389
+ capabilities: contractObj["capabilities"] ?? {},
390
+ meta: contractObj["meta"] ?? {},
391
+ sources: contractObj["sources"] ?? {}
392
+ };
393
+ const modelsObj = rawModels;
394
+ const format = detectFormat(modelsObj);
395
+ let normalizedModels;
396
+ let roots;
397
+ let topRelations;
398
+ if (format === "new") {
399
+ const result = enrichNewFormatModels(modelsObj);
400
+ normalizedModels = result.enrichedModels;
401
+ topRelations = {
402
+ ...contractObj["relations"] ?? {},
403
+ ...result.topRelations
404
+ };
405
+ roots = contractObj["roots"] ?? {};
406
+ } else {
407
+ const storageObj = { tables: (normalizedStorage && typeof normalizedStorage === "object" ? normalizedStorage : {})["tables"] ?? {} };
408
+ const existingRelations = contractObj["relations"] ?? {};
409
+ const result = enrichOldFormatModels(modelsObj, storageObj, existingRelations);
410
+ normalizedModels = result.enrichedModels;
411
+ roots = result.roots;
412
+ topRelations = existingRelations;
413
+ }
414
+ return {
415
+ ...contractObj,
416
+ roots,
417
+ models: normalizedModels,
418
+ relations: topRelations,
419
+ storage: normalizedStorage,
420
+ extensionPacks: contractObj["extensionPacks"] ?? {},
421
+ capabilities: contractObj["capabilities"] ?? {},
422
+ meta: contractObj["meta"] ?? {},
423
+ sources: contractObj["sources"] ?? {}
424
+ };
425
+ }
426
+ function validateContract(value) {
427
+ const structurallyValid = validateSqlContract(normalizeContract(value));
428
+ validateContractDomain(extractDomainShape(structurallyValid));
429
+ validateContractLogic(structurallyValid);
430
+ const semanticErrors = validateStorageSemantics(structurallyValid.storage);
431
+ if (semanticErrors.length > 0) throw new Error(`Contract semantic validation failed: ${semanticErrors.join("; ")}`);
432
+ return decodeContractDefaults(constructContract(structurallyValid));
433
+ }
434
+
435
+ //#endregion
436
+ export { decodeContractDefaults, isBigIntColumn, normalizeContract, validateContract };
437
+ //# sourceMappingURL=validate.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate.mjs","names":["modelToTable: Record<string, string>","tableToModel: Record<string, string>","fieldToColumn: Record<string, Record<string, string>>","columnToField: Record<string, Record<string, string>>","modelFieldToColumn: Record<string, string>","decodedTables: Record<string, StorageTable>","decodedColumns: Record<string, StorageColumn>","normalizedTables: Record<string, unknown>","normalizedColumns: Record<string, unknown>","map: Record<string, string>","roots: Record<string, string>","tableToModel: Record<string, string>","enrichedModels: Record<string, RawModel>","enrichedFields: Record<string, RawField>","modelStorageFields: Record<string, { column: string }>","targetColumnToField: Record<string, Record<string, string>>","modelRelations: Record<string, unknown>","topRelations: Record<string, Record<string, unknown>>","modelToTable: Record<string, string>","normalizedModels: Record<string, RawModel>"],"sources":["../src/construct.ts","../src/validate.ts"],"sourcesContent":["import type { ModelDefinition, SqlContract, SqlMappings, SqlStorage } from './types';\n\ntype ResolvedMappings = {\n modelToTable: Record<string, string>;\n tableToModel: Record<string, string>;\n fieldToColumn: Record<string, Record<string, string>>;\n columnToField: Record<string, Record<string, string>>;\n};\n\nfunction computeDefaultMappings(models: Record<string, ModelDefinition>): ResolvedMappings {\n const modelToTable: Record<string, string> = {};\n const tableToModel: Record<string, string> = {};\n const fieldToColumn: Record<string, Record<string, string>> = {};\n const columnToField: Record<string, Record<string, string>> = {};\n\n for (const [modelName, model] of Object.entries(models)) {\n const tableName = model.storage.table;\n modelToTable[modelName] = tableName;\n tableToModel[tableName] = modelName;\n\n const modelFieldToColumn: Record<string, string> = {};\n for (const [fieldName, field] of Object.entries(model.fields)) {\n const columnName = field.column;\n modelFieldToColumn[fieldName] = columnName;\n if (!columnToField[tableName]) {\n columnToField[tableName] = {};\n }\n columnToField[tableName][columnName] = fieldName;\n }\n\n fieldToColumn[modelName] = modelFieldToColumn;\n }\n\n return {\n modelToTable,\n tableToModel,\n fieldToColumn,\n columnToField,\n };\n}\n\nfunction assertInverseModelMappings(\n modelToTable: Record<string, string>,\n tableToModel: Record<string, string>,\n): void {\n for (const [model, table] of Object.entries(modelToTable)) {\n if (tableToModel[table] !== model) {\n throw new Error(\n `Mappings override mismatch: modelToTable.${model}=\"${table}\" is not mirrored in tableToModel`,\n );\n }\n }\n for (const [table, model] of Object.entries(tableToModel)) {\n if (modelToTable[model] !== table) {\n throw new Error(\n `Mappings override mismatch: tableToModel.${table}=\"${model}\" is not mirrored in modelToTable`,\n );\n }\n }\n}\n\nfunction assertInverseFieldMappings(\n fieldToColumn: Record<string, Record<string, string>>,\n columnToField: Record<string, Record<string, string>>,\n modelToTable: Record<string, string>,\n tableToModel: Record<string, string>,\n): void {\n for (const [model, fields] of Object.entries(fieldToColumn)) {\n const table = modelToTable[model];\n if (!table) {\n throw new Error(\n `Mappings override mismatch: fieldToColumn references unknown model \"${model}\"`,\n );\n }\n const reverseFields = columnToField[table];\n if (!reverseFields) {\n throw new Error(\n `Mappings override mismatch: columnToField is missing table \"${table}\" for model \"${model}\"`,\n );\n }\n for (const [field, column] of Object.entries(fields)) {\n if (reverseFields[column] !== field) {\n throw new Error(\n `Mappings override mismatch: fieldToColumn.${model}.${field}=\"${column}\" is not mirrored in columnToField.${table}`,\n );\n }\n }\n }\n\n for (const [table, columns] of Object.entries(columnToField)) {\n const model = tableToModel[table];\n if (!model) {\n throw new Error(\n `Mappings override mismatch: columnToField references unknown table \"${table}\"`,\n );\n }\n const forwardFields = fieldToColumn[model];\n if (!forwardFields) {\n throw new Error(\n `Mappings override mismatch: fieldToColumn is missing model \"${model}\" for table \"${table}\"`,\n );\n }\n for (const [column, field] of Object.entries(columns)) {\n if (forwardFields[field] !== column) {\n throw new Error(\n `Mappings override mismatch: columnToField.${table}.${column}=\"${field}\" is not mirrored in fieldToColumn.${model}`,\n );\n }\n }\n }\n}\n\nfunction mergeMappings(\n defaults: ResolvedMappings,\n existingMappings?: Partial<SqlMappings>,\n): ResolvedMappings {\n const hasModelToTable = existingMappings?.modelToTable !== undefined;\n const hasTableToModel = existingMappings?.tableToModel !== undefined;\n if (hasModelToTable !== hasTableToModel) {\n throw new Error(\n 'Mappings override mismatch: modelToTable and tableToModel must be provided together',\n );\n }\n\n const hasFieldToColumn = existingMappings?.fieldToColumn !== undefined;\n const hasColumnToField = existingMappings?.columnToField !== undefined;\n if (hasFieldToColumn !== hasColumnToField) {\n throw new Error(\n 'Mappings override mismatch: fieldToColumn and columnToField must be provided together',\n );\n }\n\n const modelToTable: Record<string, string> = hasModelToTable\n ? (existingMappings?.modelToTable ?? {})\n : defaults.modelToTable;\n const tableToModel: Record<string, string> = hasTableToModel\n ? (existingMappings?.tableToModel ?? {})\n : defaults.tableToModel;\n assertInverseModelMappings(modelToTable, tableToModel);\n\n const fieldToColumn: Record<string, Record<string, string>> = hasFieldToColumn\n ? (existingMappings?.fieldToColumn ?? {})\n : defaults.fieldToColumn;\n const columnToField: Record<string, Record<string, string>> = hasColumnToField\n ? (existingMappings?.columnToField ?? {})\n : defaults.columnToField;\n assertInverseFieldMappings(fieldToColumn, columnToField, modelToTable, tableToModel);\n\n return {\n modelToTable,\n tableToModel,\n fieldToColumn,\n columnToField,\n };\n}\n\ntype ValidatedContractInput = SqlContract<SqlStorage> & { _generated?: unknown };\n\nfunction stripGenerated(obj: ValidatedContractInput): Omit<ValidatedContractInput, '_generated'> {\n const input = obj as unknown as Record<string, unknown>;\n const { _generated: _, ...rest } = input;\n return rest as Omit<ValidatedContractInput, '_generated'>;\n}\n\nexport function constructContract<TContract extends SqlContract<SqlStorage>>(\n input: ValidatedContractInput,\n): TContract {\n const existingMappings = (input as { mappings?: Partial<SqlMappings> }).mappings;\n const defaultMappings = computeDefaultMappings(input.models as Record<string, ModelDefinition>);\n const mappings = mergeMappings(defaultMappings, existingMappings);\n\n const stripped = stripGenerated(input);\n\n const contractWithMappings = {\n ...stripped,\n mappings,\n roots: stripped.roots,\n };\n\n return contractWithMappings as TContract;\n}\n","import type { ColumnDefaultLiteralInputValue } from '@prisma-next/contract/types';\nimport { isTaggedBigInt, isTaggedRaw } from '@prisma-next/contract/types';\nimport type { DomainContractShape, DomainModelShape } from '@prisma-next/contract/validate-domain';\nimport { validateContractDomain } from '@prisma-next/contract/validate-domain';\nimport { constructContract } from './construct';\nimport type { SqlContract, SqlStorage, StorageColumn, StorageTable } from './types';\nimport { applyFkDefaults } from './types';\nimport { validateSqlContract, validateStorageSemantics } from './validators';\n\nfunction extractDomainShape(contract: SqlContract<SqlStorage>): DomainContractShape {\n return {\n roots: contract.roots,\n models: contract.models as Record<string, DomainModelShape>,\n };\n}\n\nfunction validateContractLogic(contract: SqlContract<SqlStorage>): void {\n const tableNames = new Set(Object.keys(contract.storage.tables));\n\n for (const [tableName, table] of Object.entries(contract.storage.tables)) {\n const columnNames = new Set(Object.keys(table.columns));\n\n if (table.primaryKey) {\n for (const colName of table.primaryKey.columns) {\n if (!columnNames.has(colName)) {\n throw new Error(\n `Table \"${tableName}\" primaryKey references non-existent column \"${colName}\"`,\n );\n }\n }\n }\n\n for (const unique of table.uniques) {\n for (const colName of unique.columns) {\n if (!columnNames.has(colName)) {\n throw new Error(\n `Table \"${tableName}\" unique constraint references non-existent column \"${colName}\"`,\n );\n }\n }\n }\n\n for (const index of table.indexes) {\n for (const colName of index.columns) {\n if (!columnNames.has(colName)) {\n throw new Error(`Table \"${tableName}\" index references non-existent column \"${colName}\"`);\n }\n }\n }\n\n for (const [colName, column] of Object.entries(table.columns)) {\n if (!column.nullable && column.default?.kind === 'literal' && column.default.value === null) {\n throw new Error(\n `Table \"${tableName}\" column \"${colName}\" is NOT NULL but has a literal null default`,\n );\n }\n }\n\n for (const fk of table.foreignKeys) {\n for (const colName of fk.columns) {\n if (!columnNames.has(colName)) {\n throw new Error(\n `Table \"${tableName}\" foreignKey references non-existent column \"${colName}\"`,\n );\n }\n }\n\n if (!tableNames.has(fk.references.table)) {\n throw new Error(\n `Table \"${tableName}\" foreignKey references non-existent table \"${fk.references.table}\"`,\n );\n }\n\n const referencedTable = contract.storage.tables[\n fk.references.table\n ] as (typeof contract.storage.tables)[string];\n const referencedColumnNames = new Set(Object.keys(referencedTable.columns));\n for (const colName of fk.references.columns) {\n if (!referencedColumnNames.has(colName)) {\n throw new Error(\n `Table \"${tableName}\" foreignKey references non-existent column \"${colName}\" in table \"${fk.references.table}\"`,\n );\n }\n }\n\n if (fk.columns.length !== fk.references.columns.length) {\n throw new Error(\n `Table \"${tableName}\" foreignKey column count (${fk.columns.length}) does not match referenced column count (${fk.references.columns.length})`,\n );\n }\n }\n }\n}\n\nconst BIGINT_NATIVE_TYPES = new Set(['bigint', 'int8']);\n\nexport function isBigIntColumn(column: StorageColumn): boolean {\n const nativeType = column.nativeType?.toLowerCase() ?? '';\n if (BIGINT_NATIVE_TYPES.has(nativeType)) return true;\n const codecId = column.codecId?.toLowerCase() ?? '';\n return codecId.includes('int8') || codecId.includes('bigint');\n}\n\nexport function decodeDefaultLiteralValue(\n value: ColumnDefaultLiteralInputValue,\n column: StorageColumn,\n tableName: string,\n columnName: string,\n): ColumnDefaultLiteralInputValue {\n if (value instanceof Date) {\n return value;\n }\n if (isTaggedRaw(value)) {\n return value.value;\n }\n if (isTaggedBigInt(value)) {\n if (!isBigIntColumn(column)) {\n return value;\n }\n try {\n return BigInt(value.value);\n } catch {\n throw new Error(\n `Invalid tagged bigint for default value on \"${tableName}.${columnName}\": \"${value.value}\" is not a valid integer`,\n );\n }\n }\n return value;\n}\n\nexport function decodeContractDefaults<T extends SqlContract<SqlStorage>>(contract: T): T {\n const tables = contract.storage.tables;\n let tablesChanged = false;\n const decodedTables: Record<string, StorageTable> = {};\n\n for (const [tableName, table] of Object.entries(tables)) {\n let columnsChanged = false;\n const decodedColumns: Record<string, StorageColumn> = {};\n\n for (const [columnName, column] of Object.entries(table.columns)) {\n if (column.default?.kind === 'literal') {\n const decodedValue = decodeDefaultLiteralValue(\n column.default.value,\n column,\n tableName,\n columnName,\n );\n if (decodedValue !== column.default.value) {\n columnsChanged = true;\n decodedColumns[columnName] = {\n ...column,\n default: { kind: 'literal', value: decodedValue },\n };\n continue;\n }\n }\n decodedColumns[columnName] = column;\n }\n\n if (columnsChanged) {\n tablesChanged = true;\n decodedTables[tableName] = { ...table, columns: decodedColumns };\n } else {\n decodedTables[tableName] = table;\n }\n }\n\n if (!tablesChanged) {\n return contract;\n }\n\n // The spread widens to SqlContract<SqlStorage>, but this transformation only\n // decodes tagged bigint defaults for bigint-like columns and preserves all\n // other properties of T.\n return {\n ...contract,\n storage: {\n ...contract.storage,\n tables: decodedTables,\n },\n } as T;\n}\n\nfunction normalizeStorage(contractObj: Record<string, unknown>): Record<string, unknown> {\n const normalizedStorage = contractObj['storage'];\n if (!normalizedStorage || typeof normalizedStorage !== 'object')\n return normalizedStorage as Record<string, unknown>;\n\n const storage = normalizedStorage as Record<string, unknown>;\n const tables = storage['tables'] as Record<string, unknown> | undefined;\n if (!tables) return storage;\n\n const normalizedTables: Record<string, unknown> = {};\n for (const [tableName, table] of Object.entries(tables)) {\n const tableObj = table as Record<string, unknown>;\n const columns = tableObj['columns'] as Record<string, unknown> | undefined;\n\n if (columns) {\n const normalizedColumns: Record<string, unknown> = {};\n for (const [columnName, column] of Object.entries(columns)) {\n const columnObj = column as Record<string, unknown>;\n normalizedColumns[columnName] = {\n ...columnObj,\n nullable: columnObj['nullable'] ?? false,\n };\n }\n\n const rawForeignKeys = (tableObj['foreignKeys'] ?? []) as Array<Record<string, unknown>>;\n const normalizedForeignKeys = rawForeignKeys.map((fk) => ({\n ...fk,\n ...applyFkDefaults({\n constraint: typeof fk['constraint'] === 'boolean' ? fk['constraint'] : undefined,\n index: typeof fk['index'] === 'boolean' ? fk['index'] : undefined,\n }),\n }));\n\n normalizedTables[tableName] = {\n ...tableObj,\n columns: normalizedColumns,\n uniques: tableObj['uniques'] ?? [],\n indexes: tableObj['indexes'] ?? [],\n foreignKeys: normalizedForeignKeys,\n };\n } else {\n normalizedTables[tableName] = tableObj;\n }\n }\n\n return { ...storage, tables: normalizedTables };\n}\n\ntype RawModel = Record<string, unknown>;\ntype RawField = Record<string, unknown>;\ntype RawRelation = Record<string, unknown>;\ntype RawStorageObj = { tables: Record<string, Record<string, unknown>> };\n\nfunction detectFormat(models: Record<string, RawModel>): 'old' | 'new' {\n for (const model of Object.values(models)) {\n const fields = model['fields'] as Record<string, RawField> | undefined;\n if (!fields) continue;\n for (const field of Object.values(fields)) {\n if ('column' in field) return 'old';\n if ('codecId' in field) return 'new';\n }\n }\n return 'old';\n}\n\nfunction buildColumnToFieldMap(\n fields: Record<string, RawField>,\n modelName: string,\n): Record<string, string> {\n const map: Record<string, string> = {};\n for (const [fieldName, field] of Object.entries(fields)) {\n const col = field['column'] as string | undefined;\n if (!col) continue;\n if (Object.hasOwn(map, col)) {\n throw new Error(\n `Model \"${modelName}\" has duplicate column mapping: fields \"${map[col]}\" and \"${fieldName}\" both map to column \"${col}\"`,\n );\n }\n map[col] = fieldName;\n }\n return map;\n}\n\nfunction enrichOldFormatModels(\n models: Record<string, RawModel>,\n storageObj: RawStorageObj,\n topRelations: Record<string, Record<string, RawRelation>>,\n): { enrichedModels: Record<string, RawModel>; roots: Record<string, string> } {\n const roots: Record<string, string> = {};\n const tableToModel: Record<string, string> = {};\n\n for (const [modelName, model] of Object.entries(models)) {\n const modelStorage = model['storage'] as Record<string, unknown> | undefined;\n const tableName = modelStorage?.['table'] as string | undefined;\n if (tableName) {\n if (!model['owner']) {\n roots[modelName] = modelName;\n }\n tableToModel[tableName] = modelName;\n }\n }\n\n const enrichedModels: Record<string, RawModel> = {};\n\n for (const [modelName, model] of Object.entries(models)) {\n const fields = (model['fields'] ?? {}) as Record<string, RawField>;\n const modelStorage = model['storage'] as Record<string, unknown> | undefined;\n const tableName = modelStorage?.['table'] as string | undefined;\n const storageTable = tableName\n ? (storageObj.tables[tableName] as Record<string, unknown> | undefined)\n : undefined;\n const storageColumns = (storageTable?.['columns'] ?? {}) as Record<\n string,\n Record<string, unknown>\n >;\n\n const enrichedFields: Record<string, RawField> = {};\n const modelStorageFields: Record<string, { column: string }> = {};\n\n const hasStorageColumns = Object.keys(storageColumns).length > 0;\n for (const [fieldName, field] of Object.entries(fields)) {\n const colName = field['column'] as string;\n const storageCol = storageColumns[colName];\n if (!storageCol && hasStorageColumns && colName) {\n throw new Error(\n `Model \"${modelName}\" field \"${fieldName}\" references non-existent column \"${colName}\" in table \"${tableName}\"`,\n );\n }\n enrichedFields[fieldName] = {\n ...field,\n nullable: storageCol?.['nullable'] ?? false,\n codecId: storageCol?.['codecId'] ?? '',\n };\n modelStorageFields[fieldName] = { column: colName };\n }\n\n const enrichedStorage = {\n ...(modelStorage ?? {}),\n fields: modelStorageFields,\n };\n\n enrichedModels[modelName] = {\n ...model,\n fields: enrichedFields,\n storage: enrichedStorage,\n relations: model['relations'] ?? {},\n };\n }\n\n for (const [tableName, tableRels] of Object.entries(topRelations)) {\n const modelName = tableToModel[tableName];\n if (!modelName) continue;\n const existingModel = enrichedModels[modelName];\n if (!existingModel) continue;\n\n const existingRels = (existingModel['relations'] ?? {}) as Record<string, unknown>;\n const targetColumnToField: Record<string, Record<string, string>> = {};\n\n const modelRelations: Record<string, unknown> = { ...existingRels };\n for (const [relName, rel] of Object.entries(tableRels)) {\n const on = rel['on'] as { childCols?: string[]; parentCols?: string[] } | undefined;\n const parentCols = on?.['parentCols'] ?? [];\n const childCols = on?.['childCols'] ?? [];\n\n const toModel = rel['to'] as string;\n const sourceFields = (existingModel['fields'] ?? {}) as Record<string, RawField>;\n const sourceColToField = buildColumnToFieldMap(sourceFields, modelName);\n\n if (!targetColumnToField[toModel]) {\n const targetModelObj = enrichedModels[toModel];\n if (targetModelObj) {\n targetColumnToField[toModel] = buildColumnToFieldMap(\n (targetModelObj['fields'] ?? {}) as Record<string, RawField>,\n toModel,\n );\n } else {\n targetColumnToField[toModel] = {};\n }\n }\n const targetColToField = targetColumnToField[toModel] ?? {};\n\n // Old format: parentCols = columns on FK-holding table (local), childCols = columns on referenced table (target)\n const localFields = parentCols.map((c: string) => sourceColToField[c] ?? c);\n const targetFields = childCols.map((c: string) => targetColToField[c] ?? c);\n\n modelRelations[relName] = {\n to: toModel,\n cardinality: rel['cardinality'],\n on: { localFields, targetFields },\n };\n }\n\n enrichedModels[modelName] = {\n ...existingModel,\n relations: modelRelations,\n };\n }\n\n return { enrichedModels, roots };\n}\n\nfunction enrichNewFormatModels(models: Record<string, RawModel>): {\n enrichedModels: Record<string, RawModel>;\n topRelations: Record<string, Record<string, unknown>>;\n} {\n const enrichedModels: Record<string, RawModel> = {};\n const topRelations: Record<string, Record<string, unknown>> = {};\n const modelToTable: Record<string, string> = {};\n\n for (const [modelName, model] of Object.entries(models)) {\n const modelStorage = model['storage'] as Record<string, unknown> | undefined;\n const tableName = modelStorage?.['table'] as string | undefined;\n if (tableName) modelToTable[modelName] = tableName;\n }\n\n for (const [modelName, model] of Object.entries(models)) {\n const fields = (model['fields'] ?? {}) as Record<string, RawField>;\n const modelStorage = model['storage'] as Record<string, unknown> | undefined;\n const storageFields = (modelStorage?.['fields'] ?? {}) as Record<\n string,\n Record<string, unknown>\n >;\n\n const enrichedFields: Record<string, RawField> = {};\n for (const [fieldName, field] of Object.entries(fields)) {\n const sfEntry = storageFields[fieldName];\n const column = sfEntry?.['column'] as string | undefined;\n enrichedFields[fieldName] = column ? { ...field, column } : { ...field };\n }\n\n enrichedModels[modelName] = {\n ...model,\n fields: enrichedFields,\n relations: model['relations'] ?? {},\n };\n\n const modelRels = (model['relations'] ?? {}) as Record<string, RawRelation>;\n const tableName = modelToTable[modelName];\n if (!tableName) continue;\n\n for (const [relName, rel] of Object.entries(modelRels)) {\n const on = rel['on'] as { localFields?: string[]; targetFields?: string[] } | undefined;\n if (!on) continue;\n const toModel = rel['to'] as string;\n const toTable = modelToTable[toModel];\n if (!toTable) continue;\n\n const sourceFields = enrichedFields;\n const targetModelObj = models[toModel];\n const targetFields = (targetModelObj?.['fields'] ?? {}) as Record<string, RawField>;\n const targetStorageObj = targetModelObj?.['storage'] as Record<string, unknown> | undefined;\n const targetStorageFields = (targetStorageObj?.['fields'] ?? {}) as Record<\n string,\n Record<string, unknown>\n >;\n\n const parentCols = (on.localFields ?? []).map((f: string) => {\n const sf = storageFields[f];\n return (\n (sf?.['column'] as string | undefined) ??\n (sourceFields[f]?.['column'] as string | undefined) ??\n f\n );\n });\n\n const childCols = (on.targetFields ?? []).map((f: string) => {\n const tsf = targetStorageFields[f];\n return (\n (tsf?.['column'] as string | undefined) ??\n (targetFields[f]?.['column'] as string | undefined) ??\n f\n );\n });\n\n if (!topRelations[tableName]) topRelations[tableName] = {};\n topRelations[tableName][relName] = {\n to: toModel,\n cardinality: rel['cardinality'],\n on: { parentCols, childCols },\n };\n }\n }\n\n return { enrichedModels, topRelations };\n}\n\nexport function normalizeContract(contract: unknown): SqlContract<SqlStorage> {\n if (typeof contract !== 'object' || contract === null) {\n return contract as SqlContract<SqlStorage>;\n }\n\n const contractObj = contract as Record<string, unknown>;\n const normalizedStorage = normalizeStorage(contractObj);\n\n const rawModels = contractObj['models'];\n if (!rawModels || typeof rawModels !== 'object' || rawModels === null) {\n return {\n ...contractObj,\n roots: contractObj['roots'] ?? {},\n models: rawModels ?? {},\n relations: contractObj['relations'] ?? {},\n storage: normalizedStorage,\n extensionPacks: contractObj['extensionPacks'] ?? {},\n capabilities: contractObj['capabilities'] ?? {},\n meta: contractObj['meta'] ?? {},\n sources: contractObj['sources'] ?? {},\n } as SqlContract<SqlStorage>;\n }\n\n const modelsObj = rawModels as Record<string, RawModel>;\n const format = detectFormat(modelsObj);\n\n let normalizedModels: Record<string, RawModel>;\n let roots: Record<string, string>;\n let topRelations: Record<string, Record<string, unknown>>;\n\n if (format === 'new') {\n const result = enrichNewFormatModels(modelsObj);\n normalizedModels = result.enrichedModels;\n topRelations = {\n ...((contractObj['relations'] ?? {}) as Record<string, Record<string, unknown>>),\n ...result.topRelations,\n };\n roots = (contractObj['roots'] as Record<string, string>) ?? {};\n } else {\n const rawStorageObj =\n normalizedStorage && typeof normalizedStorage === 'object'\n ? (normalizedStorage as Record<string, unknown>)\n : {};\n const storageObj = {\n tables: ((rawStorageObj as Record<string, unknown>)['tables'] ?? {}) as Record<\n string,\n Record<string, unknown>\n >,\n };\n const existingRelations = (contractObj['relations'] ?? {}) as Record<\n string,\n Record<string, RawRelation>\n >;\n const result = enrichOldFormatModels(modelsObj, storageObj, existingRelations);\n normalizedModels = result.enrichedModels;\n roots = result.roots;\n topRelations = existingRelations;\n }\n\n return {\n ...contractObj,\n roots,\n models: normalizedModels,\n relations: topRelations,\n storage: normalizedStorage,\n extensionPacks: contractObj['extensionPacks'] ?? {},\n capabilities: contractObj['capabilities'] ?? {},\n meta: contractObj['meta'] ?? {},\n sources: contractObj['sources'] ?? {},\n } as SqlContract<SqlStorage>;\n}\n\nexport function validateContract<TContract extends SqlContract<SqlStorage>>(\n value: unknown,\n): TContract {\n const normalized = normalizeContract(value);\n\n const structurallyValid = validateSqlContract<SqlContract<SqlStorage>>(normalized);\n\n validateContractDomain(extractDomainShape(structurallyValid));\n\n validateContractLogic(structurallyValid);\n\n const semanticErrors = validateStorageSemantics(structurallyValid.storage);\n if (semanticErrors.length > 0) {\n throw new Error(`Contract semantic validation failed: ${semanticErrors.join('; ')}`);\n }\n\n const constructed = constructContract<TContract>(structurallyValid);\n return decodeContractDefaults(constructed) as TContract;\n}\n"],"mappings":"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"}