@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.
Files changed (330) hide show
  1. package/LICENSE.md +21 -157
  2. package/README.md +652 -15
  3. package/dist/core.cjs +2638 -1085
  4. package/dist/core.esm.cjs +2636 -1070
  5. package/lib/esm/identity/decorators.d.ts +52 -14
  6. package/lib/esm/identity/decorators.js +54 -16
  7. package/lib/esm/identity/utils.d.ts +20 -1
  8. package/lib/esm/identity/utils.js +22 -2
  9. package/lib/esm/index.d.ts +11 -15
  10. package/lib/esm/index.js +17 -19
  11. package/lib/esm/interfaces/ErrorParser.d.ts +12 -0
  12. package/lib/esm/interfaces/ErrorParser.js +1 -1
  13. package/lib/esm/interfaces/Executor.d.ts +10 -13
  14. package/lib/esm/interfaces/Executor.js +1 -1
  15. package/lib/esm/interfaces/Observable.d.ts +20 -18
  16. package/lib/esm/interfaces/Observable.js +1 -1
  17. package/lib/esm/interfaces/Observer.d.ts +7 -8
  18. package/lib/esm/interfaces/Observer.js +1 -1
  19. package/lib/esm/interfaces/Paginatable.d.ts +18 -2
  20. package/lib/esm/interfaces/Paginatable.js +1 -1
  21. package/lib/esm/interfaces/Queriable.d.ts +44 -3
  22. package/lib/esm/interfaces/Queriable.js +1 -1
  23. package/lib/esm/interfaces/RawExecutor.d.ts +10 -13
  24. package/lib/esm/interfaces/RawExecutor.js +1 -1
  25. package/lib/esm/interfaces/SequenceOptions.d.ts +50 -5
  26. package/lib/esm/interfaces/SequenceOptions.js +19 -1
  27. package/lib/esm/interfaces/index.d.ts +0 -1
  28. package/lib/esm/interfaces/index.js +1 -2
  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 +442 -9
  32. package/lib/esm/model/construction.js +441 -2
  33. package/lib/esm/model/decorators.d.ts +166 -42
  34. package/lib/esm/model/decorators.js +161 -37
  35. package/lib/esm/model/index.js +1 -2
  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 +384 -40
  39. package/lib/esm/persistence/Adapter.js +415 -59
  40. package/lib/esm/persistence/Dispatch.d.ts +131 -0
  41. package/lib/esm/persistence/Dispatch.js +187 -0
  42. package/lib/esm/persistence/ObserverHandler.d.ts +109 -0
  43. package/lib/esm/persistence/ObserverHandler.js +137 -0
  44. package/lib/esm/persistence/Sequence.d.ts +89 -8
  45. package/lib/esm/persistence/Sequence.js +91 -1
  46. package/lib/esm/persistence/constants.d.ts +22 -5
  47. package/lib/esm/persistence/constants.js +23 -7
  48. package/lib/esm/persistence/decorators.d.ts +10 -0
  49. package/lib/esm/persistence/decorators.js +11 -1
  50. package/lib/esm/persistence/errors.d.ts +23 -3
  51. package/lib/esm/persistence/errors.js +25 -7
  52. package/lib/esm/persistence/index.d.ts +3 -0
  53. package/lib/esm/persistence/index.js +4 -1
  54. package/lib/esm/persistence/types.d.ts +21 -0
  55. package/lib/esm/persistence/types.js +2 -0
  56. package/lib/esm/query/Condition.d.ts +88 -44
  57. package/lib/esm/query/Condition.js +144 -62
  58. package/lib/esm/query/Paginator.d.ts +67 -10
  59. package/lib/esm/query/Paginator.js +64 -10
  60. package/lib/esm/query/Statement.d.ts +82 -47
  61. package/lib/esm/query/Statement.js +175 -122
  62. package/lib/esm/query/constants.d.ts +25 -64
  63. package/lib/esm/query/constants.js +26 -68
  64. package/lib/esm/query/errors.d.ts +14 -0
  65. package/lib/esm/query/errors.js +15 -1
  66. package/lib/esm/query/index.d.ts +0 -5
  67. package/lib/esm/query/index.js +1 -6
  68. package/lib/esm/query/options.d.ts +69 -178
  69. package/lib/esm/query/options.js +1 -1
  70. package/lib/esm/query/selectors.d.ts +20 -24
  71. package/lib/esm/query/selectors.js +1 -1
  72. package/lib/esm/ram/RamAdapter.d.ts +322 -20
  73. package/lib/esm/ram/RamAdapter.js +360 -140
  74. package/lib/esm/ram/RamContext.d.ts +16 -1
  75. package/lib/esm/ram/RamContext.js +18 -3
  76. package/lib/esm/ram/RamPaginator.d.ts +51 -6
  77. package/lib/esm/ram/RamPaginator.js +58 -6
  78. package/lib/esm/ram/RamSequence.d.ts +49 -24
  79. package/lib/esm/ram/RamSequence.js +52 -40
  80. package/lib/esm/ram/RamStatement.d.ts +84 -6
  81. package/lib/esm/ram/RamStatement.js +175 -6
  82. package/lib/esm/ram/constants.d.ts +9 -0
  83. package/lib/esm/ram/constants.js +10 -0
  84. package/lib/esm/ram/handlers.d.ts +25 -0
  85. package/lib/esm/ram/handlers.js +27 -0
  86. package/lib/esm/ram/index.d.ts +4 -4
  87. package/lib/esm/ram/index.js +9 -5
  88. package/lib/esm/ram/model/RamSequence.d.ts +21 -9
  89. package/lib/esm/ram/model/RamSequence.js +19 -1
  90. package/lib/esm/ram/types.d.ts +47 -5
  91. package/lib/esm/ram/types.js +1 -1
  92. package/lib/esm/repository/Repository.d.ts +381 -22
  93. package/lib/esm/repository/Repository.js +446 -43
  94. package/lib/esm/repository/constants.d.ts +23 -13
  95. package/lib/esm/repository/constants.js +24 -14
  96. package/lib/esm/repository/decorators.d.ts +27 -0
  97. package/lib/esm/repository/decorators.js +28 -1
  98. package/lib/esm/repository/errors.d.ts +12 -11
  99. package/lib/esm/repository/errors.js +13 -16
  100. package/lib/esm/repository/injectables.d.ts +18 -0
  101. package/lib/esm/repository/injectables.js +19 -1
  102. package/lib/esm/repository/types.d.ts +13 -1
  103. package/lib/esm/repository/types.js +1 -1
  104. package/lib/esm/repository/utils.d.ts +11 -0
  105. package/lib/esm/repository/utils.js +12 -1
  106. package/lib/esm/utils/decorators.d.ts +9 -0
  107. package/lib/esm/utils/decorators.js +19 -0
  108. package/lib/esm/utils/errors.d.ts +56 -0
  109. package/lib/esm/utils/errors.js +63 -0
  110. package/lib/esm/utils/index.d.ts +2 -0
  111. package/lib/esm/utils/index.js +3 -0
  112. package/lib/identity/decorators.cjs +54 -16
  113. package/lib/identity/decorators.d.ts +52 -14
  114. package/lib/identity/utils.cjs +22 -2
  115. package/lib/identity/utils.d.ts +20 -1
  116. package/lib/index.cjs +17 -19
  117. package/lib/index.d.ts +11 -15
  118. package/lib/interfaces/ErrorParser.cjs +1 -1
  119. package/lib/interfaces/ErrorParser.d.ts +12 -0
  120. package/lib/interfaces/Executor.cjs +1 -1
  121. package/lib/interfaces/Executor.d.ts +10 -13
  122. package/lib/interfaces/Observable.cjs +1 -1
  123. package/lib/interfaces/Observable.d.ts +20 -18
  124. package/lib/interfaces/Observer.cjs +1 -1
  125. package/lib/interfaces/Observer.d.ts +7 -8
  126. package/lib/interfaces/Paginatable.cjs +1 -1
  127. package/lib/interfaces/Paginatable.d.ts +18 -2
  128. package/lib/interfaces/Queriable.cjs +1 -1
  129. package/lib/interfaces/Queriable.d.ts +44 -3
  130. package/lib/interfaces/RawExecutor.cjs +1 -1
  131. package/lib/interfaces/RawExecutor.d.ts +10 -13
  132. package/lib/interfaces/SequenceOptions.cjs +19 -1
  133. package/lib/interfaces/SequenceOptions.d.ts +50 -5
  134. package/lib/interfaces/index.cjs +1 -2
  135. package/lib/interfaces/index.d.ts +0 -1
  136. package/lib/model/BaseModel.cjs +24 -1
  137. package/lib/model/BaseModel.d.ts +31 -0
  138. package/lib/model/construction.cjs +441 -2
  139. package/lib/model/construction.d.ts +442 -9
  140. package/lib/model/decorators.cjs +161 -37
  141. package/lib/model/decorators.d.ts +166 -42
  142. package/lib/model/index.cjs +1 -2
  143. package/lib/model/types.cjs +1 -1
  144. package/lib/model/types.d.ts +9 -0
  145. package/lib/persistence/Adapter.cjs +414 -58
  146. package/lib/persistence/Adapter.d.ts +384 -40
  147. package/lib/persistence/Dispatch.cjs +191 -0
  148. package/lib/persistence/Dispatch.d.ts +131 -0
  149. package/lib/persistence/ObserverHandler.cjs +141 -0
  150. package/lib/persistence/ObserverHandler.d.ts +109 -0
  151. package/lib/persistence/Sequence.cjs +91 -1
  152. package/lib/persistence/Sequence.d.ts +89 -8
  153. package/lib/persistence/constants.cjs +24 -8
  154. package/lib/persistence/constants.d.ts +22 -5
  155. package/lib/persistence/decorators.cjs +11 -1
  156. package/lib/persistence/decorators.d.ts +10 -0
  157. package/lib/persistence/errors.cjs +26 -9
  158. package/lib/persistence/errors.d.ts +23 -3
  159. package/lib/persistence/index.cjs +4 -1
  160. package/lib/persistence/index.d.ts +3 -0
  161. package/lib/persistence/types.cjs +3 -0
  162. package/lib/persistence/types.d.ts +21 -0
  163. package/lib/query/Condition.cjs +143 -61
  164. package/lib/query/Condition.d.ts +88 -44
  165. package/lib/query/Paginator.cjs +64 -10
  166. package/lib/query/Paginator.d.ts +67 -10
  167. package/lib/query/Statement.cjs +174 -121
  168. package/lib/query/Statement.d.ts +82 -47
  169. package/lib/query/constants.cjs +27 -69
  170. package/lib/query/constants.d.ts +25 -64
  171. package/lib/query/errors.cjs +15 -1
  172. package/lib/query/errors.d.ts +14 -0
  173. package/lib/query/index.cjs +1 -6
  174. package/lib/query/index.d.ts +0 -5
  175. package/lib/query/options.cjs +1 -1
  176. package/lib/query/options.d.ts +69 -178
  177. package/lib/query/selectors.cjs +1 -1
  178. package/lib/query/selectors.d.ts +20 -24
  179. package/lib/ram/RamAdapter.cjs +358 -172
  180. package/lib/ram/RamAdapter.d.ts +322 -20
  181. package/lib/ram/RamContext.cjs +18 -3
  182. package/lib/ram/RamContext.d.ts +16 -1
  183. package/lib/ram/RamPaginator.cjs +58 -6
  184. package/lib/ram/RamPaginator.d.ts +51 -6
  185. package/lib/ram/RamSequence.cjs +52 -41
  186. package/lib/ram/RamSequence.d.ts +49 -24
  187. package/lib/ram/RamStatement.cjs +175 -6
  188. package/lib/ram/RamStatement.d.ts +84 -6
  189. package/lib/ram/constants.cjs +13 -0
  190. package/lib/ram/constants.d.ts +9 -0
  191. package/lib/ram/handlers.cjs +30 -0
  192. package/lib/ram/handlers.d.ts +25 -0
  193. package/lib/ram/index.cjs +9 -5
  194. package/lib/ram/index.d.ts +4 -4
  195. package/lib/ram/model/RamSequence.cjs +19 -1
  196. package/lib/ram/model/RamSequence.d.ts +21 -9
  197. package/lib/ram/types.cjs +1 -1
  198. package/lib/ram/types.d.ts +47 -5
  199. package/lib/repository/Repository.cjs +445 -42
  200. package/lib/repository/Repository.d.ts +381 -22
  201. package/lib/repository/constants.cjs +24 -14
  202. package/lib/repository/constants.d.ts +23 -13
  203. package/lib/repository/decorators.cjs +28 -1
  204. package/lib/repository/decorators.d.ts +27 -0
  205. package/lib/repository/errors.cjs +14 -19
  206. package/lib/repository/errors.d.ts +12 -11
  207. package/lib/repository/injectables.cjs +19 -1
  208. package/lib/repository/injectables.d.ts +18 -0
  209. package/lib/repository/types.cjs +1 -1
  210. package/lib/repository/types.d.ts +13 -1
  211. package/lib/repository/utils.cjs +12 -1
  212. package/lib/repository/utils.d.ts +11 -0
  213. package/lib/utils/decorators.cjs +22 -0
  214. package/lib/utils/decorators.d.ts +9 -0
  215. package/lib/utils/errors.cjs +69 -0
  216. package/lib/utils/errors.d.ts +56 -0
  217. package/lib/{validators → utils}/index.cjs +2 -2
  218. package/lib/utils/index.d.ts +2 -0
  219. package/package.json +5 -5
  220. package/lib/esm/interfaces/Builder.d.ts +0 -16
  221. package/lib/esm/interfaces/Builder.js +0 -2
  222. package/lib/esm/model/IdentifiedBaseModel.d.ts +0 -7
  223. package/lib/esm/model/IdentifiedBaseModel.js +0 -25
  224. package/lib/esm/query/Clause.d.ts +0 -50
  225. package/lib/esm/query/Clause.js +0 -82
  226. package/lib/esm/query/ClauseFactory.d.ts +0 -71
  227. package/lib/esm/query/ClauseFactory.js +0 -6
  228. package/lib/esm/query/Query.d.ts +0 -43
  229. package/lib/esm/query/Query.js +0 -54
  230. package/lib/esm/query/clauses/FromClause.d.ts +0 -45
  231. package/lib/esm/query/clauses/FromClause.js +0 -59
  232. package/lib/esm/query/clauses/GroupByClause.d.ts +0 -21
  233. package/lib/esm/query/clauses/GroupByClause.js +0 -19
  234. package/lib/esm/query/clauses/InsertClause.d.ts +0 -37
  235. package/lib/esm/query/clauses/InsertClause.js +0 -55
  236. package/lib/esm/query/clauses/LimitClause.d.ts +0 -29
  237. package/lib/esm/query/clauses/LimitClause.js +0 -27
  238. package/lib/esm/query/clauses/OffsetClause.d.ts +0 -21
  239. package/lib/esm/query/clauses/OffsetClause.js +0 -19
  240. package/lib/esm/query/clauses/OrderByClause.d.ts +0 -37
  241. package/lib/esm/query/clauses/OrderByClause.js +0 -39
  242. package/lib/esm/query/clauses/SelectClause.d.ts +0 -47
  243. package/lib/esm/query/clauses/SelectClause.js +0 -62
  244. package/lib/esm/query/clauses/SelectorBasedClause.d.ts +0 -25
  245. package/lib/esm/query/clauses/SelectorBasedClause.js +0 -44
  246. package/lib/esm/query/clauses/ValuesClause.d.ts +0 -21
  247. package/lib/esm/query/clauses/ValuesClause.js +0 -36
  248. package/lib/esm/query/clauses/WhereClause.d.ts +0 -46
  249. package/lib/esm/query/clauses/WhereClause.js +0 -71
  250. package/lib/esm/query/clauses/index.d.ts +0 -10
  251. package/lib/esm/query/clauses/index.js +0 -11
  252. package/lib/esm/query/types.d.ts +0 -2
  253. package/lib/esm/query/types.js +0 -2
  254. package/lib/esm/ram/RamClauseFactory.d.ts +0 -17
  255. package/lib/esm/ram/RamClauseFactory.js +0 -92
  256. package/lib/esm/ram/clauses/FromClause.d.ts +0 -7
  257. package/lib/esm/ram/clauses/FromClause.js +0 -11
  258. package/lib/esm/ram/clauses/InsertClause.d.ts +0 -7
  259. package/lib/esm/ram/clauses/InsertClause.js +0 -13
  260. package/lib/esm/ram/clauses/OrderByClause.d.ts +0 -7
  261. package/lib/esm/ram/clauses/OrderByClause.js +0 -39
  262. package/lib/esm/ram/clauses/SelectClause.d.ts +0 -7
  263. package/lib/esm/ram/clauses/SelectClause.js +0 -16
  264. package/lib/esm/ram/clauses/ValuesClause.d.ts +0 -7
  265. package/lib/esm/ram/clauses/ValuesClause.js +0 -12
  266. package/lib/esm/ram/clauses/WhereClause.d.ts +0 -7
  267. package/lib/esm/ram/clauses/WhereClause.js +0 -11
  268. package/lib/esm/ram/clauses/index.d.ts +0 -6
  269. package/lib/esm/ram/clauses/index.js +0 -7
  270. package/lib/esm/validators/ClauseSequenceValidator.d.ts +0 -28
  271. package/lib/esm/validators/ClauseSequenceValidator.js +0 -95
  272. package/lib/esm/validators/decorators.d.ts +0 -10
  273. package/lib/esm/validators/decorators.js +0 -24
  274. package/lib/esm/validators/index.d.ts +0 -2
  275. package/lib/esm/validators/index.js +0 -3
  276. package/lib/interfaces/Builder.cjs +0 -3
  277. package/lib/interfaces/Builder.d.ts +0 -16
  278. package/lib/model/IdentifiedBaseModel.cjs +0 -29
  279. package/lib/model/IdentifiedBaseModel.d.ts +0 -7
  280. package/lib/query/Clause.cjs +0 -86
  281. package/lib/query/Clause.d.ts +0 -50
  282. package/lib/query/ClauseFactory.cjs +0 -10
  283. package/lib/query/ClauseFactory.d.ts +0 -71
  284. package/lib/query/Query.cjs +0 -58
  285. package/lib/query/Query.d.ts +0 -43
  286. package/lib/query/clauses/FromClause.cjs +0 -63
  287. package/lib/query/clauses/FromClause.d.ts +0 -45
  288. package/lib/query/clauses/GroupByClause.cjs +0 -23
  289. package/lib/query/clauses/GroupByClause.d.ts +0 -21
  290. package/lib/query/clauses/InsertClause.cjs +0 -59
  291. package/lib/query/clauses/InsertClause.d.ts +0 -37
  292. package/lib/query/clauses/LimitClause.cjs +0 -31
  293. package/lib/query/clauses/LimitClause.d.ts +0 -29
  294. package/lib/query/clauses/OffsetClause.cjs +0 -23
  295. package/lib/query/clauses/OffsetClause.d.ts +0 -21
  296. package/lib/query/clauses/OrderByClause.cjs +0 -43
  297. package/lib/query/clauses/OrderByClause.d.ts +0 -37
  298. package/lib/query/clauses/SelectClause.cjs +0 -66
  299. package/lib/query/clauses/SelectClause.d.ts +0 -47
  300. package/lib/query/clauses/SelectorBasedClause.cjs +0 -48
  301. package/lib/query/clauses/SelectorBasedClause.d.ts +0 -25
  302. package/lib/query/clauses/ValuesClause.cjs +0 -40
  303. package/lib/query/clauses/ValuesClause.d.ts +0 -21
  304. package/lib/query/clauses/WhereClause.cjs +0 -75
  305. package/lib/query/clauses/WhereClause.d.ts +0 -46
  306. package/lib/query/clauses/index.cjs +0 -27
  307. package/lib/query/clauses/index.d.ts +0 -10
  308. package/lib/query/types.cjs +0 -3
  309. package/lib/query/types.d.ts +0 -2
  310. package/lib/ram/RamClauseFactory.cjs +0 -96
  311. package/lib/ram/RamClauseFactory.d.ts +0 -17
  312. package/lib/ram/clauses/FromClause.cjs +0 -15
  313. package/lib/ram/clauses/FromClause.d.ts +0 -7
  314. package/lib/ram/clauses/InsertClause.cjs +0 -17
  315. package/lib/ram/clauses/InsertClause.d.ts +0 -7
  316. package/lib/ram/clauses/OrderByClause.cjs +0 -43
  317. package/lib/ram/clauses/OrderByClause.d.ts +0 -7
  318. package/lib/ram/clauses/SelectClause.cjs +0 -20
  319. package/lib/ram/clauses/SelectClause.d.ts +0 -7
  320. package/lib/ram/clauses/ValuesClause.cjs +0 -16
  321. package/lib/ram/clauses/ValuesClause.d.ts +0 -7
  322. package/lib/ram/clauses/WhereClause.cjs +0 -15
  323. package/lib/ram/clauses/WhereClause.d.ts +0 -7
  324. package/lib/ram/clauses/index.cjs +0 -23
  325. package/lib/ram/clauses/index.d.ts +0 -6
  326. package/lib/validators/ClauseSequenceValidator.cjs +0 -98
  327. package/lib/validators/ClauseSequenceValidator.d.ts +0 -28
  328. package/lib/validators/decorators.cjs +0 -27
  329. package/lib/validators/decorators.d.ts +0 -10
  330. package/lib/validators/index.d.ts +0 -2
@@ -0,0 +1,191 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Dispatch = void 0;
4
+ const db_decorators_1 = require("@decaf-ts/db-decorators");
5
+ const Adapter_1 = require("./Adapter.cjs");
6
+ const errors_1 = require("./errors.cjs");
7
+ const logging_1 = require("@decaf-ts/logging");
8
+ /**
9
+ * @description Dispatches database operation events to observers
10
+ * @summary The Dispatch class implements the Observable interface and is responsible for intercepting
11
+ * database operations from an Adapter and notifying observers when changes occur. It uses proxies to
12
+ * wrap the adapter's CRUD methods and automatically trigger observer updates after operations complete.
13
+ * @template Y - The native database driver type
14
+ * @param {void} - No constructor parameters
15
+ * @class Dispatch
16
+ * @example
17
+ * ```typescript
18
+ * // Creating and using a Dispatch instance
19
+ * const dispatch = new Dispatch<PostgresDriver>();
20
+ *
21
+ * // Connect it to an adapter
22
+ * const adapter = new PostgresAdapter(connection);
23
+ * dispatch.observe(adapter);
24
+ *
25
+ * // Now any CRUD operations on the adapter will automatically
26
+ * // trigger observer notifications
27
+ * await adapter.create('users', 123, userModel);
28
+ * // Observers will be notified about the creation
29
+ *
30
+ * // When done, you can disconnect
31
+ * dispatch.unObserve(adapter);
32
+ * ```
33
+ */
34
+ class Dispatch {
35
+ /**
36
+ * @description Accessor for the logger
37
+ * @summary Gets or initializes the logger for this dispatch instance
38
+ * @return {Logger} The logger instance
39
+ */
40
+ get log() {
41
+ if (!this.logger)
42
+ this.logger = logging_1.Logging.for(this).for(this.adapter);
43
+ return this.logger;
44
+ }
45
+ /**
46
+ * @description Creates a new Dispatch instance
47
+ * @summary Initializes a new Dispatch instance without any adapter
48
+ */
49
+ constructor() { }
50
+ /**
51
+ * @description Initializes the dispatch by proxying adapter methods
52
+ * @summary Sets up proxies on the adapter's CRUD methods to intercept operations and notify observers.
53
+ * This method is called automatically when an adapter is observed.
54
+ * @return {Promise<void>} A promise that resolves when initialization is complete
55
+ * @mermaid
56
+ * sequenceDiagram
57
+ * participant Dispatch
58
+ * participant Adapter
59
+ * participant Proxy
60
+ *
61
+ * Dispatch->>Dispatch: initialize()
62
+ * Dispatch->>Dispatch: Check if adapter exists
63
+ * alt No adapter
64
+ * Dispatch-->>Dispatch: Throw InternalError
65
+ * end
66
+ *
67
+ * loop For each CRUD method
68
+ * Dispatch->>Adapter: Check if method exists
69
+ * alt Method doesn't exist
70
+ * Dispatch-->>Dispatch: Throw InternalError
71
+ * end
72
+ *
73
+ * Dispatch->>Adapter: Get property descriptor
74
+ * loop While descriptor not found
75
+ * Dispatch->>Adapter: Check prototype chain
76
+ * end
77
+ *
78
+ * alt Descriptor not found or not writable
79
+ * Dispatch->>Dispatch: Log error and continue
80
+ * else Descriptor found and writable
81
+ * Dispatch->>Proxy: Create proxy for method
82
+ * Dispatch->>Adapter: Replace method with proxy
83
+ * end
84
+ * end
85
+ */
86
+ async initialize() {
87
+ if (!this.adapter)
88
+ throw new db_decorators_1.InternalError(`No adapter observed for dispatch`);
89
+ const adapter = this.adapter;
90
+ [
91
+ db_decorators_1.OperationKeys.CREATE,
92
+ db_decorators_1.OperationKeys.UPDATE,
93
+ db_decorators_1.OperationKeys.DELETE,
94
+ db_decorators_1.BulkCrudOperationKeys.CREATE_ALL,
95
+ db_decorators_1.BulkCrudOperationKeys.UPDATE_ALL,
96
+ db_decorators_1.BulkCrudOperationKeys.DELETE_ALL,
97
+ ].forEach((method) => {
98
+ if (!adapter[method])
99
+ throw new db_decorators_1.InternalError(`Method ${method} not found in ${adapter.alias} adapter to bind Observables Dispatch`);
100
+ let descriptor = Object.getOwnPropertyDescriptor(adapter, method);
101
+ let proto = adapter;
102
+ while (!descriptor && proto !== Object.prototype) {
103
+ proto = Object.getPrototypeOf(proto);
104
+ descriptor = Object.getOwnPropertyDescriptor(proto, method);
105
+ }
106
+ if (!descriptor || !descriptor.writable) {
107
+ this.log.error(`Could not find method ${method} to bind Observables Dispatch`);
108
+ return;
109
+ }
110
+ function bulkToSingle(method) {
111
+ switch (method) {
112
+ case db_decorators_1.BulkCrudOperationKeys.CREATE_ALL:
113
+ return db_decorators_1.OperationKeys.CREATE;
114
+ case db_decorators_1.BulkCrudOperationKeys.UPDATE_ALL:
115
+ return db_decorators_1.OperationKeys.UPDATE;
116
+ case db_decorators_1.BulkCrudOperationKeys.DELETE_ALL:
117
+ return db_decorators_1.OperationKeys.DELETE;
118
+ default:
119
+ return method;
120
+ }
121
+ }
122
+ // @ts-expect-error because there are read only properties
123
+ adapter[method] = new Proxy(adapter[method], {
124
+ apply: async (target, thisArg, argArray) => {
125
+ const [tableName, ids] = argArray;
126
+ const result = await target.apply(thisArg, argArray);
127
+ this.updateObservers(tableName, bulkToSingle(method), ids)
128
+ .then(() => {
129
+ this.log.verbose(`Observer refresh dispatched by ${method} for ${tableName}`);
130
+ this.log.debug(`pks: ${ids}`);
131
+ })
132
+ .catch((e) => this.log.error(`Failed to dispatch observer refresh for ${method} on ${tableName}: ${e}`));
133
+ return result;
134
+ },
135
+ });
136
+ });
137
+ }
138
+ /**
139
+ * @description Closes the dispatch
140
+ * @summary Performs any necessary cleanup when the dispatch is no longer needed
141
+ * @return {Promise<void>} A promise that resolves when closing is complete
142
+ */
143
+ async close() {
144
+ // to nothing in this instance but may be required for closing connections
145
+ }
146
+ /**
147
+ * @description Starts observing an adapter
148
+ * @summary Connects this dispatch to an adapter to monitor its operations
149
+ * @param {Adapter<Y, any, any, any>} observer - The adapter to observe
150
+ * @return {void}
151
+ */
152
+ observe(observer) {
153
+ if (!(observer instanceof Adapter_1.Adapter))
154
+ throw new errors_1.UnsupportedError("Only Adapters can be observed by dispatch");
155
+ this.adapter = observer;
156
+ this.native = observer.native;
157
+ this.models = Adapter_1.Adapter.models(this.adapter.alias);
158
+ this.initialize().then(() => this.log.verbose(`Dispatch initialized for ${this.adapter.alias} adapter`));
159
+ }
160
+ /**
161
+ * @description Stops observing an adapter
162
+ * @summary Disconnects this dispatch from an adapter
163
+ * @param {Observer} observer - The adapter to stop observing
164
+ * @return {void}
165
+ */
166
+ unObserve(observer) {
167
+ if (this.adapter !== observer)
168
+ throw new errors_1.UnsupportedError("Only the adapter that was used to observe can be unobserved");
169
+ this.adapter = undefined;
170
+ }
171
+ /**
172
+ * @description Updates observers about a database event
173
+ * @summary Notifies observers about a change in the database
174
+ * @param {string} table - The name of the table where the change occurred
175
+ * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred
176
+ * @param {EventIds} id - The identifier(s) of the affected record(s)
177
+ * @return {Promise<void>} A promise that resolves when all observers have been notified
178
+ */
179
+ async updateObservers(table, event, id) {
180
+ if (!this.adapter)
181
+ throw new db_decorators_1.InternalError(`No adapter observed for dispatch`);
182
+ try {
183
+ await this.adapter.refresh(table, event, id);
184
+ }
185
+ catch (e) {
186
+ throw new db_decorators_1.InternalError(`Failed to refresh dispatch: ${e}`);
187
+ }
188
+ }
189
+ }
190
+ exports.Dispatch = Dispatch;
191
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Dispatch.js","sourceRoot":"","sources":["../../src/persistence/Dispatch.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  InternalError,\n  OperationKeys,\n  BulkCrudOperationKeys,\n} from \"@decaf-ts/db-decorators\";\nimport { ModelConstructor } from \"@decaf-ts/decorator-validation\";\nimport { Observable, Observer } from \"../interfaces\";\nimport { Adapter } from \"./Adapter\";\nimport { UnsupportedError } from \"./errors\";\nimport { Logger, Logging } from \"@decaf-ts/logging\";\nimport { EventIds } from \"./types\";\n\n/**\n * @description Dispatches database operation events to observers\n * @summary The Dispatch class implements the Observable interface and is responsible for intercepting\n * database operations from an Adapter and notifying observers when changes occur. It uses proxies to\n * wrap the adapter's CRUD methods and automatically trigger observer updates after operations complete.\n * @template Y - The native database driver type\n * @param {void} - No constructor parameters\n * @class Dispatch\n * @example\n * ```typescript\n * // Creating and using a Dispatch instance\n * const dispatch = new Dispatch<PostgresDriver>();\n *\n * // Connect it to an adapter\n * const adapter = new PostgresAdapter(connection);\n * dispatch.observe(adapter);\n *\n * // Now any CRUD operations on the adapter will automatically\n * // trigger observer notifications\n * await adapter.create('users', 123, userModel);\n * // Observers will be notified about the creation\n *\n * // When done, you can disconnect\n * dispatch.unObserve(adapter);\n * ```\n */\nexport class Dispatch<Y> implements Observable {\n  /**\n   * @description The adapter being observed\n   * @summary Reference to the database adapter whose operations are being monitored\n   */\n  protected adapter?: Adapter<Y, any, any, any>;\n\n  /**\n   * @description The native database driver\n   * @summary Reference to the underlying database driver from the adapter\n   */\n  protected native?: Y;\n\n  /**\n   * @description List of model constructors\n   * @summary Array of model constructors that are registered with the adapter\n   */\n  protected models!: ModelConstructor<any>[];\n\n  /**\n   * @description Logger instance\n   * @summary Logger for recording dispatch activities\n   */\n  private logger!: Logger;\n\n  /**\n   * @description Accessor for the logger\n   * @summary Gets or initializes the logger for this dispatch instance\n   * @return {Logger} The logger instance\n   */\n  protected get log() {\n    if (!this.logger)\n      this.logger = Logging.for(this as any).for(this.adapter as any);\n    return this.logger;\n  }\n\n  /**\n   * @description Creates a new Dispatch instance\n   * @summary Initializes a new Dispatch instance without any adapter\n   */\n  constructor() {}\n\n  /**\n   * @description Initializes the dispatch by proxying adapter methods\n   * @summary Sets up proxies on the adapter's CRUD methods to intercept operations and notify observers.\n   * This method is called automatically when an adapter is observed.\n   * @return {Promise<void>} A promise that resolves when initialization is complete\n   * @mermaid\n   * sequenceDiagram\n   *   participant Dispatch\n   *   participant Adapter\n   *   participant Proxy\n   *\n   *   Dispatch->>Dispatch: initialize()\n   *   Dispatch->>Dispatch: Check if adapter exists\n   *   alt No adapter\n   *     Dispatch-->>Dispatch: Throw InternalError\n   *   end\n   *\n   *   loop For each CRUD method\n   *     Dispatch->>Adapter: Check if method exists\n   *     alt Method doesn't exist\n   *       Dispatch-->>Dispatch: Throw InternalError\n   *     end\n   *\n   *     Dispatch->>Adapter: Get property descriptor\n   *     loop While descriptor not found\n   *       Dispatch->>Adapter: Check prototype chain\n   *     end\n   *\n   *     alt Descriptor not found or not writable\n   *       Dispatch->>Dispatch: Log error and continue\n   *     else Descriptor found and writable\n   *       Dispatch->>Proxy: Create proxy for method\n   *       Dispatch->>Adapter: Replace method with proxy\n   *     end\n   *   end\n   */\n  protected async initialize(): Promise<void> {\n    if (!this.adapter)\n      throw new InternalError(`No adapter observed for dispatch`);\n    const adapter = this.adapter as Adapter<Y, any, any, any>;\n    (\n      [\n        OperationKeys.CREATE,\n        OperationKeys.UPDATE,\n        OperationKeys.DELETE,\n        BulkCrudOperationKeys.CREATE_ALL,\n        BulkCrudOperationKeys.UPDATE_ALL,\n        BulkCrudOperationKeys.DELETE_ALL,\n      ] as (keyof Adapter<Y, any, any, any>)[]\n    ).forEach((method) => {\n      if (!adapter[method])\n        throw new InternalError(\n          `Method ${method} not found in ${adapter.alias} adapter to bind Observables Dispatch`\n        );\n\n      let descriptor = Object.getOwnPropertyDescriptor(adapter, method);\n      let proto: any = adapter;\n      while (!descriptor && proto !== Object.prototype) {\n        proto = Object.getPrototypeOf(proto);\n        descriptor = Object.getOwnPropertyDescriptor(proto, method);\n      }\n\n      if (!descriptor || !descriptor.writable) {\n        this.log.error(\n          `Could not find method ${method} to bind Observables Dispatch`\n        );\n        return;\n      }\n      function bulkToSingle(method: string) {\n        switch (method) {\n          case BulkCrudOperationKeys.CREATE_ALL:\n            return OperationKeys.CREATE;\n          case BulkCrudOperationKeys.UPDATE_ALL:\n            return OperationKeys.UPDATE;\n          case BulkCrudOperationKeys.DELETE_ALL:\n            return OperationKeys.DELETE;\n          default:\n            return method;\n        }\n      }\n      // @ts-expect-error because there are read only properties\n      adapter[method] = new Proxy(adapter[method], {\n        apply: async (target: any, thisArg, argArray: any[]) => {\n          const [tableName, ids] = argArray;\n          const result = await target.apply(thisArg, argArray);\n          this.updateObservers(tableName, bulkToSingle(method), ids as EventIds)\n            .then(() => {\n              this.log.verbose(\n                `Observer refresh dispatched by ${method} for ${tableName}`\n              );\n              this.log.debug(`pks: ${ids}`);\n            })\n            .catch((e: unknown) =>\n              this.log.error(\n                `Failed to dispatch observer refresh for ${method} on ${tableName}: ${e}`\n              )\n            );\n          return result;\n        },\n      });\n    });\n  }\n\n  /**\n   * @description Closes the dispatch\n   * @summary Performs any necessary cleanup when the dispatch is no longer needed\n   * @return {Promise<void>} A promise that resolves when closing is complete\n   */\n  async close() {\n    // to nothing in this instance but may be required for closing connections\n  }\n\n  /**\n   * @description Starts observing an adapter\n   * @summary Connects this dispatch to an adapter to monitor its operations\n   * @param {Adapter<Y, any, any, any>} observer - The adapter to observe\n   * @return {void}\n   */\n  observe(observer: Adapter<Y, any, any, any>): void {\n    if (!(observer instanceof Adapter))\n      throw new UnsupportedError(\"Only Adapters can be observed by dispatch\");\n    this.adapter = observer;\n    this.native = observer.native;\n    this.models = Adapter.models(this.adapter.alias);\n    this.initialize().then(() =>\n      this.log.verbose(\n        `Dispatch initialized for ${this.adapter!.alias} adapter`\n      )\n    );\n  }\n\n  /**\n   * @description Stops observing an adapter\n   * @summary Disconnects this dispatch from an adapter\n   * @param {Observer} observer - The adapter to stop observing\n   * @return {void}\n   */\n  unObserve(observer: Observer): void {\n    if (this.adapter !== observer)\n      throw new UnsupportedError(\n        \"Only the adapter that was used to observe can be unobserved\"\n      );\n    this.adapter = undefined;\n  }\n\n  /**\n   * @description Updates observers about a database event\n   * @summary Notifies observers about a change in the database\n   * @param {string} table - The name of the table where the change occurred\n   * @param {OperationKeys|BulkCrudOperationKeys|string} event - The type of operation that occurred\n   * @param {EventIds} id - The identifier(s) of the affected record(s)\n   * @return {Promise<void>} A promise that resolves when all observers have been notified\n   */\n  async updateObservers(\n    table: string,\n    event: OperationKeys | BulkCrudOperationKeys | string,\n    id: EventIds\n  ): Promise<void> {\n    if (!this.adapter)\n      throw new InternalError(`No adapter observed for dispatch`);\n    try {\n      await this.adapter.refresh(table, event, id);\n    } catch (e: unknown) {\n      throw new InternalError(`Failed to refresh dispatch: ${e}`);\n    }\n  }\n}\n"]}
@@ -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,141 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ObserverHandler = void 0;
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
+ */
34
+ class ObserverHandler {
35
+ constructor() {
36
+ /**
37
+ * @description Collection of registered observers
38
+ * @summary Array of observer objects along with their optional filters
39
+ */
40
+ this.observers = [];
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
+ */
47
+ count() {
48
+ return this.observers.length;
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
+ */
57
+ observe(observer, filter) {
58
+ const index = this.observers.map((o) => o.observer).indexOf(observer);
59
+ if (index !== -1)
60
+ throw new db_decorators_1.InternalError("Observer already registered");
61
+ this.observers.push({ observer: observer, filter: filter });
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
+ */
69
+ unObserve(observer) {
70
+ const index = this.observers.map((o) => o.observer).indexOf(observer);
71
+ if (index === -1)
72
+ throw new db_decorators_1.InternalError("Failed to find Observer");
73
+ this.observers.splice(index, 1);
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
+ */
119
+ async updateObservers(log, table, event, id, ...args) {
120
+ const results = await Promise.allSettled(this.observers
121
+ .filter((o) => {
122
+ const { filter } = o;
123
+ if (!filter)
124
+ return true;
125
+ try {
126
+ return filter(table, event, id);
127
+ }
128
+ catch (e) {
129
+ log.error(`Failed to filter observer ${o.observer.toString()}: ${e}`);
130
+ return false;
131
+ }
132
+ })
133
+ .map((o) => o.observer.refresh(table, event, id, ...args)));
134
+ results.forEach((result, i) => {
135
+ if (result.status === "rejected")
136
+ log.error(`Failed to update observable ${this.observers[i].toString()}: ${result.reason}`);
137
+ });
138
+ }
139
+ }
140
+ exports.ObserverHandler = ObserverHandler;
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"]}
@@ -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
+ }