@futdevpro/nts-dynamo 1.15.74 → 1.15.75

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 (357) 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/.dynamo/logs/cicd-pipeline/output.log +2798 -0
  17. package/.dynamo/logs/cicd-pipeline/status.json +94 -0
  18. package/.vscode/settings.json +10 -10
  19. package/HOWTO.md +15 -15
  20. package/LICENSE +21 -21
  21. package/__documentations/nts-integration-tests-2026-03-17.md +26 -26
  22. package/__documentations/plans/BEDROCK-HYPERPLAN.md +95 -95
  23. package/_specifications/BACKLOG.md +92 -92
  24. package/_specifications/TODO.md +15 -15
  25. package/_specifications/agent.md +138 -138
  26. package/eslint.config.js +3 -3
  27. package/nodemon.json +24 -24
  28. package/package.json +2 -2
  29. package/pnpm-workspace.yaml +5 -5
  30. package/scripts/run-coverage-tests.js +28 -28
  31. package/spec/support/helpers/spec-reporter-loader.js +359 -359
  32. package/spec/support/helpers/ts-node-helper.js +93 -93
  33. package/spec/support/jasmine.coverage.json +24 -24
  34. package/spec/support/jasmine.json +24 -24
  35. package/src/_collections/archive.util.spec.ts +57 -57
  36. package/src/_collections/archive.util.ts +18 -18
  37. package/src/_collections/atlas-default-db-options.const.ts +9 -9
  38. package/src/_collections/default-fallback-cache-max-age.const.spec.ts +11 -11
  39. package/src/_collections/default-fallback-cache-max-age.const.ts +2 -2
  40. package/src/_collections/default-not-found-page.const.spec.ts +19 -19
  41. package/src/_collections/default-not-found-page.const.ts +22 -22
  42. package/src/_collections/default-socket-path.const.spec.ts +12 -12
  43. package/src/_collections/default-socket-path.const.ts +2 -2
  44. package/src/_collections/get-environment-settings.util.spec.ts +210 -210
  45. package/src/_collections/get-environment-settings.util.ts +48 -48
  46. package/src/_collections/global-settings.const.ts +97 -97
  47. package/src/_collections/sample.env +21 -21
  48. package/src/_collections/star.controller.spec.ts +224 -224
  49. package/src/_collections/star.controller.ts +129 -129
  50. package/src/_enums/data-model-type.enum.ts +14 -14
  51. package/src/_enums/data-service-function.enum.ts +24 -24
  52. package/src/_enums/predefined-data-types.enum.ts +16 -16
  53. package/src/_enums/route-security.enum.ts +12 -12
  54. package/src/_models/control-models/api-call-params.control-model.spec.ts +152 -152
  55. package/src/_models/control-models/api-call-params.control-model.ts +142 -142
  56. package/src/_models/control-models/app-ext-system-controls.control-model.spec.ts +52 -52
  57. package/src/_models/control-models/app-ext-system-controls.control-model.ts +9 -9
  58. package/src/_models/control-models/app-params.control-model.spec.ts +225 -225
  59. package/src/_models/control-models/app-params.control-model.ts +136 -136
  60. package/src/_models/control-models/app-system-controls.control-model.spec.ts +31 -31
  61. package/src/_models/control-models/app-system-controls.control-model.ts +9 -9
  62. package/src/_models/control-models/endpoint-params.control-model.spec.ts +627 -627
  63. package/src/_models/control-models/endpoint-params.control-model.ts +627 -627
  64. package/src/_models/control-models/http-settings.control-model.spec.ts +77 -77
  65. package/src/_models/control-models/http-settings.control-model.ts +37 -37
  66. package/src/_models/control-models/system-control.control-model.spec.ts +27 -27
  67. package/src/_models/control-models/system-control.control-model.ts +12 -12
  68. package/src/_models/interfaces/certification-settings.interface.ts +7 -7
  69. package/src/_models/interfaces/environment-settings.interface.ts +59 -59
  70. package/src/_models/interfaces/global-log-settings.interface.ts +171 -171
  71. package/src/_models/interfaces/global-service-settings.interface.ts +47 -47
  72. package/src/_models/interfaces/global-settings.interface.ts +216 -216
  73. package/src/_models/interfaces/routing-module-settings.interface.ts +21 -21
  74. package/src/_models/interfaces/static-client-settings.interface.spec.ts +29 -29
  75. package/src/_models/interfaces/static-client-settings.interface.ts +28 -28
  76. package/src/_models/types/db-update.type.ts +100 -100
  77. package/src/_modules/ai/_models/ai-input-interfaces.ts +117 -117
  78. package/src/_modules/ai/_models/ai-test-generation-result.interface.ts +16 -16
  79. package/src/_modules/ai/_modules/anthropic/_services/aai-user-key.control-service.ts +138 -138
  80. package/src/_modules/ai/_modules/anthropic/index.ts +5 -5
  81. package/src/_modules/ai/_modules/document-ai/_collections/dai-chunking.util.spec.ts +242 -242
  82. package/src/_modules/ai/_modules/document-ai/_collections/dai-chunking.util.ts +639 -639
  83. package/src/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.spec.ts +295 -295
  84. package/src/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.ts +518 -518
  85. package/src/_modules/ai/_modules/document-ai/_collections/dai-document.util.spec.ts +209 -209
  86. package/src/_modules/ai/_modules/document-ai/_collections/dai-document.util.ts +85 -85
  87. package/src/_modules/ai/_modules/document-ai/_enums/dai-compare-result-type.enum.ts +7 -7
  88. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-doc-chunk.data-model.ts +146 -146
  89. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-doc-page.data-model.ts +162 -162
  90. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-document.data-model.ts +99 -99
  91. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.ts +68 -68
  92. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-doc-chunk-compare-result.interface.ts +18 -18
  93. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-doc-page-compare-result.interface.ts +19 -19
  94. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-document-compare-result.interface.ts +25 -25
  95. package/src/_modules/ai/_modules/document-ai/index.ts +30 -30
  96. package/src/_modules/ai/_modules/fdp-ai/_services/fdpai-user-key.control-service.ts +189 -189
  97. package/src/_modules/ai/_modules/fdp-ai/index.ts +5 -5
  98. package/src/_modules/ai/_modules/open-ai/_collections/oai-global-settings.const.ts +9 -9
  99. package/src/_modules/ai/_modules/open-ai/_collections/oai-llm-predefined-requests-hu.conts.ts +82 -82
  100. package/src/_modules/ai/_modules/open-ai/_collections/oai-llm-predefined-requests.conts.ts +75 -75
  101. package/src/_modules/ai/_modules/open-ai/_enums/oai-gpt-message-role.enum.ts +45 -45
  102. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-global-settings.interface.ts +7 -7
  103. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-gpt-message.interface.ts +7 -7
  104. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-llm-predefined-requests.interface.ts +57 -57
  105. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-doc-chunk-data.service.ts +292 -292
  106. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-document.data-service.spec.ts +342 -342
  107. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-vector-data.service.spec.ts +550 -550
  108. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-vector-data.service.ts +630 -630
  109. package/src/_modules/ai/_modules/open-ai/_services/oai-embedding.control-service.spec.ts +332 -332
  110. package/src/_modules/ai/_modules/open-ai/_services/oai-llm-chat.service-base.spec.ts +462 -462
  111. package/src/_modules/ai/_modules/open-ai/_services/oai-llm-chat.service-base.ts +634 -634
  112. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.spec.ts +489 -489
  113. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.tools.spec.ts +173 -173
  114. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.ts +1033 -1033
  115. package/src/_modules/ai/_modules/open-ai/_services/oai-user-key.control-service.ts +157 -157
  116. package/src/_modules/ai/_services/ai-embedding-mock.service.spec.ts +115 -115
  117. package/src/_modules/ai/_services/ai-embedding-mock.service.ts +212 -212
  118. package/src/_modules/ai/_services/ai-embedding-provider.registry.spec.ts +110 -110
  119. package/src/_modules/ai/_services/ai-embedding-provider.registry.ts +110 -110
  120. package/src/_modules/ai/_services/ai-embedding.service-base.spec.ts +98 -98
  121. package/src/_modules/ai/_services/ai-embedding.service-base.ts +48 -48
  122. package/src/_modules/ai/_services/ai-llm-chat.service-base.spec.ts +229 -229
  123. package/src/_modules/ai/_services/ai-llm-chat.service-base.ts +68 -68
  124. package/src/_modules/ai/_services/ai-llm.service-base.spec.ts +250 -250
  125. package/src/_modules/ai/_services/ai-llm.service-base.ts +519 -519
  126. package/src/_modules/ai/_services/ai-provider.service-base.spec.ts +158 -158
  127. package/src/_modules/ai/_services/ai-user-key.service-base.ts +59 -59
  128. package/src/_modules/ai/_services/lmstudio-embedding.control-service.spec.ts +197 -197
  129. package/src/_modules/ai/_services/lmstudio-embedding.control-service.ts +371 -371
  130. package/src/_modules/ai/index.ts +23 -23
  131. package/src/_modules/assistant/_collections/ass-global-settings.const.ts +13 -13
  132. package/src/_modules/assistant/_collections/ass.util.spec.ts +176 -176
  133. package/src/_modules/assistant/_collections/ass.util.ts +50 -50
  134. package/src/_modules/assistant/_models/ass-global-settings.interface.ts +15 -15
  135. package/src/_modules/assistant/_services/ass-io.control-service.spec.ts +140 -140
  136. package/src/_modules/assistant/_services/ass-main.control-service.spec.ts +192 -192
  137. package/src/_modules/assistant/_services/ass-main.control-service.ts +107 -107
  138. package/src/_modules/bot/_collections/bot-default-commands.const.ts +12 -12
  139. package/src/_modules/bot/_collections/bot-global-settings.const.ts +39 -39
  140. package/src/_modules/bot/_models/bot-channel-wrapper.interface.ts +62 -62
  141. package/src/_modules/bot/_models/bot-command.interface.ts +8 -8
  142. package/src/_modules/bot/_models/bot-global-settings.interface.ts +96 -96
  143. package/src/_modules/bot/_models/bot-last-mention-date.interface.ts +6 -6
  144. package/src/_modules/bot/_models/bot-last-message-date.interface.ts +5 -5
  145. package/src/_modules/bot/_models/bot-user-wrapper.interface.ts +41 -41
  146. package/src/_modules/bot/_modules/discord-bot/_models/dib-platform.types.ts +9 -9
  147. package/src/_modules/bot/_modules/discord-bot/_services/dib-messaging-provider.control-service.spec.ts +431 -431
  148. package/src/_modules/bot/_modules/dynamo-bot/_collections/dyb-operations.util.spec.ts +160 -160
  149. package/src/_modules/bot/_modules/dynamo-bot/_collections/dyb-operations.util.ts +55 -55
  150. package/src/_modules/bot/_modules/dynamo-bot/_models/dyb-platform.types.ts +15 -15
  151. package/src/_modules/bot/_modules/dynamo-bot/_services/dyb-messaging-provider.control-service.spec.ts +374 -374
  152. package/src/_modules/bot/_modules/dynamo-bot/_services/dyb-messaging-provider.control-service.ts +447 -447
  153. package/src/_modules/bot/_modules/dynamo-bot/index.ts +15 -15
  154. package/src/_modules/bot/_modules/slack-bot/_models/slb-platform.types.ts +9 -9
  155. package/src/_modules/bot/_modules/slack-bot/_services/slb-messaging-provider.control-service.spec.ts +344 -344
  156. package/src/_modules/bot/_modules/slack-bot/_services/slb-messaging-provider.control-service.ts +197 -197
  157. package/src/_modules/bot/_modules/teams-bot/_models/teb-platform.types.ts +9 -9
  158. package/src/_modules/bot/_modules/teams-bot/_services/teb-messaging-provider.control-service.spec.ts +345 -345
  159. package/src/_modules/bot/_modules/teams-bot/_services/teb-messaging-provider.control-service.ts +197 -197
  160. package/src/_modules/bot/_services/bot-commands.control-service.spec.ts +116 -116
  161. package/src/_modules/bot/_services/bot-io.control-service.spec.ts +285 -285
  162. package/src/_modules/bot/_services/bot-main.control-service.spec.ts +208 -208
  163. package/src/_modules/bot/_services/bot-messaging-provider.service-base.spec.ts +349 -349
  164. package/src/_modules/bot/_services/bot-routines.control-service.spec.ts +111 -111
  165. package/src/_modules/custom-data/custom-data.controller.spec.ts +49 -49
  166. package/src/_modules/custom-data/custom-data.controller.ts +67 -67
  167. package/src/_modules/custom-data/custom-data.data-service.spec.ts +54 -54
  168. package/src/_modules/custom-data/custom-data.data-service.ts +21 -21
  169. package/src/_modules/custom-data/get-custom-data-routing-module.util.spec.ts +28 -28
  170. package/src/_modules/custom-data/get-custom-data-routing-module.util.ts +24 -24
  171. package/src/_modules/custom-data/index.ts +9 -9
  172. package/src/_modules/data-readers/_collections/dynts-sqlite-reader.util.spec.ts +161 -161
  173. package/src/_modules/data-readers/_collections/dynts-sqlite-reader.util.ts +203 -203
  174. package/src/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.ts +33 -33
  175. package/src/_modules/data-readers/index.ts +11 -11
  176. package/src/_modules/defaults/_collections/default-endpoints.util.ts +487 -487
  177. package/src/_modules/defaults/_models/default-user.data-model.ts +72 -72
  178. package/src/_modules/defaults/_services/default-auth.service.spec.ts +269 -269
  179. package/src/_modules/defaults/_services/default-auth.service.ts +177 -177
  180. package/src/_modules/defaults/_services/default-socket-events.service.spec.ts +42 -42
  181. package/src/_modules/defaults/_services/default-socket-events.service.ts +61 -61
  182. package/src/_modules/defaults/_services/default-user.data-service.spec.ts +187 -187
  183. package/src/_modules/defaults/_services/default-user.data-service.ts +98 -98
  184. package/src/_modules/defaults/index.ts +17 -17
  185. package/src/_modules/discord-assistant/_collections/dias-global-settings.const.ts +19 -19
  186. package/src/_modules/discord-assistant/_collections/dias.util.spec.ts +366 -366
  187. package/src/_modules/discord-assistant/_collections/dias.util.ts +132 -132
  188. package/src/_modules/discord-assistant/_models/dias-global-settings.interface.ts +19 -19
  189. package/src/_modules/discord-assistant/_models/dias-knowledge.data-model.ts +52 -52
  190. package/src/_modules/discord-assistant/_services/dias-chunk.data-service.ts +177 -177
  191. package/src/_modules/discord-assistant/_services/dias-io.control-service.spec.ts +108 -108
  192. package/src/_modules/discord-assistant/_services/dias-io.control-service.ts +69 -69
  193. package/src/_modules/discord-assistant/_services/dias-main.control-service.spec.ts +22 -22
  194. package/src/_modules/discord-assistant/_services/dias-main.control-service.ts +27 -27
  195. package/src/_modules/discord-assistant/_services/dias.service-base.spec.ts +195 -195
  196. package/src/_modules/discord-assistant/_services/dias.service-base.ts +76 -76
  197. package/src/_modules/discord-assistant/index.ts +38 -38
  198. package/src/_modules/discord-assistant-voiced/_services/dias-discord-bot.control-service.spec.ts +34 -34
  199. package/src/_modules/discord-assistant-voiced/_services/dias-discord-bot.control-service.ts +11 -11
  200. package/src/_modules/discord-assistant-voiced/index.ts +36 -36
  201. package/src/_modules/discord-bot/_collections/dibo-default-commands.const.ts +16 -16
  202. package/src/_modules/discord-bot/_collections/dibo-global-settings.conts.ts +55 -55
  203. package/src/_modules/discord-bot/_collections/dibo-operations.util.spec.ts +214 -214
  204. package/src/_modules/discord-bot/_collections/dibo-operations.util.ts +387 -387
  205. package/src/_modules/discord-bot/_models/dibo-command.interface.ts +12 -12
  206. package/src/_modules/discord-bot/_models/dibo-global-settings.interface.ts +98 -98
  207. package/src/_modules/discord-bot/_models/dibo-last-mention-date.inteface.ts +7 -7
  208. package/src/_modules/discord-bot/_models/dibo-last-message-date.interface.ts +6 -6
  209. package/src/_modules/discord-bot/_services/dibo-commands.control-service.spec.ts +154 -154
  210. package/src/_modules/discord-bot/_services/dibo-commands.control-service.ts +153 -153
  211. package/src/_modules/discord-bot/_services/dibo-io.control-service.spec.ts +264 -264
  212. package/src/_modules/discord-bot/_services/dibo-io.control-service.ts +306 -306
  213. package/src/_modules/discord-bot/_services/dibo-main.control-service.spec.ts +408 -408
  214. package/src/_modules/discord-bot/_services/dibo-main.control-service.ts +487 -487
  215. package/src/_modules/discord-bot/_services/dibo-routines.control-service.spec.ts +105 -105
  216. package/src/_modules/discord-bot/index.ts +36 -36
  217. package/src/_modules/local-vector-search/_enums/lvs-search-mode.enum.ts +35 -35
  218. package/src/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.ts +59 -59
  219. package/src/_modules/local-vector-search/_models/lvs-search-result.interface.ts +17 -17
  220. package/src/_modules/local-vector-search/_services/lvs-doc-chunk-data.service.spec.ts +418 -418
  221. package/src/_modules/local-vector-search/_services/lvs-doc-chunk-data.service.ts +276 -276
  222. package/src/_modules/local-vector-search/_services/lvs-local-vector-search.data-service.spec.ts +480 -480
  223. package/src/_modules/local-vector-search/_services/lvs-local-vector-search.data-service.ts +416 -416
  224. package/src/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.spec.ts +198 -198
  225. package/src/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.ts +146 -146
  226. package/src/_modules/local-vector-search/_services/lvs-vector-persist.data-service.spec.ts +167 -167
  227. package/src/_modules/local-vector-search/_services/lvs-vector-persist.data-service.ts +106 -106
  228. package/src/_modules/local-vector-search/_services/lvs-vector-pool.control-service.spec.ts +507 -507
  229. package/src/_modules/local-vector-search/_services/lvs-vector-pool.control-service.ts +272 -272
  230. package/src/_modules/local-vector-search/index.ts +16 -16
  231. package/src/_modules/logs/index.ts +11 -11
  232. package/src/_modules/mcp/_models/interfaces/dynts-mcp.interface.ts +111 -111
  233. package/src/_modules/mcp/_services/dynts-mcp-server.service-base.spec.ts +142 -142
  234. package/src/_modules/mcp/_services/dynts-mcp-server.service-base.ts +120 -120
  235. package/src/_modules/mcp/_services/dynts-mcp.adapter.ts +168 -168
  236. package/src/_modules/mcp/index.ts +13 -13
  237. package/src/_modules/messaging/README.md +354 -354
  238. package/src/_modules/messaging/_collections/get-messaging-routing-module.util.ts +26 -26
  239. package/src/_modules/messaging/_collections/msg-global-settings.const.ts +22 -22
  240. package/src/_modules/messaging/_collections/msg.util.spec.ts +226 -226
  241. package/src/_modules/messaging/_models/msg-global-settings.interface.ts +37 -37
  242. package/src/_modules/messaging/_services/msg-conversation.data-service.ts +146 -146
  243. package/src/_modules/messaging/_services/msg-events.service.spec.ts +219 -219
  244. package/src/_modules/messaging/_services/msg-events.service.ts +267 -267
  245. package/src/_modules/messaging/_services/msg-integration.control-service.ts +179 -179
  246. package/src/_modules/messaging/_services/msg-main.control-service.spec.ts +147 -147
  247. package/src/_modules/messaging/_services/msg-main.control-service.ts +571 -571
  248. package/src/_modules/messaging/_services/msg-message.data-service.ts +129 -129
  249. package/src/_modules/messaging/_services/msg.controller.spec.ts +201 -201
  250. package/src/_modules/messaging/index.ts +30 -30
  251. package/src/_modules/mock/app-extended-server.mock.ts +201 -201
  252. package/src/_modules/mock/app-integration-test.mock.ts +51 -51
  253. package/src/_modules/mock/app-params.mock.spec.ts +21 -21
  254. package/src/_modules/mock/app-params.mock.ts +9 -9
  255. package/src/_modules/mock/app-server.mock.ts +188 -188
  256. package/src/_modules/mock/auth-service.mock.spec.ts +47 -47
  257. package/src/_modules/mock/auth-service.mock.ts +28 -28
  258. package/src/_modules/mock/controller.mock.spec.ts +26 -26
  259. package/src/_modules/mock/controller.mock.ts +16 -16
  260. package/src/_modules/mock/data-model.mock.spec.ts +111 -111
  261. package/src/_modules/mock/data-model.mock.ts +82 -82
  262. package/src/_modules/mock/email-service-collection.mock.spec.ts +24 -24
  263. package/src/_modules/mock/email-service-collection.mock.ts +15 -15
  264. package/src/_modules/mock/email-service.mock.spec.ts +17 -17
  265. package/src/_modules/mock/email-service.mock.ts +20 -20
  266. package/src/_modules/mock/email-template.mock.html +14 -14
  267. package/src/_modules/mock/endpoint.mock.ts +91 -91
  268. package/src/_modules/mock/socket-client.mock.spec.ts +40 -40
  269. package/src/_modules/mock/socket-client.mock.ts +45 -45
  270. package/src/_modules/mock/socket-server.mock.spec.ts +44 -44
  271. package/src/_modules/mock/socket-server.mock.ts +46 -46
  272. package/src/_modules/oauth2/_routes/oauth2.controller.spec.ts +107 -107
  273. package/src/_modules/oauth2/_routes/oauth2.controller.ts +98 -98
  274. package/src/_modules/oauth2/_services/oauth2.auth-service.spec.ts +254 -254
  275. package/src/_modules/oauth2/_services/oauth2.auth-service.ts +232 -232
  276. package/src/_modules/oauth2/_services/oauth2.control-service.spec.ts +585 -585
  277. package/src/_modules/oauth2/_services/oauth2.control-service.ts +653 -653
  278. package/src/_modules/oauth2/index.ts +17 -17
  279. package/src/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.ts +22 -22
  280. package/src/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.ts +81 -81
  281. package/src/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.ts +107 -107
  282. package/src/_modules/scoped-config/_services/dynts-scoped-config.control-service.spec.ts +306 -306
  283. package/src/_modules/scoped-config/_services/dynts-scoped-config.control-service.ts +295 -295
  284. package/src/_modules/scoped-config/_services/dynts-scoped-config.data-service.spec.ts +118 -118
  285. package/src/_modules/scoped-config/_services/dynts-scoped-config.data-service.ts +105 -105
  286. package/src/_modules/scoped-config/index.ts +17 -17
  287. package/src/_modules/server/errors/errors.control-service.spec.ts +238 -238
  288. package/src/_modules/server/errors/errors.control-service.ts +100 -100
  289. package/src/_modules/server/errors/errors.controller.spec.ts +241 -241
  290. package/src/_modules/server/errors/errors.controller.ts +489 -489
  291. package/src/_modules/server/errors/errors.data-service.spec.ts +480 -480
  292. package/src/_modules/server/index.ts +30 -30
  293. package/src/_modules/server/server-status/server-status-snapshot.control-service.spec.ts +70 -70
  294. package/src/_modules/server/server-status/server-status-snapshot.control-service.ts +17 -17
  295. package/src/_modules/server/server-status/server-status-snapshot.data-service.spec.ts +77 -77
  296. package/src/_modules/server/server-status/server-status-snapshot.data-service.ts +37 -37
  297. package/src/_modules/server/server-status/server-status.control-service.spec.ts +576 -576
  298. package/src/_modules/server/server-status/server-status.control-service.ts +396 -396
  299. package/src/_modules/server/server-status/server-status.controller.spec.ts +240 -240
  300. package/src/_modules/server/server-status/server-status.controller.ts +253 -253
  301. package/src/_modules/socket/_enums/socket-security.enum.ts +11 -11
  302. package/src/_modules/socket/_models/socket-client-service-params.control-model.spec.ts +32 -32
  303. package/src/_modules/socket/_models/socket-client-service-params.control-model.ts +22 -22
  304. package/src/_modules/socket/_models/socket-presence.control-model.spec.ts +164 -164
  305. package/src/_modules/socket/_models/socket-presence.control-model.ts +210 -210
  306. package/src/_modules/socket/_models/socket-server-service-params.control-model.spec.ts +46 -46
  307. package/src/_modules/socket/_models/socket-server-service-params.control-model.ts +22 -22
  308. package/src/_modules/socket/_services/socket-client.service.spec.ts +15 -15
  309. package/src/_modules/socket/_services/socket-client.service.ts +260 -260
  310. package/src/_modules/socket/_services/socket-server.service.spec.ts +11 -11
  311. package/src/_modules/socket/app-extended.integration.spec.ts +85 -85
  312. package/src/_modules/socket/app-extended.server.ts +630 -630
  313. package/src/_modules/socket/index.ts +42 -42
  314. package/src/_modules/test/get-test-routing-module.util.spec.ts +28 -28
  315. package/src/_modules/test/get-test-routing-module.util.ts +23 -23
  316. package/src/_modules/test/index.ts +11 -11
  317. package/src/_modules/test/test.controller.spec.ts +72 -72
  318. package/src/_modules/test/test.controller.ts +115 -115
  319. package/src/_modules/usage/get-usage-routing-module.util.ts +22 -22
  320. package/src/_modules/usage/index.ts +15 -15
  321. package/src/_modules/usage/usage.controller.spec.ts +81 -81
  322. package/src/_modules/usage/usage.controller.ts +126 -126
  323. package/src/_modules/usage/usage.data-service.spec.ts +332 -332
  324. package/src/_modules/usage/usage.data-service.ts +185 -185
  325. package/src/_services/base/api.service-base.spec.ts +125 -125
  326. package/src/_services/base/api.service-base.ts +74 -74
  327. package/src/_services/base/archive-data.service.spec.ts +196 -196
  328. package/src/_services/base/archive-data.service.ts +216 -216
  329. package/src/_services/base/data.service.spec.ts +674 -674
  330. package/src/_services/base/data.service.ts +2719 -2719
  331. package/src/_services/base/db.service.spec.ts +73 -73
  332. package/src/_services/base/db.service.ts +1575 -1575
  333. package/src/_services/base/singleton.service-base.spec.ts +28 -28
  334. package/src/_services/base/singleton.service-base.ts +24 -24
  335. package/src/_services/base/singleton.service.spec.ts +114 -114
  336. package/src/_services/base/singleton.service.ts +38 -38
  337. package/src/_services/core/api.service.spec.ts +140 -140
  338. package/src/_services/core/auth.service.spec.ts +159 -159
  339. package/src/_services/core/auth.service.ts +174 -174
  340. package/src/_services/core/email.service.spec.ts +85 -85
  341. package/src/_services/core/email.service.ts +742 -742
  342. package/src/_services/core/global.service.spec.ts +292 -292
  343. package/src/_services/core/global.service.ts +475 -475
  344. package/src/_services/core/memory-guard.service.spec.ts +245 -245
  345. package/src/_services/core/memory-guard.service.ts +481 -481
  346. package/src/_services/core/service-collection.service.spec.ts +46 -46
  347. package/src/_services/core/service-collection.service.ts +6 -6
  348. package/src/_services/route/controller.service.spec.ts +53 -53
  349. package/src/_services/route/controller.service.ts +148 -148
  350. package/src/_services/route/routing-module.service.spec.ts +98 -98
  351. package/src/_services/route/routing-module.service.ts +330 -330
  352. package/src/_services/server/app.server.ts +1747 -1747
  353. package/src/_services/shared.static-service.spec.ts +99 -99
  354. package/src/_services/shared.static-service.ts +78 -78
  355. package/src/index.ts +96 -96
  356. package/tsconfig.app.json +12 -12
  357. package/tsconfig.json +42 -42
@@ -1,489 +1,489 @@
1
-
2
- import { DyNTS_OAI_LLM_ServiceBase } from './oai-llm.service-base';
3
- import { DyFM_OAI_Settings, DyFM_OAI_Model, DyFM_OAI_CallSettings } from '@futdevpro/fsm-dynamo/ai/open-ai';
4
- import { DyFM_AI_MessageRole, DyFM_AI_Message } from '@futdevpro/fsm-dynamo/ai';
5
- import { DyFM_Error, DyFM_Log } from '@futdevpro/fsm-dynamo';
6
- import { ChatCompletion } from 'openai/resources';
7
-
8
- describe('| DyNTS_OAI_LLM_ServiceBase', () => {
9
- let service: DyNTS_OAI_LLM_ServiceBase;
10
- let mockOpenAI: any;
11
- let mockChatCompletionsCreate: jasmine.Spy;
12
- let mockModelsList: jasmine.Spy;
13
-
14
- const mockSettings: DyFM_OAI_Settings = {
15
- config: {
16
- apiKey: 'test-api-key',
17
- organization: 'test-org',
18
- },
19
- defaultSettings: {
20
- useModel: DyFM_OAI_Model.gpt4o_mini,
21
- systemPrompt: 'Test system prompt',
22
- temperature: 0.7,
23
- maxTokens: 1000,
24
- },
25
- };
26
-
27
- beforeEach(() => {
28
- mockChatCompletionsCreate = jasmine.createSpy('chat.completions.create').and.returnValue({
29
- id: 'chatcmpl-123',
30
- object: 'chat.completion',
31
- created: 1234567890,
32
- model: 'gpt-4o-mini',
33
- choices: [
34
- {
35
- index: 0,
36
- message: {
37
- role: 'assistant',
38
- content: 'Test response',
39
- },
40
- finish_reason: 'stop',
41
- },
42
- ],
43
- usage: {
44
- prompt_tokens: 10,
45
- completion_tokens: 5,
46
- total_tokens: 15,
47
- },
48
- });
49
-
50
- mockModelsList = jasmine.createSpy('models.list').and.returnValue({
51
- data: [],
52
- });
53
-
54
- mockOpenAI = {
55
- chat: {
56
- completions: {
57
- create: mockChatCompletionsCreate,
58
- },
59
- },
60
- models: {
61
- list: mockModelsList,
62
- },
63
- };
64
-
65
- service = new DyNTS_OAI_LLM_ServiceBase(mockSettings);
66
- (service as any).openai = mockOpenAI;
67
- });
68
-
69
- describe('| constructor', () => {
70
- it('| should initialize with provided settings', () => {
71
- expect(service.defaultSettings).toEqual(mockSettings.defaultSettings);
72
- expect(service.aiProvider).toBeDefined();
73
- expect(service.capabilities).toBeDefined();
74
- });
75
-
76
- it('| should initialize with default settings when not provided', () => {
77
- // Creating service without settings requires OPENAI_API_KEY env var or throws
78
- // Test that service creation with settings works and uses defaults for missing values
79
- const serviceWithPartialSettings = new DyNTS_OAI_LLM_ServiceBase({
80
- config: { apiKey: 'test-key' },
81
- });
82
- expect(serviceWithPartialSettings.defaultSettings).toBeDefined();
83
- });
84
-
85
- it('| should set defaultModel from defaultSettings', () => {
86
- expect(service.defaultModel).toBe(DyFM_OAI_Model.gpt4o_mini);
87
- });
88
- });
89
-
90
- describe('| setup', () => {
91
- it('| should reinitialize OpenAI client with new config', () => {
92
- const newConfig = {
93
- apiKey: 'new-api-key',
94
- organization: 'new-org',
95
- };
96
-
97
- service.setup(newConfig);
98
-
99
- expect((service as any).openai).toBeDefined();
100
- });
101
- });
102
-
103
- describe('| requestYesNo', () => {
104
- it('| should return true when answer contains YES', async () => {
105
- mockChatCompletionsCreate.and.returnValue({
106
- choices: [
107
- {
108
- message: {
109
- content: 'YES',
110
- },
111
- },
112
- ],
113
- });
114
-
115
- spyOn(service as any, 'convertAnswerToBoolean').and.returnValue(true);
116
-
117
- const result = await service.requestYesNo({
118
- message: 'Is this a test?',
119
- issuer: 'test-issuer',
120
- });
121
-
122
- expect(result).toBe(true);
123
- expect((service as any).convertAnswerToBoolean).toHaveBeenCalled();
124
- });
125
-
126
- it('| should append predefined request to message', async () => {
127
- spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve('YES'));
128
-
129
- await service.requestYesNo({
130
- message: 'Is this a test?',
131
- issuer: 'test-issuer',
132
- });
133
-
134
- expect(service.requestSimpleMessage).toHaveBeenCalled();
135
- const callArgs = (service.requestSimpleMessage as jasmine.Spy).calls.mostRecent().args[0];
136
- expect(callArgs.message).toContain('Is this a test?');
137
- });
138
- });
139
-
140
- describe('| requestPercentage', () => {
141
- it('| should return number from answer', async () => {
142
- spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve('75'));
143
- spyOn(service as any, 'convertAnswerToNumber').and.returnValue(75);
144
-
145
- const result = await service.requestPercentage({
146
- message: 'What percentage?',
147
- issuer: 'test-issuer',
148
- });
149
-
150
- expect(result).toBe(75);
151
- expect((service as any).convertAnswerToNumber).toHaveBeenCalled();
152
- });
153
- });
154
-
155
- describe('| requestSelect', () => {
156
- it('| should return selected option', async () => {
157
- const options = ['option1', 'option2', 'option3'];
158
- spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve('option1'));
159
- spyOn(service as any, 'stringifySelectOptions').and.returnValue(['option1', 'option2', 'option3']);
160
- spyOn(service as any, 'getTextListAsText').and.returnValue('option1, option2, option3');
161
- spyOn(service as any, 'convertAnswerToSelectOption').and.returnValue('option1');
162
-
163
- const result = await service.requestSelect({
164
- message: 'Select an option',
165
- selectFrom: options,
166
- issuer: 'test-issuer',
167
- });
168
-
169
- expect(result).toBe('option1');
170
- });
171
- });
172
-
173
- describe('| requestMultiselect', () => {
174
- it('| should return multiple selected options', async () => {
175
- const options = ['option1', 'option2', 'option3'];
176
- spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve('option1 and option2'));
177
- spyOn(service as any, 'stringifySelectOptions').and.returnValue(['option1', 'option2', 'option3']);
178
- spyOn(service as any, 'getTextListAsText').and.returnValue('option1, option2, option3');
179
- spyOn(service as any, 'convertAnswerToSelectOptions').and.returnValue(['option1', 'option2']);
180
-
181
- const result = await service.requestMultiselect({
182
- message: 'Select options',
183
- options: options,
184
- issuer: 'test-issuer',
185
- });
186
-
187
- expect(result).toEqual(['option1', 'option2']);
188
- });
189
- });
190
-
191
- describe('| requestJSON', () => {
192
- it('| should return parsed JSON', async () => {
193
- const jsonString = '{"key": "value"}';
194
- spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve(jsonString));
195
- spyOn(service as any, 'convertAnswerToJSON').and.returnValue({ key: 'value' });
196
-
197
- const result = await service.requestJSON<{ key: string }>({
198
- message: 'Return JSON',
199
- issuer: 'test-issuer',
200
- });
201
-
202
- expect(result).toEqual({ key: 'value' });
203
- });
204
- });
205
-
206
- describe('| requestList', () => {
207
- it('| should return list of items', async () => {
208
- spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve('item1, item2, item3'));
209
- spyOn(service as any, 'convertAnswerToList').and.returnValue(['item1', 'item2', 'item3']);
210
-
211
- const result = await service.requestList<string>({
212
- message: 'Return a list',
213
- issuer: 'test-issuer',
214
- });
215
-
216
- expect(result).toEqual(['item1', 'item2', 'item3']);
217
- });
218
- });
219
-
220
- describe('| requestSimpleMessage', () => {
221
- it('| should resolve message and return answer', async () => {
222
- spyOn(service as any, 'logQuestion');
223
- spyOn(service as any, 'logAnswer');
224
- spyOn(service, 'resolveMessage').and.returnValue(Promise.resolve('Test response'));
225
-
226
- const result = await service.requestSimpleMessage({
227
- message: 'Test question',
228
- issuer: 'test-issuer',
229
- });
230
-
231
- expect(result).toBe('Test response');
232
- expect((service as any).logQuestion).toHaveBeenCalled();
233
- expect((service as any).logAnswer).toHaveBeenCalled();
234
- });
235
-
236
- it('| should add user message to conversation', async () => {
237
- const conversation: DyFM_AI_Message[] = [];
238
- spyOn(service, 'resolveMessage').and.returnValue(Promise.resolve('Test response'));
239
-
240
- await service.requestSimpleMessage({
241
- message: 'Test question',
242
- conversation: conversation,
243
- issuer: 'test-issuer',
244
- });
245
-
246
- expect(conversation.length).toBe(1);
247
- expect(conversation[0].role).toBe(DyFM_AI_MessageRole.user);
248
- expect(conversation[0].content).toBe('Test question');
249
- });
250
- });
251
-
252
- describe('| resolveMessage', () => {
253
- it('| should return string answer when getFullResponse is false', async () => {
254
- const conversation: DyFM_AI_Message[] = [
255
- { role: DyFM_AI_MessageRole.user, content: 'Test question' },
256
- ];
257
-
258
- const result = await service.resolveMessage({
259
- conversation: conversation,
260
- issuer: 'test-issuer',
261
- getFullResponse: false,
262
- });
263
-
264
- expect(result).toBe('Test response');
265
- expect(mockChatCompletionsCreate).toHaveBeenCalled();
266
- });
267
-
268
- it('| should return full ChatCompletion when getFullResponse is true', async () => {
269
- const conversation: DyFM_AI_Message[] = [
270
- { role: DyFM_AI_MessageRole.user, content: 'Test question' },
271
- ];
272
-
273
- const fullResponse: ChatCompletion = {
274
- id: 'chatcmpl-123',
275
- object: 'chat.completion',
276
- created: 1234567890,
277
- model: 'gpt-4o-mini',
278
- choices: [
279
- {
280
- index: 0,
281
- message: {
282
- role: 'assistant',
283
- content: 'Test response',
284
- refusal: null,
285
- },
286
- logprobs: null,
287
- finish_reason: 'stop',
288
- },
289
- ],
290
- usage: {
291
- prompt_tokens: 10,
292
- completion_tokens: 5,
293
- total_tokens: 15,
294
- },
295
- };
296
-
297
- mockChatCompletionsCreate.and.returnValue(fullResponse);
298
-
299
- const result = await service.resolveMessage({
300
- conversation: conversation,
301
- issuer: 'test-issuer',
302
- getFullResponse: true,
303
- });
304
-
305
- expect(result).toEqual(fullResponse);
306
- });
307
-
308
- it('| should add system message to conversation', async () => {
309
- const conversation: DyFM_AI_Message[] = [
310
- { role: DyFM_AI_MessageRole.user, content: 'Test question' },
311
- ];
312
-
313
- spyOn(service as any, 'getDefaultSystemMessage').and.returnValue({
314
- role: DyFM_AI_MessageRole.system,
315
- content: 'System prompt',
316
- });
317
-
318
- await service.resolveMessage({
319
- conversation: conversation,
320
- issuer: 'test-issuer',
321
- });
322
-
323
- expect(conversation[0].role).toBe(DyFM_AI_MessageRole.system);
324
- });
325
-
326
- it('| should handle API errors', async () => {
327
- const conversation: DyFM_AI_Message[] = [
328
- { role: DyFM_AI_MessageRole.user, content: 'Test question' },
329
- ];
330
-
331
- mockChatCompletionsCreate.and.throwError(new Error('API Error'));
332
-
333
- try {
334
- await service.resolveMessage({
335
- conversation: conversation,
336
- issuer: 'test-issuer',
337
- });
338
- fail('Should have thrown an error');
339
- } catch (error) {
340
- expect(error).toBeInstanceOf(DyFM_Error);
341
- expect((error as DyFM_Error)._errorCode).toContain('DyNTS-OLSB-RM0');
342
- }
343
- });
344
- });
345
-
346
- describe('| FR-002 cost-event emit', () => {
347
-
348
- it('| should emit cost-event with llm-completion callType for single user message', async () => {
349
- const received: any[] = [];
350
- (service as any).onCostEvent = (e: any) => { received.push(e); };
351
-
352
- await service.resolveMessage({
353
- conversation: [{ role: DyFM_AI_MessageRole.user, content: 'Q?' }],
354
- issuer: 'llm-issuer-1',
355
- });
356
-
357
- expect(received.length).toBe(1);
358
- expect(received[0].callType).toBe('llm-completion');
359
- expect(received[0].provider).toBe('open-ai');
360
- expect(received[0].model).toBe(DyFM_OAI_Model.gpt4o_mini);
361
- expect(received[0].tokensUsed.input).toBe(10);
362
- expect(received[0].tokensUsed.output).toBe(5);
363
- expect(received[0].tokensUsed.total).toBe(15);
364
- expect(received[0].issuer).toBe('llm-issuer-1');
365
- expect(received[0].timestamp).toBeInstanceOf(Date);
366
- expect(received[0].durationMs).toBeGreaterThanOrEqual(0);
367
- expect(received[0].estimatedCostUsd).toBeUndefined();
368
- });
369
-
370
- it('| should emit cost-event with llm-chat callType for multi-message conversation', async () => {
371
- const received: any[] = [];
372
- (service as any).onCostEvent = (e: any) => { received.push(e); };
373
-
374
- await service.resolveMessage({
375
- conversation: [
376
- { role: DyFM_AI_MessageRole.user, content: 'Q1?' },
377
- { role: DyFM_AI_MessageRole.assistant, content: 'A1.' },
378
- { role: DyFM_AI_MessageRole.user, content: 'Q2?' },
379
- ],
380
- issuer: 'llm-issuer-chat',
381
- });
382
-
383
- expect(received.length).toBe(1);
384
- expect(received[0].callType).toBe('llm-chat');
385
- });
386
-
387
- it('| constructor accepts onCostEvent option (non-breaking additive)', () => {
388
- const cb: any = jasmine.createSpy('onCostEvent');
389
- const s: DyNTS_OAI_LLM_ServiceBase = new DyNTS_OAI_LLM_ServiceBase({
390
- ...mockSettings,
391
- onCostEvent: cb,
392
- });
393
-
394
- expect((s as any).onCostEvent).toBe(cb);
395
- });
396
- });
397
-
398
- describe('| getMessageCreateInput', () => {
399
- it('| should create input with default settings', () => {
400
- const conversation: DyFM_AI_Message[] = [
401
- { role: DyFM_AI_MessageRole.user, content: 'Test question' },
402
- ];
403
-
404
- const result = (service as any).getMessageCreateInput({
405
- conversation: conversation,
406
- issuer: 'test-issuer',
407
- });
408
-
409
- expect(result.model).toBe(DyFM_OAI_Model.gpt4o_mini);
410
- expect(result.messages).toBeDefined();
411
- expect(result.temperature).toBe(0.7);
412
- expect(result.max_completion_tokens).toBe(1000);
413
- });
414
-
415
- it('| should use provided settings when available', () => {
416
- const conversation: DyFM_AI_Message[] = [
417
- { role: DyFM_AI_MessageRole.user, content: 'Test question' },
418
- ];
419
-
420
- const customSettings: DyFM_OAI_CallSettings = {
421
- useModel: DyFM_OAI_Model.gpt4o,
422
- temperature: 0.9,
423
- maxTokens: 2000,
424
- };
425
-
426
- const result = (service as any).getMessageCreateInput({
427
- conversation: conversation,
428
- issuer: 'test-issuer',
429
- settings: customSettings,
430
- });
431
-
432
- expect(result.model).toBe(DyFM_OAI_Model.gpt4o);
433
- expect(result.temperature).toBe(0.9);
434
- expect(result.max_completion_tokens).toBe(2000);
435
- });
436
-
437
- it('| should handle errors', () => {
438
- // Test that the method handles edge cases gracefully
439
- // Passing null conversation may or may not throw depending on implementation
440
- try {
441
- const result = (service as any).getMessageCreateInput({
442
- conversation: null as any,
443
- issuer: 'test-issuer',
444
- });
445
- // If it doesn't throw, verify it returns something
446
- expect(result).toBeDefined();
447
- } catch (error) {
448
- // If it throws, it should be a DyFM_Error
449
- expect(error).toBeInstanceOf(DyFM_Error);
450
- }
451
- });
452
- });
453
-
454
- describe('| testConnection', () => {
455
- it('| should return true when connection is successful', async () => {
456
- const result = await service.testConnection('test-issuer');
457
-
458
- expect(result).toBe(true);
459
- expect(mockModelsList).toHaveBeenCalled();
460
- });
461
-
462
- it('| should return false when connection fails', async () => {
463
- const logSpy = spyOn(DyFM_Log, 'error');
464
- mockModelsList.and.throwError(new Error('Connection failed'));
465
-
466
- const result = await service.testConnection('test-issuer');
467
-
468
- expect(result).toBe(false);
469
- expect(logSpy).toHaveBeenCalled();
470
- });
471
- });
472
-
473
- describe('| properties', () => {
474
- it('| should have aiProvider property', () => {
475
- expect(service.aiProvider).toBeDefined();
476
- });
477
-
478
- it('| should have capabilities property', () => {
479
- expect(service.capabilities).toBeDefined();
480
- expect(service.capabilities.chat).toBe(true);
481
- expect(service.capabilities.embeddings).toBe(true);
482
- });
483
-
484
- it('| should have predefinedRequests property', () => {
485
- expect(service.predefinedRequests).toBeDefined();
486
- });
487
- });
488
- });
489
-
1
+
2
+ import { DyNTS_OAI_LLM_ServiceBase } from './oai-llm.service-base';
3
+ import { DyFM_OAI_Settings, DyFM_OAI_Model, DyFM_OAI_CallSettings } from '@futdevpro/fsm-dynamo/ai/open-ai';
4
+ import { DyFM_AI_MessageRole, DyFM_AI_Message } from '@futdevpro/fsm-dynamo/ai';
5
+ import { DyFM_Error, DyFM_Log } from '@futdevpro/fsm-dynamo';
6
+ import { ChatCompletion } from 'openai/resources';
7
+
8
+ describe('| DyNTS_OAI_LLM_ServiceBase', () => {
9
+ let service: DyNTS_OAI_LLM_ServiceBase;
10
+ let mockOpenAI: any;
11
+ let mockChatCompletionsCreate: jasmine.Spy;
12
+ let mockModelsList: jasmine.Spy;
13
+
14
+ const mockSettings: DyFM_OAI_Settings = {
15
+ config: {
16
+ apiKey: 'test-api-key',
17
+ organization: 'test-org',
18
+ },
19
+ defaultSettings: {
20
+ useModel: DyFM_OAI_Model.gpt4o_mini,
21
+ systemPrompt: 'Test system prompt',
22
+ temperature: 0.7,
23
+ maxTokens: 1000,
24
+ },
25
+ };
26
+
27
+ beforeEach(() => {
28
+ mockChatCompletionsCreate = jasmine.createSpy('chat.completions.create').and.returnValue({
29
+ id: 'chatcmpl-123',
30
+ object: 'chat.completion',
31
+ created: 1234567890,
32
+ model: 'gpt-4o-mini',
33
+ choices: [
34
+ {
35
+ index: 0,
36
+ message: {
37
+ role: 'assistant',
38
+ content: 'Test response',
39
+ },
40
+ finish_reason: 'stop',
41
+ },
42
+ ],
43
+ usage: {
44
+ prompt_tokens: 10,
45
+ completion_tokens: 5,
46
+ total_tokens: 15,
47
+ },
48
+ });
49
+
50
+ mockModelsList = jasmine.createSpy('models.list').and.returnValue({
51
+ data: [],
52
+ });
53
+
54
+ mockOpenAI = {
55
+ chat: {
56
+ completions: {
57
+ create: mockChatCompletionsCreate,
58
+ },
59
+ },
60
+ models: {
61
+ list: mockModelsList,
62
+ },
63
+ };
64
+
65
+ service = new DyNTS_OAI_LLM_ServiceBase(mockSettings);
66
+ (service as any).openai = mockOpenAI;
67
+ });
68
+
69
+ describe('| constructor', () => {
70
+ it('| should initialize with provided settings', () => {
71
+ expect(service.defaultSettings).toEqual(mockSettings.defaultSettings);
72
+ expect(service.aiProvider).toBeDefined();
73
+ expect(service.capabilities).toBeDefined();
74
+ });
75
+
76
+ it('| should initialize with default settings when not provided', () => {
77
+ // Creating service without settings requires OPENAI_API_KEY env var or throws
78
+ // Test that service creation with settings works and uses defaults for missing values
79
+ const serviceWithPartialSettings = new DyNTS_OAI_LLM_ServiceBase({
80
+ config: { apiKey: 'test-key' },
81
+ });
82
+ expect(serviceWithPartialSettings.defaultSettings).toBeDefined();
83
+ });
84
+
85
+ it('| should set defaultModel from defaultSettings', () => {
86
+ expect(service.defaultModel).toBe(DyFM_OAI_Model.gpt4o_mini);
87
+ });
88
+ });
89
+
90
+ describe('| setup', () => {
91
+ it('| should reinitialize OpenAI client with new config', () => {
92
+ const newConfig = {
93
+ apiKey: 'new-api-key',
94
+ organization: 'new-org',
95
+ };
96
+
97
+ service.setup(newConfig);
98
+
99
+ expect((service as any).openai).toBeDefined();
100
+ });
101
+ });
102
+
103
+ describe('| requestYesNo', () => {
104
+ it('| should return true when answer contains YES', async () => {
105
+ mockChatCompletionsCreate.and.returnValue({
106
+ choices: [
107
+ {
108
+ message: {
109
+ content: 'YES',
110
+ },
111
+ },
112
+ ],
113
+ });
114
+
115
+ spyOn(service as any, 'convertAnswerToBoolean').and.returnValue(true);
116
+
117
+ const result = await service.requestYesNo({
118
+ message: 'Is this a test?',
119
+ issuer: 'test-issuer',
120
+ });
121
+
122
+ expect(result).toBe(true);
123
+ expect((service as any).convertAnswerToBoolean).toHaveBeenCalled();
124
+ });
125
+
126
+ it('| should append predefined request to message', async () => {
127
+ spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve('YES'));
128
+
129
+ await service.requestYesNo({
130
+ message: 'Is this a test?',
131
+ issuer: 'test-issuer',
132
+ });
133
+
134
+ expect(service.requestSimpleMessage).toHaveBeenCalled();
135
+ const callArgs = (service.requestSimpleMessage as jasmine.Spy).calls.mostRecent().args[0];
136
+ expect(callArgs.message).toContain('Is this a test?');
137
+ });
138
+ });
139
+
140
+ describe('| requestPercentage', () => {
141
+ it('| should return number from answer', async () => {
142
+ spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve('75'));
143
+ spyOn(service as any, 'convertAnswerToNumber').and.returnValue(75);
144
+
145
+ const result = await service.requestPercentage({
146
+ message: 'What percentage?',
147
+ issuer: 'test-issuer',
148
+ });
149
+
150
+ expect(result).toBe(75);
151
+ expect((service as any).convertAnswerToNumber).toHaveBeenCalled();
152
+ });
153
+ });
154
+
155
+ describe('| requestSelect', () => {
156
+ it('| should return selected option', async () => {
157
+ const options = ['option1', 'option2', 'option3'];
158
+ spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve('option1'));
159
+ spyOn(service as any, 'stringifySelectOptions').and.returnValue(['option1', 'option2', 'option3']);
160
+ spyOn(service as any, 'getTextListAsText').and.returnValue('option1, option2, option3');
161
+ spyOn(service as any, 'convertAnswerToSelectOption').and.returnValue('option1');
162
+
163
+ const result = await service.requestSelect({
164
+ message: 'Select an option',
165
+ selectFrom: options,
166
+ issuer: 'test-issuer',
167
+ });
168
+
169
+ expect(result).toBe('option1');
170
+ });
171
+ });
172
+
173
+ describe('| requestMultiselect', () => {
174
+ it('| should return multiple selected options', async () => {
175
+ const options = ['option1', 'option2', 'option3'];
176
+ spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve('option1 and option2'));
177
+ spyOn(service as any, 'stringifySelectOptions').and.returnValue(['option1', 'option2', 'option3']);
178
+ spyOn(service as any, 'getTextListAsText').and.returnValue('option1, option2, option3');
179
+ spyOn(service as any, 'convertAnswerToSelectOptions').and.returnValue(['option1', 'option2']);
180
+
181
+ const result = await service.requestMultiselect({
182
+ message: 'Select options',
183
+ options: options,
184
+ issuer: 'test-issuer',
185
+ });
186
+
187
+ expect(result).toEqual(['option1', 'option2']);
188
+ });
189
+ });
190
+
191
+ describe('| requestJSON', () => {
192
+ it('| should return parsed JSON', async () => {
193
+ const jsonString = '{"key": "value"}';
194
+ spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve(jsonString));
195
+ spyOn(service as any, 'convertAnswerToJSON').and.returnValue({ key: 'value' });
196
+
197
+ const result = await service.requestJSON<{ key: string }>({
198
+ message: 'Return JSON',
199
+ issuer: 'test-issuer',
200
+ });
201
+
202
+ expect(result).toEqual({ key: 'value' });
203
+ });
204
+ });
205
+
206
+ describe('| requestList', () => {
207
+ it('| should return list of items', async () => {
208
+ spyOn(service, 'requestSimpleMessage').and.returnValue(Promise.resolve('item1, item2, item3'));
209
+ spyOn(service as any, 'convertAnswerToList').and.returnValue(['item1', 'item2', 'item3']);
210
+
211
+ const result = await service.requestList<string>({
212
+ message: 'Return a list',
213
+ issuer: 'test-issuer',
214
+ });
215
+
216
+ expect(result).toEqual(['item1', 'item2', 'item3']);
217
+ });
218
+ });
219
+
220
+ describe('| requestSimpleMessage', () => {
221
+ it('| should resolve message and return answer', async () => {
222
+ spyOn(service as any, 'logQuestion');
223
+ spyOn(service as any, 'logAnswer');
224
+ spyOn(service, 'resolveMessage').and.returnValue(Promise.resolve('Test response'));
225
+
226
+ const result = await service.requestSimpleMessage({
227
+ message: 'Test question',
228
+ issuer: 'test-issuer',
229
+ });
230
+
231
+ expect(result).toBe('Test response');
232
+ expect((service as any).logQuestion).toHaveBeenCalled();
233
+ expect((service as any).logAnswer).toHaveBeenCalled();
234
+ });
235
+
236
+ it('| should add user message to conversation', async () => {
237
+ const conversation: DyFM_AI_Message[] = [];
238
+ spyOn(service, 'resolveMessage').and.returnValue(Promise.resolve('Test response'));
239
+
240
+ await service.requestSimpleMessage({
241
+ message: 'Test question',
242
+ conversation: conversation,
243
+ issuer: 'test-issuer',
244
+ });
245
+
246
+ expect(conversation.length).toBe(1);
247
+ expect(conversation[0].role).toBe(DyFM_AI_MessageRole.user);
248
+ expect(conversation[0].content).toBe('Test question');
249
+ });
250
+ });
251
+
252
+ describe('| resolveMessage', () => {
253
+ it('| should return string answer when getFullResponse is false', async () => {
254
+ const conversation: DyFM_AI_Message[] = [
255
+ { role: DyFM_AI_MessageRole.user, content: 'Test question' },
256
+ ];
257
+
258
+ const result = await service.resolveMessage({
259
+ conversation: conversation,
260
+ issuer: 'test-issuer',
261
+ getFullResponse: false,
262
+ });
263
+
264
+ expect(result).toBe('Test response');
265
+ expect(mockChatCompletionsCreate).toHaveBeenCalled();
266
+ });
267
+
268
+ it('| should return full ChatCompletion when getFullResponse is true', async () => {
269
+ const conversation: DyFM_AI_Message[] = [
270
+ { role: DyFM_AI_MessageRole.user, content: 'Test question' },
271
+ ];
272
+
273
+ const fullResponse: ChatCompletion = {
274
+ id: 'chatcmpl-123',
275
+ object: 'chat.completion',
276
+ created: 1234567890,
277
+ model: 'gpt-4o-mini',
278
+ choices: [
279
+ {
280
+ index: 0,
281
+ message: {
282
+ role: 'assistant',
283
+ content: 'Test response',
284
+ refusal: null,
285
+ },
286
+ logprobs: null,
287
+ finish_reason: 'stop',
288
+ },
289
+ ],
290
+ usage: {
291
+ prompt_tokens: 10,
292
+ completion_tokens: 5,
293
+ total_tokens: 15,
294
+ },
295
+ };
296
+
297
+ mockChatCompletionsCreate.and.returnValue(fullResponse);
298
+
299
+ const result = await service.resolveMessage({
300
+ conversation: conversation,
301
+ issuer: 'test-issuer',
302
+ getFullResponse: true,
303
+ });
304
+
305
+ expect(result).toEqual(fullResponse);
306
+ });
307
+
308
+ it('| should add system message to conversation', async () => {
309
+ const conversation: DyFM_AI_Message[] = [
310
+ { role: DyFM_AI_MessageRole.user, content: 'Test question' },
311
+ ];
312
+
313
+ spyOn(service as any, 'getDefaultSystemMessage').and.returnValue({
314
+ role: DyFM_AI_MessageRole.system,
315
+ content: 'System prompt',
316
+ });
317
+
318
+ await service.resolveMessage({
319
+ conversation: conversation,
320
+ issuer: 'test-issuer',
321
+ });
322
+
323
+ expect(conversation[0].role).toBe(DyFM_AI_MessageRole.system);
324
+ });
325
+
326
+ it('| should handle API errors', async () => {
327
+ const conversation: DyFM_AI_Message[] = [
328
+ { role: DyFM_AI_MessageRole.user, content: 'Test question' },
329
+ ];
330
+
331
+ mockChatCompletionsCreate.and.throwError(new Error('API Error'));
332
+
333
+ try {
334
+ await service.resolveMessage({
335
+ conversation: conversation,
336
+ issuer: 'test-issuer',
337
+ });
338
+ fail('Should have thrown an error');
339
+ } catch (error) {
340
+ expect(error).toBeInstanceOf(DyFM_Error);
341
+ expect((error as DyFM_Error)._errorCode).toContain('DyNTS-OLSB-RM0');
342
+ }
343
+ });
344
+ });
345
+
346
+ describe('| FR-002 cost-event emit', () => {
347
+
348
+ it('| should emit cost-event with llm-completion callType for single user message', async () => {
349
+ const received: any[] = [];
350
+ (service as any).onCostEvent = (e: any) => { received.push(e); };
351
+
352
+ await service.resolveMessage({
353
+ conversation: [{ role: DyFM_AI_MessageRole.user, content: 'Q?' }],
354
+ issuer: 'llm-issuer-1',
355
+ });
356
+
357
+ expect(received.length).toBe(1);
358
+ expect(received[0].callType).toBe('llm-completion');
359
+ expect(received[0].provider).toBe('open-ai');
360
+ expect(received[0].model).toBe(DyFM_OAI_Model.gpt4o_mini);
361
+ expect(received[0].tokensUsed.input).toBe(10);
362
+ expect(received[0].tokensUsed.output).toBe(5);
363
+ expect(received[0].tokensUsed.total).toBe(15);
364
+ expect(received[0].issuer).toBe('llm-issuer-1');
365
+ expect(received[0].timestamp).toBeInstanceOf(Date);
366
+ expect(received[0].durationMs).toBeGreaterThanOrEqual(0);
367
+ expect(received[0].estimatedCostUsd).toBeUndefined();
368
+ });
369
+
370
+ it('| should emit cost-event with llm-chat callType for multi-message conversation', async () => {
371
+ const received: any[] = [];
372
+ (service as any).onCostEvent = (e: any) => { received.push(e); };
373
+
374
+ await service.resolveMessage({
375
+ conversation: [
376
+ { role: DyFM_AI_MessageRole.user, content: 'Q1?' },
377
+ { role: DyFM_AI_MessageRole.assistant, content: 'A1.' },
378
+ { role: DyFM_AI_MessageRole.user, content: 'Q2?' },
379
+ ],
380
+ issuer: 'llm-issuer-chat',
381
+ });
382
+
383
+ expect(received.length).toBe(1);
384
+ expect(received[0].callType).toBe('llm-chat');
385
+ });
386
+
387
+ it('| constructor accepts onCostEvent option (non-breaking additive)', () => {
388
+ const cb: any = jasmine.createSpy('onCostEvent');
389
+ const s: DyNTS_OAI_LLM_ServiceBase = new DyNTS_OAI_LLM_ServiceBase({
390
+ ...mockSettings,
391
+ onCostEvent: cb,
392
+ });
393
+
394
+ expect((s as any).onCostEvent).toBe(cb);
395
+ });
396
+ });
397
+
398
+ describe('| getMessageCreateInput', () => {
399
+ it('| should create input with default settings', () => {
400
+ const conversation: DyFM_AI_Message[] = [
401
+ { role: DyFM_AI_MessageRole.user, content: 'Test question' },
402
+ ];
403
+
404
+ const result = (service as any).getMessageCreateInput({
405
+ conversation: conversation,
406
+ issuer: 'test-issuer',
407
+ });
408
+
409
+ expect(result.model).toBe(DyFM_OAI_Model.gpt4o_mini);
410
+ expect(result.messages).toBeDefined();
411
+ expect(result.temperature).toBe(0.7);
412
+ expect(result.max_completion_tokens).toBe(1000);
413
+ });
414
+
415
+ it('| should use provided settings when available', () => {
416
+ const conversation: DyFM_AI_Message[] = [
417
+ { role: DyFM_AI_MessageRole.user, content: 'Test question' },
418
+ ];
419
+
420
+ const customSettings: DyFM_OAI_CallSettings = {
421
+ useModel: DyFM_OAI_Model.gpt4o,
422
+ temperature: 0.9,
423
+ maxTokens: 2000,
424
+ };
425
+
426
+ const result = (service as any).getMessageCreateInput({
427
+ conversation: conversation,
428
+ issuer: 'test-issuer',
429
+ settings: customSettings,
430
+ });
431
+
432
+ expect(result.model).toBe(DyFM_OAI_Model.gpt4o);
433
+ expect(result.temperature).toBe(0.9);
434
+ expect(result.max_completion_tokens).toBe(2000);
435
+ });
436
+
437
+ it('| should handle errors', () => {
438
+ // Test that the method handles edge cases gracefully
439
+ // Passing null conversation may or may not throw depending on implementation
440
+ try {
441
+ const result = (service as any).getMessageCreateInput({
442
+ conversation: null as any,
443
+ issuer: 'test-issuer',
444
+ });
445
+ // If it doesn't throw, verify it returns something
446
+ expect(result).toBeDefined();
447
+ } catch (error) {
448
+ // If it throws, it should be a DyFM_Error
449
+ expect(error).toBeInstanceOf(DyFM_Error);
450
+ }
451
+ });
452
+ });
453
+
454
+ describe('| testConnection', () => {
455
+ it('| should return true when connection is successful', async () => {
456
+ const result = await service.testConnection('test-issuer');
457
+
458
+ expect(result).toBe(true);
459
+ expect(mockModelsList).toHaveBeenCalled();
460
+ });
461
+
462
+ it('| should return false when connection fails', async () => {
463
+ const logSpy = spyOn(DyFM_Log, 'error');
464
+ mockModelsList.and.throwError(new Error('Connection failed'));
465
+
466
+ const result = await service.testConnection('test-issuer');
467
+
468
+ expect(result).toBe(false);
469
+ expect(logSpy).toHaveBeenCalled();
470
+ });
471
+ });
472
+
473
+ describe('| properties', () => {
474
+ it('| should have aiProvider property', () => {
475
+ expect(service.aiProvider).toBeDefined();
476
+ });
477
+
478
+ it('| should have capabilities property', () => {
479
+ expect(service.capabilities).toBeDefined();
480
+ expect(service.capabilities.chat).toBe(true);
481
+ expect(service.capabilities.embeddings).toBe(true);
482
+ });
483
+
484
+ it('| should have predefinedRequests property', () => {
485
+ expect(service.predefinedRequests).toBeDefined();
486
+ });
487
+ });
488
+ });
489
+