@decaf-ts/core 0.5.0 → 0.5.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE.md +21 -157
- package/README.md +652 -15
- package/dist/core.cjs +2638 -1085
- package/dist/core.esm.cjs +2636 -1070
- package/lib/esm/identity/decorators.d.ts +52 -14
- package/lib/esm/identity/decorators.js +54 -16
- package/lib/esm/identity/utils.d.ts +20 -1
- package/lib/esm/identity/utils.js +22 -2
- package/lib/esm/index.d.ts +11 -15
- package/lib/esm/index.js +17 -19
- package/lib/esm/interfaces/ErrorParser.d.ts +12 -0
- package/lib/esm/interfaces/ErrorParser.js +1 -1
- package/lib/esm/interfaces/Executor.d.ts +10 -13
- package/lib/esm/interfaces/Executor.js +1 -1
- package/lib/esm/interfaces/Observable.d.ts +20 -18
- package/lib/esm/interfaces/Observable.js +1 -1
- package/lib/esm/interfaces/Observer.d.ts +7 -8
- package/lib/esm/interfaces/Observer.js +1 -1
- package/lib/esm/interfaces/Paginatable.d.ts +18 -2
- package/lib/esm/interfaces/Paginatable.js +1 -1
- package/lib/esm/interfaces/Queriable.d.ts +44 -3
- package/lib/esm/interfaces/Queriable.js +1 -1
- package/lib/esm/interfaces/RawExecutor.d.ts +10 -13
- package/lib/esm/interfaces/RawExecutor.js +1 -1
- package/lib/esm/interfaces/SequenceOptions.d.ts +50 -5
- package/lib/esm/interfaces/SequenceOptions.js +19 -1
- package/lib/esm/interfaces/index.d.ts +0 -1
- package/lib/esm/interfaces/index.js +1 -2
- package/lib/esm/model/BaseModel.d.ts +31 -0
- package/lib/esm/model/BaseModel.js +24 -1
- package/lib/esm/model/construction.d.ts +442 -9
- package/lib/esm/model/construction.js +441 -2
- package/lib/esm/model/decorators.d.ts +166 -42
- package/lib/esm/model/decorators.js +161 -37
- package/lib/esm/model/index.js +1 -2
- package/lib/esm/model/types.d.ts +9 -0
- package/lib/esm/model/types.js +1 -1
- package/lib/esm/persistence/Adapter.d.ts +384 -40
- package/lib/esm/persistence/Adapter.js +415 -59
- package/lib/esm/persistence/Dispatch.d.ts +131 -0
- package/lib/esm/persistence/Dispatch.js +187 -0
- package/lib/esm/persistence/ObserverHandler.d.ts +109 -0
- package/lib/esm/persistence/ObserverHandler.js +137 -0
- package/lib/esm/persistence/Sequence.d.ts +89 -8
- package/lib/esm/persistence/Sequence.js +91 -1
- package/lib/esm/persistence/constants.d.ts +22 -5
- package/lib/esm/persistence/constants.js +23 -7
- package/lib/esm/persistence/decorators.d.ts +10 -0
- package/lib/esm/persistence/decorators.js +11 -1
- package/lib/esm/persistence/errors.d.ts +23 -3
- package/lib/esm/persistence/errors.js +25 -7
- package/lib/esm/persistence/index.d.ts +3 -0
- package/lib/esm/persistence/index.js +4 -1
- package/lib/esm/persistence/types.d.ts +21 -0
- package/lib/esm/persistence/types.js +2 -0
- package/lib/esm/query/Condition.d.ts +88 -44
- package/lib/esm/query/Condition.js +144 -62
- package/lib/esm/query/Paginator.d.ts +67 -10
- package/lib/esm/query/Paginator.js +64 -10
- package/lib/esm/query/Statement.d.ts +82 -47
- package/lib/esm/query/Statement.js +175 -122
- package/lib/esm/query/constants.d.ts +25 -64
- package/lib/esm/query/constants.js +26 -68
- package/lib/esm/query/errors.d.ts +14 -0
- package/lib/esm/query/errors.js +15 -1
- package/lib/esm/query/index.d.ts +0 -5
- package/lib/esm/query/index.js +1 -6
- package/lib/esm/query/options.d.ts +69 -178
- package/lib/esm/query/options.js +1 -1
- package/lib/esm/query/selectors.d.ts +20 -24
- package/lib/esm/query/selectors.js +1 -1
- package/lib/esm/ram/RamAdapter.d.ts +322 -20
- package/lib/esm/ram/RamAdapter.js +360 -140
- package/lib/esm/ram/RamContext.d.ts +16 -1
- package/lib/esm/ram/RamContext.js +18 -3
- package/lib/esm/ram/RamPaginator.d.ts +51 -6
- package/lib/esm/ram/RamPaginator.js +58 -6
- package/lib/esm/ram/RamSequence.d.ts +49 -24
- package/lib/esm/ram/RamSequence.js +52 -40
- package/lib/esm/ram/RamStatement.d.ts +84 -6
- package/lib/esm/ram/RamStatement.js +175 -6
- package/lib/esm/ram/constants.d.ts +9 -0
- package/lib/esm/ram/constants.js +10 -0
- package/lib/esm/ram/handlers.d.ts +25 -0
- package/lib/esm/ram/handlers.js +27 -0
- package/lib/esm/ram/index.d.ts +4 -4
- package/lib/esm/ram/index.js +9 -5
- package/lib/esm/ram/model/RamSequence.d.ts +21 -9
- package/lib/esm/ram/model/RamSequence.js +19 -1
- package/lib/esm/ram/types.d.ts +47 -5
- package/lib/esm/ram/types.js +1 -1
- package/lib/esm/repository/Repository.d.ts +381 -22
- package/lib/esm/repository/Repository.js +446 -43
- package/lib/esm/repository/constants.d.ts +23 -13
- package/lib/esm/repository/constants.js +24 -14
- package/lib/esm/repository/decorators.d.ts +27 -0
- package/lib/esm/repository/decorators.js +28 -1
- package/lib/esm/repository/errors.d.ts +12 -11
- package/lib/esm/repository/errors.js +13 -16
- package/lib/esm/repository/injectables.d.ts +18 -0
- package/lib/esm/repository/injectables.js +19 -1
- package/lib/esm/repository/types.d.ts +13 -1
- package/lib/esm/repository/types.js +1 -1
- package/lib/esm/repository/utils.d.ts +11 -0
- package/lib/esm/repository/utils.js +12 -1
- package/lib/esm/utils/decorators.d.ts +9 -0
- package/lib/esm/utils/decorators.js +19 -0
- package/lib/esm/utils/errors.d.ts +56 -0
- package/lib/esm/utils/errors.js +63 -0
- package/lib/esm/utils/index.d.ts +2 -0
- package/lib/esm/utils/index.js +3 -0
- package/lib/identity/decorators.cjs +54 -16
- package/lib/identity/decorators.d.ts +52 -14
- package/lib/identity/utils.cjs +22 -2
- package/lib/identity/utils.d.ts +20 -1
- package/lib/index.cjs +17 -19
- package/lib/index.d.ts +11 -15
- 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 +10 -13
- package/lib/interfaces/Observable.cjs +1 -1
- package/lib/interfaces/Observable.d.ts +20 -18
- package/lib/interfaces/Observer.cjs +1 -1
- package/lib/interfaces/Observer.d.ts +7 -8
- package/lib/interfaces/Paginatable.cjs +1 -1
- package/lib/interfaces/Paginatable.d.ts +18 -2
- package/lib/interfaces/Queriable.cjs +1 -1
- package/lib/interfaces/Queriable.d.ts +44 -3
- package/lib/interfaces/RawExecutor.cjs +1 -1
- package/lib/interfaces/RawExecutor.d.ts +10 -13
- package/lib/interfaces/SequenceOptions.cjs +19 -1
- package/lib/interfaces/SequenceOptions.d.ts +50 -5
- package/lib/interfaces/index.cjs +1 -2
- package/lib/interfaces/index.d.ts +0 -1
- 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 +442 -9
- package/lib/model/decorators.cjs +161 -37
- package/lib/model/decorators.d.ts +166 -42
- package/lib/model/index.cjs +1 -2
- package/lib/model/types.cjs +1 -1
- package/lib/model/types.d.ts +9 -0
- package/lib/persistence/Adapter.cjs +414 -58
- package/lib/persistence/Adapter.d.ts +384 -40
- package/lib/persistence/Dispatch.cjs +191 -0
- package/lib/persistence/Dispatch.d.ts +131 -0
- package/lib/persistence/ObserverHandler.cjs +141 -0
- package/lib/persistence/ObserverHandler.d.ts +109 -0
- package/lib/persistence/Sequence.cjs +91 -1
- package/lib/persistence/Sequence.d.ts +89 -8
- package/lib/persistence/constants.cjs +24 -8
- package/lib/persistence/constants.d.ts +22 -5
- package/lib/persistence/decorators.cjs +11 -1
- package/lib/persistence/decorators.d.ts +10 -0
- package/lib/persistence/errors.cjs +26 -9
- package/lib/persistence/errors.d.ts +23 -3
- package/lib/persistence/index.cjs +4 -1
- package/lib/persistence/index.d.ts +3 -0
- package/lib/persistence/types.cjs +3 -0
- package/lib/persistence/types.d.ts +21 -0
- package/lib/query/Condition.cjs +143 -61
- package/lib/query/Condition.d.ts +88 -44
- package/lib/query/Paginator.cjs +64 -10
- package/lib/query/Paginator.d.ts +67 -10
- package/lib/query/Statement.cjs +174 -121
- package/lib/query/Statement.d.ts +82 -47
- package/lib/query/constants.cjs +27 -69
- package/lib/query/constants.d.ts +25 -64
- package/lib/query/errors.cjs +15 -1
- package/lib/query/errors.d.ts +14 -0
- package/lib/query/index.cjs +1 -6
- package/lib/query/index.d.ts +0 -5
- package/lib/query/options.cjs +1 -1
- package/lib/query/options.d.ts +69 -178
- package/lib/query/selectors.cjs +1 -1
- package/lib/query/selectors.d.ts +20 -24
- package/lib/ram/RamAdapter.cjs +358 -172
- package/lib/ram/RamAdapter.d.ts +322 -20
- package/lib/ram/RamContext.cjs +18 -3
- package/lib/ram/RamContext.d.ts +16 -1
- package/lib/ram/RamPaginator.cjs +58 -6
- package/lib/ram/RamPaginator.d.ts +51 -6
- package/lib/ram/RamSequence.cjs +52 -41
- package/lib/ram/RamSequence.d.ts +49 -24
- package/lib/ram/RamStatement.cjs +175 -6
- package/lib/ram/RamStatement.d.ts +84 -6
- package/lib/ram/constants.cjs +13 -0
- package/lib/ram/constants.d.ts +9 -0
- package/lib/ram/handlers.cjs +30 -0
- package/lib/ram/handlers.d.ts +25 -0
- package/lib/ram/index.cjs +9 -5
- package/lib/ram/index.d.ts +4 -4
- package/lib/ram/model/RamSequence.cjs +19 -1
- package/lib/ram/model/RamSequence.d.ts +21 -9
- package/lib/ram/types.cjs +1 -1
- package/lib/ram/types.d.ts +47 -5
- package/lib/repository/Repository.cjs +445 -42
- package/lib/repository/Repository.d.ts +381 -22
- package/lib/repository/constants.cjs +24 -14
- package/lib/repository/constants.d.ts +23 -13
- package/lib/repository/decorators.cjs +28 -1
- package/lib/repository/decorators.d.ts +27 -0
- package/lib/repository/errors.cjs +14 -19
- package/lib/repository/errors.d.ts +12 -11
- 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 +13 -1
- package/lib/repository/utils.cjs +12 -1
- package/lib/repository/utils.d.ts +11 -0
- package/lib/utils/decorators.cjs +22 -0
- package/lib/utils/decorators.d.ts +9 -0
- package/lib/utils/errors.cjs +69 -0
- package/lib/utils/errors.d.ts +56 -0
- package/lib/{validators → utils}/index.cjs +2 -2
- package/lib/utils/index.d.ts +2 -0
- package/package.json +5 -5
- package/lib/esm/interfaces/Builder.d.ts +0 -16
- package/lib/esm/interfaces/Builder.js +0 -2
- package/lib/esm/model/IdentifiedBaseModel.d.ts +0 -7
- package/lib/esm/model/IdentifiedBaseModel.js +0 -25
- package/lib/esm/query/Clause.d.ts +0 -50
- package/lib/esm/query/Clause.js +0 -82
- package/lib/esm/query/ClauseFactory.d.ts +0 -71
- package/lib/esm/query/ClauseFactory.js +0 -6
- package/lib/esm/query/Query.d.ts +0 -43
- package/lib/esm/query/Query.js +0 -54
- package/lib/esm/query/clauses/FromClause.d.ts +0 -45
- package/lib/esm/query/clauses/FromClause.js +0 -59
- package/lib/esm/query/clauses/GroupByClause.d.ts +0 -21
- package/lib/esm/query/clauses/GroupByClause.js +0 -19
- package/lib/esm/query/clauses/InsertClause.d.ts +0 -37
- package/lib/esm/query/clauses/InsertClause.js +0 -55
- package/lib/esm/query/clauses/LimitClause.d.ts +0 -29
- package/lib/esm/query/clauses/LimitClause.js +0 -27
- package/lib/esm/query/clauses/OffsetClause.d.ts +0 -21
- package/lib/esm/query/clauses/OffsetClause.js +0 -19
- package/lib/esm/query/clauses/OrderByClause.d.ts +0 -37
- package/lib/esm/query/clauses/OrderByClause.js +0 -39
- package/lib/esm/query/clauses/SelectClause.d.ts +0 -47
- package/lib/esm/query/clauses/SelectClause.js +0 -62
- package/lib/esm/query/clauses/SelectorBasedClause.d.ts +0 -25
- package/lib/esm/query/clauses/SelectorBasedClause.js +0 -44
- package/lib/esm/query/clauses/ValuesClause.d.ts +0 -21
- package/lib/esm/query/clauses/ValuesClause.js +0 -36
- package/lib/esm/query/clauses/WhereClause.d.ts +0 -46
- package/lib/esm/query/clauses/WhereClause.js +0 -71
- package/lib/esm/query/clauses/index.d.ts +0 -10
- package/lib/esm/query/clauses/index.js +0 -11
- package/lib/esm/query/types.d.ts +0 -2
- package/lib/esm/query/types.js +0 -2
- package/lib/esm/ram/RamClauseFactory.d.ts +0 -17
- package/lib/esm/ram/RamClauseFactory.js +0 -92
- package/lib/esm/ram/clauses/FromClause.d.ts +0 -7
- package/lib/esm/ram/clauses/FromClause.js +0 -11
- package/lib/esm/ram/clauses/InsertClause.d.ts +0 -7
- package/lib/esm/ram/clauses/InsertClause.js +0 -13
- package/lib/esm/ram/clauses/OrderByClause.d.ts +0 -7
- package/lib/esm/ram/clauses/OrderByClause.js +0 -39
- package/lib/esm/ram/clauses/SelectClause.d.ts +0 -7
- package/lib/esm/ram/clauses/SelectClause.js +0 -16
- package/lib/esm/ram/clauses/ValuesClause.d.ts +0 -7
- package/lib/esm/ram/clauses/ValuesClause.js +0 -12
- package/lib/esm/ram/clauses/WhereClause.d.ts +0 -7
- package/lib/esm/ram/clauses/WhereClause.js +0 -11
- package/lib/esm/ram/clauses/index.d.ts +0 -6
- package/lib/esm/ram/clauses/index.js +0 -7
- package/lib/esm/validators/ClauseSequenceValidator.d.ts +0 -28
- package/lib/esm/validators/ClauseSequenceValidator.js +0 -95
- package/lib/esm/validators/decorators.d.ts +0 -10
- package/lib/esm/validators/decorators.js +0 -24
- package/lib/esm/validators/index.d.ts +0 -2
- package/lib/esm/validators/index.js +0 -3
- package/lib/interfaces/Builder.cjs +0 -3
- package/lib/interfaces/Builder.d.ts +0 -16
- package/lib/model/IdentifiedBaseModel.cjs +0 -29
- package/lib/model/IdentifiedBaseModel.d.ts +0 -7
- package/lib/query/Clause.cjs +0 -86
- package/lib/query/Clause.d.ts +0 -50
- package/lib/query/ClauseFactory.cjs +0 -10
- package/lib/query/ClauseFactory.d.ts +0 -71
- package/lib/query/Query.cjs +0 -58
- package/lib/query/Query.d.ts +0 -43
- package/lib/query/clauses/FromClause.cjs +0 -63
- package/lib/query/clauses/FromClause.d.ts +0 -45
- package/lib/query/clauses/GroupByClause.cjs +0 -23
- package/lib/query/clauses/GroupByClause.d.ts +0 -21
- package/lib/query/clauses/InsertClause.cjs +0 -59
- package/lib/query/clauses/InsertClause.d.ts +0 -37
- package/lib/query/clauses/LimitClause.cjs +0 -31
- package/lib/query/clauses/LimitClause.d.ts +0 -29
- package/lib/query/clauses/OffsetClause.cjs +0 -23
- package/lib/query/clauses/OffsetClause.d.ts +0 -21
- package/lib/query/clauses/OrderByClause.cjs +0 -43
- package/lib/query/clauses/OrderByClause.d.ts +0 -37
- package/lib/query/clauses/SelectClause.cjs +0 -66
- package/lib/query/clauses/SelectClause.d.ts +0 -47
- package/lib/query/clauses/SelectorBasedClause.cjs +0 -48
- package/lib/query/clauses/SelectorBasedClause.d.ts +0 -25
- package/lib/query/clauses/ValuesClause.cjs +0 -40
- package/lib/query/clauses/ValuesClause.d.ts +0 -21
- package/lib/query/clauses/WhereClause.cjs +0 -75
- package/lib/query/clauses/WhereClause.d.ts +0 -46
- package/lib/query/clauses/index.cjs +0 -27
- package/lib/query/clauses/index.d.ts +0 -10
- package/lib/query/types.cjs +0 -3
- package/lib/query/types.d.ts +0 -2
- package/lib/ram/RamClauseFactory.cjs +0 -96
- package/lib/ram/RamClauseFactory.d.ts +0 -17
- package/lib/ram/clauses/FromClause.cjs +0 -15
- package/lib/ram/clauses/FromClause.d.ts +0 -7
- package/lib/ram/clauses/InsertClause.cjs +0 -17
- package/lib/ram/clauses/InsertClause.d.ts +0 -7
- package/lib/ram/clauses/OrderByClause.cjs +0 -43
- package/lib/ram/clauses/OrderByClause.d.ts +0 -7
- package/lib/ram/clauses/SelectClause.cjs +0 -20
- package/lib/ram/clauses/SelectClause.d.ts +0 -7
- package/lib/ram/clauses/ValuesClause.cjs +0 -16
- package/lib/ram/clauses/ValuesClause.d.ts +0 -7
- package/lib/ram/clauses/WhereClause.cjs +0 -15
- package/lib/ram/clauses/WhereClause.d.ts +0 -7
- package/lib/ram/clauses/index.cjs +0 -23
- package/lib/ram/clauses/index.d.ts +0 -6
- package/lib/validators/ClauseSequenceValidator.cjs +0 -98
- package/lib/validators/ClauseSequenceValidator.d.ts +0 -28
- package/lib/validators/decorators.cjs +0 -27
- package/lib/validators/decorators.d.ts +0 -10
- package/lib/validators/index.d.ts +0 -2
@@ -0,0 +1,131 @@
|
|
1
|
+
import { OperationKeys, BulkCrudOperationKeys } from "@decaf-ts/db-decorators";
|
2
|
+
import { ModelConstructor } from "@decaf-ts/decorator-validation";
|
3
|
+
import { Observable, Observer } from "../interfaces";
|
4
|
+
import { Adapter } from "./Adapter";
|
5
|
+
import { Logger } from "@decaf-ts/logging";
|
6
|
+
import { EventIds } from "./types";
|
7
|
+
/**
|
8
|
+
* @description Dispatches database operation events to observers
|
9
|
+
* @summary The Dispatch class implements the Observable interface and is responsible for intercepting
|
10
|
+
* database operations from an Adapter and notifying observers when changes occur. It uses proxies to
|
11
|
+
* wrap the adapter's CRUD methods and automatically trigger observer updates after operations complete.
|
12
|
+
* @template Y - The native database driver type
|
13
|
+
* @param {void} - No constructor parameters
|
14
|
+
* @class Dispatch
|
15
|
+
* @example
|
16
|
+
* ```typescript
|
17
|
+
* // Creating and using a Dispatch instance
|
18
|
+
* const dispatch = new Dispatch<PostgresDriver>();
|
19
|
+
*
|
20
|
+
* // Connect it to an adapter
|
21
|
+
* const adapter = new PostgresAdapter(connection);
|
22
|
+
* dispatch.observe(adapter);
|
23
|
+
*
|
24
|
+
* // Now any CRUD operations on the adapter will automatically
|
25
|
+
* // trigger observer notifications
|
26
|
+
* await adapter.create('users', 123, userModel);
|
27
|
+
* // Observers will be notified about the creation
|
28
|
+
*
|
29
|
+
* // When done, you can disconnect
|
30
|
+
* dispatch.unObserve(adapter);
|
31
|
+
* ```
|
32
|
+
*/
|
33
|
+
export declare class Dispatch<Y> implements Observable {
|
34
|
+
/**
|
35
|
+
* @description The adapter being observed
|
36
|
+
* @summary Reference to the database adapter whose operations are being monitored
|
37
|
+
*/
|
38
|
+
protected adapter?: Adapter<Y, any, any, any>;
|
39
|
+
/**
|
40
|
+
* @description The native database driver
|
41
|
+
* @summary Reference to the underlying database driver from the adapter
|
42
|
+
*/
|
43
|
+
protected native?: Y;
|
44
|
+
/**
|
45
|
+
* @description List of model constructors
|
46
|
+
* @summary Array of model constructors that are registered with the adapter
|
47
|
+
*/
|
48
|
+
protected models: ModelConstructor<any>[];
|
49
|
+
/**
|
50
|
+
* @description Logger instance
|
51
|
+
* @summary Logger for recording dispatch activities
|
52
|
+
*/
|
53
|
+
private logger;
|
54
|
+
/**
|
55
|
+
* @description Accessor for the logger
|
56
|
+
* @summary Gets or initializes the logger for this dispatch instance
|
57
|
+
* @return {Logger} The logger instance
|
58
|
+
*/
|
59
|
+
protected get log(): Logger;
|
60
|
+
/**
|
61
|
+
* @description Creates a new Dispatch instance
|
62
|
+
* @summary Initializes a new Dispatch instance without any adapter
|
63
|
+
*/
|
64
|
+
constructor();
|
65
|
+
/**
|
66
|
+
* @description Initializes the dispatch by proxying adapter methods
|
67
|
+
* @summary Sets up proxies on the adapter's CRUD methods to intercept operations and notify observers.
|
68
|
+
* This method is called automatically when an adapter is observed.
|
69
|
+
* @return {Promise<void>} A promise that resolves when initialization is complete
|
70
|
+
* @mermaid
|
71
|
+
* sequenceDiagram
|
72
|
+
* participant Dispatch
|
73
|
+
* participant Adapter
|
74
|
+
* participant Proxy
|
75
|
+
*
|
76
|
+
* Dispatch->>Dispatch: initialize()
|
77
|
+
* Dispatch->>Dispatch: Check if adapter exists
|
78
|
+
* alt No adapter
|
79
|
+
* Dispatch-->>Dispatch: Throw InternalError
|
80
|
+
* end
|
81
|
+
*
|
82
|
+
* loop For each CRUD method
|
83
|
+
* Dispatch->>Adapter: Check if method exists
|
84
|
+
* alt Method doesn't exist
|
85
|
+
* Dispatch-->>Dispatch: Throw InternalError
|
86
|
+
* end
|
87
|
+
*
|
88
|
+
* Dispatch->>Adapter: Get property descriptor
|
89
|
+
* loop While descriptor not found
|
90
|
+
* Dispatch->>Adapter: Check prototype chain
|
91
|
+
* end
|
92
|
+
*
|
93
|
+
* alt Descriptor not found or not writable
|
94
|
+
* Dispatch->>Dispatch: Log error and continue
|
95
|
+
* else Descriptor found and writable
|
96
|
+
* Dispatch->>Proxy: Create proxy for method
|
97
|
+
* Dispatch->>Adapter: Replace method with proxy
|
98
|
+
* end
|
99
|
+
* end
|
100
|
+
*/
|
101
|
+
protected initialize(): Promise<void>;
|
102
|
+
/**
|
103
|
+
* @description Closes the dispatch
|
104
|
+
* @summary Performs any necessary cleanup when the dispatch is no longer needed
|
105
|
+
* @return {Promise<void>} A promise that resolves when closing is complete
|
106
|
+
*/
|
107
|
+
close(): Promise<void>;
|
108
|
+
/**
|
109
|
+
* @description Starts observing an adapter
|
110
|
+
* @summary Connects this dispatch to an adapter to monitor its operations
|
111
|
+
* @param {Adapter<Y, any, any, any>} observer - The adapter to observe
|
112
|
+
* @return {void}
|
113
|
+
*/
|
114
|
+
observe(observer: Adapter<Y, any, any, any>): void;
|
115
|
+
/**
|
116
|
+
* @description Stops observing an adapter
|
117
|
+
* @summary Disconnects this dispatch from an adapter
|
118
|
+
* @param {Observer} observer - The adapter to stop observing
|
119
|
+
* @return {void}
|
120
|
+
*/
|
121
|
+
unObserve(observer: Observer): void;
|
122
|
+
/**
|
123
|
+
* @description Updates observers about a database event
|
124
|
+
* @summary Notifies observers about a change in the database
|
125
|
+
* @param {string} table - The name of the table where the change occurred
|
126
|
+
* @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
|
127
|
+
* @param {EventIds} id - The identifier(s) of the affected record(s)
|
128
|
+
* @return {Promise<void>} A promise that resolves when all observers have been notified
|
129
|
+
*/
|
130
|
+
updateObservers(table: string, event: OperationKeys | BulkCrudOperationKeys | string, id: EventIds): Promise<void>;
|
131
|
+
}
|
@@ -0,0 +1,187 @@
|
|
1
|
+
import { InternalError, OperationKeys, BulkCrudOperationKeys, } from "@decaf-ts/db-decorators";
|
2
|
+
import { Adapter } from "./Adapter";
|
3
|
+
import { UnsupportedError } from "./errors";
|
4
|
+
import { Logging } from "@decaf-ts/logging";
|
5
|
+
/**
|
6
|
+
* @description Dispatches database operation events to observers
|
7
|
+
* @summary The Dispatch class implements the Observable interface and is responsible for intercepting
|
8
|
+
* database operations from an Adapter and notifying observers when changes occur. It uses proxies to
|
9
|
+
* wrap the adapter's CRUD methods and automatically trigger observer updates after operations complete.
|
10
|
+
* @template Y - The native database driver type
|
11
|
+
* @param {void} - No constructor parameters
|
12
|
+
* @class Dispatch
|
13
|
+
* @example
|
14
|
+
* ```typescript
|
15
|
+
* // Creating and using a Dispatch instance
|
16
|
+
* const dispatch = new Dispatch<PostgresDriver>();
|
17
|
+
*
|
18
|
+
* // Connect it to an adapter
|
19
|
+
* const adapter = new PostgresAdapter(connection);
|
20
|
+
* dispatch.observe(adapter);
|
21
|
+
*
|
22
|
+
* // Now any CRUD operations on the adapter will automatically
|
23
|
+
* // trigger observer notifications
|
24
|
+
* await adapter.create('users', 123, userModel);
|
25
|
+
* // Observers will be notified about the creation
|
26
|
+
*
|
27
|
+
* // When done, you can disconnect
|
28
|
+
* dispatch.unObserve(adapter);
|
29
|
+
* ```
|
30
|
+
*/
|
31
|
+
export class Dispatch {
|
32
|
+
/**
|
33
|
+
* @description Accessor for the logger
|
34
|
+
* @summary Gets or initializes the logger for this dispatch instance
|
35
|
+
* @return {Logger} The logger instance
|
36
|
+
*/
|
37
|
+
get log() {
|
38
|
+
if (!this.logger)
|
39
|
+
this.logger = Logging.for(this).for(this.adapter);
|
40
|
+
return this.logger;
|
41
|
+
}
|
42
|
+
/**
|
43
|
+
* @description Creates a new Dispatch instance
|
44
|
+
* @summary Initializes a new Dispatch instance without any adapter
|
45
|
+
*/
|
46
|
+
constructor() { }
|
47
|
+
/**
|
48
|
+
* @description Initializes the dispatch by proxying adapter methods
|
49
|
+
* @summary Sets up proxies on the adapter's CRUD methods to intercept operations and notify observers.
|
50
|
+
* This method is called automatically when an adapter is observed.
|
51
|
+
* @return {Promise<void>} A promise that resolves when initialization is complete
|
52
|
+
* @mermaid
|
53
|
+
* sequenceDiagram
|
54
|
+
* participant Dispatch
|
55
|
+
* participant Adapter
|
56
|
+
* participant Proxy
|
57
|
+
*
|
58
|
+
* Dispatch->>Dispatch: initialize()
|
59
|
+
* Dispatch->>Dispatch: Check if adapter exists
|
60
|
+
* alt No adapter
|
61
|
+
* Dispatch-->>Dispatch: Throw InternalError
|
62
|
+
* end
|
63
|
+
*
|
64
|
+
* loop For each CRUD method
|
65
|
+
* Dispatch->>Adapter: Check if method exists
|
66
|
+
* alt Method doesn't exist
|
67
|
+
* Dispatch-->>Dispatch: Throw InternalError
|
68
|
+
* end
|
69
|
+
*
|
70
|
+
* Dispatch->>Adapter: Get property descriptor
|
71
|
+
* loop While descriptor not found
|
72
|
+
* Dispatch->>Adapter: Check prototype chain
|
73
|
+
* end
|
74
|
+
*
|
75
|
+
* alt Descriptor not found or not writable
|
76
|
+
* Dispatch->>Dispatch: Log error and continue
|
77
|
+
* else Descriptor found and writable
|
78
|
+
* Dispatch->>Proxy: Create proxy for method
|
79
|
+
* Dispatch->>Adapter: Replace method with proxy
|
80
|
+
* end
|
81
|
+
* end
|
82
|
+
*/
|
83
|
+
async initialize() {
|
84
|
+
if (!this.adapter)
|
85
|
+
throw new InternalError(`No adapter observed for dispatch`);
|
86
|
+
const adapter = this.adapter;
|
87
|
+
[
|
88
|
+
OperationKeys.CREATE,
|
89
|
+
OperationKeys.UPDATE,
|
90
|
+
OperationKeys.DELETE,
|
91
|
+
BulkCrudOperationKeys.CREATE_ALL,
|
92
|
+
BulkCrudOperationKeys.UPDATE_ALL,
|
93
|
+
BulkCrudOperationKeys.DELETE_ALL,
|
94
|
+
].forEach((method) => {
|
95
|
+
if (!adapter[method])
|
96
|
+
throw new InternalError(`Method ${method} not found in ${adapter.alias} adapter to bind Observables Dispatch`);
|
97
|
+
let descriptor = Object.getOwnPropertyDescriptor(adapter, method);
|
98
|
+
let proto = adapter;
|
99
|
+
while (!descriptor && proto !== Object.prototype) {
|
100
|
+
proto = Object.getPrototypeOf(proto);
|
101
|
+
descriptor = Object.getOwnPropertyDescriptor(proto, method);
|
102
|
+
}
|
103
|
+
if (!descriptor || !descriptor.writable) {
|
104
|
+
this.log.error(`Could not find method ${method} to bind Observables Dispatch`);
|
105
|
+
return;
|
106
|
+
}
|
107
|
+
function bulkToSingle(method) {
|
108
|
+
switch (method) {
|
109
|
+
case BulkCrudOperationKeys.CREATE_ALL:
|
110
|
+
return OperationKeys.CREATE;
|
111
|
+
case BulkCrudOperationKeys.UPDATE_ALL:
|
112
|
+
return OperationKeys.UPDATE;
|
113
|
+
case BulkCrudOperationKeys.DELETE_ALL:
|
114
|
+
return OperationKeys.DELETE;
|
115
|
+
default:
|
116
|
+
return method;
|
117
|
+
}
|
118
|
+
}
|
119
|
+
// @ts-expect-error because there are read only properties
|
120
|
+
adapter[method] = new Proxy(adapter[method], {
|
121
|
+
apply: async (target, thisArg, argArray) => {
|
122
|
+
const [tableName, ids] = argArray;
|
123
|
+
const result = await target.apply(thisArg, argArray);
|
124
|
+
this.updateObservers(tableName, bulkToSingle(method), ids)
|
125
|
+
.then(() => {
|
126
|
+
this.log.verbose(`Observer refresh dispatched by ${method} for ${tableName}`);
|
127
|
+
this.log.debug(`pks: ${ids}`);
|
128
|
+
})
|
129
|
+
.catch((e) => this.log.error(`Failed to dispatch observer refresh for ${method} on ${tableName}: ${e}`));
|
130
|
+
return result;
|
131
|
+
},
|
132
|
+
});
|
133
|
+
});
|
134
|
+
}
|
135
|
+
/**
|
136
|
+
* @description Closes the dispatch
|
137
|
+
* @summary Performs any necessary cleanup when the dispatch is no longer needed
|
138
|
+
* @return {Promise<void>} A promise that resolves when closing is complete
|
139
|
+
*/
|
140
|
+
async close() {
|
141
|
+
// to nothing in this instance but may be required for closing connections
|
142
|
+
}
|
143
|
+
/**
|
144
|
+
* @description Starts observing an adapter
|
145
|
+
* @summary Connects this dispatch to an adapter to monitor its operations
|
146
|
+
* @param {Adapter<Y, any, any, any>} observer - The adapter to observe
|
147
|
+
* @return {void}
|
148
|
+
*/
|
149
|
+
observe(observer) {
|
150
|
+
if (!(observer instanceof Adapter))
|
151
|
+
throw new UnsupportedError("Only Adapters can be observed by dispatch");
|
152
|
+
this.adapter = observer;
|
153
|
+
this.native = observer.native;
|
154
|
+
this.models = Adapter.models(this.adapter.alias);
|
155
|
+
this.initialize().then(() => this.log.verbose(`Dispatch initialized for ${this.adapter.alias} adapter`));
|
156
|
+
}
|
157
|
+
/**
|
158
|
+
* @description Stops observing an adapter
|
159
|
+
* @summary Disconnects this dispatch from an adapter
|
160
|
+
* @param {Observer} observer - The adapter to stop observing
|
161
|
+
* @return {void}
|
162
|
+
*/
|
163
|
+
unObserve(observer) {
|
164
|
+
if (this.adapter !== observer)
|
165
|
+
throw new UnsupportedError("Only the adapter that was used to observe can be unobserved");
|
166
|
+
this.adapter = undefined;
|
167
|
+
}
|
168
|
+
/**
|
169
|
+
* @description Updates observers about a database event
|
170
|
+
* @summary Notifies observers about a change in the database
|
171
|
+
* @param {string} table - The name of the table where the change occurred
|
172
|
+
* @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
|
173
|
+
* @param {EventIds} id - The identifier(s) of the affected record(s)
|
174
|
+
* @return {Promise<void>} A promise that resolves when all observers have been notified
|
175
|
+
*/
|
176
|
+
async updateObservers(table, event, id) {
|
177
|
+
if (!this.adapter)
|
178
|
+
throw new InternalError(`No adapter observed for dispatch`);
|
179
|
+
try {
|
180
|
+
await this.adapter.refresh(table, event, id);
|
181
|
+
}
|
182
|
+
catch (e) {
|
183
|
+
throw new InternalError(`Failed to refresh dispatch: ${e}`);
|
184
|
+
}
|
185
|
+
}
|
186
|
+
}
|
187
|
+
//# 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"]}
|
@@ -0,0 +1,109 @@
|
|
1
|
+
import { Observable, Observer } from "../interfaces";
|
2
|
+
import { EventIds, ObserverFilter } from "./types";
|
3
|
+
import { BulkCrudOperationKeys, OperationKeys } from "@decaf-ts/db-decorators";
|
4
|
+
import { Logger } from "@decaf-ts/logging";
|
5
|
+
/**
|
6
|
+
* @description Manages a collection of observers for database events
|
7
|
+
* @summary The ObserverHandler class implements the Observable interface and provides a centralized
|
8
|
+
* way to manage multiple observers. It allows registering observers with optional filters to control
|
9
|
+
* which events they receive notifications for, and handles the process of notifying all relevant
|
10
|
+
* observers when database events occur.
|
11
|
+
* @class ObserverHandler
|
12
|
+
* @example
|
13
|
+
* ```typescript
|
14
|
+
* // Create an observer handler
|
15
|
+
* const handler = new ObserverHandler();
|
16
|
+
*
|
17
|
+
* // Register an observer
|
18
|
+
* const myObserver = {
|
19
|
+
* refresh: async (table, event, id) => {
|
20
|
+
* console.log(`Change in ${table}: ${event} for ID ${id}`);
|
21
|
+
* }
|
22
|
+
* };
|
23
|
+
*
|
24
|
+
* // Add observer with a filter for only user table events
|
25
|
+
* handler.observe(myObserver, (table, event, id) => table === 'users');
|
26
|
+
*
|
27
|
+
* // Notify observers about an event
|
28
|
+
* await handler.updateObservers(logger, 'users', 'CREATE', 123);
|
29
|
+
*
|
30
|
+
* // Remove an observer when no longer needed
|
31
|
+
* handler.unObserve(myObserver);
|
32
|
+
* ```
|
33
|
+
*/
|
34
|
+
export declare class ObserverHandler implements Observable {
|
35
|
+
/**
|
36
|
+
* @description Collection of registered observers
|
37
|
+
* @summary Array of observer objects along with their optional filters
|
38
|
+
*/
|
39
|
+
protected readonly observers: {
|
40
|
+
observer: Observer;
|
41
|
+
filter?: ObserverFilter;
|
42
|
+
}[];
|
43
|
+
/**
|
44
|
+
* @description Gets the number of registered observers
|
45
|
+
* @summary Returns the count of observers currently registered with this handler
|
46
|
+
* @return {number} The number of registered observers
|
47
|
+
*/
|
48
|
+
count(): number;
|
49
|
+
/**
|
50
|
+
* @description Registers a new observer
|
51
|
+
* @summary Adds an observer to the collection with an optional filter function
|
52
|
+
* @param {Observer} observer - The observer to register
|
53
|
+
* @param {ObserverFilter} [filter] - Optional filter function to determine which events the observer receives
|
54
|
+
* @return {void}
|
55
|
+
*/
|
56
|
+
observe(observer: Observer, filter?: ObserverFilter): void;
|
57
|
+
/**
|
58
|
+
* @description Unregisters an observer
|
59
|
+
* @summary Removes an observer from the collection
|
60
|
+
* @param {Observer} observer - The observer to unregister
|
61
|
+
* @return {void}
|
62
|
+
*/
|
63
|
+
unObserve(observer: Observer): void;
|
64
|
+
/**
|
65
|
+
* @description Notifies all relevant observers about a database event
|
66
|
+
* @summary Filters observers based on their filter functions and calls refresh on each matching observer
|
67
|
+
* @param {Logger} log - Logger for recording notification activities
|
68
|
+
* @param {string} table - The name of the table where the event occurred
|
69
|
+
* @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
|
70
|
+
* @param {EventIds} id - The identifier(s) of the affected record(s)
|
71
|
+
* @param {...any[]} args - Additional arguments to pass to the observers
|
72
|
+
* @return {Promise<void>} A promise that resolves when all observers have been notified
|
73
|
+
* @mermaid
|
74
|
+
* sequenceDiagram
|
75
|
+
* participant Client
|
76
|
+
* participant ObserverHandler
|
77
|
+
* participant Observer
|
78
|
+
*
|
79
|
+
* Client->>ObserverHandler: updateObservers(log, table, event, id, ...args)
|
80
|
+
*
|
81
|
+
* ObserverHandler->>ObserverHandler: Filter observers
|
82
|
+
*
|
83
|
+
* loop For each observer with matching filter
|
84
|
+
* alt Observer has filter
|
85
|
+
* ObserverHandler->>Observer: Apply filter(table, event, id)
|
86
|
+
* alt Filter throws error
|
87
|
+
* ObserverHandler->>Logger: Log error
|
88
|
+
* ObserverHandler-->>ObserverHandler: Skip observer
|
89
|
+
* else Filter returns true
|
90
|
+
* ObserverHandler->>Observer: refresh(table, event, id, ...args)
|
91
|
+
* else Filter returns false
|
92
|
+
* ObserverHandler-->>ObserverHandler: Skip observer
|
93
|
+
* end
|
94
|
+
* else No filter
|
95
|
+
* ObserverHandler->>Observer: refresh(table, event, id, ...args)
|
96
|
+
* end
|
97
|
+
* end
|
98
|
+
*
|
99
|
+
* ObserverHandler->>ObserverHandler: Process results
|
100
|
+
* loop For each result
|
101
|
+
* alt Result is rejected
|
102
|
+
* ObserverHandler->>Logger: Log error
|
103
|
+
* end
|
104
|
+
* end
|
105
|
+
*
|
106
|
+
* ObserverHandler-->>Client: Return
|
107
|
+
*/
|
108
|
+
updateObservers(log: Logger, table: string, event: OperationKeys | BulkCrudOperationKeys | string, id: EventIds, ...args: any[]): Promise<void>;
|
109
|
+
}
|
@@ -0,0 +1,137 @@
|
|
1
|
+
import { InternalError, } from "@decaf-ts/db-decorators";
|
2
|
+
/**
|
3
|
+
* @description Manages a collection of observers for database events
|
4
|
+
* @summary The ObserverHandler class implements the Observable interface and provides a centralized
|
5
|
+
* way to manage multiple observers. It allows registering observers with optional filters to control
|
6
|
+
* which events they receive notifications for, and handles the process of notifying all relevant
|
7
|
+
* observers when database events occur.
|
8
|
+
* @class ObserverHandler
|
9
|
+
* @example
|
10
|
+
* ```typescript
|
11
|
+
* // Create an observer handler
|
12
|
+
* const handler = new ObserverHandler();
|
13
|
+
*
|
14
|
+
* // Register an observer
|
15
|
+
* const myObserver = {
|
16
|
+
* refresh: async (table, event, id) => {
|
17
|
+
* console.log(`Change in ${table}: ${event} for ID ${id}`);
|
18
|
+
* }
|
19
|
+
* };
|
20
|
+
*
|
21
|
+
* // Add observer with a filter for only user table events
|
22
|
+
* handler.observe(myObserver, (table, event, id) => table === 'users');
|
23
|
+
*
|
24
|
+
* // Notify observers about an event
|
25
|
+
* await handler.updateObservers(logger, 'users', 'CREATE', 123);
|
26
|
+
*
|
27
|
+
* // Remove an observer when no longer needed
|
28
|
+
* handler.unObserve(myObserver);
|
29
|
+
* ```
|
30
|
+
*/
|
31
|
+
export class ObserverHandler {
|
32
|
+
constructor() {
|
33
|
+
/**
|
34
|
+
* @description Collection of registered observers
|
35
|
+
* @summary Array of observer objects along with their optional filters
|
36
|
+
*/
|
37
|
+
this.observers = [];
|
38
|
+
}
|
39
|
+
/**
|
40
|
+
* @description Gets the number of registered observers
|
41
|
+
* @summary Returns the count of observers currently registered with this handler
|
42
|
+
* @return {number} The number of registered observers
|
43
|
+
*/
|
44
|
+
count() {
|
45
|
+
return this.observers.length;
|
46
|
+
}
|
47
|
+
/**
|
48
|
+
* @description Registers a new observer
|
49
|
+
* @summary Adds an observer to the collection with an optional filter function
|
50
|
+
* @param {Observer} observer - The observer to register
|
51
|
+
* @param {ObserverFilter} [filter] - Optional filter function to determine which events the observer receives
|
52
|
+
* @return {void}
|
53
|
+
*/
|
54
|
+
observe(observer, filter) {
|
55
|
+
const index = this.observers.map((o) => o.observer).indexOf(observer);
|
56
|
+
if (index !== -1)
|
57
|
+
throw new InternalError("Observer already registered");
|
58
|
+
this.observers.push({ observer: observer, filter: filter });
|
59
|
+
}
|
60
|
+
/**
|
61
|
+
* @description Unregisters an observer
|
62
|
+
* @summary Removes an observer from the collection
|
63
|
+
* @param {Observer} observer - The observer to unregister
|
64
|
+
* @return {void}
|
65
|
+
*/
|
66
|
+
unObserve(observer) {
|
67
|
+
const index = this.observers.map((o) => o.observer).indexOf(observer);
|
68
|
+
if (index === -1)
|
69
|
+
throw new InternalError("Failed to find Observer");
|
70
|
+
this.observers.splice(index, 1);
|
71
|
+
}
|
72
|
+
/**
|
73
|
+
* @description Notifies all relevant observers about a database event
|
74
|
+
* @summary Filters observers based on their filter functions and calls refresh on each matching observer
|
75
|
+
* @param {Logger} log - Logger for recording notification activities
|
76
|
+
* @param {string} table - The name of the table where the event occurred
|
77
|
+
* @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
|
78
|
+
* @param {EventIds} id - The identifier(s) of the affected record(s)
|
79
|
+
* @param {...any[]} args - Additional arguments to pass to the observers
|
80
|
+
* @return {Promise<void>} A promise that resolves when all observers have been notified
|
81
|
+
* @mermaid
|
82
|
+
* sequenceDiagram
|
83
|
+
* participant Client
|
84
|
+
* participant ObserverHandler
|
85
|
+
* participant Observer
|
86
|
+
*
|
87
|
+
* Client->>ObserverHandler: updateObservers(log, table, event, id, ...args)
|
88
|
+
*
|
89
|
+
* ObserverHandler->>ObserverHandler: Filter observers
|
90
|
+
*
|
91
|
+
* loop For each observer with matching filter
|
92
|
+
* alt Observer has filter
|
93
|
+
* ObserverHandler->>Observer: Apply filter(table, event, id)
|
94
|
+
* alt Filter throws error
|
95
|
+
* ObserverHandler->>Logger: Log error
|
96
|
+
* ObserverHandler-->>ObserverHandler: Skip observer
|
97
|
+
* else Filter returns true
|
98
|
+
* ObserverHandler->>Observer: refresh(table, event, id, ...args)
|
99
|
+
* else Filter returns false
|
100
|
+
* ObserverHandler-->>ObserverHandler: Skip observer
|
101
|
+
* end
|
102
|
+
* else No filter
|
103
|
+
* ObserverHandler->>Observer: refresh(table, event, id, ...args)
|
104
|
+
* end
|
105
|
+
* end
|
106
|
+
*
|
107
|
+
* ObserverHandler->>ObserverHandler: Process results
|
108
|
+
* loop For each result
|
109
|
+
* alt Result is rejected
|
110
|
+
* ObserverHandler->>Logger: Log error
|
111
|
+
* end
|
112
|
+
* end
|
113
|
+
*
|
114
|
+
* ObserverHandler-->>Client: Return
|
115
|
+
*/
|
116
|
+
async updateObservers(log, table, event, id, ...args) {
|
117
|
+
const results = await Promise.allSettled(this.observers
|
118
|
+
.filter((o) => {
|
119
|
+
const { filter } = o;
|
120
|
+
if (!filter)
|
121
|
+
return true;
|
122
|
+
try {
|
123
|
+
return filter(table, event, id);
|
124
|
+
}
|
125
|
+
catch (e) {
|
126
|
+
log.error(`Failed to filter observer ${o.observer.toString()}: ${e}`);
|
127
|
+
return false;
|
128
|
+
}
|
129
|
+
})
|
130
|
+
.map((o) => o.observer.refresh(table, event, id, ...args)));
|
131
|
+
results.forEach((result, i) => {
|
132
|
+
if (result.status === "rejected")
|
133
|
+
log.error(`Failed to update observable ${this.observers[i].toString()}: ${result.reason}`);
|
134
|
+
});
|
135
|
+
}
|
136
|
+
}
|
137
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ObserverHandler.js","sourceRoot":"","sources":["../../../src/persistence/ObserverHandler.ts"],"names":[],"mappings":"AAEA,OAAO,EAEL,aAAa,GAEd,MAAM,yBAAyB,CAAC;AAGjC;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4BG;AACH,MAAM,OAAO,eAAe;IAA5B;QACE;;;WAGG;QACgB,cAAS,GAGtB,EAAE,CAAC;IA8GX,CAAC;IA5GC;;;;OAIG;IACH,KAAK;QACH,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC;IAC/B,CAAC;IAED;;;;;;OAMG;IACH,OAAO,CAAC,QAAkB,EAAE,MAAuB;QACjD,MAAM,KAAK,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QACtE,IAAI,KAAK,KAAK,CAAC,CAAC;YAAE,MAAM,IAAI,aAAa,CAAC,6BAA6B,CAAC,CAAC;QACzE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;IAC9D,CAAC;IAED;;;;;OAKG;IACH,SAAS,CAAC,QAAkB;QAC1B,MAAM,KAAK,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QACtE,IAAI,KAAK,KAAK,CAAC,CAAC;YAAE,MAAM,IAAI,aAAa,CAAC,yBAAyB,CAAC,CAAC;QACrE,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;IAClC,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA2CG;IACH,KAAK,CAAC,eAAe,CACnB,GAAW,EACX,KAAa,EACb,KAAqD,EACrD,EAAY,EACZ,GAAG,IAAW;QAEd,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,UAAU,CACtC,IAAI,CAAC,SAAS;aACX,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE;YACZ,MAAM,EAAE,MAAM,EAAE,GAAG,CAAC,CAAC;YACrB,IAAI,CAAC,MAAM;gBAAE,OAAO,IAAI,CAAC;YACzB,IAAI,CAAC;gBACH,OAAO,MAAM,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE,CAAC,CAAC;YAClC,CAAC;YAAC,OAAO,CAAU,EAAE,CAAC;gBACpB,GAAG,CAAC,KAAK,CACP,6BAA6B,CAAC,CAAC,QAAQ,CAAC,QAAQ,EAAE,KAAK,CAAC,EAAE,CAC3D,CAAC;gBACF,OAAO,KAAK,CAAC;YACf,CAAC;QACH,CAAC,CAAC;aACD,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE,EAAE,GAAG,IAAI,CAAC,CAAC,CAC7D,CAAC;QACF,OAAO,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;YAC5B,IAAI,MAAM,CAAC,MAAM,KAAK,UAAU;gBAC9B,GAAG,CAAC,KAAK,CACP,+BAA+B,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,KAAK,MAAM,CAAC,MAAM,EAAE,CAChF,CAAC;QACN,CAAC,CAAC,CAAC;IACL,CAAC;CACF","sourcesContent":["import { Observable, Observer } from \"../interfaces\";\nimport { EventIds, ObserverFilter } from \"./types\";\nimport {\n  BulkCrudOperationKeys,\n  InternalError,\n  OperationKeys,\n} from \"@decaf-ts/db-decorators\";\nimport { Logger } from \"@decaf-ts/logging\";\n\n/**\n * @description Manages a collection of observers for database events\n * @summary The ObserverHandler class implements the Observable interface and provides a centralized\n * way to manage multiple observers. It allows registering observers with optional filters to control\n * which events they receive notifications for, and handles the process of notifying all relevant\n * observers when database events occur.\n * @class ObserverHandler\n * @example\n * ```typescript\n * // Create an observer handler\n * const handler = new ObserverHandler();\n * \n * // Register an observer\n * const myObserver = {\n *   refresh: async (table, event, id) => {\n *     console.log(`Change in ${table}: ${event} for ID ${id}`);\n *   }\n * };\n * \n * // Add observer with a filter for only user table events\n * handler.observe(myObserver, (table, event, id) => table === 'users');\n * \n * // Notify observers about an event\n * await handler.updateObservers(logger, 'users', 'CREATE', 123);\n * \n * // Remove an observer when no longer needed\n * handler.unObserve(myObserver);\n * ```\n */\nexport class ObserverHandler implements Observable {\n  /**\n   * @description Collection of registered observers\n   * @summary Array of observer objects along with their optional filters\n   */\n  protected readonly observers: {\n    observer: Observer;\n    filter?: ObserverFilter;\n  }[] = [];\n\n  /**\n   * @description Gets the number of registered observers\n   * @summary Returns the count of observers currently registered with this handler\n   * @return {number} The number of registered observers\n   */\n  count() {\n    return this.observers.length;\n  }\n\n  /**\n   * @description Registers a new observer\n   * @summary Adds an observer to the collection with an optional filter function\n   * @param {Observer} observer - The observer to register\n   * @param {ObserverFilter} [filter] - Optional filter function to determine which events the observer receives\n   * @return {void}\n   */\n  observe(observer: Observer, filter?: ObserverFilter): void {\n    const index = this.observers.map((o) => o.observer).indexOf(observer);\n    if (index !== -1) throw new InternalError(\"Observer already registered\");\n    this.observers.push({ observer: observer, filter: filter });\n  }\n\n  /**\n   * @description Unregisters an observer\n   * @summary Removes an observer from the collection\n   * @param {Observer} observer - The observer to unregister\n   * @return {void}\n   */\n  unObserve(observer: Observer): void {\n    const index = this.observers.map((o) => o.observer).indexOf(observer);\n    if (index === -1) throw new InternalError(\"Failed to find Observer\");\n    this.observers.splice(index, 1);\n  }\n\n  /**\n   * @description Notifies all relevant observers about a database event\n   * @summary Filters observers based on their filter functions and calls refresh on each matching observer\n   * @param {Logger} log - Logger for recording notification activities\n   * @param {string} table - The name of the table where the event occurred\n   * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred\n   * @param {EventIds} id - The identifier(s) of the affected record(s)\n   * @param {...any[]} args - Additional arguments to pass to the observers\n   * @return {Promise<void>} A promise that resolves when all observers have been notified\n   * @mermaid\n   * sequenceDiagram\n   *   participant Client\n   *   participant ObserverHandler\n   *   participant Observer\n   *   \n   *   Client->>ObserverHandler: updateObservers(log, table, event, id, ...args)\n   *   \n   *   ObserverHandler->>ObserverHandler: Filter observers\n   *   \n   *   loop For each observer with matching filter\n   *     alt Observer has filter\n   *       ObserverHandler->>Observer: Apply filter(table, event, id)\n   *       alt Filter throws error\n   *         ObserverHandler->>Logger: Log error\n   *         ObserverHandler-->>ObserverHandler: Skip observer\n   *       else Filter returns true\n   *         ObserverHandler->>Observer: refresh(table, event, id, ...args)\n   *       else Filter returns false\n   *         ObserverHandler-->>ObserverHandler: Skip observer\n   *       end\n   *     else No filter\n   *       ObserverHandler->>Observer: refresh(table, event, id, ...args)\n   *     end\n   *   end\n   *   \n   *   ObserverHandler->>ObserverHandler: Process results\n   *   loop For each result\n   *     alt Result is rejected\n   *       ObserverHandler->>Logger: Log error\n   *     end\n   *   end\n   *   \n   *   ObserverHandler-->>Client: Return\n   */\n  async updateObservers(\n    log: Logger,\n    table: string,\n    event: OperationKeys | BulkCrudOperationKeys | string,\n    id: EventIds,\n    ...args: any[]\n  ): Promise<void> {\n    const results = await Promise.allSettled(\n      this.observers\n        .filter((o) => {\n          const { filter } = o;\n          if (!filter) return true;\n          try {\n            return filter(table, event, id);\n          } catch (e: unknown) {\n            log.error(\n              `Failed to filter observer ${o.observer.toString()}: ${e}`\n            );\n            return false;\n          }\n        })\n        .map((o) => o.observer.refresh(table, event, id, ...args))\n    );\n    results.forEach((result, i) => {\n      if (result.status === \"rejected\")\n        log.error(\n          `Failed to update observable ${this.observers[i].toString()}: ${result.reason}`\n        );\n    });\n  }\n}\n"]}
|