@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.
- package/LICENSE.md +21 -157
- package/README.md +652 -15
- package/dist/core.cjs +2111 -133
- package/dist/core.esm.cjs +2112 -134
- package/lib/esm/identity/decorators.d.ts +52 -7
- package/lib/esm/identity/decorators.js +58 -13
- package/lib/esm/identity/index.js +3 -3
- package/lib/esm/identity/utils.d.ts +19 -0
- package/lib/esm/identity/utils.js +22 -3
- package/lib/esm/index.d.ts +10 -3
- package/lib/esm/index.js +19 -12
- package/lib/esm/interfaces/ErrorParser.d.ts +12 -0
- package/lib/esm/interfaces/ErrorParser.js +1 -1
- package/lib/esm/interfaces/Executor.d.ts +13 -0
- package/lib/esm/interfaces/Executor.js +1 -1
- package/lib/esm/interfaces/Observable.d.ts +27 -0
- package/lib/esm/interfaces/Observable.js +1 -1
- package/lib/esm/interfaces/Observer.d.ts +12 -0
- package/lib/esm/interfaces/Observer.js +1 -1
- package/lib/esm/interfaces/Paginatable.d.ts +15 -0
- package/lib/esm/interfaces/Paginatable.js +1 -1
- package/lib/esm/interfaces/Queriable.d.ts +34 -9
- package/lib/esm/interfaces/Queriable.js +1 -1
- package/lib/esm/interfaces/RawExecutor.d.ts +14 -0
- package/lib/esm/interfaces/RawExecutor.js +1 -1
- package/lib/esm/interfaces/SequenceOptions.d.ts +52 -0
- package/lib/esm/interfaces/SequenceOptions.js +19 -1
- package/lib/esm/interfaces/index.js +8 -8
- package/lib/esm/model/BaseModel.d.ts +31 -0
- package/lib/esm/model/BaseModel.js +24 -1
- package/lib/esm/model/construction.d.ts +433 -0
- package/lib/esm/model/construction.js +444 -5
- package/lib/esm/model/decorators.d.ts +159 -29
- package/lib/esm/model/decorators.js +167 -37
- package/lib/esm/model/index.js +5 -5
- package/lib/esm/model/types.d.ts +9 -0
- package/lib/esm/model/types.js +1 -1
- package/lib/esm/persistence/Adapter.d.ts +358 -17
- package/lib/esm/persistence/Adapter.js +292 -24
- package/lib/esm/persistence/Dispatch.d.ts +114 -1
- package/lib/esm/persistence/Dispatch.js +104 -6
- package/lib/esm/persistence/ObserverHandler.d.ts +95 -0
- package/lib/esm/persistence/ObserverHandler.js +96 -1
- package/lib/esm/persistence/Sequence.d.ts +89 -0
- package/lib/esm/persistence/Sequence.js +71 -2
- package/lib/esm/persistence/constants.d.ts +22 -0
- package/lib/esm/persistence/constants.js +23 -1
- package/lib/esm/persistence/decorators.d.ts +10 -0
- package/lib/esm/persistence/decorators.js +13 -3
- package/lib/esm/persistence/errors.d.ts +23 -0
- package/lib/esm/persistence/errors.js +24 -1
- package/lib/esm/persistence/index.js +9 -9
- package/lib/esm/persistence/types.d.ts +18 -0
- package/lib/esm/persistence/types.js +1 -1
- package/lib/esm/query/Condition.d.ts +78 -31
- package/lib/esm/query/Condition.js +134 -55
- package/lib/esm/query/Paginator.d.ts +56 -0
- package/lib/esm/query/Paginator.js +58 -2
- package/lib/esm/query/Statement.d.ts +51 -0
- package/lib/esm/query/Statement.js +55 -4
- package/lib/esm/query/constants.d.ts +25 -0
- package/lib/esm/query/constants.js +26 -1
- package/lib/esm/query/errors.d.ts +14 -0
- package/lib/esm/query/errors.js +15 -1
- package/lib/esm/query/index.js +8 -8
- package/lib/esm/query/options.d.ts +21 -3
- package/lib/esm/query/options.js +1 -1
- package/lib/esm/query/selectors.d.ts +26 -0
- package/lib/esm/query/selectors.js +1 -1
- package/lib/esm/ram/RamAdapter.d.ts +311 -0
- package/lib/esm/ram/RamAdapter.js +319 -8
- package/lib/esm/ram/RamContext.d.ts +16 -1
- package/lib/esm/ram/RamContext.js +18 -3
- package/lib/esm/ram/RamPaginator.d.ts +43 -0
- package/lib/esm/ram/RamPaginator.js +55 -3
- package/lib/esm/ram/RamSequence.d.ts +61 -0
- package/lib/esm/ram/RamSequence.js +66 -5
- package/lib/esm/ram/RamStatement.d.ts +74 -0
- package/lib/esm/ram/RamStatement.js +78 -4
- package/lib/esm/ram/constants.d.ts +8 -0
- package/lib/esm/ram/constants.js +9 -1
- package/lib/esm/ram/handlers.d.ts +19 -0
- package/lib/esm/ram/handlers.js +21 -2
- package/lib/esm/ram/index.js +11 -11
- package/lib/esm/ram/model/RamSequence.d.ts +25 -0
- package/lib/esm/ram/model/RamSequence.js +21 -3
- package/lib/esm/ram/model/index.js +2 -2
- package/lib/esm/ram/types.d.ts +42 -0
- package/lib/esm/ram/types.js +1 -1
- package/lib/esm/repository/Repository.d.ts +363 -8
- package/lib/esm/repository/Repository.js +369 -24
- package/lib/esm/repository/constants.d.ts +25 -0
- package/lib/esm/repository/constants.js +26 -1
- package/lib/esm/repository/decorators.d.ts +27 -0
- package/lib/esm/repository/decorators.js +29 -2
- package/lib/esm/repository/errors.d.ts +12 -5
- package/lib/esm/repository/errors.js +13 -6
- package/lib/esm/repository/index.js +8 -8
- package/lib/esm/repository/injectables.d.ts +18 -0
- package/lib/esm/repository/injectables.js +23 -5
- package/lib/esm/repository/types.d.ts +15 -0
- package/lib/esm/repository/types.js +1 -1
- package/lib/esm/repository/utils.d.ts +11 -0
- package/lib/esm/repository/utils.js +15 -4
- package/lib/esm/utils/decorators.d.ts +8 -0
- package/lib/esm/utils/decorators.js +9 -1
- package/lib/esm/utils/errors.d.ts +46 -0
- package/lib/esm/utils/errors.js +47 -1
- package/lib/esm/utils/index.js +3 -3
- package/lib/identity/decorators.cjs +53 -8
- package/lib/identity/decorators.d.ts +52 -7
- package/lib/identity/utils.cjs +20 -1
- package/lib/identity/utils.d.ts +19 -0
- package/lib/index.cjs +11 -4
- package/lib/index.d.ts +10 -3
- package/lib/interfaces/ErrorParser.cjs +1 -1
- package/lib/interfaces/ErrorParser.d.ts +12 -0
- package/lib/interfaces/Executor.cjs +1 -1
- package/lib/interfaces/Executor.d.ts +13 -0
- package/lib/interfaces/Observable.cjs +1 -1
- package/lib/interfaces/Observable.d.ts +27 -0
- package/lib/interfaces/Observer.cjs +1 -1
- package/lib/interfaces/Observer.d.ts +12 -0
- package/lib/interfaces/Paginatable.cjs +1 -1
- package/lib/interfaces/Paginatable.d.ts +15 -0
- package/lib/interfaces/Queriable.cjs +1 -1
- package/lib/interfaces/Queriable.d.ts +34 -9
- package/lib/interfaces/RawExecutor.cjs +1 -1
- package/lib/interfaces/RawExecutor.d.ts +14 -0
- package/lib/interfaces/SequenceOptions.cjs +19 -1
- package/lib/interfaces/SequenceOptions.d.ts +52 -0
- package/lib/model/BaseModel.cjs +24 -1
- package/lib/model/BaseModel.d.ts +31 -0
- package/lib/model/construction.cjs +441 -2
- package/lib/model/construction.d.ts +433 -0
- package/lib/model/decorators.cjs +160 -30
- package/lib/model/decorators.d.ts +159 -29
- package/lib/model/types.cjs +1 -1
- package/lib/model/types.d.ts +9 -0
- package/lib/persistence/Adapter.cjs +287 -19
- package/lib/persistence/Adapter.d.ts +358 -17
- package/lib/persistence/Dispatch.cjs +102 -4
- package/lib/persistence/Dispatch.d.ts +114 -1
- package/lib/persistence/ObserverHandler.cjs +96 -1
- package/lib/persistence/ObserverHandler.d.ts +95 -0
- package/lib/persistence/Sequence.cjs +70 -1
- package/lib/persistence/Sequence.d.ts +89 -0
- package/lib/persistence/constants.cjs +23 -1
- package/lib/persistence/constants.d.ts +22 -0
- package/lib/persistence/decorators.cjs +11 -1
- package/lib/persistence/decorators.d.ts +10 -0
- package/lib/persistence/errors.cjs +24 -1
- package/lib/persistence/errors.d.ts +23 -0
- package/lib/persistence/types.cjs +1 -1
- package/lib/persistence/types.d.ts +18 -0
- package/lib/query/Condition.cjs +132 -53
- package/lib/query/Condition.d.ts +78 -31
- package/lib/query/Paginator.cjs +57 -1
- package/lib/query/Paginator.d.ts +56 -0
- package/lib/query/Statement.cjs +52 -1
- package/lib/query/Statement.d.ts +51 -0
- package/lib/query/constants.cjs +26 -1
- package/lib/query/constants.d.ts +25 -0
- package/lib/query/errors.cjs +15 -1
- package/lib/query/errors.d.ts +14 -0
- package/lib/query/options.cjs +1 -1
- package/lib/query/options.d.ts +21 -3
- package/lib/query/selectors.cjs +1 -1
- package/lib/query/selectors.d.ts +26 -0
- package/lib/ram/RamAdapter.cjs +312 -1
- package/lib/ram/RamAdapter.d.ts +311 -0
- package/lib/ram/RamContext.cjs +18 -3
- package/lib/ram/RamContext.d.ts +16 -1
- package/lib/ram/RamPaginator.cjs +54 -2
- package/lib/ram/RamPaginator.d.ts +43 -0
- package/lib/ram/RamSequence.cjs +63 -2
- package/lib/ram/RamSequence.d.ts +61 -0
- package/lib/ram/RamStatement.cjs +75 -1
- package/lib/ram/RamStatement.d.ts +74 -0
- package/lib/ram/constants.cjs +9 -1
- package/lib/ram/constants.d.ts +8 -0
- package/lib/ram/handlers.cjs +20 -1
- package/lib/ram/handlers.d.ts +19 -0
- package/lib/ram/model/RamSequence.cjs +19 -1
- package/lib/ram/model/RamSequence.d.ts +25 -0
- package/lib/ram/types.cjs +1 -1
- package/lib/ram/types.d.ts +42 -0
- package/lib/repository/Repository.cjs +360 -15
- package/lib/repository/Repository.d.ts +363 -8
- package/lib/repository/constants.cjs +26 -1
- package/lib/repository/constants.d.ts +25 -0
- package/lib/repository/decorators.cjs +28 -1
- package/lib/repository/decorators.d.ts +27 -0
- package/lib/repository/errors.cjs +13 -6
- package/lib/repository/errors.d.ts +12 -5
- package/lib/repository/injectables.cjs +19 -1
- package/lib/repository/injectables.d.ts +18 -0
- package/lib/repository/types.cjs +1 -1
- package/lib/repository/types.d.ts +15 -0
- package/lib/repository/utils.cjs +12 -1
- package/lib/repository/utils.d.ts +11 -0
- package/lib/utils/decorators.cjs +9 -1
- package/lib/utils/decorators.d.ts +8 -0
- package/lib/utils/errors.cjs +47 -1
- package/lib/utils/errors.d.ts +46 -0
- 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
|
-
* @
|
17
|
-
* @
|
18
|
-
*
|
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
|
-
*
|
21
|
-
*
|
37
|
+
* async initialize() {
|
38
|
+
* // Set up the adapter
|
39
|
+
* await this.native.connect();
|
40
|
+
* }
|
22
41
|
*
|
23
|
-
*
|
24
|
-
*
|
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
|
-
*
|
27
|
-
*
|
28
|
-
*
|
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
|
-
|
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
|
-
* @
|
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
|
-
* @
|
76
|
-
* @
|
77
|
-
*
|
78
|
-
* @
|
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
|
-
|
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":"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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\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"]}
|