@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.
- package/LICENSE.md +21 -0
- package/README.md +93 -0
- package/dist/for-typeorm.cjs +2553 -0
- package/dist/for-typeorm.esm.cjs +2538 -0
- package/lib/TypeORMAdapter.cjs +1129 -0
- package/lib/TypeORMAdapter.d.ts +221 -0
- package/lib/TypeORMDispatch.cjs +134 -0
- package/lib/TypeORMDispatch.d.ts +87 -0
- package/lib/TypeORMEventSubscriber.cjs +96 -0
- package/lib/TypeORMEventSubscriber.d.ts +56 -0
- package/lib/TypeORMRepository.cjs +209 -0
- package/lib/TypeORMRepository.d.ts +125 -0
- package/lib/constants.cjs +43 -0
- package/lib/constants.d.ts +39 -0
- package/lib/errors.cjs +28 -0
- package/lib/errors.d.ts +21 -0
- package/lib/esm/TypeORMAdapter.d.ts +221 -0
- package/lib/esm/TypeORMAdapter.js +1124 -0
- package/lib/esm/TypeORMDispatch.d.ts +87 -0
- package/lib/esm/TypeORMDispatch.js +130 -0
- package/lib/esm/TypeORMEventSubscriber.d.ts +56 -0
- package/lib/esm/TypeORMEventSubscriber.js +93 -0
- package/lib/esm/TypeORMRepository.d.ts +125 -0
- package/lib/esm/TypeORMRepository.js +206 -0
- package/lib/esm/constants.d.ts +39 -0
- package/lib/esm/constants.js +40 -0
- package/lib/esm/errors.d.ts +21 -0
- package/lib/esm/errors.js +24 -0
- package/lib/esm/index.d.ts +22 -0
- package/lib/esm/index.js +25 -0
- package/lib/esm/indexes/generator.d.ts +50 -0
- package/lib/esm/indexes/generator.js +95 -0
- package/lib/esm/indexes/index.d.ts +1 -0
- package/lib/esm/indexes/index.js +2 -0
- package/lib/esm/overrides/Column.d.ts +74 -0
- package/lib/esm/overrides/Column.js +70 -0
- package/lib/esm/overrides/CreateDateColumn.d.ts +2 -0
- package/lib/esm/overrides/CreateDateColumn.js +9 -0
- package/lib/esm/overrides/Entity.d.ts +11 -0
- package/lib/esm/overrides/Entity.js +28 -0
- package/lib/esm/overrides/PrimaryColumn.d.ts +20 -0
- package/lib/esm/overrides/PrimaryColumn.js +53 -0
- package/lib/esm/overrides/PrimaryGeneratedColumn.d.ts +24 -0
- package/lib/esm/overrides/PrimaryGeneratedColumn.js +51 -0
- package/lib/esm/overrides/UpdateDateColumn.d.ts +2 -0
- package/lib/esm/overrides/UpdateDateColumn.js +9 -0
- package/lib/esm/overrides/utils.d.ts +2 -0
- package/lib/esm/overrides/utils.js +29 -0
- package/lib/esm/query/Paginator.d.ts +86 -0
- package/lib/esm/query/Paginator.js +124 -0
- package/lib/esm/query/Statement.d.ts +131 -0
- package/lib/esm/query/Statement.js +242 -0
- package/lib/esm/query/constants.d.ts +52 -0
- package/lib/esm/query/constants.js +74 -0
- package/lib/esm/query/index.d.ts +4 -0
- package/lib/esm/query/index.js +5 -0
- package/lib/esm/query/translate.d.ts +34 -0
- package/lib/esm/query/translate.js +42 -0
- package/lib/esm/raw/postgres.d.ts +36 -0
- package/lib/esm/raw/postgres.js +2 -0
- package/lib/esm/sequences/Sequence.d.ts +67 -0
- package/lib/esm/sequences/Sequence.js +117 -0
- package/lib/esm/sequences/index.d.ts +1 -0
- package/lib/esm/sequences/index.js +2 -0
- package/lib/esm/types.d.ts +67 -0
- package/lib/esm/types.js +28 -0
- package/lib/esm/utils.d.ts +16 -0
- package/lib/esm/utils.js +29 -0
- package/lib/index.cjs +42 -0
- package/lib/index.d.ts +22 -0
- package/lib/indexes/generator.cjs +98 -0
- package/lib/indexes/generator.d.ts +50 -0
- package/lib/indexes/index.cjs +18 -0
- package/lib/indexes/index.d.ts +1 -0
- package/lib/overrides/Column.cjs +73 -0
- package/lib/overrides/Column.d.ts +74 -0
- package/lib/overrides/CreateDateColumn.cjs +12 -0
- package/lib/overrides/CreateDateColumn.d.ts +2 -0
- package/lib/overrides/Entity.cjs +31 -0
- package/lib/overrides/Entity.d.ts +11 -0
- package/lib/overrides/PrimaryColumn.cjs +56 -0
- package/lib/overrides/PrimaryColumn.d.ts +20 -0
- package/lib/overrides/PrimaryGeneratedColumn.cjs +54 -0
- package/lib/overrides/PrimaryGeneratedColumn.d.ts +24 -0
- package/lib/overrides/UpdateDateColumn.cjs +12 -0
- package/lib/overrides/UpdateDateColumn.d.ts +2 -0
- package/lib/overrides/utils.cjs +32 -0
- package/lib/overrides/utils.d.ts +2 -0
- package/lib/query/Paginator.cjs +128 -0
- package/lib/query/Paginator.d.ts +86 -0
- package/lib/query/Statement.cjs +246 -0
- package/lib/query/Statement.d.ts +131 -0
- package/lib/query/constants.cjs +77 -0
- package/lib/query/constants.d.ts +52 -0
- package/lib/query/index.cjs +21 -0
- package/lib/query/index.d.ts +4 -0
- package/lib/query/translate.cjs +45 -0
- package/lib/query/translate.d.ts +34 -0
- package/lib/raw/postgres.cjs +3 -0
- package/lib/raw/postgres.d.ts +36 -0
- package/lib/sequences/Sequence.cjs +121 -0
- package/lib/sequences/Sequence.d.ts +67 -0
- package/lib/sequences/index.cjs +18 -0
- package/lib/sequences/index.d.ts +1 -0
- package/lib/types.cjs +31 -0
- package/lib/types.d.ts +67 -0
- package/lib/utils.cjs +32 -0
- package/lib/utils.d.ts +16 -0
- 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"]}
|