@decaf-ts/core 0.5.1 → 0.5.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (197) hide show
  1. package/LICENSE.md +21 -157
  2. package/README.md +652 -15
  3. package/dist/core.cjs +2110 -132
  4. package/dist/core.esm.cjs +2111 -133
  5. package/lib/esm/identity/decorators.d.ts +52 -7
  6. package/lib/esm/identity/decorators.js +53 -8
  7. package/lib/esm/identity/utils.d.ts +19 -0
  8. package/lib/esm/identity/utils.js +20 -1
  9. package/lib/esm/index.d.ts +9 -2
  10. package/lib/esm/index.js +10 -3
  11. package/lib/esm/interfaces/ErrorParser.d.ts +12 -0
  12. package/lib/esm/interfaces/ErrorParser.js +1 -1
  13. package/lib/esm/interfaces/Executor.d.ts +13 -0
  14. package/lib/esm/interfaces/Executor.js +1 -1
  15. package/lib/esm/interfaces/Observable.d.ts +27 -0
  16. package/lib/esm/interfaces/Observable.js +1 -1
  17. package/lib/esm/interfaces/Observer.d.ts +12 -0
  18. package/lib/esm/interfaces/Observer.js +1 -1
  19. package/lib/esm/interfaces/Paginatable.d.ts +15 -0
  20. package/lib/esm/interfaces/Paginatable.js +1 -1
  21. package/lib/esm/interfaces/Queriable.d.ts +34 -9
  22. package/lib/esm/interfaces/Queriable.js +1 -1
  23. package/lib/esm/interfaces/RawExecutor.d.ts +14 -0
  24. package/lib/esm/interfaces/RawExecutor.js +1 -1
  25. package/lib/esm/interfaces/SequenceOptions.d.ts +52 -0
  26. package/lib/esm/interfaces/SequenceOptions.js +19 -1
  27. package/lib/esm/model/BaseModel.d.ts +31 -0
  28. package/lib/esm/model/BaseModel.js +24 -1
  29. package/lib/esm/model/construction.d.ts +433 -0
  30. package/lib/esm/model/construction.js +441 -2
  31. package/lib/esm/model/decorators.d.ts +159 -29
  32. package/lib/esm/model/decorators.js +160 -30
  33. package/lib/esm/model/types.d.ts +9 -0
  34. package/lib/esm/model/types.js +1 -1
  35. package/lib/esm/persistence/Adapter.d.ts +358 -17
  36. package/lib/esm/persistence/Adapter.js +287 -19
  37. package/lib/esm/persistence/Dispatch.d.ts +114 -1
  38. package/lib/esm/persistence/Dispatch.js +102 -4
  39. package/lib/esm/persistence/ObserverHandler.d.ts +95 -0
  40. package/lib/esm/persistence/ObserverHandler.js +96 -1
  41. package/lib/esm/persistence/Sequence.d.ts +89 -0
  42. package/lib/esm/persistence/Sequence.js +70 -1
  43. package/lib/esm/persistence/constants.d.ts +22 -0
  44. package/lib/esm/persistence/constants.js +23 -1
  45. package/lib/esm/persistence/decorators.d.ts +10 -0
  46. package/lib/esm/persistence/decorators.js +11 -1
  47. package/lib/esm/persistence/errors.d.ts +23 -0
  48. package/lib/esm/persistence/errors.js +24 -1
  49. package/lib/esm/persistence/types.d.ts +18 -0
  50. package/lib/esm/persistence/types.js +1 -1
  51. package/lib/esm/query/Condition.d.ts +78 -31
  52. package/lib/esm/query/Condition.js +132 -53
  53. package/lib/esm/query/Paginator.d.ts +56 -0
  54. package/lib/esm/query/Paginator.js +57 -1
  55. package/lib/esm/query/Statement.d.ts +51 -0
  56. package/lib/esm/query/Statement.js +52 -1
  57. package/lib/esm/query/constants.d.ts +25 -0
  58. package/lib/esm/query/constants.js +26 -1
  59. package/lib/esm/query/errors.d.ts +14 -0
  60. package/lib/esm/query/errors.js +15 -1
  61. package/lib/esm/query/options.d.ts +21 -3
  62. package/lib/esm/query/options.js +1 -1
  63. package/lib/esm/query/selectors.d.ts +26 -0
  64. package/lib/esm/query/selectors.js +1 -1
  65. package/lib/esm/ram/RamAdapter.d.ts +311 -0
  66. package/lib/esm/ram/RamAdapter.js +312 -1
  67. package/lib/esm/ram/RamContext.d.ts +16 -1
  68. package/lib/esm/ram/RamContext.js +18 -3
  69. package/lib/esm/ram/RamPaginator.d.ts +43 -0
  70. package/lib/esm/ram/RamPaginator.js +54 -2
  71. package/lib/esm/ram/RamSequence.d.ts +61 -0
  72. package/lib/esm/ram/RamSequence.js +63 -2
  73. package/lib/esm/ram/RamStatement.d.ts +74 -0
  74. package/lib/esm/ram/RamStatement.js +75 -1
  75. package/lib/esm/ram/constants.d.ts +8 -0
  76. package/lib/esm/ram/constants.js +9 -1
  77. package/lib/esm/ram/handlers.d.ts +19 -0
  78. package/lib/esm/ram/handlers.js +20 -1
  79. package/lib/esm/ram/model/RamSequence.d.ts +25 -0
  80. package/lib/esm/ram/model/RamSequence.js +19 -1
  81. package/lib/esm/ram/types.d.ts +42 -0
  82. package/lib/esm/ram/types.js +1 -1
  83. package/lib/esm/repository/Repository.d.ts +363 -8
  84. package/lib/esm/repository/Repository.js +361 -16
  85. package/lib/esm/repository/constants.d.ts +25 -0
  86. package/lib/esm/repository/constants.js +26 -1
  87. package/lib/esm/repository/decorators.d.ts +27 -0
  88. package/lib/esm/repository/decorators.js +28 -1
  89. package/lib/esm/repository/errors.d.ts +12 -5
  90. package/lib/esm/repository/errors.js +13 -6
  91. package/lib/esm/repository/injectables.d.ts +18 -0
  92. package/lib/esm/repository/injectables.js +19 -1
  93. package/lib/esm/repository/types.d.ts +15 -0
  94. package/lib/esm/repository/types.js +1 -1
  95. package/lib/esm/repository/utils.d.ts +11 -0
  96. package/lib/esm/repository/utils.js +12 -1
  97. package/lib/esm/utils/decorators.d.ts +8 -0
  98. package/lib/esm/utils/decorators.js +9 -1
  99. package/lib/esm/utils/errors.d.ts +46 -0
  100. package/lib/esm/utils/errors.js +47 -1
  101. package/lib/identity/decorators.cjs +53 -8
  102. package/lib/identity/decorators.d.ts +52 -7
  103. package/lib/identity/utils.cjs +20 -1
  104. package/lib/identity/utils.d.ts +19 -0
  105. package/lib/index.cjs +10 -3
  106. package/lib/index.d.ts +9 -2
  107. package/lib/interfaces/ErrorParser.cjs +1 -1
  108. package/lib/interfaces/ErrorParser.d.ts +12 -0
  109. package/lib/interfaces/Executor.cjs +1 -1
  110. package/lib/interfaces/Executor.d.ts +13 -0
  111. package/lib/interfaces/Observable.cjs +1 -1
  112. package/lib/interfaces/Observable.d.ts +27 -0
  113. package/lib/interfaces/Observer.cjs +1 -1
  114. package/lib/interfaces/Observer.d.ts +12 -0
  115. package/lib/interfaces/Paginatable.cjs +1 -1
  116. package/lib/interfaces/Paginatable.d.ts +15 -0
  117. package/lib/interfaces/Queriable.cjs +1 -1
  118. package/lib/interfaces/Queriable.d.ts +34 -9
  119. package/lib/interfaces/RawExecutor.cjs +1 -1
  120. package/lib/interfaces/RawExecutor.d.ts +14 -0
  121. package/lib/interfaces/SequenceOptions.cjs +19 -1
  122. package/lib/interfaces/SequenceOptions.d.ts +52 -0
  123. package/lib/model/BaseModel.cjs +24 -1
  124. package/lib/model/BaseModel.d.ts +31 -0
  125. package/lib/model/construction.cjs +441 -2
  126. package/lib/model/construction.d.ts +433 -0
  127. package/lib/model/decorators.cjs +160 -30
  128. package/lib/model/decorators.d.ts +159 -29
  129. package/lib/model/types.cjs +1 -1
  130. package/lib/model/types.d.ts +9 -0
  131. package/lib/persistence/Adapter.cjs +287 -19
  132. package/lib/persistence/Adapter.d.ts +358 -17
  133. package/lib/persistence/Dispatch.cjs +102 -4
  134. package/lib/persistence/Dispatch.d.ts +114 -1
  135. package/lib/persistence/ObserverHandler.cjs +96 -1
  136. package/lib/persistence/ObserverHandler.d.ts +95 -0
  137. package/lib/persistence/Sequence.cjs +70 -1
  138. package/lib/persistence/Sequence.d.ts +89 -0
  139. package/lib/persistence/constants.cjs +23 -1
  140. package/lib/persistence/constants.d.ts +22 -0
  141. package/lib/persistence/decorators.cjs +11 -1
  142. package/lib/persistence/decorators.d.ts +10 -0
  143. package/lib/persistence/errors.cjs +24 -1
  144. package/lib/persistence/errors.d.ts +23 -0
  145. package/lib/persistence/types.cjs +1 -1
  146. package/lib/persistence/types.d.ts +18 -0
  147. package/lib/query/Condition.cjs +132 -53
  148. package/lib/query/Condition.d.ts +78 -31
  149. package/lib/query/Paginator.cjs +57 -1
  150. package/lib/query/Paginator.d.ts +56 -0
  151. package/lib/query/Statement.cjs +52 -1
  152. package/lib/query/Statement.d.ts +51 -0
  153. package/lib/query/constants.cjs +26 -1
  154. package/lib/query/constants.d.ts +25 -0
  155. package/lib/query/errors.cjs +15 -1
  156. package/lib/query/errors.d.ts +14 -0
  157. package/lib/query/options.cjs +1 -1
  158. package/lib/query/options.d.ts +21 -3
  159. package/lib/query/selectors.cjs +1 -1
  160. package/lib/query/selectors.d.ts +26 -0
  161. package/lib/ram/RamAdapter.cjs +312 -1
  162. package/lib/ram/RamAdapter.d.ts +311 -0
  163. package/lib/ram/RamContext.cjs +18 -3
  164. package/lib/ram/RamContext.d.ts +16 -1
  165. package/lib/ram/RamPaginator.cjs +54 -2
  166. package/lib/ram/RamPaginator.d.ts +43 -0
  167. package/lib/ram/RamSequence.cjs +63 -2
  168. package/lib/ram/RamSequence.d.ts +61 -0
  169. package/lib/ram/RamStatement.cjs +75 -1
  170. package/lib/ram/RamStatement.d.ts +74 -0
  171. package/lib/ram/constants.cjs +9 -1
  172. package/lib/ram/constants.d.ts +8 -0
  173. package/lib/ram/handlers.cjs +20 -1
  174. package/lib/ram/handlers.d.ts +19 -0
  175. package/lib/ram/model/RamSequence.cjs +19 -1
  176. package/lib/ram/model/RamSequence.d.ts +25 -0
  177. package/lib/ram/types.cjs +1 -1
  178. package/lib/ram/types.d.ts +42 -0
  179. package/lib/repository/Repository.cjs +360 -15
  180. package/lib/repository/Repository.d.ts +363 -8
  181. package/lib/repository/constants.cjs +26 -1
  182. package/lib/repository/constants.d.ts +25 -0
  183. package/lib/repository/decorators.cjs +28 -1
  184. package/lib/repository/decorators.d.ts +27 -0
  185. package/lib/repository/errors.cjs +13 -6
  186. package/lib/repository/errors.d.ts +12 -5
  187. package/lib/repository/injectables.cjs +19 -1
  188. package/lib/repository/injectables.d.ts +18 -0
  189. package/lib/repository/types.cjs +1 -1
  190. package/lib/repository/types.d.ts +15 -0
  191. package/lib/repository/utils.cjs +12 -1
  192. package/lib/repository/utils.d.ts +11 -0
  193. package/lib/utils/decorators.cjs +9 -1
  194. package/lib/utils/decorators.d.ts +8 -0
  195. package/lib/utils/errors.cjs +47 -1
  196. package/lib/utils/errors.d.ts +46 -0
  197. package/package.json +5 -5
@@ -26,40 +26,146 @@ Decoration.setFlavourResolver((obj) => {
26
26
  }
27
27
  });
28
28
  /**
29
- * @summary Abstract Decaf-ts Persistence Adapter Class
30
- * @description Offers the base implementation for all Adapter Classes
31
- * and manages them various registered {@link Adapter}s
29
+ * @description Abstract base class for database adapters
30
+ * @summary Provides the foundation for all database adapters in the persistence layer. This class
31
+ * implements several interfaces to provide a consistent API for database operations, observer
32
+ * pattern support, and error handling. It manages adapter registration, CRUD operations, and
33
+ * observer notifications.
34
+ * @template Y - The underlying database driver type
35
+ * @template Q - The query object type used by the adapter
36
+ * @template F - The repository flags type
37
+ * @template C - The context type
38
+ * @param {Y} _native - The underlying database driver instance
39
+ * @param {string} flavour - The identifier for this adapter type
40
+ * @param {string} [_alias] - Optional alternative name for this adapter
41
+ * @class Adapter
42
+ * @example
43
+ * ```typescript
44
+ * // Implementing a concrete adapter
45
+ * class PostgresAdapter extends Adapter<pg.Client, pg.Query, PostgresFlags, PostgresContext> {
46
+ * constructor(client: pg.Client) {
47
+ * super(client, 'postgres');
48
+ * }
32
49
  *
33
- * @typedef Y the underlying persistence object type or the required config to set it up
34
- * @typedef Q The query object the adapter uses
50
+ * async initialize() {
51
+ * // Set up the adapter
52
+ * await this.native.connect();
53
+ * }
35
54
  *
36
- * @param {Y} native the underlying persistence object
37
- * @param {string} flavour the under witch the persistence adapter should be stored
55
+ * async create(tableName, id, model) {
56
+ * // Implementation for creating records
57
+ * const columns = Object.keys(model).join(', ');
58
+ * const values = Object.values(model);
59
+ * const placeholders = values.map((_, i) => `$${i+1}`).join(', ');
38
60
  *
39
- * @class Adapter
40
- * @implements RawExecutor
41
- * @implements Observable
61
+ * const query = `INSERT INTO ${tableName} (${columns}) VALUES (${placeholders}) RETURNING *`;
62
+ * const result = await this.native.query(query, values);
63
+ * return result.rows[0];
64
+ * }
65
+ *
66
+ * // Other required method implementations...
67
+ * }
68
+ *
69
+ * // Using the adapter
70
+ * const pgClient = new pg.Client(connectionString);
71
+ * const adapter = new PostgresAdapter(pgClient);
72
+ * await adapter.initialize();
73
+ *
74
+ * // Set as the default adapter
75
+ * Adapter.setCurrent('postgres');
76
+ *
77
+ * // Perform operations
78
+ * const user = await adapter.create('users', 1, { name: 'John', email: 'john@example.com' });
79
+ * ```
80
+ * @mermaid
81
+ * classDiagram
82
+ * class Adapter {
83
+ * +Y native
84
+ * +string flavour
85
+ * +string alias
86
+ * +create(tableName, id, model)
87
+ * +read(tableName, id)
88
+ * +update(tableName, id, model)
89
+ * +delete(tableName, id)
90
+ * +observe(observer, filter)
91
+ * +unObserve(observer)
92
+ * +static current
93
+ * +static get(flavour)
94
+ * +static setCurrent(flavour)
95
+ * }
96
+ *
97
+ * class RawExecutor {
98
+ * +raw(query)
99
+ * }
100
+ *
101
+ * class Observable {
102
+ * +observe(observer, filter)
103
+ * +unObserve(observer)
104
+ * +updateObservers(table, event, id)
105
+ * }
106
+ *
107
+ * class Observer {
108
+ * +refresh(table, event, id)
109
+ * }
110
+ *
111
+ * class ErrorParser {
112
+ * +parseError(err)
113
+ * }
114
+ *
115
+ * Adapter --|> RawExecutor
116
+ * Adapter --|> Observable
117
+ * Adapter --|> Observer
118
+ * Adapter --|> ErrorParser
42
119
  */
43
120
  export class Adapter {
44
121
  static { this._cache = {}; }
122
+ /**
123
+ * @description Logger accessor
124
+ * @summary Gets or initializes the logger for this adapter instance
125
+ * @return {Logger} The logger instance
126
+ */
45
127
  get log() {
46
128
  if (!this.logger)
47
129
  this.logger = Logging.for(this);
48
130
  return this.logger;
49
131
  }
132
+ /**
133
+ * @description Gets the native database driver
134
+ * @summary Provides access to the underlying database driver instance
135
+ * @return {Y} The native database driver
136
+ */
50
137
  get native() {
51
138
  return this._native;
52
139
  }
140
+ /**
141
+ * @description Gets the adapter's alias or flavor name
142
+ * @summary Returns the alias if set, otherwise returns the flavor name
143
+ * @return {string} The adapter's identifier
144
+ */
53
145
  get alias() {
54
146
  return this._alias || this.flavour;
55
147
  }
148
+ /**
149
+ * @description Gets the repository constructor for this adapter
150
+ * @summary Returns the constructor for creating repositories that work with this adapter
151
+ * @template M - The model type
152
+ * @return {Constructor<Repository<M, Q, Adapter<Y, Q, F, C>, F, C>>} The repository constructor
153
+ */
56
154
  repository() {
57
155
  return Repository;
58
156
  }
157
+ /**
158
+ * @description Creates a new adapter instance
159
+ * @summary Initializes the adapter with the native driver and registers it in the adapter cache
160
+ */
59
161
  constructor(_native, flavour, _alias) {
60
162
  this._native = _native;
61
163
  this.flavour = flavour;
62
164
  this._alias = _alias;
165
+ /**
166
+ * @description The context constructor for this adapter
167
+ * @summary Reference to the context class constructor used by this adapter
168
+ */
63
169
  this.Context = (Context);
64
170
  if (this.flavour in Adapter._cache)
65
171
  throw new InternalError(`${this.alias} persistence adapter ${this._alias ? `(${this.flavour}) ` : ""} already registered`);
@@ -70,15 +176,42 @@ export class Adapter {
70
176
  Adapter._current = this;
71
177
  }
72
178
  }
179
+ /**
180
+ * @description Creates a new dispatch instance
181
+ * @summary Factory method that creates a dispatch instance for this adapter
182
+ * @return {Dispatch<Y>} A new dispatch instance
183
+ */
73
184
  Dispatch() {
74
185
  return new Dispatch();
75
186
  }
187
+ /**
188
+ * @description Creates a new observer handler
189
+ * @summary Factory method that creates an observer handler for this adapter
190
+ * @return {ObserverHandler} A new observer handler instance
191
+ */
76
192
  ObserverHandler() {
77
193
  return new ObserverHandler();
78
194
  }
195
+ /**
196
+ * @description Checks if an attribute name is reserved
197
+ * @summary Determines if a given attribute name is reserved and cannot be used as a column name
198
+ * @param {string} attr - The attribute name to check
199
+ * @return {boolean} True if the attribute is reserved, false otherwise
200
+ */
79
201
  isReserved(attr) {
80
202
  return !attr;
81
203
  }
204
+ /**
205
+ * @description Creates repository flags for an operation
206
+ * @summary Generates a set of flags that describe a database operation, combining default flags with overrides
207
+ * @template F - The Repository Flags type
208
+ * @template M - The model type
209
+ * @param {OperationKeys} operation - The type of operation being performed
210
+ * @param {Constructor<M>} model - The model constructor
211
+ * @param {Partial<F>} flags - Custom flag overrides
212
+ * @param {...any[]} args - Additional arguments
213
+ * @return {F} The complete set of flags
214
+ */
82
215
  flags(operation, model, flags,
83
216
  // eslint-disable-next-line @typescript-eslint/no-unused-vars
84
217
  ...args) {
@@ -89,12 +222,32 @@ export class Adapter {
89
222
  operation: operation,
90
223
  });
91
224
  }
225
+ /**
226
+ * @description Creates a context for a database operation
227
+ * @summary Generates a context object that describes a database operation, used for tracking and auditing
228
+ * @template F - The Repository flags type
229
+ * @template M - The model type
230
+ * @param {OperationKeys.CREATE|OperationKeys.READ|OperationKeys.UPDATE|OperationKeys.DELETE} operation - The type of operation
231
+ * @param {Partial<F>} overrides - Custom flag overrides
232
+ * @param {Constructor<M>} model - The model constructor
233
+ * @param {...any[]} args - Additional arguments
234
+ * @return {Promise<C>} A promise that resolves to the context object
235
+ */
92
236
  async context(operation, overrides, model, ...args) {
93
237
  this.log
94
238
  .for(this.context)
95
- .debug(`Creating new context for ${operation} operation on ${model.name} model with flags: ${JSON.stringify(overrides)}`);
96
- return new this.Context(this.flags(operation, model, overrides, ...args));
239
+ .debug(`Creating new context for ${operation} operation on ${model.name} model with flag overrides: ${JSON.stringify(overrides)}`);
240
+ return new this.Context().accumulate(this.flags(operation, model, overrides, ...args));
97
241
  }
242
+ /**
243
+ * @description Prepares a model for persistence
244
+ * @summary Converts a model instance into a format suitable for database storage,
245
+ * handling column mapping and separating transient properties
246
+ * @template M - The model type
247
+ * @param {M} model - The model instance to prepare
248
+ * @param pk - The primary key property name
249
+ * @return The prepared data
250
+ */
98
251
  prepare(model, pk) {
99
252
  const log = this.log.for(this.prepare);
100
253
  log.silly(`Preparing model ${model.constructor.name} before persisting`);
@@ -123,6 +276,18 @@ export class Adapter {
123
276
  transient: split.transient,
124
277
  };
125
278
  }
279
+ /**
280
+ * @description Converts database data back into a model instance
281
+ * @summary Reconstructs a model instance from database data, handling column mapping
282
+ * and reattaching transient properties
283
+ * @template M - The model type
284
+ * @param obj - The database record
285
+ * @param {string|Constructor<M>} clazz - The model class or name
286
+ * @param pk - The primary key property name
287
+ * @param {string|number|bigint} id - The primary key value
288
+ * @param [transient] - Transient properties to reattach
289
+ * @return {M} The reconstructed model instance
290
+ */
126
291
  revert(obj, clazz, pk, id, transient) {
127
292
  const log = this.log.for(this.revert);
128
293
  const ob = {};
@@ -155,6 +320,15 @@ export class Adapter {
155
320
  }
156
321
  return result;
157
322
  }
323
+ /**
324
+ * @description Creates multiple records in the database
325
+ * @summary Inserts multiple records with the given IDs and data into the specified table
326
+ * @param {string} tableName - The name of the table to insert into
327
+ * @param id - The identifiers for the new records
328
+ * @param model - The data to insert for each record
329
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
330
+ * @return A promise that resolves to an array of created records
331
+ */
158
332
  async createAll(tableName, id, model, ...args) {
159
333
  if (id.length !== model.length)
160
334
  throw new InternalError("Ids and models must have the same length");
@@ -163,12 +337,29 @@ export class Adapter {
163
337
  log.debug(`pks: ${id}`);
164
338
  return Promise.all(id.map((i, count) => this.create(tableName, i, model[count], ...args)));
165
339
  }
340
+ /**
341
+ * @description Retrieves multiple records from the database
342
+ * @summary Fetches multiple records with the given IDs from the specified table
343
+ * @param {string} tableName - The name of the table to read from
344
+ * @param id - The identifiers of the records to retrieve
345
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
346
+ * @return A promise that resolves to an array of retrieved records
347
+ */
166
348
  async readAll(tableName, id, ...args) {
167
349
  const log = this.log.for(this.readAll);
168
350
  log.verbose(`Reading ${id.length} entries ${tableName} table`);
169
351
  log.debug(`pks: ${id}`);
170
352
  return Promise.all(id.map((i) => this.read(tableName, i, ...args)));
171
353
  }
354
+ /**
355
+ * @description Updates multiple records in the database
356
+ * @summary Modifies multiple existing records with the given IDs in the specified table
357
+ * @param {string} tableName - The name of the table to update
358
+ * @param {string[]|number[]} id - The identifiers of the records to update
359
+ * @param model - The new data for each record
360
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
361
+ * @return A promise that resolves to an array of updated records
362
+ */
172
363
  async updateAll(tableName, id, model, ...args) {
173
364
  if (id.length !== model.length)
174
365
  throw new InternalError("Ids and models must have the same length");
@@ -177,6 +368,14 @@ export class Adapter {
177
368
  log.debug(`pks: ${id}`);
178
369
  return Promise.all(id.map((i, count) => this.update(tableName, i, model[count], ...args)));
179
370
  }
371
+ /**
372
+ * @description Deletes multiple records from the database
373
+ * @summary Removes multiple records with the given IDs from the specified table
374
+ * @param {string} tableName - The name of the table to delete from
375
+ * @param id - The identifiers of the records to delete
376
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
377
+ * @return A promise that resolves to an array of deleted records
378
+ */
180
379
  async deleteAll(tableName, id, ...args) {
181
380
  const log = this.log.for(this.createAll);
182
381
  log.verbose(`Deleting ${id.length} entries ${tableName} table`);
@@ -184,8 +383,12 @@ export class Adapter {
184
383
  return Promise.all(id.map((i) => this.delete(tableName, i, ...args)));
185
384
  }
186
385
  /**
187
- *
188
- * @see {Observable#observe}
386
+ * @description Registers an observer for database events
387
+ * @summary Adds an observer to be notified about database changes. The observer can optionally
388
+ * provide a filter function to receive only specific events.
389
+ * @param {Observer} observer - The observer to register
390
+ * @param {ObserverFilter} [filter] - Optional filter function to determine which events the observer receives
391
+ * @return {void}
189
392
  */
190
393
  observe(observer, filter) {
191
394
  if (!this.observerHandler)
@@ -204,10 +407,10 @@ export class Adapter {
204
407
  }
205
408
  }
206
409
  /**
207
- * @summary Unregisters an {@link Observer}
208
- * @param {Observer} observer
209
- *
210
- * @see {Observable#unObserve}
410
+ * @description Unregisters an observer
411
+ * @summary Removes a previously registered observer so it no longer receives database event notifications
412
+ * @param {Observer} observer - The observer to unregister
413
+ * @return {void}
211
414
  */
212
415
  unObserve(observer) {
213
416
  if (!this.observerHandler)
@@ -217,6 +420,16 @@ export class Adapter {
217
420
  .for(this.unObserve)
218
421
  .verbose(`Observer ${observer.toString()} removed`);
219
422
  }
423
+ /**
424
+ * @description Notifies all observers about a database event
425
+ * @summary Sends notifications to all registered observers about a change in the database,
426
+ * filtering based on each observer's filter function
427
+ * @param {string} table - The name of the table where the change occurred
428
+ * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
429
+ * @param {EventIds} id - The identifier(s) of the affected record(s)
430
+ * @param {...any[]} args - Additional arguments to pass to the observers
431
+ * @return {Promise<void>} A promise that resolves when all observers have been notified
432
+ */
220
433
  async updateObservers(table, event, id, ...args) {
221
434
  if (!this.observerHandler)
222
435
  throw new InternalError("ObserverHandler not initialized. Did you register any observables?");
@@ -224,35 +437,90 @@ export class Adapter {
224
437
  log.verbose(`Updating ${this.observerHandler.count()} observers for adapter ${this.alias}`);
225
438
  await this.observerHandler.updateObservers(this.log, table, event, id, ...args);
226
439
  }
440
+ /**
441
+ * @description Refreshes data based on a database event
442
+ * @summary Implementation of the Observer interface method that delegates to updateObservers
443
+ * @param {string} table - The name of the table where the change occurred
444
+ * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
445
+ * @param {EventIds} id - The identifier(s) of the affected record(s)
446
+ * @param {...any[]} args - Additional arguments related to the event
447
+ * @return {Promise<void>} A promise that resolves when the refresh is complete
448
+ */
227
449
  async refresh(table, event, id, ...args) {
228
450
  return this.updateObservers(table, event, id, ...args);
229
451
  }
452
+ /**
453
+ * @description Gets a string representation of the adapter
454
+ * @summary Returns a human-readable string identifying this adapter
455
+ * @return {string} A string representation of the adapter
456
+ */
230
457
  toString() {
231
458
  return `${this.flavour} persistence Adapter`;
232
459
  }
460
+ /**
461
+ * @description Gets the adapter flavor associated with a model
462
+ * @summary Retrieves the adapter flavor that should be used for a specific model class
463
+ * @template M - The model type
464
+ * @param {Constructor<M>} model - The model constructor
465
+ * @return {string} The adapter flavor name
466
+ */
233
467
  static flavourOf(model) {
234
468
  return (Reflect.getMetadata(this.key(PersistenceKeys.ADAPTER), model) ||
235
469
  this.current.flavour);
236
470
  }
471
+ /**
472
+ * @description Gets the current default adapter
473
+ * @summary Retrieves the adapter that is currently set as the default for operations
474
+ * @return {Adapter<any, any, any, any>} The current adapter
475
+ */
237
476
  static get current() {
238
477
  if (!Adapter._current)
239
478
  throw new InternalError(`No persistence flavour set. Please initialize your adapter`);
240
479
  return Adapter._current;
241
480
  }
481
+ /**
482
+ * @description Gets an adapter by flavor
483
+ * @summary Retrieves a registered adapter by its flavor name
484
+ * @template Y - The database driver type
485
+ * @template Q - The query type
486
+ * @template C - The context type
487
+ * @template F - The repository flags type
488
+ * @param {string} flavour - The flavor name of the adapter to retrieve
489
+ * @return {Adapter<Y, Q, F, C> | undefined} The adapter instance or undefined if not found
490
+ */
242
491
  static get(flavour) {
243
492
  if (flavour in this._cache)
244
493
  return this._cache[flavour];
245
494
  throw new InternalError(`No Adapter registered under ${flavour}.`);
246
495
  }
496
+ /**
497
+ * @description Sets the current default adapter
498
+ * @summary Changes which adapter is used as the default for operations
499
+ * @param {string} flavour - The flavor name of the adapter to set as current
500
+ * @return {void}
501
+ */
247
502
  static setCurrent(flavour) {
248
503
  const adapter = Adapter.get(flavour);
249
504
  if (!adapter)
250
505
  throw new NotFoundError(`No persistence flavour ${flavour} registered`);
251
506
  this._current = adapter;
252
507
  }
508
+ /**
509
+ * @description Creates a metadata key
510
+ * @summary Generates a standardized metadata key for persistence-related metadata
511
+ * @param {string} key - The base key name
512
+ * @return {string} The formatted metadata key
513
+ */
253
514
  static key(key) {
254
515
  return Repository.key(key);
255
516
  }
517
+ /**
518
+ * @description Gets all models associated with an adapter flavor
519
+ * @summary Retrieves all model constructors that are configured to use a specific adapter flavor
520
+ * @template M - The model type
521
+ * @param {string} flavour - The adapter flavor to find models for
522
+ * @return An array of model constructors
523
+ */
256
524
  static models(flavour) {
257
525
  try {
258
526
  const registry = Model.getRegistry();
@@ -297,4 +565,4 @@ __decorate([
297
565
  __metadata("design:paramtypes", [Object]),
298
566
  __metadata("design:returntype", void 0)
299
567
  ], Adapter.prototype, "unObserve", null);
300
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Adapter.js","sourceRoot":"","sources":["../../../src/persistence/Adapter.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BaseError,\n  DBKeys,\n  InternalError,\n  NotFoundError,\n  Context,\n  OperationKeys,\n  RepositoryFlags,\n  DefaultRepositoryFlags,\n  Contextual,\n  BulkCrudOperationKeys,\n  modelToTransient,\n} from \"@decaf-ts/db-decorators\";\nimport { type Observer } from \"../interfaces/Observer\";\nimport {\n  type Constructor,\n  Decoration,\n  DefaultFlavour,\n  Model,\n  ModelConstructor,\n  ModelRegistry,\n} from \"@decaf-ts/decorator-validation\";\nimport { SequenceOptions } from \"../interfaces/SequenceOptions\";\nimport { RawExecutor } from \"../interfaces/RawExecutor\";\nimport { Observable } from \"../interfaces/Observable\";\nimport { PersistenceKeys } from \"./constants\";\nimport { Repository } from \"../repository/Repository\";\nimport { Sequence } from \"./Sequence\";\nimport { ErrorParser } from \"../interfaces\";\nimport { Statement } from \"../query/Statement\";\nimport { Logger, Logging } from \"@decaf-ts/logging\";\nimport { final } from \"../utils\";\nimport { Dispatch } from \"./Dispatch\";\nimport { type EventIds, type ObserverFilter } from \"./types\";\nimport { ObserverHandler } from \"./ObserverHandler\";\n\nDecoration.setFlavourResolver((obj: object) => {\n  try {\n    return (\n      Adapter.flavourOf(Model.isModel(obj) ? obj.constructor : (obj as any)) ||\n      DefaultFlavour\n    );\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  } catch (e: unknown) {\n    return DefaultFlavour;\n  }\n});\n\n/**\n * @summary Abstract Decaf-ts Persistence Adapter Class\n * @description Offers the base implementation for all Adapter Classes\n * and manages them various registered {@link Adapter}s\n *\n * @typedef Y the underlying persistence object type or the required config to set it up\n * @typedef Q The query object the adapter uses\n *\n * @param {Y} native the underlying persistence object\n * @param {string} flavour the under witch the persistence adapter should be stored\n *\n * @class Adapter\n * @implements RawExecutor\n * @implements Observable\n */\nexport abstract class Adapter<\n    Y,\n    Q,\n    F extends RepositoryFlags,\n    C extends Context<F>,\n  >\n  implements RawExecutor<Q>, Contextual<F, C>, Observable, Observer, ErrorParser\n{\n  private static _current: Adapter<any, any, any, any>;\n  private static _cache: Record<string, Adapter<any, any, any, any>> = {};\n\n  private logger!: Logger;\n\n  protected dispatch?: Dispatch<Y>;\n\n  protected readonly observerHandler?: ObserverHandler;\n\n  protected get log() {\n    if (!this.logger) this.logger = Logging.for(this as any);\n    return this.logger;\n  }\n\n  get native() {\n    return this._native;\n  }\n\n  get alias() {\n    return this._alias || this.flavour;\n  }\n\n  repository<M extends Model>(): Constructor<\n    Repository<M, Q, Adapter<Y, Q, F, C>, F, C>\n  > {\n    return Repository;\n  }\n\n  protected constructor(\n    private readonly _native: Y,\n    readonly flavour: string,\n    private readonly _alias?: string\n  ) {\n    if (this.flavour in Adapter._cache)\n      throw new InternalError(\n        `${this.alias} persistence adapter ${this._alias ? `(${this.flavour}) ` : \"\"} already registered`\n      );\n    Adapter._cache[this.alias] = this;\n    this.log.info(\n      `Created ${this.alias} persistence adapter ${this._alias ? `(${this.flavour}) ` : \"\"} persistence adapter`\n    );\n    if (!Adapter._current) {\n      this.log.verbose(`Defined ${this.alias} persistence adapter as current`);\n      Adapter._current = this;\n    }\n  }\n\n  abstract Statement<M extends Model>(): Statement<Q, M, any>;\n\n  protected Dispatch(): Dispatch<Y> {\n    return new Dispatch();\n  }\n\n  protected ObserverHandler() {\n    return new ObserverHandler();\n  }\n\n  protected isReserved(attr: string) {\n    return !attr;\n  }\n\n  abstract parseError(err: Error): BaseError;\n\n  abstract initialize(...args: any[]): Promise<void>;\n\n  abstract Sequence(options: SequenceOptions): Promise<Sequence>;\n\n  protected flags<M extends Model>(\n    operation: OperationKeys,\n    model: Constructor<M>,\n    flags: Partial<F>,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): F {\n    return Object.assign({}, DefaultRepositoryFlags, flags, {\n      affectedTables: Repository.table(model),\n      writeOperation: operation !== OperationKeys.READ,\n      timestamp: new Date(),\n      operation: operation,\n    }) as F;\n  }\n\n  protected Context: Constructor<C> = Context<F> as any;\n\n  @final()\n  async context<M extends Model>(\n    operation:\n      | OperationKeys.CREATE\n      | OperationKeys.READ\n      | OperationKeys.UPDATE\n      | OperationKeys.DELETE,\n    overrides: Partial<F>,\n    model: Constructor<M>,\n    ...args: any[]\n  ): Promise<C> {\n    this.log\n      .for(this.context)\n      .debug(\n        `Creating new context for ${operation} operation on ${model.name} model with flags: ${JSON.stringify(overrides)}`\n      );\n    return new this.Context(\n      this.flags(operation, model, overrides, ...args)\n    ) as unknown as C;\n  }\n\n  prepare<M extends Model>(\n    model: M,\n    pk: keyof M\n  ): {\n    record: Record<string, any>;\n    id: string;\n    transient?: Record<string, any>;\n  } {\n    const log = this.log.for(this.prepare);\n    log.silly(`Preparing model ${model.constructor.name} before persisting`);\n    const split = modelToTransient(model);\n    const result = Object.entries(split.model).reduce(\n      (accum: Record<string, any>, [key, val]) => {\n        if (typeof val === \"undefined\") return accum;\n        const mappedProp = Repository.column(model, key);\n        if (this.isReserved(mappedProp))\n          throw new InternalError(`Property name ${mappedProp} is reserved`);\n        accum[mappedProp] = val;\n        return accum;\n      },\n      {}\n    );\n    if ((model as any)[PersistenceKeys.METADATA]) {\n      log.silly(\n        `Passing along persistence metadata for ${(model as any)[PersistenceKeys.METADATA]}`\n      );\n      Object.defineProperty(result, PersistenceKeys.METADATA, {\n        enumerable: false,\n        writable: false,\n        configurable: true,\n        value: (model as any)[PersistenceKeys.METADATA],\n      });\n    }\n\n    return {\n      record: result,\n      id: model[pk] as string,\n      transient: split.transient,\n    };\n  }\n\n  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    const ob: Record<string, any> = {};\n    ob[pk as string] = id;\n    const m = (\n      typeof clazz === \"string\" ? Model.build(ob, clazz) : new clazz(ob)\n    ) as M;\n    log.silly(`Rebuilding model ${m.constructor.name} id ${id}`);\n    const metadata = obj[PersistenceKeys.METADATA];\n    const result = Object.keys(m).reduce((accum: M, key) => {\n      if (key === pk) return accum;\n      (accum as Record<string, any>)[key] = obj[Repository.column(accum, key)];\n      return accum;\n    }, m);\n\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 result)\n          throw new InternalError(\n            `Transient property ${key} already exists on model ${m.constructor.name}. should be impossible`\n          );\n        result[key as keyof M] = val;\n      });\n    }\n\n    if (metadata) {\n      log.silly(\n        `Passing along ${this.flavour} persistence metadata for ${m.constructor.name} id ${id}: ${metadata}`\n      );\n      Object.defineProperty(result, PersistenceKeys.METADATA, {\n        enumerable: false,\n        configurable: false,\n        writable: false,\n        value: metadata,\n      });\n    }\n\n    return result;\n  }\n\n  abstract create(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  async createAll(\n    tableName: string,\n    id: (string | number)[],\n    model: Record<string, any>[],\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    if (id.length !== model.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n    const log = this.log.for(this.createAll);\n    log.verbose(`Creating ${id.length} entries ${tableName} table`);\n    log.debug(`pks: ${id}`);\n    return Promise.all(\n      id.map((i, count) => this.create(tableName, i, model[count], ...args))\n    );\n  }\n\n  abstract read(\n    tableName: string,\n    id: string | number | bigint,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  async readAll(\n    tableName: string,\n    id: (string | number | bigint)[],\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    const log = this.log.for(this.readAll);\n    log.verbose(`Reading ${id.length} entries ${tableName} table`);\n    log.debug(`pks: ${id}`);\n    return Promise.all(id.map((i) => this.read(tableName, i, ...args)));\n  }\n\n  abstract update(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  async updateAll(\n    tableName: string,\n    id: string[] | number[],\n    model: Record<string, any>[],\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    if (id.length !== model.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n    const log = this.log.for(this.updateAll);\n    log.verbose(`Updating ${id.length} entries ${tableName} table`);\n    log.debug(`pks: ${id}`);\n    return Promise.all(\n      id.map((i, count) => this.update(tableName, i, model[count], ...args))\n    );\n  }\n\n  abstract delete(\n    tableName: string,\n    id: string | number | bigint,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  async deleteAll(\n    tableName: string,\n    id: (string | number | bigint)[],\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    const log = this.log.for(this.createAll);\n    log.verbose(`Deleting ${id.length} entries ${tableName} table`);\n    log.debug(`pks: ${id}`);\n    return Promise.all(id.map((i) => this.delete(tableName, i, ...args)));\n  }\n\n  abstract raw<R>(rawInput: Q, ...args: any[]): Promise<R>;\n\n  /**\n   *\n   * @see {Observable#observe}\n   */\n  @final()\n  observe(observer: Observer, filter?: ObserverFilter): void {\n    if (!this.observerHandler)\n      Object.defineProperty(this, \"observerHandler\", {\n        value: this.ObserverHandler(),\n        writable: false,\n      });\n    this.observerHandler!.observe(observer, filter);\n    this.log\n      .for(this.observe)\n      .verbose(`Registering new observer ${observer.toString()}`);\n    if (!this.dispatch) {\n      this.log.for(this.observe).info(`Creating dispatch for ${this.alias}`);\n      this.dispatch = this.Dispatch();\n      this.dispatch.observe(this);\n    }\n  }\n\n  /**\n   * @summary Unregisters an {@link Observer}\n   * @param {Observer} observer\n   *\n   * @see {Observable#unObserve}\n   */\n  @final()\n  unObserve(observer: Observer): void {\n    if (!this.observerHandler)\n      throw new InternalError(\n        \"ObserverHandler not initialized. Did you register any observables?\"\n      );\n    this.observerHandler.unObserve(observer);\n    this.log\n      .for(this.unObserve)\n      .verbose(`Observer ${observer.toString()} removed`);\n  }\n\n  async updateObservers(\n    table: string,\n    event: OperationKeys | BulkCrudOperationKeys | string,\n    id: EventIds,\n    ...args: any[]\n  ): Promise<void> {\n    if (!this.observerHandler)\n      throw new InternalError(\n        \"ObserverHandler not initialized. Did you register any observables?\"\n      );\n    const log = this.log.for(this.updateObservers);\n    log.verbose(\n      `Updating ${this.observerHandler.count()} observers for adapter ${this.alias}`\n    );\n    await this.observerHandler.updateObservers(\n      this.log,\n      table,\n      event,\n      id,\n      ...args\n    );\n  }\n\n  async refresh(\n    table: string,\n    event: OperationKeys | BulkCrudOperationKeys | string,\n    id: EventIds,\n    ...args: any[]\n  ) {\n    return this.updateObservers(table, event, id, ...args);\n  }\n\n  toString() {\n    return `${this.flavour} persistence Adapter`;\n  }\n\n  static flavourOf<M extends Model>(model: Constructor<M>): string {\n    return (\n      Reflect.getMetadata(this.key(PersistenceKeys.ADAPTER), model) ||\n      this.current.flavour\n    );\n  }\n\n  static get current() {\n    if (!Adapter._current)\n      throw new InternalError(\n        `No persistence flavour set. Please initialize your adapter`\n      );\n    return Adapter._current;\n  }\n\n  static get<Y, Q, C extends Context<F>, F extends RepositoryFlags>(\n    flavour: any\n  ): Adapter<Y, Q, F, C> | undefined {\n    if (flavour in this._cache) return this._cache[flavour];\n    throw new InternalError(`No Adapter registered under ${flavour}.`);\n  }\n\n  static setCurrent(flavour: string) {\n    const adapter = Adapter.get(flavour);\n    if (!adapter)\n      throw new NotFoundError(`No persistence flavour ${flavour} registered`);\n    this._current = adapter;\n  }\n\n  static key(key: string) {\n    return Repository.key(key);\n  }\n\n  static models<M extends Model>(flavour: string) {\n    try {\n      const registry = (Model as any).getRegistry() as ModelRegistry<any>;\n      const cache = (\n        registry as unknown as { cache: Record<string, ModelConstructor<any>> }\n      ).cache;\n      const managedModels: ModelConstructor<any>[] = Object.values(cache)\n        .map((m: ModelConstructor<M>) => {\n          let f = Reflect.getMetadata(\n            Adapter.key(PersistenceKeys.ADAPTER),\n            m as ModelConstructor<any>\n          );\n          if (f && f === flavour) return m;\n          if (!f) {\n            const repo = Reflect.getMetadata(\n              Repository.key(DBKeys.REPOSITORY),\n              m as ModelConstructor<any>\n            );\n            if (!repo) return;\n            const repository = Repository.forModel(m);\n\n            f = Reflect.getMetadata(\n              Adapter.key(PersistenceKeys.ADAPTER),\n              repository\n            );\n            return f;\n          }\n        })\n        .filter((m) => !!m);\n      return managedModels;\n    } catch (e: any) {\n      throw new InternalError(e);\n    }\n  }\n}\n"]}
568
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Adapter.js","sourceRoot":"","sources":["../../../src/persistence/Adapter.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BaseError,\n  DBKeys,\n  InternalError,\n  NotFoundError,\n  Context,\n  OperationKeys,\n  RepositoryFlags,\n  DefaultRepositoryFlags,\n  Contextual,\n  BulkCrudOperationKeys,\n  modelToTransient,\n} from \"@decaf-ts/db-decorators\";\nimport { type Observer } from \"../interfaces/Observer\";\nimport {\n  type Constructor,\n  Decoration,\n  DefaultFlavour,\n  Model,\n  ModelConstructor,\n  ModelRegistry,\n} from \"@decaf-ts/decorator-validation\";\nimport { SequenceOptions } from \"../interfaces/SequenceOptions\";\nimport { RawExecutor } from \"../interfaces/RawExecutor\";\nimport { Observable } from \"../interfaces/Observable\";\nimport { PersistenceKeys } from \"./constants\";\nimport { Repository } from \"../repository/Repository\";\nimport { Sequence } from \"./Sequence\";\nimport { ErrorParser } from \"../interfaces\";\nimport { Statement } from \"../query/Statement\";\nimport { Logger, Logging } from \"@decaf-ts/logging\";\nimport { final } from \"../utils\";\nimport { Dispatch } from \"./Dispatch\";\nimport { type EventIds, type ObserverFilter } from \"./types\";\nimport { ObserverHandler } from \"./ObserverHandler\";\n\nDecoration.setFlavourResolver((obj: object) => {\n  try {\n    return (\n      Adapter.flavourOf(Model.isModel(obj) ? obj.constructor : (obj as any)) ||\n      DefaultFlavour\n    );\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  } catch (e: unknown) {\n    return DefaultFlavour;\n  }\n});\n\n/**\n * @description Abstract base class for database adapters\n * @summary Provides the foundation for all database adapters in the persistence layer. This class\n * implements several interfaces to provide a consistent API for database operations, observer\n * pattern support, and error handling. It manages adapter registration, CRUD operations, and\n * observer notifications.\n * @template Y - The underlying database driver type\n * @template Q - The query object type used by the adapter\n * @template F - The repository flags type\n * @template C - The context type\n * @param {Y} _native - The underlying database driver instance\n * @param {string} flavour - The identifier for this adapter type\n * @param {string} [_alias] - Optional alternative name for this adapter\n * @class Adapter\n * @example\n * ```typescript\n * // Implementing a concrete adapter\n * class PostgresAdapter extends Adapter<pg.Client, pg.Query, PostgresFlags, PostgresContext> {\n *   constructor(client: pg.Client) {\n *     super(client, 'postgres');\n *   }\n *\n *   async initialize() {\n *     // Set up the adapter\n *     await this.native.connect();\n *   }\n *\n *   async create(tableName, id, model) {\n *     // Implementation for creating records\n *     const columns = Object.keys(model).join(', ');\n *     const values = Object.values(model);\n *     const placeholders = values.map((_, i) => `$${i+1}`).join(', ');\n *\n *     const query = `INSERT INTO ${tableName} (${columns}) VALUES (${placeholders}) RETURNING *`;\n *     const result = await this.native.query(query, values);\n *     return result.rows[0];\n *   }\n *\n *   // Other required method implementations...\n * }\n *\n * // Using the adapter\n * const pgClient = new pg.Client(connectionString);\n * const adapter = new PostgresAdapter(pgClient);\n * await adapter.initialize();\n *\n * // Set as the default adapter\n * Adapter.setCurrent('postgres');\n *\n * // Perform operations\n * const user = await adapter.create('users', 1, { name: 'John', email: 'john@example.com' });\n * ```\n * @mermaid\n * classDiagram\n *   class Adapter {\n *     +Y native\n *     +string flavour\n *     +string alias\n *     +create(tableName, id, model)\n *     +read(tableName, id)\n *     +update(tableName, id, model)\n *     +delete(tableName, id)\n *     +observe(observer, filter)\n *     +unObserve(observer)\n *     +static current\n *     +static get(flavour)\n *     +static setCurrent(flavour)\n *   }\n *\n *   class RawExecutor {\n *     +raw(query)\n *   }\n *\n *   class Observable {\n *     +observe(observer, filter)\n *     +unObserve(observer)\n *     +updateObservers(table, event, id)\n *   }\n *\n *   class Observer {\n *     +refresh(table, event, id)\n *   }\n *\n *   class ErrorParser {\n *     +parseError(err)\n *   }\n *\n *   Adapter --|> RawExecutor\n *   Adapter --|> Observable\n *   Adapter --|> Observer\n *   Adapter --|> ErrorParser\n */\nexport abstract class Adapter<\n    Y,\n    Q,\n    F extends RepositoryFlags,\n    C extends Context<F>,\n  >\n  implements RawExecutor<Q>, Contextual<F, C>, Observable, Observer, ErrorParser\n{\n  private static _current: Adapter<any, any, any, any>;\n  private static _cache: Record<string, Adapter<any, any, any, any>> = {};\n\n  private logger!: Logger;\n\n  protected dispatch?: Dispatch<Y>;\n\n  protected readonly observerHandler?: ObserverHandler;\n\n  /**\n   * @description Logger accessor\n   * @summary Gets or initializes the logger for this adapter instance\n   * @return {Logger} The logger instance\n   */\n  protected get log() {\n    if (!this.logger) this.logger = Logging.for(this as any);\n    return this.logger;\n  }\n\n  /**\n   * @description Gets the native database driver\n   * @summary Provides access to the underlying database driver instance\n   * @return {Y} The native database driver\n   */\n  get native() {\n    return this._native;\n  }\n\n  /**\n   * @description Gets the adapter's alias or flavor name\n   * @summary Returns the alias if set, otherwise returns the flavor name\n   * @return {string} The adapter's identifier\n   */\n  get alias() {\n    return this._alias || this.flavour;\n  }\n\n  /**\n   * @description Gets the repository constructor for this adapter\n   * @summary Returns the constructor for creating repositories that work with this adapter\n   * @template M - The model type\n   * @return {Constructor<Repository<M, Q, Adapter<Y, Q, F, C>, F, C>>} The repository constructor\n   */\n  repository<M extends Model>(): Constructor<\n    Repository<M, Q, Adapter<Y, Q, F, C>, F, C>\n  > {\n    return Repository;\n  }\n\n  /**\n   * @description Creates a new adapter instance\n   * @summary Initializes the adapter with the native driver and registers it in the adapter cache\n   */\n  protected constructor(\n    private readonly _native: Y,\n    readonly flavour: string,\n    private readonly _alias?: string\n  ) {\n    if (this.flavour in Adapter._cache)\n      throw new InternalError(\n        `${this.alias} persistence adapter ${this._alias ? `(${this.flavour}) ` : \"\"} already registered`\n      );\n    Adapter._cache[this.alias] = this;\n    this.log.info(\n      `Created ${this.alias} persistence adapter ${this._alias ? `(${this.flavour}) ` : \"\"} persistence adapter`\n    );\n    if (!Adapter._current) {\n      this.log.verbose(`Defined ${this.alias} persistence adapter as current`);\n      Adapter._current = this;\n    }\n  }\n\n  /**\n   * @description Creates a new statement builder for a model\n   * @summary Returns a statement builder that can be used to construct queries for a specific model\n   * @template M - The model type\n   * @return {Statement} A statement builder for the model\n   */\n  abstract Statement<M extends Model>(): Statement<Q, M, any>;\n\n  /**\n   * @description Creates a new dispatch instance\n   * @summary Factory method that creates a dispatch instance for this adapter\n   * @return {Dispatch<Y>} A new dispatch instance\n   */\n  protected Dispatch(): Dispatch<Y> {\n    return new Dispatch();\n  }\n\n  /**\n   * @description Creates a new observer handler\n   * @summary Factory method that creates an observer handler for this adapter\n   * @return {ObserverHandler} A new observer handler instance\n   */\n  protected ObserverHandler() {\n    return new ObserverHandler();\n  }\n\n  /**\n   * @description Checks if an attribute name is reserved\n   * @summary Determines if a given attribute name is reserved and cannot be used as a column name\n   * @param {string} attr - The attribute name to check\n   * @return {boolean} True if the attribute is reserved, false otherwise\n   */\n  protected isReserved(attr: string) {\n    return !attr;\n  }\n\n  /**\n   * @description Parses a database error into a standardized error\n   * @summary Converts database-specific errors into standardized application errors\n   * @param {Error} err - The original database error\n   * @return {BaseError} A standardized error\n   */\n  abstract parseError(err: Error): BaseError;\n\n  /**\n   * @description Initializes the adapter\n   * @summary Performs any necessary setup for the adapter, such as establishing connections\n   * @param {...any[]} args - Initialization arguments\n   * @return {Promise<void>} A promise that resolves when initialization is complete\n   */\n  abstract initialize(...args: any[]): Promise<void>;\n\n  /**\n   * @description Creates a sequence generator\n   * @summary Factory method that creates a sequence generator for generating sequential values\n   * @param {SequenceOptions} options - Configuration options for the sequence\n   * @return {Promise<Sequence>} A promise that resolves to a new sequence instance\n   */\n  abstract Sequence(options: SequenceOptions): Promise<Sequence>;\n\n  /**\n   * @description Creates repository flags for an operation\n   * @summary Generates a set of flags that describe a database operation, combining default flags with overrides\n   * @template F - The Repository Flags type\n   * @template M - The model type\n   * @param {OperationKeys} operation - The type of operation being performed\n   * @param {Constructor<M>} model - The model constructor\n   * @param {Partial<F>} flags - Custom flag overrides\n   * @param {...any[]} args - Additional arguments\n   * @return {F} The complete set of flags\n   */\n  protected flags<M extends Model>(\n    operation: OperationKeys,\n    model: Constructor<M>,\n    flags: Partial<F>,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): F {\n    return Object.assign({}, DefaultRepositoryFlags, flags, {\n      affectedTables: Repository.table(model),\n      writeOperation: operation !== OperationKeys.READ,\n      timestamp: new Date(),\n      operation: operation,\n    }) as F;\n  }\n\n  /**\n   * @description The context constructor for this adapter\n   * @summary Reference to the context class constructor used by this adapter\n   */\n  protected Context = Context<F>;\n\n  /**\n   * @description Creates a context for a database operation\n   * @summary Generates a context object that describes a database operation, used for tracking and auditing\n   * @template F - The Repository flags type\n   * @template M - The model type\n   * @param {OperationKeys.CREATE|OperationKeys.READ|OperationKeys.UPDATE|OperationKeys.DELETE} operation - The type of operation\n   * @param {Partial<F>} overrides - Custom flag overrides\n   * @param {Constructor<M>} model - The model constructor\n   * @param {...any[]} args - Additional arguments\n   * @return {Promise<C>} A promise that resolves to the context object\n   */\n  @final()\n  async context<M extends Model>(\n    operation:\n      | OperationKeys.CREATE\n      | OperationKeys.READ\n      | OperationKeys.UPDATE\n      | OperationKeys.DELETE,\n    overrides: Partial<F>,\n    model: Constructor<M>,\n    ...args: any[]\n  ): Promise<C> {\n    this.log\n      .for(this.context)\n      .debug(\n        `Creating new context for ${operation} operation on ${model.name} model with flag overrides: ${JSON.stringify(overrides)}`\n      );\n    return new this.Context().accumulate(\n      this.flags(operation, model, overrides, ...args)\n    ) as unknown as C;\n  }\n\n  /**\n   * @description Prepares a model for persistence\n   * @summary Converts a model instance into a format suitable for database storage,\n   * handling column mapping and separating transient properties\n   * @template M - The model type\n   * @param {M} model - The model instance to prepare\n   * @param pk - The primary key property name\n   * @return The prepared data\n   */\n  prepare<M extends Model>(\n    model: M,\n    pk: keyof M\n  ): {\n    record: Record<string, any>;\n    id: string;\n    transient?: Record<string, any>;\n  } {\n    const log = this.log.for(this.prepare);\n    log.silly(`Preparing model ${model.constructor.name} before persisting`);\n    const split = modelToTransient(model);\n    const result = Object.entries(split.model).reduce(\n      (accum: Record<string, any>, [key, val]) => {\n        if (typeof val === \"undefined\") return accum;\n        const mappedProp = Repository.column(model, key);\n        if (this.isReserved(mappedProp))\n          throw new InternalError(`Property name ${mappedProp} is reserved`);\n        accum[mappedProp] = val;\n        return accum;\n      },\n      {}\n    );\n    if ((model as any)[PersistenceKeys.METADATA]) {\n      log.silly(\n        `Passing along persistence metadata for ${(model as any)[PersistenceKeys.METADATA]}`\n      );\n      Object.defineProperty(result, PersistenceKeys.METADATA, {\n        enumerable: false,\n        writable: false,\n        configurable: true,\n        value: (model as any)[PersistenceKeys.METADATA],\n      });\n    }\n\n    return {\n      record: result,\n      id: model[pk] as string,\n      transient: split.transient,\n    };\n  }\n\n  /**\n   * @description Converts database data back into a model instance\n   * @summary Reconstructs a model instance from database data, handling column mapping\n   * and reattaching transient properties\n   * @template M - The model type\n   * @param obj - The database record\n   * @param {string|Constructor<M>} clazz - The model class or name\n   * @param pk - The primary key property name\n   * @param {string|number|bigint} id - The primary key value\n   * @param [transient] - Transient properties to reattach\n   * @return {M} The reconstructed model instance\n   */\n  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    const ob: Record<string, any> = {};\n    ob[pk as string] = id;\n    const m = (\n      typeof clazz === \"string\" ? Model.build(ob, clazz) : new clazz(ob)\n    ) as M;\n    log.silly(`Rebuilding model ${m.constructor.name} id ${id}`);\n    const metadata = obj[PersistenceKeys.METADATA];\n    const result = Object.keys(m).reduce((accum: M, key) => {\n      if (key === pk) return accum;\n      (accum as Record<string, any>)[key] = obj[Repository.column(accum, key)];\n      return accum;\n    }, m);\n\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 result)\n          throw new InternalError(\n            `Transient property ${key} already exists on model ${m.constructor.name}. should be impossible`\n          );\n        result[key as keyof M] = val;\n      });\n    }\n\n    if (metadata) {\n      log.silly(\n        `Passing along ${this.flavour} persistence metadata for ${m.constructor.name} id ${id}: ${metadata}`\n      );\n      Object.defineProperty(result, PersistenceKeys.METADATA, {\n        enumerable: false,\n        configurable: false,\n        writable: false,\n        value: metadata,\n      });\n    }\n\n    return result;\n  }\n\n  /**\n   * @description Creates a new record in the database\n   * @summary Inserts a new record with the given ID and data into the specified table\n   * @param {string} tableName - The name of the table to insert into\n   * @param {string|number} id - The identifier for the new record\n   * @param model - The data to insert\n   * @param {any[]} args - Additional arguments specific to the adapter implementation\n   * @return A promise that resolves to the created record\n   */\n  abstract create(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Creates multiple records in the database\n   * @summary Inserts multiple records with the given IDs and data into the specified table\n   * @param {string} tableName - The name of the table to insert into\n   * @param id - The identifiers for the new records\n   * @param model - The data to insert for each record\n   * @param {...any[]} args - Additional arguments specific to the adapter implementation\n   * @return A promise that resolves to an array of created records\n   */\n  async createAll(\n    tableName: string,\n    id: (string | number)[],\n    model: Record<string, any>[],\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    if (id.length !== model.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n    const log = this.log.for(this.createAll);\n    log.verbose(`Creating ${id.length} entries ${tableName} table`);\n    log.debug(`pks: ${id}`);\n    return Promise.all(\n      id.map((i, count) => this.create(tableName, i, model[count], ...args))\n    );\n  }\n\n  /**\n   * @description Retrieves a record from the database\n   * @summary Fetches a record with the given ID from the specified table\n   * @param {string} tableName - The name of the table to read from\n   * @param {string|number|bigint} id - The identifier of the record to retrieve\n   * @param {...any[]} args - Additional arguments specific to the adapter implementation\n   * @return A promise that resolves to the retrieved record\n   */\n  abstract read(\n    tableName: string,\n    id: string | number | bigint,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Retrieves multiple records from the database\n   * @summary Fetches multiple records with the given IDs from the specified table\n   * @param {string} tableName - The name of the table to read from\n   * @param id - The identifiers of the records to retrieve\n   * @param {...any[]} args - Additional arguments specific to the adapter implementation\n   * @return A promise that resolves to an array of retrieved records\n   */\n  async readAll(\n    tableName: string,\n    id: (string | number | bigint)[],\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    const log = this.log.for(this.readAll);\n    log.verbose(`Reading ${id.length} entries ${tableName} table`);\n    log.debug(`pks: ${id}`);\n    return Promise.all(id.map((i) => this.read(tableName, i, ...args)));\n  }\n\n  /**\n   * @description Updates a record in the database\n   * @summary Modifies an existing record with the given ID in the specified table\n   * @param {string} tableName - The name of the table to update\n   * @param {string|number} id - The identifier of the record to update\n   * @param  model - The new data for the record\n   * @param {...any[]} args - Additional arguments specific to the adapter implementation\n   * @return A promise that resolves to the updated record\n   */\n  abstract update(\n    tableName: string,\n    id: string | number,\n    model: Record<string, any>,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Updates multiple records in the database\n   * @summary Modifies multiple existing records with the given IDs in the specified table\n   * @param {string} tableName - The name of the table to update\n   * @param {string[]|number[]} id - The identifiers of the records to update\n   * @param model - The new data for each record\n   * @param {...any[]} args - Additional arguments specific to the adapter implementation\n   * @return A promise that resolves to an array of updated records\n   */\n  async updateAll(\n    tableName: string,\n    id: string[] | number[],\n    model: Record<string, any>[],\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    if (id.length !== model.length)\n      throw new InternalError(\"Ids and models must have the same length\");\n    const log = this.log.for(this.updateAll);\n    log.verbose(`Updating ${id.length} entries ${tableName} table`);\n    log.debug(`pks: ${id}`);\n    return Promise.all(\n      id.map((i, count) => this.update(tableName, i, model[count], ...args))\n    );\n  }\n\n  /**\n   * @description Deletes a record from the database\n   * @summary Removes a record with the given ID from the specified table\n   * @param {string} tableName - The name of the table to delete from\n   * @param {string|number|bigint} id - The identifier of the record to delete\n   * @param {...any[]} args - Additional arguments specific to the adapter implementation\n   * @return A promise that resolves to the deleted record\n   */\n  abstract delete(\n    tableName: string,\n    id: string | number | bigint,\n    ...args: any[]\n  ): Promise<Record<string, any>>;\n\n  /**\n   * @description Deletes multiple records from the database\n   * @summary Removes multiple records with the given IDs from the specified table\n   * @param {string} tableName - The name of the table to delete from\n   * @param id - The identifiers of the records to delete\n   * @param {...any[]} args - Additional arguments specific to the adapter implementation\n   * @return A promise that resolves to an array of deleted records\n   */\n  async deleteAll(\n    tableName: string,\n    id: (string | number | bigint)[],\n    ...args: any[]\n  ): Promise<Record<string, any>[]> {\n    const log = this.log.for(this.createAll);\n    log.verbose(`Deleting ${id.length} entries ${tableName} table`);\n    log.debug(`pks: ${id}`);\n    return Promise.all(id.map((i) => this.delete(tableName, i, ...args)));\n  }\n\n  /**\n   * @description Executes a raw query against the database\n   * @summary Allows executing database-specific queries directly\n   * @template Q - The raw query type\n   * @template R - The return type of the query\n   * @param {Q} rawInput - The query to execute\n   * @param {...any[]} args - Additional arguments specific to the adapter implementation\n   * @return {Promise<R>} A promise that resolves to the query result\n   */\n  abstract raw<R>(rawInput: Q, ...args: any[]): Promise<R>;\n\n  /**\n   * @description Registers an observer for database events\n   * @summary Adds an observer to be notified about database changes. The observer can optionally\n   * provide a filter function to receive only specific events.\n   * @param {Observer} observer - The observer to register\n   * @param {ObserverFilter} [filter] - Optional filter function to determine which events the observer receives\n   * @return {void}\n   */\n  @final()\n  observe(observer: Observer, filter?: ObserverFilter): void {\n    if (!this.observerHandler)\n      Object.defineProperty(this, \"observerHandler\", {\n        value: this.ObserverHandler(),\n        writable: false,\n      });\n    this.observerHandler!.observe(observer, filter);\n    this.log\n      .for(this.observe)\n      .verbose(`Registering new observer ${observer.toString()}`);\n    if (!this.dispatch) {\n      this.log.for(this.observe).info(`Creating dispatch for ${this.alias}`);\n      this.dispatch = this.Dispatch();\n      this.dispatch.observe(this);\n    }\n  }\n\n  /**\n   * @description Unregisters an observer\n   * @summary Removes a previously registered observer so it no longer receives database event notifications\n   * @param {Observer} observer - The observer to unregister\n   * @return {void}\n   */\n  @final()\n  unObserve(observer: Observer): void {\n    if (!this.observerHandler)\n      throw new InternalError(\n        \"ObserverHandler not initialized. Did you register any observables?\"\n      );\n    this.observerHandler.unObserve(observer);\n    this.log\n      .for(this.unObserve)\n      .verbose(`Observer ${observer.toString()} removed`);\n  }\n\n  /**\n   * @description Notifies all observers about a database event\n   * @summary Sends notifications to all registered observers about a change in the database,\n   * filtering based on each observer's filter function\n   * @param {string} table - The name of the table where the change occurred\n   * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred\n   * @param {EventIds} id - The identifier(s) of the affected record(s)\n   * @param {...any[]} args - Additional arguments to pass to the observers\n   * @return {Promise<void>} A promise that resolves when all observers have been notified\n   */\n  async updateObservers(\n    table: string,\n    event: OperationKeys | BulkCrudOperationKeys | string,\n    id: EventIds,\n    ...args: any[]\n  ): Promise<void> {\n    if (!this.observerHandler)\n      throw new InternalError(\n        \"ObserverHandler not initialized. Did you register any observables?\"\n      );\n    const log = this.log.for(this.updateObservers);\n    log.verbose(\n      `Updating ${this.observerHandler.count()} observers for adapter ${this.alias}`\n    );\n    await this.observerHandler.updateObservers(\n      this.log,\n      table,\n      event,\n      id,\n      ...args\n    );\n  }\n\n  /**\n   * @description Refreshes data based on a database event\n   * @summary Implementation of the Observer interface method that delegates to updateObservers\n   * @param {string} table - The name of the table where the change occurred\n   * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred\n   * @param {EventIds} id - The identifier(s) of the affected record(s)\n   * @param {...any[]} args - Additional arguments related to the event\n   * @return {Promise<void>} A promise that resolves when the refresh is complete\n   */\n  async refresh(\n    table: string,\n    event: OperationKeys | BulkCrudOperationKeys | string,\n    id: EventIds,\n    ...args: any[]\n  ) {\n    return this.updateObservers(table, event, id, ...args);\n  }\n\n  /**\n   * @description Gets a string representation of the adapter\n   * @summary Returns a human-readable string identifying this adapter\n   * @return {string} A string representation of the adapter\n   */\n  toString() {\n    return `${this.flavour} persistence Adapter`;\n  }\n\n  /**\n   * @description Gets the adapter flavor associated with a model\n   * @summary Retrieves the adapter flavor that should be used for a specific model class\n   * @template M - The model type\n   * @param {Constructor<M>} model - The model constructor\n   * @return {string} The adapter flavor name\n   */\n  static flavourOf<M extends Model>(model: Constructor<M>): string {\n    return (\n      Reflect.getMetadata(this.key(PersistenceKeys.ADAPTER), model) ||\n      this.current.flavour\n    );\n  }\n\n  /**\n   * @description Gets the current default adapter\n   * @summary Retrieves the adapter that is currently set as the default for operations\n   * @return {Adapter<any, any, any, any>} The current adapter\n   */\n  static get current() {\n    if (!Adapter._current)\n      throw new InternalError(\n        `No persistence flavour set. Please initialize your adapter`\n      );\n    return Adapter._current;\n  }\n\n  /**\n   * @description Gets an adapter by flavor\n   * @summary Retrieves a registered adapter by its flavor name\n   * @template Y - The database driver type\n   * @template Q - The query type\n   * @template C - The context type\n   * @template F - The repository flags type\n   * @param {string} flavour - The flavor name of the adapter to retrieve\n   * @return {Adapter<Y, Q, F, C> | undefined} The adapter instance or undefined if not found\n   */\n  static get<Y, Q, C extends Context<F>, F extends RepositoryFlags>(\n    flavour: any\n  ): Adapter<Y, Q, F, C> | undefined {\n    if (flavour in this._cache) return this._cache[flavour];\n    throw new InternalError(`No Adapter registered under ${flavour}.`);\n  }\n\n  /**\n   * @description Sets the current default adapter\n   * @summary Changes which adapter is used as the default for operations\n   * @param {string} flavour - The flavor name of the adapter to set as current\n   * @return {void}\n   */\n  static setCurrent(flavour: string) {\n    const adapter = Adapter.get(flavour);\n    if (!adapter)\n      throw new NotFoundError(`No persistence flavour ${flavour} registered`);\n    this._current = adapter;\n  }\n\n  /**\n   * @description Creates a metadata key\n   * @summary Generates a standardized metadata key for persistence-related metadata\n   * @param {string} key - The base key name\n   * @return {string} The formatted metadata key\n   */\n  static key(key: string) {\n    return Repository.key(key);\n  }\n\n  /**\n   * @description Gets all models associated with an adapter flavor\n   * @summary Retrieves all model constructors that are configured to use a specific adapter flavor\n   * @template M - The model type\n   * @param {string} flavour - The adapter flavor to find models for\n   * @return An array of model constructors\n   */\n  static models<M extends Model>(flavour: string) {\n    try {\n      const registry = (Model as any).getRegistry() as ModelRegistry<any>;\n      const cache = (\n        registry as unknown as { cache: Record<string, ModelConstructor<any>> }\n      ).cache;\n      const managedModels: ModelConstructor<any>[] = Object.values(cache)\n        .map((m: ModelConstructor<M>) => {\n          let f = Reflect.getMetadata(\n            Adapter.key(PersistenceKeys.ADAPTER),\n            m as ModelConstructor<any>\n          );\n          if (f && f === flavour) return m;\n          if (!f) {\n            const repo = Reflect.getMetadata(\n              Repository.key(DBKeys.REPOSITORY),\n              m as ModelConstructor<any>\n            );\n            if (!repo) return;\n            const repository = Repository.forModel(m);\n\n            f = Reflect.getMetadata(\n              Adapter.key(PersistenceKeys.ADAPTER),\n              repository\n            );\n            return f;\n          }\n        })\n        .filter((m) => !!m);\n      return managedModels;\n    } catch (e: any) {\n      throw new InternalError(e);\n    }\n  }\n}\n"]}