@decaf-ts/core 0.5.1 → 0.5.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (206) hide show
  1. package/LICENSE.md +21 -157
  2. package/README.md +652 -15
  3. package/dist/core.cjs +2111 -133
  4. package/dist/core.esm.cjs +2112 -134
  5. package/lib/esm/identity/decorators.d.ts +52 -7
  6. package/lib/esm/identity/decorators.js +58 -13
  7. package/lib/esm/identity/index.js +3 -3
  8. package/lib/esm/identity/utils.d.ts +19 -0
  9. package/lib/esm/identity/utils.js +22 -3
  10. package/lib/esm/index.d.ts +10 -3
  11. package/lib/esm/index.js +19 -12
  12. package/lib/esm/interfaces/ErrorParser.d.ts +12 -0
  13. package/lib/esm/interfaces/ErrorParser.js +1 -1
  14. package/lib/esm/interfaces/Executor.d.ts +13 -0
  15. package/lib/esm/interfaces/Executor.js +1 -1
  16. package/lib/esm/interfaces/Observable.d.ts +27 -0
  17. package/lib/esm/interfaces/Observable.js +1 -1
  18. package/lib/esm/interfaces/Observer.d.ts +12 -0
  19. package/lib/esm/interfaces/Observer.js +1 -1
  20. package/lib/esm/interfaces/Paginatable.d.ts +15 -0
  21. package/lib/esm/interfaces/Paginatable.js +1 -1
  22. package/lib/esm/interfaces/Queriable.d.ts +34 -9
  23. package/lib/esm/interfaces/Queriable.js +1 -1
  24. package/lib/esm/interfaces/RawExecutor.d.ts +14 -0
  25. package/lib/esm/interfaces/RawExecutor.js +1 -1
  26. package/lib/esm/interfaces/SequenceOptions.d.ts +52 -0
  27. package/lib/esm/interfaces/SequenceOptions.js +19 -1
  28. package/lib/esm/interfaces/index.js +8 -8
  29. package/lib/esm/model/BaseModel.d.ts +31 -0
  30. package/lib/esm/model/BaseModel.js +24 -1
  31. package/lib/esm/model/construction.d.ts +433 -0
  32. package/lib/esm/model/construction.js +444 -5
  33. package/lib/esm/model/decorators.d.ts +159 -29
  34. package/lib/esm/model/decorators.js +167 -37
  35. package/lib/esm/model/index.js +5 -5
  36. package/lib/esm/model/types.d.ts +9 -0
  37. package/lib/esm/model/types.js +1 -1
  38. package/lib/esm/persistence/Adapter.d.ts +358 -17
  39. package/lib/esm/persistence/Adapter.js +292 -24
  40. package/lib/esm/persistence/Dispatch.d.ts +114 -1
  41. package/lib/esm/persistence/Dispatch.js +104 -6
  42. package/lib/esm/persistence/ObserverHandler.d.ts +95 -0
  43. package/lib/esm/persistence/ObserverHandler.js +96 -1
  44. package/lib/esm/persistence/Sequence.d.ts +89 -0
  45. package/lib/esm/persistence/Sequence.js +71 -2
  46. package/lib/esm/persistence/constants.d.ts +22 -0
  47. package/lib/esm/persistence/constants.js +23 -1
  48. package/lib/esm/persistence/decorators.d.ts +10 -0
  49. package/lib/esm/persistence/decorators.js +13 -3
  50. package/lib/esm/persistence/errors.d.ts +23 -0
  51. package/lib/esm/persistence/errors.js +24 -1
  52. package/lib/esm/persistence/index.js +9 -9
  53. package/lib/esm/persistence/types.d.ts +18 -0
  54. package/lib/esm/persistence/types.js +1 -1
  55. package/lib/esm/query/Condition.d.ts +78 -31
  56. package/lib/esm/query/Condition.js +134 -55
  57. package/lib/esm/query/Paginator.d.ts +56 -0
  58. package/lib/esm/query/Paginator.js +58 -2
  59. package/lib/esm/query/Statement.d.ts +51 -0
  60. package/lib/esm/query/Statement.js +55 -4
  61. package/lib/esm/query/constants.d.ts +25 -0
  62. package/lib/esm/query/constants.js +26 -1
  63. package/lib/esm/query/errors.d.ts +14 -0
  64. package/lib/esm/query/errors.js +15 -1
  65. package/lib/esm/query/index.js +8 -8
  66. package/lib/esm/query/options.d.ts +21 -3
  67. package/lib/esm/query/options.js +1 -1
  68. package/lib/esm/query/selectors.d.ts +26 -0
  69. package/lib/esm/query/selectors.js +1 -1
  70. package/lib/esm/ram/RamAdapter.d.ts +311 -0
  71. package/lib/esm/ram/RamAdapter.js +319 -8
  72. package/lib/esm/ram/RamContext.d.ts +16 -1
  73. package/lib/esm/ram/RamContext.js +18 -3
  74. package/lib/esm/ram/RamPaginator.d.ts +43 -0
  75. package/lib/esm/ram/RamPaginator.js +55 -3
  76. package/lib/esm/ram/RamSequence.d.ts +61 -0
  77. package/lib/esm/ram/RamSequence.js +66 -5
  78. package/lib/esm/ram/RamStatement.d.ts +74 -0
  79. package/lib/esm/ram/RamStatement.js +78 -4
  80. package/lib/esm/ram/constants.d.ts +8 -0
  81. package/lib/esm/ram/constants.js +9 -1
  82. package/lib/esm/ram/handlers.d.ts +19 -0
  83. package/lib/esm/ram/handlers.js +21 -2
  84. package/lib/esm/ram/index.js +11 -11
  85. package/lib/esm/ram/model/RamSequence.d.ts +25 -0
  86. package/lib/esm/ram/model/RamSequence.js +21 -3
  87. package/lib/esm/ram/model/index.js +2 -2
  88. package/lib/esm/ram/types.d.ts +42 -0
  89. package/lib/esm/ram/types.js +1 -1
  90. package/lib/esm/repository/Repository.d.ts +363 -8
  91. package/lib/esm/repository/Repository.js +369 -24
  92. package/lib/esm/repository/constants.d.ts +25 -0
  93. package/lib/esm/repository/constants.js +26 -1
  94. package/lib/esm/repository/decorators.d.ts +27 -0
  95. package/lib/esm/repository/decorators.js +29 -2
  96. package/lib/esm/repository/errors.d.ts +12 -5
  97. package/lib/esm/repository/errors.js +13 -6
  98. package/lib/esm/repository/index.js +8 -8
  99. package/lib/esm/repository/injectables.d.ts +18 -0
  100. package/lib/esm/repository/injectables.js +23 -5
  101. package/lib/esm/repository/types.d.ts +15 -0
  102. package/lib/esm/repository/types.js +1 -1
  103. package/lib/esm/repository/utils.d.ts +11 -0
  104. package/lib/esm/repository/utils.js +15 -4
  105. package/lib/esm/utils/decorators.d.ts +8 -0
  106. package/lib/esm/utils/decorators.js +9 -1
  107. package/lib/esm/utils/errors.d.ts +46 -0
  108. package/lib/esm/utils/errors.js +47 -1
  109. package/lib/esm/utils/index.js +3 -3
  110. package/lib/identity/decorators.cjs +53 -8
  111. package/lib/identity/decorators.d.ts +52 -7
  112. package/lib/identity/utils.cjs +20 -1
  113. package/lib/identity/utils.d.ts +19 -0
  114. package/lib/index.cjs +11 -4
  115. package/lib/index.d.ts +10 -3
  116. package/lib/interfaces/ErrorParser.cjs +1 -1
  117. package/lib/interfaces/ErrorParser.d.ts +12 -0
  118. package/lib/interfaces/Executor.cjs +1 -1
  119. package/lib/interfaces/Executor.d.ts +13 -0
  120. package/lib/interfaces/Observable.cjs +1 -1
  121. package/lib/interfaces/Observable.d.ts +27 -0
  122. package/lib/interfaces/Observer.cjs +1 -1
  123. package/lib/interfaces/Observer.d.ts +12 -0
  124. package/lib/interfaces/Paginatable.cjs +1 -1
  125. package/lib/interfaces/Paginatable.d.ts +15 -0
  126. package/lib/interfaces/Queriable.cjs +1 -1
  127. package/lib/interfaces/Queriable.d.ts +34 -9
  128. package/lib/interfaces/RawExecutor.cjs +1 -1
  129. package/lib/interfaces/RawExecutor.d.ts +14 -0
  130. package/lib/interfaces/SequenceOptions.cjs +19 -1
  131. package/lib/interfaces/SequenceOptions.d.ts +52 -0
  132. package/lib/model/BaseModel.cjs +24 -1
  133. package/lib/model/BaseModel.d.ts +31 -0
  134. package/lib/model/construction.cjs +441 -2
  135. package/lib/model/construction.d.ts +433 -0
  136. package/lib/model/decorators.cjs +160 -30
  137. package/lib/model/decorators.d.ts +159 -29
  138. package/lib/model/types.cjs +1 -1
  139. package/lib/model/types.d.ts +9 -0
  140. package/lib/persistence/Adapter.cjs +287 -19
  141. package/lib/persistence/Adapter.d.ts +358 -17
  142. package/lib/persistence/Dispatch.cjs +102 -4
  143. package/lib/persistence/Dispatch.d.ts +114 -1
  144. package/lib/persistence/ObserverHandler.cjs +96 -1
  145. package/lib/persistence/ObserverHandler.d.ts +95 -0
  146. package/lib/persistence/Sequence.cjs +70 -1
  147. package/lib/persistence/Sequence.d.ts +89 -0
  148. package/lib/persistence/constants.cjs +23 -1
  149. package/lib/persistence/constants.d.ts +22 -0
  150. package/lib/persistence/decorators.cjs +11 -1
  151. package/lib/persistence/decorators.d.ts +10 -0
  152. package/lib/persistence/errors.cjs +24 -1
  153. package/lib/persistence/errors.d.ts +23 -0
  154. package/lib/persistence/types.cjs +1 -1
  155. package/lib/persistence/types.d.ts +18 -0
  156. package/lib/query/Condition.cjs +132 -53
  157. package/lib/query/Condition.d.ts +78 -31
  158. package/lib/query/Paginator.cjs +57 -1
  159. package/lib/query/Paginator.d.ts +56 -0
  160. package/lib/query/Statement.cjs +52 -1
  161. package/lib/query/Statement.d.ts +51 -0
  162. package/lib/query/constants.cjs +26 -1
  163. package/lib/query/constants.d.ts +25 -0
  164. package/lib/query/errors.cjs +15 -1
  165. package/lib/query/errors.d.ts +14 -0
  166. package/lib/query/options.cjs +1 -1
  167. package/lib/query/options.d.ts +21 -3
  168. package/lib/query/selectors.cjs +1 -1
  169. package/lib/query/selectors.d.ts +26 -0
  170. package/lib/ram/RamAdapter.cjs +312 -1
  171. package/lib/ram/RamAdapter.d.ts +311 -0
  172. package/lib/ram/RamContext.cjs +18 -3
  173. package/lib/ram/RamContext.d.ts +16 -1
  174. package/lib/ram/RamPaginator.cjs +54 -2
  175. package/lib/ram/RamPaginator.d.ts +43 -0
  176. package/lib/ram/RamSequence.cjs +63 -2
  177. package/lib/ram/RamSequence.d.ts +61 -0
  178. package/lib/ram/RamStatement.cjs +75 -1
  179. package/lib/ram/RamStatement.d.ts +74 -0
  180. package/lib/ram/constants.cjs +9 -1
  181. package/lib/ram/constants.d.ts +8 -0
  182. package/lib/ram/handlers.cjs +20 -1
  183. package/lib/ram/handlers.d.ts +19 -0
  184. package/lib/ram/model/RamSequence.cjs +19 -1
  185. package/lib/ram/model/RamSequence.d.ts +25 -0
  186. package/lib/ram/types.cjs +1 -1
  187. package/lib/ram/types.d.ts +42 -0
  188. package/lib/repository/Repository.cjs +360 -15
  189. package/lib/repository/Repository.d.ts +363 -8
  190. package/lib/repository/constants.cjs +26 -1
  191. package/lib/repository/constants.d.ts +25 -0
  192. package/lib/repository/decorators.cjs +28 -1
  193. package/lib/repository/decorators.d.ts +27 -0
  194. package/lib/repository/errors.cjs +13 -6
  195. package/lib/repository/errors.d.ts +12 -5
  196. package/lib/repository/injectables.cjs +19 -1
  197. package/lib/repository/injectables.d.ts +18 -0
  198. package/lib/repository/types.cjs +1 -1
  199. package/lib/repository/types.d.ts +15 -0
  200. package/lib/repository/utils.cjs +12 -1
  201. package/lib/repository/utils.d.ts +11 -0
  202. package/lib/utils/decorators.cjs +9 -1
  203. package/lib/utils/decorators.d.ts +8 -0
  204. package/lib/utils/errors.cjs +47 -1
  205. package/lib/utils/errors.d.ts +46 -0
  206. package/package.json +5 -5
@@ -4,15 +4,128 @@ import { Observable, Observer } from "../interfaces";
4
4
  import { Adapter } from "./Adapter";
5
5
  import { Logger } from "@decaf-ts/logging";
6
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
+ */
7
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
+ */
8
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
+ */
9
43
  protected native?: Y;
44
+ /**
45
+ * @description List of model constructors
46
+ * @summary Array of model constructors that are registered with the adapter
47
+ */
10
48
  protected models: ModelConstructor<any>[];
49
+ /**
50
+ * @description Logger instance
51
+ * @summary Logger for recording dispatch activities
52
+ */
11
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
+ */
12
59
  protected get log(): Logger;
60
+ /**
61
+ * @description Creates a new Dispatch instance
62
+ * @summary Initializes a new Dispatch instance without any adapter
63
+ */
13
64
  constructor();
14
- protected initialize(): void;
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
+ */
15
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
+ */
16
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
+ */
17
130
  updateObservers(table: string, event: OperationKeys | BulkCrudOperationKeys | string, id: EventIds): Promise<void>;
18
131
  }
@@ -2,25 +2,120 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.ObserverHandler = void 0;
4
4
  const db_decorators_1 = require("@decaf-ts/db-decorators");
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
+ */
5
34
  class ObserverHandler {
6
35
  constructor() {
36
+ /**
37
+ * @description Collection of registered observers
38
+ * @summary Array of observer objects along with their optional filters
39
+ */
7
40
  this.observers = [];
8
41
  }
42
+ /**
43
+ * @description Gets the number of registered observers
44
+ * @summary Returns the count of observers currently registered with this handler
45
+ * @return {number} The number of registered observers
46
+ */
9
47
  count() {
10
48
  return this.observers.length;
11
49
  }
50
+ /**
51
+ * @description Registers a new observer
52
+ * @summary Adds an observer to the collection with an optional filter function
53
+ * @param {Observer} observer - The observer to register
54
+ * @param {ObserverFilter} [filter] - Optional filter function to determine which events the observer receives
55
+ * @return {void}
56
+ */
12
57
  observe(observer, filter) {
13
58
  const index = this.observers.map((o) => o.observer).indexOf(observer);
14
59
  if (index !== -1)
15
60
  throw new db_decorators_1.InternalError("Observer already registered");
16
61
  this.observers.push({ observer: observer, filter: filter });
17
62
  }
63
+ /**
64
+ * @description Unregisters an observer
65
+ * @summary Removes an observer from the collection
66
+ * @param {Observer} observer - The observer to unregister
67
+ * @return {void}
68
+ */
18
69
  unObserve(observer) {
19
70
  const index = this.observers.map((o) => o.observer).indexOf(observer);
20
71
  if (index === -1)
21
72
  throw new db_decorators_1.InternalError("Failed to find Observer");
22
73
  this.observers.splice(index, 1);
23
74
  }
75
+ /**
76
+ * @description Notifies all relevant observers about a database event
77
+ * @summary Filters observers based on their filter functions and calls refresh on each matching observer
78
+ * @param {Logger} log - Logger for recording notification activities
79
+ * @param {string} table - The name of the table where the event occurred
80
+ * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
81
+ * @param {EventIds} id - The identifier(s) of the affected record(s)
82
+ * @param {...any[]} args - Additional arguments to pass to the observers
83
+ * @return {Promise<void>} A promise that resolves when all observers have been notified
84
+ * @mermaid
85
+ * sequenceDiagram
86
+ * participant Client
87
+ * participant ObserverHandler
88
+ * participant Observer
89
+ *
90
+ * Client->>ObserverHandler: updateObservers(log, table, event, id, ...args)
91
+ *
92
+ * ObserverHandler->>ObserverHandler: Filter observers
93
+ *
94
+ * loop For each observer with matching filter
95
+ * alt Observer has filter
96
+ * ObserverHandler->>Observer: Apply filter(table, event, id)
97
+ * alt Filter throws error
98
+ * ObserverHandler->>Logger: Log error
99
+ * ObserverHandler-->>ObserverHandler: Skip observer
100
+ * else Filter returns true
101
+ * ObserverHandler->>Observer: refresh(table, event, id, ...args)
102
+ * else Filter returns false
103
+ * ObserverHandler-->>ObserverHandler: Skip observer
104
+ * end
105
+ * else No filter
106
+ * ObserverHandler->>Observer: refresh(table, event, id, ...args)
107
+ * end
108
+ * end
109
+ *
110
+ * ObserverHandler->>ObserverHandler: Process results
111
+ * loop For each result
112
+ * alt Result is rejected
113
+ * ObserverHandler->>Logger: Log error
114
+ * end
115
+ * end
116
+ *
117
+ * ObserverHandler-->>Client: Return
118
+ */
24
119
  async updateObservers(log, table, event, id, ...args) {
25
120
  const results = await Promise.allSettled(this.observers
26
121
  .filter((o) => {
@@ -43,4 +138,4 @@ class ObserverHandler {
43
138
  }
44
139
  }
45
140
  exports.ObserverHandler = ObserverHandler;
46
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiT2JzZXJ2ZXJIYW5kbGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3BlcnNpc3RlbmNlL09ic2VydmVySGFuZGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFFQSwyREFJaUM7QUFHakMsTUFBYSxlQUFlO0lBQTVCO1FBQ3FCLGNBQVMsR0FHdEIsRUFBRSxDQUFDO0lBZ0RYLENBQUM7SUE5Q0MsS0FBSztRQUNILE9BQU8sSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUM7SUFDL0IsQ0FBQztJQUVELE9BQU8sQ0FBQyxRQUFrQixFQUFFLE1BQXVCO1FBQ2pELE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsUUFBUSxDQUFDLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQ3RFLElBQUksS0FBSyxLQUFLLENBQUMsQ0FBQztZQUFFLE1BQU0sSUFBSSw2QkFBYSxDQUFDLDZCQUE2QixDQUFDLENBQUM7UUFDekUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsRUFBRSxRQUFRLEVBQUUsUUFBUSxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsQ0FBQyxDQUFDO0lBQzlELENBQUM7SUFFRCxTQUFTLENBQUMsUUFBa0I7UUFDMUIsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxRQUFRLENBQUMsQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLENBQUM7UUFDdEUsSUFBSSxLQUFLLEtBQUssQ0FBQyxDQUFDO1lBQUUsTUFBTSxJQUFJLDZCQUFhLENBQUMseUJBQXlCLENBQUMsQ0FBQztRQUNyRSxJQUFJLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQyxLQUFLLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFDbEMsQ0FBQztJQUVELEtBQUssQ0FBQyxlQUFlLENBQ25CLEdBQVcsRUFDWCxLQUFhLEVBQ2IsS0FBcUQsRUFDckQsRUFBWSxFQUNaLEdBQUcsSUFBVztRQUVkLE1BQU0sT0FBTyxHQUFHLE1BQU0sT0FBTyxDQUFDLFVBQVUsQ0FDdEMsSUFBSSxDQUFDLFNBQVM7YUFDWCxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRTtZQUNaLE1BQU0sRUFBRSxNQUFNLEVBQUUsR0FBRyxDQUFDLENBQUM7WUFDckIsSUFBSSxDQUFDLE1BQU07Z0JBQUUsT0FBTyxJQUFJLENBQUM7WUFDekIsSUFBSSxDQUFDO2dCQUNILE9BQU8sTUFBTSxDQUFDLEtBQUssRUFBRSxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7WUFDbEMsQ0FBQztZQUFDLE9BQU8sQ0FBVSxFQUFFLENBQUM7Z0JBQ3BCLEdBQUcsQ0FBQyxLQUFLLENBQ1AsNkJBQTZCLENBQUMsQ0FBQyxRQUFRLENBQUMsUUFBUSxFQUFFLEtBQUssQ0FBQyxFQUFFLENBQzNELENBQUM7Z0JBQ0YsT0FBTyxLQUFLLENBQUM7WUFDZixDQUFDO1FBQ0gsQ0FBQyxDQUFDO2FBQ0QsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxLQUFLLEVBQUUsS0FBSyxFQUFFLEVBQUUsRUFBRSxHQUFHLElBQUksQ0FBQyxDQUFDLENBQzdELENBQUM7UUFDRixPQUFPLENBQUMsT0FBTyxDQUFDLENBQUMsTUFBTSxFQUFFLENBQUMsRUFBRSxFQUFFO1lBQzVCLElBQUksTUFBTSxDQUFDLE1BQU0sS0FBSyxVQUFVO2dCQUM5QixHQUFHLENBQUMsS0FBSyxDQUNQLCtCQUErQixJQUFJLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLFFBQVEsRUFBRSxLQUFLLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FDaEYsQ0FBQztRQUNOLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBcERELDBDQW9EQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE9ic2VydmFibGUsIE9ic2VydmVyIH0gZnJvbSBcIi4uL2ludGVyZmFjZXNcIjtcbmltcG9ydCB7IEV2ZW50SWRzLCBPYnNlcnZlckZpbHRlciB9IGZyb20gXCIuL3R5cGVzXCI7XG5pbXBvcnQge1xuICBCdWxrQ3J1ZE9wZXJhdGlvbktleXMsXG4gIEludGVybmFsRXJyb3IsXG4gIE9wZXJhdGlvbktleXMsXG59IGZyb20gXCJAZGVjYWYtdHMvZGItZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHsgTG9nZ2VyIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5cbmV4cG9ydCBjbGFzcyBPYnNlcnZlckhhbmRsZXIgaW1wbGVtZW50cyBPYnNlcnZhYmxlIHtcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IG9ic2VydmVyczoge1xuICAgIG9ic2VydmVyOiBPYnNlcnZlcjtcbiAgICBmaWx0ZXI/OiBPYnNlcnZlckZpbHRlcjtcbiAgfVtdID0gW107XG5cbiAgY291bnQoKSB7XG4gICAgcmV0dXJuIHRoaXMub2JzZXJ2ZXJzLmxlbmd0aDtcbiAgfVxuXG4gIG9ic2VydmUob2JzZXJ2ZXI6IE9ic2VydmVyLCBmaWx0ZXI/OiBPYnNlcnZlckZpbHRlcik6IHZvaWQge1xuICAgIGNvbnN0IGluZGV4ID0gdGhpcy5vYnNlcnZlcnMubWFwKChvKSA9PiBvLm9ic2VydmVyKS5pbmRleE9mKG9ic2VydmVyKTtcbiAgICBpZiAoaW5kZXggIT09IC0xKSB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihcIk9ic2VydmVyIGFscmVhZHkgcmVnaXN0ZXJlZFwiKTtcbiAgICB0aGlzLm9ic2VydmVycy5wdXNoKHsgb2JzZXJ2ZXI6IG9ic2VydmVyLCBmaWx0ZXI6IGZpbHRlciB9KTtcbiAgfVxuXG4gIHVuT2JzZXJ2ZShvYnNlcnZlcjogT2JzZXJ2ZXIpOiB2b2lkIHtcbiAgICBjb25zdCBpbmRleCA9IHRoaXMub2JzZXJ2ZXJzLm1hcCgobykgPT4gby5vYnNlcnZlcikuaW5kZXhPZihvYnNlcnZlcik7XG4gICAgaWYgKGluZGV4ID09PSAtMSkgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoXCJGYWlsZWQgdG8gZmluZCBPYnNlcnZlclwiKTtcbiAgICB0aGlzLm9ic2VydmVycy5zcGxpY2UoaW5kZXgsIDEpO1xuICB9XG5cbiAgYXN5bmMgdXBkYXRlT2JzZXJ2ZXJzKFxuICAgIGxvZzogTG9nZ2VyLFxuICAgIHRhYmxlOiBzdHJpbmcsXG4gICAgZXZlbnQ6IE9wZXJhdGlvbktleXMgfCBCdWxrQ3J1ZE9wZXJhdGlvbktleXMgfCBzdHJpbmcsXG4gICAgaWQ6IEV2ZW50SWRzLFxuICAgIC4uLmFyZ3M6IGFueVtdXG4gICk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IHJlc3VsdHMgPSBhd2FpdCBQcm9taXNlLmFsbFNldHRsZWQoXG4gICAgICB0aGlzLm9ic2VydmVyc1xuICAgICAgICAuZmlsdGVyKChvKSA9PiB7XG4gICAgICAgICAgY29uc3QgeyBmaWx0ZXIgfSA9IG87XG4gICAgICAgICAgaWYgKCFmaWx0ZXIpIHJldHVybiB0cnVlO1xuICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICByZXR1cm4gZmlsdGVyKHRhYmxlLCBldmVudCwgaWQpO1xuICAgICAgICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgICAgICAgIGxvZy5lcnJvcihcbiAgICAgICAgICAgICAgYEZhaWxlZCB0byBmaWx0ZXIgb2JzZXJ2ZXIgJHtvLm9ic2VydmVyLnRvU3RyaW5nKCl9OiAke2V9YFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgICAgICB9XG4gICAgICAgIH0pXG4gICAgICAgIC5tYXAoKG8pID0+IG8ub2JzZXJ2ZXIucmVmcmVzaCh0YWJsZSwgZXZlbnQsIGlkLCAuLi5hcmdzKSlcbiAgICApO1xuICAgIHJlc3VsdHMuZm9yRWFjaCgocmVzdWx0LCBpKSA9PiB7XG4gICAgICBpZiAocmVzdWx0LnN0YXR1cyA9PT0gXCJyZWplY3RlZFwiKVxuICAgICAgICBsb2cuZXJyb3IoXG4gICAgICAgICAgYEZhaWxlZCB0byB1cGRhdGUgb2JzZXJ2YWJsZSAke3RoaXMub2JzZXJ2ZXJzW2ldLnRvU3RyaW5nKCl9OiAke3Jlc3VsdC5yZWFzb259YFxuICAgICAgICApO1xuICAgIH0pO1xuICB9XG59XG4iXX0=
141
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ObserverHandler.js","sourceRoot":"","sources":["../../src/persistence/ObserverHandler.ts"],"names":[],"mappings":";;;AAEA,2DAIiC;AAGjC;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4BG;AACH,MAAa,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,6BAAa,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,6BAAa,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;AAtHD,0CAsHC","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"]}
@@ -2,13 +2,108 @@ import { Observable, Observer } from "../interfaces";
2
2
  import { EventIds, ObserverFilter } from "./types";
3
3
  import { BulkCrudOperationKeys, OperationKeys } from "@decaf-ts/db-decorators";
4
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
+ */
5
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
+ */
6
39
  protected readonly observers: {
7
40
  observer: Observer;
8
41
  filter?: ObserverFilter;
9
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
+ */
10
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
+ */
11
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
+ */
12
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
+ */
13
108
  updateObservers(log: Logger, table: string, event: OperationKeys | BulkCrudOperationKeys | string, id: EventIds, ...args: any[]): Promise<void>;
14
109
  }
@@ -4,18 +4,87 @@ exports.Sequence = void 0;
4
4
  const utils_1 = require("./../identity/utils.cjs");
5
5
  const db_decorators_1 = require("@decaf-ts/db-decorators");
6
6
  const logging_1 = require("@decaf-ts/logging");
7
+ /**
8
+ * @description Abstract base class for sequence generation
9
+ * @summary Provides a framework for generating sequential values (like primary keys) in the persistence layer.
10
+ * Implementations of this class handle the specifics of how sequences are stored and incremented in different
11
+ * database systems.
12
+ * @param {SequenceOptions} options - Configuration options for the sequence generator
13
+ * @class Sequence
14
+ * @example
15
+ * ```typescript
16
+ * // Example implementation for a specific database
17
+ * class PostgresSequence extends Sequence {
18
+ * constructor(options: SequenceOptions) {
19
+ * super(options);
20
+ * }
21
+ *
22
+ * async next(): Promise<number> {
23
+ * // Implementation to get next value from PostgreSQL sequence
24
+ * const result = await this.options.executor.raw(`SELECT nextval('${this.options.name}')`);
25
+ * return parseInt(result.rows[0].nextval);
26
+ * }
27
+ *
28
+ * async current(): Promise<number> {
29
+ * // Implementation to get current value from PostgreSQL sequence
30
+ * const result = await this.options.executor.raw(`SELECT currval('${this.options.name}')`);
31
+ * return parseInt(result.rows[0].currval);
32
+ * }
33
+ *
34
+ * async range(count: number): Promise<number[]> {
35
+ * // Implementation to get a range of values
36
+ * const values: number[] = [];
37
+ * for (let i = 0; i < count; i++) {
38
+ * values.push(await this.next());
39
+ * }
40
+ * return values;
41
+ * }
42
+ * }
43
+ *
44
+ * // Usage
45
+ * const sequence = new PostgresSequence({
46
+ * name: 'user_id_seq',
47
+ * executor: dbExecutor
48
+ * });
49
+ *
50
+ * const nextId = await sequence.next();
51
+ * ```
52
+ */
7
53
  class Sequence {
54
+ /**
55
+ * @description Accessor for the logger instance
56
+ * @summary Gets or initializes the logger for this sequence
57
+ * @return {Logger} The logger instance
58
+ */
8
59
  get log() {
9
60
  if (!this.logger)
10
61
  this.logger = logging_1.Logging.for(this);
11
62
  return this.logger;
12
63
  }
64
+ /**
65
+ * @description Creates a new sequence instance
66
+ * @summary Protected constructor that initializes the sequence with the provided options
67
+ */
13
68
  constructor(options) {
14
69
  this.options = options;
15
70
  }
71
+ /**
72
+ * @description Gets the primary key sequence name for a model
73
+ * @summary Utility method that returns the standardized sequence name for a model's primary key
74
+ * @template M - The model type
75
+ * @param {M|Constructor<M>} model - The model instance or constructor
76
+ * @return {string} The sequence name for the model's primary key
77
+ */
16
78
  static pk(model) {
17
79
  return (0, utils_1.sequenceNameForModel)(model, "pk");
18
80
  }
81
+ /**
82
+ * @description Parses a sequence value to the appropriate type
83
+ * @summary Converts a sequence value to the specified type (Number or BigInt)
84
+ * @param {"Number"|"BigInt"|undefined} type - The target type to convert to
85
+ * @param {string|number|bigint} value - The value to convert
86
+ * @return {string|number|bigint} The converted value
87
+ */
19
88
  static parseValue(type, value) {
20
89
  switch (type) {
21
90
  case "Number":
@@ -32,4 +101,4 @@ class Sequence {
32
101
  }
33
102
  }
34
103
  exports.Sequence = Sequence;
35
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU2VxdWVuY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcGVyc2lzdGVuY2UvU2VxdWVuY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EsbURBQXlEO0FBRXpELDJEQUF3RDtBQUN4RCwrQ0FBb0Q7QUFFcEQsTUFBc0IsUUFBUTtJQUc1QixJQUFjLEdBQUc7UUFDZixJQUFJLENBQUMsSUFBSSxDQUFDLE1BQU07WUFBRSxJQUFJLENBQUMsTUFBTSxHQUFHLGlCQUFPLENBQUMsR0FBRyxDQUFDLElBQVcsQ0FBQyxDQUFDO1FBQ3pELE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQztJQUNyQixDQUFDO0lBRUQsWUFBeUMsT0FBd0I7UUFBeEIsWUFBTyxHQUFQLE9BQU8sQ0FBaUI7SUFBRyxDQUFDO0lBTXJFLE1BQU0sQ0FBQyxFQUFFLENBQWtCLEtBQXlCO1FBQ2xELE9BQU8sSUFBQSw0QkFBb0IsRUFBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDM0MsQ0FBQztJQUVELE1BQU0sQ0FBQyxVQUFVLENBQ2YsSUFBcUMsRUFDckMsS0FBK0I7UUFFL0IsUUFBUSxJQUFJLEVBQUUsQ0FBQztZQUNiLEtBQUssUUFBUTtnQkFDWCxPQUFPLE9BQU8sS0FBSyxLQUFLLFFBQVE7b0JBQzlCLENBQUMsQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDO29CQUNqQixDQUFDLENBQUMsT0FBTyxLQUFLLEtBQUssUUFBUTt3QkFDekIsQ0FBQyxDQUFDLEtBQUs7d0JBQ1AsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUN0QixLQUFLLFFBQVE7Z0JBQ1gsT0FBTyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUM7WUFDdkI7Z0JBQ0UsTUFBTSxJQUFJLDZCQUFhLENBQUMscUJBQXFCLENBQUMsQ0FBQztRQUNuRCxDQUFDO0lBQ0gsQ0FBQztDQUNGO0FBbkNELDRCQW1DQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbnN0cnVjdG9yLCBNb2RlbCB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIjtcbmltcG9ydCB7IHNlcXVlbmNlTmFtZUZvck1vZGVsIH0gZnJvbSBcIi4uL2lkZW50aXR5L3V0aWxzXCI7XG5pbXBvcnQgeyBTZXF1ZW5jZU9wdGlvbnMgfSBmcm9tIFwiLi4vaW50ZXJmYWNlcy9TZXF1ZW5jZU9wdGlvbnNcIjtcbmltcG9ydCB7IEludGVybmFsRXJyb3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcbmltcG9ydCB7IExvZ2dlciwgTG9nZ2luZyB9IGZyb20gXCJAZGVjYWYtdHMvbG9nZ2luZ1wiO1xuXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgU2VxdWVuY2Uge1xuICBwcml2YXRlIGxvZ2dlciE6IExvZ2dlcjtcblxuICBwcm90ZWN0ZWQgZ2V0IGxvZygpIHtcbiAgICBpZiAoIXRoaXMubG9nZ2VyKSB0aGlzLmxvZ2dlciA9IExvZ2dpbmcuZm9yKHRoaXMgYXMgYW55KTtcbiAgICByZXR1cm4gdGhpcy5sb2dnZXI7XG4gIH1cblxuICBwcm90ZWN0ZWQgY29uc3RydWN0b3IocHJvdGVjdGVkIHJlYWRvbmx5IG9wdGlvbnM6IFNlcXVlbmNlT3B0aW9ucykge31cblxuICBhYnN0cmFjdCBuZXh0KCk6IFByb21pc2U8c3RyaW5nIHwgbnVtYmVyIHwgYmlnaW50PjtcbiAgYWJzdHJhY3QgY3VycmVudCgpOiBQcm9taXNlPHN0cmluZyB8IG51bWJlciB8IGJpZ2ludD47XG4gIGFic3RyYWN0IHJhbmdlKGNvdW50OiBudW1iZXIpOiBQcm9taXNlPChudW1iZXIgfCBzdHJpbmcgfCBiaWdpbnQpW10+O1xuXG4gIHN0YXRpYyBwazxNIGV4dGVuZHMgTW9kZWw+KG1vZGVsOiBNIHwgQ29uc3RydWN0b3I8TT4pIHtcbiAgICByZXR1cm4gc2VxdWVuY2VOYW1lRm9yTW9kZWwobW9kZWwsIFwicGtcIik7XG4gIH1cblxuICBzdGF0aWMgcGFyc2VWYWx1ZShcbiAgICB0eXBlOiBcIk51bWJlclwiIHwgXCJCaWdJbnRcIiB8IHVuZGVmaW5lZCxcbiAgICB2YWx1ZTogc3RyaW5nIHwgbnVtYmVyIHwgYmlnaW50XG4gICk6IHN0cmluZyB8IG51bWJlciB8IGJpZ2ludCB7XG4gICAgc3dpdGNoICh0eXBlKSB7XG4gICAgICBjYXNlIFwiTnVtYmVyXCI6XG4gICAgICAgIHJldHVybiB0eXBlb2YgdmFsdWUgPT09IFwic3RyaW5nXCJcbiAgICAgICAgICA/IHBhcnNlSW50KHZhbHVlKVxuICAgICAgICAgIDogdHlwZW9mIHZhbHVlID09PSBcIm51bWJlclwiXG4gICAgICAgICAgICA/IHZhbHVlXG4gICAgICAgICAgICA6IEJpZ0ludCh2YWx1ZSk7XG4gICAgICBjYXNlIFwiQmlnSW50XCI6XG4gICAgICAgIHJldHVybiBCaWdJbnQodmFsdWUpO1xuICAgICAgZGVmYXVsdDpcbiAgICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoXCJTaG91bGQgbmV2ZXIgaGFwcGVuXCIpO1xuICAgIH1cbiAgfVxufVxuIl19
104
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Sequence.js","sourceRoot":"","sources":["../../src/persistence/Sequence.ts"],"names":[],"mappings":";;;AACA,mDAAyD;AAEzD,2DAAwD;AACxD,+CAAoD;AAEpD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6CG;AACH,MAAsB,QAAQ;IAO5B;;;;OAIG;IACH,IAAc,GAAG;QACf,IAAI,CAAC,IAAI,CAAC,MAAM;YAAE,IAAI,CAAC,MAAM,GAAG,iBAAO,CAAC,GAAG,CAAC,IAAW,CAAC,CAAC;QACzD,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED;;;OAGG;IACH,YAAyC,OAAwB;QAAxB,YAAO,GAAP,OAAO,CAAiB;IAAG,CAAC;IAwBrE;;;;;;OAMG;IACH,MAAM,CAAC,EAAE,CAAkB,KAAyB;QAClD,OAAO,IAAA,4BAAoB,EAAC,KAAK,EAAE,IAAI,CAAC,CAAC;IAC3C,CAAC;IAED;;;;;;OAMG;IACH,MAAM,CAAC,UAAU,CACf,IAAqC,EACrC,KAA+B;QAE/B,QAAQ,IAAI,EAAE,CAAC;YACb,KAAK,QAAQ;gBACX,OAAO,OAAO,KAAK,KAAK,QAAQ;oBAC9B,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC;oBACjB,CAAC,CAAC,OAAO,KAAK,KAAK,QAAQ;wBACzB,CAAC,CAAC,KAAK;wBACP,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YACtB,KAAK,QAAQ;gBACX,OAAO,MAAM,CAAC,KAAK,CAAC,CAAC;YACvB;gBACE,MAAM,IAAI,6BAAa,CAAC,qBAAqB,CAAC,CAAC;QACnD,CAAC;IACH,CAAC;CACF;AAhFD,4BAgFC","sourcesContent":["import { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport { sequenceNameForModel } from \"../identity/utils\";\nimport { SequenceOptions } from \"../interfaces/SequenceOptions\";\nimport { InternalError } from \"@decaf-ts/db-decorators\";\nimport { Logger, Logging } from \"@decaf-ts/logging\";\n\n/**\n * @description Abstract base class for sequence generation\n * @summary Provides a framework for generating sequential values (like primary keys) in the persistence layer.\n * Implementations of this class handle the specifics of how sequences are stored and incremented in different\n * database systems.\n * @param {SequenceOptions} options - Configuration options for the sequence generator\n * @class Sequence\n * @example\n * ```typescript\n * // Example implementation for a specific database\n * class PostgresSequence extends Sequence {\n *   constructor(options: SequenceOptions) {\n *     super(options);\n *   }\n *\n *   async next(): Promise<number> {\n *     // Implementation to get next value from PostgreSQL sequence\n *     const result = await this.options.executor.raw(`SELECT nextval('${this.options.name}')`);\n *     return parseInt(result.rows[0].nextval);\n *   }\n *\n *   async current(): Promise<number> {\n *     // Implementation to get current value from PostgreSQL sequence\n *     const result = await this.options.executor.raw(`SELECT currval('${this.options.name}')`);\n *     return parseInt(result.rows[0].currval);\n *   }\n *\n *   async range(count: number): Promise<number[]> {\n *     // Implementation to get a range of values\n *     const values: number[] = [];\n *     for (let i = 0; i < count; i++) {\n *       values.push(await this.next());\n *     }\n *     return values;\n *   }\n * }\n *\n * // Usage\n * const sequence = new PostgresSequence({\n *   name: 'user_id_seq',\n *   executor: dbExecutor\n * });\n *\n * const nextId = await sequence.next();\n * ```\n */\nexport abstract class Sequence {\n  /**\n   * @description Logger instance for this sequence\n   * @summary Lazily initialized logger for the sequence instance\n   */\n  private logger!: Logger;\n\n  /**\n   * @description Accessor for the logger instance\n   * @summary Gets or initializes the logger for this sequence\n   * @return {Logger} The logger instance\n   */\n  protected get log() {\n    if (!this.logger) this.logger = Logging.for(this as any);\n    return this.logger;\n  }\n\n  /**\n   * @description Creates a new sequence instance\n   * @summary Protected constructor that initializes the sequence with the provided options\n   */\n  protected constructor(protected readonly options: SequenceOptions) {}\n\n  /**\n   * @description Gets the next value in the sequence\n   * @summary Retrieves the next value from the sequence, incrementing it in the process\n   * @return A promise that resolves to the next value in the sequence\n   */\n  abstract next(): Promise<string | number | bigint>;\n\n  /**\n   * @description Gets the current value of the sequence\n   * @summary Retrieves the current value of the sequence without incrementing it\n   * @return A promise that resolves to the current value in the sequence\n   */\n  abstract current(): Promise<string | number | bigint>;\n\n  /**\n   * @description Gets a range of sequential values\n   * @summary Retrieves multiple sequential values at once, which can be more efficient than calling next() multiple times\n   * @param {number} count - The number of sequential values to retrieve\n   * @return A promise that resolves to an array of sequential values\n   */\n  abstract range(count: number): Promise<(number | string | bigint)[]>;\n\n  /**\n   * @description Gets the primary key sequence name for a model\n   * @summary Utility method that returns the standardized sequence name for a model's primary key\n   * @template M - The model type\n   * @param {M|Constructor<M>} model - The model instance or constructor\n   * @return {string} The sequence name for the model's primary key\n   */\n  static pk<M extends Model>(model: M | Constructor<M>) {\n    return sequenceNameForModel(model, \"pk\");\n  }\n\n  /**\n   * @description Parses a sequence value to the appropriate type\n   * @summary Converts a sequence value to the specified type (Number or BigInt)\n   * @param {\"Number\"|\"BigInt\"|undefined} type - The target type to convert to\n   * @param {string|number|bigint} value - The value to convert\n   * @return {string|number|bigint} The converted value\n   */\n  static parseValue(\n    type: \"Number\" | \"BigInt\" | undefined,\n    value: string | number | bigint\n  ): string | number | bigint {\n    switch (type) {\n      case \"Number\":\n        return typeof value === \"string\"\n          ? parseInt(value)\n          : typeof value === \"number\"\n            ? value\n            : BigInt(value);\n      case \"BigInt\":\n        return BigInt(value);\n      default:\n        throw new InternalError(\"Should never happen\");\n    }\n  }\n}\n"]}
@@ -1,14 +1,103 @@
1
1
  import { Constructor, Model } from "@decaf-ts/decorator-validation";
2
2
  import { SequenceOptions } from "../interfaces/SequenceOptions";
3
3
  import { Logger } from "@decaf-ts/logging";
4
+ /**
5
+ * @description Abstract base class for sequence generation
6
+ * @summary Provides a framework for generating sequential values (like primary keys) in the persistence layer.
7
+ * Implementations of this class handle the specifics of how sequences are stored and incremented in different
8
+ * database systems.
9
+ * @param {SequenceOptions} options - Configuration options for the sequence generator
10
+ * @class Sequence
11
+ * @example
12
+ * ```typescript
13
+ * // Example implementation for a specific database
14
+ * class PostgresSequence extends Sequence {
15
+ * constructor(options: SequenceOptions) {
16
+ * super(options);
17
+ * }
18
+ *
19
+ * async next(): Promise<number> {
20
+ * // Implementation to get next value from PostgreSQL sequence
21
+ * const result = await this.options.executor.raw(`SELECT nextval('${this.options.name}')`);
22
+ * return parseInt(result.rows[0].nextval);
23
+ * }
24
+ *
25
+ * async current(): Promise<number> {
26
+ * // Implementation to get current value from PostgreSQL sequence
27
+ * const result = await this.options.executor.raw(`SELECT currval('${this.options.name}')`);
28
+ * return parseInt(result.rows[0].currval);
29
+ * }
30
+ *
31
+ * async range(count: number): Promise<number[]> {
32
+ * // Implementation to get a range of values
33
+ * const values: number[] = [];
34
+ * for (let i = 0; i < count; i++) {
35
+ * values.push(await this.next());
36
+ * }
37
+ * return values;
38
+ * }
39
+ * }
40
+ *
41
+ * // Usage
42
+ * const sequence = new PostgresSequence({
43
+ * name: 'user_id_seq',
44
+ * executor: dbExecutor
45
+ * });
46
+ *
47
+ * const nextId = await sequence.next();
48
+ * ```
49
+ */
4
50
  export declare abstract class Sequence {
5
51
  protected readonly options: SequenceOptions;
52
+ /**
53
+ * @description Logger instance for this sequence
54
+ * @summary Lazily initialized logger for the sequence instance
55
+ */
6
56
  private logger;
57
+ /**
58
+ * @description Accessor for the logger instance
59
+ * @summary Gets or initializes the logger for this sequence
60
+ * @return {Logger} The logger instance
61
+ */
7
62
  protected get log(): Logger;
63
+ /**
64
+ * @description Creates a new sequence instance
65
+ * @summary Protected constructor that initializes the sequence with the provided options
66
+ */
8
67
  protected constructor(options: SequenceOptions);
68
+ /**
69
+ * @description Gets the next value in the sequence
70
+ * @summary Retrieves the next value from the sequence, incrementing it in the process
71
+ * @return A promise that resolves to the next value in the sequence
72
+ */
9
73
  abstract next(): Promise<string | number | bigint>;
74
+ /**
75
+ * @description Gets the current value of the sequence
76
+ * @summary Retrieves the current value of the sequence without incrementing it
77
+ * @return A promise that resolves to the current value in the sequence
78
+ */
10
79
  abstract current(): Promise<string | number | bigint>;
80
+ /**
81
+ * @description Gets a range of sequential values
82
+ * @summary Retrieves multiple sequential values at once, which can be more efficient than calling next() multiple times
83
+ * @param {number} count - The number of sequential values to retrieve
84
+ * @return A promise that resolves to an array of sequential values
85
+ */
11
86
  abstract range(count: number): Promise<(number | string | bigint)[]>;
87
+ /**
88
+ * @description Gets the primary key sequence name for a model
89
+ * @summary Utility method that returns the standardized sequence name for a model's primary key
90
+ * @template M - The model type
91
+ * @param {M|Constructor<M>} model - The model instance or constructor
92
+ * @return {string} The sequence name for the model's primary key
93
+ */
12
94
  static pk<M extends Model>(model: M | Constructor<M>): string;
95
+ /**
96
+ * @description Parses a sequence value to the appropriate type
97
+ * @summary Converts a sequence value to the specified type (Number or BigInt)
98
+ * @param {"Number"|"BigInt"|undefined} type - The target type to convert to
99
+ * @param {string|number|bigint} value - The value to convert
100
+ * @return {string|number|bigint} The converted value
101
+ */
13
102
  static parseValue(type: "Number" | "BigInt" | undefined, value: string | number | bigint): string | number | bigint;
14
103
  }