@decaf-ts/core 0.5.1 → 0.5.3

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 (206) hide show
  1. package/LICENSE.md +21 -157
  2. package/README.md +652 -15
  3. package/dist/core.cjs +2111 -133
  4. package/dist/core.esm.cjs +2112 -134
  5. package/lib/esm/identity/decorators.d.ts +52 -7
  6. package/lib/esm/identity/decorators.js +58 -13
  7. package/lib/esm/identity/index.js +3 -3
  8. package/lib/esm/identity/utils.d.ts +19 -0
  9. package/lib/esm/identity/utils.js +22 -3
  10. package/lib/esm/index.d.ts +10 -3
  11. package/lib/esm/index.js +19 -12
  12. package/lib/esm/interfaces/ErrorParser.d.ts +12 -0
  13. package/lib/esm/interfaces/ErrorParser.js +1 -1
  14. package/lib/esm/interfaces/Executor.d.ts +13 -0
  15. package/lib/esm/interfaces/Executor.js +1 -1
  16. package/lib/esm/interfaces/Observable.d.ts +27 -0
  17. package/lib/esm/interfaces/Observable.js +1 -1
  18. package/lib/esm/interfaces/Observer.d.ts +12 -0
  19. package/lib/esm/interfaces/Observer.js +1 -1
  20. package/lib/esm/interfaces/Paginatable.d.ts +15 -0
  21. package/lib/esm/interfaces/Paginatable.js +1 -1
  22. package/lib/esm/interfaces/Queriable.d.ts +34 -9
  23. package/lib/esm/interfaces/Queriable.js +1 -1
  24. package/lib/esm/interfaces/RawExecutor.d.ts +14 -0
  25. package/lib/esm/interfaces/RawExecutor.js +1 -1
  26. package/lib/esm/interfaces/SequenceOptions.d.ts +52 -0
  27. package/lib/esm/interfaces/SequenceOptions.js +19 -1
  28. package/lib/esm/interfaces/index.js +8 -8
  29. package/lib/esm/model/BaseModel.d.ts +31 -0
  30. package/lib/esm/model/BaseModel.js +24 -1
  31. package/lib/esm/model/construction.d.ts +433 -0
  32. package/lib/esm/model/construction.js +444 -5
  33. package/lib/esm/model/decorators.d.ts +159 -29
  34. package/lib/esm/model/decorators.js +167 -37
  35. package/lib/esm/model/index.js +5 -5
  36. package/lib/esm/model/types.d.ts +9 -0
  37. package/lib/esm/model/types.js +1 -1
  38. package/lib/esm/persistence/Adapter.d.ts +358 -17
  39. package/lib/esm/persistence/Adapter.js +292 -24
  40. package/lib/esm/persistence/Dispatch.d.ts +114 -1
  41. package/lib/esm/persistence/Dispatch.js +104 -6
  42. package/lib/esm/persistence/ObserverHandler.d.ts +95 -0
  43. package/lib/esm/persistence/ObserverHandler.js +96 -1
  44. package/lib/esm/persistence/Sequence.d.ts +89 -0
  45. package/lib/esm/persistence/Sequence.js +71 -2
  46. package/lib/esm/persistence/constants.d.ts +22 -0
  47. package/lib/esm/persistence/constants.js +23 -1
  48. package/lib/esm/persistence/decorators.d.ts +10 -0
  49. package/lib/esm/persistence/decorators.js +13 -3
  50. package/lib/esm/persistence/errors.d.ts +23 -0
  51. package/lib/esm/persistence/errors.js +24 -1
  52. package/lib/esm/persistence/index.js +9 -9
  53. package/lib/esm/persistence/types.d.ts +18 -0
  54. package/lib/esm/persistence/types.js +1 -1
  55. package/lib/esm/query/Condition.d.ts +78 -31
  56. package/lib/esm/query/Condition.js +134 -55
  57. package/lib/esm/query/Paginator.d.ts +56 -0
  58. package/lib/esm/query/Paginator.js +58 -2
  59. package/lib/esm/query/Statement.d.ts +51 -0
  60. package/lib/esm/query/Statement.js +55 -4
  61. package/lib/esm/query/constants.d.ts +25 -0
  62. package/lib/esm/query/constants.js +26 -1
  63. package/lib/esm/query/errors.d.ts +14 -0
  64. package/lib/esm/query/errors.js +15 -1
  65. package/lib/esm/query/index.js +8 -8
  66. package/lib/esm/query/options.d.ts +21 -3
  67. package/lib/esm/query/options.js +1 -1
  68. package/lib/esm/query/selectors.d.ts +26 -0
  69. package/lib/esm/query/selectors.js +1 -1
  70. package/lib/esm/ram/RamAdapter.d.ts +311 -0
  71. package/lib/esm/ram/RamAdapter.js +319 -8
  72. package/lib/esm/ram/RamContext.d.ts +16 -1
  73. package/lib/esm/ram/RamContext.js +18 -3
  74. package/lib/esm/ram/RamPaginator.d.ts +43 -0
  75. package/lib/esm/ram/RamPaginator.js +55 -3
  76. package/lib/esm/ram/RamSequence.d.ts +61 -0
  77. package/lib/esm/ram/RamSequence.js +66 -5
  78. package/lib/esm/ram/RamStatement.d.ts +74 -0
  79. package/lib/esm/ram/RamStatement.js +78 -4
  80. package/lib/esm/ram/constants.d.ts +8 -0
  81. package/lib/esm/ram/constants.js +9 -1
  82. package/lib/esm/ram/handlers.d.ts +19 -0
  83. package/lib/esm/ram/handlers.js +21 -2
  84. package/lib/esm/ram/index.js +11 -11
  85. package/lib/esm/ram/model/RamSequence.d.ts +25 -0
  86. package/lib/esm/ram/model/RamSequence.js +21 -3
  87. package/lib/esm/ram/model/index.js +2 -2
  88. package/lib/esm/ram/types.d.ts +42 -0
  89. package/lib/esm/ram/types.js +1 -1
  90. package/lib/esm/repository/Repository.d.ts +363 -8
  91. package/lib/esm/repository/Repository.js +369 -24
  92. package/lib/esm/repository/constants.d.ts +25 -0
  93. package/lib/esm/repository/constants.js +26 -1
  94. package/lib/esm/repository/decorators.d.ts +27 -0
  95. package/lib/esm/repository/decorators.js +29 -2
  96. package/lib/esm/repository/errors.d.ts +12 -5
  97. package/lib/esm/repository/errors.js +13 -6
  98. package/lib/esm/repository/index.js +8 -8
  99. package/lib/esm/repository/injectables.d.ts +18 -0
  100. package/lib/esm/repository/injectables.js +23 -5
  101. package/lib/esm/repository/types.d.ts +15 -0
  102. package/lib/esm/repository/types.js +1 -1
  103. package/lib/esm/repository/utils.d.ts +11 -0
  104. package/lib/esm/repository/utils.js +15 -4
  105. package/lib/esm/utils/decorators.d.ts +8 -0
  106. package/lib/esm/utils/decorators.js +9 -1
  107. package/lib/esm/utils/errors.d.ts +46 -0
  108. package/lib/esm/utils/errors.js +47 -1
  109. package/lib/esm/utils/index.js +3 -3
  110. package/lib/identity/decorators.cjs +53 -8
  111. package/lib/identity/decorators.d.ts +52 -7
  112. package/lib/identity/utils.cjs +20 -1
  113. package/lib/identity/utils.d.ts +19 -0
  114. package/lib/index.cjs +11 -4
  115. package/lib/index.d.ts +10 -3
  116. package/lib/interfaces/ErrorParser.cjs +1 -1
  117. package/lib/interfaces/ErrorParser.d.ts +12 -0
  118. package/lib/interfaces/Executor.cjs +1 -1
  119. package/lib/interfaces/Executor.d.ts +13 -0
  120. package/lib/interfaces/Observable.cjs +1 -1
  121. package/lib/interfaces/Observable.d.ts +27 -0
  122. package/lib/interfaces/Observer.cjs +1 -1
  123. package/lib/interfaces/Observer.d.ts +12 -0
  124. package/lib/interfaces/Paginatable.cjs +1 -1
  125. package/lib/interfaces/Paginatable.d.ts +15 -0
  126. package/lib/interfaces/Queriable.cjs +1 -1
  127. package/lib/interfaces/Queriable.d.ts +34 -9
  128. package/lib/interfaces/RawExecutor.cjs +1 -1
  129. package/lib/interfaces/RawExecutor.d.ts +14 -0
  130. package/lib/interfaces/SequenceOptions.cjs +19 -1
  131. package/lib/interfaces/SequenceOptions.d.ts +52 -0
  132. package/lib/model/BaseModel.cjs +24 -1
  133. package/lib/model/BaseModel.d.ts +31 -0
  134. package/lib/model/construction.cjs +441 -2
  135. package/lib/model/construction.d.ts +433 -0
  136. package/lib/model/decorators.cjs +160 -30
  137. package/lib/model/decorators.d.ts +159 -29
  138. package/lib/model/types.cjs +1 -1
  139. package/lib/model/types.d.ts +9 -0
  140. package/lib/persistence/Adapter.cjs +287 -19
  141. package/lib/persistence/Adapter.d.ts +358 -17
  142. package/lib/persistence/Dispatch.cjs +102 -4
  143. package/lib/persistence/Dispatch.d.ts +114 -1
  144. package/lib/persistence/ObserverHandler.cjs +96 -1
  145. package/lib/persistence/ObserverHandler.d.ts +95 -0
  146. package/lib/persistence/Sequence.cjs +70 -1
  147. package/lib/persistence/Sequence.d.ts +89 -0
  148. package/lib/persistence/constants.cjs +23 -1
  149. package/lib/persistence/constants.d.ts +22 -0
  150. package/lib/persistence/decorators.cjs +11 -1
  151. package/lib/persistence/decorators.d.ts +10 -0
  152. package/lib/persistence/errors.cjs +24 -1
  153. package/lib/persistence/errors.d.ts +23 -0
  154. package/lib/persistence/types.cjs +1 -1
  155. package/lib/persistence/types.d.ts +18 -0
  156. package/lib/query/Condition.cjs +132 -53
  157. package/lib/query/Condition.d.ts +78 -31
  158. package/lib/query/Paginator.cjs +57 -1
  159. package/lib/query/Paginator.d.ts +56 -0
  160. package/lib/query/Statement.cjs +52 -1
  161. package/lib/query/Statement.d.ts +51 -0
  162. package/lib/query/constants.cjs +26 -1
  163. package/lib/query/constants.d.ts +25 -0
  164. package/lib/query/errors.cjs +15 -1
  165. package/lib/query/errors.d.ts +14 -0
  166. package/lib/query/options.cjs +1 -1
  167. package/lib/query/options.d.ts +21 -3
  168. package/lib/query/selectors.cjs +1 -1
  169. package/lib/query/selectors.d.ts +26 -0
  170. package/lib/ram/RamAdapter.cjs +312 -1
  171. package/lib/ram/RamAdapter.d.ts +311 -0
  172. package/lib/ram/RamContext.cjs +18 -3
  173. package/lib/ram/RamContext.d.ts +16 -1
  174. package/lib/ram/RamPaginator.cjs +54 -2
  175. package/lib/ram/RamPaginator.d.ts +43 -0
  176. package/lib/ram/RamSequence.cjs +63 -2
  177. package/lib/ram/RamSequence.d.ts +61 -0
  178. package/lib/ram/RamStatement.cjs +75 -1
  179. package/lib/ram/RamStatement.d.ts +74 -0
  180. package/lib/ram/constants.cjs +9 -1
  181. package/lib/ram/constants.d.ts +8 -0
  182. package/lib/ram/handlers.cjs +20 -1
  183. package/lib/ram/handlers.d.ts +19 -0
  184. package/lib/ram/model/RamSequence.cjs +19 -1
  185. package/lib/ram/model/RamSequence.d.ts +25 -0
  186. package/lib/ram/types.cjs +1 -1
  187. package/lib/ram/types.d.ts +42 -0
  188. package/lib/repository/Repository.cjs +360 -15
  189. package/lib/repository/Repository.d.ts +363 -8
  190. package/lib/repository/constants.cjs +26 -1
  191. package/lib/repository/constants.d.ts +25 -0
  192. package/lib/repository/decorators.cjs +28 -1
  193. package/lib/repository/decorators.d.ts +27 -0
  194. package/lib/repository/errors.cjs +13 -6
  195. package/lib/repository/errors.d.ts +12 -5
  196. package/lib/repository/injectables.cjs +19 -1
  197. package/lib/repository/injectables.d.ts +18 -0
  198. package/lib/repository/types.cjs +1 -1
  199. package/lib/repository/types.d.ts +15 -0
  200. package/lib/repository/utils.cjs +12 -1
  201. package/lib/repository/utils.d.ts +11 -0
  202. package/lib/utils/decorators.cjs +9 -1
  203. package/lib/utils/decorators.d.ts +8 -0
  204. package/lib/utils/errors.cjs +47 -1
  205. package/lib/utils/errors.d.ts +46 -0
  206. package/package.json +5 -5
@@ -13,19 +13,96 @@ import { Dispatch } from "./Dispatch";
13
13
  import { type EventIds, type ObserverFilter } from "./types";
14
14
  import { ObserverHandler } from "./ObserverHandler";
15
15
  /**
16
- * @summary Abstract Decaf-ts Persistence Adapter Class
17
- * @description Offers the base implementation for all Adapter Classes
18
- * and manages them various registered {@link Adapter}s
16
+ * @description Abstract base class for database adapters
17
+ * @summary Provides the foundation for all database adapters in the persistence layer. This class
18
+ * implements several interfaces to provide a consistent API for database operations, observer
19
+ * pattern support, and error handling. It manages adapter registration, CRUD operations, and
20
+ * observer notifications.
21
+ * @template Y - The underlying database driver type
22
+ * @template Q - The query object type used by the adapter
23
+ * @template F - The repository flags type
24
+ * @template C - The context type
25
+ * @param {Y} _native - The underlying database driver instance
26
+ * @param {string} flavour - The identifier for this adapter type
27
+ * @param {string} [_alias] - Optional alternative name for this adapter
28
+ * @class Adapter
29
+ * @example
30
+ * ```typescript
31
+ * // Implementing a concrete adapter
32
+ * class PostgresAdapter extends Adapter<pg.Client, pg.Query, PostgresFlags, PostgresContext> {
33
+ * constructor(client: pg.Client) {
34
+ * super(client, 'postgres');
35
+ * }
19
36
  *
20
- * @typedef Y the underlying persistence object type or the required config to set it up
21
- * @typedef Q The query object the adapter uses
37
+ * async initialize() {
38
+ * // Set up the adapter
39
+ * await this.native.connect();
40
+ * }
22
41
  *
23
- * @param {Y} native the underlying persistence object
24
- * @param {string} flavour the under witch the persistence adapter should be stored
42
+ * async create(tableName, id, model) {
43
+ * // Implementation for creating records
44
+ * const columns = Object.keys(model).join(', ');
45
+ * const values = Object.values(model);
46
+ * const placeholders = values.map((_, i) => `$${i+1}`).join(', ');
25
47
  *
26
- * @class Adapter
27
- * @implements RawExecutor
28
- * @implements Observable
48
+ * const query = `INSERT INTO ${tableName} (${columns}) VALUES (${placeholders}) RETURNING *`;
49
+ * const result = await this.native.query(query, values);
50
+ * return result.rows[0];
51
+ * }
52
+ *
53
+ * // Other required method implementations...
54
+ * }
55
+ *
56
+ * // Using the adapter
57
+ * const pgClient = new pg.Client(connectionString);
58
+ * const adapter = new PostgresAdapter(pgClient);
59
+ * await adapter.initialize();
60
+ *
61
+ * // Set as the default adapter
62
+ * Adapter.setCurrent('postgres');
63
+ *
64
+ * // Perform operations
65
+ * const user = await adapter.create('users', 1, { name: 'John', email: 'john@example.com' });
66
+ * ```
67
+ * @mermaid
68
+ * classDiagram
69
+ * class Adapter {
70
+ * +Y native
71
+ * +string flavour
72
+ * +string alias
73
+ * +create(tableName, id, model)
74
+ * +read(tableName, id)
75
+ * +update(tableName, id, model)
76
+ * +delete(tableName, id)
77
+ * +observe(observer, filter)
78
+ * +unObserve(observer)
79
+ * +static current
80
+ * +static get(flavour)
81
+ * +static setCurrent(flavour)
82
+ * }
83
+ *
84
+ * class RawExecutor {
85
+ * +raw(query)
86
+ * }
87
+ *
88
+ * class Observable {
89
+ * +observe(observer, filter)
90
+ * +unObserve(observer)
91
+ * +updateObservers(table, event, id)
92
+ * }
93
+ *
94
+ * class Observer {
95
+ * +refresh(table, event, id)
96
+ * }
97
+ *
98
+ * class ErrorParser {
99
+ * +parseError(err)
100
+ * }
101
+ *
102
+ * Adapter --|> RawExecutor
103
+ * Adapter --|> Observable
104
+ * Adapter --|> Observer
105
+ * Adapter --|> ErrorParser
29
106
  */
30
107
  export declare abstract class Adapter<Y, Q, F extends RepositoryFlags, C extends Context<F>> implements RawExecutor<Q>, Contextual<F, C>, Observable, Observer, ErrorParser {
31
108
  private readonly _native;
@@ -36,55 +113,319 @@ export declare abstract class Adapter<Y, Q, F extends RepositoryFlags, C extends
36
113
  private logger;
37
114
  protected dispatch?: Dispatch<Y>;
38
115
  protected readonly observerHandler?: ObserverHandler;
116
+ /**
117
+ * @description Logger accessor
118
+ * @summary Gets or initializes the logger for this adapter instance
119
+ * @return {Logger} The logger instance
120
+ */
39
121
  protected get log(): Logger;
122
+ /**
123
+ * @description Gets the native database driver
124
+ * @summary Provides access to the underlying database driver instance
125
+ * @return {Y} The native database driver
126
+ */
40
127
  get native(): Y;
128
+ /**
129
+ * @description Gets the adapter's alias or flavor name
130
+ * @summary Returns the alias if set, otherwise returns the flavor name
131
+ * @return {string} The adapter's identifier
132
+ */
41
133
  get alias(): string;
134
+ /**
135
+ * @description Gets the repository constructor for this adapter
136
+ * @summary Returns the constructor for creating repositories that work with this adapter
137
+ * @template M - The model type
138
+ * @return {Constructor<Repository<M, Q, Adapter<Y, Q, F, C>, F, C>>} The repository constructor
139
+ */
42
140
  repository<M extends Model>(): Constructor<Repository<M, Q, Adapter<Y, Q, F, C>, F, C>>;
141
+ /**
142
+ * @description Creates a new adapter instance
143
+ * @summary Initializes the adapter with the native driver and registers it in the adapter cache
144
+ */
43
145
  protected constructor(_native: Y, flavour: string, _alias?: string | undefined);
146
+ /**
147
+ * @description Creates a new statement builder for a model
148
+ * @summary Returns a statement builder that can be used to construct queries for a specific model
149
+ * @template M - The model type
150
+ * @return {Statement} A statement builder for the model
151
+ */
44
152
  abstract Statement<M extends Model>(): Statement<Q, M, any>;
153
+ /**
154
+ * @description Creates a new dispatch instance
155
+ * @summary Factory method that creates a dispatch instance for this adapter
156
+ * @return {Dispatch<Y>} A new dispatch instance
157
+ */
45
158
  protected Dispatch(): Dispatch<Y>;
159
+ /**
160
+ * @description Creates a new observer handler
161
+ * @summary Factory method that creates an observer handler for this adapter
162
+ * @return {ObserverHandler} A new observer handler instance
163
+ */
46
164
  protected ObserverHandler(): ObserverHandler;
165
+ /**
166
+ * @description Checks if an attribute name is reserved
167
+ * @summary Determines if a given attribute name is reserved and cannot be used as a column name
168
+ * @param {string} attr - The attribute name to check
169
+ * @return {boolean} True if the attribute is reserved, false otherwise
170
+ */
47
171
  protected isReserved(attr: string): boolean;
172
+ /**
173
+ * @description Parses a database error into a standardized error
174
+ * @summary Converts database-specific errors into standardized application errors
175
+ * @param {Error} err - The original database error
176
+ * @return {BaseError} A standardized error
177
+ */
48
178
  abstract parseError(err: Error): BaseError;
179
+ /**
180
+ * @description Initializes the adapter
181
+ * @summary Performs any necessary setup for the adapter, such as establishing connections
182
+ * @param {...any[]} args - Initialization arguments
183
+ * @return {Promise<void>} A promise that resolves when initialization is complete
184
+ */
49
185
  abstract initialize(...args: any[]): Promise<void>;
186
+ /**
187
+ * @description Creates a sequence generator
188
+ * @summary Factory method that creates a sequence generator for generating sequential values
189
+ * @param {SequenceOptions} options - Configuration options for the sequence
190
+ * @return {Promise<Sequence>} A promise that resolves to a new sequence instance
191
+ */
50
192
  abstract Sequence(options: SequenceOptions): Promise<Sequence>;
193
+ /**
194
+ * @description Creates repository flags for an operation
195
+ * @summary Generates a set of flags that describe a database operation, combining default flags with overrides
196
+ * @template F - The Repository Flags type
197
+ * @template M - The model type
198
+ * @param {OperationKeys} operation - The type of operation being performed
199
+ * @param {Constructor<M>} model - The model constructor
200
+ * @param {Partial<F>} flags - Custom flag overrides
201
+ * @param {...any[]} args - Additional arguments
202
+ * @return {F} The complete set of flags
203
+ */
51
204
  protected flags<M extends Model>(operation: OperationKeys, model: Constructor<M>, flags: Partial<F>, ...args: any[]): F;
52
- protected Context: Constructor<C>;
205
+ /**
206
+ * @description The context constructor for this adapter
207
+ * @summary Reference to the context class constructor used by this adapter
208
+ */
209
+ protected Context: {
210
+ new (): Context<F>;
211
+ factory: import("@decaf-ts/db-decorators").ContextFactory<any>;
212
+ childFrom<F_1 extends RepositoryFlags, C_1 extends Context<F_1>>(context: C_1, overrides?: Partial<F_1>): C_1;
213
+ from<M extends Model, F_1 extends RepositoryFlags, C_1 extends Context<F_1>>(operation: OperationKeys.CREATE | OperationKeys.READ | OperationKeys.UPDATE | OperationKeys.DELETE, overrides: Partial<F_1>, model: Constructor<M>, ...args: any[]): Promise<C_1>;
214
+ args<M extends Model, C_1 extends Context<F_1>, F_1 extends RepositoryFlags>(operation: OperationKeys.CREATE | OperationKeys.READ | OperationKeys.UPDATE | OperationKeys.DELETE, model: Constructor<M>, args: any[], contextual?: Contextual<F_1>, overrides?: Partial<F_1>): Promise<import("@decaf-ts/db-decorators").ContextArgs<F_1, C_1>>;
215
+ };
216
+ /**
217
+ * @description Creates a context for a database operation
218
+ * @summary Generates a context object that describes a database operation, used for tracking and auditing
219
+ * @template F - The Repository flags type
220
+ * @template M - The model type
221
+ * @param {OperationKeys.CREATE|OperationKeys.READ|OperationKeys.UPDATE|OperationKeys.DELETE} operation - The type of operation
222
+ * @param {Partial<F>} overrides - Custom flag overrides
223
+ * @param {Constructor<M>} model - The model constructor
224
+ * @param {...any[]} args - Additional arguments
225
+ * @return {Promise<C>} A promise that resolves to the context object
226
+ */
53
227
  context<M extends Model>(operation: OperationKeys.CREATE | OperationKeys.READ | OperationKeys.UPDATE | OperationKeys.DELETE, overrides: Partial<F>, model: Constructor<M>, ...args: any[]): Promise<C>;
228
+ /**
229
+ * @description Prepares a model for persistence
230
+ * @summary Converts a model instance into a format suitable for database storage,
231
+ * handling column mapping and separating transient properties
232
+ * @template M - The model type
233
+ * @param {M} model - The model instance to prepare
234
+ * @param pk - The primary key property name
235
+ * @return The prepared data
236
+ */
54
237
  prepare<M extends Model>(model: M, pk: keyof M): {
55
238
  record: Record<string, any>;
56
239
  id: string;
57
240
  transient?: Record<string, any>;
58
241
  };
242
+ /**
243
+ * @description Converts database data back into a model instance
244
+ * @summary Reconstructs a model instance from database data, handling column mapping
245
+ * and reattaching transient properties
246
+ * @template M - The model type
247
+ * @param obj - The database record
248
+ * @param {string|Constructor<M>} clazz - The model class or name
249
+ * @param pk - The primary key property name
250
+ * @param {string|number|bigint} id - The primary key value
251
+ * @param [transient] - Transient properties to reattach
252
+ * @return {M} The reconstructed model instance
253
+ */
59
254
  revert<M extends Model>(obj: Record<string, any>, clazz: string | Constructor<M>, pk: keyof M, id: string | number | bigint, transient?: Record<string, any>): M;
255
+ /**
256
+ * @description Creates a new record in the database
257
+ * @summary Inserts a new record with the given ID and data into the specified table
258
+ * @param {string} tableName - The name of the table to insert into
259
+ * @param {string|number} id - The identifier for the new record
260
+ * @param model - The data to insert
261
+ * @param {any[]} args - Additional arguments specific to the adapter implementation
262
+ * @return A promise that resolves to the created record
263
+ */
60
264
  abstract create(tableName: string, id: string | number, model: Record<string, any>, ...args: any[]): Promise<Record<string, any>>;
265
+ /**
266
+ * @description Creates multiple records in the database
267
+ * @summary Inserts multiple records with the given IDs and data into the specified table
268
+ * @param {string} tableName - The name of the table to insert into
269
+ * @param id - The identifiers for the new records
270
+ * @param model - The data to insert for each record
271
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
272
+ * @return A promise that resolves to an array of created records
273
+ */
61
274
  createAll(tableName: string, id: (string | number)[], model: Record<string, any>[], ...args: any[]): Promise<Record<string, any>[]>;
275
+ /**
276
+ * @description Retrieves a record from the database
277
+ * @summary Fetches a record with the given ID from the specified table
278
+ * @param {string} tableName - The name of the table to read from
279
+ * @param {string|number|bigint} id - The identifier of the record to retrieve
280
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
281
+ * @return A promise that resolves to the retrieved record
282
+ */
62
283
  abstract read(tableName: string, id: string | number | bigint, ...args: any[]): Promise<Record<string, any>>;
284
+ /**
285
+ * @description Retrieves multiple records from the database
286
+ * @summary Fetches multiple records with the given IDs from the specified table
287
+ * @param {string} tableName - The name of the table to read from
288
+ * @param id - The identifiers of the records to retrieve
289
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
290
+ * @return A promise that resolves to an array of retrieved records
291
+ */
63
292
  readAll(tableName: string, id: (string | number | bigint)[], ...args: any[]): Promise<Record<string, any>[]>;
293
+ /**
294
+ * @description Updates a record in the database
295
+ * @summary Modifies an existing record with the given ID in the specified table
296
+ * @param {string} tableName - The name of the table to update
297
+ * @param {string|number} id - The identifier of the record to update
298
+ * @param model - The new data for the record
299
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
300
+ * @return A promise that resolves to the updated record
301
+ */
64
302
  abstract update(tableName: string, id: string | number, model: Record<string, any>, ...args: any[]): Promise<Record<string, any>>;
303
+ /**
304
+ * @description Updates multiple records in the database
305
+ * @summary Modifies multiple existing records with the given IDs in the specified table
306
+ * @param {string} tableName - The name of the table to update
307
+ * @param {string[]|number[]} id - The identifiers of the records to update
308
+ * @param model - The new data for each record
309
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
310
+ * @return A promise that resolves to an array of updated records
311
+ */
65
312
  updateAll(tableName: string, id: string[] | number[], model: Record<string, any>[], ...args: any[]): Promise<Record<string, any>[]>;
313
+ /**
314
+ * @description Deletes a record from the database
315
+ * @summary Removes a record with the given ID from the specified table
316
+ * @param {string} tableName - The name of the table to delete from
317
+ * @param {string|number|bigint} id - The identifier of the record to delete
318
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
319
+ * @return A promise that resolves to the deleted record
320
+ */
66
321
  abstract delete(tableName: string, id: string | number | bigint, ...args: any[]): Promise<Record<string, any>>;
322
+ /**
323
+ * @description Deletes multiple records from the database
324
+ * @summary Removes multiple records with the given IDs from the specified table
325
+ * @param {string} tableName - The name of the table to delete from
326
+ * @param id - The identifiers of the records to delete
327
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
328
+ * @return A promise that resolves to an array of deleted records
329
+ */
67
330
  deleteAll(tableName: string, id: (string | number | bigint)[], ...args: any[]): Promise<Record<string, any>[]>;
331
+ /**
332
+ * @description Executes a raw query against the database
333
+ * @summary Allows executing database-specific queries directly
334
+ * @template Q - The raw query type
335
+ * @template R - The return type of the query
336
+ * @param {Q} rawInput - The query to execute
337
+ * @param {...any[]} args - Additional arguments specific to the adapter implementation
338
+ * @return {Promise<R>} A promise that resolves to the query result
339
+ */
68
340
  abstract raw<R>(rawInput: Q, ...args: any[]): Promise<R>;
69
341
  /**
70
- *
71
- * @see {Observable#observe}
342
+ * @description Registers an observer for database events
343
+ * @summary Adds an observer to be notified about database changes. The observer can optionally
344
+ * provide a filter function to receive only specific events.
345
+ * @param {Observer} observer - The observer to register
346
+ * @param {ObserverFilter} [filter] - Optional filter function to determine which events the observer receives
347
+ * @return {void}
72
348
  */
73
349
  observe(observer: Observer, filter?: ObserverFilter): void;
74
350
  /**
75
- * @summary Unregisters an {@link Observer}
76
- * @param {Observer} observer
77
- *
78
- * @see {Observable#unObserve}
351
+ * @description Unregisters an observer
352
+ * @summary Removes a previously registered observer so it no longer receives database event notifications
353
+ * @param {Observer} observer - The observer to unregister
354
+ * @return {void}
79
355
  */
80
356
  unObserve(observer: Observer): void;
357
+ /**
358
+ * @description Notifies all observers about a database event
359
+ * @summary Sends notifications to all registered observers about a change in the database,
360
+ * filtering based on each observer's filter function
361
+ * @param {string} table - The name of the table where the change occurred
362
+ * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
363
+ * @param {EventIds} id - The identifier(s) of the affected record(s)
364
+ * @param {...any[]} args - Additional arguments to pass to the observers
365
+ * @return {Promise<void>} A promise that resolves when all observers have been notified
366
+ */
81
367
  updateObservers(table: string, event: OperationKeys | BulkCrudOperationKeys | string, id: EventIds, ...args: any[]): Promise<void>;
368
+ /**
369
+ * @description Refreshes data based on a database event
370
+ * @summary Implementation of the Observer interface method that delegates to updateObservers
371
+ * @param {string} table - The name of the table where the change occurred
372
+ * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
373
+ * @param {EventIds} id - The identifier(s) of the affected record(s)
374
+ * @param {...any[]} args - Additional arguments related to the event
375
+ * @return {Promise<void>} A promise that resolves when the refresh is complete
376
+ */
82
377
  refresh(table: string, event: OperationKeys | BulkCrudOperationKeys | string, id: EventIds, ...args: any[]): Promise<void>;
378
+ /**
379
+ * @description Gets a string representation of the adapter
380
+ * @summary Returns a human-readable string identifying this adapter
381
+ * @return {string} A string representation of the adapter
382
+ */
83
383
  toString(): string;
384
+ /**
385
+ * @description Gets the adapter flavor associated with a model
386
+ * @summary Retrieves the adapter flavor that should be used for a specific model class
387
+ * @template M - The model type
388
+ * @param {Constructor<M>} model - The model constructor
389
+ * @return {string} The adapter flavor name
390
+ */
84
391
  static flavourOf<M extends Model>(model: Constructor<M>): string;
392
+ /**
393
+ * @description Gets the current default adapter
394
+ * @summary Retrieves the adapter that is currently set as the default for operations
395
+ * @return {Adapter<any, any, any, any>} The current adapter
396
+ */
85
397
  static get current(): Adapter<any, any, any, any>;
398
+ /**
399
+ * @description Gets an adapter by flavor
400
+ * @summary Retrieves a registered adapter by its flavor name
401
+ * @template Y - The database driver type
402
+ * @template Q - The query type
403
+ * @template C - The context type
404
+ * @template F - The repository flags type
405
+ * @param {string} flavour - The flavor name of the adapter to retrieve
406
+ * @return {Adapter<Y, Q, F, C> | undefined} The adapter instance or undefined if not found
407
+ */
86
408
  static get<Y, Q, C extends Context<F>, F extends RepositoryFlags>(flavour: any): Adapter<Y, Q, F, C> | undefined;
409
+ /**
410
+ * @description Sets the current default adapter
411
+ * @summary Changes which adapter is used as the default for operations
412
+ * @param {string} flavour - The flavor name of the adapter to set as current
413
+ * @return {void}
414
+ */
87
415
  static setCurrent(flavour: string): void;
416
+ /**
417
+ * @description Creates a metadata key
418
+ * @summary Generates a standardized metadata key for persistence-related metadata
419
+ * @param {string} key - The base key name
420
+ * @return {string} The formatted metadata key
421
+ */
88
422
  static key(key: string): string;
423
+ /**
424
+ * @description Gets all models associated with an adapter flavor
425
+ * @summary Retrieves all model constructors that are configured to use a specific adapter flavor
426
+ * @template M - The model type
427
+ * @param {string} flavour - The adapter flavor to find models for
428
+ * @return An array of model constructors
429
+ */
89
430
  static models<M extends Model>(flavour: string): ModelConstructor<any>[];
90
431
  }
@@ -5,14 +5,85 @@ const db_decorators_1 = require("@decaf-ts/db-decorators");
5
5
  const Adapter_1 = require("./Adapter.cjs");
6
6
  const errors_1 = require("./errors.cjs");
7
7
  const logging_1 = require("@decaf-ts/logging");
8
+ /**
9
+ * @description Dispatches database operation events to observers
10
+ * @summary The Dispatch class implements the Observable interface and is responsible for intercepting
11
+ * database operations from an Adapter and notifying observers when changes occur. It uses proxies to
12
+ * wrap the adapter's CRUD methods and automatically trigger observer updates after operations complete.
13
+ * @template Y - The native database driver type
14
+ * @param {void} - No constructor parameters
15
+ * @class Dispatch
16
+ * @example
17
+ * ```typescript
18
+ * // Creating and using a Dispatch instance
19
+ * const dispatch = new Dispatch<PostgresDriver>();
20
+ *
21
+ * // Connect it to an adapter
22
+ * const adapter = new PostgresAdapter(connection);
23
+ * dispatch.observe(adapter);
24
+ *
25
+ * // Now any CRUD operations on the adapter will automatically
26
+ * // trigger observer notifications
27
+ * await adapter.create('users', 123, userModel);
28
+ * // Observers will be notified about the creation
29
+ *
30
+ * // When done, you can disconnect
31
+ * dispatch.unObserve(adapter);
32
+ * ```
33
+ */
8
34
  class Dispatch {
35
+ /**
36
+ * @description Accessor for the logger
37
+ * @summary Gets or initializes the logger for this dispatch instance
38
+ * @return {Logger} The logger instance
39
+ */
9
40
  get log() {
10
41
  if (!this.logger)
11
42
  this.logger = logging_1.Logging.for(this).for(this.adapter);
12
43
  return this.logger;
13
44
  }
45
+ /**
46
+ * @description Creates a new Dispatch instance
47
+ * @summary Initializes a new Dispatch instance without any adapter
48
+ */
14
49
  constructor() { }
15
- initialize() {
50
+ /**
51
+ * @description Initializes the dispatch by proxying adapter methods
52
+ * @summary Sets up proxies on the adapter's CRUD methods to intercept operations and notify observers.
53
+ * This method is called automatically when an adapter is observed.
54
+ * @return {Promise<void>} A promise that resolves when initialization is complete
55
+ * @mermaid
56
+ * sequenceDiagram
57
+ * participant Dispatch
58
+ * participant Adapter
59
+ * participant Proxy
60
+ *
61
+ * Dispatch->>Dispatch: initialize()
62
+ * Dispatch->>Dispatch: Check if adapter exists
63
+ * alt No adapter
64
+ * Dispatch-->>Dispatch: Throw InternalError
65
+ * end
66
+ *
67
+ * loop For each CRUD method
68
+ * Dispatch->>Adapter: Check if method exists
69
+ * alt Method doesn't exist
70
+ * Dispatch-->>Dispatch: Throw InternalError
71
+ * end
72
+ *
73
+ * Dispatch->>Adapter: Get property descriptor
74
+ * loop While descriptor not found
75
+ * Dispatch->>Adapter: Check prototype chain
76
+ * end
77
+ *
78
+ * alt Descriptor not found or not writable
79
+ * Dispatch->>Dispatch: Log error and continue
80
+ * else Descriptor found and writable
81
+ * Dispatch->>Proxy: Create proxy for method
82
+ * Dispatch->>Adapter: Replace method with proxy
83
+ * end
84
+ * end
85
+ */
86
+ async initialize() {
16
87
  if (!this.adapter)
17
88
  throw new db_decorators_1.InternalError(`No adapter observed for dispatch`);
18
89
  const adapter = this.adapter;
@@ -64,20 +135,47 @@ class Dispatch {
64
135
  });
65
136
  });
66
137
  }
138
+ /**
139
+ * @description Closes the dispatch
140
+ * @summary Performs any necessary cleanup when the dispatch is no longer needed
141
+ * @return {Promise<void>} A promise that resolves when closing is complete
142
+ */
143
+ async close() {
144
+ // to nothing in this instance but may be required for closing connections
145
+ }
146
+ /**
147
+ * @description Starts observing an adapter
148
+ * @summary Connects this dispatch to an adapter to monitor its operations
149
+ * @param {Adapter<Y, any, any, any>} observer - The adapter to observe
150
+ * @return {void}
151
+ */
67
152
  observe(observer) {
68
153
  if (!(observer instanceof Adapter_1.Adapter))
69
154
  throw new errors_1.UnsupportedError("Only Adapters can be observed by dispatch");
70
155
  this.adapter = observer;
71
156
  this.native = observer.native;
72
157
  this.models = Adapter_1.Adapter.models(this.adapter.alias);
73
- this.initialize();
74
- this.log.verbose(`Dispatch initialized for ${this.adapter.alias} adapter`);
158
+ this.initialize().then(() => this.log.verbose(`Dispatch initialized for ${this.adapter.alias} adapter`));
75
159
  }
160
+ /**
161
+ * @description Stops observing an adapter
162
+ * @summary Disconnects this dispatch from an adapter
163
+ * @param {Observer} observer - The adapter to stop observing
164
+ * @return {void}
165
+ */
76
166
  unObserve(observer) {
77
167
  if (this.adapter !== observer)
78
168
  throw new errors_1.UnsupportedError("Only the adapter that was used to observe can be unobserved");
79
169
  this.adapter = undefined;
80
170
  }
171
+ /**
172
+ * @description Updates observers about a database event
173
+ * @summary Notifies observers about a change in the database
174
+ * @param {string} table - The name of the table where the change occurred
175
+ * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
176
+ * @param {EventIds} id - The identifier(s) of the affected record(s)
177
+ * @return {Promise<void>} A promise that resolves when all observers have been notified
178
+ */
81
179
  async updateObservers(table, event, id) {
82
180
  if (!this.adapter)
83
181
  throw new db_decorators_1.InternalError(`No adapter observed for dispatch`);
@@ -90,4 +188,4 @@ class Dispatch {
90
188
  }
91
189
  }
92
190
  exports.Dispatch = Dispatch;
93
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Dispatch.js","sourceRoot":"","sources":["../../src/persistence/Dispatch.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  InternalError,\n  OperationKeys,\n  BulkCrudOperationKeys,\n} from \"@decaf-ts/db-decorators\";\nimport { ModelConstructor } from \"@decaf-ts/decorator-validation\";\nimport { Observable, Observer } from \"../interfaces\";\nimport { Adapter } from \"./Adapter\";\nimport { UnsupportedError } from \"./errors\";\nimport { Logger, Logging } from \"@decaf-ts/logging\";\nimport { EventIds } from \"./types\";\n\nexport class Dispatch<Y> implements Observable {\n  protected adapter?: Adapter<Y, any, any, any>;\n  protected native?: Y;\n  protected models!: ModelConstructor<any>[];\n\n  private logger!: Logger;\n\n  protected get log() {\n    if (!this.logger)\n      this.logger = Logging.for(this as any).for(this.adapter as any);\n    return this.logger;\n  }\n\n  constructor() {}\n\n  protected initialize(): void {\n    if (!this.adapter)\n      throw new InternalError(`No adapter observed for dispatch`);\n    const adapter = this.adapter as Adapter<Y, any, any, any>;\n    (\n      [\n        OperationKeys.CREATE,\n        OperationKeys.UPDATE,\n        OperationKeys.DELETE,\n        BulkCrudOperationKeys.CREATE_ALL,\n        BulkCrudOperationKeys.UPDATE_ALL,\n        BulkCrudOperationKeys.DELETE_ALL,\n      ] as (keyof Adapter<Y, any, any, any>)[]\n    ).forEach((method) => {\n      if (!adapter[method])\n        throw new InternalError(\n          `Method ${method} not found in ${adapter.alias} adapter to bind Observables Dispatch`\n        );\n\n      let descriptor = Object.getOwnPropertyDescriptor(adapter, method);\n      let proto: any = adapter;\n      while (!descriptor && proto !== Object.prototype) {\n        proto = Object.getPrototypeOf(proto);\n        descriptor = Object.getOwnPropertyDescriptor(proto, method);\n      }\n\n      if (!descriptor || !descriptor.writable) {\n        this.log.error(\n          `Could not find method ${method} to bind Observables Dispatch`\n        );\n        return;\n      }\n      function bulkToSingle(method: string) {\n        switch (method) {\n          case BulkCrudOperationKeys.CREATE_ALL:\n            return OperationKeys.CREATE;\n          case BulkCrudOperationKeys.UPDATE_ALL:\n            return OperationKeys.UPDATE;\n          case BulkCrudOperationKeys.DELETE_ALL:\n            return OperationKeys.DELETE;\n          default:\n            return method;\n        }\n      }\n      // @ts-expect-error because there are read only properties\n      adapter[method] = new Proxy(adapter[method], {\n        apply: async (target: any, thisArg, argArray: any[]) => {\n          const [tableName, ids] = argArray;\n          const result = await target.apply(thisArg, argArray);\n          this.updateObservers(tableName, bulkToSingle(method), ids as EventIds)\n            .then(() => {\n              this.log.verbose(\n                `Observer refresh dispatched by ${method} for ${tableName}`\n              );\n              this.log.debug(`pks: ${ids}`);\n            })\n            .catch((e: unknown) =>\n              this.log.error(\n                `Failed to dispatch observer refresh for ${method} on ${tableName}: ${e}`\n              )\n            );\n          return result;\n        },\n      });\n    });\n  }\n\n  observe(observer: Adapter<Y, any, any, any>): void {\n    if (!(observer instanceof Adapter))\n      throw new UnsupportedError(\"Only Adapters can be observed by dispatch\");\n    this.adapter = observer;\n    this.native = observer.native;\n    this.models = Adapter.models(this.adapter.alias);\n    this.initialize();\n    this.log.verbose(`Dispatch initialized for ${this.adapter.alias} adapter`);\n  }\n\n  unObserve(observer: Observer): void {\n    if (this.adapter !== observer)\n      throw new UnsupportedError(\n        \"Only the adapter that was used to observe can be unobserved\"\n      );\n    this.adapter = undefined;\n  }\n\n  async updateObservers(\n    table: string,\n    event: OperationKeys | BulkCrudOperationKeys | string,\n    id: EventIds\n  ): Promise<void> {\n    if (!this.adapter)\n      throw new InternalError(`No adapter observed for dispatch`);\n    try {\n      await this.adapter.refresh(table, event, id);\n    } catch (e: unknown) {\n      throw new InternalError(`Failed to refresh dispatch: ${e}`);\n    }\n  }\n}\n"]}
191
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Dispatch.js","sourceRoot":"","sources":["../../src/persistence/Dispatch.ts"],"names":[],"mappings":";;;AAAA,2DAIiC;AAGjC,2CAAoC;AACpC,yCAA4C;AAC5C,+CAAoD;AAGpD;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AACH,MAAa,QAAQ;IAyBnB;;;;OAIG;IACH,IAAc,GAAG;QACf,IAAI,CAAC,IAAI,CAAC,MAAM;YACd,IAAI,CAAC,MAAM,GAAG,iBAAO,CAAC,GAAG,CAAC,IAAW,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,OAAc,CAAC,CAAC;QAClE,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED;;;OAGG;IACH,gBAAe,CAAC;IAEhB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OAmCG;IACO,KAAK,CAAC,UAAU;QACxB,IAAI,CAAC,IAAI,CAAC,OAAO;YACf,MAAM,IAAI,6BAAa,CAAC,kCAAkC,CAAC,CAAC;QAC9D,MAAM,OAAO,GAAG,IAAI,CAAC,OAAoC,CAAC;QAExD;YACE,6BAAa,CAAC,MAAM;YACpB,6BAAa,CAAC,MAAM;YACpB,6BAAa,CAAC,MAAM;YACpB,qCAAqB,CAAC,UAAU;YAChC,qCAAqB,CAAC,UAAU;YAChC,qCAAqB,CAAC,UAAU;SAEnC,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE;YACnB,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC;gBAClB,MAAM,IAAI,6BAAa,CACrB,UAAU,MAAM,iBAAiB,OAAO,CAAC,KAAK,uCAAuC,CACtF,CAAC;YAEJ,IAAI,UAAU,GAAG,MAAM,CAAC,wBAAwB,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;YAClE,IAAI,KAAK,GAAQ,OAAO,CAAC;YACzB,OAAO,CAAC,UAAU,IAAI,KAAK,KAAK,MAAM,CAAC,SAAS,EAAE,CAAC;gBACjD,KAAK,GAAG,MAAM,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;gBACrC,UAAU,GAAG,MAAM,CAAC,wBAAwB,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;YAC9D,CAAC;YAED,IAAI,CAAC,UAAU,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;gBACxC,IAAI,CAAC,GAAG,CAAC,KAAK,CACZ,yBAAyB,MAAM,+BAA+B,CAC/D,CAAC;gBACF,OAAO;YACT,CAAC;YACD,SAAS,YAAY,CAAC,MAAc;gBAClC,QAAQ,MAAM,EAAE,CAAC;oBACf,KAAK,qCAAqB,CAAC,UAAU;wBACnC,OAAO,6BAAa,CAAC,MAAM,CAAC;oBAC9B,KAAK,qCAAqB,CAAC,UAAU;wBACnC,OAAO,6BAAa,CAAC,MAAM,CAAC;oBAC9B,KAAK,qCAAqB,CAAC,UAAU;wBACnC,OAAO,6BAAa,CAAC,MAAM,CAAC;oBAC9B;wBACE,OAAO,MAAM,CAAC;gBAClB,CAAC;YACH,CAAC;YACD,0DAA0D;YAC1D,OAAO,CAAC,MAAM,CAAC,GAAG,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;gBAC3C,KAAK,EAAE,KAAK,EAAE,MAAW,EAAE,OAAO,EAAE,QAAe,EAAE,EAAE;oBACrD,MAAM,CAAC,SAAS,EAAE,GAAG,CAAC,GAAG,QAAQ,CAAC;oBAClC,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;oBACrD,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,YAAY,CAAC,MAAM,CAAC,EAAE,GAAe,CAAC;yBACnE,IAAI,CAAC,GAAG,EAAE;wBACT,IAAI,CAAC,GAAG,CAAC,OAAO,CACd,kCAAkC,MAAM,QAAQ,SAAS,EAAE,CAC5D,CAAC;wBACF,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,GAAG,EAAE,CAAC,CAAC;oBAChC,CAAC,CAAC;yBACD,KAAK,CAAC,CAAC,CAAU,EAAE,EAAE,CACpB,IAAI,CAAC,GAAG,CAAC,KAAK,CACZ,2CAA2C,MAAM,OAAO,SAAS,KAAK,CAAC,EAAE,CAC1E,CACF,CAAC;oBACJ,OAAO,MAAM,CAAC;gBAChB,CAAC;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,KAAK;QACT,0EAA0E;IAC5E,CAAC;IAED;;;;;OAKG;IACH,OAAO,CAAC,QAAmC;QACzC,IAAI,CAAC,CAAC,QAAQ,YAAY,iBAAO,CAAC;YAChC,MAAM,IAAI,yBAAgB,CAAC,2CAA2C,CAAC,CAAC;QAC1E,IAAI,CAAC,OAAO,GAAG,QAAQ,CAAC;QACxB,IAAI,CAAC,MAAM,GAAG,QAAQ,CAAC,MAAM,CAAC;QAC9B,IAAI,CAAC,MAAM,GAAG,iBAAO,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QACjD,IAAI,CAAC,UAAU,EAAE,CAAC,IAAI,CAAC,GAAG,EAAE,CAC1B,IAAI,CAAC,GAAG,CAAC,OAAO,CACd,4BAA4B,IAAI,CAAC,OAAQ,CAAC,KAAK,UAAU,CAC1D,CACF,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACH,SAAS,CAAC,QAAkB;QAC1B,IAAI,IAAI,CAAC,OAAO,KAAK,QAAQ;YAC3B,MAAM,IAAI,yBAAgB,CACxB,6DAA6D,CAC9D,CAAC;QACJ,IAAI,CAAC,OAAO,GAAG,SAAS,CAAC;IAC3B,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,eAAe,CACnB,KAAa,EACb,KAAqD,EACrD,EAAY;QAEZ,IAAI,CAAC,IAAI,CAAC,OAAO;YACf,MAAM,IAAI,6BAAa,CAAC,kCAAkC,CAAC,CAAC;QAC9D,IAAI,CAAC;YACH,MAAM,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE,CAAC,CAAC;QAC/C,CAAC;QAAC,OAAO,CAAU,EAAE,CAAC;YACpB,MAAM,IAAI,6BAAa,CAAC,+BAA+B,CAAC,EAAE,CAAC,CAAC;QAC9D,CAAC;IACH,CAAC;CACF;AAhND,4BAgNC","sourcesContent":["import {\n  InternalError,\n  OperationKeys,\n  BulkCrudOperationKeys,\n} from \"@decaf-ts/db-decorators\";\nimport { ModelConstructor } from \"@decaf-ts/decorator-validation\";\nimport { Observable, Observer } from \"../interfaces\";\nimport { Adapter } from \"./Adapter\";\nimport { UnsupportedError } from \"./errors\";\nimport { Logger, Logging } from \"@decaf-ts/logging\";\nimport { EventIds } from \"./types\";\n\n/**\n * @description Dispatches database operation events to observers\n * @summary The Dispatch class implements the Observable interface and is responsible for intercepting\n * database operations from an Adapter and notifying observers when changes occur. It uses proxies to\n * wrap the adapter's CRUD methods and automatically trigger observer updates after operations complete.\n * @template Y - The native database driver type\n * @param {void} - No constructor parameters\n * @class Dispatch\n * @example\n * ```typescript\n * // Creating and using a Dispatch instance\n * const dispatch = new Dispatch<PostgresDriver>();\n *\n * // Connect it to an adapter\n * const adapter = new PostgresAdapter(connection);\n * dispatch.observe(adapter);\n *\n * // Now any CRUD operations on the adapter will automatically\n * // trigger observer notifications\n * await adapter.create('users', 123, userModel);\n * // Observers will be notified about the creation\n *\n * // When done, you can disconnect\n * dispatch.unObserve(adapter);\n * ```\n */\nexport class Dispatch<Y> implements Observable {\n  /**\n   * @description The adapter being observed\n   * @summary Reference to the database adapter whose operations are being monitored\n   */\n  protected adapter?: Adapter<Y, any, any, any>;\n\n  /**\n   * @description The native database driver\n   * @summary Reference to the underlying database driver from the adapter\n   */\n  protected native?: Y;\n\n  /**\n   * @description List of model constructors\n   * @summary Array of model constructors that are registered with the adapter\n   */\n  protected models!: ModelConstructor<any>[];\n\n  /**\n   * @description Logger instance\n   * @summary Logger for recording dispatch activities\n   */\n  private logger!: Logger;\n\n  /**\n   * @description Accessor for the logger\n   * @summary Gets or initializes the logger for this dispatch instance\n   * @return {Logger} The logger instance\n   */\n  protected get log() {\n    if (!this.logger)\n      this.logger = Logging.for(this as any).for(this.adapter as any);\n    return this.logger;\n  }\n\n  /**\n   * @description Creates a new Dispatch instance\n   * @summary Initializes a new Dispatch instance without any adapter\n   */\n  constructor() {}\n\n  /**\n   * @description Initializes the dispatch by proxying adapter methods\n   * @summary Sets up proxies on the adapter's CRUD methods to intercept operations and notify observers.\n   * This method is called automatically when an adapter is observed.\n   * @return {Promise<void>} A promise that resolves when initialization is complete\n   * @mermaid\n   * sequenceDiagram\n   *   participant Dispatch\n   *   participant Adapter\n   *   participant Proxy\n   *\n   *   Dispatch->>Dispatch: initialize()\n   *   Dispatch->>Dispatch: Check if adapter exists\n   *   alt No adapter\n   *     Dispatch-->>Dispatch: Throw InternalError\n   *   end\n   *\n   *   loop For each CRUD method\n   *     Dispatch->>Adapter: Check if method exists\n   *     alt Method doesn't exist\n   *       Dispatch-->>Dispatch: Throw InternalError\n   *     end\n   *\n   *     Dispatch->>Adapter: Get property descriptor\n   *     loop While descriptor not found\n   *       Dispatch->>Adapter: Check prototype chain\n   *     end\n   *\n   *     alt Descriptor not found or not writable\n   *       Dispatch->>Dispatch: Log error and continue\n   *     else Descriptor found and writable\n   *       Dispatch->>Proxy: Create proxy for method\n   *       Dispatch->>Adapter: Replace method with proxy\n   *     end\n   *   end\n   */\n  protected async initialize(): Promise<void> {\n    if (!this.adapter)\n      throw new InternalError(`No adapter observed for dispatch`);\n    const adapter = this.adapter as Adapter<Y, any, any, any>;\n    (\n      [\n        OperationKeys.CREATE,\n        OperationKeys.UPDATE,\n        OperationKeys.DELETE,\n        BulkCrudOperationKeys.CREATE_ALL,\n        BulkCrudOperationKeys.UPDATE_ALL,\n        BulkCrudOperationKeys.DELETE_ALL,\n      ] as (keyof Adapter<Y, any, any, any>)[]\n    ).forEach((method) => {\n      if (!adapter[method])\n        throw new InternalError(\n          `Method ${method} not found in ${adapter.alias} adapter to bind Observables Dispatch`\n        );\n\n      let descriptor = Object.getOwnPropertyDescriptor(adapter, method);\n      let proto: any = adapter;\n      while (!descriptor && proto !== Object.prototype) {\n        proto = Object.getPrototypeOf(proto);\n        descriptor = Object.getOwnPropertyDescriptor(proto, method);\n      }\n\n      if (!descriptor || !descriptor.writable) {\n        this.log.error(\n          `Could not find method ${method} to bind Observables Dispatch`\n        );\n        return;\n      }\n      function bulkToSingle(method: string) {\n        switch (method) {\n          case BulkCrudOperationKeys.CREATE_ALL:\n            return OperationKeys.CREATE;\n          case BulkCrudOperationKeys.UPDATE_ALL:\n            return OperationKeys.UPDATE;\n          case BulkCrudOperationKeys.DELETE_ALL:\n            return OperationKeys.DELETE;\n          default:\n            return method;\n        }\n      }\n      // @ts-expect-error because there are read only properties\n      adapter[method] = new Proxy(adapter[method], {\n        apply: async (target: any, thisArg, argArray: any[]) => {\n          const [tableName, ids] = argArray;\n          const result = await target.apply(thisArg, argArray);\n          this.updateObservers(tableName, bulkToSingle(method), ids as EventIds)\n            .then(() => {\n              this.log.verbose(\n                `Observer refresh dispatched by ${method} for ${tableName}`\n              );\n              this.log.debug(`pks: ${ids}`);\n            })\n            .catch((e: unknown) =>\n              this.log.error(\n                `Failed to dispatch observer refresh for ${method} on ${tableName}: ${e}`\n              )\n            );\n          return result;\n        },\n      });\n    });\n  }\n\n  /**\n   * @description Closes the dispatch\n   * @summary Performs any necessary cleanup when the dispatch is no longer needed\n   * @return {Promise<void>} A promise that resolves when closing is complete\n   */\n  async close() {\n    // to nothing in this instance but may be required for closing connections\n  }\n\n  /**\n   * @description Starts observing an adapter\n   * @summary Connects this dispatch to an adapter to monitor its operations\n   * @param {Adapter<Y, any, any, any>} observer - The adapter to observe\n   * @return {void}\n   */\n  observe(observer: Adapter<Y, any, any, any>): void {\n    if (!(observer instanceof Adapter))\n      throw new UnsupportedError(\"Only Adapters can be observed by dispatch\");\n    this.adapter = observer;\n    this.native = observer.native;\n    this.models = Adapter.models(this.adapter.alias);\n    this.initialize().then(() =>\n      this.log.verbose(\n        `Dispatch initialized for ${this.adapter!.alias} adapter`\n      )\n    );\n  }\n\n  /**\n   * @description Stops observing an adapter\n   * @summary Disconnects this dispatch from an adapter\n   * @param {Observer} observer - The adapter to stop observing\n   * @return {void}\n   */\n  unObserve(observer: Observer): void {\n    if (this.adapter !== observer)\n      throw new UnsupportedError(\n        \"Only the adapter that was used to observe can be unobserved\"\n      );\n    this.adapter = undefined;\n  }\n\n  /**\n   * @description Updates observers about a database event\n   * @summary Notifies observers about a change in the database\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   * @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  ): Promise<void> {\n    if (!this.adapter)\n      throw new InternalError(`No adapter observed for dispatch`);\n    try {\n      await this.adapter.refresh(table, event, id);\n    } catch (e: unknown) {\n      throw new InternalError(`Failed to refresh dispatch: ${e}`);\n    }\n  }\n}\n"]}