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
@@ -0,0 +1,382 @@
1
+ import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, it, jest } from '@jest/globals';
2
+ import SMTPTransport from 'nodemailer/lib/smtp-transport';
3
+
4
+ import { EmailService } from './email.service';
5
+ import { CreateEmailData, QueueEmailData } from './models/create-email-data.model';
6
+ import { EmailPriority } from './models/email-priority.enum';
7
+ import { EmailStatus } from './models/email-status.enum';
8
+ import { Email } from './models/email.model';
9
+ import { defaultTestServerProviders } from '../__testing__/test-server/providers';
10
+ import { startTestServer, StartedTestServer } from '../__testing__/test-server/start-test-server.function';
11
+ import { Repository } from '../data-source/repository';
12
+ import { EmailConfig } from './models/email-config.model';
13
+ import { InjectRepository, repositoryTokenFor } from '../di/decorators/inject-repository.decorator';
14
+ import { Inject } from '../di/decorators/inject.decorator';
15
+ import { Injectable } from '../di/decorators/injectable.decorator';
16
+ import { ZIBRI_DI_TOKENS } from '../di/default/zibri-di-tokens.default';
17
+ import { inject } from '../di/inject.function';
18
+ import { type LoggerInterface } from '../logging/logger.interface';
19
+
20
+ // ---------- Test email service (mocks the transporter) ----------
21
+
22
+ // eslint-disable-next-line typescript/typedef
23
+ const mockSendMail = jest.fn<() => Promise<SMTPTransport.SentMessageInfo>>();
24
+
25
+ const testConfig: EmailConfig = {
26
+ host: 'smtp.example.com',
27
+ port: 587,
28
+ maxEmailsPerHour: 1000,
29
+ defaultSender: 'noreply@example.com',
30
+ pool: false,
31
+ auth: { user: 'user', pass: 'pass' }
32
+ };
33
+
34
+ @Injectable({ register: 'onUse' })
35
+ class TestEmailService extends EmailService {
36
+ constructor(
37
+ @Inject(ZIBRI_DI_TOKENS.LOGGER) logger: LoggerInterface,
38
+ @InjectRepository(Email) emailRepository: Repository<Email, CreateEmailData>
39
+ ) {
40
+ super(logger, emailRepository, testConfig);
41
+ }
42
+
43
+ protected override async send(email: Email): Promise<void> {
44
+ await this.validateAttachments(email.attachments);
45
+ const res: SMTPTransport.SentMessageInfo = await mockSendMail();
46
+ const status: EmailStatus = res.rejected.length ? EmailStatus.FAILED : EmailStatus.SENT;
47
+
48
+ if (status === EmailStatus.FAILED) {
49
+ const rejectedRecipients: string[] = res.rejected.map((e) => typeof e === 'string' ? e : e.address);
50
+ await this['logger'].warn(`mail to ${rejectedRecipients.join(', ')} was rejected`);
51
+ }
52
+
53
+ if (email.persist) {
54
+ await this['emailRepository'].updateById(email.id, { status });
55
+ return;
56
+ }
57
+
58
+ await this['emailRepository'].deleteById(email.id);
59
+ }
60
+ }
61
+
62
+ // ---------- Helpers ----------
63
+
64
+ function sentMessageInfo(overrides: Partial<SMTPTransport.SentMessageInfo> = {}): SMTPTransport.SentMessageInfo {
65
+ return {
66
+ accepted: ['recipient@example.com'],
67
+ rejected: [],
68
+ pending: [],
69
+ response: '250 OK',
70
+ envelope: { from: 'noreply@example.com', to: ['recipient@example.com'] },
71
+ messageId: 'test-message-id',
72
+ ...overrides
73
+ } as SMTPTransport.SentMessageInfo;
74
+ }
75
+
76
+ function makeQueueData(overrides: Partial<QueueEmailData> = {}): QueueEmailData {
77
+ return {
78
+ html: '<p>Hello</p>',
79
+ subject: 'Test subject',
80
+ recipients: ['recipient@example.com'],
81
+ ...overrides
82
+ } as QueueEmailData;
83
+ }
84
+
85
+ // ---------- Test setup ----------
86
+
87
+ let server: StartedTestServer;
88
+ let emailService: TestEmailService;
89
+ let emailRepo: Repository<Email, CreateEmailData>;
90
+
91
+ beforeAll(async () => {
92
+ server = await startTestServer({
93
+ providers: [
94
+ ...defaultTestServerProviders,
95
+ { token: ZIBRI_DI_TOKENS.EMAIL_CONFIG, useValue: testConfig },
96
+ { token: ZIBRI_DI_TOKENS.EMAIL_SERVICE, useClass: TestEmailService }
97
+ ]
98
+ });
99
+ await server.start();
100
+
101
+ emailService = inject(TestEmailService);
102
+ emailRepo = inject(repositoryTokenFor(Email));
103
+ }, 15000);
104
+
105
+ afterAll(async () => {
106
+ await server.shutdown();
107
+ });
108
+
109
+ beforeEach(async () => {
110
+ await emailRepo.deleteAll({});
111
+ mockSendMail.mockReset();
112
+ mockSendMail.mockResolvedValue(sentMessageInfo());
113
+ });
114
+
115
+ // ---------- Tests ----------
116
+
117
+ describe('EmailService.queue', () => {
118
+ it('creates an email with QUEUED status', async () => {
119
+ await emailService.queue(makeQueueData());
120
+
121
+ const emails: Email[] = await emailRepo.findAll({});
122
+ expect(emails).toHaveLength(1);
123
+ expect(emails[0].status).toBe(EmailStatus.QUEUED);
124
+ });
125
+
126
+ it('defaults to NORMAL priority', async () => {
127
+ await emailService.queue(makeQueueData());
128
+
129
+ const emails: Email[] = await emailRepo.findAll({});
130
+ expect(emails[0].priority).toBe(EmailPriority.NORMAL);
131
+ });
132
+
133
+ it('respects provided priority', async () => {
134
+ await emailService.queue(makeQueueData({ priority: EmailPriority.HIGH }));
135
+
136
+ const emails: Email[] = await emailRepo.findAll({});
137
+ expect(emails[0].priority).toBe(EmailPriority.HIGH);
138
+ });
139
+
140
+ it('defaults to defaultSender from config when no sender provided', async () => {
141
+ await emailService.queue(makeQueueData());
142
+
143
+ const emails: Email[] = await emailRepo.findAll({});
144
+ expect(emails[0].sender).toBe(testConfig.defaultSender);
145
+ });
146
+
147
+ it('respects provided sender', async () => {
148
+ await emailService.queue(makeQueueData({ sender: 'custom@example.com' }));
149
+
150
+ const emails: Email[] = await emailRepo.findAll({});
151
+ expect(emails[0].sender).toBe('custom@example.com');
152
+ });
153
+
154
+ it('defaults persist to false', async () => {
155
+ await emailService.queue(makeQueueData());
156
+
157
+ const emails: Email[] = await emailRepo.findAll({});
158
+ expect(emails[0].persist).toBe(false);
159
+ });
160
+
161
+ it('respects provided persist flag', async () => {
162
+ await emailService.queue(makeQueueData({ persist: true }));
163
+
164
+ const emails: Email[] = await emailRepo.findAll({});
165
+ expect(emails[0].persist).toBe(true);
166
+ });
167
+ });
168
+
169
+ describe('EmailService.sendQueuedEmails', () => {
170
+ afterEach(() => void jest.restoreAllMocks());
171
+
172
+ it('returns false and sends nothing when no emails are queued', async () => {
173
+ const result: boolean = await emailService.sendQueuedEmails();
174
+
175
+ expect(result).toBe(false);
176
+ expect(mockSendMail).not.toHaveBeenCalled();
177
+ });
178
+
179
+ it('returns true after sending emails (indicating there may be more)', async () => {
180
+ await emailService.queue(makeQueueData());
181
+
182
+ const result: boolean = await emailService.sendQueuedEmails();
183
+
184
+ expect(result).toBe(true);
185
+ expect(mockSendMail).toHaveBeenCalledTimes(1);
186
+ });
187
+
188
+ it('deletes email after sending when persist is false', async () => {
189
+ await emailService.queue(makeQueueData({ persist: false }));
190
+
191
+ await emailService.sendQueuedEmails();
192
+
193
+ const emails: Email[] = await emailRepo.findAll({});
194
+ expect(emails).toHaveLength(0);
195
+ });
196
+
197
+ it('updates email status to SENT when persist is true', async () => {
198
+ await emailService.queue(makeQueueData({ persist: true }));
199
+
200
+ await emailService.sendQueuedEmails();
201
+
202
+ const emails: Email[] = await emailRepo.findAll({});
203
+ expect(emails).toHaveLength(1);
204
+ expect(emails[0].status).toBe(EmailStatus.SENT);
205
+ });
206
+
207
+ it('updates email status to FAILED when the send is rejected', async () => {
208
+ mockSendMail.mockResolvedValue(sentMessageInfo({
209
+ rejected: ['recipient@example.com'],
210
+ accepted: []
211
+ }));
212
+ await emailService.queue(makeQueueData({ persist: true }));
213
+
214
+ await emailService.sendQueuedEmails();
215
+
216
+ const emails: Email[] = await emailRepo.findAll({});
217
+ expect(emails[0].status).toBe(EmailStatus.FAILED);
218
+ });
219
+
220
+ it('sends HIGH priority emails before NORMAL and LOW', async () => {
221
+ // eslint-disable-next-line typescript/require-await
222
+ mockSendMail.mockImplementation(async () => {
223
+ return sentMessageInfo();
224
+ });
225
+
226
+ // Queue in reverse priority order to ensure ordering is by priority, not insertion
227
+ await emailService.queue(makeQueueData({ priority: EmailPriority.LOW, subject: 'low' }));
228
+ await emailService.queue(makeQueueData({ priority: EmailPriority.NORMAL, subject: 'normal' }));
229
+ await emailService.queue(makeQueueData({ priority: EmailPriority.HIGH, subject: 'high' }));
230
+
231
+ // Capture send order by intercepting the underlying send call
232
+ const calls: Email[] = [];
233
+ jest.spyOn(emailService as unknown as { send: (e: Email) => Promise<void> }, 'send')
234
+ // eslint-disable-next-line typescript/require-await
235
+ .mockImplementation(async (email: Email) => {
236
+ calls.push(email);
237
+ });
238
+
239
+ await emailService.sendQueuedEmails();
240
+
241
+ expect(calls[0].priority).toBe(EmailPriority.HIGH);
242
+ expect(calls[1].priority).toBe(EmailPriority.NORMAL);
243
+ expect(calls[2].priority).toBe(EmailPriority.LOW);
244
+ });
245
+
246
+ it('sends at most 10 emails per batch across all priorities', async () => {
247
+ // Need all three priorities to fill 10 slots: HIGH(8) + NORMAL(1 reserved) + LOW(1 reserved)
248
+ for (let i: number = 0; i < 10; i++) {
249
+ await emailService.queue(makeQueueData({ priority: EmailPriority.HIGH, subject: `High ${i}` }));
250
+ }
251
+ for (let i: number = 0; i < 5; i++) {
252
+ await emailService.queue(makeQueueData({ priority: EmailPriority.NORMAL, subject: `Normal ${i}` }));
253
+ }
254
+ for (let i: number = 0; i < 5; i++) {
255
+ await emailService.queue(makeQueueData({ priority: EmailPriority.LOW, subject: `Low ${i}` }));
256
+ }
257
+
258
+ await emailService.sendQueuedEmails();
259
+
260
+ // 8 high + 1 normal + 1 low = 10 (normal is capped at 10 - 1 - highCount)
261
+ expect(mockSendMail).toHaveBeenCalledTimes(10);
262
+ });
263
+
264
+ it('sends at most 8 HIGH priority emails per batch', async () => {
265
+ for (let i: number = 0; i < 10; i++) {
266
+ await emailService.queue(makeQueueData({ priority: EmailPriority.HIGH, subject: `High ${i}` }));
267
+ }
268
+
269
+ await emailService.sendQueuedEmails();
270
+
271
+ // HIGH is capped at 8; no NORMAL or LOW queued, so total = 8
272
+ expect(mockSendMail).toHaveBeenCalledTimes(8);
273
+ });
274
+
275
+ it('fills remaining slots with NORMAL and LOW after HIGH', async () => {
276
+ for (let i: number = 0; i < 8; i++) {
277
+ await emailService.queue(makeQueueData({ priority: EmailPriority.HIGH, subject: `High ${i}` }));
278
+ }
279
+ for (let i: number = 0; i < 5; i++) {
280
+ await emailService.queue(makeQueueData({ priority: EmailPriority.NORMAL, subject: `Normal ${i}` }));
281
+ }
282
+ for (let i: number = 0; i < 5; i++) {
283
+ await emailService.queue(makeQueueData({ priority: EmailPriority.LOW, subject: `Low ${i}` }));
284
+ }
285
+
286
+ await emailService.sendQueuedEmails();
287
+
288
+ // 8 high + 1 normal (10 - 1 - 8 = 1) + 1 low (10 - 8 - 1 = 1) = 10
289
+ expect(mockSendMail).toHaveBeenCalledTimes(10);
290
+ });
291
+ });
292
+
293
+ // describe('EmailService rate limiting', () => {
294
+ // // A separate server with a very low rate limit so we can exhaust it with real sends
295
+ // const rateLimitConfig: EmailConfig = { ...testConfig, maxEmailsPerHour: 3 };
296
+
297
+ // let rateLimitServer: StartedTestServer;
298
+ // let rateLimitEmailService: TestEmailService;
299
+ // let rateLimitEmailRepo: Repository<Email, CreateEmailData>;
300
+
301
+ // beforeAll(async () => {
302
+ // rateLimitServer = await startTestServer({
303
+ // dataSources: [
304
+ // createTestDataSource({
305
+ // entities: [...defaultTestServerEntities, Email]
306
+ // })
307
+ // ],
308
+ // controllers: [],
309
+ // cronJobs: [],
310
+ // providers: [
311
+ // ...defaultTestServerProviders,
312
+ // { token: ZIBRI_DI_TOKENS.EMAIL_CONFIG, useValue: rateLimitConfig },
313
+ // { token: ZIBRI_DI_TOKENS.EMAIL_SERVICE, useClass: TestEmailService }
314
+ // ]
315
+ // });
316
+ // await rateLimitServer.start();
317
+ // rateLimitEmailService = inject(TestEmailService);
318
+ // rateLimitEmailRepo = inject(repositoryTokenFor(Email));
319
+ // }, 15000);
320
+
321
+ // afterAll(async () => {
322
+ // await rateLimitServer.shutdown();
323
+ // });
324
+
325
+ // beforeEach(async () => {
326
+ // await rateLimitEmailRepo.deleteAll({});
327
+ // mockSendMail.mockReset();
328
+ // mockSendMail.mockResolvedValue(sentMessageInfo());
329
+ // });
330
+
331
+ // it('returns false and sends nothing when the rate limit is exhausted', async () => {
332
+ // // Exhaust the limit: queue and send 3 emails (maxEmailsPerHour: 3)
333
+ // for (let i = 0; i < 3; i++) {
334
+ // await rateLimitEmailService.queue(makeQueueData({ subject: `Email ${i}` }));
335
+ // }
336
+ // await rateLimitEmailService.sendQueuedEmails();
337
+ // expect(mockSendMail).toHaveBeenCalledTimes(3);
338
+
339
+ // // Queue one more — rate limiter should now block it
340
+ // await rateLimitEmailService.queue(makeQueueData({ subject: 'One too many' }));
341
+ // mockSendMail.mockReset();
342
+
343
+ // const result = await rateLimitEmailService.sendQueuedEmails();
344
+
345
+ // expect(result).toBe(false);
346
+ // expect(mockSendMail).not.toHaveBeenCalled();
347
+ // });
348
+ // });
349
+
350
+ describe('EmailService attachment validation', () => {
351
+ it('throws when an attachment path does not exist', async () => {
352
+ await emailService.queue(makeQueueData({
353
+ persist: true,
354
+ attachments: [{ filename: 'missing.pdf', path: '/nonexistent/missing.pdf' as never }]
355
+ }));
356
+ mockSendMail.mockResolvedValue(sentMessageInfo());
357
+
358
+ await expect(emailService.sendQueuedEmails()).rejects.toThrow('does not exist');
359
+ });
360
+
361
+ it('does not throw when attachments is undefined', async () => {
362
+ await emailService.queue(makeQueueData({ persist: true }));
363
+
364
+ await expect(emailService.sendQueuedEmails()).resolves.not.toThrow();
365
+ });
366
+
367
+ it('does not throw when attachments is an empty array', async () => {
368
+ await emailService.queue(makeQueueData({ persist: true, attachments: [] }));
369
+
370
+ await expect(emailService.sendQueuedEmails()).resolves.not.toThrow();
371
+ });
372
+
373
+ it('does not throw when attachment path exists', async () => {
374
+ // __filename is always a real path
375
+ await emailService.queue(makeQueueData({
376
+ persist: true,
377
+ attachments: [{ filename: 'test.ts', path: __filename as never }]
378
+ }));
379
+
380
+ await expect(emailService.sendQueuedEmails()).resolves.not.toThrow();
381
+ });
382
+ });
@@ -11,7 +11,6 @@ import { InjectRepository } from '../di/decorators/inject-repository.decorator';
11
11
  import { Inject } from '../di/decorators/inject.decorator';
12
12
  import { Injectable } from '../di/decorators/injectable.decorator';
13
13
  import { ZIBRI_DI_TOKENS } from '../di/default/zibri-di-tokens.default';
14
- import { inject } from '../di/inject.function';
15
14
  import { type LoggerInterface } from '../logging/logger.interface';
16
15
  import { RateLimiter } from '../rate-limiting/rate-limiter';
17
16
  import { FsUtilities } from '../utilities/fs.utilities';
@@ -44,9 +43,10 @@ export class EmailService implements EmailServiceInterface, OnAppInit, OnAppShut
44
43
  @Inject(ZIBRI_DI_TOKENS.LOGGER)
45
44
  private readonly logger: LoggerInterface,
46
45
  @InjectRepository(Email)
47
- private readonly emailRepository: Repository<Email, CreateEmailData>
46
+ private readonly emailRepository: Repository<Email, CreateEmailData>,
47
+ @Inject(ZIBRI_DI_TOKENS.EMAIL_CONFIG)
48
+ config: EmailConfigInput | undefined
48
49
  ) {
49
- const config: EmailConfigInput | undefined = inject(ZIBRI_DI_TOKENS.EMAIL_CONFIG);
50
50
  if (!config) {
51
51
  throw new Error('no email config was provided for the token "ZIBRI_DI_TOKENS.MAIL_CONFIG"');
52
52
  }
@@ -158,8 +158,7 @@ export class EmailService implements EmailServiceInterface, OnAppInit, OnAppShut
158
158
  status: EmailStatus.QUEUED,
159
159
  priority
160
160
  },
161
- take: amount,
162
- order: { createdAt: 'ASC' }
161
+ take: amount
163
162
  });
164
163
  }
165
164
  }
@@ -8,7 +8,7 @@ import { Property } from '../../entity/decorators/property.decorator';
8
8
  /**
9
9
  * Definition of a Email.
10
10
  */
11
- @Entity()
11
+ @Entity({ defaultOrder: { createdAt: 'ASC' } })
12
12
  export class Email extends BaseEntity {
13
13
  /**
14
14
  * The createdAt date. Is set to now by default.
@@ -1,5 +1,8 @@
1
+ import { FindOptionsOrder, FindOptionsOrderValue, FindOptionsRelations } from 'typeorm';
2
+
1
3
  import { GlobalRegistry } from '../../global/global-registry';
2
4
  import { Newable } from '../../types/newable.type';
5
+ import { OmitStrict } from '../../types/omit-strict.type';
3
6
  import { MetadataUtilities } from '../../utilities/metadata.utilities';
4
7
  import { toSnakeCase } from '../../utilities/to-snake-case.function';
5
8
  import { type BaseEntity } from '../base-entity.model';
@@ -15,19 +18,55 @@ export type EntityMetadata = {
15
18
  /**
16
19
  * Whether or not this entity is allowed to exist without belonging to a data source.
17
20
  */
18
- allowOrphan: boolean
21
+ allowOrphan: boolean,
22
+ /**
23
+ * Default ordering to apply to results when nothing has been specified.
24
+ */
25
+ // eslint-disable-next-line typescript/no-explicit-any
26
+ defaultOrder?: FindOptionsOrder<any>,
27
+ /**
28
+ * Default relations to include in results when nothing has been specified.
29
+ */
30
+ // eslint-disable-next-line typescript/no-explicit-any
31
+ defaultRelations?: FindOptionsRelations<any>
32
+ };
33
+
34
+ // eslint-disable-next-line jsdoc/require-jsdoc
35
+ type EntityMetadataInput<
36
+ TOrderKey extends string = never,
37
+ TRelationKey extends string = never
38
+ > = OmitStrict<Partial<EntityMetadata>, 'defaultOrder' | 'defaultRelations'> & {
39
+ /**
40
+ * Default ordering to apply to results when nothing has been specified.
41
+ */
42
+ defaultOrder?: { [K in TOrderKey]?: FindOptionsOrderValue },
43
+ /**
44
+ * Default relations to include in results when nothing has been specified.
45
+ */
46
+ // eslint-disable-next-line typescript/no-explicit-any
47
+ defaultRelations?: { [K in TRelationKey]?: FindOptionsRelations<any> | boolean }
19
48
  };
20
49
 
50
+ // eslint-disable-next-line jsdoc/require-jsdoc, typescript/no-explicit-any
51
+ type ConstrainKeys<TKey extends string> = [TKey] extends [never] ? object : Record<TKey, any>;
52
+
53
+ // eslint-disable-next-line jsdoc/require-returns
21
54
  /**
22
55
  * Marks an entity.
23
56
  * @param options - Configuration options for the entity.
24
57
  */
25
- export function Entity(options: Partial<EntityMetadata> = {}): ClassDecorator {
26
- const { tableName, allowOrphan = false } = options;
27
- return target => {
58
+ export function Entity<TOrderKey extends string, TRelationKey extends string>(
59
+ options: EntityMetadataInput<TOrderKey, TRelationKey> = {}
60
+ ) {
61
+ const { tableName, defaultOrder, defaultRelations, allowOrphan = false } = options;
62
+ return <T extends ConstrainKeys<TOrderKey> & ConstrainKeys<TRelationKey>>(
63
+ target: Newable<T>
64
+ ): void => {
28
65
  const metadata: EntityMetadata = {
29
66
  tableName: tableName ?? toSnakeCase(target.name),
30
- allowOrphan
67
+ allowOrphan,
68
+ defaultOrder,
69
+ defaultRelations
31
70
  };
32
71
  MetadataUtilities.setEntityMetadata(target as unknown as Newable<BaseEntity>, metadata);
33
72
  GlobalRegistry.entityClasses.push(target as unknown as Newable<BaseEntity>);