zibri 2.4.0 → 2.4.1

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 (517) hide show
  1. package/dist/cjs/application.d.ts.map +1 -1
  2. package/dist/cjs/application.js +8 -2
  3. package/dist/cjs/application.js.map +1 -1
  4. package/dist/cjs/auth/encryption/encryption-key.model.d.ts +5 -1
  5. package/dist/cjs/auth/encryption/encryption-key.model.d.ts.map +1 -1
  6. package/dist/cjs/auth/encryption/encryption-key.model.js +9 -1
  7. package/dist/cjs/auth/encryption/encryption-key.model.js.map +1 -1
  8. package/dist/cjs/auth/strategies/cookie/cookie-auth.auth-strategy.d.ts.map +1 -1
  9. package/dist/cjs/auth/strategies/cookie/cookie-auth.auth-strategy.js.map +1 -1
  10. package/dist/cjs/auth/strategies/jwt/jwt-credentials.model.d.ts +5 -1
  11. package/dist/cjs/auth/strategies/jwt/jwt-credentials.model.d.ts.map +1 -1
  12. package/dist/cjs/auth/strategies/jwt/jwt-credentials.model.js +9 -1
  13. package/dist/cjs/auth/strategies/jwt/jwt-credentials.model.js.map +1 -1
  14. package/dist/cjs/auth/strategies/jwt/jwt.auth-strategy.d.ts.map +1 -1
  15. package/dist/cjs/auth/strategies/jwt/jwt.auth-strategy.js +2 -1
  16. package/dist/cjs/auth/strategies/jwt/jwt.auth-strategy.js.map +1 -1
  17. package/dist/cjs/backup/backup-resource-entity.model.d.ts +5 -1
  18. package/dist/cjs/backup/backup-resource-entity.model.d.ts.map +1 -1
  19. package/dist/cjs/backup/backup-resource-entity.model.js +9 -1
  20. package/dist/cjs/backup/backup-resource-entity.model.js.map +1 -1
  21. package/dist/cjs/backup/backup.service.js +2 -2
  22. package/dist/cjs/backup/backup.service.js.map +1 -1
  23. package/dist/cjs/caching/cache/base-cache.model.js.map +1 -1
  24. package/dist/cjs/caching/cache/multi-tier.cache.js.map +1 -1
  25. package/dist/cjs/caching/cache/read-aside/read-aside.cache.js.map +1 -1
  26. package/dist/cjs/caching/cache/read-aside/write-around-read-aside.cache.js.map +1 -1
  27. package/dist/cjs/caching/cache/read-aside/write-behind-read-aside.cache.js.map +1 -1
  28. package/dist/cjs/caching/cache/read-aside/write-invalidate-read-aside-args-only.cache.js.map +1 -1
  29. package/dist/cjs/caching/cache/read-aside/write-invalidate-read-aside-with-result.cache.js.map +1 -1
  30. package/dist/cjs/caching/cache/read-aside/write-through-read-aside.cache.js.map +1 -1
  31. package/dist/cjs/caching/cache/read-through/read-through.cache.js.map +1 -1
  32. package/dist/cjs/caching/cache/read-through/write-around-read-through.cache.js.map +1 -1
  33. package/dist/cjs/caching/cache/read-through/write-behind-read-through.cache.js.map +1 -1
  34. package/dist/cjs/caching/cache/read-through/write-invalidate-read-through-args-only.cache.js.map +1 -1
  35. package/dist/cjs/caching/cache/read-through/write-invalidate-read-through-with-result.cache.js.map +1 -1
  36. package/dist/cjs/caching/cache/read-through/write-through-read-through.cache.js.map +1 -1
  37. package/dist/cjs/change-sets/change-set-repository.d.ts +5 -3
  38. package/dist/cjs/change-sets/change-set-repository.d.ts.map +1 -1
  39. package/dist/cjs/change-sets/change-set-repository.js +16 -16
  40. package/dist/cjs/change-sets/change-set-repository.js.map +1 -1
  41. package/dist/cjs/change-sets/models/change-set-entity.model.d.ts +2 -2
  42. package/dist/cjs/change-sets/models/change-set-entity.model.d.ts.map +1 -1
  43. package/dist/cjs/change-sets/models/change-set-entity.model.js +29 -1
  44. package/dist/cjs/change-sets/models/change-set-entity.model.js.map +1 -1
  45. package/dist/cjs/change-sets/models/change-set.model.js +1 -1
  46. package/dist/cjs/change-sets/models/change-set.model.js.map +1 -1
  47. package/dist/cjs/change-sets/models/change.model.d.ts +5 -1
  48. package/dist/cjs/change-sets/models/change.model.d.ts.map +1 -1
  49. package/dist/cjs/change-sets/models/change.model.js +9 -1
  50. package/dist/cjs/change-sets/models/change.model.js.map +1 -1
  51. package/dist/cjs/change-sets/models/soft-delete-entity.model.d.ts +2 -2
  52. package/dist/cjs/change-sets/models/soft-delete-entity.model.d.ts.map +1 -1
  53. package/dist/cjs/change-sets/models/soft-delete-entity.model.js +26 -0
  54. package/dist/cjs/change-sets/models/soft-delete-entity.model.js.map +1 -1
  55. package/dist/cjs/change-sets/soft-delete-repository.d.ts +6 -5
  56. package/dist/cjs/change-sets/soft-delete-repository.d.ts.map +1 -1
  57. package/dist/cjs/change-sets/soft-delete-repository.js +22 -6
  58. package/dist/cjs/change-sets/soft-delete-repository.js.map +1 -1
  59. package/dist/cjs/context/als.utilities.d.ts +3 -3
  60. package/dist/cjs/context/als.utilities.d.ts.map +1 -1
  61. package/dist/cjs/context/als.utilities.js.map +1 -1
  62. package/dist/cjs/context/cache/cache.context.d.ts +27 -0
  63. package/dist/cjs/context/cache/cache.context.d.ts.map +1 -0
  64. package/dist/cjs/context/cache/cache.context.js +61 -0
  65. package/dist/cjs/context/cache/cache.context.js.map +1 -0
  66. package/dist/cjs/data-source/data-sources/data-source-initialization.error.d.ts +7 -0
  67. package/dist/cjs/data-source/data-sources/data-source-initialization.error.d.ts.map +1 -0
  68. package/dist/cjs/data-source/data-sources/data-source-initialization.error.js +14 -0
  69. package/dist/cjs/data-source/data-sources/data-source-initialization.error.js.map +1 -0
  70. package/dist/cjs/data-source/data-sources/data-source.interface.d.ts +9 -1
  71. package/dist/cjs/data-source/data-sources/data-source.interface.d.ts.map +1 -1
  72. package/dist/cjs/data-source/data-sources/data-source.interface.js.map +1 -1
  73. package/dist/cjs/data-source/data-sources/postgres-typeorm-data-source.model.d.ts +44 -0
  74. package/dist/cjs/data-source/data-sources/postgres-typeorm-data-source.model.d.ts.map +1 -0
  75. package/dist/cjs/data-source/data-sources/postgres-typeorm-data-source.model.js +289 -0
  76. package/dist/cjs/data-source/data-sources/postgres-typeorm-data-source.model.js.map +1 -0
  77. package/dist/cjs/data-source/data-sources/sql-data-source.interface.d.ts +32 -0
  78. package/dist/cjs/data-source/data-sources/sql-data-source.interface.d.ts.map +1 -0
  79. package/dist/cjs/{entity/models/one-to-one-property-metadata.model.js → data-source/data-sources/sql-data-source.interface.js} +1 -1
  80. package/dist/cjs/data-source/data-sources/sql-data-source.interface.js.map +1 -0
  81. package/dist/cjs/data-source/data-sources/typeorm-base-data-source.model.d.ts +156 -0
  82. package/dist/cjs/data-source/data-sources/typeorm-base-data-source.model.d.ts.map +1 -0
  83. package/dist/cjs/data-source/data-sources/typeorm-base-data-source.model.js +378 -0
  84. package/dist/cjs/data-source/data-sources/typeorm-base-data-source.model.js.map +1 -0
  85. package/dist/cjs/data-source/data-sources/where-converter/postgres-typeorm-where-filter.converter.d.ts +25 -0
  86. package/dist/cjs/data-source/data-sources/where-converter/postgres-typeorm-where-filter.converter.d.ts.map +1 -0
  87. package/dist/cjs/data-source/data-sources/where-converter/postgres-typeorm-where-filter.converter.js +353 -0
  88. package/dist/cjs/data-source/data-sources/where-converter/postgres-typeorm-where-filter.converter.js.map +1 -0
  89. package/dist/cjs/data-source/data-sources/where-converter/typeorm-where-filter.converter.d.ts +70 -0
  90. package/dist/cjs/data-source/data-sources/where-converter/typeorm-where-filter.converter.d.ts.map +1 -0
  91. package/dist/cjs/data-source/data-sources/where-converter/typeorm-where-filter.converter.js +248 -0
  92. package/dist/cjs/data-source/data-sources/where-converter/typeorm-where-filter.converter.js.map +1 -0
  93. package/dist/cjs/data-source/models/options/count-options.model.d.ts +2 -7
  94. package/dist/cjs/data-source/models/options/count-options.model.d.ts.map +1 -1
  95. package/dist/cjs/data-source/models/options/delete-all-options.model.d.ts +2 -4
  96. package/dist/cjs/data-source/models/options/delete-all-options.model.d.ts.map +1 -1
  97. package/dist/cjs/data-source/models/options/find-all-options.model.d.ts +2 -2
  98. package/dist/cjs/data-source/models/options/find-all-options.model.d.ts.map +1 -1
  99. package/dist/cjs/data-source/models/options/find-by-id-options.model.d.ts +1 -6
  100. package/dist/cjs/data-source/models/options/find-by-id-options.model.d.ts.map +1 -1
  101. package/dist/cjs/data-source/models/options/find-one-options.model.d.ts +2 -11
  102. package/dist/cjs/data-source/models/options/find-one-options.model.d.ts.map +1 -1
  103. package/dist/cjs/data-source/models/options/update-all-options.model.d.ts +1 -7
  104. package/dist/cjs/data-source/models/options/update-all-options.model.d.ts.map +1 -1
  105. package/dist/cjs/data-source/models/options/update-by-id-options.model.d.ts +1 -7
  106. package/dist/cjs/data-source/models/options/update-by-id-options.model.d.ts.map +1 -1
  107. package/dist/cjs/data-source/models/where/array-where-filter.model.d.ts +142 -3
  108. package/dist/cjs/data-source/models/where/array-where-filter.model.d.ts.map +1 -1
  109. package/dist/cjs/data-source/models/where/base-where-filter.model.d.ts +18 -1
  110. package/dist/cjs/data-source/models/where/base-where-filter.model.d.ts.map +1 -1
  111. package/dist/cjs/data-source/models/where/boolean-where-filter.model.d.ts +4 -2
  112. package/dist/cjs/data-source/models/where/boolean-where-filter.model.d.ts.map +1 -1
  113. package/dist/cjs/data-source/models/where/date-where-filter.model.d.ts +8 -15
  114. package/dist/cjs/data-source/models/where/date-where-filter.model.d.ts.map +1 -1
  115. package/dist/cjs/data-source/models/where/number-where-filter.model.d.ts +8 -15
  116. package/dist/cjs/data-source/models/where/number-where-filter.model.d.ts.map +1 -1
  117. package/dist/cjs/data-source/models/where/object-where-filter.model.d.ts +52 -7
  118. package/dist/cjs/data-source/models/where/object-where-filter.model.d.ts.map +1 -1
  119. package/dist/cjs/data-source/models/where/string-where-filter.model.d.ts +8 -15
  120. package/dist/cjs/data-source/models/where/string-where-filter.model.d.ts.map +1 -1
  121. package/dist/cjs/data-source/models/where/where-filter-keys.model.d.ts +46 -0
  122. package/dist/cjs/data-source/models/where/where-filter-keys.model.d.ts.map +1 -0
  123. package/dist/cjs/data-source/models/where/where-filter-keys.model.js +36 -0
  124. package/dist/cjs/data-source/models/where/where-filter-keys.model.js.map +1 -0
  125. package/dist/cjs/data-source/models/where/where-filter.model.d.ts +2 -2
  126. package/dist/cjs/data-source/models/where/where-filter.model.d.ts.map +1 -1
  127. package/dist/cjs/data-source/query-failed.error.js +27 -1
  128. package/dist/cjs/data-source/query-failed.error.js.map +1 -1
  129. package/dist/cjs/data-source/repository.d.ts +8 -3
  130. package/dist/cjs/data-source/repository.d.ts.map +1 -1
  131. package/dist/cjs/data-source/repository.js +42 -16
  132. package/dist/cjs/data-source/repository.js.map +1 -1
  133. package/dist/cjs/di/decorators/inject-repository.decorator.d.ts +2 -2
  134. package/dist/cjs/di/decorators/inject-repository.decorator.d.ts.map +1 -1
  135. package/dist/cjs/di/decorators/inject-repository.decorator.js.map +1 -1
  136. package/dist/cjs/di/default/zibri-di-tokens.default.d.ts +2 -2
  137. package/dist/cjs/di/default/zibri-di-tokens.default.d.ts.map +1 -1
  138. package/dist/cjs/di/default/zibri-di-tokens.default.js.map +1 -1
  139. package/dist/cjs/email/email.service.d.ts +2 -2
  140. package/dist/cjs/email/email.service.d.ts.map +1 -1
  141. package/dist/cjs/email/email.service.js +4 -6
  142. package/dist/cjs/email/email.service.js.map +1 -1
  143. package/dist/cjs/email/models/email.model.js +1 -1
  144. package/dist/cjs/email/models/email.model.js.map +1 -1
  145. package/dist/cjs/entity/decorators/entity.decorator.d.ts +28 -1
  146. package/dist/cjs/entity/decorators/entity.decorator.d.ts.map +1 -1
  147. package/dist/cjs/entity/decorators/entity.decorator.js +6 -3
  148. package/dist/cjs/entity/decorators/entity.decorator.js.map +1 -1
  149. package/dist/cjs/entity/decorators/property.decorator.d.ts +5 -8
  150. package/dist/cjs/entity/decorators/property.decorator.d.ts.map +1 -1
  151. package/dist/cjs/entity/decorators/property.decorator.js +34 -20
  152. package/dist/cjs/entity/decorators/property.decorator.js.map +1 -1
  153. package/dist/cjs/entity/entity-metadata-missing.error.d.ts +9 -0
  154. package/dist/cjs/entity/entity-metadata-missing.error.d.ts.map +1 -0
  155. package/dist/cjs/entity/entity-metadata-missing.error.js +17 -0
  156. package/dist/cjs/entity/entity-metadata-missing.error.js.map +1 -0
  157. package/dist/cjs/entity/models/base-relation-metadata.model.d.ts +1 -1
  158. package/dist/cjs/entity/models/belongs-to-one-property-metadata.model.d.ts +28 -0
  159. package/dist/cjs/entity/models/belongs-to-one-property-metadata.model.d.ts.map +1 -0
  160. package/dist/cjs/entity/models/belongs-to-one-property-metadata.model.js +3 -0
  161. package/dist/cjs/entity/models/belongs-to-one-property-metadata.model.js.map +1 -0
  162. package/dist/cjs/entity/models/has-one-property-metadata.model.d.ts +18 -0
  163. package/dist/cjs/entity/models/has-one-property-metadata.model.d.ts.map +1 -0
  164. package/dist/{esm/entity/models/one-to-one-property-metadata.model.js → cjs/entity/models/has-one-property-metadata.model.js} +1 -1
  165. package/dist/cjs/entity/models/has-one-property-metadata.model.js.map +1 -0
  166. package/dist/cjs/entity/models/many-to-many-property-metadata.model.d.ts +3 -3
  167. package/dist/cjs/entity/models/many-to-many-property-metadata.model.d.ts.map +1 -1
  168. package/dist/cjs/entity/models/many-to-one-property-metadata.model.d.ts +11 -1
  169. package/dist/cjs/entity/models/many-to-one-property-metadata.model.d.ts.map +1 -1
  170. package/dist/cjs/entity/models/relation.enum.d.ts +2 -1
  171. package/dist/cjs/entity/models/relation.enum.d.ts.map +1 -1
  172. package/dist/cjs/entity/models/relation.enum.js +2 -1
  173. package/dist/cjs/entity/models/relation.enum.js.map +1 -1
  174. package/dist/cjs/entity/partial-class.model.js +1 -1
  175. package/dist/cjs/entity/partial-class.model.js.map +1 -1
  176. package/dist/cjs/event/event-cleanup.cron-job.d.ts.map +1 -1
  177. package/dist/cjs/event/event-cleanup.cron-job.js +4 -6
  178. package/dist/cjs/event/event-cleanup.cron-job.js.map +1 -1
  179. package/dist/cjs/event/event-subscriber-run.model.d.ts +6 -8
  180. package/dist/cjs/event/event-subscriber-run.model.d.ts.map +1 -1
  181. package/dist/cjs/event/event-subscriber-run.model.js +10 -17
  182. package/dist/cjs/event/event-subscriber-run.model.js.map +1 -1
  183. package/dist/cjs/event/event.model.js +1 -1
  184. package/dist/cjs/event/event.model.js.map +1 -1
  185. package/dist/cjs/event/event.service.d.ts.map +1 -1
  186. package/dist/cjs/event/event.service.js +6 -3
  187. package/dist/cjs/event/event.service.js.map +1 -1
  188. package/dist/cjs/global/model-registry/default-descriptor.d.ts.map +1 -1
  189. package/dist/cjs/global/model-registry/default-descriptor.js +2 -1
  190. package/dist/cjs/global/model-registry/default-descriptor.js.map +1 -1
  191. package/dist/cjs/global/model-registry/encryption-descriptor.d.ts.map +1 -1
  192. package/dist/cjs/global/model-registry/encryption-descriptor.js +2 -1
  193. package/dist/cjs/global/model-registry/encryption-descriptor.js.map +1 -1
  194. package/dist/cjs/global/model-registry/exclude-descriptor.d.ts.map +1 -1
  195. package/dist/cjs/global/model-registry/exclude-descriptor.js +2 -1
  196. package/dist/cjs/global/model-registry/exclude-descriptor.js.map +1 -1
  197. package/dist/cjs/global/model-registry/hash-descriptor.d.ts.map +1 -1
  198. package/dist/cjs/global/model-registry/hash-descriptor.js +2 -1
  199. package/dist/cjs/global/model-registry/hash-descriptor.js.map +1 -1
  200. package/dist/cjs/index.d.ts +12 -3
  201. package/dist/cjs/index.d.ts.map +1 -1
  202. package/dist/cjs/index.js +12 -3
  203. package/dist/cjs/index.js.map +1 -1
  204. package/dist/cjs/logging/log-context.model.d.ts +2 -27
  205. package/dist/cjs/logging/log-context.model.d.ts.map +1 -1
  206. package/dist/cjs/logging/log-context.model.js +3 -49
  207. package/dist/cjs/logging/log-context.model.js.map +1 -1
  208. package/dist/cjs/open-api/open-api.service.d.ts.map +1 -1
  209. package/dist/cjs/open-api/open-api.service.js +20 -11
  210. package/dist/cjs/open-api/open-api.service.js.map +1 -1
  211. package/dist/cjs/parsing/form-data/form-data.body-parser.d.ts.map +1 -1
  212. package/dist/cjs/parsing/form-data/form-data.body-parser.js +2 -1
  213. package/dist/cjs/parsing/form-data/form-data.body-parser.js.map +1 -1
  214. package/dist/cjs/parsing/functions/parse-boolean.function.d.ts.map +1 -1
  215. package/dist/cjs/parsing/functions/parse-boolean.function.js.map +1 -1
  216. package/dist/cjs/parsing/functions/parse-object.function.d.ts.map +1 -1
  217. package/dist/cjs/parsing/functions/parse-object.function.js +2 -1
  218. package/dist/cjs/parsing/functions/parse-object.function.js.map +1 -1
  219. package/dist/cjs/parsing/parser.d.ts.map +1 -1
  220. package/dist/cjs/parsing/parser.js +2 -1
  221. package/dist/cjs/parsing/parser.js.map +1 -1
  222. package/dist/cjs/routing/decorators/body.decorator.js +2 -1
  223. package/dist/cjs/routing/decorators/body.decorator.js.map +1 -1
  224. package/dist/cjs/routing/resolve-route-params.function.js +1 -1
  225. package/dist/cjs/routing/resolve-route-params.function.js.map +1 -1
  226. package/dist/cjs/routing/router.d.ts.map +1 -1
  227. package/dist/cjs/routing/router.js +38 -2
  228. package/dist/cjs/routing/router.js.map +1 -1
  229. package/dist/cjs/utilities/metadata-injection-keys.enum.d.ts +1 -0
  230. package/dist/cjs/utilities/metadata-injection-keys.enum.d.ts.map +1 -1
  231. package/dist/cjs/utilities/metadata-injection-keys.enum.js +1 -0
  232. package/dist/cjs/utilities/metadata-injection-keys.enum.js.map +1 -1
  233. package/dist/cjs/utilities/typeorm.utilities.d.ts +39 -0
  234. package/dist/cjs/utilities/typeorm.utilities.d.ts.map +1 -0
  235. package/dist/cjs/utilities/typeorm.utilities.js +47 -0
  236. package/dist/cjs/utilities/typeorm.utilities.js.map +1 -0
  237. package/dist/cjs/validation/validation-problem.model.d.ts.map +1 -1
  238. package/dist/cjs/validation/validation-problem.model.js +4 -7
  239. package/dist/cjs/validation/validation-problem.model.js.map +1 -1
  240. package/dist/cjs/validation/validation.service.d.ts.map +1 -1
  241. package/dist/cjs/validation/validation.service.js +6 -4
  242. package/dist/cjs/validation/validation.service.js.map +1 -1
  243. package/dist/cjs/websocket/models/websocket-message.model.js +1 -1
  244. package/dist/cjs/websocket/models/websocket-message.model.js.map +1 -1
  245. package/dist/cjs/websocket/models/websocket-request.model.d.ts +5 -11
  246. package/dist/cjs/websocket/models/websocket-request.model.d.ts.map +1 -1
  247. package/dist/cjs/websocket/models/websocket-request.model.js.map +1 -1
  248. package/dist/cjs/websocket/services/websocket.service.d.ts.map +1 -1
  249. package/dist/cjs/websocket/services/websocket.service.js +1 -2
  250. package/dist/cjs/websocket/services/websocket.service.js.map +1 -1
  251. package/dist/esm/application.js +8 -2
  252. package/dist/esm/application.js.map +1 -1
  253. package/dist/esm/auth/encryption/encryption-key.model.js +9 -1
  254. package/dist/esm/auth/encryption/encryption-key.model.js.map +1 -1
  255. package/dist/esm/auth/strategies/cookie/cookie-auth.auth-strategy.js.map +1 -1
  256. package/dist/esm/auth/strategies/jwt/jwt-credentials.model.js +9 -1
  257. package/dist/esm/auth/strategies/jwt/jwt-credentials.model.js.map +1 -1
  258. package/dist/esm/auth/strategies/jwt/jwt.auth-strategy.js +2 -1
  259. package/dist/esm/auth/strategies/jwt/jwt.auth-strategy.js.map +1 -1
  260. package/dist/esm/backup/backup-resource-entity.model.js +9 -1
  261. package/dist/esm/backup/backup-resource-entity.model.js.map +1 -1
  262. package/dist/esm/backup/backup.service.js +2 -2
  263. package/dist/esm/backup/backup.service.js.map +1 -1
  264. package/dist/esm/caching/cache/base-cache.model.js.map +1 -1
  265. package/dist/esm/caching/cache/multi-tier.cache.js.map +1 -1
  266. package/dist/esm/caching/cache/read-aside/read-aside.cache.js.map +1 -1
  267. package/dist/esm/caching/cache/read-aside/write-around-read-aside.cache.js.map +1 -1
  268. package/dist/esm/caching/cache/read-aside/write-behind-read-aside.cache.js.map +1 -1
  269. package/dist/esm/caching/cache/read-aside/write-invalidate-read-aside-args-only.cache.js.map +1 -1
  270. package/dist/esm/caching/cache/read-aside/write-invalidate-read-aside-with-result.cache.js.map +1 -1
  271. package/dist/esm/caching/cache/read-aside/write-through-read-aside.cache.js.map +1 -1
  272. package/dist/esm/caching/cache/read-through/read-through.cache.js.map +1 -1
  273. package/dist/esm/caching/cache/read-through/write-around-read-through.cache.js.map +1 -1
  274. package/dist/esm/caching/cache/read-through/write-behind-read-through.cache.js.map +1 -1
  275. package/dist/esm/caching/cache/read-through/write-invalidate-read-through-args-only.cache.js.map +1 -1
  276. package/dist/esm/caching/cache/read-through/write-invalidate-read-through-with-result.cache.js.map +1 -1
  277. package/dist/esm/caching/cache/read-through/write-through-read-through.cache.js.map +1 -1
  278. package/dist/esm/change-sets/change-set-repository.js +16 -16
  279. package/dist/esm/change-sets/change-set-repository.js.map +1 -1
  280. package/dist/esm/change-sets/models/change-set-entity.model.js +29 -1
  281. package/dist/esm/change-sets/models/change-set-entity.model.js.map +1 -1
  282. package/dist/esm/change-sets/models/change-set.model.js +1 -1
  283. package/dist/esm/change-sets/models/change-set.model.js.map +1 -1
  284. package/dist/esm/change-sets/models/change.model.js +9 -1
  285. package/dist/esm/change-sets/models/change.model.js.map +1 -1
  286. package/dist/esm/change-sets/models/soft-delete-entity.model.js +26 -0
  287. package/dist/esm/change-sets/models/soft-delete-entity.model.js.map +1 -1
  288. package/dist/esm/change-sets/soft-delete-repository.js +22 -6
  289. package/dist/esm/change-sets/soft-delete-repository.js.map +1 -1
  290. package/dist/esm/context/als.utilities.js.map +1 -1
  291. package/dist/esm/context/cache/cache.context.js +61 -0
  292. package/dist/esm/context/cache/cache.context.js.map +1 -0
  293. package/dist/esm/data-source/data-sources/data-source-initialization.error.js +14 -0
  294. package/dist/esm/data-source/data-sources/data-source-initialization.error.js.map +1 -0
  295. package/dist/esm/data-source/data-sources/data-source.interface.js.map +1 -1
  296. package/dist/esm/data-source/data-sources/postgres-typeorm-data-source.model.js +289 -0
  297. package/dist/esm/data-source/data-sources/postgres-typeorm-data-source.model.js.map +1 -0
  298. package/dist/esm/data-source/data-sources/sql-data-source.interface.js +3 -0
  299. package/dist/esm/data-source/data-sources/sql-data-source.interface.js.map +1 -0
  300. package/dist/esm/data-source/data-sources/typeorm-base-data-source.model.js +378 -0
  301. package/dist/esm/data-source/data-sources/typeorm-base-data-source.model.js.map +1 -0
  302. package/dist/esm/data-source/data-sources/where-converter/postgres-typeorm-where-filter.converter.js +353 -0
  303. package/dist/esm/data-source/data-sources/where-converter/postgres-typeorm-where-filter.converter.js.map +1 -0
  304. package/dist/esm/data-source/data-sources/where-converter/typeorm-where-filter.converter.js +248 -0
  305. package/dist/esm/data-source/data-sources/where-converter/typeorm-where-filter.converter.js.map +1 -0
  306. package/dist/esm/data-source/models/where/where-filter-keys.model.js +36 -0
  307. package/dist/esm/data-source/models/where/where-filter-keys.model.js.map +1 -0
  308. package/dist/esm/data-source/query-failed.error.js +27 -1
  309. package/dist/esm/data-source/query-failed.error.js.map +1 -1
  310. package/dist/esm/data-source/repository.js +42 -16
  311. package/dist/esm/data-source/repository.js.map +1 -1
  312. package/dist/esm/di/decorators/inject-repository.decorator.js.map +1 -1
  313. package/dist/esm/di/default/zibri-di-tokens.default.js.map +1 -1
  314. package/dist/esm/email/email.service.js +4 -6
  315. package/dist/esm/email/email.service.js.map +1 -1
  316. package/dist/esm/email/models/email.model.js +1 -1
  317. package/dist/esm/email/models/email.model.js.map +1 -1
  318. package/dist/esm/entity/decorators/entity.decorator.js +6 -3
  319. package/dist/esm/entity/decorators/entity.decorator.js.map +1 -1
  320. package/dist/esm/entity/decorators/property.decorator.js +34 -20
  321. package/dist/esm/entity/decorators/property.decorator.js.map +1 -1
  322. package/dist/esm/entity/entity-metadata-missing.error.js +17 -0
  323. package/dist/esm/entity/entity-metadata-missing.error.js.map +1 -0
  324. package/dist/esm/entity/models/belongs-to-one-property-metadata.model.js +3 -0
  325. package/dist/esm/entity/models/belongs-to-one-property-metadata.model.js.map +1 -0
  326. package/dist/esm/entity/models/has-one-property-metadata.model.js +3 -0
  327. package/dist/esm/entity/models/has-one-property-metadata.model.js.map +1 -0
  328. package/dist/esm/entity/models/relation.enum.js +2 -1
  329. package/dist/esm/entity/models/relation.enum.js.map +1 -1
  330. package/dist/esm/entity/partial-class.model.js +1 -1
  331. package/dist/esm/entity/partial-class.model.js.map +1 -1
  332. package/dist/esm/event/event-cleanup.cron-job.js +4 -6
  333. package/dist/esm/event/event-cleanup.cron-job.js.map +1 -1
  334. package/dist/esm/event/event-subscriber-run.model.js +10 -17
  335. package/dist/esm/event/event-subscriber-run.model.js.map +1 -1
  336. package/dist/esm/event/event.model.js +1 -1
  337. package/dist/esm/event/event.model.js.map +1 -1
  338. package/dist/esm/event/event.service.js +6 -3
  339. package/dist/esm/event/event.service.js.map +1 -1
  340. package/dist/esm/global/model-registry/default-descriptor.js +2 -1
  341. package/dist/esm/global/model-registry/default-descriptor.js.map +1 -1
  342. package/dist/esm/global/model-registry/encryption-descriptor.js +2 -1
  343. package/dist/esm/global/model-registry/encryption-descriptor.js.map +1 -1
  344. package/dist/esm/global/model-registry/exclude-descriptor.js +2 -1
  345. package/dist/esm/global/model-registry/exclude-descriptor.js.map +1 -1
  346. package/dist/esm/global/model-registry/hash-descriptor.js +2 -1
  347. package/dist/esm/global/model-registry/hash-descriptor.js.map +1 -1
  348. package/dist/esm/index.js +12 -3
  349. package/dist/esm/index.js.map +1 -1
  350. package/dist/esm/logging/log-context.model.js +3 -49
  351. package/dist/esm/logging/log-context.model.js.map +1 -1
  352. package/dist/esm/open-api/open-api.service.js +20 -11
  353. package/dist/esm/open-api/open-api.service.js.map +1 -1
  354. package/dist/esm/parsing/form-data/form-data.body-parser.js +2 -1
  355. package/dist/esm/parsing/form-data/form-data.body-parser.js.map +1 -1
  356. package/dist/esm/parsing/functions/parse-boolean.function.js.map +1 -1
  357. package/dist/esm/parsing/functions/parse-object.function.js +2 -1
  358. package/dist/esm/parsing/functions/parse-object.function.js.map +1 -1
  359. package/dist/esm/parsing/parser.js +2 -1
  360. package/dist/esm/parsing/parser.js.map +1 -1
  361. package/dist/esm/routing/decorators/body.decorator.js +2 -1
  362. package/dist/esm/routing/decorators/body.decorator.js.map +1 -1
  363. package/dist/esm/routing/resolve-route-params.function.js +1 -1
  364. package/dist/esm/routing/resolve-route-params.function.js.map +1 -1
  365. package/dist/esm/routing/router.js +38 -2
  366. package/dist/esm/routing/router.js.map +1 -1
  367. package/dist/esm/utilities/metadata-injection-keys.enum.js +1 -0
  368. package/dist/esm/utilities/metadata-injection-keys.enum.js.map +1 -1
  369. package/dist/esm/utilities/typeorm.utilities.js +47 -0
  370. package/dist/esm/utilities/typeorm.utilities.js.map +1 -0
  371. package/dist/esm/validation/validation-problem.model.js +4 -7
  372. package/dist/esm/validation/validation-problem.model.js.map +1 -1
  373. package/dist/esm/validation/validation.service.js +6 -4
  374. package/dist/esm/validation/validation.service.js.map +1 -1
  375. package/dist/esm/websocket/models/websocket-message.model.js +1 -1
  376. package/dist/esm/websocket/models/websocket-message.model.js.map +1 -1
  377. package/dist/esm/websocket/models/websocket-request.model.js.map +1 -1
  378. package/dist/esm/websocket/services/websocket.service.js +1 -2
  379. package/dist/esm/websocket/services/websocket.service.js.map +1 -1
  380. package/package.json +9 -9
  381. package/src/__testing__/mocks/entities/child.entity.ts +4 -1
  382. package/src/__testing__/mocks/entities/company.entity.ts +5 -2
  383. package/src/__testing__/mocks/entities/profile.entity.ts +5 -2
  384. package/src/__testing__/mocks/entities/role.entity.ts +1 -1
  385. package/src/__testing__/mocks/entities/user.entity.ts +1 -1
  386. package/src/__testing__/test-server/create-test-data-source.function.ts +5 -5
  387. package/src/__testing__/test-server/start-test-server.function.ts +8 -6
  388. package/src/__testing__/test-server/user-repository.ts +4 -3
  389. package/src/application.ts +8 -2
  390. package/src/auth/2fa/two-factor.service.test.ts +151 -0
  391. package/src/auth/auth.service.test.ts +381 -0
  392. package/src/auth/encryption/encryption-key.model.ts +7 -2
  393. package/src/auth/strategies/cookie/cookie-auth.auth-strategy.ts +3 -2
  394. package/src/auth/strategies/jwt/jwt-credentials.model.ts +7 -1
  395. package/src/auth/strategies/jwt/jwt.auth-strategy.ts +5 -3
  396. package/src/backup/backup-resource-entity.model.ts +7 -2
  397. package/src/backup/backup-service.test.ts +1 -1
  398. package/src/backup/backup.service.ts +1 -1
  399. package/src/caching/cache/base-cache.model.ts +3 -3
  400. package/src/caching/cache/multi-tier.cache.ts +3 -3
  401. package/src/caching/cache/read-aside/read-aside.cache.ts +2 -2
  402. package/src/caching/cache/read-aside/write-around-read-aside.cache.ts +2 -2
  403. package/src/caching/cache/read-aside/write-behind-read-aside.cache.ts +2 -2
  404. package/src/caching/cache/read-aside/write-invalidate-read-aside-args-only.cache.ts +2 -2
  405. package/src/caching/cache/read-aside/write-invalidate-read-aside-with-result.cache.ts +2 -2
  406. package/src/caching/cache/read-aside/write-through-read-aside.cache.ts +2 -2
  407. package/src/caching/cache/read-through/read-through.cache.ts +2 -2
  408. package/src/caching/cache/read-through/write-around-read-through.cache.ts +2 -2
  409. package/src/caching/cache/read-through/write-behind-read-through.cache.ts +2 -2
  410. package/src/caching/cache/read-through/write-invalidate-read-through-args-only.cache.ts +2 -2
  411. package/src/caching/cache/read-through/write-invalidate-read-through-with-result.cache.ts +2 -2
  412. package/src/caching/cache/read-through/write-through-read-through.cache.ts +2 -2
  413. package/src/change-sets/change-set-repository.test.ts +317 -0
  414. package/src/change-sets/change-set-repository.ts +17 -17
  415. package/src/change-sets/models/change-set-entity.model.ts +6 -4
  416. package/src/change-sets/models/change-set.model.ts +1 -1
  417. package/src/change-sets/models/change.model.ts +7 -2
  418. package/src/change-sets/models/soft-delete-entity.model.ts +5 -3
  419. package/src/change-sets/soft-delete-repository.test.ts +326 -0
  420. package/src/change-sets/soft-delete-repository.ts +29 -10
  421. package/src/context/als.utilities.ts +5 -5
  422. package/src/context/cache/cache.context.ts +33 -0
  423. package/src/cron/cron.test.ts +421 -0
  424. package/src/data-source/array-where-filter.test.ts +332 -0
  425. package/src/data-source/data-sources/data-source-initialization.error.ts +9 -0
  426. package/src/data-source/data-sources/data-source.interface.ts +14 -1
  427. package/src/data-source/data-sources/postgres-typeorm-data-source.model.ts +330 -0
  428. package/src/data-source/data-sources/sql-data-source.interface.ts +35 -0
  429. package/src/data-source/data-sources/typeorm-base-data-source.model.ts +544 -0
  430. package/src/data-source/data-sources/where-converter/postgres-typeorm-where-filter.converter.ts +451 -0
  431. package/src/data-source/data-sources/where-converter/typeorm-where-filter.converter.ts +376 -0
  432. package/src/data-source/exclude-property.test.ts +4 -1
  433. package/src/data-source/hooks/hooks.test.ts +268 -0
  434. package/src/data-source/migration/migration.test.ts +4 -3
  435. package/src/data-source/models/options/count-options.model.ts +2 -7
  436. package/src/data-source/models/options/delete-all-options.model.ts +2 -4
  437. package/src/data-source/models/options/find-all-options.model.ts +2 -2
  438. package/src/data-source/models/options/find-by-id-options.model.ts +1 -6
  439. package/src/data-source/models/options/find-one-options.model.ts +2 -11
  440. package/src/data-source/models/options/update-all-options.model.ts +2 -5
  441. package/src/data-source/models/options/update-by-id-options.model.ts +2 -5
  442. package/src/data-source/models/where/array-where-filter.model.ts +147 -5
  443. package/src/data-source/models/where/base-where-filter.model.ts +19 -1
  444. package/src/data-source/models/where/boolean-where-filter.model.ts +7 -2
  445. package/src/data-source/models/where/date-where-filter.model.ts +9 -16
  446. package/src/data-source/models/where/number-where-filter.model.ts +9 -16
  447. package/src/data-source/models/where/object-where-filter.model.ts +66 -7
  448. package/src/data-source/models/where/string-where-filter.model.ts +9 -16
  449. package/src/data-source/models/where/where-filter-keys.model.ts +88 -0
  450. package/src/data-source/models/where/where-filter-to-find-options-where-function.test.ts +35 -13
  451. package/src/data-source/models/where/where-filter.model.ts +7 -4
  452. package/src/data-source/nested-where-filter.test.ts +344 -0
  453. package/src/data-source/query-failed.error.ts +38 -1
  454. package/src/data-source/repository-relation-pitfalls.test.ts +232 -0
  455. package/src/data-source/repository.test.ts +274 -37
  456. package/src/data-source/repository.ts +50 -16
  457. package/src/data-source/transaction/transaction.test.ts +1 -1
  458. package/src/data-source/where-filter.test.ts +479 -0
  459. package/src/di/decorators/inject-repository.decorator.ts +3 -2
  460. package/src/di/default/zibri-di-tokens.default.ts +2 -2
  461. package/src/email/email.service.test.ts +382 -0
  462. package/src/email/email.service.ts +4 -5
  463. package/src/email/models/email.model.ts +1 -1
  464. package/src/entity/decorators/entity.decorator.ts +44 -5
  465. package/src/entity/decorators/property.decorator.ts +54 -36
  466. package/src/entity/entity-metadata-missing.error.ts +15 -0
  467. package/src/entity/models/base-relation-metadata.model.ts +1 -1
  468. package/src/entity/models/belongs-to-one-property-metadata.model.ts +34 -0
  469. package/src/entity/models/has-one-property-metadata.model.ts +20 -0
  470. package/src/entity/models/many-to-many-property-metadata.model.ts +3 -3
  471. package/src/entity/models/many-to-one-property-metadata.model.ts +17 -3
  472. package/src/entity/models/relation.enum.ts +2 -1
  473. package/src/entity/partial-class.model.ts +1 -1
  474. package/src/event/event-cleanup.cron-job.ts +4 -6
  475. package/src/event/event-subscriber-run.model.ts +8 -9
  476. package/src/event/event.model.ts +1 -1
  477. package/src/event/event.service.ts +6 -3
  478. package/src/global/model-registry/default-descriptor.ts +2 -1
  479. package/src/global/model-registry/encryption-descriptor.ts +2 -1
  480. package/src/global/model-registry/exclude-descriptor.ts +2 -1
  481. package/src/global/model-registry/hash-descriptor.ts +2 -1
  482. package/src/index.ts +15 -3
  483. package/src/logging/log-context.model.ts +3 -34
  484. package/src/open-api/open-api.service.ts +32 -13
  485. package/src/parsing/form-data/form-data.body-parser.ts +2 -1
  486. package/src/parsing/functions/parse-boolean.function.ts +0 -1
  487. package/src/parsing/functions/parse-object.function.ts +2 -1
  488. package/src/parsing/parser.ts +2 -1
  489. package/src/routing/decorators/body.decorator.ts +2 -1
  490. package/src/routing/resolve-route-params.function.ts +1 -1
  491. package/src/routing/router.ts +41 -7
  492. package/src/utilities/metadata-injection-keys.enum.ts +1 -0
  493. package/src/utilities/typeorm.utilities.ts +75 -0
  494. package/src/validation/validation-problem.model.ts +7 -9
  495. package/src/validation/validation.service.ts +6 -4
  496. package/src/websocket/models/websocket-message.model.ts +1 -1
  497. package/src/websocket/models/websocket-request.model.ts +17 -9
  498. package/src/websocket/services/websocket.service.ts +1 -2
  499. package/dist/cjs/data-source/data-sources/postgres-data-source.model.d.ts +0 -129
  500. package/dist/cjs/data-source/data-sources/postgres-data-source.model.d.ts.map +0 -1
  501. package/dist/cjs/data-source/data-sources/postgres-data-source.model.js +0 -534
  502. package/dist/cjs/data-source/data-sources/postgres-data-source.model.js.map +0 -1
  503. package/dist/cjs/data-source/models/where/where-filter-to-find-options-where.function.d.ts +0 -11
  504. package/dist/cjs/data-source/models/where/where-filter-to-find-options-where.function.d.ts.map +0 -1
  505. package/dist/cjs/data-source/models/where/where-filter-to-find-options-where.function.js +0 -229
  506. package/dist/cjs/data-source/models/where/where-filter-to-find-options-where.function.js.map +0 -1
  507. package/dist/cjs/entity/models/one-to-one-property-metadata.model.d.ts +0 -30
  508. package/dist/cjs/entity/models/one-to-one-property-metadata.model.d.ts.map +0 -1
  509. package/dist/cjs/entity/models/one-to-one-property-metadata.model.js.map +0 -1
  510. package/dist/esm/data-source/data-sources/postgres-data-source.model.js +0 -534
  511. package/dist/esm/data-source/data-sources/postgres-data-source.model.js.map +0 -1
  512. package/dist/esm/data-source/models/where/where-filter-to-find-options-where.function.js +0 -229
  513. package/dist/esm/data-source/models/where/where-filter-to-find-options-where.function.js.map +0 -1
  514. package/dist/esm/entity/models/one-to-one-property-metadata.model.js.map +0 -1
  515. package/src/data-source/data-sources/postgres-data-source.model.ts +0 -675
  516. package/src/data-source/models/where/where-filter-to-find-options-where.function.ts +0 -307
  517. package/src/entity/models/one-to-one-property-metadata.model.ts +0 -35
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../context/als.utilities';
2
- import { LogCacheContext } from '../../logging/log-context.model';
2
+ import { CacheContext } from '../../context/cache/cache.context';
3
3
  import { LoggerInterface } from '../../logging/logger.interface';
4
4
  import { MetricsServiceInterface } from '../../metrics/metrics-service.interface';
5
5
  import { OmitStrict } from '../../types/omit-strict.type';
@@ -54,7 +54,7 @@ implements OmitStrict<CacheInterface<K, V, CacheTag, WriteResultAvailable, N>, '
54
54
  options?: CacheWrapDeleteOptions<TArgs, CacheTag>
55
55
  ): (...args: TArgs) => Promise<TReturn> {
56
56
  return async (...args) => {
57
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.DELETE };
57
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.DELETE };
58
58
 
59
59
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
60
60
  const sourceStart: number = performance.now();
@@ -95,7 +95,7 @@ implements OmitStrict<CacheInterface<K, V, CacheTag, WriteResultAvailable, N>, '
95
95
  options: CacheWrapInvalidateOptions<TArgs, CacheTag>
96
96
  ): (...args: TArgs) => Promise<TReturn> {
97
97
  return async (...args) => {
98
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.INVALIDATE };
98
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.INVALIDATE };
99
99
 
100
100
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
101
101
  const sourceStart: number = performance.now();
@@ -3,7 +3,7 @@ import { CacheOperation } from './cache-operation.enum';
3
3
  import { CacheWrapOptions, CacheWrapWriteOptionsWithResult, CacheWrapWriteOptionsArgsOnly, CacheWrapDeleteOptions, CacheWrapInvalidateOptions, CacheKeyProvider, ResultCacheKeyProvider, OnInvalidationFailure, CacheTagsProvider, CacheSetDirectOptions } from './cache-options.model';
4
4
  import { CacheInterface } from './cache.interface';
5
5
  import { AlsUtilities } from '../../context/als.utilities';
6
- import { LogCacheContext } from '../../logging/log-context.model';
6
+ import { CacheContext } from '../../context/cache/cache.context';
7
7
  import { LoggerInterface } from '../../logging/logger.interface';
8
8
  import { MetricsServiceInterface } from '../../metrics/metrics-service.interface';
9
9
  import { CacheMetrics } from '../cache-metrics.model';
@@ -243,7 +243,7 @@ export abstract class MultiTierCache<
243
243
  options?: CacheWrapDeleteOptions<TArgs, CacheTag>
244
244
  ): (...args: TArgs) => Promise<TReturn> {
245
245
  return async (...args) => {
246
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.DELETE };
246
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.DELETE };
247
247
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
248
248
  const res: TReturn = await fn(...args);
249
249
  const key: K = await keyFn(...args);
@@ -267,7 +267,7 @@ export abstract class MultiTierCache<
267
267
  options: CacheWrapInvalidateOptions<TArgs, CacheTag>
268
268
  ): (...args: TArgs) => Promise<TReturn> {
269
269
  return async (...args) => {
270
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.INVALIDATE };
270
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.INVALIDATE };
271
271
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
272
272
  const result: TReturn = await fn(...args);
273
273
  const tags: CacheTag[] = typeof options.invalidatesTags === 'function'
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CachedValue } from '../../store/cached-value.model';
4
4
  import { BaseCache } from '../base-cache.model';
5
5
  import { CacheOperation } from '../cache-operation.enum';
@@ -30,7 +30,7 @@ export abstract class ReadAsideCache<
30
30
  options?: CacheWrapOptions<V, TArgs, CacheTag>
31
31
  ): (...args: TArgs) => Promise<V> {
32
32
  return async (...args) => {
33
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRAP };
33
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRAP };
34
34
  const label: Record<string, string> = { cache: this.name };
35
35
 
36
36
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CacheOperation } from '../cache-operation.enum';
4
4
  import { CacheKeyProvider, CacheWrapWriteOptionsArgsOnly } from '../cache-options.model';
5
5
  import { CacheInterface } from '../cache.interface';
@@ -25,7 +25,7 @@ export abstract class WriteAroundReadAsideCache<K, V, N extends string, CacheTag
25
25
  options?: CacheWrapWriteOptionsArgsOnly<TArgs, CacheTag>
26
26
  ): (...args: TArgs) => Promise<V> {
27
27
  return async (...args) => {
28
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRITE };
28
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRITE };
29
29
 
30
30
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
31
31
  const sourceStart: number = performance.now();
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CacheOperation } from '../cache-operation.enum';
4
4
  import { ResultCacheKeyProvider, CacheWrapWriteOptionsWithResult, CacheSetDirectOptions } from '../cache-options.model';
5
5
  import { CacheInterface } from '../cache.interface';
@@ -25,7 +25,7 @@ export abstract class WriteBehindReadAsideCache<K, V, N extends string, CacheTag
25
25
  options?: CacheWrapWriteOptionsWithResult<V, TArgs, CacheTag>
26
26
  ): (...args: TArgs) => Promise<V> {
27
27
  return async (...args) => {
28
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRITE };
28
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRITE };
29
29
 
30
30
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
31
31
  const sourceStart: number = performance.now();
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CacheOperation } from '../cache-operation.enum';
4
4
  import { CacheKeyProvider, CacheWrapWriteOptionsArgsOnly } from '../cache-options.model';
5
5
  import { CacheInterface } from '../cache.interface';
@@ -25,7 +25,7 @@ export abstract class WriteInvalidateReadAsideArgsOnlyCache<K, V, N extends stri
25
25
  options?: CacheWrapWriteOptionsArgsOnly<TArgs, CacheTag>
26
26
  ): (...args: TArgs) => Promise<V> {
27
27
  return async (...args) => {
28
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRITE };
28
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRITE };
29
29
 
30
30
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
31
31
  const sourceStart: number = performance.now();
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CacheOperation } from '../cache-operation.enum';
4
4
  import { ResultCacheKeyProvider, CacheWrapWriteOptionsWithResult } from '../cache-options.model';
5
5
  import { CacheInterface } from '../cache.interface';
@@ -25,7 +25,7 @@ export abstract class WriteInvalidateReadAsideWithResultCache<K, V, N extends st
25
25
  options?: CacheWrapWriteOptionsWithResult<V, TArgs, CacheTag>
26
26
  ): (...args: TArgs) => Promise<V> {
27
27
  return async (...args) => {
28
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRITE };
28
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRITE };
29
29
 
30
30
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
31
31
  const sourceStart: number = performance.now();
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CacheOperation } from '../cache-operation.enum';
4
4
  import { ResultCacheKeyProvider, CacheWrapWriteOptionsWithResult, CacheSetDirectOptions } from '../cache-options.model';
5
5
  import { CacheInterface } from '../cache.interface';
@@ -25,7 +25,7 @@ export abstract class WriteThroughReadAsideCache<K, V, N extends string, CacheTa
25
25
  options?: CacheWrapWriteOptionsWithResult<V, TArgs, CacheTag>
26
26
  ): (...args: TArgs) => Promise<V> {
27
27
  return async (...args) => {
28
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRITE };
28
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRITE };
29
29
 
30
30
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
31
31
  const sourceStart: number = performance.now();
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CachedValue } from '../../store/cached-value.model';
4
4
  import { BaseCache } from '../base-cache.model';
5
5
  import { CacheOperation } from '../cache-operation.enum';
@@ -22,7 +22,7 @@ export abstract class ReadThroughCache<
22
22
  options?: CacheWrapOptions<V, TArgs, CacheTag>
23
23
  ): (...args: TArgs) => Promise<V> {
24
24
  return async (...args) => {
25
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRAP };
25
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRAP };
26
26
  const label: Record<string, string> = { cache: this.name };
27
27
 
28
28
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CacheOperation } from '../cache-operation.enum';
4
4
  import { CacheKeyProvider, CacheWrapWriteOptionsArgsOnly } from '../cache-options.model';
5
5
  import { CacheInterface } from '../cache.interface';
@@ -23,7 +23,7 @@ export abstract class WriteAroundReadThroughCache<K, V, N extends string, CacheT
23
23
  options?: CacheWrapWriteOptionsArgsOnly<TArgs, CacheTag>
24
24
  ): (...args: TArgs) => Promise<V> {
25
25
  return async (...args) => {
26
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRITE };
26
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRITE };
27
27
 
28
28
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
29
29
  const sourceStart: number = performance.now();
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CacheOperation } from '../cache-operation.enum';
4
4
  import { ResultCacheKeyProvider, CacheWrapWriteOptionsWithResult, CacheSetDirectOptions } from '../cache-options.model';
5
5
  import { CacheInterface } from '../cache.interface';
@@ -26,7 +26,7 @@ export abstract class WriteBehindReadThroughCache<K, V, N extends string, CacheT
26
26
  options?: CacheWrapWriteOptionsWithResult<V, TArgs, CacheTag>
27
27
  ): (...args: TArgs) => Promise<V> {
28
28
  return async (...args) => {
29
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRITE };
29
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRITE };
30
30
 
31
31
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
32
32
  const sourceStart: number = performance.now();
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CacheOperation } from '../cache-operation.enum';
4
4
  import { CacheKeyProvider, CacheWrapWriteOptionsArgsOnly } from '../cache-options.model';
5
5
  import { CacheInterface } from '../cache.interface';
@@ -26,7 +26,7 @@ export abstract class WriteInvalidateReadThroughArgsOnlyCache<K, V, N extends st
26
26
  options?: CacheWrapWriteOptionsArgsOnly<TArgs, CacheTag>
27
27
  ): (...args: TArgs) => Promise<V> {
28
28
  return async (...args) => {
29
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRITE };
29
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRITE };
30
30
 
31
31
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
32
32
  const sourceStart: number = performance.now();
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CacheOperation } from '../cache-operation.enum';
4
4
  import { ResultCacheKeyProvider, CacheWrapWriteOptionsWithResult } from '../cache-options.model';
5
5
  import { CacheInterface } from '../cache.interface';
@@ -25,7 +25,7 @@ export abstract class WriteInvalidateReadThroughWithResultCache<K, V, N extends
25
25
  options?: CacheWrapWriteOptionsWithResult<V, TArgs, CacheTag>
26
26
  ): (...args: TArgs) => Promise<V> {
27
27
  return async (...args) => {
28
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRITE };
28
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRITE };
29
29
 
30
30
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
31
31
  const sourceStart: number = performance.now();
@@ -1,5 +1,5 @@
1
1
  import { AlsUtilities } from '../../../context/als.utilities';
2
- import { LogCacheContext } from '../../../logging/log-context.model';
2
+ import { CacheContext } from '../../../context/cache/cache.context';
3
3
  import { CacheOperation } from '../cache-operation.enum';
4
4
  import { ResultCacheKeyProvider, CacheWrapWriteOptionsWithResult, CacheSetDirectOptions } from '../cache-options.model';
5
5
  import { CacheInterface } from '../cache.interface';
@@ -22,7 +22,7 @@ export abstract class WriteThroughReadThroughCache<K, V, N extends string, Cache
22
22
  options?: CacheWrapWriteOptionsWithResult<V, TArgs, CacheTag>
23
23
  ): (...args: TArgs) => Promise<V> {
24
24
  return async (...args) => {
25
- const cacheCtx: LogCacheContext = { cache: this.name, operation: CacheOperation.WRITE };
25
+ const cacheCtx: CacheContext = { cache: this.name, operation: CacheOperation.WRITE };
26
26
 
27
27
  return AlsUtilities.runWithCacheContext(cacheCtx, async () => {
28
28
  const sourceStart: number = performance.now();
@@ -0,0 +1,317 @@
1
+ import { afterAll, beforeAll, beforeEach, describe, expect, it } from '@jest/globals';
2
+
3
+ import { ChangeSetRepository, ResetChangeSetResult } from './change-set-repository';
4
+ import { ChangeSet, CreateChangeSetData } from './models/change-set.model';
5
+ import { Change } from './models/change.model';
6
+ import { Roles } from '../__testing__/mocks/entities/roles.enum';
7
+ import { createTestDataSource, defaultTestServerEntities } from '../__testing__/test-server/create-test-data-source.function';
8
+ import { startTestServer, StartedTestServer } from '../__testing__/test-server/start-test-server.function';
9
+ import { DefaultTestServerUserRepository } from '../__testing__/test-server/user-repository'; // adjust if needed
10
+ import { AuthServiceInterface } from '../auth/auth-service.interface';
11
+ import { Auth } from '../auth/decorators/auth.decorator';
12
+ import { BaseUserEntity } from '../auth/models/base-user.model';
13
+ import { JwtAuthData } from '../auth/strategies/jwt/jwt-auth-data.model';
14
+ import { JwtCredentials } from '../auth/strategies/jwt/jwt-credentials.model';
15
+ import { JwtAuthStrategy } from '../auth/strategies/jwt/jwt.auth-strategy';
16
+ import { Repository } from '../data-source/repository';
17
+ import { InjectRepository, repositoryTokenFor } from '../di/decorators/inject-repository.decorator';
18
+ import { ZIBRI_DI_TOKENS } from '../di/default/zibri-di-tokens.default';
19
+ import { inject } from '../di/inject.function';
20
+ import { BaseEntity } from '../entity/base-entity.model';
21
+ import { ChangeSetEntity } from './models/change-set-entity.model';
22
+ import { ChangeSetType } from './models/change-set-type.enum';
23
+ import { Entity } from '../entity/decorators/entity.decorator';
24
+ import { Property } from '../entity/decorators/property.decorator';
25
+ import { OmitClass } from '../entity/omit-class.model';
26
+ import { PartialClass } from '../entity/partial-class.model';
27
+ import { Body } from '../routing/decorators/body.decorator';
28
+ import { Controller } from '../routing/decorators/controller.decorator';
29
+ import { Delete } from '../routing/decorators/delete.decorator';
30
+ import { Param } from '../routing/decorators/param.decorator';
31
+ import { Patch } from '../routing/decorators/patch.decorator';
32
+ import { Post } from '../routing/decorators/post.decorator';
33
+ import { JsonUtilities } from '../utilities/json.utilities';
34
+
35
+ // ---- Test entity that extends ChangeSetEntity ----
36
+ @Entity()
37
+ class Widget extends BaseEntity implements ChangeSetEntity {
38
+ @Property.string()
39
+ name!: string;
40
+
41
+ @Property.number()
42
+ value!: number;
43
+
44
+ @Property.string({ excludeFromChangeSets: true })
45
+ internalNote!: string;
46
+
47
+ // required by ChangeSetEntity
48
+ @Property.oneToMany({ target: () => ChangeSet, inverseSide: 'changeSetEntityId' }) // dummy, real implementation may differ
49
+ changeSets!: ChangeSet[];
50
+ }
51
+
52
+ class CreateWidgetDto extends OmitClass(Widget, ['changeSets', 'id']) {}
53
+
54
+ class UpdateWidgetDto extends PartialClass(OmitClass(Widget, ['changeSets', 'id'])) {}
55
+
56
+ @Entity()
57
+ class TestUser extends BaseUserEntity(Roles) {
58
+ @Property.string({ hash: true })
59
+ password!: string;
60
+ }
61
+
62
+ // ---- Controller that uses the repository ----
63
+ @Controller('/widgets')
64
+ class WidgetController {
65
+ constructor(
66
+ @InjectRepository(Widget)
67
+ private readonly repo: ChangeSetRepository<Widget>
68
+ ) {}
69
+
70
+ @Post('/')
71
+ @Auth.isLoggedIn([JwtAuthStrategy])
72
+ async create(@Body(CreateWidgetDto) body: CreateWidgetDto): Promise<Widget> {
73
+ return await this.repo.create(body);
74
+ }
75
+
76
+ @Patch('/:id')
77
+ @Auth.isLoggedIn([JwtAuthStrategy])
78
+ async update(@Param.path('id') id: string, @Body(UpdateWidgetDto) body: UpdateWidgetDto): Promise<Widget> {
79
+ return this.repo.updateById(id, body);
80
+ }
81
+
82
+ @Delete('/:id')
83
+ @Auth.isLoggedIn([JwtAuthStrategy])
84
+ async delete(@Param.path('id') id: string): Promise<Widget> {
85
+ return this.repo.deleteById(id);
86
+ }
87
+
88
+ @Post('/:id/reset/:changeSetId')
89
+ @Auth.isLoggedIn([JwtAuthStrategy])
90
+ async resetSingle(
91
+ @Param.path('id') id: string,
92
+ @Param.path('changeSetId') changeSetId: string
93
+ ): Promise<ResetChangeSetResult<Widget>> {
94
+ const entity: Widget = await this.repo.findById(id);
95
+ return this.repo.resetSingleChangeSet(entity, changeSetId);
96
+ }
97
+
98
+ @Post('/:id/rollback/:changeSetId')
99
+ @Auth.isLoggedIn([JwtAuthStrategy])
100
+ async rollbackTo(
101
+ @Param.path('id') id: string,
102
+ @Param.path('changeSetId') changeSetId: string
103
+ ): Promise<Widget> {
104
+ const entity: Widget = await this.repo.findById(id);
105
+ return this.repo.rollbackToChangeSet(entity, changeSetId);
106
+ }
107
+ }
108
+
109
+ let server: StartedTestServer;
110
+ let baseUrl: string;
111
+ let accessToken: string;
112
+ let userId: string;
113
+
114
+ let changeSetRepo: Repository<ChangeSet, CreateChangeSetData>;
115
+ let changeRepo: Repository<Change>;
116
+ let widgetRepo: ChangeSetRepository<Widget>;
117
+
118
+ beforeAll(async () => {
119
+ server = await startTestServer({
120
+ dataSources: [
121
+ createTestDataSource({
122
+ entities: [...defaultTestServerEntities, Widget, TestUser]
123
+ })
124
+ ],
125
+ controllers: [WidgetController]
126
+ });
127
+ baseUrl = await server.start();
128
+
129
+ // Inject repository instances
130
+ changeSetRepo = inject(repositoryTokenFor(ChangeSet));
131
+ changeRepo = inject(repositoryTokenFor(Change));
132
+ widgetRepo = inject(repositoryTokenFor(Widget)) as ChangeSetRepository<Widget>;
133
+
134
+ // Create a user and login to get a token
135
+ const userRepo: DefaultTestServerUserRepository = inject(DefaultTestServerUserRepository);
136
+ const credentialsRepo: Repository<JwtCredentials> = inject(repositoryTokenFor(JwtCredentials));
137
+
138
+ const testEmail: string = 'widget-test@example.com';
139
+ const testPassword: string = 'test123';
140
+ await userRepo.create({ email: testEmail, roles: [Roles.USER] });
141
+ await credentialsRepo.create({ email: testEmail, password: testPassword, userId: (await userRepo.findOne({ where: { email: testEmail } }, true)).id });
142
+
143
+ userId = (await userRepo.findOne({ where: { email: testEmail } }, true)).id;
144
+
145
+ const authService: AuthServiceInterface = inject(ZIBRI_DI_TOKENS.AUTH_SERVICE);
146
+ const authData: JwtAuthData<Roles> = await authService.login(JwtAuthStrategy<Roles>, { email: testEmail, password: testPassword });
147
+ accessToken = authData.accessToken.value;
148
+ }, 15000);
149
+
150
+ afterAll(async () => {
151
+ await server.shutdown();
152
+ });
153
+
154
+ beforeEach(async () => {
155
+ await changeSetRepo.deleteAll({});
156
+ await changeRepo.deleteAll({});
157
+ await widgetRepo.deleteAll({});
158
+ });
159
+
160
+ // Helper for authorized requests
161
+ async function authFetch(path: string, options: RequestInit = {}): Promise<Response> {
162
+ const res: Response = await fetch(`${baseUrl}${path}`, {
163
+ ...options,
164
+ headers: {
165
+ ...options.headers,
166
+ Authorization: `Bearer ${accessToken}`,
167
+ 'Content-Type': 'application/json'
168
+ }
169
+ });
170
+
171
+ if (!res.ok) {
172
+ const body: unknown = await res.json();
173
+ throw new Error(`Request "${options.method ?? 'GET'} ${path}" failed:\n${JsonUtilities.stringify(body, undefined, 4)}`);
174
+ }
175
+
176
+ return res;
177
+ }
178
+
179
+ describe('ChangeSetRepository behavior', () => {
180
+ describe('on create', () => {
181
+ it('creates a CREATE change set with correct changes (excluding internalNote)', async () => {
182
+ const res: Response = await authFetch('/widgets', {
183
+ method: 'POST',
184
+ body: JSON.stringify({ name: 'TestWidget', value: 42, internalNote: 'secret' })
185
+ });
186
+ expect(res.status).toBe(200);
187
+ const widget: Widget = await res.json() as Widget;
188
+
189
+ const changeSets: ChangeSet[] = await changeSetRepo.findAll({ where: { changeSetEntityId: widget.id } });
190
+ expect(changeSets).toHaveLength(1);
191
+ const cs: ChangeSet = changeSets[0];
192
+ expect(cs.type).toBe(ChangeSetType.CREATE);
193
+ expect(cs.createdBy).toBe(userId);
194
+
195
+ const changes: Change[] = await changeRepo.findAll({ where: { changeSetId: cs.id } });
196
+ expect(changes).toHaveLength(3); // id, name and value. not internalNote
197
+ expect(changes.map(c => c.key).sort()).toEqual(['id', 'name', 'value']);
198
+ expect(changes.find(c => c.key === 'name')?.previousValue).toBeNull();
199
+ expect(changes.find(c => c.key === 'name')?.newValue).toBe('TestWidget');
200
+ });
201
+ });
202
+
203
+ describe('on update', () => {
204
+ let widgetId: string;
205
+
206
+ beforeEach(async () => {
207
+ const res: Response = await authFetch('/widgets', {
208
+ method: 'POST',
209
+ body: JSON.stringify({ name: 'Original', value: 100, internalNote: 'old-secret' })
210
+ });
211
+ const widget: Widget = await res.json() as Widget;
212
+ widgetId = widget.id;
213
+ // clear change sets from creation to isolate update test
214
+ await changeSetRepo.deleteAll({});
215
+ });
216
+
217
+ it('creates an UPDATE change set only with changed fields', async () => {
218
+ await authFetch(`/widgets/${widgetId}`, {
219
+ method: 'PATCH',
220
+ body: JSON.stringify({ name: 'Updated', value: 100 }) // value unchanged
221
+ });
222
+
223
+ const changeSets: ChangeSet[] = await changeSetRepo.findAll({ where: { changeSetEntityId: widgetId } });
224
+ expect(changeSets).toHaveLength(1);
225
+ expect(changeSets[0].type).toBe(ChangeSetType.UPDATE);
226
+
227
+ const changes: Change[] = await changeRepo.findAll({ where: { changeSetId: changeSets[0].id } });
228
+ expect(changes).toHaveLength(1); // only name changed
229
+ expect(changes[0].key).toBe('name');
230
+ expect(changes[0].previousValue).toBe('Original');
231
+ expect(changes[0].newValue).toBe('Updated');
232
+ });
233
+
234
+ it('does not create a change set when no values actually change', async () => {
235
+ await authFetch(`/widgets/${widgetId}`, {
236
+ method: 'PATCH',
237
+ body: JSON.stringify({ name: 'Original', value: 100 })
238
+ });
239
+
240
+ const changeSets: ChangeSet[] = await changeSetRepo.findAll({ where: { changeSetEntityId: widgetId } });
241
+ expect(changeSets).toHaveLength(0);
242
+ });
243
+ });
244
+
245
+ describe('resetSingleChangeSet', () => {
246
+ let widgetId: string;
247
+ let nameUpdateChangeSetId: string;
248
+
249
+ beforeEach(async () => {
250
+ // Create widget, this causes
251
+ const res1: Response = await authFetch('/widgets', {
252
+ method: 'POST',
253
+ body: JSON.stringify({ name: 'Initial', value: 10, internalNote: 'test' })
254
+ });
255
+ const w: Widget = await res1.json() as Widget;
256
+ widgetId = w.id;
257
+
258
+ // First update: change name
259
+ await authFetch(`/widgets/${widgetId}`, { method: 'PATCH', body: JSON.stringify({ name: 'FirstEdit' }) });
260
+
261
+ // Second update: change value
262
+ await authFetch(`/widgets/${widgetId}`, { method: 'PATCH', body: JSON.stringify({ value: 20 }) });
263
+
264
+ const sets: ChangeSet[] = await changeSetRepo.findAll({
265
+ where: { changeSetEntityId: widgetId },
266
+ order: { createdAt: 'ASC' },
267
+ relations: ['changes']
268
+ });
269
+ // sets[0] = CREATE, sets[1] = UPDATE name, sets[2] = UPDATE value
270
+ nameUpdateChangeSetId = sets[1].id; // target the name update
271
+ });
272
+
273
+ it('reverts only the targeted change, preserving later changes', async () => {
274
+ await authFetch(`/widgets/${widgetId}/reset/${nameUpdateChangeSetId}`, { method: 'POST' });
275
+
276
+ const widget: Widget = await widgetRepo.findById(widgetId);
277
+ expect(widget.name).toBe('Initial'); // reverted
278
+ expect(widget.value).toBe(20); // preserved (later change untouched)
279
+
280
+ // A new RESET change set was created
281
+ const allSets: ChangeSet[] = await changeSetRepo.findAll({ where: { changeSetEntityId: widgetId } });
282
+ expect(allSets.some(s => s.type === ChangeSetType.RESET)).toBe(true);
283
+ });
284
+ });
285
+
286
+ describe('rollbackToChangeSet', () => {
287
+ let widgetId: string;
288
+ let createSetId: string;
289
+
290
+ beforeEach(async () => {
291
+ const res: Response = await authFetch('/widgets', {
292
+ method: 'POST',
293
+ body: JSON.stringify({ name: 'Base', value: 1, internalNote: 'base' })
294
+ });
295
+ const w: Widget = await res.json() as Widget;
296
+ widgetId = w.id;
297
+ createSetId = (await changeSetRepo.findOne({ where: { changeSetEntityId: widgetId, type: ChangeSetType.CREATE } }, true)).id;
298
+
299
+ // Two subsequent updates
300
+ await authFetch(`/widgets/${widgetId}`, { method: 'PATCH', body: JSON.stringify({ name: 'Version2' }) });
301
+ await authFetch(`/widgets/${widgetId}`, { method: 'PATCH', body: JSON.stringify({ value: 99 }) });
302
+ });
303
+
304
+ it('rolls back all changes after the given change set', async () => {
305
+ await authFetch(`/widgets/${widgetId}/rollback/${createSetId}`, { method: 'POST' });
306
+
307
+ const widget: Widget = await widgetRepo.findById(widgetId);
308
+ expect(widget.name).toBe('Base');
309
+ expect(widget.value).toBe(1);
310
+
311
+ // Only the CREATE and the new RESET change set remain (intermediate ones deleted)
312
+ const remaining: ChangeSet[] = await changeSetRepo.findAll({ where: { changeSetEntityId: widgetId } });
313
+ expect(remaining).toHaveLength(2);
314
+ expect(remaining.map(s => s.type)).toEqual(expect.arrayContaining([ChangeSetType.CREATE, ChangeSetType.RESET]));
315
+ });
316
+ });
317
+ });