@decaf-ts/for-typeorm 0.0.6

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 (109) hide show
  1. package/LICENSE.md +21 -0
  2. package/README.md +93 -0
  3. package/dist/for-typeorm.cjs +2553 -0
  4. package/dist/for-typeorm.esm.cjs +2538 -0
  5. package/lib/TypeORMAdapter.cjs +1129 -0
  6. package/lib/TypeORMAdapter.d.ts +221 -0
  7. package/lib/TypeORMDispatch.cjs +134 -0
  8. package/lib/TypeORMDispatch.d.ts +87 -0
  9. package/lib/TypeORMEventSubscriber.cjs +96 -0
  10. package/lib/TypeORMEventSubscriber.d.ts +56 -0
  11. package/lib/TypeORMRepository.cjs +209 -0
  12. package/lib/TypeORMRepository.d.ts +125 -0
  13. package/lib/constants.cjs +43 -0
  14. package/lib/constants.d.ts +39 -0
  15. package/lib/errors.cjs +28 -0
  16. package/lib/errors.d.ts +21 -0
  17. package/lib/esm/TypeORMAdapter.d.ts +221 -0
  18. package/lib/esm/TypeORMAdapter.js +1124 -0
  19. package/lib/esm/TypeORMDispatch.d.ts +87 -0
  20. package/lib/esm/TypeORMDispatch.js +130 -0
  21. package/lib/esm/TypeORMEventSubscriber.d.ts +56 -0
  22. package/lib/esm/TypeORMEventSubscriber.js +93 -0
  23. package/lib/esm/TypeORMRepository.d.ts +125 -0
  24. package/lib/esm/TypeORMRepository.js +206 -0
  25. package/lib/esm/constants.d.ts +39 -0
  26. package/lib/esm/constants.js +40 -0
  27. package/lib/esm/errors.d.ts +21 -0
  28. package/lib/esm/errors.js +24 -0
  29. package/lib/esm/index.d.ts +22 -0
  30. package/lib/esm/index.js +25 -0
  31. package/lib/esm/indexes/generator.d.ts +50 -0
  32. package/lib/esm/indexes/generator.js +95 -0
  33. package/lib/esm/indexes/index.d.ts +1 -0
  34. package/lib/esm/indexes/index.js +2 -0
  35. package/lib/esm/overrides/Column.d.ts +74 -0
  36. package/lib/esm/overrides/Column.js +70 -0
  37. package/lib/esm/overrides/CreateDateColumn.d.ts +2 -0
  38. package/lib/esm/overrides/CreateDateColumn.js +9 -0
  39. package/lib/esm/overrides/Entity.d.ts +11 -0
  40. package/lib/esm/overrides/Entity.js +28 -0
  41. package/lib/esm/overrides/PrimaryColumn.d.ts +20 -0
  42. package/lib/esm/overrides/PrimaryColumn.js +53 -0
  43. package/lib/esm/overrides/PrimaryGeneratedColumn.d.ts +24 -0
  44. package/lib/esm/overrides/PrimaryGeneratedColumn.js +51 -0
  45. package/lib/esm/overrides/UpdateDateColumn.d.ts +2 -0
  46. package/lib/esm/overrides/UpdateDateColumn.js +9 -0
  47. package/lib/esm/overrides/utils.d.ts +2 -0
  48. package/lib/esm/overrides/utils.js +29 -0
  49. package/lib/esm/query/Paginator.d.ts +86 -0
  50. package/lib/esm/query/Paginator.js +124 -0
  51. package/lib/esm/query/Statement.d.ts +131 -0
  52. package/lib/esm/query/Statement.js +242 -0
  53. package/lib/esm/query/constants.d.ts +52 -0
  54. package/lib/esm/query/constants.js +74 -0
  55. package/lib/esm/query/index.d.ts +4 -0
  56. package/lib/esm/query/index.js +5 -0
  57. package/lib/esm/query/translate.d.ts +34 -0
  58. package/lib/esm/query/translate.js +42 -0
  59. package/lib/esm/raw/postgres.d.ts +36 -0
  60. package/lib/esm/raw/postgres.js +2 -0
  61. package/lib/esm/sequences/Sequence.d.ts +67 -0
  62. package/lib/esm/sequences/Sequence.js +117 -0
  63. package/lib/esm/sequences/index.d.ts +1 -0
  64. package/lib/esm/sequences/index.js +2 -0
  65. package/lib/esm/types.d.ts +67 -0
  66. package/lib/esm/types.js +28 -0
  67. package/lib/esm/utils.d.ts +16 -0
  68. package/lib/esm/utils.js +29 -0
  69. package/lib/index.cjs +42 -0
  70. package/lib/index.d.ts +22 -0
  71. package/lib/indexes/generator.cjs +98 -0
  72. package/lib/indexes/generator.d.ts +50 -0
  73. package/lib/indexes/index.cjs +18 -0
  74. package/lib/indexes/index.d.ts +1 -0
  75. package/lib/overrides/Column.cjs +73 -0
  76. package/lib/overrides/Column.d.ts +74 -0
  77. package/lib/overrides/CreateDateColumn.cjs +12 -0
  78. package/lib/overrides/CreateDateColumn.d.ts +2 -0
  79. package/lib/overrides/Entity.cjs +31 -0
  80. package/lib/overrides/Entity.d.ts +11 -0
  81. package/lib/overrides/PrimaryColumn.cjs +56 -0
  82. package/lib/overrides/PrimaryColumn.d.ts +20 -0
  83. package/lib/overrides/PrimaryGeneratedColumn.cjs +54 -0
  84. package/lib/overrides/PrimaryGeneratedColumn.d.ts +24 -0
  85. package/lib/overrides/UpdateDateColumn.cjs +12 -0
  86. package/lib/overrides/UpdateDateColumn.d.ts +2 -0
  87. package/lib/overrides/utils.cjs +32 -0
  88. package/lib/overrides/utils.d.ts +2 -0
  89. package/lib/query/Paginator.cjs +128 -0
  90. package/lib/query/Paginator.d.ts +86 -0
  91. package/lib/query/Statement.cjs +246 -0
  92. package/lib/query/Statement.d.ts +131 -0
  93. package/lib/query/constants.cjs +77 -0
  94. package/lib/query/constants.d.ts +52 -0
  95. package/lib/query/index.cjs +21 -0
  96. package/lib/query/index.d.ts +4 -0
  97. package/lib/query/translate.cjs +45 -0
  98. package/lib/query/translate.d.ts +34 -0
  99. package/lib/raw/postgres.cjs +3 -0
  100. package/lib/raw/postgres.d.ts +36 -0
  101. package/lib/sequences/Sequence.cjs +121 -0
  102. package/lib/sequences/Sequence.d.ts +67 -0
  103. package/lib/sequences/index.cjs +18 -0
  104. package/lib/sequences/index.d.ts +1 -0
  105. package/lib/types.cjs +31 -0
  106. package/lib/types.d.ts +67 -0
  107. package/lib/utils.cjs +32 -0
  108. package/lib/utils.d.ts +16 -0
  109. package/package.json +128 -0
@@ -0,0 +1,1124 @@
1
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
2
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
3
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
4
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
5
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
6
+ };
7
+ var __metadata = (this && this.__metadata) || function (k, v) {
8
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
9
+ };
10
+ import { Adapter, Cascade, ConnectionError, final, PersistenceKeys, Repository, } from "@decaf-ts/core";
11
+ import { reservedAttributes, TypeORMFlavour } from "./constants.js";
12
+ import { BaseError, ConflictError, DBKeys, DEFAULT_ERROR_MESSAGES as DB_DEFAULT_ERROR_MESSAGES, findPrimaryKey, InternalError, NotFoundError, OperationKeys, readonly, UpdateValidationKeys, } from "@decaf-ts/db-decorators";
13
+ import "reflect-metadata";
14
+ import { date, Decoration, DEFAULT_ERROR_MESSAGES, list, Model, ModelKeys, prop, propMetadata, required, type, Validation, ValidationKeys, } from "@decaf-ts/decorator-validation";
15
+ import { IndexError } from "./errors.js";
16
+ import { TypeORMStatement } from "./query/index.js";
17
+ import { TypeORMSequence } from "./sequences/index.js";
18
+ import { generateIndexes } from "./indexes/index.js";
19
+ import { apply, Reflection } from "@decaf-ts/reflection";
20
+ import { TypeORMRepository } from "./TypeORMRepository.js";
21
+ import { Logging } from "@decaf-ts/logging";
22
+ import { TypeORMDispatch } from "./TypeORMDispatch.js";
23
+ import { convertJsRegexToPostgres } from "./utils.js";
24
+ import { DataSource, In, OneToOne, JoinColumn, ManyToMany, VersionColumn, OneToMany, ManyToOne, JoinTable, } from "typeorm";
25
+ import { Column } from "./overrides/Column.js";
26
+ import { UpdateDateColumn } from "./overrides/UpdateDateColumn.js";
27
+ import { CreateDateColumn } from "./overrides/CreateDateColumn.js";
28
+ import { PrimaryGeneratedColumn } from "./overrides/PrimaryGeneratedColumn.js";
29
+ import { PrimaryColumn } from "./overrides/PrimaryColumn.js";
30
+ import { Entity } from "./overrides/Entity.js";
31
+ export async function createdByOnPostgresCreateUpdate(context, data, key, model) {
32
+ try {
33
+ const user = context.get("user");
34
+ model[key] = user;
35
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
36
+ }
37
+ catch (e) {
38
+ throw new InternalError("No User found in context. Please provide a user in the context");
39
+ }
40
+ }
41
+ /**
42
+ * @description Adapter for TypeORM-backed persistence operations.
43
+ * @summary Implements the Decaf.ts Adapter over a TypeORM DataSource, providing CRUD operations, query/statement factories, sequence management, error parsing, and decoration helpers.
44
+ * @template Y The native configuration type (TypeORM DataSourceOptions).
45
+ * @template F The repository flags type.
46
+ * @template C The context type.
47
+ * @param {DataSourceOptions} scope The DataSource options for the adapter.
48
+ * @param {string} flavour The flavour of the adapter.
49
+ * @param {string} [alias] Optional alias for the adapter.
50
+ * @class TypeORMAdapter
51
+ * @example
52
+ * const adapter = new TypeORMAdapter({ type: 'postgres', /* ... *\/ });
53
+ * await adapter.initialize();
54
+ * const repo = new (adapter.repository<User>())(adapter, User);
55
+ * const created = await repo.create(new User({ name: 'Alice' }));
56
+ *
57
+ * @mermaid
58
+ * sequenceDiagram
59
+ * participant App
60
+ * participant Adapter as TypeORMAdapter
61
+ * participant Repo as TypeORMRepository
62
+ * participant DS as TypeORM DataSource
63
+ *
64
+ * App->>Adapter: new TypeORMAdapter(opts)
65
+ * Adapter->>DS: initialize()
66
+ * App->>Adapter: repository()
67
+ * Adapter-->>App: TypeORMRepository
68
+ * App->>Repo: create(model)
69
+ * Repo->>Adapter: prepare/create/revert
70
+ * Adapter-->>Repo: Model
71
+ * Repo-->>App: Model
72
+ */
73
+ export class TypeORMAdapter extends Adapter {
74
+ get dataSource() {
75
+ if (!this._dataSource) {
76
+ const models = Adapter.models(this.flavour);
77
+ this._dataSource = new DataSource(Object.assign(this.native, {
78
+ entities: models.map((c) => c[ModelKeys.ANCHOR]),
79
+ }));
80
+ }
81
+ return this._dataSource;
82
+ }
83
+ // protected dataSou
84
+ constructor(options, alias) {
85
+ super(options, TypeORMFlavour, alias);
86
+ }
87
+ async flags(operation, model, flags) {
88
+ const f = await super.flags(operation, model, flags);
89
+ const newObj = {
90
+ user: (await TypeORMAdapter.getCurrentUser(this.dataSource)),
91
+ };
92
+ const m = new model();
93
+ const exceptions = [];
94
+ if (operation === OperationKeys.CREATE) {
95
+ const pk = findPrimaryKey(m).id;
96
+ exceptions.push(pk);
97
+ }
98
+ if (operation === OperationKeys.CREATE ||
99
+ operation === OperationKeys.UPDATE) {
100
+ const decs = Object.keys(m).reduce((accum, key) => {
101
+ const decs = Reflection.getPropertyDecorators(ValidationKeys.REFLECT, m, key, true);
102
+ const dec = decs.decorators.find((dec) => dec.key === DBKeys.TIMESTAMP &&
103
+ dec.props.operation.indexOf(operation) !== -1);
104
+ if (dec) {
105
+ accum[key] = dec.props;
106
+ }
107
+ return accum;
108
+ }, {});
109
+ exceptions.push(...Object.keys(decs));
110
+ }
111
+ newObj.ignoredValidationProperties = (f.ignoredValidationProperties ? f.ignoredValidationProperties : []).concat(...exceptions);
112
+ return Object.assign(f, newObj);
113
+ }
114
+ Dispatch() {
115
+ return new TypeORMDispatch();
116
+ }
117
+ repository() {
118
+ return TypeORMRepository;
119
+ }
120
+ /**
121
+ * @description Creates a new Postgres statement for querying
122
+ * @summary Factory method that creates a new PostgresStatement instance for building queries
123
+ * @template M - The model type
124
+ * @return {TypeORMStatement<M, any>} A new PostgresStatement instance
125
+ */
126
+ Statement() {
127
+ return new TypeORMStatement(this);
128
+ }
129
+ /**
130
+ * @description Creates a new PostgreSQL sequence
131
+ * @summary Factory method that creates a new PostgreSQLSequence instance for managing sequences
132
+ * @param {SequenceOptions} options - The options for the sequence
133
+ * @return {Promise<Sequence>} A promise that resolves to a new Sequence instance
134
+ */
135
+ async Sequence(options) {
136
+ return new TypeORMSequence(options, this);
137
+ }
138
+ /**
139
+ * @description Initializes the adapter by creating indexes for all managed models
140
+ * @summary Sets up the necessary database indexes for all models managed by this adapter
141
+ * @return {Promise<void>} A promise that resolves when initialization is complete
142
+ */
143
+ async initialize() {
144
+ const ds = this.dataSource;
145
+ try {
146
+ await ds.initialize();
147
+ }
148
+ catch (e) {
149
+ throw this.parseError(e);
150
+ }
151
+ const log = this.log.for(this.initialize);
152
+ log.verbose(`${this.flavour} adapter initialized`);
153
+ }
154
+ /**
155
+ * @description Creates indexes for the given models
156
+ * @summary Abstract method that must be implemented to create database indexes for the specified models
157
+ * @template M - The model type
158
+ * @param {...Constructor<M>} models - The model constructors to create indexes for
159
+ * @return {Promise<void>} A promise that resolves when all indexes are created
160
+ */
161
+ async index(...models) {
162
+ const indexes = generateIndexes(models);
163
+ try {
164
+ await this.dataSource.query("BEGIN");
165
+ for (const index of indexes) {
166
+ await this.dataSource.query(index.query, index.values);
167
+ }
168
+ await this.dataSource.query("COMMIT");
169
+ }
170
+ catch (e) {
171
+ await this.dataSource.query("ROLLBACK");
172
+ throw this.parseError(e);
173
+ }
174
+ }
175
+ /**
176
+ * @description Executes a raw SQL query against the database
177
+ * @summary Abstract method that must be implemented to execute raw SQL queries
178
+ * @template R - The result type
179
+ * @param {TypeORMQuery} q - The query to execute
180
+ * @return {Promise<R>} A promise that resolves to the query result
181
+ */
182
+ async raw(q) {
183
+ const log = this.log.for(this.raw);
184
+ try {
185
+ if (!this.dataSource.isInitialized)
186
+ await this.dataSource.initialize();
187
+ }
188
+ catch (e) {
189
+ throw this.parseError(e);
190
+ }
191
+ try {
192
+ const { query, values } = q;
193
+ log.debug(`executing query: ${query.getSql()}`);
194
+ const response = await this.dataSource.query(query, values);
195
+ return response;
196
+ }
197
+ catch (e) {
198
+ throw this.parseError(e);
199
+ }
200
+ }
201
+ prepare(model, pk, child = false) {
202
+ const prepared = super.prepare(model, pk);
203
+ prepared.record = Object.entries(prepared.record).reduce((accum, [key, value]) => {
204
+ if (key === PersistenceKeys.METADATA || this.isReserved(key))
205
+ return accum;
206
+ if (value === undefined) {
207
+ return accum;
208
+ }
209
+ if (value instanceof Date) {
210
+ value = new Date(value.getTime());
211
+ }
212
+ else if (Model.isModel(value)) {
213
+ value = this.prepare(value, findPrimaryKey(value).id, true).record;
214
+ }
215
+ else {
216
+ switch (typeof value) {
217
+ case "string":
218
+ value = `${value}`;
219
+ break;
220
+ default:
221
+ //do nothing;
222
+ }
223
+ }
224
+ accum[key] = value;
225
+ return accum;
226
+ }, {});
227
+ const constr = Model.get(model.constructor.name);
228
+ if (!constr)
229
+ throw new InternalError(`Model ${model.constructor.name} not found in registry`);
230
+ const result = child
231
+ ? new constr[ModelKeys.ANCHOR]()
232
+ : new constr();
233
+ if (child)
234
+ Object.defineProperty(result, "constructor", {
235
+ configurable: false,
236
+ enumerable: false,
237
+ value: constr[ModelKeys.ANCHOR],
238
+ writable: false,
239
+ });
240
+ Object.entries(prepared.record).forEach(([key, val]) => (result[key] = val));
241
+ prepared.record = result;
242
+ return prepared;
243
+ }
244
+ revert(obj, clazz, pk, id, transient) {
245
+ const log = this.log.for(this.revert);
246
+ if (transient) {
247
+ log.verbose(`re-adding transient properties: ${Object.keys(transient).join(", ")}`);
248
+ Object.entries(transient).forEach(([key, val]) => {
249
+ if (key in obj)
250
+ throw new InternalError(`Transient property ${key} already exists on model ${typeof clazz === "string" ? clazz : clazz.name}. should be impossible`);
251
+ obj[key] = val;
252
+ });
253
+ }
254
+ return new clazz(obj);
255
+ }
256
+ /**
257
+ * @description Creates a new record in the database
258
+ * @summary Abstract method that must be implemented to create a new record
259
+ * @param {string} tableName - The name of the table
260
+ * @param {string|number} id - The ID of the record
261
+ * @param {Record<string, any>} model - The model to create
262
+ * @param {...any[]} args - Additional arguments
263
+ * @return {Promise<Record<string, any>>} A promise that resolves to the created record
264
+ */
265
+ async create(tableName, id, model,
266
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
267
+ ...args) {
268
+ const m = tableName;
269
+ try {
270
+ const repo = this.dataSource.getRepository(m);
271
+ return await repo.save(model);
272
+ }
273
+ catch (e) {
274
+ throw this.parseError(e);
275
+ }
276
+ }
277
+ /**
278
+ * @description Reads a record from the database
279
+ * @summary Abstract method that must be implemented to read a record
280
+ * @param {string} tableName - The name of the table
281
+ * @param {string|number} id - The ID of the record
282
+ * @param {string} pk - primary key colum
283
+ * @return {Promise<Record<string, any>>} A promise that resolves to the read record
284
+ */
285
+ async read(tableName, id, pk) {
286
+ const m = tableName;
287
+ let result;
288
+ try {
289
+ const repo = this.dataSource.getRepository(m);
290
+ const q = {
291
+ where: {
292
+ [pk]: id,
293
+ },
294
+ };
295
+ result = (await repo.findOne(q));
296
+ }
297
+ catch (e) {
298
+ throw this.parseError(e);
299
+ }
300
+ if (!result)
301
+ throw new NotFoundError(`Record with id: ${id} not found in table ${typeof tableName === "string" ? tableName : Repository.table(tableName)}`);
302
+ return result;
303
+ }
304
+ /**
305
+ * @description Updates a record in the database
306
+ * @summary Abstract method that must be implemented to update a record
307
+ * @param {string} tableName - The name of the table
308
+ * @param {string|number} id - The ID of the record
309
+ * @param {Record<string, any>} model - The model to update
310
+ * @param {string} pk - Additional arguments
311
+ * @return A promise that resolves to the updated record
312
+ */
313
+ async update(tableName, id, model,
314
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
315
+ ...args) {
316
+ const m = tableName;
317
+ try {
318
+ const repo = this.dataSource.getRepository(m);
319
+ return repo.save(model);
320
+ }
321
+ catch (e) {
322
+ throw this.parseError(e);
323
+ }
324
+ }
325
+ /**
326
+ * @description Deletes a record from the database
327
+ * @summary Abstract method that must be implemented to delete a record
328
+ * @param {string} tableName - The name of the table
329
+ * @param {string|number} id - The ID of the record
330
+ * @param {string} pk - Additional arguments
331
+ * @return A promise that resolves to the deleted record
332
+ */
333
+ async delete(tableName, id, pk,
334
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
335
+ ...args) {
336
+ const m = tableName;
337
+ try {
338
+ const repo = this.dataSource.getRepository(m);
339
+ const model = await this.read(tableName, id, pk);
340
+ const res = await repo.delete(id);
341
+ return model;
342
+ }
343
+ catch (e) {
344
+ throw this.parseError(e);
345
+ }
346
+ }
347
+ async createAll(tableName, id, model,
348
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
349
+ ...args) {
350
+ const m = tableName;
351
+ try {
352
+ const repo = this.dataSource.getRepository(m);
353
+ const result = await repo.insert(model);
354
+ return this.readAll(tableName, result.identifiers.map((id) => id.id), "id");
355
+ }
356
+ catch (e) {
357
+ throw this.parseError(e);
358
+ }
359
+ }
360
+ async readAll(tableName, id, pk,
361
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
362
+ ...args) {
363
+ if (!id.length)
364
+ return [];
365
+ const m = tableName;
366
+ try {
367
+ const repo = this.dataSource.getRepository(m);
368
+ return repo.findBy({ [pk]: In(id) });
369
+ }
370
+ catch (e) {
371
+ throw this.parseError(e);
372
+ }
373
+ }
374
+ async updateAll(tableName, ids, model, pk, ...args) {
375
+ const result = [];
376
+ for (const m of model) {
377
+ result.push(await this.update(tableName, m[pk], m, ...args));
378
+ }
379
+ return result;
380
+ }
381
+ async deleteAll(tableName, ids, pk,
382
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
383
+ ...args) {
384
+ if (!ids.length)
385
+ return [];
386
+ const m = tableName;
387
+ try {
388
+ const repo = this.dataSource.getRepository(m);
389
+ const models = await this.readAll(tableName, ids, pk);
390
+ await repo.delete(ids);
391
+ return models;
392
+ }
393
+ catch (e) {
394
+ throw this.parseError(e);
395
+ }
396
+ }
397
+ /**
398
+ * @description Parses an error and converts it to a BaseError
399
+ * @summary Converts various error types to appropriate BaseError subtypes
400
+ * @param {Error|string} err - The error to parse
401
+ * @param {string} [reason] - Optional reason for the error
402
+ * @return {BaseError} The parsed error as a BaseError
403
+ */
404
+ parseError(err, reason) {
405
+ return TypeORMAdapter.parseError(err, reason);
406
+ }
407
+ /**
408
+ * @description Checks if an attribute is reserved
409
+ * @summary Determines if an attribute name is reserved in PostgreSQL
410
+ * @param {string} attr - The attribute name to check
411
+ * @return {boolean} True if the attribute is reserved, false otherwise
412
+ */
413
+ isReserved(attr) {
414
+ return !!attr.match(reservedAttributes);
415
+ }
416
+ /**
417
+ * @description Static method to parse an error and convert it to a BaseError
418
+ * @summary Converts various error types to appropriate BaseError subtypes based on PostgreSQL error codes and messages
419
+ * @param {Error|string} err - The error to parse
420
+ * @param {string} [reason] - Optional reason for the error
421
+ * @return {BaseError} The parsed error as a BaseError
422
+ * @mermaid
423
+ * sequenceDiagram
424
+ * participant Caller
425
+ * participant parseError
426
+ * participant ErrorTypes
427
+ *
428
+ * Caller->>parseError: err, reason
429
+ * Note over parseError: Check if err is already a BaseError
430
+ * alt err is BaseError
431
+ * parseError-->>Caller: return err
432
+ * else err is string
433
+ * Note over parseError: Extract code from string
434
+ * alt code matches "duplicate key|already exists"
435
+ * parseError->>ErrorTypes: new ConflictError(code)
436
+ * ErrorTypes-->>Caller: ConflictError
437
+ * else code matches "does not exist|not found"
438
+ * parseError->>ErrorTypes: new NotFoundError(code)
439
+ * ErrorTypes-->>Caller: NotFoundError
440
+ * end
441
+ * else err has code property
442
+ * Note over parseError: Extract code and reason
443
+ * else
444
+ * Note over parseError: Use err.message as code
445
+ * end
446
+ *
447
+ * Note over parseError: Switch on PostgreSQL error code
448
+ * alt code is 23505 (unique_violation)
449
+ * parseError->>ErrorTypes: new ConflictError(reason)
450
+ * ErrorTypes-->>Caller: ConflictError
451
+ * else code is 23503 (foreign_key_violation)
452
+ * parseError->>ErrorTypes: new ConflictError(reason)
453
+ * ErrorTypes-->>Caller: ConflictError
454
+ * else code is 42P01 (undefined_table)
455
+ * parseError->>ErrorTypes: new NotFoundError(reason)
456
+ * ErrorTypes-->>Caller: NotFoundError
457
+ * else code is 42703 (undefined_column)
458
+ * parseError->>ErrorTypes: new NotFoundError(reason)
459
+ * ErrorTypes-->>Caller: NotFoundError
460
+ * else code is 42P07 (duplicate_table)
461
+ * parseError->>ErrorTypes: new ConflictError(reason)
462
+ * ErrorTypes-->>Caller: ConflictError
463
+ * else code is 42P16 (invalid_table_definition)
464
+ * parseError->>ErrorTypes: new IndexError(err)
465
+ * ErrorTypes-->>Caller: IndexError
466
+ * else code matches "ECONNREFUSED"
467
+ * parseError->>ErrorTypes: new ConnectionError(err)
468
+ * ErrorTypes-->>Caller: ConnectionError
469
+ * else
470
+ * parseError->>ErrorTypes: new InternalError(err)
471
+ * ErrorTypes-->>Caller: InternalError
472
+ * end
473
+ */
474
+ static parseError(err, reason) {
475
+ if (err instanceof BaseError)
476
+ return err;
477
+ const code = typeof err === "string" ? err : err.message;
478
+ if (code.match(/duplicate key|already exists/g))
479
+ return new ConflictError(code);
480
+ if (code.match(/does not exist|not found/g))
481
+ return new NotFoundError(code);
482
+ // PostgreSQL error codes: https://www.postgresql.org/docs/current/errcodes-appendix.html
483
+ switch (code.toString()) {
484
+ // Integrity constraint violations
485
+ case "23505": // unique_violation
486
+ case "23503": // foreign_key_violation
487
+ case "42P07": // duplicate_table
488
+ return new ConflictError(reason);
489
+ // Object not found errors
490
+ case "42P01": // undefined_table
491
+ case "42703": // undefined_column
492
+ return new NotFoundError(reason);
493
+ // Invalid object definition
494
+ case "42P16": // invalid_table_definition
495
+ return new IndexError(err);
496
+ // Connection errors
497
+ default:
498
+ if (code.toString().match(/ECONNREFUSED/g))
499
+ return new ConnectionError(err);
500
+ return new InternalError(err);
501
+ }
502
+ }
503
+ static async connect(config) {
504
+ const con = new DataSource(config);
505
+ if (!con.isInitialized)
506
+ await con.initialize();
507
+ return con;
508
+ }
509
+ static async createDatabase(dataSource, dbName) {
510
+ const log = Logging.for(this.createDatabase);
511
+ log.verbose(`Creating database ${dbName}`);
512
+ try {
513
+ await dataSource.query(`CREATE DATABASE ${dbName}`);
514
+ log.info(`Created database ${dbName}`);
515
+ }
516
+ catch (e) {
517
+ throw this.parseError(e);
518
+ }
519
+ }
520
+ static async createNotifyFunction(dataSource, user) {
521
+ const log = Logging.for(this.createNotifyFunction);
522
+ log.verbose(`Creating notify function`);
523
+ try {
524
+ await dataSource.query(`CREATE OR REPLACE FUNCTION notify_table_changes()
525
+ RETURNS trigger AS $$
526
+ BEGIN
527
+ PERFORM pg_notify(
528
+ 'table_changes',
529
+ json_build_object(
530
+ 'table', TG_TABLE_NAME,
531
+ 'action', TG_OP,
532
+ 'data', row_to_json(NEW),
533
+ 'old_data', row_to_json(OLD)
534
+ )::text
535
+ );
536
+ RETURN NEW;
537
+ END;
538
+ $$ LANGUAGE plpgsql SECURITY DEFINER
539
+ ;`);
540
+ await dataSource.query(`ALTER FUNCTION notify_table_changes() OWNER TO ${user};`);
541
+ await dataSource.query(`
542
+ GRANT EXECUTE ON FUNCTION notify_table_changes() TO public;
543
+ `);
544
+ log.info(`Created notify function`);
545
+ }
546
+ catch (e) {
547
+ throw this.parseError(e);
548
+ }
549
+ }
550
+ static async deleteDatabase(dataSource, dbName, user) {
551
+ try {
552
+ if (user)
553
+ await dataSource.query(`DROP OWNED BY ${user} CASCADE;`);
554
+ await dataSource.query(`DROP DATABASE ${dbName}`);
555
+ }
556
+ catch (e) {
557
+ throw this.parseError(e);
558
+ }
559
+ }
560
+ static async createUser(dataSource, dbName, user, password) {
561
+ try {
562
+ await dataSource.query(`CREATE USER ${user} WITH PASSWORD '${password}'`);
563
+ await dataSource.query(`GRANT CONNECT ON DATABASE ${dbName} TO ${user}`);
564
+ await dataSource.query(`GRANT USAGE ON SCHEMA public TO ${user}`);
565
+ await dataSource.query(`GRANT CREATE ON SCHEMA public TO ${user}`);
566
+ await dataSource.query(`GRANT ALL PRIVILEGES ON ALL TABLES IN SCHEMA public TO ${user}`);
567
+ await dataSource.query(`GRANT ALL PRIVILEGES ON ALL SEQUENCES IN SCHEMA public TO ${user}`);
568
+ await dataSource.query(`GRANT ALL PRIVILEGES ON ALL FUNCTIONS IN SCHEMA public TO ${user}`);
569
+ await dataSource.query(`ALTER DEFAULT PRIVILEGES IN SCHEMA public GRANT ALL PRIVILEGES ON TABLES TO ${user}`);
570
+ await dataSource.query(`ALTER DEFAULT PRIVILEGES IN SCHEMA public GRANT ALL PRIVILEGES ON SEQUENCES TO ${user}`);
571
+ }
572
+ catch (e) {
573
+ throw this.parseError(e);
574
+ }
575
+ }
576
+ static async deleteUser(client, user, admin) {
577
+ try {
578
+ await client.query(`REASSIGN OWNED BY ${user} TO ${admin}`);
579
+ await client.query(`REVOKE ALL ON ALL TABLES IN SCHEMA public FROM ${user}`);
580
+ await client.query(`REVOKE ALL ON SCHEMA public FROM ${user}`);
581
+ await client.query(`REVOKE ALL PRIVILEGES ON ALL SEQUENCES IN SCHEMA public FROM ${user}`);
582
+ await client.query(`REVOKE ALL PRIVILEGES ON ALL FUNCTIONS IN SCHEMA public FROM ${user}`);
583
+ await client.query(`ALTER DEFAULT PRIVILEGES FOR ROLE ${admin} IN SCHEMA public REVOKE ALL ON TABLES FROM ${user}`);
584
+ await client.query(`ALTER DEFAULT PRIVILEGES FOR ROLE ${admin} IN SCHEMA public REVOKE ALL ON SEQUENCES FROM ${user};`);
585
+ await client.query(`ALTER DEFAULT PRIVILEGES FOR ROLE ${admin} IN SCHEMA public REVOKE ALL ON FUNCTIONS FROM ${user}`);
586
+ await client.query(`DROP OWNED BY ${user} CASCADE`);
587
+ await client.query(`DROP USER IF EXISTS "${user}"`);
588
+ }
589
+ catch (e) {
590
+ throw this.parseError(e);
591
+ }
592
+ }
593
+ static parseTypeToPostgres(type, isPk, isFk = false) {
594
+ switch (type.toLowerCase()) {
595
+ case "string":
596
+ return isPk ? "TEXT PRIMARY KEY" : isFk ? "TEXT" : "VARCHAR";
597
+ case "number":
598
+ return isPk ? "SERIAL PRIMARY KEY" : "INTEGER";
599
+ case "boolean":
600
+ return "BOOLEAN";
601
+ case "date":
602
+ return "TIMESTAMP";
603
+ case "bigint":
604
+ return isPk ? "BIGINT PRIMARY KEY" : "BIGINT";
605
+ default: {
606
+ const m = Model.get(type);
607
+ if (m) {
608
+ const mm = new m();
609
+ const type = Reflection.getTypeFromDecorator(mm, findPrimaryKey(mm).id);
610
+ return {
611
+ model: m,
612
+ pkType: type,
613
+ };
614
+ }
615
+ throw new InternalError(`Unsupported type: ${type}`);
616
+ }
617
+ }
618
+ }
619
+ static parseValidationToPostgres(prop, type, isPk, key, options) {
620
+ switch (key) {
621
+ case ValidationKeys.REQUIRED:
622
+ return "NOT NULL";
623
+ case ValidationKeys.MAX_LENGTH:
624
+ if (isPk || !options || type.toLowerCase() !== "string") {
625
+ return "";
626
+ }
627
+ return `(${options[ValidationKeys.MAX_LENGTH]})`;
628
+ case ValidationKeys.MIN_LENGTH:
629
+ return `CONSTRAINT ${prop}_min_length_check CHECK (LENGTH(${prop}) >= ${options[ValidationKeys.MIN_LENGTH]})`;
630
+ case ValidationKeys.PATTERN:
631
+ case ValidationKeys.URL:
632
+ case ValidationKeys.EMAIL:
633
+ return `CONSTRAINT ${prop}_pattern_check CHECK (${prop} ~ '${convertJsRegexToPostgres(options[ValidationKeys.PATTERN])}')`;
634
+ case ValidationKeys.TYPE:
635
+ case ValidationKeys.DATE:
636
+ return "";
637
+ case ValidationKeys.MIN:
638
+ return `CONSTRAINT ${prop}_${key}_check CHECK (${prop} >= ${options[ValidationKeys.MIN]})`;
639
+ case ValidationKeys.MAX:
640
+ return `CONSTRAINT ${prop}_${key}_check CHECK (${prop} <= ${options[ValidationKeys.MAX]})`;
641
+ case ValidationKeys.PASSWORD:
642
+ default:
643
+ throw new InternalError(`Unsupported type: ${key}`);
644
+ }
645
+ }
646
+ static parseRelationsToPostgres(prop, clazz, pk, key, options) {
647
+ const tableName = Repository.table(clazz);
648
+ const { cascade } = options;
649
+ const cascadeStr = `${cascade.update ? " ON UPDATE CASCADE" : ""}${cascade.delete ? " ON DELETE CASCADE" : ""}`;
650
+ switch (`relations${key}`) {
651
+ case PersistenceKeys.ONE_TO_ONE:
652
+ return `FOREIGN KEY (${prop}) REFERENCES ${tableName}(${pk})${cascadeStr}`;
653
+ default:
654
+ throw new InternalError(`Unsupported operation: ${key}`);
655
+ }
656
+ }
657
+ static async createTable(client, model) {
658
+ const result = {};
659
+ const m = new model({});
660
+ const tableName = Repository.table(model);
661
+ const { id } = findPrimaryKey(m);
662
+ let isPk, column;
663
+ const properties = Object.getOwnPropertyNames(m);
664
+ for (const prop of properties) {
665
+ if (typeof this[prop] === "function" ||
666
+ prop.toString().startsWith("_") ||
667
+ prop === "constructor") {
668
+ continue;
669
+ }
670
+ isPk = prop === id;
671
+ column = Repository.column(m, prop.toString());
672
+ const allDecs = Reflection.getPropertyDecorators(ValidationKeys.REFLECT, m, prop.toString(), false, true);
673
+ const decoratorData = allDecs.decorators.reduce((accum, el) => {
674
+ const { key, props } = el;
675
+ if (key === ModelKeys.TYPE && !accum[ValidationKeys.TYPE]) {
676
+ accum[ValidationKeys.TYPE] = {
677
+ customTypes: [props.name],
678
+ message: DEFAULT_ERROR_MESSAGES.TYPE,
679
+ description: "defines the accepted types for the attribute",
680
+ };
681
+ }
682
+ else if (key !== ValidationKeys.TYPE) {
683
+ // do nothing. we can only support basis ctypes at this time
684
+ accum[key] = props;
685
+ }
686
+ return accum;
687
+ }, {});
688
+ const dbDecs = Reflection.getPropertyDecorators(Repository.key("relations"), m, prop.toString(), true, true);
689
+ const query = [];
690
+ const constraints = [];
691
+ const foreignKeys = [];
692
+ let typeData = undefined;
693
+ let childClass = undefined;
694
+ let childPk;
695
+ if (Object.keys(decoratorData).length) {
696
+ typeData = decoratorData[ValidationKeys.TYPE];
697
+ if (!typeData) {
698
+ throw new Error(`Missing type information`);
699
+ }
700
+ let parsedType = this.parseTypeToPostgres(typeof typeData.customTypes[0] === "function"
701
+ ? typeData.customTypes[0]()
702
+ : typeData.customTypes[0], isPk);
703
+ if (typeof parsedType === "string") {
704
+ parsedType = { model: parsedType };
705
+ }
706
+ let typeStr = parsedType.model;
707
+ if (typeof typeStr !== "string") {
708
+ if (Array.isArray(typeStr)) {
709
+ console.log(typeStr);
710
+ }
711
+ // continue;
712
+ // const res: Record<string, PostgresTableSpec> = await this.createTable(pool, typeStr);
713
+ try {
714
+ childClass = parsedType.model;
715
+ const m = new childClass();
716
+ childPk = findPrimaryKey(m);
717
+ typeStr = this.parseTypeToPostgres(parsedType.pkType, false, true);
718
+ await this.createTable(client, childClass);
719
+ }
720
+ catch (e) {
721
+ if (!(e instanceof ConflictError))
722
+ throw e;
723
+ }
724
+ }
725
+ let tp = Array.isArray(typeData.customTypes)
726
+ ? typeData.customTypes[0]
727
+ : typeData.customTypes;
728
+ tp = typeof tp === "function" && !tp.name ? tp() : tp;
729
+ const validationStr = this.parseValidationToPostgres(column, tp, isPk, ValidationKeys.MAX_LENGTH, decoratorData[ValidationKeys.MAX_LENGTH] || {
730
+ [ValidationKeys.MAX_LENGTH]: 255,
731
+ });
732
+ const q = `${column} ${typeStr}${validationStr}`;
733
+ if (isPk) {
734
+ query.unshift(q);
735
+ }
736
+ else {
737
+ query.push(q);
738
+ }
739
+ for (const [key, props] of Object.entries(decoratorData).filter(([k]) => ![ValidationKeys.TYPE, ValidationKeys.MAX_LENGTH].includes(k))) {
740
+ const validation = this.parseValidationToPostgres(column, tp, isPk, key, props);
741
+ if (validation.startsWith("CONSTRAINT")) {
742
+ constraints.push(validation);
743
+ }
744
+ else {
745
+ if (validation) {
746
+ query.push(validation);
747
+ }
748
+ }
749
+ }
750
+ }
751
+ // TODO ignore for now. this leaves foreign keys out
752
+ // eslint-disable-next-line no-constant-binary-expression
753
+ if (false || (dbDecs && dbDecs.decorators.length)) {
754
+ if (!typeData)
755
+ throw new Error(`Missing type information`);
756
+ for (const decorator of dbDecs.decorators) {
757
+ const { key, props } = decorator;
758
+ const validation = this.parseRelationsToPostgres(column, childClass, childPk.id, key, props);
759
+ if (validation.startsWith("FOREIGN")) {
760
+ foreignKeys.push(validation);
761
+ }
762
+ else {
763
+ throw new InternalError(`Unsupported relation: ${key}`);
764
+ }
765
+ }
766
+ }
767
+ result[prop.toString()] = {
768
+ query: query.join(" "),
769
+ values: [],
770
+ primaryKey: isPk,
771
+ constraints: constraints,
772
+ foreignKeys: foreignKeys,
773
+ };
774
+ }
775
+ const values = Object.values(result);
776
+ const query = values.map((r) => r.query).join(",\n");
777
+ const constraints = values
778
+ .filter((c) => !!c.constraints.length)
779
+ .map((r) => r.constraints)
780
+ .join(",\n");
781
+ const foreignKeys = values
782
+ .filter((c) => !!c.foreignKeys.length)
783
+ .map((r) => r.foreignKeys)
784
+ .join(",\n");
785
+ const vals = [query, constraints];
786
+ if (foreignKeys) {
787
+ vals.push(foreignKeys);
788
+ }
789
+ const queryString = `CREATE TABLE ${tableName} (${vals.filter((v) => !!v).join(",\n")})`;
790
+ try {
791
+ await client.query(queryString);
792
+ await client.query(`CREATE TRIGGER notify_changes_${tableName}
793
+ AFTER INSERT OR UPDATE OR DELETE ON ${tableName}
794
+ FOR EACH ROW
795
+ EXECUTE FUNCTION notify_table_changes();`);
796
+ }
797
+ catch (e) {
798
+ throw this.parseError(e);
799
+ }
800
+ return result;
801
+ }
802
+ static async getCurrentUser(client) {
803
+ const queryString = `SELECT CURRENT_USER;`;
804
+ try {
805
+ const result = await client.query(queryString);
806
+ return result[0].current_user;
807
+ }
808
+ catch (e) {
809
+ throw this.parseError(e);
810
+ }
811
+ }
812
+ static decoration() {
813
+ // @table() => @Entity()
814
+ const tableKey = Adapter.key(PersistenceKeys.TABLE);
815
+ Decoration.flavouredAs(TypeORMFlavour)
816
+ .for(tableKey)
817
+ .extend((original) => Entity()(original[ModelKeys.ANCHOR] || original))
818
+ .apply();
819
+ // @pk() => @PrimaryGeneratedColumn() | @PrimaryColumn()
820
+ const pkKey = Repository.key(DBKeys.ID);
821
+ function pkDec(options) {
822
+ const decorators = [
823
+ required(),
824
+ readonly(),
825
+ propMetadata(pkKey, options),
826
+ ];
827
+ if (options.type)
828
+ decorators.push(PrimaryGeneratedColumn());
829
+ else
830
+ decorators.push(PrimaryColumn({ unique: true }));
831
+ return apply(...decorators);
832
+ }
833
+ Decoration.flavouredAs(TypeORMFlavour)
834
+ .for(pkKey)
835
+ .define({
836
+ decorator: pkDec,
837
+ })
838
+ .apply();
839
+ // @column("columnName") => @Column({name: "columnName"})
840
+ const columnKey = Adapter.key(PersistenceKeys.COLUMN);
841
+ Decoration.flavouredAs(TypeORMFlavour)
842
+ .for(columnKey)
843
+ .extend({
844
+ decorator: function columm(name) {
845
+ return function column(obj, prop) {
846
+ return Column({
847
+ name: name || prop,
848
+ nullable: true,
849
+ })(obj, prop);
850
+ };
851
+ },
852
+ transform: (args) => {
853
+ const columnName = args[1];
854
+ return [columnName];
855
+ },
856
+ })
857
+ .apply();
858
+ // @unique => @Column({unique: true})
859
+ const uniqueKey = Adapter.key(PersistenceKeys.UNIQUE);
860
+ Decoration.flavouredAs(TypeORMFlavour)
861
+ .for(uniqueKey)
862
+ .define(propMetadata(uniqueKey, {}))
863
+ .extend(Column({ unique: true }))
864
+ .apply();
865
+ // @required => @Column({ nullable: false })
866
+ const requiredKey = Validation.key(ValidationKeys.REQUIRED);
867
+ Decoration.flavouredAs(TypeORMFlavour)
868
+ .for(requiredKey)
869
+ .extend(Column({ nullable: false }))
870
+ .apply();
871
+ // @version => @VersionColumn()
872
+ const versionKey = Repository.key(DBKeys.VERSION);
873
+ Decoration.flavouredAs(TypeORMFlavour)
874
+ .for(versionKey)
875
+ .define(type(Number.name), VersionColumn())
876
+ .apply();
877
+ function ValidationUpdateKey(key) {
878
+ return UpdateValidationKeys.REFLECT + key;
879
+ }
880
+ // @timestamp(op) => @CreateDateColumn() || @UpdateDateColumn()
881
+ const timestampKey = ValidationUpdateKey(DBKeys.TIMESTAMP);
882
+ function ts(operation, format) {
883
+ const decorators = [
884
+ date(format, DB_DEFAULT_ERROR_MESSAGES.TIMESTAMP.DATE),
885
+ required(DB_DEFAULT_ERROR_MESSAGES.TIMESTAMP.REQUIRED),
886
+ propMetadata(Validation.key(DBKeys.TIMESTAMP), {
887
+ operation: operation,
888
+ format: format,
889
+ }),
890
+ ];
891
+ if (operation.indexOf(OperationKeys.UPDATE) !== -1)
892
+ decorators.push(propMetadata(timestampKey, {
893
+ message: DB_DEFAULT_ERROR_MESSAGES.TIMESTAMP.INVALID,
894
+ }));
895
+ else
896
+ decorators.push(readonly());
897
+ return apply(...decorators);
898
+ }
899
+ Decoration.flavouredAs(TypeORMFlavour)
900
+ .for(timestampKey)
901
+ .define({
902
+ decorator: ts,
903
+ })
904
+ .extend({
905
+ decorator: function timestamp(...ops) {
906
+ return function timestamp(obj, prop) {
907
+ if (ops.indexOf(OperationKeys.UPDATE) !== -1)
908
+ return UpdateDateColumn()(obj, prop);
909
+ return CreateDateColumn()(obj, prop);
910
+ };
911
+ },
912
+ transform: (args) => {
913
+ return args[0];
914
+ },
915
+ })
916
+ .apply();
917
+ // @oneToOne(clazz) => @OneToOne(() => clazz)
918
+ const oneToOneKey = Repository.key(PersistenceKeys.ONE_TO_ONE);
919
+ Decoration.flavouredAs(TypeORMFlavour)
920
+ .for(oneToOneKey)
921
+ .define({
922
+ decorator: function oneToOne(clazz, cascade, populate) {
923
+ const metadata = {
924
+ class: (clazz.name ? clazz.name : clazz),
925
+ cascade: cascade,
926
+ populate: populate,
927
+ };
928
+ const ormMeta = {
929
+ cascade: cascade.update === Cascade.CASCADE ||
930
+ cascade.delete === Cascade.CASCADE,
931
+ onDelete: cascade.delete ? "CASCADE" : "DEFAULT",
932
+ onUpdate: cascade.update ? "CASCADE" : "DEFAULT",
933
+ nullable: true,
934
+ eager: populate,
935
+ };
936
+ return apply(prop(PersistenceKeys.RELATIONS), type([
937
+ (typeof clazz === "function" && !clazz.name
938
+ ? clazz
939
+ : clazz.name),
940
+ String.name,
941
+ Number.name,
942
+ BigInt.name,
943
+ ]), propMetadata(oneToOneKey, metadata), OneToOne(() => {
944
+ if (!clazz.name)
945
+ clazz = clazz();
946
+ if (!clazz[ModelKeys.ANCHOR])
947
+ throw new InternalError("Original Model not found in constructor");
948
+ return clazz[ModelKeys.ANCHOR];
949
+ }, (model) => {
950
+ const pk = findPrimaryKey(new clazz()).id;
951
+ return model[pk];
952
+ }, ormMeta), JoinColumn());
953
+ },
954
+ })
955
+ .apply();
956
+ // @oneToMany(clazz) => @OneToMany(() => clazz)
957
+ const oneToManyKey = Repository.key(PersistenceKeys.ONE_TO_MANY);
958
+ Decoration.flavouredAs(TypeORMFlavour)
959
+ .for(oneToManyKey)
960
+ .define({
961
+ decorator: function oneToMany(clazz, cascade, populate) {
962
+ const metadata = {
963
+ class: (clazz.name ? clazz.name : clazz),
964
+ cascade: cascade,
965
+ populate: populate,
966
+ };
967
+ return apply(prop(PersistenceKeys.RELATIONS), list(clazz), propMetadata(oneToManyKey, metadata), function OneToManyWrapper(obj, prop) {
968
+ const ormMeta = {
969
+ cascade: cascade.update === Cascade.CASCADE ||
970
+ cascade.delete === Cascade.CASCADE,
971
+ onDelete: cascade.delete ? "CASCADE" : "DEFAULT",
972
+ onUpdate: cascade.update ? "CASCADE" : "DEFAULT",
973
+ nullable: true,
974
+ eager: populate,
975
+ };
976
+ return OneToMany(() => {
977
+ if (!clazz.name)
978
+ clazz = clazz();
979
+ if (!clazz[ModelKeys.ANCHOR])
980
+ throw new InternalError("Original Model not found in constructor");
981
+ return clazz[ModelKeys.ANCHOR];
982
+ }, (model) => {
983
+ if (!clazz.name)
984
+ clazz = clazz();
985
+ const m = new clazz();
986
+ const crossRelationKey = Object.keys(m).find((k) => {
987
+ const decs = Reflection.getPropertyDecorators(Repository.key(PersistenceKeys.MANY_TO_ONE), m, k, true);
988
+ if (!decs || !decs.decorators || !decs.decorators.length)
989
+ return false;
990
+ const designType = Reflect.getMetadata(ModelKeys.TYPE, m, k);
991
+ if (!designType)
992
+ throw new InternalError(`No Type Definition found for ${k} in ${m.constructor.name}`);
993
+ return designType.name === obj.constructor.name;
994
+ });
995
+ if (!crossRelationKey)
996
+ throw new InternalError(`Cross relation not found. Did you use @manyToOne on the ${clazz.name}?`);
997
+ return model[crossRelationKey];
998
+ }, ormMeta)(obj, prop);
999
+ });
1000
+ },
1001
+ })
1002
+ .apply();
1003
+ // @manyToOne(clazz) => @ManyToOne(() => clazz)
1004
+ const manyToOneKey = Repository.key(PersistenceKeys.MANY_TO_ONE);
1005
+ Decoration.flavouredAs(TypeORMFlavour)
1006
+ .for(manyToOneKey)
1007
+ .define({
1008
+ decorator: function manyToOne(clazz, cascade, populate) {
1009
+ const metadata = {
1010
+ class: (clazz.name ? clazz.name : clazz),
1011
+ cascade: cascade,
1012
+ populate: populate,
1013
+ };
1014
+ const ormMeta = {
1015
+ cascade: cascade.update === Cascade.CASCADE ||
1016
+ cascade.delete === Cascade.CASCADE,
1017
+ onDelete: cascade.delete ? "CASCADE" : "DEFAULT",
1018
+ onUpdate: cascade.update ? "CASCADE" : "DEFAULT",
1019
+ nullable: true,
1020
+ eager: populate,
1021
+ };
1022
+ return apply(prop(PersistenceKeys.RELATIONS), type([
1023
+ (typeof clazz === "function" && !clazz.name
1024
+ ? clazz
1025
+ : clazz.name),
1026
+ String.name,
1027
+ Number.name,
1028
+ BigInt.name,
1029
+ ]), propMetadata(manyToOneKey, metadata), function ManyToOneWrapper(obj, prop) {
1030
+ return ManyToOne(() => {
1031
+ if (!clazz.name)
1032
+ clazz = clazz();
1033
+ if (!clazz[ModelKeys.ANCHOR])
1034
+ throw new InternalError("Original Model not found in constructor");
1035
+ return clazz[ModelKeys.ANCHOR];
1036
+ }, (model) => {
1037
+ if (!clazz.name)
1038
+ clazz = clazz();
1039
+ const m = new clazz();
1040
+ const crossRelationKey = Object.keys(m).find((k) => {
1041
+ const decs = Reflection.getPropertyDecorators(Repository.key(PersistenceKeys.ONE_TO_MANY), m, k, true);
1042
+ if (!decs || !decs.decorators || !decs.decorators.length)
1043
+ return false;
1044
+ const listDec = Reflect.getMetadata(Validation.key(ValidationKeys.LIST), m, k);
1045
+ if (!listDec)
1046
+ throw new InternalError(`No Type Definition found for ${k} in ${m.constructor.name}`);
1047
+ const name = listDec.clazz[0]().name;
1048
+ return name === obj.constructor.name;
1049
+ });
1050
+ if (!crossRelationKey)
1051
+ throw new InternalError(`Cross relation not found. Did you use @manyToOne on the ${clazz.name}?`);
1052
+ return model[crossRelationKey];
1053
+ })(obj, prop);
1054
+ });
1055
+ },
1056
+ })
1057
+ .apply();
1058
+ // @manyToMany(clazz) => @ManyToMany(() => clazz)
1059
+ const manyToManyKey = Repository.key(PersistenceKeys.MANY_TO_MANY);
1060
+ Decoration.flavouredAs(TypeORMFlavour)
1061
+ .for(manyToManyKey)
1062
+ .define({
1063
+ decorator: function manyToMany(clazz, cascade, populate) {
1064
+ const metadata = {
1065
+ class: clazz.name,
1066
+ cascade: cascade,
1067
+ populate: populate,
1068
+ };
1069
+ const ormMeta = {
1070
+ cascade: cascade.update === Cascade.CASCADE ||
1071
+ cascade.delete === Cascade.CASCADE,
1072
+ onDelete: cascade.delete ? "CASCADE" : "DEFAULT",
1073
+ onUpdate: cascade.update ? "CASCADE" : "DEFAULT",
1074
+ nullable: true,
1075
+ eager: populate,
1076
+ };
1077
+ return apply(prop(PersistenceKeys.RELATIONS), list(clazz), propMetadata(manyToManyKey, metadata), ManyToMany(() => {
1078
+ if (!clazz.name)
1079
+ clazz = clazz();
1080
+ if (!clazz[ModelKeys.ANCHOR])
1081
+ throw new InternalError("Original Model not found in constructor");
1082
+ return clazz[ModelKeys.ANCHOR];
1083
+ }, (model) => {
1084
+ if (!clazz.name)
1085
+ clazz = clazz();
1086
+ const pk = findPrimaryKey(new clazz()).id;
1087
+ return model[pk];
1088
+ }, ormMeta), JoinTable());
1089
+ },
1090
+ })
1091
+ .apply();
1092
+ }
1093
+ }
1094
+ __decorate([
1095
+ final(),
1096
+ __metadata("design:type", Function),
1097
+ __metadata("design:paramtypes", []),
1098
+ __metadata("design:returntype", TypeORMDispatch)
1099
+ ], TypeORMAdapter.prototype, "Dispatch", null);
1100
+ __decorate([
1101
+ final(),
1102
+ __metadata("design:type", Function),
1103
+ __metadata("design:paramtypes", []),
1104
+ __metadata("design:returntype", Object)
1105
+ ], TypeORMAdapter.prototype, "repository", null);
1106
+ __decorate([
1107
+ final(),
1108
+ __metadata("design:type", Function),
1109
+ __metadata("design:paramtypes", []),
1110
+ __metadata("design:returntype", TypeORMStatement)
1111
+ ], TypeORMAdapter.prototype, "Statement", null);
1112
+ __decorate([
1113
+ final(),
1114
+ __metadata("design:type", Function),
1115
+ __metadata("design:paramtypes", [Object]),
1116
+ __metadata("design:returntype", Promise)
1117
+ ], TypeORMAdapter.prototype, "Sequence", null);
1118
+ __decorate([
1119
+ final(),
1120
+ __metadata("design:type", Function),
1121
+ __metadata("design:paramtypes", [Object]),
1122
+ __metadata("design:returntype", Promise)
1123
+ ], TypeORMAdapter.prototype, "index", null);
1124
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"TypeORMAdapter.js","sourceRoot":"","sources":["../../src/TypeORMAdapter.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Adapter,\n  Cascade,\n  CascadeMetadata,\n  ConnectionError,\n  final,\n  PersistenceKeys,\n  RelationsMetadata,\n  Repository,\n  Sequence,\n  type SequenceOptions,\n} from \"@decaf-ts/core\";\nimport { reservedAttributes, TypeORMFlavour } from \"./constants\";\nimport {\n  BaseError,\n  ConflictError,\n  Context,\n  DBKeys,\n  DEFAULT_ERROR_MESSAGES as DB_DEFAULT_ERROR_MESSAGES,\n  findPrimaryKey,\n  InternalError,\n  NotFoundError,\n  OperationKeys,\n  readonly,\n  UpdateValidationKeys,\n} from \"@decaf-ts/db-decorators\";\nimport \"reflect-metadata\";\nimport {\n  type Constructor,\n  date,\n  Decoration,\n  DEFAULT_ERROR_MESSAGES,\n  list,\n  MaxLengthValidatorOptions,\n  MaxValidatorOptions,\n  MinLengthValidatorOptions,\n  MinValidatorOptions,\n  Model,\n  ModelKeys,\n  PatternValidatorOptions,\n  prop,\n  propMetadata,\n  required,\n  type,\n  TypeMetadata,\n  Validation,\n  ValidationKeys,\n  ValidatorOptions,\n} from \"@decaf-ts/decorator-validation\";\nimport { IndexError } from \"./errors\";\nimport { TypeORMStatement } from \"./query\";\nimport { TypeORMSequence } from \"./sequences\";\nimport { generateIndexes } from \"./indexes\";\nimport { TypeORMFlags, TypeORMQuery, TypeORMTableSpec } from \"./types\";\nimport { apply, Reflection } from \"@decaf-ts/reflection\";\nimport { TypeORMRepository } from \"./TypeORMRepository\";\nimport { Logging } from \"@decaf-ts/logging\";\nimport { TypeORMDispatch } from \"./TypeORMDispatch\";\nimport { convertJsRegexToPostgres } from \"./utils\";\nimport {\n  DataSource,\n  FindOneOptions,\n  In,\n  InsertResult,\n  RelationOptions,\n  OneToOne,\n  JoinColumn,\n  ManyToMany,\n  SelectQueryBuilder,\n  VersionColumn,\n  OneToMany,\n  ManyToOne,\n  JoinTable,\n} from \"typeorm\";\nimport { DataSourceOptions } from \"typeorm/data-source/DataSourceOptions\";\nimport { Column } from \"./overrides/Column\";\nimport { UpdateDateColumn } from \"./overrides/UpdateDateColumn\";\nimport { CreateDateColumn } from \"./overrides/CreateDateColumn\";\nimport { PrimaryGeneratedColumn } from \"./overrides/PrimaryGeneratedColumn\";\nimport { PrimaryColumn } from \"./overrides/PrimaryColumn\";\nimport { Entity } from \"./overrides/Entity\";\n\nexport async function createdByOnPostgresCreateUpdate<\n  M extends Model,\n  R extends TypeORMRepository<M>,\n  V extends RelationsMetadata,\n>(\n  this: R,\n  context: Context<TypeORMFlags>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  try {\n    const user = context.get(\"user\");\n    model[key] = user as M[typeof key];\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  } catch (e: unknown) {\n    throw new InternalError(\n      \"No User found in context. Please provide a user in the context\"\n    );\n  }\n}\n\n/**\n * @description Adapter for TypeORM-backed persistence operations.\n * @summary Implements the Decaf.ts Adapter over a TypeORM DataSource, providing CRUD operations, query/statement factories, sequence management, error parsing, and decoration helpers.\n * @template Y The native configuration type (TypeORM DataSourceOptions).\n * @template F The repository flags type.\n * @template C The context type.\n * @param {DataSourceOptions} scope The DataSource options for the adapter.\n * @param {string} flavour The flavour of the adapter.\n * @param {string} [alias] Optional alias for the adapter.\n * @class TypeORMAdapter\n * @example\n * const adapter = new TypeORMAdapter({ type: 'postgres', /* ... *\\/ });\n * await adapter.initialize();\n * const repo = new (adapter.repository<User>())(adapter, User);\n * const created = await repo.create(new User({ name: 'Alice' }));\n *\n * @mermaid\n * sequenceDiagram\n *   participant App\n *   participant Adapter as TypeORMAdapter\n *   participant Repo as TypeORMRepository\n *   participant DS as TypeORM DataSource\n *\n *   App->>Adapter: new TypeORMAdapter(opts)\n *   Adapter->>DS: initialize()\n *   App->>Adapter: repository()\n *   Adapter-->>App: TypeORMRepository\n *   App->>Repo: create(model)\n *   Repo->>Adapter: prepare/create/revert\n *   Adapter-->>Repo: Model\n *   Repo-->>App: Model\n */\nexport class TypeORMAdapter extends Adapter<\n  DataSourceOptions,\n  TypeORMQuery,\n  TypeORMFlags,\n  Context<TypeORMFlags>\n> {\n  private _dataSource?: DataSource;\n\n  get dataSource(): DataSource {\n    if (!this._dataSource) {\n      const models = Adapter.models(this.flavour);\n      this._dataSource = new DataSource(\n        Object.assign(this.native, {\n          entities: models.map((c) => c[ModelKeys.ANCHOR as keyof typeof c]),\n        })\n      );\n    }\n    return this._dataSource;\n  }\n  // protected dataSou\n\n  constructor(options: DataSourceOptions, alias?: string) {\n    super(options, TypeORMFlavour, alias);\n  }\n\n  protected override async flags<M extends Model>(\n    operation: OperationKeys,\n    model: Constructor<M>,\n    flags: Partial<TypeORMFlags>\n  ): Promise<TypeORMFlags> {\n    const f = await super.flags(operation, model, flags);\n    const newObj: any = {\n      user: (await TypeORMAdapter.getCurrentUser(this.dataSource)) as string,\n    };\n    const m = new model();\n\n    const exceptions: string[] = [];\n    if (operation === OperationKeys.CREATE) {\n      const pk = findPrimaryKey(m).id;\n      exceptions.push(pk as string);\n    }\n\n    if (\n      operation === OperationKeys.CREATE ||\n      operation === OperationKeys.UPDATE\n    ) {\n      const decs = Object.keys(m).reduce((accum: Record<string, any>, key) => {\n        const decs = Reflection.getPropertyDecorators(\n          ValidationKeys.REFLECT,\n          m,\n          key,\n          true\n        );\n        const dec = decs.decorators.find(\n          (dec: any) =>\n            dec.key === DBKeys.TIMESTAMP &&\n            dec.props.operation.indexOf(operation) !== -1\n        );\n        if (dec) {\n          accum[key] = dec.props;\n        }\n        return accum;\n      }, {});\n\n      exceptions.push(...Object.keys(decs));\n    }\n\n    newObj.ignoredValidationProperties = (\n      f.ignoredValidationProperties ? f.ignoredValidationProperties : []\n    ).concat(...exceptions);\n    return Object.assign(f, newObj) as TypeORMFlags;\n  }\n\n  @final()\n  protected override Dispatch(): TypeORMDispatch {\n    return new TypeORMDispatch();\n  }\n\n  @final()\n  override repository<M extends Model>(): Constructor<TypeORMRepository<M>> {\n    return TypeORMRepository;\n  }\n\n  /**\n   * @description Creates a new Postgres statement for querying\n   * @summary Factory method that creates a new PostgresStatement instance for building queries\n   * @template M - The model type\n   * @return {TypeORMStatement<M, any>} A new PostgresStatement instance\n   */\n  @final()\n  Statement<M extends Model>(): TypeORMStatement<M, any> {\n    return new TypeORMStatement(this);\n  }\n\n  /**\n   * @description Creates a new PostgreSQL sequence\n   * @summary Factory method that creates a new PostgreSQLSequence instance for managing sequences\n   * @param {SequenceOptions} options - The options for the sequence\n   * @return {Promise<Sequence>} A promise that resolves to a new Sequence instance\n   */\n  @final()\n  async Sequence(options: SequenceOptions): Promise<Sequence> {\n    return new TypeORMSequence(options, this);\n  }\n\n  /**\n   * @description Initializes the adapter by creating indexes for all managed models\n   * @summary Sets up the necessary database indexes for all models managed by this adapter\n   * @return {Promise<void>} A promise that resolves when initialization is complete\n   */\n  async initialize(): Promise<void> {\n    const ds = this.dataSource;\n    try {\n      await ds.initialize();\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n    const log = this.log.for(this.initialize);\n    log.verbose(`${this.flavour} adapter initialized`);\n  }\n\n  /**\n   * @description Creates indexes for the given models\n   * @summary Abstract method that must be implemented to create database indexes for the specified models\n   * @template M - The model type\n   * @param {...Constructor<M>} models - The model constructors to create indexes for\n   * @return {Promise<void>} A promise that resolves when all indexes are created\n   */\n  @final()\n  protected async index<M extends Model>(\n    ...models: Constructor<M>[]\n  ): Promise<void> {\n    const indexes: TypeORMQuery[] = generateIndexes(models);\n\n    try {\n      await this.dataSource.query(\"BEGIN\");\n\n      for (const index of indexes) {\n        await this.dataSource.query(index.query, index.values);\n      }\n\n      await this.dataSource.query(\"COMMIT\");\n    } catch (e: unknown) {\n      await this.dataSource.query(\"ROLLBACK\");\n      throw this.parseError(e as Error);\n    }\n  }\n\n  /**\n   * @description Executes a raw SQL query against the database\n   * @summary Abstract method that must be implemented to execute raw SQL queries\n   * @template R - The result type\n   * @param {TypeORMQuery} q - The query to execute\n   * @return {Promise<R>} A promise that resolves to the query result\n   */\n  override async raw<R>(q: TypeORMQuery): Promise<R> {\n    const log = this.log.for(this.raw);\n    try {\n      if (!this.dataSource.isInitialized) await this.dataSource.initialize();\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n    try {\n      const { query, values } = q;\n      log.debug(\n        `executing query: ${(query as unknown as SelectQueryBuilder<any>).getSql()}`\n      );\n      const response = await this.dataSource.query(query, values);\n      return response as R;\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  override prepare<M extends Model>(\n    model: M,\n    pk: keyof M,\n    child = false\n  ): {\n    record: Record<string, any>;\n    id: string;\n    transient?: Record<string, any>;\n  } {\n    const prepared = super.prepare(model, pk);\n\n    prepared.record = Object.entries(prepared.record).reduce(\n      (accum: Record<string, any>, [key, value]) => {\n        if (key === PersistenceKeys.METADATA || this.isReserved(key))\n          return accum;\n        if (value === undefined) {\n          return accum;\n        }\n\n        if (value instanceof Date) {\n          value = new Date(value.getTime());\n        } else if (Model.isModel(value)) {\n          value = this.prepare(value, findPrimaryKey(value).id, true).record;\n        } else {\n          switch (typeof value) {\n            case \"string\":\n              value = `${value}`;\n              break;\n            default:\n            //do nothing;\n          }\n        }\n        accum[key] = value;\n        return accum;\n      },\n      {}\n    );\n    const constr: Constructor<any> | undefined = Model.get(\n      model.constructor.name\n    );\n    if (!constr)\n      throw new InternalError(\n        `Model ${model.constructor.name} not found in registry`\n      );\n    const result = child\n      ? new (constr as any)[ModelKeys.ANCHOR as keyof typeof constr]()\n      : new constr();\n    if (child)\n      Object.defineProperty(result, \"constructor\", {\n        configurable: false,\n        enumerable: false,\n        value: (constr as any)[ModelKeys.ANCHOR as keyof typeof constr],\n        writable: false,\n      });\n    Object.entries(prepared.record).forEach(\n      ([key, val]) => (result[key as keyof typeof result] = val)\n    );\n    prepared.record = result;\n    return prepared;\n  }\n\n  override revert<M extends Model>(\n    obj: Record<string, any>,\n    clazz: string | Constructor<M>,\n    pk: keyof M,\n    id: string | number | bigint,\n    transient?: Record<string, any>\n  ): M {\n    const log = this.log.for(this.revert);\n    if (transient) {\n      log.verbose(\n        `re-adding transient properties: ${Object.keys(transient).join(\", \")}`\n      );\n      Object.entries(transient).forEach(([key, val]) => {\n        if (key in obj)\n          throw new InternalError(\n            `Transient property ${key} already exists on model ${typeof clazz === \"string\" ? clazz : clazz.name}. should be impossible`\n          );\n        (obj as M)[key as keyof M] = val;\n      });\n    }\n\n    return new (clazz as Constructor<M>)(obj);\n  }\n\n  /**\n   * @description Creates a new record in the database\n   * @summary Abstract method that must be implemented to create a new record\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @param {Record<string, any>} model - The model to create\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<Record<string, any>>} A promise that resolves to the created record\n   */\n  override async create(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): Promise<Record<string, any>> {\n    const m: Constructor<Model> = tableName as unknown as Constructor<Model>;\n    try {\n      const repo = this.dataSource.getRepository(m);\n      return await repo.save(model);\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  /**\n   * @description Reads a record from the database\n   * @summary Abstract method that must be implemented to read a record\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @param {string} pk - primary key colum\n   * @return {Promise<Record<string, any>>} A promise that resolves to the read record\n   */\n  override async read(\n    tableName: string,\n    id: string | number,\n    pk: string\n  ): Promise<Record<string, any>> {\n    const m: Constructor<Model> = tableName as unknown as Constructor<Model>;\n    let result: any;\n    try {\n      const repo = this.dataSource.getRepository(m);\n      const q: FindOneOptions = {\n        where: {\n          [pk]: id,\n        },\n      };\n      result = (await repo.findOne(q)) as Record<string, any>;\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n    if (!result)\n      throw new NotFoundError(\n        `Record with id: ${id} not found in table ${typeof tableName === \"string\" ? tableName : Repository.table(tableName)}`\n      );\n    return result;\n  }\n\n  /**\n   * @description Updates a record in the database\n   * @summary Abstract method that must be implemented to update a record\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @param {Record<string, any>} model - The model to update\n   * @param {string} pk - Additional arguments\n   * @return A promise that resolves to the updated record\n   */\n  override async update(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): Promise<Record<string, any>> {\n    const m: Constructor<Model> = tableName as unknown as Constructor<Model>;\n    try {\n      const repo = this.dataSource.getRepository(m);\n      return repo.save(model);\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  /**\n   * @description Deletes a record from the database\n   * @summary Abstract method that must be implemented to delete a record\n   * @param {string} tableName - The name of the table\n   * @param {string|number} id - The ID of the record\n   * @param {string} pk - Additional arguments\n   * @return A promise that resolves to the deleted record\n   */\n  override async delete(\n    tableName: string,\n    id: string | number,\n    pk: string,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): Promise<Record<string, any>> {\n    const m: Constructor<Model> = tableName as unknown as Constructor<Model>;\n    try {\n      const repo = this.dataSource.getRepository(m);\n      const model = await this.read(tableName, id, pk);\n      const res = await repo.delete(id);\n      return model;\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  override async createAll(\n    tableName: string,\n    id: (string | number)[],\n    model: Record<string, any>[],\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    const m: Constructor<Model> = tableName as unknown as Constructor<Model>;\n    try {\n      const repo = this.dataSource.getRepository(m);\n      const result: InsertResult = await repo.insert(model);\n      return this.readAll(\n        tableName,\n        result.identifiers.map((id) => id.id),\n        \"id\"\n      );\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  override async readAll(\n    tableName: string,\n    id: (string | number | bigint)[],\n    pk: string,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    if (!id.length) return [];\n\n    const m: Constructor<Model> = tableName as unknown as Constructor<Model>;\n    try {\n      const repo = this.dataSource.getRepository(m);\n      return repo.findBy({ [pk]: In(id) });\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  override async updateAll(\n    tableName: string,\n    ids: string[] | number[],\n    model: Record<string, any>[],\n    pk: string,\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    const result = [];\n    for (const m of model) {\n      result.push(await this.update(tableName, m[pk], m, ...args));\n    }\n    return result;\n  }\n\n  override async deleteAll(\n    tableName: string,\n    ids: (string | number | bigint)[],\n    pk: string,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    if (!ids.length) return [];\n    const m: Constructor<Model> = tableName as unknown as Constructor<Model>;\n    try {\n      const repo = this.dataSource.getRepository(m);\n      const models = await this.readAll(tableName, ids, pk);\n      await repo.delete(ids);\n      return models;\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  /**\n   * @description Parses an error and converts it to a BaseError\n   * @summary Converts various error types to appropriate BaseError subtypes\n   * @param {Error|string} err - The error to parse\n   * @param {string} [reason] - Optional reason for the error\n   * @return {BaseError} The parsed error as a BaseError\n   */\n  parseError(err: Error | string, reason?: string): BaseError {\n    return TypeORMAdapter.parseError(err, reason);\n  }\n\n  /**\n   * @description Checks if an attribute is reserved\n   * @summary Determines if an attribute name is reserved in PostgreSQL\n   * @param {string} attr - The attribute name to check\n   * @return {boolean} True if the attribute is reserved, false otherwise\n   */\n  protected override isReserved(attr: string): boolean {\n    return !!attr.match(reservedAttributes);\n  }\n\n  /**\n   * @description Static method to parse an error and convert it to a BaseError\n   * @summary Converts various error types to appropriate BaseError subtypes based on PostgreSQL error codes and messages\n   * @param {Error|string} err - The error to parse\n   * @param {string} [reason] - Optional reason for the error\n   * @return {BaseError} The parsed error as a BaseError\n   * @mermaid\n   * sequenceDiagram\n   *   participant Caller\n   *   participant parseError\n   *   participant ErrorTypes\n   *\n   *   Caller->>parseError: err, reason\n   *   Note over parseError: Check if err is already a BaseError\n   *   alt err is BaseError\n   *     parseError-->>Caller: return err\n   *   else err is string\n   *     Note over parseError: Extract code from string\n   *     alt code matches \"duplicate key|already exists\"\n   *       parseError->>ErrorTypes: new ConflictError(code)\n   *       ErrorTypes-->>Caller: ConflictError\n   *     else code matches \"does not exist|not found\"\n   *       parseError->>ErrorTypes: new NotFoundError(code)\n   *       ErrorTypes-->>Caller: NotFoundError\n   *     end\n   *   else err has code property\n   *     Note over parseError: Extract code and reason\n   *   else\n   *     Note over parseError: Use err.message as code\n   *   end\n   *\n   *   Note over parseError: Switch on PostgreSQL error code\n   *   alt code is 23505 (unique_violation)\n   *     parseError->>ErrorTypes: new ConflictError(reason)\n   *     ErrorTypes-->>Caller: ConflictError\n   *   else code is 23503 (foreign_key_violation)\n   *     parseError->>ErrorTypes: new ConflictError(reason)\n   *     ErrorTypes-->>Caller: ConflictError\n   *   else code is 42P01 (undefined_table)\n   *     parseError->>ErrorTypes: new NotFoundError(reason)\n   *     ErrorTypes-->>Caller: NotFoundError\n   *   else code is 42703 (undefined_column)\n   *     parseError->>ErrorTypes: new NotFoundError(reason)\n   *     ErrorTypes-->>Caller: NotFoundError\n   *   else code is 42P07 (duplicate_table)\n   *     parseError->>ErrorTypes: new ConflictError(reason)\n   *     ErrorTypes-->>Caller: ConflictError\n   *   else code is 42P16 (invalid_table_definition)\n   *     parseError->>ErrorTypes: new IndexError(err)\n   *     ErrorTypes-->>Caller: IndexError\n   *   else code matches \"ECONNREFUSED\"\n   *     parseError->>ErrorTypes: new ConnectionError(err)\n   *     ErrorTypes-->>Caller: ConnectionError\n   *   else\n   *     parseError->>ErrorTypes: new InternalError(err)\n   *     ErrorTypes-->>Caller: InternalError\n   *   end\n   */\n  protected static parseError(err: Error | string, reason?: string): BaseError {\n    if (err instanceof BaseError) return err as any;\n    const code: string = typeof err === \"string\" ? err : err.message;\n\n    if (code.match(/duplicate key|already exists/g))\n      return new ConflictError(code);\n    if (code.match(/does not exist|not found/g)) return new NotFoundError(code);\n\n    // PostgreSQL error codes: https://www.postgresql.org/docs/current/errcodes-appendix.html\n    switch (code.toString()) {\n      // Integrity constraint violations\n      case \"23505\": // unique_violation\n      case \"23503\": // foreign_key_violation\n      case \"42P07\": // duplicate_table\n        return new ConflictError(reason as string);\n\n      // Object not found errors\n      case \"42P01\": // undefined_table\n      case \"42703\": // undefined_column\n        return new NotFoundError(reason as string);\n\n      // Invalid object definition\n      case \"42P16\": // invalid_table_definition\n        return new IndexError(err);\n\n      // Connection errors\n      default:\n        if (code.toString().match(/ECONNREFUSED/g))\n          return new ConnectionError(err);\n        return new InternalError(err);\n    }\n  }\n\n  static async connect(config: DataSourceOptions): Promise<DataSource> {\n    const con = new DataSource(config);\n    if (!con.isInitialized) await con.initialize();\n    return con;\n  }\n\n  static async createDatabase(\n    dataSource: DataSource,\n    dbName: string\n  ): Promise<void> {\n    const log = Logging.for(this.createDatabase);\n    log.verbose(`Creating database ${dbName}`);\n    try {\n      await dataSource.query(`CREATE DATABASE ${dbName}`);\n      log.info(`Created database ${dbName}`);\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  static async createNotifyFunction(\n    dataSource: DataSource,\n    user: string\n  ): Promise<void> {\n    const log = Logging.for(this.createNotifyFunction);\n    log.verbose(`Creating notify function`);\n    try {\n      await dataSource.query(\n        `CREATE OR REPLACE FUNCTION notify_table_changes()\nRETURNS trigger AS $$\nBEGIN\n    PERFORM pg_notify(\n        'table_changes',\n        json_build_object(\n            'table', TG_TABLE_NAME,\n            'action', TG_OP,\n            'data', row_to_json(NEW),\n            'old_data', row_to_json(OLD)\n        )::text\n    );\n    RETURN NEW;\nEND;\n$$ LANGUAGE plpgsql SECURITY DEFINER\n;`\n      );\n      await dataSource.query(\n        `ALTER FUNCTION notify_table_changes() OWNER TO ${user};`\n      );\n      await dataSource.query(`\n            GRANT EXECUTE ON FUNCTION notify_table_changes() TO public;\n        `);\n      log.info(`Created notify function`);\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  static async deleteDatabase(\n    dataSource: DataSource,\n    dbName: string,\n    user?: string\n  ): Promise<void> {\n    try {\n      if (user) await dataSource.query(`DROP OWNED BY ${user} CASCADE;`);\n      await dataSource.query(`DROP DATABASE ${dbName}`);\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  static async createUser(\n    dataSource: DataSource,\n    dbName: string,\n    user: string,\n    password: string\n  ): Promise<void> {\n    try {\n      await dataSource.query(`CREATE USER ${user} WITH PASSWORD '${password}'`);\n      await dataSource.query(`GRANT CONNECT ON DATABASE ${dbName} TO ${user}`);\n\n      await dataSource.query(`GRANT USAGE ON SCHEMA public TO ${user}`);\n      await dataSource.query(`GRANT CREATE ON SCHEMA public TO ${user}`);\n      await dataSource.query(\n        `GRANT ALL PRIVILEGES ON ALL TABLES IN SCHEMA public TO ${user}`\n      );\n      await dataSource.query(\n        `GRANT ALL PRIVILEGES ON ALL SEQUENCES IN SCHEMA public TO ${user}`\n      );\n      await dataSource.query(\n        `GRANT ALL PRIVILEGES ON ALL FUNCTIONS IN SCHEMA public TO ${user}`\n      );\n      await dataSource.query(\n        `ALTER DEFAULT PRIVILEGES IN SCHEMA public GRANT ALL PRIVILEGES ON TABLES TO ${user}`\n      );\n      await dataSource.query(\n        `ALTER DEFAULT PRIVILEGES IN SCHEMA public GRANT ALL PRIVILEGES ON SEQUENCES TO ${user}`\n      );\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  static async deleteUser(\n    client: DataSource,\n    user: string,\n    admin: string\n  ): Promise<void> {\n    try {\n      await client.query(`REASSIGN OWNED BY ${user} TO ${admin}`);\n      await client.query(\n        `REVOKE ALL ON ALL TABLES IN SCHEMA public FROM ${user}`\n      );\n      await client.query(`REVOKE ALL ON SCHEMA public FROM ${user}`);\n      await client.query(\n        `REVOKE ALL PRIVILEGES ON ALL SEQUENCES IN SCHEMA public FROM ${user}`\n      );\n      await client.query(\n        `REVOKE ALL PRIVILEGES ON ALL FUNCTIONS IN SCHEMA public FROM ${user}`\n      );\n      await client.query(\n        `ALTER DEFAULT PRIVILEGES FOR ROLE ${admin} IN SCHEMA public REVOKE ALL ON TABLES FROM ${user}`\n      );\n      await client.query(\n        `ALTER DEFAULT PRIVILEGES FOR ROLE ${admin} IN SCHEMA public REVOKE ALL ON SEQUENCES FROM ${user};`\n      );\n      await client.query(\n        `ALTER DEFAULT PRIVILEGES FOR ROLE ${admin} IN SCHEMA public REVOKE ALL ON FUNCTIONS FROM ${user}`\n      );\n      await client.query(`DROP OWNED BY ${user} CASCADE`);\n      await client.query(`DROP USER IF EXISTS \"${user}\"`);\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  private static parseTypeToPostgres(\n    type: string,\n    isPk: boolean,\n    isFk = false\n  ) {\n    switch (type.toLowerCase()) {\n      case \"string\":\n        return isPk ? \"TEXT PRIMARY KEY\" : isFk ? \"TEXT\" : \"VARCHAR\";\n      case \"number\":\n        return isPk ? \"SERIAL PRIMARY KEY\" : \"INTEGER\";\n      case \"boolean\":\n        return \"BOOLEAN\";\n      case \"date\":\n        return \"TIMESTAMP\";\n      case \"bigint\":\n        return isPk ? \"BIGINT PRIMARY KEY\" : \"BIGINT\";\n      default: {\n        const m = Model.get(type);\n        if (m) {\n          const mm = new m();\n          const type = Reflection.getTypeFromDecorator(\n            mm,\n            findPrimaryKey(mm).id\n          );\n          return {\n            model: m,\n            pkType: type,\n          };\n        }\n        throw new InternalError(`Unsupported type: ${type}`);\n      }\n    }\n  }\n\n  private static parseValidationToPostgres(\n    prop: string,\n    type: string,\n    isPk: boolean,\n    key: string,\n    options: ValidatorOptions\n  ) {\n    switch (key) {\n      case ValidationKeys.REQUIRED:\n        return \"NOT NULL\";\n      case ValidationKeys.MAX_LENGTH:\n        if (isPk || !options || type.toLowerCase() !== \"string\") {\n          return \"\";\n        }\n        return `(${(options as MaxLengthValidatorOptions)[ValidationKeys.MAX_LENGTH]})`;\n      case ValidationKeys.MIN_LENGTH:\n        return `CONSTRAINT ${prop}_min_length_check CHECK (LENGTH(${prop}) >= ${(options as MinLengthValidatorOptions)[ValidationKeys.MIN_LENGTH]})`;\n      case ValidationKeys.PATTERN:\n      case ValidationKeys.URL:\n      case ValidationKeys.EMAIL:\n        return `CONSTRAINT ${prop}_pattern_check CHECK (${prop} ~ '${convertJsRegexToPostgres((options as PatternValidatorOptions)[ValidationKeys.PATTERN] as string)}')`;\n      case ValidationKeys.TYPE:\n      case ValidationKeys.DATE:\n        return \"\";\n      case ValidationKeys.MIN:\n        return `CONSTRAINT ${prop}_${key}_check CHECK (${prop} >= ${(options as MinValidatorOptions)[ValidationKeys.MIN]})`;\n      case ValidationKeys.MAX:\n        return `CONSTRAINT ${prop}_${key}_check CHECK (${prop} <= ${(options as MaxValidatorOptions)[ValidationKeys.MAX]})`;\n      case ValidationKeys.PASSWORD:\n      default:\n        throw new InternalError(`Unsupported type: ${key}`);\n    }\n  }\n\n  private static parseRelationsToPostgres(\n    prop: string,\n    clazz: Constructor<Model>,\n    pk: string,\n    key: PersistenceKeys,\n    options: RelationsMetadata\n  ) {\n    const tableName = Repository.table(clazz);\n    const { cascade } = options;\n    const cascadeStr = `${cascade.update ? \" ON UPDATE CASCADE\" : \"\"}${cascade.delete ? \" ON DELETE CASCADE\" : \"\"}`;\n    switch (`relations${key}`) {\n      case PersistenceKeys.ONE_TO_ONE:\n        return `FOREIGN KEY (${prop}) REFERENCES ${tableName}(${pk})${cascadeStr}`;\n      default:\n        throw new InternalError(`Unsupported operation: ${key}`);\n    }\n  }\n\n  static async createTable<M extends Model>(\n    client: DataSource,\n    model: Constructor<M>\n  ): Promise<Record<string, TypeORMTableSpec>> {\n    const result: Record<string, TypeORMTableSpec> = {};\n    const m = new model({});\n    const tableName = Repository.table(model);\n    const { id } = findPrimaryKey(m);\n\n    let isPk: boolean, column: string;\n    const properties = Object.getOwnPropertyNames(m) as (keyof M)[];\n    for (const prop of properties) {\n      if (\n        typeof (this as any)[prop] === \"function\" ||\n        prop.toString().startsWith(\"_\") ||\n        prop === \"constructor\"\n      ) {\n        continue;\n      }\n\n      isPk = prop === id;\n      column = Repository.column(m, prop.toString());\n\n      const allDecs = Reflection.getPropertyDecorators(\n        ValidationKeys.REFLECT,\n        m,\n        prop.toString(),\n        false,\n        true\n      );\n\n      const decoratorData = allDecs.decorators.reduce(\n        (accum: Record<string, any>, el) => {\n          const { key, props } = el;\n          if (key === ModelKeys.TYPE && !accum[ValidationKeys.TYPE]) {\n            accum[ValidationKeys.TYPE] = {\n              customTypes: [props.name as string],\n              message: DEFAULT_ERROR_MESSAGES.TYPE,\n              description: \"defines the accepted types for the attribute\",\n            };\n          } else if (key !== ValidationKeys.TYPE) {\n            // do nothing. we can only support basis ctypes at this time\n            accum[key] = props;\n          }\n          return accum;\n        },\n        {}\n      );\n\n      const dbDecs = Reflection.getPropertyDecorators(\n        Repository.key(\"relations\"),\n        m,\n        prop.toString(),\n        true,\n        true\n      );\n\n      const query: string[] = [];\n      const constraints: string[] = [];\n      const foreignKeys: string[] = [];\n      let typeData: TypeMetadata | undefined = undefined;\n      let childClass: Constructor<Model> | undefined = undefined;\n      let childPk: any;\n\n      if (Object.keys(decoratorData).length) {\n        typeData = decoratorData[ValidationKeys.TYPE] as TypeMetadata;\n\n        if (!typeData) {\n          throw new Error(`Missing type information`);\n        }\n\n        let parsedType:\n          | string\n          | { model: Constructor<Model> | string; pkType?: string } =\n          this.parseTypeToPostgres(\n            typeof (typeData.customTypes as any[])[0] === \"function\"\n              ? (typeData.customTypes as any)[0]()\n              : (typeData.customTypes as any)[0],\n            isPk\n          );\n        if (typeof parsedType === \"string\") {\n          parsedType = { model: parsedType };\n        }\n        let typeStr:\n          | string\n          | { model: Constructor<Model> | string; pkType?: string } =\n          parsedType.model as\n            | string\n            | { model: Constructor<Model> | string; pkType?: string };\n\n        if (typeof typeStr !== \"string\") {\n          if (Array.isArray(typeStr)) {\n            console.log(typeStr);\n          }\n\n          // continue;\n          // const res: Record<string, PostgresTableSpec> = await this.createTable(pool, typeStr);\n          try {\n            childClass = parsedType.model as Constructor<Model>;\n            const m = new childClass();\n            childPk = findPrimaryKey(m);\n            typeStr = this.parseTypeToPostgres(\n              parsedType.pkType as string,\n              false,\n              true\n            );\n            await this.createTable(client, childClass);\n          } catch (e: unknown) {\n            if (!(e instanceof ConflictError)) throw e;\n          }\n        }\n\n        let tp = Array.isArray(typeData.customTypes)\n          ? typeData.customTypes[0]\n          : typeData.customTypes;\n        tp = typeof tp === \"function\" && !tp.name ? tp() : tp;\n        const validationStr = this.parseValidationToPostgres(\n          column,\n          tp as any,\n          isPk,\n          ValidationKeys.MAX_LENGTH,\n          (decoratorData[\n            ValidationKeys.MAX_LENGTH\n          ] as MaxLengthValidatorOptions) || {\n            [ValidationKeys.MAX_LENGTH]: 255,\n          }\n        );\n\n        const q = `${column} ${typeStr}${validationStr}`;\n\n        if (isPk) {\n          query.unshift(q);\n        } else {\n          query.push(q);\n        }\n\n        for (const [key, props] of Object.entries(decoratorData).filter(\n          ([k]) =>\n            ![ValidationKeys.TYPE, ValidationKeys.MAX_LENGTH].includes(k as any)\n        )) {\n          const validation = this.parseValidationToPostgres(\n            column,\n            tp as any,\n            isPk,\n            key,\n            props\n          );\n          if (validation.startsWith(\"CONSTRAINT\")) {\n            constraints.push(validation);\n          } else {\n            if (validation) {\n              query.push(validation);\n            }\n          }\n        }\n      }\n\n      // TODO ignore for now. this leaves foreign keys out\n      // eslint-disable-next-line no-constant-binary-expression\n      if (false || (dbDecs && dbDecs.decorators.length)) {\n        if (!typeData) throw new Error(`Missing type information`);\n        for (const decorator of dbDecs.decorators) {\n          const { key, props } = decorator;\n          const validation = this.parseRelationsToPostgres(\n            column,\n            childClass as Constructor<Model>,\n            childPk.id,\n            key as PersistenceKeys,\n            props as unknown as RelationsMetadata\n          );\n          if (validation.startsWith(\"FOREIGN\")) {\n            foreignKeys.push(validation);\n          } else {\n            throw new InternalError(`Unsupported relation: ${key}`);\n          }\n        }\n      }\n\n      result[prop.toString()] = {\n        query: query.join(\" \"),\n        values: [],\n        primaryKey: isPk,\n        constraints: constraints,\n        foreignKeys: foreignKeys,\n      };\n    }\n\n    const values = Object.values(result);\n    const query = values.map((r) => r.query).join(\",\\n\");\n    const constraints = values\n      .filter((c) => !!c.constraints.length)\n      .map((r) => r.constraints)\n      .join(\",\\n\");\n    const foreignKeys = values\n      .filter((c) => !!c.foreignKeys.length)\n      .map((r) => r.foreignKeys)\n      .join(\",\\n\");\n    const vals = [query, constraints];\n    if (foreignKeys) {\n      vals.push(foreignKeys);\n    }\n    const queryString = `CREATE TABLE ${tableName} (${vals.filter((v) => !!v).join(\",\\n\")})`;\n    try {\n      await client.query(queryString);\n      await client.query(\n        `CREATE TRIGGER notify_changes_${tableName}\nAFTER INSERT OR UPDATE OR DELETE ON ${tableName}\n    FOR EACH ROW\n    EXECUTE FUNCTION notify_table_changes();`\n      );\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n    return result;\n  }\n\n  static async getCurrentUser(client: DataSource): Promise<string> {\n    const queryString = `SELECT CURRENT_USER;`;\n    try {\n      const result = await client.query(queryString);\n      return result[0].current_user;\n    } catch (e: unknown) {\n      throw this.parseError(e as Error);\n    }\n  }\n\n  static decoration() {\n    // @table() => @Entity()\n    const tableKey = Adapter.key(PersistenceKeys.TABLE);\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(tableKey)\n      .extend((original: any) =>\n        Entity()(original[ModelKeys.ANCHOR] || original)\n      )\n      .apply();\n\n    // @pk() => @PrimaryGeneratedColumn() | @PrimaryColumn()\n    const pkKey = Repository.key(DBKeys.ID);\n\n    function pkDec(options: SequenceOptions) {\n      const decorators: any[] = [\n        required(),\n        readonly(),\n        propMetadata(pkKey, options),\n      ];\n      if (options.type) decorators.push(PrimaryGeneratedColumn());\n      else decorators.push(PrimaryColumn({ unique: true }));\n      return apply(...decorators);\n    }\n\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(pkKey)\n      .define({\n        decorator: pkDec,\n      })\n      .apply();\n\n    // @column(\"columnName\") => @Column({name: \"columnName\"})\n    const columnKey = Adapter.key(PersistenceKeys.COLUMN);\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(columnKey)\n      .extend({\n        decorator: function columm(name: string) {\n          return function column(obj: any, prop: any) {\n            return Column({\n              name: name || prop,\n              nullable: true,\n            })(obj, prop);\n          };\n        },\n        transform: (args: any[]) => {\n          const columnName = args[1];\n          return [columnName];\n        },\n      })\n      .apply();\n\n    // @unique => @Column({unique: true})\n    const uniqueKey = Adapter.key(PersistenceKeys.UNIQUE);\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(uniqueKey)\n      .define(propMetadata(uniqueKey, {}))\n      .extend(Column({ unique: true }))\n      .apply();\n\n    // @required => @Column({ nullable: false })\n    const requiredKey = Validation.key(ValidationKeys.REQUIRED);\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(requiredKey)\n      .extend(Column({ nullable: false }))\n      .apply();\n\n    // @version => @VersionColumn()\n    const versionKey = Repository.key(DBKeys.VERSION);\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(versionKey)\n      .define(type(Number.name), VersionColumn())\n      .apply();\n\n    function ValidationUpdateKey(key: string) {\n      return UpdateValidationKeys.REFLECT + key;\n    }\n\n    // @timestamp(op) => @CreateDateColumn() || @UpdateDateColumn()\n    const timestampKey = ValidationUpdateKey(DBKeys.TIMESTAMP);\n\n    function ts(operation: OperationKeys[], format: string) {\n      const decorators: any[] = [\n        date(format, DB_DEFAULT_ERROR_MESSAGES.TIMESTAMP.DATE),\n        required(DB_DEFAULT_ERROR_MESSAGES.TIMESTAMP.REQUIRED),\n        propMetadata(Validation.key(DBKeys.TIMESTAMP), {\n          operation: operation,\n          format: format,\n        }),\n      ];\n      if (operation.indexOf(OperationKeys.UPDATE) !== -1)\n        decorators.push(\n          propMetadata(timestampKey, {\n            message: DB_DEFAULT_ERROR_MESSAGES.TIMESTAMP.INVALID,\n          })\n        );\n      else decorators.push(readonly());\n      return apply(...decorators);\n    }\n\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(timestampKey)\n      .define({\n        decorator: ts,\n      })\n      .extend({\n        decorator: function timestamp(...ops: OperationKeys[]) {\n          return function timestamp(obj: any, prop: any) {\n            if (ops.indexOf(OperationKeys.UPDATE) !== -1)\n              return UpdateDateColumn()(obj, prop);\n            return CreateDateColumn()(obj, prop);\n          };\n        },\n        transform: (args: any[]) => {\n          return args[0];\n        },\n      })\n      .apply();\n\n    // @oneToOne(clazz) => @OneToOne(() => clazz)\n    const oneToOneKey = Repository.key(PersistenceKeys.ONE_TO_ONE);\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(oneToOneKey)\n      .define({\n        decorator: function oneToOne(\n          clazz: Constructor<any> | (() => Constructor<any>),\n          cascade: CascadeMetadata,\n          populate: boolean\n        ) {\n          const metadata: RelationsMetadata = {\n            class: (clazz.name ? clazz.name : clazz) as string,\n            cascade: cascade,\n            populate: populate,\n          };\n          const ormMeta: RelationOptions = {\n            cascade:\n              cascade.update === Cascade.CASCADE ||\n              cascade.delete === Cascade.CASCADE,\n            onDelete: cascade.delete ? \"CASCADE\" : \"DEFAULT\",\n            onUpdate: cascade.update ? \"CASCADE\" : \"DEFAULT\",\n            nullable: true,\n            eager: populate,\n          };\n          return apply(\n            prop(PersistenceKeys.RELATIONS),\n            type([\n              (typeof clazz === \"function\" && !clazz.name\n                ? clazz\n                : clazz.name) as any,\n              String.name,\n              Number.name,\n              BigInt.name,\n            ]),\n            propMetadata(oneToOneKey, metadata),\n            OneToOne(\n              () => {\n                if (!clazz.name) clazz = (clazz as any)();\n                if (!clazz[ModelKeys.ANCHOR as keyof typeof clazz])\n                  throw new InternalError(\n                    \"Original Model not found in constructor\"\n                  );\n                return clazz[ModelKeys.ANCHOR as keyof typeof clazz];\n              },\n              (model: any) => {\n                const pk = findPrimaryKey(new (clazz as Constructor<any>)()).id;\n                return model[pk];\n              },\n              ormMeta\n            ),\n            JoinColumn()\n          );\n        },\n      })\n      .apply();\n\n    // @oneToMany(clazz) => @OneToMany(() => clazz)\n    const oneToManyKey = Repository.key(PersistenceKeys.ONE_TO_MANY);\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(oneToManyKey)\n      .define({\n        decorator: function oneToMany(\n          clazz: Constructor<any> | (() => Constructor<any>),\n          cascade: CascadeMetadata,\n          populate: boolean\n        ) {\n          const metadata: RelationsMetadata = {\n            class: (clazz.name ? clazz.name : clazz) as string,\n            cascade: cascade,\n            populate: populate,\n          };\n          return apply(\n            prop(PersistenceKeys.RELATIONS),\n            list(clazz),\n            propMetadata(oneToManyKey, metadata),\n            function OneToManyWrapper(obj: any, prop: any): any {\n              const ormMeta: RelationOptions = {\n                cascade:\n                  cascade.update === Cascade.CASCADE ||\n                  cascade.delete === Cascade.CASCADE,\n                onDelete: cascade.delete ? \"CASCADE\" : \"DEFAULT\",\n                onUpdate: cascade.update ? \"CASCADE\" : \"DEFAULT\",\n                nullable: true,\n                eager: populate,\n              };\n              return OneToMany(\n                () => {\n                  if (!clazz.name) clazz = (clazz as any)();\n                  if (!clazz[ModelKeys.ANCHOR as keyof typeof clazz])\n                    throw new InternalError(\n                      \"Original Model not found in constructor\"\n                    );\n                  return clazz[ModelKeys.ANCHOR as keyof typeof clazz];\n                },\n                (model: any) => {\n                  if (!clazz.name) clazz = (clazz as any)();\n                  const m = new (clazz as Constructor<any>)();\n                  const crossRelationKey = Object.keys(m).find((k) => {\n                    const decs = Reflection.getPropertyDecorators(\n                      Repository.key(PersistenceKeys.MANY_TO_ONE),\n                      m,\n                      k,\n                      true\n                    );\n                    if (!decs || !decs.decorators || !decs.decorators.length)\n                      return false;\n                    const designType = Reflect.getMetadata(\n                      ModelKeys.TYPE,\n                      m,\n                      k\n                    );\n                    if (!designType)\n                      throw new InternalError(\n                        `No Type Definition found for ${k} in ${m.constructor.name}`\n                      );\n                    return designType.name === obj.constructor.name;\n                  });\n                  if (!crossRelationKey)\n                    throw new InternalError(\n                      `Cross relation not found. Did you use @manyToOne on the ${clazz.name}?`\n                    );\n                  return model[crossRelationKey];\n                },\n                ormMeta\n              )(obj, prop);\n            }\n          );\n        },\n      })\n      .apply();\n\n    // @manyToOne(clazz) => @ManyToOne(() => clazz)\n    const manyToOneKey = Repository.key(PersistenceKeys.MANY_TO_ONE);\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(manyToOneKey)\n      .define({\n        decorator: function manyToOne(\n          clazz: Constructor<any> | (() => Constructor<any>),\n          cascade: CascadeMetadata,\n          populate: boolean\n        ) {\n          const metadata: RelationsMetadata = {\n            class: (clazz.name ? clazz.name : clazz) as string,\n            cascade: cascade,\n            populate: populate,\n          };\n          const ormMeta: RelationOptions = {\n            cascade:\n              cascade.update === Cascade.CASCADE ||\n              cascade.delete === Cascade.CASCADE,\n            onDelete: cascade.delete ? \"CASCADE\" : \"DEFAULT\",\n            onUpdate: cascade.update ? \"CASCADE\" : \"DEFAULT\",\n            nullable: true,\n            eager: populate,\n          };\n          return apply(\n            prop(PersistenceKeys.RELATIONS),\n            type([\n              (typeof clazz === \"function\" && !clazz.name\n                ? clazz\n                : clazz.name) as any,\n              String.name,\n              Number.name,\n              BigInt.name,\n            ]),\n            propMetadata(manyToOneKey, metadata),\n            function ManyToOneWrapper(obj: any, prop: any): any {\n              return ManyToOne(\n                () => {\n                  if (!clazz.name) clazz = (clazz as any)();\n                  if (!clazz[ModelKeys.ANCHOR as keyof typeof clazz])\n                    throw new InternalError(\n                      \"Original Model not found in constructor\"\n                    );\n                  return clazz[ModelKeys.ANCHOR as keyof typeof clazz];\n                },\n                (model: any) => {\n                  if (!clazz.name) clazz = (clazz as any)();\n                  const m = new (clazz as Constructor<any>)();\n                  const crossRelationKey = Object.keys(m).find((k) => {\n                    const decs = Reflection.getPropertyDecorators(\n                      Repository.key(PersistenceKeys.ONE_TO_MANY),\n                      m,\n                      k,\n                      true\n                    );\n                    if (!decs || !decs.decorators || !decs.decorators.length)\n                      return false;\n                    const listDec = Reflect.getMetadata(\n                      Validation.key(ValidationKeys.LIST),\n                      m,\n                      k\n                    );\n                    if (!listDec)\n                      throw new InternalError(\n                        `No Type Definition found for ${k} in ${m.constructor.name}`\n                      );\n                    const name = listDec.clazz[0]().name;\n                    return name === obj.constructor.name;\n                  });\n                  if (!crossRelationKey)\n                    throw new InternalError(\n                      `Cross relation not found. Did you use @manyToOne on the ${clazz.name}?`\n                    );\n                  return model[crossRelationKey];\n                }\n              )(obj, prop);\n            }\n          );\n        },\n      })\n      .apply();\n\n    // @manyToMany(clazz) => @ManyToMany(() => clazz)\n    const manyToManyKey = Repository.key(PersistenceKeys.MANY_TO_MANY);\n    Decoration.flavouredAs(TypeORMFlavour)\n      .for(manyToManyKey)\n      .define({\n        decorator: function manyToMany(\n          clazz: Constructor<any> | (() => Constructor<any>),\n          cascade: CascadeMetadata,\n          populate: boolean\n        ) {\n          const metadata: RelationsMetadata = {\n            class: clazz.name,\n            cascade: cascade,\n            populate: populate,\n          };\n          const ormMeta: RelationOptions = {\n            cascade:\n              cascade.update === Cascade.CASCADE ||\n              cascade.delete === Cascade.CASCADE,\n            onDelete: cascade.delete ? \"CASCADE\" : \"DEFAULT\",\n            onUpdate: cascade.update ? \"CASCADE\" : \"DEFAULT\",\n            nullable: true,\n            eager: populate,\n          };\n          return apply(\n            prop(PersistenceKeys.RELATIONS),\n            list(clazz),\n            propMetadata(manyToManyKey, metadata),\n            ManyToMany(\n              () => {\n                if (!clazz.name) clazz = (clazz as any)();\n                if (!clazz[ModelKeys.ANCHOR as keyof typeof clazz])\n                  throw new InternalError(\n                    \"Original Model not found in constructor\"\n                  );\n                return clazz[ModelKeys.ANCHOR as keyof typeof clazz];\n              },\n              (model: any) => {\n                if (!clazz.name) clazz = (clazz as any)();\n                const pk = findPrimaryKey(new (clazz as Constructor<any>)()).id;\n                return model[pk];\n              },\n              ormMeta\n            ),\n            JoinTable()\n          );\n        },\n      })\n      .apply();\n  }\n}\n"]}