@futdevpro/nts-dynamo 1.15.82 → 1.15.84

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 (365) hide show
  1. package/.c8rc.json +26 -26
  2. package/.copilot/patterns.json +7 -7
  3. package/.cursor/rules/__assistant_guide.mdc +30 -30
  4. package/.cursor/rules/_ag_backend-structure.mdc +85 -85
  5. package/.cursor/rules/_ag_backend.mdc +16 -16
  6. package/.cursor/rules/_ag_frontend-structure.mdc +86 -86
  7. package/.cursor/rules/_ag_frontend.mdc +39 -39
  8. package/.cursor/rules/_ag_import-rules.mdc +44 -44
  9. package/.cursor/rules/_ag_naming.mdc +115 -115
  10. package/.cursor/rules/_ag_should-be.mdc +6 -6
  11. package/.cursor/rules/ai_development_guide.md +60 -60
  12. package/.cursor/rules/cursor-rules.md +160 -160
  13. package/.cursor/rules/default-command.mdc +464 -464
  14. package/.cursor/rules/error_code_pattern.md +39 -39
  15. package/.cursor/rules/saved rule mcp server use.md +15 -15
  16. package/.vscode/settings.json +10 -10
  17. package/HOWTO.md +15 -15
  18. package/LICENSE +21 -21
  19. package/__documentations/nts-integration-tests-2026-03-17.md +26 -26
  20. package/__documentations/plans/BEDROCK-HYPERPLAN.md +95 -95
  21. package/_specifications/BACKLOG.md +92 -92
  22. package/_specifications/TODO.md +15 -15
  23. package/_specifications/agent.md +138 -138
  24. package/build/_modules/server/errors/errors.controller.d.ts +15 -0
  25. package/build/_modules/server/errors/errors.controller.d.ts.map +1 -1
  26. package/build/_modules/server/errors/errors.controller.js +51 -13
  27. package/build/_modules/server/errors/errors.controller.js.map +1 -1
  28. package/build/_modules/server/server-status/server-status.controller.d.ts +8 -0
  29. package/build/_modules/server/server-status/server-status.controller.d.ts.map +1 -1
  30. package/build/_modules/server/server-status/server-status.controller.js +25 -5
  31. package/build/_modules/server/server-status/server-status.controller.js.map +1 -1
  32. package/eslint.config.js +3 -3
  33. package/nodemon.json +24 -24
  34. package/package.json +3 -3
  35. package/pnpm-workspace.yaml +5 -5
  36. package/scripts/run-coverage-tests.js +28 -28
  37. package/spec/support/helpers/spec-reporter-loader.js +359 -359
  38. package/spec/support/helpers/ts-node-helper.js +93 -93
  39. package/spec/support/jasmine.coverage.json +24 -24
  40. package/spec/support/jasmine.json +24 -24
  41. package/src/_collections/archive.util.spec.ts +57 -57
  42. package/src/_collections/archive.util.ts +18 -18
  43. package/src/_collections/atlas-default-db-options.const.ts +9 -9
  44. package/src/_collections/default-fallback-cache-max-age.const.spec.ts +11 -11
  45. package/src/_collections/default-fallback-cache-max-age.const.ts +2 -2
  46. package/src/_collections/default-not-found-page.const.spec.ts +19 -19
  47. package/src/_collections/default-not-found-page.const.ts +22 -22
  48. package/src/_collections/default-socket-path.const.spec.ts +12 -12
  49. package/src/_collections/default-socket-path.const.ts +2 -2
  50. package/src/_collections/get-environment-settings.util.spec.ts +210 -210
  51. package/src/_collections/get-environment-settings.util.ts +48 -48
  52. package/src/_collections/global-settings.const.ts +109 -109
  53. package/src/_collections/sample.env +21 -21
  54. package/src/_collections/star.controller.spec.ts +224 -224
  55. package/src/_collections/star.controller.ts +129 -129
  56. package/src/_enums/data-model-type.enum.ts +14 -14
  57. package/src/_enums/data-service-function.enum.ts +24 -24
  58. package/src/_enums/predefined-data-types.enum.ts +16 -16
  59. package/src/_enums/route-security.enum.ts +12 -12
  60. package/src/_models/control-models/api-call-params.control-model.spec.ts +152 -152
  61. package/src/_models/control-models/api-call-params.control-model.ts +142 -142
  62. package/src/_models/control-models/app-ext-system-controls.control-model.spec.ts +52 -52
  63. package/src/_models/control-models/app-ext-system-controls.control-model.ts +9 -9
  64. package/src/_models/control-models/app-params.control-model.spec.ts +225 -225
  65. package/src/_models/control-models/app-params.control-model.ts +136 -136
  66. package/src/_models/control-models/app-system-controls.control-model.spec.ts +31 -31
  67. package/src/_models/control-models/app-system-controls.control-model.ts +9 -9
  68. package/src/_models/control-models/endpoint-params.control-model.spec.ts +627 -627
  69. package/src/_models/control-models/endpoint-params.control-model.ts +627 -627
  70. package/src/_models/control-models/http-settings.control-model.spec.ts +77 -77
  71. package/src/_models/control-models/http-settings.control-model.ts +37 -37
  72. package/src/_models/control-models/system-control.control-model.spec.ts +27 -27
  73. package/src/_models/control-models/system-control.control-model.ts +12 -12
  74. package/src/_models/interfaces/certification-settings.interface.ts +7 -7
  75. package/src/_models/interfaces/environment-settings.interface.ts +59 -59
  76. package/src/_models/interfaces/global-log-settings.interface.ts +171 -171
  77. package/src/_models/interfaces/global-service-settings.interface.ts +47 -47
  78. package/src/_models/interfaces/global-settings.interface.ts +235 -235
  79. package/src/_models/interfaces/routing-module-settings.interface.ts +21 -21
  80. package/src/_models/interfaces/static-client-settings.interface.spec.ts +29 -29
  81. package/src/_models/interfaces/static-client-settings.interface.ts +28 -28
  82. package/src/_models/types/db-update.type.ts +100 -100
  83. package/src/_modules/ai/_models/ai-input-interfaces.ts +117 -117
  84. package/src/_modules/ai/_models/ai-test-generation-result.interface.ts +16 -16
  85. package/src/_modules/ai/_modules/anthropic/_services/aai-user-key.control-service.ts +138 -138
  86. package/src/_modules/ai/_modules/anthropic/index.ts +5 -5
  87. package/src/_modules/ai/_modules/document-ai/_collections/dai-chunking.util.spec.ts +242 -242
  88. package/src/_modules/ai/_modules/document-ai/_collections/dai-chunking.util.ts +639 -639
  89. package/src/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.spec.ts +295 -295
  90. package/src/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.ts +518 -518
  91. package/src/_modules/ai/_modules/document-ai/_collections/dai-document.util.spec.ts +209 -209
  92. package/src/_modules/ai/_modules/document-ai/_collections/dai-document.util.ts +85 -85
  93. package/src/_modules/ai/_modules/document-ai/_enums/dai-compare-result-type.enum.ts +7 -7
  94. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-doc-chunk.data-model.ts +146 -146
  95. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-doc-page.data-model.ts +162 -162
  96. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-document.data-model.ts +99 -99
  97. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.ts +68 -68
  98. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-doc-chunk-compare-result.interface.ts +18 -18
  99. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-doc-page-compare-result.interface.ts +19 -19
  100. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-document-compare-result.interface.ts +25 -25
  101. package/src/_modules/ai/_modules/document-ai/index.ts +30 -30
  102. package/src/_modules/ai/_modules/fdp-ai/_services/fdpai-user-key.control-service.ts +189 -189
  103. package/src/_modules/ai/_modules/fdp-ai/index.ts +5 -5
  104. package/src/_modules/ai/_modules/open-ai/_collections/oai-global-settings.const.ts +9 -9
  105. package/src/_modules/ai/_modules/open-ai/_collections/oai-llm-predefined-requests-hu.conts.ts +82 -82
  106. package/src/_modules/ai/_modules/open-ai/_collections/oai-llm-predefined-requests.conts.ts +75 -75
  107. package/src/_modules/ai/_modules/open-ai/_enums/oai-gpt-message-role.enum.ts +45 -45
  108. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-global-settings.interface.ts +7 -7
  109. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-gpt-message.interface.ts +7 -7
  110. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-llm-predefined-requests.interface.ts +57 -57
  111. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-doc-chunk-data.service.ts +292 -292
  112. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-document.data-service.spec.ts +342 -342
  113. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-vector-data.service.spec.ts +550 -550
  114. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-vector-data.service.ts +630 -630
  115. package/src/_modules/ai/_modules/open-ai/_services/oai-embedding.control-service.spec.ts +332 -332
  116. package/src/_modules/ai/_modules/open-ai/_services/oai-llm-chat.service-base.spec.ts +462 -462
  117. package/src/_modules/ai/_modules/open-ai/_services/oai-llm-chat.service-base.ts +634 -634
  118. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.spec.ts +489 -489
  119. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.tools.spec.ts +173 -173
  120. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.ts +1033 -1033
  121. package/src/_modules/ai/_modules/open-ai/_services/oai-user-key.control-service.ts +157 -157
  122. package/src/_modules/ai/_services/ai-embedding-mock.service.spec.ts +115 -115
  123. package/src/_modules/ai/_services/ai-embedding-mock.service.ts +212 -212
  124. package/src/_modules/ai/_services/ai-embedding-provider.registry.spec.ts +110 -110
  125. package/src/_modules/ai/_services/ai-embedding-provider.registry.ts +110 -110
  126. package/src/_modules/ai/_services/ai-embedding.service-base.spec.ts +98 -98
  127. package/src/_modules/ai/_services/ai-embedding.service-base.ts +48 -48
  128. package/src/_modules/ai/_services/ai-llm-chat.service-base.spec.ts +229 -229
  129. package/src/_modules/ai/_services/ai-llm-chat.service-base.ts +68 -68
  130. package/src/_modules/ai/_services/ai-llm.service-base.spec.ts +250 -250
  131. package/src/_modules/ai/_services/ai-llm.service-base.ts +519 -519
  132. package/src/_modules/ai/_services/ai-provider.service-base.spec.ts +158 -158
  133. package/src/_modules/ai/_services/ai-user-key.service-base.ts +59 -59
  134. package/src/_modules/ai/_services/lmstudio-embedding.control-service.spec.ts +197 -197
  135. package/src/_modules/ai/_services/lmstudio-embedding.control-service.ts +371 -371
  136. package/src/_modules/ai/index.ts +23 -23
  137. package/src/_modules/assistant/_collections/ass-global-settings.const.ts +13 -13
  138. package/src/_modules/assistant/_collections/ass.util.spec.ts +176 -176
  139. package/src/_modules/assistant/_collections/ass.util.ts +50 -50
  140. package/src/_modules/assistant/_models/ass-global-settings.interface.ts +15 -15
  141. package/src/_modules/assistant/_services/ass-io.control-service.spec.ts +140 -140
  142. package/src/_modules/assistant/_services/ass-main.control-service.spec.ts +192 -192
  143. package/src/_modules/assistant/_services/ass-main.control-service.ts +107 -107
  144. package/src/_modules/bot/_collections/bot-default-commands.const.ts +12 -12
  145. package/src/_modules/bot/_collections/bot-global-settings.const.ts +39 -39
  146. package/src/_modules/bot/_models/bot-channel-wrapper.interface.ts +62 -62
  147. package/src/_modules/bot/_models/bot-command.interface.ts +8 -8
  148. package/src/_modules/bot/_models/bot-global-settings.interface.ts +96 -96
  149. package/src/_modules/bot/_models/bot-last-mention-date.interface.ts +6 -6
  150. package/src/_modules/bot/_models/bot-last-message-date.interface.ts +5 -5
  151. package/src/_modules/bot/_models/bot-user-wrapper.interface.ts +41 -41
  152. package/src/_modules/bot/_modules/discord-bot/_models/dib-platform.types.ts +9 -9
  153. package/src/_modules/bot/_modules/discord-bot/_services/dib-messaging-provider.control-service.spec.ts +431 -431
  154. package/src/_modules/bot/_modules/dynamo-bot/_collections/dyb-operations.util.spec.ts +160 -160
  155. package/src/_modules/bot/_modules/dynamo-bot/_collections/dyb-operations.util.ts +55 -55
  156. package/src/_modules/bot/_modules/dynamo-bot/_models/dyb-platform.types.ts +15 -15
  157. package/src/_modules/bot/_modules/dynamo-bot/_services/dyb-messaging-provider.control-service.spec.ts +374 -374
  158. package/src/_modules/bot/_modules/dynamo-bot/_services/dyb-messaging-provider.control-service.ts +447 -447
  159. package/src/_modules/bot/_modules/dynamo-bot/index.ts +15 -15
  160. package/src/_modules/bot/_modules/slack-bot/_models/slb-platform.types.ts +9 -9
  161. package/src/_modules/bot/_modules/slack-bot/_services/slb-messaging-provider.control-service.spec.ts +344 -344
  162. package/src/_modules/bot/_modules/slack-bot/_services/slb-messaging-provider.control-service.ts +197 -197
  163. package/src/_modules/bot/_modules/teams-bot/_models/teb-platform.types.ts +9 -9
  164. package/src/_modules/bot/_modules/teams-bot/_services/teb-messaging-provider.control-service.spec.ts +345 -345
  165. package/src/_modules/bot/_modules/teams-bot/_services/teb-messaging-provider.control-service.ts +197 -197
  166. package/src/_modules/bot/_services/bot-commands.control-service.spec.ts +116 -116
  167. package/src/_modules/bot/_services/bot-io.control-service.spec.ts +285 -285
  168. package/src/_modules/bot/_services/bot-main.control-service.spec.ts +208 -208
  169. package/src/_modules/bot/_services/bot-messaging-provider.service-base.spec.ts +349 -349
  170. package/src/_modules/bot/_services/bot-routines.control-service.spec.ts +111 -111
  171. package/src/_modules/custom-data/custom-data.controller.spec.ts +49 -49
  172. package/src/_modules/custom-data/custom-data.controller.ts +67 -67
  173. package/src/_modules/custom-data/custom-data.data-service.spec.ts +54 -54
  174. package/src/_modules/custom-data/custom-data.data-service.ts +21 -21
  175. package/src/_modules/custom-data/get-custom-data-routing-module.util.spec.ts +28 -28
  176. package/src/_modules/custom-data/get-custom-data-routing-module.util.ts +24 -24
  177. package/src/_modules/custom-data/index.ts +9 -9
  178. package/src/_modules/data-readers/_collections/dynts-sqlite-reader.util.spec.ts +161 -161
  179. package/src/_modules/data-readers/_collections/dynts-sqlite-reader.util.ts +203 -203
  180. package/src/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.ts +33 -33
  181. package/src/_modules/data-readers/index.ts +11 -11
  182. package/src/_modules/defaults/_collections/default-endpoints.util.ts +487 -487
  183. package/src/_modules/defaults/_models/default-user.data-model.ts +72 -72
  184. package/src/_modules/defaults/_services/default-auth.service.spec.ts +269 -269
  185. package/src/_modules/defaults/_services/default-auth.service.ts +177 -177
  186. package/src/_modules/defaults/_services/default-socket-events.service.spec.ts +42 -42
  187. package/src/_modules/defaults/_services/default-socket-events.service.ts +61 -61
  188. package/src/_modules/defaults/_services/default-user.data-service.spec.ts +187 -187
  189. package/src/_modules/defaults/_services/default-user.data-service.ts +98 -98
  190. package/src/_modules/defaults/index.ts +17 -17
  191. package/src/_modules/discord-assistant/_collections/dias-global-settings.const.ts +19 -19
  192. package/src/_modules/discord-assistant/_collections/dias.util.spec.ts +366 -366
  193. package/src/_modules/discord-assistant/_collections/dias.util.ts +132 -132
  194. package/src/_modules/discord-assistant/_models/dias-global-settings.interface.ts +19 -19
  195. package/src/_modules/discord-assistant/_models/dias-knowledge.data-model.ts +52 -52
  196. package/src/_modules/discord-assistant/_services/dias-chunk.data-service.ts +177 -177
  197. package/src/_modules/discord-assistant/_services/dias-io.control-service.spec.ts +108 -108
  198. package/src/_modules/discord-assistant/_services/dias-io.control-service.ts +69 -69
  199. package/src/_modules/discord-assistant/_services/dias-main.control-service.spec.ts +22 -22
  200. package/src/_modules/discord-assistant/_services/dias-main.control-service.ts +27 -27
  201. package/src/_modules/discord-assistant/_services/dias.service-base.spec.ts +195 -195
  202. package/src/_modules/discord-assistant/_services/dias.service-base.ts +76 -76
  203. package/src/_modules/discord-assistant/index.ts +38 -38
  204. package/src/_modules/discord-assistant-voiced/_services/dias-discord-bot.control-service.spec.ts +34 -34
  205. package/src/_modules/discord-assistant-voiced/_services/dias-discord-bot.control-service.ts +11 -11
  206. package/src/_modules/discord-assistant-voiced/index.ts +36 -36
  207. package/src/_modules/discord-bot/_collections/dibo-default-commands.const.ts +16 -16
  208. package/src/_modules/discord-bot/_collections/dibo-global-settings.conts.ts +55 -55
  209. package/src/_modules/discord-bot/_collections/dibo-operations.util.spec.ts +214 -214
  210. package/src/_modules/discord-bot/_collections/dibo-operations.util.ts +387 -387
  211. package/src/_modules/discord-bot/_models/dibo-command.interface.ts +12 -12
  212. package/src/_modules/discord-bot/_models/dibo-global-settings.interface.ts +98 -98
  213. package/src/_modules/discord-bot/_models/dibo-last-mention-date.inteface.ts +7 -7
  214. package/src/_modules/discord-bot/_models/dibo-last-message-date.interface.ts +6 -6
  215. package/src/_modules/discord-bot/_services/dibo-commands.control-service.spec.ts +154 -154
  216. package/src/_modules/discord-bot/_services/dibo-commands.control-service.ts +153 -153
  217. package/src/_modules/discord-bot/_services/dibo-io.control-service.spec.ts +264 -264
  218. package/src/_modules/discord-bot/_services/dibo-io.control-service.ts +306 -306
  219. package/src/_modules/discord-bot/_services/dibo-main.control-service.spec.ts +408 -408
  220. package/src/_modules/discord-bot/_services/dibo-main.control-service.ts +487 -487
  221. package/src/_modules/discord-bot/_services/dibo-routines.control-service.spec.ts +105 -105
  222. package/src/_modules/discord-bot/index.ts +36 -36
  223. package/src/_modules/local-vector-search/_enums/lvs-search-mode.enum.ts +35 -35
  224. package/src/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.ts +59 -59
  225. package/src/_modules/local-vector-search/_models/lvs-search-result.interface.ts +17 -17
  226. package/src/_modules/local-vector-search/_services/lvs-doc-chunk-data.service.spec.ts +418 -418
  227. package/src/_modules/local-vector-search/_services/lvs-doc-chunk-data.service.ts +276 -276
  228. package/src/_modules/local-vector-search/_services/lvs-local-vector-search.data-service.spec.ts +480 -480
  229. package/src/_modules/local-vector-search/_services/lvs-local-vector-search.data-service.ts +416 -416
  230. package/src/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.spec.ts +198 -198
  231. package/src/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.ts +146 -146
  232. package/src/_modules/local-vector-search/_services/lvs-vector-persist.data-service.spec.ts +167 -167
  233. package/src/_modules/local-vector-search/_services/lvs-vector-persist.data-service.ts +106 -106
  234. package/src/_modules/local-vector-search/_services/lvs-vector-pool.control-service.spec.ts +507 -507
  235. package/src/_modules/local-vector-search/_services/lvs-vector-pool.control-service.ts +272 -272
  236. package/src/_modules/local-vector-search/index.ts +16 -16
  237. package/src/_modules/logs/index.ts +11 -11
  238. package/src/_modules/mcp/_models/interfaces/dynts-mcp.interface.ts +111 -111
  239. package/src/_modules/mcp/_services/dynts-mcp-server.service-base.spec.ts +142 -142
  240. package/src/_modules/mcp/_services/dynts-mcp-server.service-base.ts +120 -120
  241. package/src/_modules/mcp/_services/dynts-mcp.adapter.ts +168 -168
  242. package/src/_modules/mcp/index.ts +13 -13
  243. package/src/_modules/messaging/README.md +354 -354
  244. package/src/_modules/messaging/_collections/get-messaging-routing-module.util.ts +26 -26
  245. package/src/_modules/messaging/_collections/msg-global-settings.const.ts +22 -22
  246. package/src/_modules/messaging/_collections/msg.util.spec.ts +226 -226
  247. package/src/_modules/messaging/_models/msg-global-settings.interface.ts +37 -37
  248. package/src/_modules/messaging/_services/msg-conversation.data-service.ts +146 -146
  249. package/src/_modules/messaging/_services/msg-events.service.spec.ts +219 -219
  250. package/src/_modules/messaging/_services/msg-events.service.ts +267 -267
  251. package/src/_modules/messaging/_services/msg-integration.control-service.ts +179 -179
  252. package/src/_modules/messaging/_services/msg-main.control-service.spec.ts +147 -147
  253. package/src/_modules/messaging/_services/msg-main.control-service.ts +571 -571
  254. package/src/_modules/messaging/_services/msg-message.data-service.ts +129 -129
  255. package/src/_modules/messaging/_services/msg.controller.spec.ts +201 -201
  256. package/src/_modules/messaging/index.ts +30 -30
  257. package/src/_modules/mock/app-extended-server.mock.ts +201 -201
  258. package/src/_modules/mock/app-integration-test.mock.ts +51 -51
  259. package/src/_modules/mock/app-params.mock.spec.ts +21 -21
  260. package/src/_modules/mock/app-params.mock.ts +9 -9
  261. package/src/_modules/mock/app-server.mock.ts +188 -188
  262. package/src/_modules/mock/auth-service.mock.spec.ts +47 -47
  263. package/src/_modules/mock/auth-service.mock.ts +28 -28
  264. package/src/_modules/mock/controller.mock.spec.ts +26 -26
  265. package/src/_modules/mock/controller.mock.ts +16 -16
  266. package/src/_modules/mock/data-model.mock.spec.ts +111 -111
  267. package/src/_modules/mock/data-model.mock.ts +82 -82
  268. package/src/_modules/mock/email-service-collection.mock.spec.ts +24 -24
  269. package/src/_modules/mock/email-service-collection.mock.ts +15 -15
  270. package/src/_modules/mock/email-service.mock.spec.ts +17 -17
  271. package/src/_modules/mock/email-service.mock.ts +20 -20
  272. package/src/_modules/mock/email-template.mock.html +14 -14
  273. package/src/_modules/mock/endpoint.mock.ts +91 -91
  274. package/src/_modules/mock/socket-client.mock.spec.ts +40 -40
  275. package/src/_modules/mock/socket-client.mock.ts +45 -45
  276. package/src/_modules/mock/socket-server.mock.spec.ts +44 -44
  277. package/src/_modules/mock/socket-server.mock.ts +46 -46
  278. package/src/_modules/oauth2/_routes/oauth2.controller.spec.ts +107 -107
  279. package/src/_modules/oauth2/_routes/oauth2.controller.ts +98 -98
  280. package/src/_modules/oauth2/_services/oauth2.auth-service.spec.ts +254 -254
  281. package/src/_modules/oauth2/_services/oauth2.auth-service.ts +232 -232
  282. package/src/_modules/oauth2/_services/oauth2.control-service.spec.ts +585 -585
  283. package/src/_modules/oauth2/_services/oauth2.control-service.ts +653 -653
  284. package/src/_modules/oauth2/index.ts +17 -17
  285. package/src/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.ts +22 -22
  286. package/src/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.ts +81 -81
  287. package/src/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.ts +107 -107
  288. package/src/_modules/scoped-config/_services/dynts-scoped-config.control-service.spec.ts +306 -306
  289. package/src/_modules/scoped-config/_services/dynts-scoped-config.control-service.ts +295 -295
  290. package/src/_modules/scoped-config/_services/dynts-scoped-config.data-service.spec.ts +118 -118
  291. package/src/_modules/scoped-config/_services/dynts-scoped-config.data-service.ts +105 -105
  292. package/src/_modules/scoped-config/index.ts +17 -17
  293. package/src/_modules/server/errors/errors.control-service.spec.ts +238 -238
  294. package/src/_modules/server/errors/errors.control-service.ts +100 -100
  295. package/src/_modules/server/errors/errors.controller.spec.ts +268 -249
  296. package/src/_modules/server/errors/errors.controller.ts +527 -489
  297. package/src/_modules/server/errors/errors.data-service.spec.ts +480 -480
  298. package/src/_modules/server/index.ts +30 -30
  299. package/src/_modules/server/server-status/server-status-snapshot.control-service.spec.ts +70 -70
  300. package/src/_modules/server/server-status/server-status-snapshot.control-service.ts +17 -17
  301. package/src/_modules/server/server-status/server-status-snapshot.data-service.spec.ts +77 -77
  302. package/src/_modules/server/server-status/server-status-snapshot.data-service.ts +37 -37
  303. package/src/_modules/server/server-status/server-status.control-service.spec.ts +576 -576
  304. package/src/_modules/server/server-status/server-status.control-service.ts +396 -396
  305. package/src/_modules/server/server-status/server-status.controller.spec.ts +255 -248
  306. package/src/_modules/server/server-status/server-status.controller.ts +272 -253
  307. package/src/_modules/socket/_enums/socket-security.enum.ts +11 -11
  308. package/src/_modules/socket/_models/socket-client-service-params.control-model.spec.ts +32 -32
  309. package/src/_modules/socket/_models/socket-client-service-params.control-model.ts +22 -22
  310. package/src/_modules/socket/_models/socket-presence.control-model.spec.ts +164 -164
  311. package/src/_modules/socket/_models/socket-presence.control-model.ts +210 -210
  312. package/src/_modules/socket/_models/socket-server-service-params.control-model.spec.ts +46 -46
  313. package/src/_modules/socket/_models/socket-server-service-params.control-model.ts +22 -22
  314. package/src/_modules/socket/_services/socket-client.service.spec.ts +15 -15
  315. package/src/_modules/socket/_services/socket-client.service.ts +260 -260
  316. package/src/_modules/socket/_services/socket-server.service.spec.ts +11 -11
  317. package/src/_modules/socket/app-extended.integration.spec.ts +85 -85
  318. package/src/_modules/socket/app-extended.server.ts +630 -630
  319. package/src/_modules/socket/index.ts +42 -42
  320. package/src/_modules/test/get-test-routing-module.util.spec.ts +28 -28
  321. package/src/_modules/test/get-test-routing-module.util.ts +23 -23
  322. package/src/_modules/test/index.ts +11 -11
  323. package/src/_modules/test/test.controller.spec.ts +72 -72
  324. package/src/_modules/test/test.controller.ts +115 -115
  325. package/src/_modules/usage/get-usage-routing-module.util.ts +22 -22
  326. package/src/_modules/usage/index.ts +15 -15
  327. package/src/_modules/usage/usage.controller.spec.ts +81 -81
  328. package/src/_modules/usage/usage.controller.ts +126 -126
  329. package/src/_modules/usage/usage.data-service.spec.ts +332 -332
  330. package/src/_modules/usage/usage.data-service.ts +185 -185
  331. package/src/_services/base/api.service-base.spec.ts +125 -125
  332. package/src/_services/base/api.service-base.ts +74 -74
  333. package/src/_services/base/archive-data.service.spec.ts +209 -209
  334. package/src/_services/base/archive-data.service.ts +224 -224
  335. package/src/_services/base/data.service.spec.ts +729 -729
  336. package/src/_services/base/data.service.ts +2740 -2740
  337. package/src/_services/base/db.service.spec.ts +73 -73
  338. package/src/_services/base/db.service.ts +1575 -1575
  339. package/src/_services/base/singleton.service-base.spec.ts +28 -28
  340. package/src/_services/base/singleton.service-base.ts +24 -24
  341. package/src/_services/base/singleton.service.spec.ts +114 -114
  342. package/src/_services/base/singleton.service.ts +38 -38
  343. package/src/_services/core/api.service.spec.ts +140 -140
  344. package/src/_services/core/auth.service.spec.ts +159 -159
  345. package/src/_services/core/auth.service.ts +174 -174
  346. package/src/_services/core/email.service.spec.ts +85 -85
  347. package/src/_services/core/email.service.ts +742 -742
  348. package/src/_services/core/global.service.spec.ts +292 -292
  349. package/src/_services/core/global.service.ts +487 -487
  350. package/src/_services/core/memory-guard.service.spec.ts +245 -245
  351. package/src/_services/core/memory-guard.service.ts +481 -481
  352. package/src/_services/core/service-collection.service.spec.ts +46 -46
  353. package/src/_services/core/service-collection.service.ts +6 -6
  354. package/src/_services/route/controller.service.spec.ts +53 -53
  355. package/src/_services/route/controller.service.ts +148 -148
  356. package/src/_services/route/routing-module.service.spec.ts +98 -98
  357. package/src/_services/route/routing-module.service.ts +330 -330
  358. package/src/_services/server/app.server.ts +1905 -1905
  359. package/src/_services/shared.static-service.spec.ts +99 -99
  360. package/src/_services/shared.static-service.ts +78 -78
  361. package/src/index.ts +97 -97
  362. package/tsconfig.app.json +12 -12
  363. package/tsconfig.json +42 -42
  364. package/.dynamo/logs/cicd-pipeline/output.log +0 -2867
  365. package/.dynamo/logs/cicd-pipeline/status.json +0 -94
@@ -1,729 +1,729 @@
1
-
2
- import { DyNTS_DataService } from './data.service';
3
- import { DyNTS_DBService } from './db.service';
4
- import { DyNTS_GlobalService } from '../core/global.service';
5
- import { DyNTS_ArchiveDataService } from './archive-data.service';
6
- import { DyFM_Metadata, DyFM_DataModel_Params, DyFM_Error, DyFM_DBFilter, DyFM_SearchQuery, DyFM_BasicProperty_Type, DyFM_EnvironmentFlag, DyFM_Log } from '@futdevpro/fsm-dynamo';
7
- import { DyNTS_global_settings } from '../../_collections/global-settings.const';
8
-
9
- // Initialize global settings before any test runs
10
- beforeAll(() => {
11
- if (!DyNTS_global_settings.systemShortCodeName) {
12
- (DyNTS_global_settings as any).systemShortCodeName = 'TEST';
13
- }
14
- if (!DyNTS_global_settings.env_settings) {
15
- (DyNTS_global_settings as any).env_settings = {
16
- environment: DyFM_EnvironmentFlag.local,
17
- };
18
- }
19
- });
20
-
21
- class TestMetadata extends DyFM_Metadata {
22
- name: string;
23
- email?: string;
24
- userId?: string;
25
- }
26
-
27
- const testDataParams = new DyFM_DataModel_Params<TestMetadata>({
28
- dataName: 'test_data',
29
- properties: {
30
- name: {
31
- key: 'name',
32
- type: DyFM_BasicProperty_Type.string,
33
- required: true,
34
- },
35
- email: {
36
- key: 'email',
37
- type: DyFM_BasicProperty_Type.string,
38
- },
39
- userId: {
40
- key: 'userId',
41
- type: DyFM_BasicProperty_Type.string,
42
- dependencyDataName: 'user_data',
43
- },
44
- },
45
- });
46
-
47
- describe('| DyNTS_DataService', () => {
48
- let mockDBService: jasmine.SpyObj<DyNTS_DBService<TestMetadata>>;
49
- let service: DyNTS_DataService<TestMetadata>;
50
- let testData: TestMetadata;
51
-
52
- beforeEach(() => {
53
- testData = new TestMetadata({
54
- _id: 'test-id-123',
55
- name: 'Test Name',
56
- email: 'test@example.com',
57
- } as TestMetadata);
58
-
59
- mockDBService = jasmine.createSpyObj('DyNTS_DBService', [
60
- 'getAll',
61
- 'getDataById',
62
- 'getDataListByIds',
63
- 'getDataByDependencyId',
64
- 'getDataListByDependencyIds',
65
- 'getDataListByDependencyId',
66
- 'findOne',
67
- 'find',
68
- 'updateOne',
69
- 'createData',
70
- 'modifyData',
71
- 'markDeletedById',
72
- 'trueDeleteDataById',
73
- 'trueDeleteAllData',
74
- 'restoreDeletedById',
75
- ]);
76
-
77
- spyOn(DyNTS_GlobalService, 'getDBService').and.returnValue(mockDBService);
78
- spyOn(DyNTS_GlobalService, 'getDBServiceByKey').and.returnValue(mockDBService);
79
-
80
- service = new DyNTS_DataService<TestMetadata>(
81
- testData,
82
- testDataParams,
83
- 'test-issuer'
84
- );
85
- });
86
-
87
- it('| should create service instance', () => {
88
- expect(service).toBeDefined();
89
- expect(service.data).toBe(testData);
90
- expect(service.dataParams).toBe(testDataParams);
91
- expect(service.issuer).toBe('test-issuer');
92
- expect(service.dataDBService).toBe(mockDBService);
93
- });
94
-
95
- it('| should set service name from constructor', () => {
96
- expect(service.serviceName).toBe('DyNTS_DataService');
97
- });
98
-
99
- it('| should have default error messages', () => {
100
- expect(service.defaultErrorUserMsg).toContain('Data Service Error');
101
- expect(service.defaultValidationErrorUserMsg).toContain('Validation Error');
102
- });
103
-
104
- it('| should throw error when DB service not found', () => {
105
- (DyNTS_GlobalService.getDBService as jasmine.Spy).and.throwError('DB Service not found');
106
-
107
- expect(() => {
108
- new DyNTS_DataService<TestMetadata>(testData, testDataParams, 'test-issuer');
109
- }).toThrow();
110
- });
111
-
112
- describe('| getAll', () => {
113
- it('| should get all data from database', async () => {
114
- const mockDataList = [testData, { ...testData, _id: 'test-id-456' }];
115
- mockDBService.getAll.and.returnValue(Promise.resolve(mockDataList));
116
-
117
- const result = await service.getAll();
118
-
119
- expect(result).toEqual(mockDataList);
120
- expect(service.dataList).toEqual(mockDataList);
121
- expect(mockDBService.getAll).toHaveBeenCalled();
122
- });
123
-
124
- it('| should not set to service when dontSetToService is true', async () => {
125
- const mockDataList = [testData];
126
- mockDBService.getAll.and.returnValue(Promise.resolve(mockDataList));
127
-
128
- const result = await service.getAll(true);
129
-
130
- expect(result).toEqual(mockDataList);
131
- expect(service.dataList).not.toEqual(mockDataList);
132
- });
133
- });
134
-
135
- describe('| getDataById', () => {
136
- it('| should get data by id', async () => {
137
- mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
138
-
139
- const result = await service.getDataById('test-id-123');
140
-
141
- expect(result).toBe(testData);
142
- expect(service.data).toBe(testData);
143
- expect(mockDBService.getDataById).toHaveBeenCalledWith('test-id-123');
144
- });
145
-
146
- it('| should use data._id when id not provided', async () => {
147
- mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
148
-
149
- const result = await service.getDataById();
150
-
151
- expect(mockDBService.getDataById).toHaveBeenCalledWith('test-id-123');
152
- });
153
-
154
- it('| should throw error when id is missing', async () => {
155
- service.data = new TestMetadata({});
156
-
157
- await expectAsync(
158
- service.getDataById()
159
- ).toBeRejected();
160
- });
161
-
162
- it('| should load from archive when not found and archive enabled', async () => {
163
- service.haveArchiveDataService = true;
164
- mockDBService.getDataById.and.returnValue(Promise.resolve(null));
165
- const mockArchiveService = jasmine.createSpyObj('DyNTS_ArchiveDataService', ['getDataByOriginalId']);
166
- mockArchiveService.getDataByOriginalId.and.returnValue(Promise.resolve(testData));
167
- spyOn(service, 'getArchiveDataService').and.returnValue(mockArchiveService);
168
-
169
- const result = await service.getDataById('test-id-123');
170
-
171
- expect(result).toBe(testData);
172
- expect(mockArchiveService.getDataByOriginalId).toHaveBeenCalled();
173
- });
174
-
175
- it('| should not load from archive when skipArchiveLoad is true', async () => {
176
- service.haveArchiveDataService = true;
177
- mockDBService.getDataById.and.returnValue(Promise.resolve(null));
178
- spyOn(service, 'getArchiveDataService');
179
-
180
- const result = await service.getDataById('test-id-123', false, true);
181
-
182
- expect(result).toBeNull();
183
- expect(service.getArchiveDataService).not.toHaveBeenCalled();
184
- });
185
- });
186
-
187
- describe('| getDataByIds', () => {
188
- it('| should get data by ids', async () => {
189
- const ids = ['id-1', 'id-2'];
190
- const mockDataList = [testData, { ...testData, _id: 'id-2' }];
191
- mockDBService.find.and.returnValue(Promise.resolve(mockDataList));
192
-
193
- const result = await service.getDataByIds(ids);
194
-
195
- expect(result).toEqual(mockDataList);
196
- expect(mockDBService.find).toHaveBeenCalledWith({ _id: { $in: ids } });
197
- });
198
-
199
- it('| should throw error when ids is missing', async () => {
200
- await expectAsync(
201
- service.getDataByIds(null as any)
202
- ).toBeRejected();
203
- });
204
-
205
- it('| should return empty array when ids is empty array', async () => {
206
- const result = await service.getDataByIds([]);
207
-
208
- expect(result).toEqual([]);
209
- });
210
- });
211
-
212
- describe('| getDataListByIds', () => {
213
- it('| should get data list by ids', async () => {
214
- const ids = ['id-1', 'id-2'];
215
- const mockDataList = [testData, { ...testData, _id: 'id-2' }];
216
- mockDBService.getDataListByIds.and.returnValue(Promise.resolve(mockDataList));
217
-
218
- const result = await service.getDataListByIds(ids);
219
-
220
- expect(result).toEqual(mockDataList);
221
- expect(service.dataList).toEqual(mockDataList);
222
- });
223
- });
224
-
225
- describe('| getDataByDependencyId', () => {
226
- it('| should get data by dependency id', async () => {
227
- const dependencyData = { ...testData, userId: 'user-123' };
228
- mockDBService.getDataByDependencyId.and.returnValue(Promise.resolve(dependencyData));
229
-
230
- const result = await service.getDataByDependencyId('user-123');
231
-
232
- expect(result).toBe(dependencyData);
233
- expect(service.data).toBe(dependencyData);
234
- });
235
-
236
- it('| should throw error when dependency settings not configured', async () => {
237
- const serviceWithoutDeps = new DyNTS_DataService<TestMetadata>(
238
- testData,
239
- new DyFM_DataModel_Params<TestMetadata>({
240
- dataName: 'test_data',
241
- properties: { name: { key: 'name', type: DyFM_BasicProperty_Type.string } },
242
- }),
243
- 'test-issuer'
244
- );
245
-
246
- await expectAsync(
247
- serviceWithoutDeps.getDataByDependencyId('user-123')
248
- ).toBeRejected();
249
- });
250
- });
251
-
252
- describe('| findData', () => {
253
- it('| should find data by filter', async () => {
254
- const filter: DyFM_DBFilter<TestMetadata> = { email: 'test@example.com' };
255
- mockDBService.findOne.and.returnValue(Promise.resolve(testData));
256
-
257
- const result = await service.findData(filter);
258
-
259
- expect(result).toBe(testData);
260
- expect(service.data).toBe(testData);
261
- expect(mockDBService.findOne).toHaveBeenCalledWith(filter);
262
- });
263
- });
264
-
265
- describe('| findDataList', () => {
266
- it('| should find data list by filter', async () => {
267
- const filter: DyFM_DBFilter<TestMetadata> = { name: 'Test' };
268
- const mockDataList = [testData];
269
- mockDBService.find.and.returnValue(Promise.resolve(mockDataList));
270
-
271
- const result = await service.findDataList(filter);
272
-
273
- expect(result).toEqual(mockDataList);
274
- expect(service.dataList).toEqual(mockDataList);
275
- });
276
-
277
- // FR-258 / SR-3 — over-fetch guard (unbounded find → heap → OOM precursor). Warns, never caps.
278
- describe('| over-fetch warning (FR-258 / SR-3)', () => {
279
- let originalThreshold: number;
280
-
281
- beforeEach(() => {
282
- originalThreshold = DyNTS_DataService.findDataListWarnThreshold;
283
- });
284
-
285
- afterEach(() => {
286
- DyNTS_DataService.findDataListWarnThreshold = originalThreshold;
287
- });
288
-
289
- function listOf(count: number): TestMetadata[] {
290
- return Array.from({ length: count }, (_v, i: number): TestMetadata =>
291
- new TestMetadata({ _id: `id-${i}`, name: `n-${i}` } as TestMetadata));
292
- }
293
-
294
- it('| warns when the result EXCEEDS the threshold', async () => {
295
- DyNTS_DataService.findDataListWarnThreshold = 3;
296
- const warnSpy = spyOn(DyFM_Log, 'warn');
297
- mockDBService.find.and.returnValue(Promise.resolve(listOf(5)));
298
-
299
- const result = await service.findDataList({ name: 'Test' });
300
-
301
- expect(result.length).toBe(5);
302
- const overFetchCall = warnSpy.calls.allArgs().find((args: any[]): boolean =>
303
- typeof args[0] === 'string' && args[0].includes('[FR-258 SR-3]'));
304
- expect(overFetchCall).toBeDefined();
305
- });
306
-
307
- it('| does NOT warn at exactly the threshold (strictly greater)', async () => {
308
- DyNTS_DataService.findDataListWarnThreshold = 5;
309
- const warnSpy = spyOn(DyFM_Log, 'warn');
310
- mockDBService.find.and.returnValue(Promise.resolve(listOf(5)));
311
-
312
- await service.findDataList({ name: 'Test' });
313
-
314
- const overFetchCall = warnSpy.calls.allArgs().find((args: any[]): boolean =>
315
- typeof args[0] === 'string' && args[0].includes('[FR-258 SR-3]'));
316
- expect(overFetchCall).toBeUndefined();
317
- });
318
-
319
- it('| disabled (threshold <= 0) → never warns regardless of size', async () => {
320
- DyNTS_DataService.findDataListWarnThreshold = 0;
321
- const warnSpy = spyOn(DyFM_Log, 'warn');
322
- mockDBService.find.and.returnValue(Promise.resolve(listOf(50000)));
323
-
324
- await service.findDataList({ name: 'Test' });
325
-
326
- const overFetchCall = warnSpy.calls.allArgs().find((args: any[]): boolean =>
327
- typeof args[0] === 'string' && args[0].includes('[FR-258 SR-3]'));
328
- expect(overFetchCall).toBeUndefined();
329
- });
330
- });
331
- });
332
-
333
- describe('| updateData', () => {
334
- it('| should update data by filterBy', async () => {
335
- const filter: DyFM_DBFilter<TestMetadata> = { email: 'test@example.com' };
336
- const update = { name: 'Updated Name' };
337
- mockDBService.updateOne.and.returnValue(Promise.resolve());
338
-
339
- await service.updateData({ filterBy: filter, update });
340
-
341
- expect(mockDBService.updateOne).toHaveBeenCalledWith(
342
- filter,
343
- update,
344
- 'test-issuer',
345
- undefined
346
- );
347
- });
348
-
349
- it('| should update data by data._id', async () => {
350
- const update = { name: 'Updated Name' };
351
- mockDBService.updateOne.and.returnValue(Promise.resolve());
352
-
353
- await service.updateData({ update });
354
-
355
- expect(mockDBService.updateOne).toHaveBeenCalledWith(
356
- { _id: 'test-id-123' } as any,
357
- update,
358
- 'test-issuer',
359
- undefined
360
- );
361
- });
362
-
363
- it('| should throw error when no usable parameter provided', async () => {
364
- service.data = new TestMetadata({});
365
- const update = { name: 'Updated Name' };
366
-
367
- await expectAsync(
368
- service.updateData({ update })
369
- ).toBeRejected();
370
- });
371
- });
372
-
373
- describe('| saveData', () => {
374
- it('| should modify existing data when _id is present', async () => {
375
- const existingData = { ...testData, name: 'Updated Name' };
376
- mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
377
- mockDBService.modifyData.and.returnValue(Promise.resolve(existingData));
378
- spyOn(service, 'validateForSave').and.returnValue(Promise.resolve());
379
-
380
- const result = await service.saveData(existingData);
381
-
382
- expect(result).toBe(existingData);
383
- expect(mockDBService.modifyData).toHaveBeenCalled();
384
- });
385
-
386
- it('| should throw error when data with _id does not exist', async () => {
387
- const dataWithId = { ...testData, _id: 'non-existent-id' };
388
- mockDBService.getDataById.and.returnValue(Promise.resolve(null));
389
- spyOn(service, 'validateForSave').and.returnValue(Promise.resolve());
390
-
391
- await expectAsync(
392
- service.saveData(dataWithId)
393
- ).toBeRejected();
394
- });
395
- });
396
-
397
- describe('| deleteData', () => {
398
- it('| should delete data by id', async () => {
399
- mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
400
- mockDBService.markDeletedById.and.returnValue(Promise.resolve());
401
-
402
- await service.deleteData('test-id-123');
403
-
404
- expect(mockDBService.markDeletedById).toHaveBeenCalledWith('test-id-123', 'test-issuer');
405
- });
406
-
407
- it('| should use data._id when id not provided', async () => {
408
- mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
409
- mockDBService.markDeletedById.and.returnValue(Promise.resolve());
410
-
411
- await service.deleteData();
412
-
413
- expect(mockDBService.markDeletedById).toHaveBeenCalledWith('test-id-123', 'test-issuer');
414
- });
415
-
416
- it('| should throw error when id is missing', async () => {
417
- service.data = new TestMetadata({});
418
-
419
- await expectAsync(
420
- service.deleteData()
421
- ).toBeRejected();
422
- });
423
-
424
- it('| should archive data when archive service enabled', async () => {
425
- service.haveArchiveDataService = true;
426
- mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
427
- mockDBService.trueDeleteDataById.and.returnValue(Promise.resolve());
428
- const mockArchiveService = jasmine.createSpyObj('DyNTS_ArchiveDataService', ['saveData']);
429
- mockArchiveService.saveData.and.returnValue(Promise.resolve());
430
- spyOn(service, 'getArchiveDataService').and.returnValue(mockArchiveService);
431
-
432
- await service.deleteData('test-id-123');
433
-
434
- expect(mockDBService.trueDeleteDataById).toHaveBeenCalled();
435
- expect(mockArchiveService.saveData).toHaveBeenCalled();
436
- });
437
-
438
- it('| should permanently delete when absolute is true', async () => {
439
- mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
440
- mockDBService.trueDeleteDataById.and.returnValue(Promise.resolve());
441
-
442
- await service.deleteData('test-id-123', true);
443
-
444
- expect(mockDBService.trueDeleteDataById).toHaveBeenCalledWith('test-id-123');
445
- });
446
- });
447
-
448
- describe('| deleteAllData', () => {
449
- it('| should delete all data', async () => {
450
- const mockDataList = [testData, { ...testData, _id: 'id-2' }];
451
- mockDBService.getAll.and.returnValue(Promise.resolve(mockDataList));
452
- mockDBService.markDeletedById.and.returnValue(Promise.resolve());
453
-
454
- await service.deleteAllData();
455
-
456
- expect(mockDBService.markDeletedById).toHaveBeenCalledTimes(2);
457
- });
458
-
459
- it('| should archive all data when archive service enabled', async () => {
460
- service.haveArchiveDataService = true;
461
- const mockDataList = [testData];
462
- mockDBService.getAll.and.returnValue(Promise.resolve(mockDataList));
463
- mockDBService.trueDeleteAllData.and.returnValue(Promise.resolve());
464
- const mockArchiveService = jasmine.createSpyObj('DyNTS_ArchiveDataService', ['saveData']);
465
- mockArchiveService.saveData.and.returnValue(Promise.resolve());
466
- spyOn(service, 'getArchiveDataService').and.returnValue(mockArchiveService);
467
-
468
- await service.deleteAllData();
469
-
470
- expect(mockDBService.trueDeleteAllData).toHaveBeenCalled();
471
- expect(mockArchiveService.saveData).toHaveBeenCalled();
472
- });
473
- });
474
-
475
- describe('| searchData', () => {
476
- it('| should search data with query', async () => {
477
- const query: DyFM_SearchQuery<TestMetadata> = {
478
- filterBy: { name: 'Test' },
479
- page: 0,
480
- pageSize: 10,
481
- sortBy: [{ key: 'name', order: 1 }],
482
- };
483
- const mockDataList = [testData];
484
- spyOn(service, 'sortAndFilterDataList').and.returnValue(Promise.resolve({
485
- results: mockDataList,
486
- totalItems: 1,
487
- }));
488
-
489
- const result = await service.searchData(query);
490
-
491
- expect(result.results).toEqual(mockDataList);
492
- expect(service.sortAndFilterDataList).toHaveBeenCalledWith(query, undefined);
493
- });
494
-
495
- it('| should apply pagination', async () => {
496
- const query: DyFM_SearchQuery<TestMetadata> = {
497
- page: 1,
498
- pageSize: 2,
499
- };
500
- const mockResults = [
501
- testData,
502
- { ...testData, _id: 'id-2' },
503
- { ...testData, _id: 'id-3' },
504
- ];
505
- spyOn(service, 'sortAndFilterDataList').and.returnValue(Promise.resolve({
506
- results: mockResults,
507
- totalItems: 3,
508
- }));
509
-
510
- const result = await service.searchData(query);
511
-
512
- expect(result.results.length).toBe(1);
513
- expect(result.results[0]._id).toBe('id-3');
514
- });
515
- });
516
-
517
- describe('| getArchiveDataService', () => {
518
- it('| should throw error when not implemented', () => {
519
- expect(() => {
520
- service.getArchiveDataService();
521
- }).toThrow();
522
- });
523
- });
524
-
525
- describe('| getDependencyDataDBService', () => {
526
- it('| should get dependency DB service when configured', () => {
527
- const depService = service.getDependencyDataDBService('user_data');
528
-
529
- expect(depService).toBeDefined();
530
- });
531
-
532
- it('| should throw error when service key not set', () => {
533
- const serviceWithoutDeps = new DyNTS_DataService<TestMetadata>(
534
- testData,
535
- new DyFM_DataModel_Params<TestMetadata>({
536
- dataName: 'test_data',
537
- properties: { name: { key: 'name', type: DyFM_BasicProperty_Type.string } },
538
- }),
539
- 'test-issuer'
540
- );
541
-
542
- expect(() => {
543
- serviceWithoutDeps.getDependencyDataDBService();
544
- }).toThrow();
545
- });
546
- });
547
-
548
- // ════════════════════════════════════════════════════════════════════════
549
- // FR-001 — Generic compareData()
550
- // ════════════════════════════════════════════════════════════════════════
551
-
552
- describe('| compareData() — FR-001', (): void => {
553
- let svc: DyNTS_DataService<TestMetadata>;
554
-
555
- beforeEach((): void => {
556
- // Re-use existing TestMetadata + testDataParams.
557
- const t: TestMetadata = new TestMetadata();
558
- svc = new DyNTS_DataService<TestMetadata>(t, testDataParams, 'test-issuer');
559
- });
560
-
561
- it('| equal POJO returns { result: "equal" } — sin changedFields', (): void => {
562
- const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x' });
563
- const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x' });
564
- const r = svc.compareData(a, b);
565
- expect(r.result).toBe('equal');
566
- expect(r.changedFields).toBeUndefined();
567
- });
568
-
569
- it('| modified single field — changedFields tartalmazza azt', (): void => {
570
- const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x' });
571
- const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'bob', email: 'a@x' });
572
- const r = svc.compareData(a, b);
573
- expect(r.result).toBe('modified');
574
- expect(r.changedFields).toEqual(['name']);
575
- });
576
-
577
- it('| modified multi-field — changedFields TELJES listat ad (nem early-return)', (): void => {
578
- const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x', userId: 'u1' });
579
- const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'bob', email: 'b@y', userId: 'u2' });
580
- const r = svc.compareData(a, b);
581
- expect(r.result).toBe('modified');
582
- expect(r.changedFields?.sort()).toEqual(['email', 'name', 'userId'] as any);
583
- });
584
-
585
- it('| auto-discovery SKIP-eli a metadata fields-et (_id, __created, __lastModified)', (): void => {
586
- const now: Date = new Date();
587
- const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', _id: 'X', __created: now, __lastModified: now });
588
- const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', _id: 'Y', __created: new Date(2000, 0, 1), __lastModified: new Date(2001, 0, 1) });
589
- const r = svc.compareData(a, b);
590
- expect(r.result).toBe('equal');
591
- });
592
-
593
- it('| explicit fields override — KIFEJEZETT _id-val VIZSGALJA a skip-listet is', (): void => {
594
- const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', _id: 'X' });
595
- const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', _id: 'Y' });
596
- const r = svc.compareData(a, b, { fields: ['_id' as keyof TestMetadata] });
597
- expect(r.result).toBe('modified');
598
- expect(r.changedFields).toEqual(['_id' as keyof TestMetadata]);
599
- });
600
-
601
- it('| explicit fields scope-szukit — csak a listazott fields szamit', (): void => {
602
- const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x' });
603
- const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'b@y' });
604
- const r = svc.compareData(a, b, { fields: ['name'] });
605
- expect(r.result).toBe('equal'); // email change ignored, only 'name' checked
606
- });
607
-
608
- it('| customComparator override (case-insensitive string)', (): void => {
609
- const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'Alice' });
610
- const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice' });
611
- const r = svc.compareData(a, b, {
612
- customComparators: { name: (x: string, y: string) => x.toLowerCase() === y.toLowerCase() },
613
- });
614
- expect(r.result).toBe('equal');
615
- });
616
-
617
- it('| customComparator — array-set-equality eltero sorrendre', (): void => {
618
- class WithArr extends DyFM_Metadata {
619
- name?: string;
620
- tags?: string[];
621
- }
622
- const arrSvc = new DyNTS_DataService<WithArr>(
623
- new WithArr(),
624
- new DyFM_DataModel_Params<WithArr>({
625
- dataName: 'with_arr',
626
- properties: { name: { key: 'name', type: DyFM_BasicProperty_Type.string } },
627
- }),
628
- 'test-issuer',
629
- );
630
- const a: WithArr = Object.assign(new WithArr(), { name: 'x', tags: ['a', 'b', 'c'] });
631
- const b: WithArr = Object.assign(new WithArr(), { name: 'x', tags: ['c', 'b', 'a'] });
632
- const r = arrSvc.compareData(a, b, {
633
- customComparators: {
634
- tags: (x: string[], y: string[]) => {
635
- const s: Set<string> = new Set<string>(x);
636
- return y.length === x.length && y.every((v: string) => s.has(v));
637
- },
638
- },
639
- });
640
- expect(r.result).toBe('equal');
641
- });
642
-
643
- it('| deep-equal: Date instances ugyanazon time-stamp-pel equal-nek szamitanak', (): void => {
644
- class WithDate extends DyFM_Metadata {
645
- when?: Date;
646
- }
647
- const dateSvc = new DyNTS_DataService<WithDate>(
648
- new WithDate(),
649
- new DyFM_DataModel_Params<WithDate>({
650
- dataName: 'with_date',
651
- properties: {},
652
- }),
653
- 'test-issuer',
654
- );
655
- const t: number = 1_700_000_000_000;
656
- const a: WithDate = Object.assign(new WithDate(), { when: new Date(t) });
657
- const b: WithDate = Object.assign(new WithDate(), { when: new Date(t) });
658
- expect(dateSvc.compareData(a, b).result).toBe('equal');
659
- });
660
-
661
- it('| deep-equal: nested object', (): void => {
662
- class WithNested extends DyFM_Metadata {
663
- nested?: { a: number; b: { c: string } };
664
- }
665
- const nSvc = new DyNTS_DataService<WithNested>(
666
- new WithNested(),
667
- new DyFM_DataModel_Params<WithNested>({ dataName: 'with_nested', properties: {} }),
668
- 'test-issuer',
669
- );
670
- const a: WithNested = Object.assign(new WithNested(), { nested: { a: 1, b: { c: 'x' } } });
671
- const b: WithNested = Object.assign(new WithNested(), { nested: { a: 1, b: { c: 'x' } } });
672
- const c: WithNested = Object.assign(new WithNested(), { nested: { a: 1, b: { c: 'y' } } });
673
- expect(nSvc.compareData(a, b).result).toBe('equal');
674
- expect(nSvc.compareData(a, c).result).toBe('modified');
675
- });
676
-
677
- it('| deep-equal: array index-szerinti compare (eltero sorrend → modified default-ban)', (): void => {
678
- class WithArr2 extends DyFM_Metadata {
679
- items?: number[];
680
- }
681
- const aSvc = new DyNTS_DataService<WithArr2>(
682
- new WithArr2(),
683
- new DyFM_DataModel_Params<WithArr2>({ dataName: 'with_arr2', properties: {} }),
684
- 'test-issuer',
685
- );
686
- const a: WithArr2 = Object.assign(new WithArr2(), { items: [1, 2, 3] });
687
- const b: WithArr2 = Object.assign(new WithArr2(), { items: [1, 2, 3] });
688
- const c: WithArr2 = Object.assign(new WithArr2(), { items: [3, 2, 1] });
689
- expect(aSvc.compareData(a, b).result).toBe('equal');
690
- expect(aSvc.compareData(a, c).result).toBe('modified');
691
- });
692
-
693
- it('| throw 400 ha newData null', (): void => {
694
- const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'x' });
695
- let thrown: any = null;
696
- try { svc.compareData(null as any, b); } catch (e) { thrown = e; }
697
- expect(thrown).not.toBeNull();
698
- expect(DyFM_Error.getErrorStatus(thrown)).toBe(400);
699
- expect(DyFM_Error.getErrorCode(thrown)).toContain('DyNTS-DS0-CD1');
700
- });
701
-
702
- it('| throw 400 ha oldData undefined', (): void => {
703
- const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'x' });
704
- let thrown: any = null;
705
- try { svc.compareData(a, undefined as any); } catch (e) { thrown = e; }
706
- expect(thrown).not.toBeNull();
707
- expect(DyFM_Error.getErrorStatus(thrown)).toBe(400);
708
- });
709
-
710
- it('| throw 400 ha options.fields ures array', (): void => {
711
- const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice' });
712
- const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice' });
713
- let thrown: any = null;
714
- try { svc.compareData(a, b, { fields: [] }); } catch (e) { thrown = e; }
715
- expect(thrown).not.toBeNull();
716
- expect(DyFM_Error.getErrorStatus(thrown)).toBe(400);
717
- expect(DyFM_Error.getErrorCode(thrown)).toContain('DyNTS-DS0-CD2');
718
- });
719
-
720
- it('| asymmetric keys: ha az egyik objektum-bol hianyzik a key, modified-nek szamit', (): void => {
721
- const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x' });
722
- const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice' }); // no email
723
- const r = svc.compareData(a, b);
724
- expect(r.result).toBe('modified');
725
- expect(r.changedFields).toEqual(['email']);
726
- });
727
- });
728
- });
729
-
1
+
2
+ import { DyNTS_DataService } from './data.service';
3
+ import { DyNTS_DBService } from './db.service';
4
+ import { DyNTS_GlobalService } from '../core/global.service';
5
+ import { DyNTS_ArchiveDataService } from './archive-data.service';
6
+ import { DyFM_Metadata, DyFM_DataModel_Params, DyFM_Error, DyFM_DBFilter, DyFM_SearchQuery, DyFM_BasicProperty_Type, DyFM_EnvironmentFlag, DyFM_Log } from '@futdevpro/fsm-dynamo';
7
+ import { DyNTS_global_settings } from '../../_collections/global-settings.const';
8
+
9
+ // Initialize global settings before any test runs
10
+ beforeAll(() => {
11
+ if (!DyNTS_global_settings.systemShortCodeName) {
12
+ (DyNTS_global_settings as any).systemShortCodeName = 'TEST';
13
+ }
14
+ if (!DyNTS_global_settings.env_settings) {
15
+ (DyNTS_global_settings as any).env_settings = {
16
+ environment: DyFM_EnvironmentFlag.local,
17
+ };
18
+ }
19
+ });
20
+
21
+ class TestMetadata extends DyFM_Metadata {
22
+ name: string;
23
+ email?: string;
24
+ userId?: string;
25
+ }
26
+
27
+ const testDataParams = new DyFM_DataModel_Params<TestMetadata>({
28
+ dataName: 'test_data',
29
+ properties: {
30
+ name: {
31
+ key: 'name',
32
+ type: DyFM_BasicProperty_Type.string,
33
+ required: true,
34
+ },
35
+ email: {
36
+ key: 'email',
37
+ type: DyFM_BasicProperty_Type.string,
38
+ },
39
+ userId: {
40
+ key: 'userId',
41
+ type: DyFM_BasicProperty_Type.string,
42
+ dependencyDataName: 'user_data',
43
+ },
44
+ },
45
+ });
46
+
47
+ describe('| DyNTS_DataService', () => {
48
+ let mockDBService: jasmine.SpyObj<DyNTS_DBService<TestMetadata>>;
49
+ let service: DyNTS_DataService<TestMetadata>;
50
+ let testData: TestMetadata;
51
+
52
+ beforeEach(() => {
53
+ testData = new TestMetadata({
54
+ _id: 'test-id-123',
55
+ name: 'Test Name',
56
+ email: 'test@example.com',
57
+ } as TestMetadata);
58
+
59
+ mockDBService = jasmine.createSpyObj('DyNTS_DBService', [
60
+ 'getAll',
61
+ 'getDataById',
62
+ 'getDataListByIds',
63
+ 'getDataByDependencyId',
64
+ 'getDataListByDependencyIds',
65
+ 'getDataListByDependencyId',
66
+ 'findOne',
67
+ 'find',
68
+ 'updateOne',
69
+ 'createData',
70
+ 'modifyData',
71
+ 'markDeletedById',
72
+ 'trueDeleteDataById',
73
+ 'trueDeleteAllData',
74
+ 'restoreDeletedById',
75
+ ]);
76
+
77
+ spyOn(DyNTS_GlobalService, 'getDBService').and.returnValue(mockDBService);
78
+ spyOn(DyNTS_GlobalService, 'getDBServiceByKey').and.returnValue(mockDBService);
79
+
80
+ service = new DyNTS_DataService<TestMetadata>(
81
+ testData,
82
+ testDataParams,
83
+ 'test-issuer'
84
+ );
85
+ });
86
+
87
+ it('| should create service instance', () => {
88
+ expect(service).toBeDefined();
89
+ expect(service.data).toBe(testData);
90
+ expect(service.dataParams).toBe(testDataParams);
91
+ expect(service.issuer).toBe('test-issuer');
92
+ expect(service.dataDBService).toBe(mockDBService);
93
+ });
94
+
95
+ it('| should set service name from constructor', () => {
96
+ expect(service.serviceName).toBe('DyNTS_DataService');
97
+ });
98
+
99
+ it('| should have default error messages', () => {
100
+ expect(service.defaultErrorUserMsg).toContain('Data Service Error');
101
+ expect(service.defaultValidationErrorUserMsg).toContain('Validation Error');
102
+ });
103
+
104
+ it('| should throw error when DB service not found', () => {
105
+ (DyNTS_GlobalService.getDBService as jasmine.Spy).and.throwError('DB Service not found');
106
+
107
+ expect(() => {
108
+ new DyNTS_DataService<TestMetadata>(testData, testDataParams, 'test-issuer');
109
+ }).toThrow();
110
+ });
111
+
112
+ describe('| getAll', () => {
113
+ it('| should get all data from database', async () => {
114
+ const mockDataList = [testData, { ...testData, _id: 'test-id-456' }];
115
+ mockDBService.getAll.and.returnValue(Promise.resolve(mockDataList));
116
+
117
+ const result = await service.getAll();
118
+
119
+ expect(result).toEqual(mockDataList);
120
+ expect(service.dataList).toEqual(mockDataList);
121
+ expect(mockDBService.getAll).toHaveBeenCalled();
122
+ });
123
+
124
+ it('| should not set to service when dontSetToService is true', async () => {
125
+ const mockDataList = [testData];
126
+ mockDBService.getAll.and.returnValue(Promise.resolve(mockDataList));
127
+
128
+ const result = await service.getAll(true);
129
+
130
+ expect(result).toEqual(mockDataList);
131
+ expect(service.dataList).not.toEqual(mockDataList);
132
+ });
133
+ });
134
+
135
+ describe('| getDataById', () => {
136
+ it('| should get data by id', async () => {
137
+ mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
138
+
139
+ const result = await service.getDataById('test-id-123');
140
+
141
+ expect(result).toBe(testData);
142
+ expect(service.data).toBe(testData);
143
+ expect(mockDBService.getDataById).toHaveBeenCalledWith('test-id-123');
144
+ });
145
+
146
+ it('| should use data._id when id not provided', async () => {
147
+ mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
148
+
149
+ const result = await service.getDataById();
150
+
151
+ expect(mockDBService.getDataById).toHaveBeenCalledWith('test-id-123');
152
+ });
153
+
154
+ it('| should throw error when id is missing', async () => {
155
+ service.data = new TestMetadata({});
156
+
157
+ await expectAsync(
158
+ service.getDataById()
159
+ ).toBeRejected();
160
+ });
161
+
162
+ it('| should load from archive when not found and archive enabled', async () => {
163
+ service.haveArchiveDataService = true;
164
+ mockDBService.getDataById.and.returnValue(Promise.resolve(null));
165
+ const mockArchiveService = jasmine.createSpyObj('DyNTS_ArchiveDataService', ['getDataByOriginalId']);
166
+ mockArchiveService.getDataByOriginalId.and.returnValue(Promise.resolve(testData));
167
+ spyOn(service, 'getArchiveDataService').and.returnValue(mockArchiveService);
168
+
169
+ const result = await service.getDataById('test-id-123');
170
+
171
+ expect(result).toBe(testData);
172
+ expect(mockArchiveService.getDataByOriginalId).toHaveBeenCalled();
173
+ });
174
+
175
+ it('| should not load from archive when skipArchiveLoad is true', async () => {
176
+ service.haveArchiveDataService = true;
177
+ mockDBService.getDataById.and.returnValue(Promise.resolve(null));
178
+ spyOn(service, 'getArchiveDataService');
179
+
180
+ const result = await service.getDataById('test-id-123', false, true);
181
+
182
+ expect(result).toBeNull();
183
+ expect(service.getArchiveDataService).not.toHaveBeenCalled();
184
+ });
185
+ });
186
+
187
+ describe('| getDataByIds', () => {
188
+ it('| should get data by ids', async () => {
189
+ const ids = ['id-1', 'id-2'];
190
+ const mockDataList = [testData, { ...testData, _id: 'id-2' }];
191
+ mockDBService.find.and.returnValue(Promise.resolve(mockDataList));
192
+
193
+ const result = await service.getDataByIds(ids);
194
+
195
+ expect(result).toEqual(mockDataList);
196
+ expect(mockDBService.find).toHaveBeenCalledWith({ _id: { $in: ids } });
197
+ });
198
+
199
+ it('| should throw error when ids is missing', async () => {
200
+ await expectAsync(
201
+ service.getDataByIds(null as any)
202
+ ).toBeRejected();
203
+ });
204
+
205
+ it('| should return empty array when ids is empty array', async () => {
206
+ const result = await service.getDataByIds([]);
207
+
208
+ expect(result).toEqual([]);
209
+ });
210
+ });
211
+
212
+ describe('| getDataListByIds', () => {
213
+ it('| should get data list by ids', async () => {
214
+ const ids = ['id-1', 'id-2'];
215
+ const mockDataList = [testData, { ...testData, _id: 'id-2' }];
216
+ mockDBService.getDataListByIds.and.returnValue(Promise.resolve(mockDataList));
217
+
218
+ const result = await service.getDataListByIds(ids);
219
+
220
+ expect(result).toEqual(mockDataList);
221
+ expect(service.dataList).toEqual(mockDataList);
222
+ });
223
+ });
224
+
225
+ describe('| getDataByDependencyId', () => {
226
+ it('| should get data by dependency id', async () => {
227
+ const dependencyData = { ...testData, userId: 'user-123' };
228
+ mockDBService.getDataByDependencyId.and.returnValue(Promise.resolve(dependencyData));
229
+
230
+ const result = await service.getDataByDependencyId('user-123');
231
+
232
+ expect(result).toBe(dependencyData);
233
+ expect(service.data).toBe(dependencyData);
234
+ });
235
+
236
+ it('| should throw error when dependency settings not configured', async () => {
237
+ const serviceWithoutDeps = new DyNTS_DataService<TestMetadata>(
238
+ testData,
239
+ new DyFM_DataModel_Params<TestMetadata>({
240
+ dataName: 'test_data',
241
+ properties: { name: { key: 'name', type: DyFM_BasicProperty_Type.string } },
242
+ }),
243
+ 'test-issuer'
244
+ );
245
+
246
+ await expectAsync(
247
+ serviceWithoutDeps.getDataByDependencyId('user-123')
248
+ ).toBeRejected();
249
+ });
250
+ });
251
+
252
+ describe('| findData', () => {
253
+ it('| should find data by filter', async () => {
254
+ const filter: DyFM_DBFilter<TestMetadata> = { email: 'test@example.com' };
255
+ mockDBService.findOne.and.returnValue(Promise.resolve(testData));
256
+
257
+ const result = await service.findData(filter);
258
+
259
+ expect(result).toBe(testData);
260
+ expect(service.data).toBe(testData);
261
+ expect(mockDBService.findOne).toHaveBeenCalledWith(filter);
262
+ });
263
+ });
264
+
265
+ describe('| findDataList', () => {
266
+ it('| should find data list by filter', async () => {
267
+ const filter: DyFM_DBFilter<TestMetadata> = { name: 'Test' };
268
+ const mockDataList = [testData];
269
+ mockDBService.find.and.returnValue(Promise.resolve(mockDataList));
270
+
271
+ const result = await service.findDataList(filter);
272
+
273
+ expect(result).toEqual(mockDataList);
274
+ expect(service.dataList).toEqual(mockDataList);
275
+ });
276
+
277
+ // FR-258 / SR-3 — over-fetch guard (unbounded find → heap → OOM precursor). Warns, never caps.
278
+ describe('| over-fetch warning (FR-258 / SR-3)', () => {
279
+ let originalThreshold: number;
280
+
281
+ beforeEach(() => {
282
+ originalThreshold = DyNTS_DataService.findDataListWarnThreshold;
283
+ });
284
+
285
+ afterEach(() => {
286
+ DyNTS_DataService.findDataListWarnThreshold = originalThreshold;
287
+ });
288
+
289
+ function listOf(count: number): TestMetadata[] {
290
+ return Array.from({ length: count }, (_v, i: number): TestMetadata =>
291
+ new TestMetadata({ _id: `id-${i}`, name: `n-${i}` } as TestMetadata));
292
+ }
293
+
294
+ it('| warns when the result EXCEEDS the threshold', async () => {
295
+ DyNTS_DataService.findDataListWarnThreshold = 3;
296
+ const warnSpy = spyOn(DyFM_Log, 'warn');
297
+ mockDBService.find.and.returnValue(Promise.resolve(listOf(5)));
298
+
299
+ const result = await service.findDataList({ name: 'Test' });
300
+
301
+ expect(result.length).toBe(5);
302
+ const overFetchCall = warnSpy.calls.allArgs().find((args: any[]): boolean =>
303
+ typeof args[0] === 'string' && args[0].includes('[FR-258 SR-3]'));
304
+ expect(overFetchCall).toBeDefined();
305
+ });
306
+
307
+ it('| does NOT warn at exactly the threshold (strictly greater)', async () => {
308
+ DyNTS_DataService.findDataListWarnThreshold = 5;
309
+ const warnSpy = spyOn(DyFM_Log, 'warn');
310
+ mockDBService.find.and.returnValue(Promise.resolve(listOf(5)));
311
+
312
+ await service.findDataList({ name: 'Test' });
313
+
314
+ const overFetchCall = warnSpy.calls.allArgs().find((args: any[]): boolean =>
315
+ typeof args[0] === 'string' && args[0].includes('[FR-258 SR-3]'));
316
+ expect(overFetchCall).toBeUndefined();
317
+ });
318
+
319
+ it('| disabled (threshold <= 0) → never warns regardless of size', async () => {
320
+ DyNTS_DataService.findDataListWarnThreshold = 0;
321
+ const warnSpy = spyOn(DyFM_Log, 'warn');
322
+ mockDBService.find.and.returnValue(Promise.resolve(listOf(50000)));
323
+
324
+ await service.findDataList({ name: 'Test' });
325
+
326
+ const overFetchCall = warnSpy.calls.allArgs().find((args: any[]): boolean =>
327
+ typeof args[0] === 'string' && args[0].includes('[FR-258 SR-3]'));
328
+ expect(overFetchCall).toBeUndefined();
329
+ });
330
+ });
331
+ });
332
+
333
+ describe('| updateData', () => {
334
+ it('| should update data by filterBy', async () => {
335
+ const filter: DyFM_DBFilter<TestMetadata> = { email: 'test@example.com' };
336
+ const update = { name: 'Updated Name' };
337
+ mockDBService.updateOne.and.returnValue(Promise.resolve());
338
+
339
+ await service.updateData({ filterBy: filter, update });
340
+
341
+ expect(mockDBService.updateOne).toHaveBeenCalledWith(
342
+ filter,
343
+ update,
344
+ 'test-issuer',
345
+ undefined
346
+ );
347
+ });
348
+
349
+ it('| should update data by data._id', async () => {
350
+ const update = { name: 'Updated Name' };
351
+ mockDBService.updateOne.and.returnValue(Promise.resolve());
352
+
353
+ await service.updateData({ update });
354
+
355
+ expect(mockDBService.updateOne).toHaveBeenCalledWith(
356
+ { _id: 'test-id-123' } as any,
357
+ update,
358
+ 'test-issuer',
359
+ undefined
360
+ );
361
+ });
362
+
363
+ it('| should throw error when no usable parameter provided', async () => {
364
+ service.data = new TestMetadata({});
365
+ const update = { name: 'Updated Name' };
366
+
367
+ await expectAsync(
368
+ service.updateData({ update })
369
+ ).toBeRejected();
370
+ });
371
+ });
372
+
373
+ describe('| saveData', () => {
374
+ it('| should modify existing data when _id is present', async () => {
375
+ const existingData = { ...testData, name: 'Updated Name' };
376
+ mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
377
+ mockDBService.modifyData.and.returnValue(Promise.resolve(existingData));
378
+ spyOn(service, 'validateForSave').and.returnValue(Promise.resolve());
379
+
380
+ const result = await service.saveData(existingData);
381
+
382
+ expect(result).toBe(existingData);
383
+ expect(mockDBService.modifyData).toHaveBeenCalled();
384
+ });
385
+
386
+ it('| should throw error when data with _id does not exist', async () => {
387
+ const dataWithId = { ...testData, _id: 'non-existent-id' };
388
+ mockDBService.getDataById.and.returnValue(Promise.resolve(null));
389
+ spyOn(service, 'validateForSave').and.returnValue(Promise.resolve());
390
+
391
+ await expectAsync(
392
+ service.saveData(dataWithId)
393
+ ).toBeRejected();
394
+ });
395
+ });
396
+
397
+ describe('| deleteData', () => {
398
+ it('| should delete data by id', async () => {
399
+ mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
400
+ mockDBService.markDeletedById.and.returnValue(Promise.resolve());
401
+
402
+ await service.deleteData('test-id-123');
403
+
404
+ expect(mockDBService.markDeletedById).toHaveBeenCalledWith('test-id-123', 'test-issuer');
405
+ });
406
+
407
+ it('| should use data._id when id not provided', async () => {
408
+ mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
409
+ mockDBService.markDeletedById.and.returnValue(Promise.resolve());
410
+
411
+ await service.deleteData();
412
+
413
+ expect(mockDBService.markDeletedById).toHaveBeenCalledWith('test-id-123', 'test-issuer');
414
+ });
415
+
416
+ it('| should throw error when id is missing', async () => {
417
+ service.data = new TestMetadata({});
418
+
419
+ await expectAsync(
420
+ service.deleteData()
421
+ ).toBeRejected();
422
+ });
423
+
424
+ it('| should archive data when archive service enabled', async () => {
425
+ service.haveArchiveDataService = true;
426
+ mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
427
+ mockDBService.trueDeleteDataById.and.returnValue(Promise.resolve());
428
+ const mockArchiveService = jasmine.createSpyObj('DyNTS_ArchiveDataService', ['saveData']);
429
+ mockArchiveService.saveData.and.returnValue(Promise.resolve());
430
+ spyOn(service, 'getArchiveDataService').and.returnValue(mockArchiveService);
431
+
432
+ await service.deleteData('test-id-123');
433
+
434
+ expect(mockDBService.trueDeleteDataById).toHaveBeenCalled();
435
+ expect(mockArchiveService.saveData).toHaveBeenCalled();
436
+ });
437
+
438
+ it('| should permanently delete when absolute is true', async () => {
439
+ mockDBService.getDataById.and.returnValue(Promise.resolve(testData));
440
+ mockDBService.trueDeleteDataById.and.returnValue(Promise.resolve());
441
+
442
+ await service.deleteData('test-id-123', true);
443
+
444
+ expect(mockDBService.trueDeleteDataById).toHaveBeenCalledWith('test-id-123');
445
+ });
446
+ });
447
+
448
+ describe('| deleteAllData', () => {
449
+ it('| should delete all data', async () => {
450
+ const mockDataList = [testData, { ...testData, _id: 'id-2' }];
451
+ mockDBService.getAll.and.returnValue(Promise.resolve(mockDataList));
452
+ mockDBService.markDeletedById.and.returnValue(Promise.resolve());
453
+
454
+ await service.deleteAllData();
455
+
456
+ expect(mockDBService.markDeletedById).toHaveBeenCalledTimes(2);
457
+ });
458
+
459
+ it('| should archive all data when archive service enabled', async () => {
460
+ service.haveArchiveDataService = true;
461
+ const mockDataList = [testData];
462
+ mockDBService.getAll.and.returnValue(Promise.resolve(mockDataList));
463
+ mockDBService.trueDeleteAllData.and.returnValue(Promise.resolve());
464
+ const mockArchiveService = jasmine.createSpyObj('DyNTS_ArchiveDataService', ['saveData']);
465
+ mockArchiveService.saveData.and.returnValue(Promise.resolve());
466
+ spyOn(service, 'getArchiveDataService').and.returnValue(mockArchiveService);
467
+
468
+ await service.deleteAllData();
469
+
470
+ expect(mockDBService.trueDeleteAllData).toHaveBeenCalled();
471
+ expect(mockArchiveService.saveData).toHaveBeenCalled();
472
+ });
473
+ });
474
+
475
+ describe('| searchData', () => {
476
+ it('| should search data with query', async () => {
477
+ const query: DyFM_SearchQuery<TestMetadata> = {
478
+ filterBy: { name: 'Test' },
479
+ page: 0,
480
+ pageSize: 10,
481
+ sortBy: [{ key: 'name', order: 1 }],
482
+ };
483
+ const mockDataList = [testData];
484
+ spyOn(service, 'sortAndFilterDataList').and.returnValue(Promise.resolve({
485
+ results: mockDataList,
486
+ totalItems: 1,
487
+ }));
488
+
489
+ const result = await service.searchData(query);
490
+
491
+ expect(result.results).toEqual(mockDataList);
492
+ expect(service.sortAndFilterDataList).toHaveBeenCalledWith(query, undefined);
493
+ });
494
+
495
+ it('| should apply pagination', async () => {
496
+ const query: DyFM_SearchQuery<TestMetadata> = {
497
+ page: 1,
498
+ pageSize: 2,
499
+ };
500
+ const mockResults = [
501
+ testData,
502
+ { ...testData, _id: 'id-2' },
503
+ { ...testData, _id: 'id-3' },
504
+ ];
505
+ spyOn(service, 'sortAndFilterDataList').and.returnValue(Promise.resolve({
506
+ results: mockResults,
507
+ totalItems: 3,
508
+ }));
509
+
510
+ const result = await service.searchData(query);
511
+
512
+ expect(result.results.length).toBe(1);
513
+ expect(result.results[0]._id).toBe('id-3');
514
+ });
515
+ });
516
+
517
+ describe('| getArchiveDataService', () => {
518
+ it('| should throw error when not implemented', () => {
519
+ expect(() => {
520
+ service.getArchiveDataService();
521
+ }).toThrow();
522
+ });
523
+ });
524
+
525
+ describe('| getDependencyDataDBService', () => {
526
+ it('| should get dependency DB service when configured', () => {
527
+ const depService = service.getDependencyDataDBService('user_data');
528
+
529
+ expect(depService).toBeDefined();
530
+ });
531
+
532
+ it('| should throw error when service key not set', () => {
533
+ const serviceWithoutDeps = new DyNTS_DataService<TestMetadata>(
534
+ testData,
535
+ new DyFM_DataModel_Params<TestMetadata>({
536
+ dataName: 'test_data',
537
+ properties: { name: { key: 'name', type: DyFM_BasicProperty_Type.string } },
538
+ }),
539
+ 'test-issuer'
540
+ );
541
+
542
+ expect(() => {
543
+ serviceWithoutDeps.getDependencyDataDBService();
544
+ }).toThrow();
545
+ });
546
+ });
547
+
548
+ // ════════════════════════════════════════════════════════════════════════
549
+ // FR-001 — Generic compareData()
550
+ // ════════════════════════════════════════════════════════════════════════
551
+
552
+ describe('| compareData() — FR-001', (): void => {
553
+ let svc: DyNTS_DataService<TestMetadata>;
554
+
555
+ beforeEach((): void => {
556
+ // Re-use existing TestMetadata + testDataParams.
557
+ const t: TestMetadata = new TestMetadata();
558
+ svc = new DyNTS_DataService<TestMetadata>(t, testDataParams, 'test-issuer');
559
+ });
560
+
561
+ it('| equal POJO returns { result: "equal" } — sin changedFields', (): void => {
562
+ const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x' });
563
+ const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x' });
564
+ const r = svc.compareData(a, b);
565
+ expect(r.result).toBe('equal');
566
+ expect(r.changedFields).toBeUndefined();
567
+ });
568
+
569
+ it('| modified single field — changedFields tartalmazza azt', (): void => {
570
+ const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x' });
571
+ const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'bob', email: 'a@x' });
572
+ const r = svc.compareData(a, b);
573
+ expect(r.result).toBe('modified');
574
+ expect(r.changedFields).toEqual(['name']);
575
+ });
576
+
577
+ it('| modified multi-field — changedFields TELJES listat ad (nem early-return)', (): void => {
578
+ const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x', userId: 'u1' });
579
+ const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'bob', email: 'b@y', userId: 'u2' });
580
+ const r = svc.compareData(a, b);
581
+ expect(r.result).toBe('modified');
582
+ expect(r.changedFields?.sort()).toEqual(['email', 'name', 'userId'] as any);
583
+ });
584
+
585
+ it('| auto-discovery SKIP-eli a metadata fields-et (_id, __created, __lastModified)', (): void => {
586
+ const now: Date = new Date();
587
+ const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', _id: 'X', __created: now, __lastModified: now });
588
+ const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', _id: 'Y', __created: new Date(2000, 0, 1), __lastModified: new Date(2001, 0, 1) });
589
+ const r = svc.compareData(a, b);
590
+ expect(r.result).toBe('equal');
591
+ });
592
+
593
+ it('| explicit fields override — KIFEJEZETT _id-val VIZSGALJA a skip-listet is', (): void => {
594
+ const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', _id: 'X' });
595
+ const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', _id: 'Y' });
596
+ const r = svc.compareData(a, b, { fields: ['_id' as keyof TestMetadata] });
597
+ expect(r.result).toBe('modified');
598
+ expect(r.changedFields).toEqual(['_id' as keyof TestMetadata]);
599
+ });
600
+
601
+ it('| explicit fields scope-szukit — csak a listazott fields szamit', (): void => {
602
+ const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x' });
603
+ const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'b@y' });
604
+ const r = svc.compareData(a, b, { fields: ['name'] });
605
+ expect(r.result).toBe('equal'); // email change ignored, only 'name' checked
606
+ });
607
+
608
+ it('| customComparator override (case-insensitive string)', (): void => {
609
+ const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'Alice' });
610
+ const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice' });
611
+ const r = svc.compareData(a, b, {
612
+ customComparators: { name: (x: string, y: string) => x.toLowerCase() === y.toLowerCase() },
613
+ });
614
+ expect(r.result).toBe('equal');
615
+ });
616
+
617
+ it('| customComparator — array-set-equality eltero sorrendre', (): void => {
618
+ class WithArr extends DyFM_Metadata {
619
+ name?: string;
620
+ tags?: string[];
621
+ }
622
+ const arrSvc = new DyNTS_DataService<WithArr>(
623
+ new WithArr(),
624
+ new DyFM_DataModel_Params<WithArr>({
625
+ dataName: 'with_arr',
626
+ properties: { name: { key: 'name', type: DyFM_BasicProperty_Type.string } },
627
+ }),
628
+ 'test-issuer',
629
+ );
630
+ const a: WithArr = Object.assign(new WithArr(), { name: 'x', tags: ['a', 'b', 'c'] });
631
+ const b: WithArr = Object.assign(new WithArr(), { name: 'x', tags: ['c', 'b', 'a'] });
632
+ const r = arrSvc.compareData(a, b, {
633
+ customComparators: {
634
+ tags: (x: string[], y: string[]) => {
635
+ const s: Set<string> = new Set<string>(x);
636
+ return y.length === x.length && y.every((v: string) => s.has(v));
637
+ },
638
+ },
639
+ });
640
+ expect(r.result).toBe('equal');
641
+ });
642
+
643
+ it('| deep-equal: Date instances ugyanazon time-stamp-pel equal-nek szamitanak', (): void => {
644
+ class WithDate extends DyFM_Metadata {
645
+ when?: Date;
646
+ }
647
+ const dateSvc = new DyNTS_DataService<WithDate>(
648
+ new WithDate(),
649
+ new DyFM_DataModel_Params<WithDate>({
650
+ dataName: 'with_date',
651
+ properties: {},
652
+ }),
653
+ 'test-issuer',
654
+ );
655
+ const t: number = 1_700_000_000_000;
656
+ const a: WithDate = Object.assign(new WithDate(), { when: new Date(t) });
657
+ const b: WithDate = Object.assign(new WithDate(), { when: new Date(t) });
658
+ expect(dateSvc.compareData(a, b).result).toBe('equal');
659
+ });
660
+
661
+ it('| deep-equal: nested object', (): void => {
662
+ class WithNested extends DyFM_Metadata {
663
+ nested?: { a: number; b: { c: string } };
664
+ }
665
+ const nSvc = new DyNTS_DataService<WithNested>(
666
+ new WithNested(),
667
+ new DyFM_DataModel_Params<WithNested>({ dataName: 'with_nested', properties: {} }),
668
+ 'test-issuer',
669
+ );
670
+ const a: WithNested = Object.assign(new WithNested(), { nested: { a: 1, b: { c: 'x' } } });
671
+ const b: WithNested = Object.assign(new WithNested(), { nested: { a: 1, b: { c: 'x' } } });
672
+ const c: WithNested = Object.assign(new WithNested(), { nested: { a: 1, b: { c: 'y' } } });
673
+ expect(nSvc.compareData(a, b).result).toBe('equal');
674
+ expect(nSvc.compareData(a, c).result).toBe('modified');
675
+ });
676
+
677
+ it('| deep-equal: array index-szerinti compare (eltero sorrend → modified default-ban)', (): void => {
678
+ class WithArr2 extends DyFM_Metadata {
679
+ items?: number[];
680
+ }
681
+ const aSvc = new DyNTS_DataService<WithArr2>(
682
+ new WithArr2(),
683
+ new DyFM_DataModel_Params<WithArr2>({ dataName: 'with_arr2', properties: {} }),
684
+ 'test-issuer',
685
+ );
686
+ const a: WithArr2 = Object.assign(new WithArr2(), { items: [1, 2, 3] });
687
+ const b: WithArr2 = Object.assign(new WithArr2(), { items: [1, 2, 3] });
688
+ const c: WithArr2 = Object.assign(new WithArr2(), { items: [3, 2, 1] });
689
+ expect(aSvc.compareData(a, b).result).toBe('equal');
690
+ expect(aSvc.compareData(a, c).result).toBe('modified');
691
+ });
692
+
693
+ it('| throw 400 ha newData null', (): void => {
694
+ const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'x' });
695
+ let thrown: any = null;
696
+ try { svc.compareData(null as any, b); } catch (e) { thrown = e; }
697
+ expect(thrown).not.toBeNull();
698
+ expect(DyFM_Error.getErrorStatus(thrown)).toBe(400);
699
+ expect(DyFM_Error.getErrorCode(thrown)).toContain('DyNTS-DS0-CD1');
700
+ });
701
+
702
+ it('| throw 400 ha oldData undefined', (): void => {
703
+ const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'x' });
704
+ let thrown: any = null;
705
+ try { svc.compareData(a, undefined as any); } catch (e) { thrown = e; }
706
+ expect(thrown).not.toBeNull();
707
+ expect(DyFM_Error.getErrorStatus(thrown)).toBe(400);
708
+ });
709
+
710
+ it('| throw 400 ha options.fields ures array', (): void => {
711
+ const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice' });
712
+ const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice' });
713
+ let thrown: any = null;
714
+ try { svc.compareData(a, b, { fields: [] }); } catch (e) { thrown = e; }
715
+ expect(thrown).not.toBeNull();
716
+ expect(DyFM_Error.getErrorStatus(thrown)).toBe(400);
717
+ expect(DyFM_Error.getErrorCode(thrown)).toContain('DyNTS-DS0-CD2');
718
+ });
719
+
720
+ it('| asymmetric keys: ha az egyik objektum-bol hianyzik a key, modified-nek szamit', (): void => {
721
+ const a: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice', email: 'a@x' });
722
+ const b: TestMetadata = Object.assign(new TestMetadata(), { name: 'alice' }); // no email
723
+ const r = svc.compareData(a, b);
724
+ expect(r.result).toBe('modified');
725
+ expect(r.changedFields).toEqual(['email']);
726
+ });
727
+ });
728
+ });
729
+