@futdevpro/nts-dynamo 1.15.40 → 1.15.41

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 (325) 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/.github/workflows/main.yml +432 -432
  17. package/.vscode/settings.json +10 -10
  18. package/HOWTO.md +15 -15
  19. package/LICENSE +21 -21
  20. package/__documentations/2026-06-01-fr047-p2p3-function-calling.md +81 -0
  21. package/__documentations/nts-integration-tests-2026-03-17.md +26 -26
  22. package/_specifications/BACKLOG.md +92 -92
  23. package/_specifications/TODO.md +15 -15
  24. package/_specifications/agent.md +138 -138
  25. package/build/_modules/server/errors/errors.data-service.d.ts.map +1 -1
  26. package/build/_modules/server/errors/errors.data-service.js +12 -1
  27. package/build/_modules/server/errors/errors.data-service.js.map +1 -1
  28. package/eslint.config.js +3 -3
  29. package/nodemon.json +24 -24
  30. package/package.json +1 -1
  31. package/pnpm-workspace.yaml +8 -8
  32. package/scripts/run-coverage-tests.js +28 -28
  33. package/spec/support/helpers/spec-reporter-loader.js +359 -359
  34. package/spec/support/helpers/ts-node-helper.js +93 -93
  35. package/spec/support/jasmine.coverage.json +24 -24
  36. package/spec/support/jasmine.json +24 -24
  37. package/src/_collections/archive.util.spec.ts +57 -57
  38. package/src/_collections/archive.util.ts +18 -18
  39. package/src/_collections/atlas-default-db-options.const.ts +9 -9
  40. package/src/_collections/default-fallback-cache-max-age.const.spec.ts +11 -11
  41. package/src/_collections/default-fallback-cache-max-age.const.ts +2 -2
  42. package/src/_collections/default-not-found-page.const.spec.ts +19 -19
  43. package/src/_collections/default-not-found-page.const.ts +22 -22
  44. package/src/_collections/default-socket-path.const.spec.ts +12 -12
  45. package/src/_collections/default-socket-path.const.ts +2 -2
  46. package/src/_collections/get-environment-settings.util.spec.ts +210 -210
  47. package/src/_collections/get-environment-settings.util.ts +48 -48
  48. package/src/_collections/sample.env +21 -21
  49. package/src/_collections/star.controller.spec.ts +224 -224
  50. package/src/_collections/star.controller.ts +129 -129
  51. package/src/_enums/data-model-type.enum.ts +14 -14
  52. package/src/_enums/data-service-function.enum.ts +24 -24
  53. package/src/_enums/predefined-data-types.enum.ts +16 -16
  54. package/src/_enums/route-security.enum.ts +12 -12
  55. package/src/_models/control-models/api-call-params.control-model.spec.ts +152 -152
  56. package/src/_models/control-models/api-call-params.control-model.ts +142 -142
  57. package/src/_models/control-models/app-ext-system-controls.control-model.spec.ts +52 -52
  58. package/src/_models/control-models/app-ext-system-controls.control-model.ts +9 -9
  59. package/src/_models/control-models/app-params.control-model.spec.ts +225 -225
  60. package/src/_models/control-models/app-params.control-model.ts +136 -136
  61. package/src/_models/control-models/app-system-controls.control-model.spec.ts +31 -31
  62. package/src/_models/control-models/app-system-controls.control-model.ts +9 -9
  63. package/src/_models/control-models/endpoint-params.control-model.spec.ts +578 -578
  64. package/src/_models/control-models/endpoint-params.control-model.ts +526 -526
  65. package/src/_models/control-models/http-settings.control-model.spec.ts +77 -77
  66. package/src/_models/control-models/http-settings.control-model.ts +37 -37
  67. package/src/_models/control-models/system-control.control-model.spec.ts +27 -27
  68. package/src/_models/control-models/system-control.control-model.ts +12 -12
  69. package/src/_models/interfaces/certification-settings.interface.ts +7 -7
  70. package/src/_models/interfaces/environment-settings.interface.ts +59 -59
  71. package/src/_models/interfaces/global-log-settings.interface.ts +144 -144
  72. package/src/_models/interfaces/global-service-settings.interface.ts +47 -47
  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-document.util.spec.ts +209 -209
  84. package/src/_modules/ai/_modules/document-ai/_collections/dai-document.util.ts +85 -85
  85. package/src/_modules/ai/_modules/document-ai/_enums/dai-compare-result-type.enum.ts +7 -7
  86. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-doc-chunk.data-model.ts +146 -146
  87. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-doc-page.data-model.ts +162 -162
  88. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-document.data-model.ts +99 -99
  89. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-doc-chunk-compare-result.interface.ts +18 -18
  90. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-doc-page-compare-result.interface.ts +19 -19
  91. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-document-compare-result.interface.ts +25 -25
  92. package/src/_modules/ai/_modules/document-ai/index.ts +28 -28
  93. package/src/_modules/ai/_modules/fdp-ai/_services/fdpai-user-key.control-service.ts +189 -189
  94. package/src/_modules/ai/_modules/fdp-ai/index.ts +5 -5
  95. package/src/_modules/ai/_modules/open-ai/_collections/oai-global-settings.const.ts +9 -9
  96. package/src/_modules/ai/_modules/open-ai/_collections/oai-llm-predefined-requests-hu.conts.ts +82 -82
  97. package/src/_modules/ai/_modules/open-ai/_collections/oai-llm-predefined-requests.conts.ts +75 -75
  98. package/src/_modules/ai/_modules/open-ai/_enums/oai-gpt-message-role.enum.ts +45 -45
  99. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-global-settings.interface.ts +7 -7
  100. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-gpt-message.interface.ts +7 -7
  101. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-llm-predefined-requests.interface.ts +57 -57
  102. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-doc-chunk-data.service.ts +292 -292
  103. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-document.data-service.spec.ts +342 -342
  104. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-vector-data.service.spec.ts +550 -550
  105. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-vector-data.service.ts +630 -630
  106. package/src/_modules/ai/_modules/open-ai/_services/oai-embedding.control-service.spec.ts +332 -332
  107. package/src/_modules/ai/_modules/open-ai/_services/oai-llm-chat.service-base.spec.ts +462 -462
  108. package/src/_modules/ai/_modules/open-ai/_services/oai-llm-chat.service-base.ts +634 -634
  109. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.spec.ts +489 -489
  110. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.tools.spec.ts +173 -173
  111. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.ts +1033 -1033
  112. package/src/_modules/ai/_modules/open-ai/_services/oai-user-key.control-service.ts +157 -157
  113. package/src/_modules/ai/_services/ai-embedding.service-base.spec.ts +98 -98
  114. package/src/_modules/ai/_services/ai-embedding.service-base.ts +48 -48
  115. package/src/_modules/ai/_services/ai-llm-chat.service-base.spec.ts +229 -229
  116. package/src/_modules/ai/_services/ai-llm-chat.service-base.ts +68 -68
  117. package/src/_modules/ai/_services/ai-llm.service-base.spec.ts +250 -250
  118. package/src/_modules/ai/_services/ai-llm.service-base.ts +519 -519
  119. package/src/_modules/ai/_services/ai-provider.service-base.spec.ts +158 -158
  120. package/src/_modules/ai/_services/ai-user-key.service-base.ts +59 -59
  121. package/src/_modules/ai/index.ts +13 -13
  122. package/src/_modules/assistant/_collections/ass-global-settings.const.ts +13 -13
  123. package/src/_modules/assistant/_collections/ass.util.spec.ts +176 -176
  124. package/src/_modules/assistant/_collections/ass.util.ts +50 -50
  125. package/src/_modules/assistant/_models/ass-global-settings.interface.ts +15 -15
  126. package/src/_modules/assistant/_services/ass-io.control-service.spec.ts +140 -140
  127. package/src/_modules/assistant/_services/ass-main.control-service.spec.ts +192 -192
  128. package/src/_modules/assistant/_services/ass-main.control-service.ts +107 -107
  129. package/src/_modules/bot/_collections/bot-default-commands.const.ts +12 -12
  130. package/src/_modules/bot/_collections/bot-global-settings.const.ts +39 -39
  131. package/src/_modules/bot/_models/bot-channel-wrapper.interface.ts +62 -62
  132. package/src/_modules/bot/_models/bot-command.interface.ts +8 -8
  133. package/src/_modules/bot/_models/bot-global-settings.interface.ts +96 -96
  134. package/src/_modules/bot/_models/bot-last-mention-date.interface.ts +6 -6
  135. package/src/_modules/bot/_models/bot-last-message-date.interface.ts +5 -5
  136. package/src/_modules/bot/_models/bot-user-wrapper.interface.ts +41 -41
  137. package/src/_modules/bot/_modules/discord-bot/_models/dib-platform.types.ts +9 -9
  138. package/src/_modules/bot/_modules/discord-bot/_services/dib-messaging-provider.control-service.spec.ts +431 -431
  139. package/src/_modules/bot/_modules/dynamo-bot/_collections/dyb-operations.util.spec.ts +160 -160
  140. package/src/_modules/bot/_modules/dynamo-bot/_collections/dyb-operations.util.ts +55 -55
  141. package/src/_modules/bot/_modules/dynamo-bot/_models/dyb-platform.types.ts +15 -15
  142. package/src/_modules/bot/_modules/dynamo-bot/_services/dyb-messaging-provider.control-service.spec.ts +374 -374
  143. package/src/_modules/bot/_modules/dynamo-bot/_services/dyb-messaging-provider.control-service.ts +447 -447
  144. package/src/_modules/bot/_modules/dynamo-bot/index.ts +15 -15
  145. package/src/_modules/bot/_modules/slack-bot/_models/slb-platform.types.ts +9 -9
  146. package/src/_modules/bot/_modules/slack-bot/_services/slb-messaging-provider.control-service.spec.ts +344 -344
  147. package/src/_modules/bot/_modules/slack-bot/_services/slb-messaging-provider.control-service.ts +197 -197
  148. package/src/_modules/bot/_modules/teams-bot/_models/teb-platform.types.ts +9 -9
  149. package/src/_modules/bot/_modules/teams-bot/_services/teb-messaging-provider.control-service.spec.ts +345 -345
  150. package/src/_modules/bot/_modules/teams-bot/_services/teb-messaging-provider.control-service.ts +197 -197
  151. package/src/_modules/bot/_services/bot-commands.control-service.spec.ts +116 -116
  152. package/src/_modules/bot/_services/bot-io.control-service.spec.ts +285 -285
  153. package/src/_modules/bot/_services/bot-main.control-service.spec.ts +208 -208
  154. package/src/_modules/bot/_services/bot-messaging-provider.service-base.spec.ts +349 -349
  155. package/src/_modules/bot/_services/bot-routines.control-service.spec.ts +111 -111
  156. package/src/_modules/custom-data/custom-data.controller.spec.ts +49 -49
  157. package/src/_modules/custom-data/custom-data.controller.ts +67 -67
  158. package/src/_modules/custom-data/custom-data.data-service.spec.ts +54 -54
  159. package/src/_modules/custom-data/custom-data.data-service.ts +21 -21
  160. package/src/_modules/custom-data/get-custom-data-routing-module.util.spec.ts +28 -28
  161. package/src/_modules/custom-data/get-custom-data-routing-module.util.ts +24 -24
  162. package/src/_modules/custom-data/index.ts +9 -9
  163. package/src/_modules/defaults/_collections/default-endpoints.util.ts +487 -487
  164. package/src/_modules/defaults/_models/default-user.data-model.ts +72 -72
  165. package/src/_modules/defaults/_services/default-auth.service.spec.ts +269 -269
  166. package/src/_modules/defaults/_services/default-auth.service.ts +177 -177
  167. package/src/_modules/defaults/_services/default-socket-events.service.spec.ts +42 -42
  168. package/src/_modules/defaults/_services/default-socket-events.service.ts +61 -61
  169. package/src/_modules/defaults/_services/default-user.data-service.spec.ts +187 -187
  170. package/src/_modules/defaults/_services/default-user.data-service.ts +98 -98
  171. package/src/_modules/defaults/index.ts +17 -17
  172. package/src/_modules/discord-assistant/_collections/dias-global-settings.const.ts +19 -19
  173. package/src/_modules/discord-assistant/_collections/dias.util.spec.ts +366 -366
  174. package/src/_modules/discord-assistant/_collections/dias.util.ts +132 -132
  175. package/src/_modules/discord-assistant/_models/dias-global-settings.interface.ts +19 -19
  176. package/src/_modules/discord-assistant/_models/dias-knowledge.data-model.ts +52 -52
  177. package/src/_modules/discord-assistant/_services/dias-chunk.data-service.ts +177 -177
  178. package/src/_modules/discord-assistant/_services/dias-io.control-service.spec.ts +108 -108
  179. package/src/_modules/discord-assistant/_services/dias-io.control-service.ts +69 -69
  180. package/src/_modules/discord-assistant/_services/dias-main.control-service.spec.ts +22 -22
  181. package/src/_modules/discord-assistant/_services/dias-main.control-service.ts +27 -27
  182. package/src/_modules/discord-assistant/_services/dias.service-base.spec.ts +195 -195
  183. package/src/_modules/discord-assistant/_services/dias.service-base.ts +76 -76
  184. package/src/_modules/discord-assistant/index.ts +38 -38
  185. package/src/_modules/discord-assistant-voiced/_services/dias-discord-bot.control-service.spec.ts +34 -34
  186. package/src/_modules/discord-assistant-voiced/_services/dias-discord-bot.control-service.ts +11 -11
  187. package/src/_modules/discord-assistant-voiced/index.ts +36 -36
  188. package/src/_modules/discord-bot/_collections/dibo-default-commands.const.ts +16 -16
  189. package/src/_modules/discord-bot/_collections/dibo-global-settings.conts.ts +55 -55
  190. package/src/_modules/discord-bot/_collections/dibo-operations.util.spec.ts +214 -214
  191. package/src/_modules/discord-bot/_collections/dibo-operations.util.ts +387 -387
  192. package/src/_modules/discord-bot/_models/dibo-command.interface.ts +12 -12
  193. package/src/_modules/discord-bot/_models/dibo-global-settings.interface.ts +98 -98
  194. package/src/_modules/discord-bot/_models/dibo-last-mention-date.inteface.ts +7 -7
  195. package/src/_modules/discord-bot/_models/dibo-last-message-date.interface.ts +6 -6
  196. package/src/_modules/discord-bot/_services/dibo-commands.control-service.spec.ts +154 -154
  197. package/src/_modules/discord-bot/_services/dibo-commands.control-service.ts +153 -153
  198. package/src/_modules/discord-bot/_services/dibo-io.control-service.spec.ts +264 -264
  199. package/src/_modules/discord-bot/_services/dibo-io.control-service.ts +306 -306
  200. package/src/_modules/discord-bot/_services/dibo-main.control-service.spec.ts +408 -408
  201. package/src/_modules/discord-bot/_services/dibo-main.control-service.ts +487 -487
  202. package/src/_modules/discord-bot/_services/dibo-routines.control-service.spec.ts +105 -105
  203. package/src/_modules/discord-bot/index.ts +36 -36
  204. package/src/_modules/local-vector-search/_enums/lvs-search-mode.enum.ts +35 -35
  205. package/src/_modules/local-vector-search/_models/lvs-search-result.interface.ts +17 -17
  206. package/src/_modules/local-vector-search/_services/lvs-doc-chunk-data.service.spec.ts +418 -418
  207. package/src/_modules/local-vector-search/_services/lvs-doc-chunk-data.service.ts +276 -276
  208. package/src/_modules/local-vector-search/_services/lvs-local-vector-search.data-service.spec.ts +480 -480
  209. package/src/_modules/local-vector-search/_services/lvs-local-vector-search.data-service.ts +416 -416
  210. package/src/_modules/local-vector-search/_services/lvs-vector-pool.control-service.spec.ts +393 -393
  211. package/src/_modules/local-vector-search/_services/lvs-vector-pool.control-service.ts +220 -220
  212. package/src/_modules/local-vector-search/index.ts +11 -11
  213. package/src/_modules/messaging/README.md +354 -354
  214. package/src/_modules/messaging/_collections/get-messaging-routing-module.util.ts +26 -26
  215. package/src/_modules/messaging/_collections/msg-global-settings.const.ts +22 -22
  216. package/src/_modules/messaging/_collections/msg.util.spec.ts +226 -226
  217. package/src/_modules/messaging/_models/msg-global-settings.interface.ts +37 -37
  218. package/src/_modules/messaging/_services/msg-conversation.data-service.ts +146 -146
  219. package/src/_modules/messaging/_services/msg-events.service.spec.ts +219 -219
  220. package/src/_modules/messaging/_services/msg-events.service.ts +267 -267
  221. package/src/_modules/messaging/_services/msg-integration.control-service.ts +179 -179
  222. package/src/_modules/messaging/_services/msg-main.control-service.spec.ts +147 -147
  223. package/src/_modules/messaging/_services/msg-main.control-service.ts +571 -571
  224. package/src/_modules/messaging/_services/msg-message.data-service.ts +129 -129
  225. package/src/_modules/messaging/_services/msg.controller.spec.ts +201 -201
  226. package/src/_modules/messaging/index.ts +30 -30
  227. package/src/_modules/mock/app-extended-server.mock.ts +201 -201
  228. package/src/_modules/mock/app-integration-test.mock.ts +51 -51
  229. package/src/_modules/mock/app-params.mock.spec.ts +21 -21
  230. package/src/_modules/mock/app-params.mock.ts +9 -9
  231. package/src/_modules/mock/app-server.mock.ts +188 -188
  232. package/src/_modules/mock/auth-service.mock.spec.ts +47 -47
  233. package/src/_modules/mock/auth-service.mock.ts +28 -28
  234. package/src/_modules/mock/controller.mock.spec.ts +26 -26
  235. package/src/_modules/mock/controller.mock.ts +16 -16
  236. package/src/_modules/mock/data-model.mock.spec.ts +111 -111
  237. package/src/_modules/mock/data-model.mock.ts +82 -82
  238. package/src/_modules/mock/email-service-collection.mock.spec.ts +24 -24
  239. package/src/_modules/mock/email-service-collection.mock.ts +15 -15
  240. package/src/_modules/mock/email-service.mock.spec.ts +17 -17
  241. package/src/_modules/mock/email-service.mock.ts +20 -20
  242. package/src/_modules/mock/email-template.mock.html +14 -14
  243. package/src/_modules/mock/endpoint.mock.ts +91 -91
  244. package/src/_modules/mock/socket-client.mock.spec.ts +40 -40
  245. package/src/_modules/mock/socket-client.mock.ts +45 -45
  246. package/src/_modules/mock/socket-server.mock.spec.ts +44 -44
  247. package/src/_modules/mock/socket-server.mock.ts +46 -46
  248. package/src/_modules/oauth2/_routes/oauth2.controller.spec.ts +107 -107
  249. package/src/_modules/oauth2/_routes/oauth2.controller.ts +98 -98
  250. package/src/_modules/oauth2/_services/oauth2.auth-service.spec.ts +254 -254
  251. package/src/_modules/oauth2/_services/oauth2.auth-service.ts +232 -232
  252. package/src/_modules/oauth2/_services/oauth2.control-service.spec.ts +585 -585
  253. package/src/_modules/oauth2/_services/oauth2.control-service.ts +653 -653
  254. package/src/_modules/oauth2/index.ts +17 -17
  255. package/src/_modules/server/errors/errors.control-service.spec.ts +238 -238
  256. package/src/_modules/server/errors/errors.control-service.ts +85 -85
  257. package/src/_modules/server/errors/errors.controller.spec.ts +241 -241
  258. package/src/_modules/server/errors/errors.controller.ts +431 -431
  259. package/src/_modules/server/errors/errors.data-service.spec.ts +355 -355
  260. package/src/_modules/server/errors/errors.data-service.ts +14 -2
  261. package/src/_modules/server/index.ts +30 -30
  262. package/src/_modules/server/server-status/server-status-snapshot.control-service.spec.ts +70 -70
  263. package/src/_modules/server/server-status/server-status-snapshot.control-service.ts +17 -17
  264. package/src/_modules/server/server-status/server-status-snapshot.data-service.spec.ts +77 -77
  265. package/src/_modules/server/server-status/server-status-snapshot.data-service.ts +37 -37
  266. package/src/_modules/server/server-status/server-status.control-service.spec.ts +524 -524
  267. package/src/_modules/server/server-status/server-status.control-service.ts +336 -336
  268. package/src/_modules/server/server-status/server-status.controller.spec.ts +162 -162
  269. package/src/_modules/server/server-status/server-status.controller.ts +131 -131
  270. package/src/_modules/socket/_enums/socket-security.enum.ts +11 -11
  271. package/src/_modules/socket/_models/socket-client-service-params.control-model.spec.ts +32 -32
  272. package/src/_modules/socket/_models/socket-client-service-params.control-model.ts +22 -22
  273. package/src/_modules/socket/_models/socket-presence.control-model.spec.ts +164 -164
  274. package/src/_modules/socket/_models/socket-presence.control-model.ts +210 -210
  275. package/src/_modules/socket/_models/socket-server-service-params.control-model.spec.ts +46 -46
  276. package/src/_modules/socket/_models/socket-server-service-params.control-model.ts +22 -22
  277. package/src/_modules/socket/_services/socket-client.service.spec.ts +15 -15
  278. package/src/_modules/socket/_services/socket-client.service.ts +260 -260
  279. package/src/_modules/socket/_services/socket-server.service.spec.ts +11 -11
  280. package/src/_modules/socket/app-extended.integration.spec.ts +85 -85
  281. package/src/_modules/socket/app-extended.server.ts +630 -630
  282. package/src/_modules/socket/index.ts +42 -42
  283. package/src/_modules/test/get-test-routing-module.util.spec.ts +28 -28
  284. package/src/_modules/test/get-test-routing-module.util.ts +23 -23
  285. package/src/_modules/test/index.ts +11 -11
  286. package/src/_modules/test/test.controller.spec.ts +72 -72
  287. package/src/_modules/test/test.controller.ts +115 -115
  288. package/src/_modules/usage/get-usage-routing-module.util.ts +22 -22
  289. package/src/_modules/usage/index.ts +15 -15
  290. package/src/_modules/usage/usage.controller.spec.ts +81 -81
  291. package/src/_modules/usage/usage.controller.ts +126 -126
  292. package/src/_modules/usage/usage.data-service.spec.ts +332 -332
  293. package/src/_modules/usage/usage.data-service.ts +185 -185
  294. package/src/_services/base/api.service-base.spec.ts +125 -125
  295. package/src/_services/base/api.service-base.ts +74 -74
  296. package/src/_services/base/archive-data.service.spec.ts +196 -196
  297. package/src/_services/base/archive-data.service.ts +216 -216
  298. package/src/_services/base/data.service.spec.ts +674 -674
  299. package/src/_services/base/data.service.ts +2719 -2719
  300. package/src/_services/base/db.service.spec.ts +73 -73
  301. package/src/_services/base/db.service.ts +1575 -1575
  302. package/src/_services/base/singleton.service-base.spec.ts +28 -28
  303. package/src/_services/base/singleton.service-base.ts +24 -24
  304. package/src/_services/base/singleton.service.spec.ts +114 -114
  305. package/src/_services/base/singleton.service.ts +38 -38
  306. package/src/_services/core/api.service.spec.ts +140 -140
  307. package/src/_services/core/auth.service.spec.ts +159 -159
  308. package/src/_services/core/auth.service.ts +174 -174
  309. package/src/_services/core/email.service.spec.ts +85 -85
  310. package/src/_services/core/email.service.ts +742 -742
  311. package/src/_services/core/global.service.spec.ts +275 -275
  312. package/src/_services/core/global.service.ts +461 -461
  313. package/src/_services/core/service-collection.service.spec.ts +46 -46
  314. package/src/_services/core/service-collection.service.ts +6 -6
  315. package/src/_services/route/controller.service.spec.ts +53 -53
  316. package/src/_services/route/controller.service.ts +148 -148
  317. package/src/_services/route/routing-module.service.spec.ts +98 -98
  318. package/src/_services/route/routing-module.service.ts +330 -330
  319. package/src/_services/shared.static-service.spec.ts +99 -99
  320. package/src/_services/shared.static-service.ts +78 -78
  321. package/src/index.ts +95 -95
  322. package/tsconfig.app.json +12 -12
  323. package/tsconfig.json +42 -42
  324. package/.dynamo/logs/cicd-pipeline/output.log +0 -2567
  325. package/.dynamo/logs/cicd-pipeline/status.json +0 -319
@@ -1,1033 +1,1033 @@
1
- import { OpenAI } from 'openai';
2
-
3
- import { DyFM_OAI_Settings, DyFM_OAI_Model, DyFM_OAI_CallSettings, DyFM_OAI_Models } from '@futdevpro/fsm-dynamo/ai/open-ai';
4
- import { DyFM_AnyError, DyFM_Error, DyFM_Error_Settings, DyFM_Log, DyFM_notNull, DyFM_Object } from '@futdevpro/fsm-dynamo';
5
- import {
6
- DyFM_AI_Message,
7
- DyFM_AI_MessageRole,
8
- DyFM_AI_Provider,
9
- DyFM_AI_ProviderCapabilities,
10
- DyFM_AI_CallSettings,
11
- DyFM_AI_Config,
12
- DyFM_AI_LLM_Response,
13
- DyFM_AI_Tool,
14
- DyFM_AI_ToolCall,
15
- DyFM_AI_ModelInfo
16
- } from '@futdevpro/fsm-dynamo/ai';
17
-
18
- import { DyNTS_global_settings } from '../../../../../_collections/global-settings.const';
19
- import { ChatCompletion } from 'openai/resources';
20
- import { ChatCompletionCreateParamsBase, ChatCompletionMessageParam, ChatCompletionTool } from 'openai/resources/chat/completions';
21
- import { DyNTS_AI_CostEventCallback } from '../../../_models/interfaces/dynts-ai-cost-event-callback.interface';
22
- import { DyNTS_OAI_LLM_Predefined_Requests } from '../_models/interfaces/oai-llm-predefined-requests.interface';
23
- import { DyNTS_OAI_global_settings } from '../_collections/oai-global-settings.const';
24
- import { DyNTS_OAI_LLMDefaultPredefined_Requests } from '../_collections/oai-llm-predefined-requests.conts';
25
- import { DyNTS_AI_LLM_ServiceBase } from '../../../_services/ai-llm.service-base';
26
- import {
27
- DyFM_AI_Message_Input,
28
- DyFM_AI_GenericSelect_Input,
29
- DyFM_AI_GenericMultiSelect_Input,
30
- DyFM_AI_JSONKeysDescription_Input,
31
- DyFM_AI_JSONExactKeys_Input,
32
- DyFM_AI_MultiSelect_Input,
33
- DyFM_AI_ConversationBase_Input
34
- } from '../../../_models/ai-input-interfaces';
35
-
36
-
37
- /**
38
- * {
39
- * organization: 'org-XY',
40
- * apiKey: 'sk-XY',
41
- * project: 'proj_XY',
42
- * }
43
- */
44
-
45
- // Di Dictionary DiModel DiEz DiAz
46
-
47
- export class DyNTS_OAI_LLM_ServiceBase extends DyNTS_AI_LLM_ServiceBase {
48
-
49
- /* readonly defaultErrorUserMsg: string =
50
- `We encountered an unhandled Control Service Error, ` +
51
- `\nplease contact the responsible development team.`; */
52
-
53
- openai: OpenAI;
54
-
55
- readonly defaultSettings: DyFM_OAI_CallSettings;
56
-
57
- // Provider properties from DyNTS_AI_Provider_ServiceBase
58
- readonly aiProvider: DyFM_AI_Provider = DyFM_AI_Provider.OpenAI;
59
-
60
- readonly capabilities: DyFM_AI_ProviderCapabilities = {
61
- chat: true,
62
- embeddings: true,
63
- imageGeneration: true,
64
- vision: true,
65
- audioGeneration: true,
66
- audioAnalysis: true,
67
- functionCalling: true,
68
- streaming: true,
69
- batchOperations: true,
70
- maxContextLength: 128000,
71
- supportedModelTypes: ['gpt-4o', 'gpt-4o-mini', 'text-embedding-3-large', 'text-embedding-3-small']
72
- };
73
-
74
- /* get _debugLog(): boolean {
75
- return this.defaultSettings.debugLog;
76
- } */
77
-
78
- /* get defaultSystemPrompt(): string {
79
- return this.defaultSettings.systemPrompt;
80
- } */
81
-
82
- override get defaultModel(): DyFM_OAI_Model | string {
83
- return this.defaultSettings.useModel;
84
- }
85
-
86
- predefinedRequests: DyNTS_OAI_LLM_Predefined_Requests = DyNTS_OAI_LLMDefaultPredefined_Requests;
87
-
88
- constructor(
89
- set?: DyFM_OAI_Settings & { onCostEvent?: DyNTS_AI_CostEventCallback }
90
- ) {
91
- super();
92
-
93
- DyNTS_global_settings.openAi_settings ??= DyNTS_OAI_global_settings;
94
-
95
- this.openai = new OpenAI(
96
- set?.config ??
97
- {
98
- organization: DyNTS_global_settings.env_settings.openAi.organization,
99
- apiKey: DyNTS_global_settings.env_settings.openAi.apiKey,
100
- project: DyNTS_global_settings.env_settings.openAi.project,
101
- }
102
- );
103
-
104
- this.defaultSettings = set?.defaultSettings ??
105
- DyNTS_OAI_global_settings.defaultSettings ??
106
- new DyFM_OAI_CallSettings();
107
-
108
- // FR-002 — opcionális per-call cost-event callback átadás a base-osztálynak
109
- if (set?.onCostEvent) {
110
- this.onCostEvent = set.onCostEvent;
111
- }
112
- }
113
-
114
- /**
115
- * Inicializálja az OpenAI client-et a megadott config-gal
116
- *
117
- * Reinitializes the OpenAI client with the provided config
118
- */
119
- setup(config: DyFM_AI_Config): void {
120
- this.openai = new OpenAI(config);
121
- }
122
-
123
- /* readonly defaultLogReplacer: string = '...long-context...'; */
124
-
125
- //#region Ask Questions
126
-
127
- /**
128
- * Asks the AI to answer a yes/no question
129
- *
130
- * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.yesNo.request} to ask the question,
131
- * and the {@link DyNTS_OAI_LLM_Predefined_Requests.yesNo.upperCaseYes} to check the answer
132
- */
133
- async requestYesNo(
134
- set: DyFM_AI_Message_Input
135
- ): Promise<boolean> {
136
- set.message += this.predefinedRequests.yesNo.request;
137
-
138
- const answer = await this.requestSimpleMessage(set);
139
-
140
- return this.convertAnswerToBoolean(answer);
141
- }
142
- /** the exact same as {@link requestYesNo} */
143
- //yesNoQuestion = this.askYesNoQuestion;
144
-
145
- /**
146
- * Asks the AI to answer a percentage question
147
- *
148
- * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.percentageOnly} to ask the question
149
- */
150
- async requestPercentage(
151
- set: DyFM_AI_Message_Input
152
- ): Promise<number> {
153
- set.message += this.predefinedRequests.percentageOnly;
154
-
155
- const answer = await this.requestSimpleMessage(set);
156
-
157
- return this.convertAnswerToNumber(answer, set.message);
158
- }
159
- /** the exact same as {@link requestPercentage} */
160
- //percentageQuestion = this.askPercentageQuestion;
161
-
162
- /**
163
- * Asks the AI to select one of the options from the list
164
- *
165
- * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.selectRequest} to ask the question
166
- */
167
- /* async askSelectQuestion(
168
- set: DyFM_AI_GenericSelect_Input<string>
169
- ): Promise<string> {
170
- // replace the {{DyNTS_selectOptions}} with the list of options
171
- const selectQuestionAddition = this.predefinedRequests.selectRequest.replace(
172
- '{{DyNTS_selectOptions}}',
173
- this.getTextListAsText(set.selectFrom)
174
- );
175
- set.message += selectQuestionAddition;
176
-
177
- let answer = await this.requestSimpleMessage(set);
178
-
179
- answer = answer.toLocaleUpperCase();
180
-
181
- for (const item of set.selectFrom) {
182
- if (answer.includes(item.toLocaleUpperCase())) return item;
183
- }
184
-
185
- return null;
186
- } */
187
- /** the exact same as {@link askSelectQuestion} */
188
- //selectQuestion = this.askSelectQuestion;
189
-
190
- /**
191
- * Asks the AI to select one of the options from the list
192
- * Similar to {@link askSelectQuestion},
193
- * but with a more generic approach that can be used for objects
194
- *
195
- * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.selectRequest} to ask the question
196
- */
197
- async requestSelect<T>(
198
- set: DyFM_AI_GenericSelect_Input<T>
199
- ): Promise<T | { unparsableResult: string }> {
200
- const options: string[] = this.stringifySelectOptions(set.selectFrom);
201
- const selectQuestionAddition = this.predefinedRequests.selectRequest.replace(
202
- '{{DyNTS_selectOptions}}',
203
- this.getTextListAsText(options)
204
- );
205
- set.message += selectQuestionAddition;
206
-
207
- const answer = await this.requestSimpleMessage(set);
208
-
209
- return this.convertAnswerToSelectOption<T>(answer, set.message, set.selectFrom);
210
- }
211
- /** the exact same as {@link requestSelect} */
212
- //selectRequest = this.requestSelect;
213
-
214
- /**
215
- * Asks the AI to select one or more of the options from the list
216
- *
217
- * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.multiselect} to ask the question
218
- */
219
- async requestMultiselect<T>(
220
- set: DyFM_AI_GenericMultiSelect_Input<T, DyFM_OAI_CallSettings>
221
- ): Promise<T[] | { unparsableResult: string }> {
222
- const options: string[] = this.stringifySelectOptions(set.options);
223
- const selectQuestionAddition = this.predefinedRequests.multiselect.replace(
224
- '{{DyNTS_selectOptions}}',
225
- this.getTextListAsText(options)
226
- );
227
- set.message += selectQuestionAddition;
228
-
229
- let answer = await this.requestSimpleMessage(set);
230
-
231
- answer = answer.toLocaleUpperCase();
232
-
233
- return this.convertAnswerToSelectOptions<T>(answer, set.message, set.options);
234
- /* const result: T[] = [];
235
-
236
- for (const item of options) {
237
- if (answer.includes(item.toLocaleUpperCase())) {
238
- const parsedItem: T | { unparsableResult: string } = DyFM_Object.safeParseJSON<T>(item);
239
- if ((parsedItem as { unparsableResult: string }).unparsableResult) {
240
- result.push(item as any);
241
- } else {
242
- result.push(parsedItem as T);
243
- }
244
-
245
- }
246
- }
247
-
248
- if (!result.length && !answer.trim().length) {
249
- return { unparsableResult: answer };
250
- }
251
-
252
- return result; */
253
- }
254
- /** the exact same as {@link askMultipleSelectQuestionWithOptions} */
255
- //multipleSelectQuestion = this.askMultipleSelectQuestionWithOptions;
256
-
257
- /**
258
- * Asks the AI to select one or more of the options from the list
259
- * Similar to {@link askMultipleSelectQuestionWithOptions},
260
- * but with a more generic approach that can be used for objects
261
- *
262
- * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.multiselect} to ask the question
263
- */
264
- /* async requestMultiselect<T>(
265
- set: DyFM_AI_GenericMultiSelect_Input<T, DyFM_OAI_CallSettings>
266
- ): Promise<T[]> {
267
- const selectQuestionAddition = this.predefinedRequests.multiselect.replace(
268
- '{{DyNTS_selectOptions}}',
269
- this.getTextListAsText(set.options.map(item => `"${JSON.stringify(item)}"`))
270
- );
271
- set.message += selectQuestionAddition;
272
-
273
- const answer = await this.requestSimpleMessage(set);
274
-
275
- return DyFM_Object.safeParseList<T[]>(answer);
276
- } */
277
- /** the exact same as {@link requestMultipleSelect} */
278
- //multipleSelectRequest = this.requestMultipleSelect;
279
-
280
- /**
281
- * Asks the AI to answer a question that must result a JSON object
282
- *
283
- * If the answer is not a valid JSON object, it will return { unparsableResult: answer }
284
- *
285
- * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.jsonRequest} to ask the question
286
- *
287
- * (uses {@link DyFM_safeParseJSON})
288
- */
289
- async requestJSON<T = any>(
290
- set: DyFM_AI_Message_Input<DyFM_OAI_CallSettings>
291
- ): Promise<T | { unparsableResult: string }> {
292
- set.message += this.predefinedRequests.jsonRequest;
293
-
294
- const answer = await this.requestSimpleMessage(set);
295
-
296
- return this.convertAnswerToJSON<T>(answer, set.message);
297
- }
298
- /** the exact same as {@link askJSONQuestion} */
299
- //jsonQuestion = this.askJSONQuestion;
300
-
301
- /**
302
- * Asks the AI to answer a question that must result a JSON object with specific key descriptions
303
- *
304
- * If the answer is not a valid JSON object, it will return { unparsableResult: answer }
305
- *
306
- * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.jsonRequestWithKeysDescription} to ask the question
307
- *
308
- * (uses {@link DyFM_safeParseJSON})
309
- */
310
- async requestJSONQuestionWithKeysDescription<T = any>(
311
- set: DyFM_AI_JSONKeysDescription_Input<DyFM_OAI_CallSettings>
312
- ): Promise<T | { unparsableResult: string }> {
313
- const jsonRequestWithKeysDescription = this.predefinedRequests.jsonRequestWithKeysDescription.replace(
314
- '{{DyNTS_jsonKeysDescription}}',
315
- set.keysDescription
316
- );
317
- set.message += jsonRequestWithKeysDescription;
318
-
319
- const answer = await this.requestSimpleMessage(set);
320
-
321
- return this.convertAnswerToJSON<T>(answer, set.message);
322
- }
323
- /** the exact same as {@link requestJSONQuestionWithKeysDescription} */
324
- //jsonQuestionWithKeysDescription = this.askJSONQuestionWithKeysDescription;
325
-
326
- /**
327
- * Asks the AI to answer a question that must result a JSON object with specific keys
328
- *
329
- * If the answer is not a valid JSON object, it will return { unparsableResult: answer }
330
- *
331
- * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.jsonRequestWithExactKeys} to ask the question
332
- *
333
- * (uses {@link DyFM_safeParseJSON})
334
- */
335
- async requestJSONWithExactKeys<T = any>(
336
- set: DyFM_AI_JSONExactKeys_Input<DyFM_OAI_CallSettings>
337
- ): Promise<T | { unparsableResult: string }> {
338
- const jsonRequestWithExactKeys = this.predefinedRequests.jsonRequestWithExactKeys.replace(
339
- '{{DyNTS_jsonKeys}}',
340
- this.getTextListAsText(set.keys)
341
- );
342
- set.message += jsonRequestWithExactKeys;
343
-
344
- const answer = await this.requestSimpleMessage(set);
345
-
346
- return this.convertAnswerToJSON<T>(answer, set.message);
347
- }
348
- /** the exact same as {@link askJSONQuestionWithExactKeys} */
349
- //jsonQuestionWithExactKeys = this.askJSONQuestionWithExactKeys;
350
-
351
- /**
352
- * Asks the AI to answer a question that must result a list of strings
353
- *
354
- * If the answer is not a valid JSON list, it will return the answer as a single item in an array
355
- *
356
- * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.listRequest} to ask the question
357
- *
358
- * (uses {@link DyFM_Object.safeParseList})
359
- */
360
- /* async requestList(
361
- set: DyFM_AI_Base_Input<DyFM_OAI_CallSettings>
362
- ): Promise<string[]> {
363
- set.message += this.predefinedRequests.listRequest;
364
-
365
- const answer = await this.requestSimpleMessage(set);
366
-
367
- return this.convertAnswerToList<string>(answer, set.message);
368
- } */
369
- /** the exact same as {@link askListQuestion} */
370
- //listQuestion = this.askListQuestion;
371
- /** the exact same as {@link askListQuestion} */
372
- //listRequest = this.askListQuestion;
373
-
374
- async requestList<T = any>(
375
- set: DyFM_AI_Message_Input<DyFM_OAI_CallSettings>
376
- ): Promise<T[] | { unparsableResult: string }> {
377
- set.message += this.predefinedRequests.listRequest;
378
-
379
- const answer = await this.requestSimpleMessage(set);
380
-
381
- return this.convertAnswerToList<T>(answer, set.message);
382
- }
383
-
384
- /**
385
- * Asks the AI to answer a question
386
- *
387
- * Uses the {@link resolveSimpleUserMessage}
388
- */
389
- async requestSimpleMessage(
390
- set: DyFM_AI_Message_Input<DyFM_OAI_CallSettings>
391
- ): Promise<string> {
392
- this.logQuestion(set);
393
-
394
- set.conversation ??= [];
395
- set.conversation.push({ role: DyFM_AI_MessageRole.user, content: set.message });
396
-
397
- const answer: string = await this.resolveMessage({
398
- ...set,
399
- conversation: set.conversation,
400
- getFullResponse: false,
401
- }) as string;
402
-
403
- this.logAnswer(answer);
404
-
405
- return answer;
406
- }
407
- /** the exact same as {@link requestSimpleMessage} */
408
- //answer = this.askQuestion;
409
- /** the exact same as {@link requestSimpleMessage} */
410
- //getQuestionAnswer = this.askQuestion;
411
- /** the exact same as {@link requestSimpleMessage} */
412
- //simpleQuestion = this.askQuestion;
413
- /** the exact same as {@link requestSimpleMessage} */
414
- //askSimpleQuestion = this.askQuestion;
415
-
416
- /**
417
- * Resolves a simple user message and returns the answer as string
418
- *
419
- * Uses the {@link resolveMessage}
420
- */
421
- /* async resolveSimpleUserMessage(
422
- set: DyFM_AI_Base_Input<DyFM_OAI_CallSettings>
423
- ): Promise<string> {
424
- return this.resolveMessage({
425
- conversation: [{ role: DyFM_AI_MessageRole.user, content: set.message }],
426
- issuer: set.issuer ?? 'unknown',
427
- settings: set.settings ?? this.defaultSettings,
428
- getFullResponse: false,
429
- }) as Promise<string>;
430
- } */
431
- /** the exact same as {@link resolveSimpleUserMessage} */
432
- //simpleUserMessage = this.resolveSimpleUserMessage;
433
-
434
- //#endregion
435
-
436
- /**
437
- * Resolves a conversation
438
- *
439
- * Uses the {@link getMessageCreateInput}
440
- */
441
- async resolveMessage(
442
- set: DyFM_AI_ConversationBase_Input<DyFM_OAI_CallSettings>
443
- /* {
444
- conversation: [{ role: DyFM_AI_MessageRole.user, content: set.message }],
445
- issuer: set.issuer ?? 'unknown',
446
- settings: set.settings ?? this.defaultSettings,
447
- getFullResponse: false,
448
- } */
449
- /* {
450
- conversation: DyFM_AI_Message[],
451
- issuer: string,
452
- settings?: DyFM_OAI_CallSettings,
453
- getFullResponse?: boolean
454
- } */
455
- ): Promise<string | ChatCompletion> {
456
- try {
457
- // FR-002 — callType meghatározás a unshift ELŐTT (a system-message hozzáadás
458
- // után már 2+ message lenne, ami a "chat" kategóriát mindig kiváltja).
459
- // 1 user-message → 'llm-completion'; több → 'llm-chat'. A 'llm-tool-use'
460
- // categoria a tools paraméterek bevezetésekor kerül használatba (jövőbeli FR).
461
- const userMessageCount: number = set.conversation.length;
462
-
463
- set.conversation.unshift(this.getDefaultSystemMessage(set.settings));
464
-
465
- // FR-002 — per-call cost-event timing: a tényleges API-call latency mérve
466
- const input: ChatCompletionCreateParamsBase = this.getMessageCreateInput(set);
467
- const start: number = Date.now();
468
- const result: ChatCompletion = await this.openai.chat.completions.create(
469
- input,
470
- ) as ChatCompletion;
471
- const durationMs: number = Date.now() - start;
472
-
473
- // FR-002 — cost-event emit (safe, no-op ha nincs onCostEvent callback)
474
- this.emitCostEvent({
475
- callType: userMessageCount === 1 ? 'llm-completion' : 'llm-chat',
476
- provider: this.aiProvider,
477
- model: String(input.model ?? this.defaultModel),
478
- tokensUsed: {
479
- input: result.usage?.prompt_tokens ?? 0,
480
- output: result.usage?.completion_tokens ?? 0,
481
- total: result.usage?.total_tokens ?? 0,
482
- },
483
- durationMs: durationMs,
484
- issuer: set.issuer,
485
- timestamp: new Date(),
486
- });
487
-
488
- if (set.getFullResponse) {
489
- return result;
490
- }
491
-
492
- return result.choices[0].message.content;
493
- } catch (error) {
494
- throw new DyFM_Error({
495
- ...this.getDefaultErrorSettings('resolveConversation', error, set.issuer),
496
-
497
- errorCode: `${DyNTS_global_settings.systemShortCodeName}|DyNTS-OLSB-RM0`,
498
- });
499
- }
500
- }
501
-
502
- /**
503
- * Creates a message create input
504
- *
505
- * Uses the {@link getDefaultSystemMessage}
506
- */
507
- protected getMessageCreateInput(
508
- set: {
509
- conversation: DyFM_AI_Message[],
510
- issuer: string,
511
- settings?: DyFM_OAI_CallSettings
512
- }
513
- ): ChatCompletionCreateParamsBase {
514
- try {
515
- const settings = set.settings ?? this.defaultSettings;
516
-
517
- const result: ChatCompletionCreateParamsBase = {
518
- model: settings?.useModel ?? this.defaultModel,
519
- messages: set.conversation as ChatCompletionMessageParam[],
520
- };
521
-
522
- result.temperature = DyFM_notNull(settings?.temperature) ?
523
- settings.temperature :
524
- this.defaultSettings.temperature;
525
-
526
- /* result.max_completion_tokens = DyFM_notNull(settings?.maxTokens) ?
527
- settings.maxTokens :
528
- this.defaultSettings.maxTokens; */
529
- result.max_completion_tokens = DyFM_notNull(settings?.maxTokens) ?
530
- settings.maxTokens :
531
- this.defaultSettings.maxTokens;
532
-
533
- result.top_p = DyFM_notNull(settings?.topP) ?
534
- settings.topP :
535
- this.defaultSettings.topP;
536
-
537
- result.frequency_penalty = DyFM_notNull(settings?.frequencyPenalty) ?
538
- settings.frequencyPenalty :
539
- this.defaultSettings.frequencyPenalty;
540
-
541
- result.presence_penalty = DyFM_notNull(settings?.presencePenalty) ?
542
- settings.presencePenalty :
543
- this.defaultSettings.presencePenalty;
544
-
545
- return result;
546
- } catch (error) {
547
- throw new DyFM_Error({
548
- ...this.getDefaultErrorSettings('getMessageCreateSettings', error, set.issuer),
549
-
550
- errorCode: `${DyNTS_global_settings.systemShortCodeName}|DyNTS-OLSB-RC0`,
551
- });
552
- }
553
- }
554
-
555
- /**
556
- * Teszteli a kapcsolatot az OpenAI API-val
557
- *
558
- * Tests the connection to OpenAI API
559
- */
560
- async testConnection(issuer: string): Promise<boolean> {
561
- try {
562
- await this.openai.models.list();
563
- return true;
564
- } catch (error) {
565
- DyFM_Log.error('DyNTS_OAI_LLM_ServiceBase', 'testConnection', 'Connection test failed', {
566
- error,
567
- issuer
568
- });
569
- return false;
570
- }
571
- }
572
-
573
- //#region Function calling (tool use) — FR-047
574
-
575
- /** OpenAI tool-kepes modell-registry (a base capability-precheck-jehez). */
576
- protected override getModelRegistry(): DyFM_AI_ModelInfo[] {
577
- return DyFM_OAI_Models;
578
- }
579
-
580
- /**
581
- * Egy OpenAI-kor a tool-loop-ban: felepiti a natív request-et (system + a teljes beszelgetes
582
- * tool-uzenetekkel + tools), meghivja az API-t, es normalizalt DyFM_AI_LLM_Response-t ad vissza
583
- * (toolCalls feltoltve, ha a model tool-t akar hivni). FR-002 cost-event: 'llm-tool-use'.
584
- */
585
- protected override async callModelWithTools(
586
- set: {
587
- conversation: DyFM_AI_Message[];
588
- tools: DyFM_AI_Tool[];
589
- settings?: DyFM_OAI_CallSettings;
590
- issuer: string;
591
- }
592
- ): Promise<DyFM_AI_LLM_Response> {
593
- try {
594
- const settings: DyFM_OAI_CallSettings = set.settings ?? this.defaultSettings;
595
-
596
- const messages: ChatCompletionMessageParam[] = [
597
- this.toOpenAIMessage(this.getDefaultSystemMessage(settings)),
598
- ...set.conversation.map((message: DyFM_AI_Message) => this.toOpenAIMessage(message)),
599
- ];
600
-
601
- const useModel: DyFM_OAI_Model | string = settings?.useModel ?? this.defaultModel;
602
-
603
- const input: ChatCompletionCreateParamsBase = {
604
- model: useModel,
605
- messages: messages,
606
- max_completion_tokens: DyFM_notNull(settings?.maxTokens) ? settings.maxTokens : this.defaultSettings.maxTokens,
607
- };
608
-
609
- // Reasoning modellek (GPT-5.x / o-series) elutasitjak a sampling-parametereket
610
- // (temperature != 1, top_p, frequency/presence_penalty) -> 400. Ezeket NEM kuldjuk
611
- // reasoning modellnel. (Ez a FR-048 per-model settings elv lokalis, OpenAI-specifikus
612
- // alkalmazasa — amig a general reconciler el nem keszul.)
613
- if (!this.isReasoningModel(String(useModel))) {
614
- input.temperature = DyFM_notNull(settings?.temperature) ? settings.temperature : this.defaultSettings.temperature;
615
- input.top_p = DyFM_notNull(settings?.topP) ? settings.topP : this.defaultSettings.topP;
616
- input.frequency_penalty = DyFM_notNull(settings?.frequencyPenalty) ? settings.frequencyPenalty : this.defaultSettings.frequencyPenalty;
617
- input.presence_penalty = DyFM_notNull(settings?.presencePenalty) ? settings.presencePenalty : this.defaultSettings.presencePenalty;
618
- }
619
-
620
- // Ures tools-tomb -> 400 ("tools must have minimum length 1"). Csak akkor kuldjuk, ha van tool.
621
- if (set.tools.length) {
622
- input.tools = set.tools.map((tool: DyFM_AI_Tool) => this.toOpenAITool(tool));
623
- input.tool_choice = 'auto';
624
- }
625
-
626
- const start: number = Date.now();
627
- const result: ChatCompletion = await this.openai.chat.completions.create(input) as ChatCompletion;
628
- const durationMs: number = Date.now() - start;
629
-
630
- const message = result.choices[0].message;
631
-
632
- const toolCalls: DyFM_AI_ToolCall[] = (message.tool_calls ?? [])
633
- .filter((toolCall) => toolCall.type === 'function')
634
- .map((toolCall) => ({
635
- id: toolCall.id,
636
- name: toolCall.function.name,
637
- arguments: this.parseToolArguments(toolCall.function.arguments),
638
- }));
639
-
640
- this.emitCostEvent({
641
- callType: 'llm-tool-use',
642
- provider: this.aiProvider,
643
- model: String(input.model ?? this.defaultModel),
644
- tokensUsed: {
645
- input: result.usage?.prompt_tokens ?? 0,
646
- output: result.usage?.completion_tokens ?? 0,
647
- total: result.usage?.total_tokens ?? 0,
648
- },
649
- durationMs: durationMs,
650
- issuer: set.issuer,
651
- timestamp: new Date(),
652
- });
653
-
654
- return {
655
- content: message.content ?? '',
656
- toolCalls: toolCalls.length ? toolCalls : undefined,
657
- usage: result.usage ? {
658
- promptTokens: result.usage.prompt_tokens,
659
- completionTokens: result.usage.completion_tokens,
660
- totalTokens: result.usage.total_tokens,
661
- } : undefined,
662
- model: result.model,
663
- finishReason: result.choices[0].finish_reason,
664
- stopReason: result.choices[0].finish_reason,
665
- rawResponse: result,
666
- };
667
- } catch (error) {
668
- throw new DyFM_Error({
669
- ...this.getDefaultErrorSettings('callModelWithTools', error, set.issuer),
670
- errorCode: `${DyNTS_global_settings.systemShortCodeName}|DyNTS-OLSB-TC0`,
671
- });
672
- }
673
- }
674
-
675
- /** Agnosztikus tool-definicio → OpenAI ChatCompletionTool. */
676
- protected toOpenAITool(tool: DyFM_AI_Tool): ChatCompletionTool {
677
- return {
678
- type: 'function',
679
- function: {
680
- name: tool.name,
681
- description: tool.description,
682
- parameters: tool.parameters as Record<string, unknown>,
683
- },
684
- };
685
- }
686
-
687
- /**
688
- * Agnosztikus DyFM_AI_Message → OpenAI ChatCompletionMessageParam.
689
- * Kezeli a tool-result (role:'tool') es a tool-hivo assistant (tool_calls) uzeneteket is.
690
- */
691
- protected toOpenAIMessage(message: DyFM_AI_Message): ChatCompletionMessageParam {
692
- if (message.role === DyFM_AI_MessageRole.tool) {
693
- return {
694
- role: 'tool',
695
- tool_call_id: message.toolCallId ?? '',
696
- content: message.content,
697
- };
698
- }
699
-
700
- if (message.role === DyFM_AI_MessageRole.assistant && message.toolCalls?.length) {
701
- return {
702
- role: 'assistant',
703
- content: message.content || null,
704
- tool_calls: message.toolCalls.map((toolCall: DyFM_AI_ToolCall) => ({
705
- id: toolCall.id,
706
- type: 'function' as const,
707
- function: {
708
- name: toolCall.name,
709
- arguments: JSON.stringify(toolCall.arguments),
710
- },
711
- })),
712
- };
713
- }
714
-
715
- return {
716
- role: message.role as 'system' | 'user' | 'assistant',
717
- content: message.content,
718
- };
719
- }
720
-
721
- /** Tool-argumentumok parse-olasa (OpenAI JSON-string → object); hibas JSON eseten ures object. */
722
- protected parseToolArguments(raw: string): Record<string, unknown> {
723
- try {
724
- return JSON.parse(raw || '{}');
725
- } catch {
726
- return {};
727
- }
728
- }
729
-
730
- /**
731
- * Reasoning modell-e (GPT-5.x sor vagy o-series, pl. o1/o3/o4)? Ezek elutasitjak a
732
- * sampling-parametereket (temperature != 1, top_p, frequency/presence_penalty). A `gpt-4o`
733
- * NEM reasoning (a 'gpt-5' prefix nem illeszti, 'o'-prefix sem).
734
- */
735
- protected isReasoningModel(modelId: string): boolean {
736
- return /^gpt-5/.test(modelId) || /^o[0-9]/.test(modelId);
737
- }
738
-
739
- //#endregion
740
-
741
- // async askJSONListQuestion(
742
- // question: string,
743
- // issuer: string,
744
- // settings?: DyFM_OAI_CallSettings,
745
- // debugLog?: boolean,
746
- // /** this is used to readably replace too long contents to eg '...' in logs */
747
- // replaceThisInLog?: string,
748
- // ): Promise<{ key: string, input: string }[]> {
749
- // question += this.predefinedRequests.jsonListRequest;
750
- //
751
- // const answer = await this.askQuestion(question, issuer, settings, debugLog, replaceThisInLog);
752
- //
753
- // return DyFM_Object.safeParseList<{ key: string, input: string }[]>(answer);
754
- // }
755
-
756
- //#region Tools
757
-
758
- /**
759
- * Logs the question if the debugLog is true
760
- */
761
- /* protected logQuestion(
762
- set: DyFM_AI_Base_Input<DyFM_OAI_CallSettings>
763
- ): void {
764
- if (set.settings?.debugLog ?? this._debugLog) {
765
- console.log('\n - ', set.message);
766
- }
767
- } */
768
-
769
- /**
770
- * olvasható mondatszerű-listaszerű formába teszi a listaelemeket
771
- * pl.: ['a', 'b', 'c'] -> '"a", "b" or "c"'
772
- */
773
- /* getTextListAsText(list: string[]): string {
774
- list = list.filter(item => item?.trim()).map(item => `"${item}"`);
775
-
776
- //list = list.map(item => item.toLocaleLowerCase());
777
-
778
- list.push(list.pop() + ' or ' + list.pop());
779
-
780
- return list.join(', ');
781
- } */
782
-
783
- /* protected getDefaultSystemMessage(settings: DyFM_OAI_CallSettings): DyFM_AI_Message {
784
- return {
785
- role: DyFM_AI_MessageRole.system,
786
- content: settings?.systemPrompt || this.defaultSystemPrompt,
787
- };
788
- } */
789
-
790
- /* protected validateConversation(conversation: DyFM_AI_Message[]): void {
791
- conversation.forEach((message: DyFM_AI_Message, index: number) => {
792
- if (!message.role) {
793
- throw new DyFM_Error({
794
- message: `Message has no role at index ${index}`,
795
- additionalContent: {
796
- invalidMessage: message,
797
- conversation: conversation,
798
- }
799
- });
800
- }
801
-
802
- if (!message.content) {
803
- throw new DyFM_Error({
804
- message: `Message has no content at index ${index}`,
805
- additionalContent: {
806
- invalidMessage: message,
807
- conversation: conversation,
808
- }
809
- });
810
- }
811
- });
812
- } */
813
-
814
- /* protected getDefaultErrorSettings(
815
- fnName: string,
816
- error: DyFM_AnyError,
817
- issuer: string
818
- ): DyFM_Error_Settings {
819
- return {
820
- status: (error as DyFM_Error)?.___status ?? 500,
821
- message: (error as Error)?.message ??
822
- (error as DyFM_Error)?._message ??
823
- `${fnName} was UNSUCCESSFUL (${DyNTS_global_settings.systemShortCodeName})`,
824
- addECToUserMsg: !(error as DyFM_Error)?.__userMessage,
825
- userMessage: (error as DyFM_Error)?.__userMessage ?? this.defaultErrorUserMsg,
826
- issuer: issuer,
827
- issuerService: this.constructor?.name,
828
- error: error,
829
- };
830
- } */
831
-
832
- //#endregion
833
-
834
- //#region Provider Methods (from DyNTS_AI_Provider_ServiceBase)
835
-
836
- //#endregion
837
-
838
- //#region Abstract LLM Methods (from DyNTS_AI_LLM_ServiceBase)
839
-
840
- /**
841
- * Meghívja az LLM-et system és user message-szel
842
- *
843
- * Calls the LLM with system and user messages
844
- */
845
- /* async callLLM(
846
- systemMessage: string,
847
- userMessage: string,
848
- settings?: DyFM_AI_CallSettings,
849
- issuer?: string
850
- ): Promise<string> {
851
- const result = await this.resolveMessage({
852
- conversation: [
853
- { role: DyFM_AI_MessageRole.system, content: systemMessage },
854
- { role: DyFM_AI_MessageRole.user, content: userMessage }
855
- ],
856
- issuer: issuer ?? 'system',
857
- settings: this.convertToOAISettings(settings),
858
- getFullResponse: false
859
- });
860
-
861
- if (typeof result === 'string') {
862
- return result;
863
- }
864
- return '';
865
- } */
866
-
867
- /**
868
- * Meghívja az LLM-et üzenet history-val
869
- *
870
- * Calls the LLM with message history
871
- */
872
- /* async callLLMWithHistory(
873
- messages: DyFM_AI_Message[],
874
- settings?: DyFM_AI_CallSettings,
875
- issuer?: string
876
- ): Promise<string> {
877
- const result = await this.resolveMessage({
878
- conversation: messages,
879
- issuer: issuer ?? 'system',
880
- settings: this.convertToOAISettings(settings),
881
- getFullResponse: false
882
- });
883
-
884
- if (typeof result === 'string') {
885
- return result;
886
- }
887
- return '';
888
- } */
889
-
890
- /**
891
- * Meghívja az LLM-et és visszaadja a raw response-t
892
- *
893
- * Calls the LLM and returns raw response
894
- */
895
- /* async callLLMRaw(
896
- messages: DyFM_AI_Message[],
897
- settings?: DyFM_AI_CallSettings,
898
- issuer?: string
899
- ): Promise<DyFM_AI_LLM_Response> {
900
- const result = await this.resolveMessage({
901
- conversation: messages,
902
- issuer: issuer ?? 'system',
903
- settings: this.convertToOAISettings(settings),
904
- getFullResponse: true
905
- }) as ChatCompletion;
906
-
907
- return {
908
- content: result.choices[0].message.content,
909
- usage: result.usage ? {
910
- promptTokens: result.usage.prompt_tokens,
911
- completionTokens: result.usage.completion_tokens,
912
- totalTokens: result.usage.total_tokens
913
- } : undefined,
914
- model: result.model,
915
- finishReason: result.choices[0].finish_reason,
916
- rawResponse: result
917
- };
918
- } */
919
-
920
- /**
921
- * Küld egy egyszerű üzenetet az LLM-nek
922
- *
923
- * Sends a simple message to the LLM
924
- */
925
- /* async sendMessage(set: DyFM_AI_Base_Input<DyFM_OAI_CallSettings>): Promise<string> {
926
- return await this.resolveSimpleUserMessage({
927
- message: set.message,
928
- issuer: set.issuer,
929
- settings: set.settings ? this.convertToOAISettings(set.settings) : undefined
930
- });
931
- } */
932
-
933
- /**
934
- * Kérdez egy kérdést és JSON választ vár exact keys-szel
935
- *
936
- * Asks a question and expects JSON response with exact keys
937
- */
938
- /* async requestJSONWithExactKeys<T = any>(
939
- set: DyFM_AI_JSONExactKeys_Input
940
- ): Promise<T> {
941
- const answer = await this.askJSONQuestionWithExactKeys({
942
- question: set.question,
943
- keys: set.keys,
944
- issuer: set.issuer,
945
- settings: set.settings ? this.convertToOAISettings(set.settings) : undefined,
946
- debugLog: set.debugLog,
947
- replaceThisInLog: set.replaceThisInLog
948
- });
949
-
950
- if ('unparsableResult' in answer) {
951
- throw new DyFM_Error({
952
- message: 'Invalid JSON response from AI',
953
- additionalContent: { answer },
954
- userMessage: 'Failed to parse AI response'
955
- });
956
- }
957
-
958
- return answer as T;
959
- } */
960
-
961
- /**
962
- * Kérdez egy kérdést és több választ vár
963
- *
964
- * Asks a question and expects multiple responses
965
- */
966
- /* async requestMultiselect<T = any>(
967
- set: DyFM_AI_GenericMultiSelect_Input<T>
968
- ): Promise<T[]> {
969
- const answer = await this.requestMultipleSelect({
970
- question: set.message,
971
- options: set.options,
972
- issuer: set.issuer,
973
- settings: set.settings ? this.convertToOAISettings(set.settings) : undefined,
974
- debugLog: set.debugLog,
975
- replaceThisInLog: set.replaceThisInLog
976
- });
977
-
978
- return answer as T[];
979
- } */
980
-
981
- /**
982
- * Konvertálja a generic AI settings-et OpenAI specifikus settings-re
983
- *
984
- * Converts generic AI settings to OpenAI-specific settings
985
- */
986
- /* private convertToOAISettings(settings?: DyFM_AI_CallSettings): DyFM_OAI_CallSettings {
987
- if (!settings) {
988
- return undefined;
989
- }
990
-
991
- const oaiSettings = new DyFM_OAI_CallSettings();
992
-
993
- if (settings.systemPrompt !== undefined) {
994
- oaiSettings.systemPrompt = settings.systemPrompt;
995
- }
996
-
997
- if (settings.useModel !== undefined) {
998
- oaiSettings.useModel = settings.useModel as DyFM_OAI_Model;
999
- }
1000
-
1001
- if (settings.project !== undefined) {
1002
- oaiSettings.project = settings.project;
1003
- }
1004
-
1005
- if (settings.debugLog !== undefined) {
1006
- oaiSettings.debugLog = settings.debugLog;
1007
- }
1008
-
1009
- if (settings.temperature !== undefined) {
1010
- oaiSettings.temperature = settings.temperature;
1011
- }
1012
-
1013
- if (settings.maxTokens !== undefined) {
1014
- oaiSettings.maxTokens = settings.maxTokens;
1015
- }
1016
-
1017
- if (settings.topP !== undefined) {
1018
- oaiSettings.topP = settings.topP;
1019
- }
1020
-
1021
- if (settings.frequencyPenalty !== undefined) {
1022
- oaiSettings.frequencyPenalty = settings.frequencyPenalty;
1023
- }
1024
-
1025
- if (settings.presencePenalty !== undefined) {
1026
- oaiSettings.presencePenalty = settings.presencePenalty;
1027
- }
1028
-
1029
- return oaiSettings;
1030
- } */
1031
-
1032
- //#endregion
1033
- }
1
+ import { OpenAI } from 'openai';
2
+
3
+ import { DyFM_OAI_Settings, DyFM_OAI_Model, DyFM_OAI_CallSettings, DyFM_OAI_Models } from '@futdevpro/fsm-dynamo/ai/open-ai';
4
+ import { DyFM_AnyError, DyFM_Error, DyFM_Error_Settings, DyFM_Log, DyFM_notNull, DyFM_Object } from '@futdevpro/fsm-dynamo';
5
+ import {
6
+ DyFM_AI_Message,
7
+ DyFM_AI_MessageRole,
8
+ DyFM_AI_Provider,
9
+ DyFM_AI_ProviderCapabilities,
10
+ DyFM_AI_CallSettings,
11
+ DyFM_AI_Config,
12
+ DyFM_AI_LLM_Response,
13
+ DyFM_AI_Tool,
14
+ DyFM_AI_ToolCall,
15
+ DyFM_AI_ModelInfo
16
+ } from '@futdevpro/fsm-dynamo/ai';
17
+
18
+ import { DyNTS_global_settings } from '../../../../../_collections/global-settings.const';
19
+ import { ChatCompletion } from 'openai/resources';
20
+ import { ChatCompletionCreateParamsBase, ChatCompletionMessageParam, ChatCompletionTool } from 'openai/resources/chat/completions';
21
+ import { DyNTS_AI_CostEventCallback } from '../../../_models/interfaces/dynts-ai-cost-event-callback.interface';
22
+ import { DyNTS_OAI_LLM_Predefined_Requests } from '../_models/interfaces/oai-llm-predefined-requests.interface';
23
+ import { DyNTS_OAI_global_settings } from '../_collections/oai-global-settings.const';
24
+ import { DyNTS_OAI_LLMDefaultPredefined_Requests } from '../_collections/oai-llm-predefined-requests.conts';
25
+ import { DyNTS_AI_LLM_ServiceBase } from '../../../_services/ai-llm.service-base';
26
+ import {
27
+ DyFM_AI_Message_Input,
28
+ DyFM_AI_GenericSelect_Input,
29
+ DyFM_AI_GenericMultiSelect_Input,
30
+ DyFM_AI_JSONKeysDescription_Input,
31
+ DyFM_AI_JSONExactKeys_Input,
32
+ DyFM_AI_MultiSelect_Input,
33
+ DyFM_AI_ConversationBase_Input
34
+ } from '../../../_models/ai-input-interfaces';
35
+
36
+
37
+ /**
38
+ * {
39
+ * organization: 'org-XY',
40
+ * apiKey: 'sk-XY',
41
+ * project: 'proj_XY',
42
+ * }
43
+ */
44
+
45
+ // Di Dictionary DiModel DiEz DiAz
46
+
47
+ export class DyNTS_OAI_LLM_ServiceBase extends DyNTS_AI_LLM_ServiceBase {
48
+
49
+ /* readonly defaultErrorUserMsg: string =
50
+ `We encountered an unhandled Control Service Error, ` +
51
+ `\nplease contact the responsible development team.`; */
52
+
53
+ openai: OpenAI;
54
+
55
+ readonly defaultSettings: DyFM_OAI_CallSettings;
56
+
57
+ // Provider properties from DyNTS_AI_Provider_ServiceBase
58
+ readonly aiProvider: DyFM_AI_Provider = DyFM_AI_Provider.OpenAI;
59
+
60
+ readonly capabilities: DyFM_AI_ProviderCapabilities = {
61
+ chat: true,
62
+ embeddings: true,
63
+ imageGeneration: true,
64
+ vision: true,
65
+ audioGeneration: true,
66
+ audioAnalysis: true,
67
+ functionCalling: true,
68
+ streaming: true,
69
+ batchOperations: true,
70
+ maxContextLength: 128000,
71
+ supportedModelTypes: ['gpt-4o', 'gpt-4o-mini', 'text-embedding-3-large', 'text-embedding-3-small']
72
+ };
73
+
74
+ /* get _debugLog(): boolean {
75
+ return this.defaultSettings.debugLog;
76
+ } */
77
+
78
+ /* get defaultSystemPrompt(): string {
79
+ return this.defaultSettings.systemPrompt;
80
+ } */
81
+
82
+ override get defaultModel(): DyFM_OAI_Model | string {
83
+ return this.defaultSettings.useModel;
84
+ }
85
+
86
+ predefinedRequests: DyNTS_OAI_LLM_Predefined_Requests = DyNTS_OAI_LLMDefaultPredefined_Requests;
87
+
88
+ constructor(
89
+ set?: DyFM_OAI_Settings & { onCostEvent?: DyNTS_AI_CostEventCallback }
90
+ ) {
91
+ super();
92
+
93
+ DyNTS_global_settings.openAi_settings ??= DyNTS_OAI_global_settings;
94
+
95
+ this.openai = new OpenAI(
96
+ set?.config ??
97
+ {
98
+ organization: DyNTS_global_settings.env_settings.openAi.organization,
99
+ apiKey: DyNTS_global_settings.env_settings.openAi.apiKey,
100
+ project: DyNTS_global_settings.env_settings.openAi.project,
101
+ }
102
+ );
103
+
104
+ this.defaultSettings = set?.defaultSettings ??
105
+ DyNTS_OAI_global_settings.defaultSettings ??
106
+ new DyFM_OAI_CallSettings();
107
+
108
+ // FR-002 — opcionális per-call cost-event callback átadás a base-osztálynak
109
+ if (set?.onCostEvent) {
110
+ this.onCostEvent = set.onCostEvent;
111
+ }
112
+ }
113
+
114
+ /**
115
+ * Inicializálja az OpenAI client-et a megadott config-gal
116
+ *
117
+ * Reinitializes the OpenAI client with the provided config
118
+ */
119
+ setup(config: DyFM_AI_Config): void {
120
+ this.openai = new OpenAI(config);
121
+ }
122
+
123
+ /* readonly defaultLogReplacer: string = '...long-context...'; */
124
+
125
+ //#region Ask Questions
126
+
127
+ /**
128
+ * Asks the AI to answer a yes/no question
129
+ *
130
+ * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.yesNo.request} to ask the question,
131
+ * and the {@link DyNTS_OAI_LLM_Predefined_Requests.yesNo.upperCaseYes} to check the answer
132
+ */
133
+ async requestYesNo(
134
+ set: DyFM_AI_Message_Input
135
+ ): Promise<boolean> {
136
+ set.message += this.predefinedRequests.yesNo.request;
137
+
138
+ const answer = await this.requestSimpleMessage(set);
139
+
140
+ return this.convertAnswerToBoolean(answer);
141
+ }
142
+ /** the exact same as {@link requestYesNo} */
143
+ //yesNoQuestion = this.askYesNoQuestion;
144
+
145
+ /**
146
+ * Asks the AI to answer a percentage question
147
+ *
148
+ * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.percentageOnly} to ask the question
149
+ */
150
+ async requestPercentage(
151
+ set: DyFM_AI_Message_Input
152
+ ): Promise<number> {
153
+ set.message += this.predefinedRequests.percentageOnly;
154
+
155
+ const answer = await this.requestSimpleMessage(set);
156
+
157
+ return this.convertAnswerToNumber(answer, set.message);
158
+ }
159
+ /** the exact same as {@link requestPercentage} */
160
+ //percentageQuestion = this.askPercentageQuestion;
161
+
162
+ /**
163
+ * Asks the AI to select one of the options from the list
164
+ *
165
+ * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.selectRequest} to ask the question
166
+ */
167
+ /* async askSelectQuestion(
168
+ set: DyFM_AI_GenericSelect_Input<string>
169
+ ): Promise<string> {
170
+ // replace the {{DyNTS_selectOptions}} with the list of options
171
+ const selectQuestionAddition = this.predefinedRequests.selectRequest.replace(
172
+ '{{DyNTS_selectOptions}}',
173
+ this.getTextListAsText(set.selectFrom)
174
+ );
175
+ set.message += selectQuestionAddition;
176
+
177
+ let answer = await this.requestSimpleMessage(set);
178
+
179
+ answer = answer.toLocaleUpperCase();
180
+
181
+ for (const item of set.selectFrom) {
182
+ if (answer.includes(item.toLocaleUpperCase())) return item;
183
+ }
184
+
185
+ return null;
186
+ } */
187
+ /** the exact same as {@link askSelectQuestion} */
188
+ //selectQuestion = this.askSelectQuestion;
189
+
190
+ /**
191
+ * Asks the AI to select one of the options from the list
192
+ * Similar to {@link askSelectQuestion},
193
+ * but with a more generic approach that can be used for objects
194
+ *
195
+ * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.selectRequest} to ask the question
196
+ */
197
+ async requestSelect<T>(
198
+ set: DyFM_AI_GenericSelect_Input<T>
199
+ ): Promise<T | { unparsableResult: string }> {
200
+ const options: string[] = this.stringifySelectOptions(set.selectFrom);
201
+ const selectQuestionAddition = this.predefinedRequests.selectRequest.replace(
202
+ '{{DyNTS_selectOptions}}',
203
+ this.getTextListAsText(options)
204
+ );
205
+ set.message += selectQuestionAddition;
206
+
207
+ const answer = await this.requestSimpleMessage(set);
208
+
209
+ return this.convertAnswerToSelectOption<T>(answer, set.message, set.selectFrom);
210
+ }
211
+ /** the exact same as {@link requestSelect} */
212
+ //selectRequest = this.requestSelect;
213
+
214
+ /**
215
+ * Asks the AI to select one or more of the options from the list
216
+ *
217
+ * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.multiselect} to ask the question
218
+ */
219
+ async requestMultiselect<T>(
220
+ set: DyFM_AI_GenericMultiSelect_Input<T, DyFM_OAI_CallSettings>
221
+ ): Promise<T[] | { unparsableResult: string }> {
222
+ const options: string[] = this.stringifySelectOptions(set.options);
223
+ const selectQuestionAddition = this.predefinedRequests.multiselect.replace(
224
+ '{{DyNTS_selectOptions}}',
225
+ this.getTextListAsText(options)
226
+ );
227
+ set.message += selectQuestionAddition;
228
+
229
+ let answer = await this.requestSimpleMessage(set);
230
+
231
+ answer = answer.toLocaleUpperCase();
232
+
233
+ return this.convertAnswerToSelectOptions<T>(answer, set.message, set.options);
234
+ /* const result: T[] = [];
235
+
236
+ for (const item of options) {
237
+ if (answer.includes(item.toLocaleUpperCase())) {
238
+ const parsedItem: T | { unparsableResult: string } = DyFM_Object.safeParseJSON<T>(item);
239
+ if ((parsedItem as { unparsableResult: string }).unparsableResult) {
240
+ result.push(item as any);
241
+ } else {
242
+ result.push(parsedItem as T);
243
+ }
244
+
245
+ }
246
+ }
247
+
248
+ if (!result.length && !answer.trim().length) {
249
+ return { unparsableResult: answer };
250
+ }
251
+
252
+ return result; */
253
+ }
254
+ /** the exact same as {@link askMultipleSelectQuestionWithOptions} */
255
+ //multipleSelectQuestion = this.askMultipleSelectQuestionWithOptions;
256
+
257
+ /**
258
+ * Asks the AI to select one or more of the options from the list
259
+ * Similar to {@link askMultipleSelectQuestionWithOptions},
260
+ * but with a more generic approach that can be used for objects
261
+ *
262
+ * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.multiselect} to ask the question
263
+ */
264
+ /* async requestMultiselect<T>(
265
+ set: DyFM_AI_GenericMultiSelect_Input<T, DyFM_OAI_CallSettings>
266
+ ): Promise<T[]> {
267
+ const selectQuestionAddition = this.predefinedRequests.multiselect.replace(
268
+ '{{DyNTS_selectOptions}}',
269
+ this.getTextListAsText(set.options.map(item => `"${JSON.stringify(item)}"`))
270
+ );
271
+ set.message += selectQuestionAddition;
272
+
273
+ const answer = await this.requestSimpleMessage(set);
274
+
275
+ return DyFM_Object.safeParseList<T[]>(answer);
276
+ } */
277
+ /** the exact same as {@link requestMultipleSelect} */
278
+ //multipleSelectRequest = this.requestMultipleSelect;
279
+
280
+ /**
281
+ * Asks the AI to answer a question that must result a JSON object
282
+ *
283
+ * If the answer is not a valid JSON object, it will return { unparsableResult: answer }
284
+ *
285
+ * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.jsonRequest} to ask the question
286
+ *
287
+ * (uses {@link DyFM_safeParseJSON})
288
+ */
289
+ async requestJSON<T = any>(
290
+ set: DyFM_AI_Message_Input<DyFM_OAI_CallSettings>
291
+ ): Promise<T | { unparsableResult: string }> {
292
+ set.message += this.predefinedRequests.jsonRequest;
293
+
294
+ const answer = await this.requestSimpleMessage(set);
295
+
296
+ return this.convertAnswerToJSON<T>(answer, set.message);
297
+ }
298
+ /** the exact same as {@link askJSONQuestion} */
299
+ //jsonQuestion = this.askJSONQuestion;
300
+
301
+ /**
302
+ * Asks the AI to answer a question that must result a JSON object with specific key descriptions
303
+ *
304
+ * If the answer is not a valid JSON object, it will return { unparsableResult: answer }
305
+ *
306
+ * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.jsonRequestWithKeysDescription} to ask the question
307
+ *
308
+ * (uses {@link DyFM_safeParseJSON})
309
+ */
310
+ async requestJSONQuestionWithKeysDescription<T = any>(
311
+ set: DyFM_AI_JSONKeysDescription_Input<DyFM_OAI_CallSettings>
312
+ ): Promise<T | { unparsableResult: string }> {
313
+ const jsonRequestWithKeysDescription = this.predefinedRequests.jsonRequestWithKeysDescription.replace(
314
+ '{{DyNTS_jsonKeysDescription}}',
315
+ set.keysDescription
316
+ );
317
+ set.message += jsonRequestWithKeysDescription;
318
+
319
+ const answer = await this.requestSimpleMessage(set);
320
+
321
+ return this.convertAnswerToJSON<T>(answer, set.message);
322
+ }
323
+ /** the exact same as {@link requestJSONQuestionWithKeysDescription} */
324
+ //jsonQuestionWithKeysDescription = this.askJSONQuestionWithKeysDescription;
325
+
326
+ /**
327
+ * Asks the AI to answer a question that must result a JSON object with specific keys
328
+ *
329
+ * If the answer is not a valid JSON object, it will return { unparsableResult: answer }
330
+ *
331
+ * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.jsonRequestWithExactKeys} to ask the question
332
+ *
333
+ * (uses {@link DyFM_safeParseJSON})
334
+ */
335
+ async requestJSONWithExactKeys<T = any>(
336
+ set: DyFM_AI_JSONExactKeys_Input<DyFM_OAI_CallSettings>
337
+ ): Promise<T | { unparsableResult: string }> {
338
+ const jsonRequestWithExactKeys = this.predefinedRequests.jsonRequestWithExactKeys.replace(
339
+ '{{DyNTS_jsonKeys}}',
340
+ this.getTextListAsText(set.keys)
341
+ );
342
+ set.message += jsonRequestWithExactKeys;
343
+
344
+ const answer = await this.requestSimpleMessage(set);
345
+
346
+ return this.convertAnswerToJSON<T>(answer, set.message);
347
+ }
348
+ /** the exact same as {@link askJSONQuestionWithExactKeys} */
349
+ //jsonQuestionWithExactKeys = this.askJSONQuestionWithExactKeys;
350
+
351
+ /**
352
+ * Asks the AI to answer a question that must result a list of strings
353
+ *
354
+ * If the answer is not a valid JSON list, it will return the answer as a single item in an array
355
+ *
356
+ * Uses the {@link DyNTS_OAI_LLM_Predefined_Requests.listRequest} to ask the question
357
+ *
358
+ * (uses {@link DyFM_Object.safeParseList})
359
+ */
360
+ /* async requestList(
361
+ set: DyFM_AI_Base_Input<DyFM_OAI_CallSettings>
362
+ ): Promise<string[]> {
363
+ set.message += this.predefinedRequests.listRequest;
364
+
365
+ const answer = await this.requestSimpleMessage(set);
366
+
367
+ return this.convertAnswerToList<string>(answer, set.message);
368
+ } */
369
+ /** the exact same as {@link askListQuestion} */
370
+ //listQuestion = this.askListQuestion;
371
+ /** the exact same as {@link askListQuestion} */
372
+ //listRequest = this.askListQuestion;
373
+
374
+ async requestList<T = any>(
375
+ set: DyFM_AI_Message_Input<DyFM_OAI_CallSettings>
376
+ ): Promise<T[] | { unparsableResult: string }> {
377
+ set.message += this.predefinedRequests.listRequest;
378
+
379
+ const answer = await this.requestSimpleMessage(set);
380
+
381
+ return this.convertAnswerToList<T>(answer, set.message);
382
+ }
383
+
384
+ /**
385
+ * Asks the AI to answer a question
386
+ *
387
+ * Uses the {@link resolveSimpleUserMessage}
388
+ */
389
+ async requestSimpleMessage(
390
+ set: DyFM_AI_Message_Input<DyFM_OAI_CallSettings>
391
+ ): Promise<string> {
392
+ this.logQuestion(set);
393
+
394
+ set.conversation ??= [];
395
+ set.conversation.push({ role: DyFM_AI_MessageRole.user, content: set.message });
396
+
397
+ const answer: string = await this.resolveMessage({
398
+ ...set,
399
+ conversation: set.conversation,
400
+ getFullResponse: false,
401
+ }) as string;
402
+
403
+ this.logAnswer(answer);
404
+
405
+ return answer;
406
+ }
407
+ /** the exact same as {@link requestSimpleMessage} */
408
+ //answer = this.askQuestion;
409
+ /** the exact same as {@link requestSimpleMessage} */
410
+ //getQuestionAnswer = this.askQuestion;
411
+ /** the exact same as {@link requestSimpleMessage} */
412
+ //simpleQuestion = this.askQuestion;
413
+ /** the exact same as {@link requestSimpleMessage} */
414
+ //askSimpleQuestion = this.askQuestion;
415
+
416
+ /**
417
+ * Resolves a simple user message and returns the answer as string
418
+ *
419
+ * Uses the {@link resolveMessage}
420
+ */
421
+ /* async resolveSimpleUserMessage(
422
+ set: DyFM_AI_Base_Input<DyFM_OAI_CallSettings>
423
+ ): Promise<string> {
424
+ return this.resolveMessage({
425
+ conversation: [{ role: DyFM_AI_MessageRole.user, content: set.message }],
426
+ issuer: set.issuer ?? 'unknown',
427
+ settings: set.settings ?? this.defaultSettings,
428
+ getFullResponse: false,
429
+ }) as Promise<string>;
430
+ } */
431
+ /** the exact same as {@link resolveSimpleUserMessage} */
432
+ //simpleUserMessage = this.resolveSimpleUserMessage;
433
+
434
+ //#endregion
435
+
436
+ /**
437
+ * Resolves a conversation
438
+ *
439
+ * Uses the {@link getMessageCreateInput}
440
+ */
441
+ async resolveMessage(
442
+ set: DyFM_AI_ConversationBase_Input<DyFM_OAI_CallSettings>
443
+ /* {
444
+ conversation: [{ role: DyFM_AI_MessageRole.user, content: set.message }],
445
+ issuer: set.issuer ?? 'unknown',
446
+ settings: set.settings ?? this.defaultSettings,
447
+ getFullResponse: false,
448
+ } */
449
+ /* {
450
+ conversation: DyFM_AI_Message[],
451
+ issuer: string,
452
+ settings?: DyFM_OAI_CallSettings,
453
+ getFullResponse?: boolean
454
+ } */
455
+ ): Promise<string | ChatCompletion> {
456
+ try {
457
+ // FR-002 — callType meghatározás a unshift ELŐTT (a system-message hozzáadás
458
+ // után már 2+ message lenne, ami a "chat" kategóriát mindig kiváltja).
459
+ // 1 user-message → 'llm-completion'; több → 'llm-chat'. A 'llm-tool-use'
460
+ // categoria a tools paraméterek bevezetésekor kerül használatba (jövőbeli FR).
461
+ const userMessageCount: number = set.conversation.length;
462
+
463
+ set.conversation.unshift(this.getDefaultSystemMessage(set.settings));
464
+
465
+ // FR-002 — per-call cost-event timing: a tényleges API-call latency mérve
466
+ const input: ChatCompletionCreateParamsBase = this.getMessageCreateInput(set);
467
+ const start: number = Date.now();
468
+ const result: ChatCompletion = await this.openai.chat.completions.create(
469
+ input,
470
+ ) as ChatCompletion;
471
+ const durationMs: number = Date.now() - start;
472
+
473
+ // FR-002 — cost-event emit (safe, no-op ha nincs onCostEvent callback)
474
+ this.emitCostEvent({
475
+ callType: userMessageCount === 1 ? 'llm-completion' : 'llm-chat',
476
+ provider: this.aiProvider,
477
+ model: String(input.model ?? this.defaultModel),
478
+ tokensUsed: {
479
+ input: result.usage?.prompt_tokens ?? 0,
480
+ output: result.usage?.completion_tokens ?? 0,
481
+ total: result.usage?.total_tokens ?? 0,
482
+ },
483
+ durationMs: durationMs,
484
+ issuer: set.issuer,
485
+ timestamp: new Date(),
486
+ });
487
+
488
+ if (set.getFullResponse) {
489
+ return result;
490
+ }
491
+
492
+ return result.choices[0].message.content;
493
+ } catch (error) {
494
+ throw new DyFM_Error({
495
+ ...this.getDefaultErrorSettings('resolveConversation', error, set.issuer),
496
+
497
+ errorCode: `${DyNTS_global_settings.systemShortCodeName}|DyNTS-OLSB-RM0`,
498
+ });
499
+ }
500
+ }
501
+
502
+ /**
503
+ * Creates a message create input
504
+ *
505
+ * Uses the {@link getDefaultSystemMessage}
506
+ */
507
+ protected getMessageCreateInput(
508
+ set: {
509
+ conversation: DyFM_AI_Message[],
510
+ issuer: string,
511
+ settings?: DyFM_OAI_CallSettings
512
+ }
513
+ ): ChatCompletionCreateParamsBase {
514
+ try {
515
+ const settings = set.settings ?? this.defaultSettings;
516
+
517
+ const result: ChatCompletionCreateParamsBase = {
518
+ model: settings?.useModel ?? this.defaultModel,
519
+ messages: set.conversation as ChatCompletionMessageParam[],
520
+ };
521
+
522
+ result.temperature = DyFM_notNull(settings?.temperature) ?
523
+ settings.temperature :
524
+ this.defaultSettings.temperature;
525
+
526
+ /* result.max_completion_tokens = DyFM_notNull(settings?.maxTokens) ?
527
+ settings.maxTokens :
528
+ this.defaultSettings.maxTokens; */
529
+ result.max_completion_tokens = DyFM_notNull(settings?.maxTokens) ?
530
+ settings.maxTokens :
531
+ this.defaultSettings.maxTokens;
532
+
533
+ result.top_p = DyFM_notNull(settings?.topP) ?
534
+ settings.topP :
535
+ this.defaultSettings.topP;
536
+
537
+ result.frequency_penalty = DyFM_notNull(settings?.frequencyPenalty) ?
538
+ settings.frequencyPenalty :
539
+ this.defaultSettings.frequencyPenalty;
540
+
541
+ result.presence_penalty = DyFM_notNull(settings?.presencePenalty) ?
542
+ settings.presencePenalty :
543
+ this.defaultSettings.presencePenalty;
544
+
545
+ return result;
546
+ } catch (error) {
547
+ throw new DyFM_Error({
548
+ ...this.getDefaultErrorSettings('getMessageCreateSettings', error, set.issuer),
549
+
550
+ errorCode: `${DyNTS_global_settings.systemShortCodeName}|DyNTS-OLSB-RC0`,
551
+ });
552
+ }
553
+ }
554
+
555
+ /**
556
+ * Teszteli a kapcsolatot az OpenAI API-val
557
+ *
558
+ * Tests the connection to OpenAI API
559
+ */
560
+ async testConnection(issuer: string): Promise<boolean> {
561
+ try {
562
+ await this.openai.models.list();
563
+ return true;
564
+ } catch (error) {
565
+ DyFM_Log.error('DyNTS_OAI_LLM_ServiceBase', 'testConnection', 'Connection test failed', {
566
+ error,
567
+ issuer
568
+ });
569
+ return false;
570
+ }
571
+ }
572
+
573
+ //#region Function calling (tool use) — FR-047
574
+
575
+ /** OpenAI tool-kepes modell-registry (a base capability-precheck-jehez). */
576
+ protected override getModelRegistry(): DyFM_AI_ModelInfo[] {
577
+ return DyFM_OAI_Models;
578
+ }
579
+
580
+ /**
581
+ * Egy OpenAI-kor a tool-loop-ban: felepiti a natív request-et (system + a teljes beszelgetes
582
+ * tool-uzenetekkel + tools), meghivja az API-t, es normalizalt DyFM_AI_LLM_Response-t ad vissza
583
+ * (toolCalls feltoltve, ha a model tool-t akar hivni). FR-002 cost-event: 'llm-tool-use'.
584
+ */
585
+ protected override async callModelWithTools(
586
+ set: {
587
+ conversation: DyFM_AI_Message[];
588
+ tools: DyFM_AI_Tool[];
589
+ settings?: DyFM_OAI_CallSettings;
590
+ issuer: string;
591
+ }
592
+ ): Promise<DyFM_AI_LLM_Response> {
593
+ try {
594
+ const settings: DyFM_OAI_CallSettings = set.settings ?? this.defaultSettings;
595
+
596
+ const messages: ChatCompletionMessageParam[] = [
597
+ this.toOpenAIMessage(this.getDefaultSystemMessage(settings)),
598
+ ...set.conversation.map((message: DyFM_AI_Message) => this.toOpenAIMessage(message)),
599
+ ];
600
+
601
+ const useModel: DyFM_OAI_Model | string = settings?.useModel ?? this.defaultModel;
602
+
603
+ const input: ChatCompletionCreateParamsBase = {
604
+ model: useModel,
605
+ messages: messages,
606
+ max_completion_tokens: DyFM_notNull(settings?.maxTokens) ? settings.maxTokens : this.defaultSettings.maxTokens,
607
+ };
608
+
609
+ // Reasoning modellek (GPT-5.x / o-series) elutasitjak a sampling-parametereket
610
+ // (temperature != 1, top_p, frequency/presence_penalty) -> 400. Ezeket NEM kuldjuk
611
+ // reasoning modellnel. (Ez a FR-048 per-model settings elv lokalis, OpenAI-specifikus
612
+ // alkalmazasa — amig a general reconciler el nem keszul.)
613
+ if (!this.isReasoningModel(String(useModel))) {
614
+ input.temperature = DyFM_notNull(settings?.temperature) ? settings.temperature : this.defaultSettings.temperature;
615
+ input.top_p = DyFM_notNull(settings?.topP) ? settings.topP : this.defaultSettings.topP;
616
+ input.frequency_penalty = DyFM_notNull(settings?.frequencyPenalty) ? settings.frequencyPenalty : this.defaultSettings.frequencyPenalty;
617
+ input.presence_penalty = DyFM_notNull(settings?.presencePenalty) ? settings.presencePenalty : this.defaultSettings.presencePenalty;
618
+ }
619
+
620
+ // Ures tools-tomb -> 400 ("tools must have minimum length 1"). Csak akkor kuldjuk, ha van tool.
621
+ if (set.tools.length) {
622
+ input.tools = set.tools.map((tool: DyFM_AI_Tool) => this.toOpenAITool(tool));
623
+ input.tool_choice = 'auto';
624
+ }
625
+
626
+ const start: number = Date.now();
627
+ const result: ChatCompletion = await this.openai.chat.completions.create(input) as ChatCompletion;
628
+ const durationMs: number = Date.now() - start;
629
+
630
+ const message = result.choices[0].message;
631
+
632
+ const toolCalls: DyFM_AI_ToolCall[] = (message.tool_calls ?? [])
633
+ .filter((toolCall) => toolCall.type === 'function')
634
+ .map((toolCall) => ({
635
+ id: toolCall.id,
636
+ name: toolCall.function.name,
637
+ arguments: this.parseToolArguments(toolCall.function.arguments),
638
+ }));
639
+
640
+ this.emitCostEvent({
641
+ callType: 'llm-tool-use',
642
+ provider: this.aiProvider,
643
+ model: String(input.model ?? this.defaultModel),
644
+ tokensUsed: {
645
+ input: result.usage?.prompt_tokens ?? 0,
646
+ output: result.usage?.completion_tokens ?? 0,
647
+ total: result.usage?.total_tokens ?? 0,
648
+ },
649
+ durationMs: durationMs,
650
+ issuer: set.issuer,
651
+ timestamp: new Date(),
652
+ });
653
+
654
+ return {
655
+ content: message.content ?? '',
656
+ toolCalls: toolCalls.length ? toolCalls : undefined,
657
+ usage: result.usage ? {
658
+ promptTokens: result.usage.prompt_tokens,
659
+ completionTokens: result.usage.completion_tokens,
660
+ totalTokens: result.usage.total_tokens,
661
+ } : undefined,
662
+ model: result.model,
663
+ finishReason: result.choices[0].finish_reason,
664
+ stopReason: result.choices[0].finish_reason,
665
+ rawResponse: result,
666
+ };
667
+ } catch (error) {
668
+ throw new DyFM_Error({
669
+ ...this.getDefaultErrorSettings('callModelWithTools', error, set.issuer),
670
+ errorCode: `${DyNTS_global_settings.systemShortCodeName}|DyNTS-OLSB-TC0`,
671
+ });
672
+ }
673
+ }
674
+
675
+ /** Agnosztikus tool-definicio → OpenAI ChatCompletionTool. */
676
+ protected toOpenAITool(tool: DyFM_AI_Tool): ChatCompletionTool {
677
+ return {
678
+ type: 'function',
679
+ function: {
680
+ name: tool.name,
681
+ description: tool.description,
682
+ parameters: tool.parameters as Record<string, unknown>,
683
+ },
684
+ };
685
+ }
686
+
687
+ /**
688
+ * Agnosztikus DyFM_AI_Message → OpenAI ChatCompletionMessageParam.
689
+ * Kezeli a tool-result (role:'tool') es a tool-hivo assistant (tool_calls) uzeneteket is.
690
+ */
691
+ protected toOpenAIMessage(message: DyFM_AI_Message): ChatCompletionMessageParam {
692
+ if (message.role === DyFM_AI_MessageRole.tool) {
693
+ return {
694
+ role: 'tool',
695
+ tool_call_id: message.toolCallId ?? '',
696
+ content: message.content,
697
+ };
698
+ }
699
+
700
+ if (message.role === DyFM_AI_MessageRole.assistant && message.toolCalls?.length) {
701
+ return {
702
+ role: 'assistant',
703
+ content: message.content || null,
704
+ tool_calls: message.toolCalls.map((toolCall: DyFM_AI_ToolCall) => ({
705
+ id: toolCall.id,
706
+ type: 'function' as const,
707
+ function: {
708
+ name: toolCall.name,
709
+ arguments: JSON.stringify(toolCall.arguments),
710
+ },
711
+ })),
712
+ };
713
+ }
714
+
715
+ return {
716
+ role: message.role as 'system' | 'user' | 'assistant',
717
+ content: message.content,
718
+ };
719
+ }
720
+
721
+ /** Tool-argumentumok parse-olasa (OpenAI JSON-string → object); hibas JSON eseten ures object. */
722
+ protected parseToolArguments(raw: string): Record<string, unknown> {
723
+ try {
724
+ return JSON.parse(raw || '{}');
725
+ } catch {
726
+ return {};
727
+ }
728
+ }
729
+
730
+ /**
731
+ * Reasoning modell-e (GPT-5.x sor vagy o-series, pl. o1/o3/o4)? Ezek elutasitjak a
732
+ * sampling-parametereket (temperature != 1, top_p, frequency/presence_penalty). A `gpt-4o`
733
+ * NEM reasoning (a 'gpt-5' prefix nem illeszti, 'o'-prefix sem).
734
+ */
735
+ protected isReasoningModel(modelId: string): boolean {
736
+ return /^gpt-5/.test(modelId) || /^o[0-9]/.test(modelId);
737
+ }
738
+
739
+ //#endregion
740
+
741
+ // async askJSONListQuestion(
742
+ // question: string,
743
+ // issuer: string,
744
+ // settings?: DyFM_OAI_CallSettings,
745
+ // debugLog?: boolean,
746
+ // /** this is used to readably replace too long contents to eg '...' in logs */
747
+ // replaceThisInLog?: string,
748
+ // ): Promise<{ key: string, input: string }[]> {
749
+ // question += this.predefinedRequests.jsonListRequest;
750
+ //
751
+ // const answer = await this.askQuestion(question, issuer, settings, debugLog, replaceThisInLog);
752
+ //
753
+ // return DyFM_Object.safeParseList<{ key: string, input: string }[]>(answer);
754
+ // }
755
+
756
+ //#region Tools
757
+
758
+ /**
759
+ * Logs the question if the debugLog is true
760
+ */
761
+ /* protected logQuestion(
762
+ set: DyFM_AI_Base_Input<DyFM_OAI_CallSettings>
763
+ ): void {
764
+ if (set.settings?.debugLog ?? this._debugLog) {
765
+ console.log('\n - ', set.message);
766
+ }
767
+ } */
768
+
769
+ /**
770
+ * olvasható mondatszerű-listaszerű formába teszi a listaelemeket
771
+ * pl.: ['a', 'b', 'c'] -> '"a", "b" or "c"'
772
+ */
773
+ /* getTextListAsText(list: string[]): string {
774
+ list = list.filter(item => item?.trim()).map(item => `"${item}"`);
775
+
776
+ //list = list.map(item => item.toLocaleLowerCase());
777
+
778
+ list.push(list.pop() + ' or ' + list.pop());
779
+
780
+ return list.join(', ');
781
+ } */
782
+
783
+ /* protected getDefaultSystemMessage(settings: DyFM_OAI_CallSettings): DyFM_AI_Message {
784
+ return {
785
+ role: DyFM_AI_MessageRole.system,
786
+ content: settings?.systemPrompt || this.defaultSystemPrompt,
787
+ };
788
+ } */
789
+
790
+ /* protected validateConversation(conversation: DyFM_AI_Message[]): void {
791
+ conversation.forEach((message: DyFM_AI_Message, index: number) => {
792
+ if (!message.role) {
793
+ throw new DyFM_Error({
794
+ message: `Message has no role at index ${index}`,
795
+ additionalContent: {
796
+ invalidMessage: message,
797
+ conversation: conversation,
798
+ }
799
+ });
800
+ }
801
+
802
+ if (!message.content) {
803
+ throw new DyFM_Error({
804
+ message: `Message has no content at index ${index}`,
805
+ additionalContent: {
806
+ invalidMessage: message,
807
+ conversation: conversation,
808
+ }
809
+ });
810
+ }
811
+ });
812
+ } */
813
+
814
+ /* protected getDefaultErrorSettings(
815
+ fnName: string,
816
+ error: DyFM_AnyError,
817
+ issuer: string
818
+ ): DyFM_Error_Settings {
819
+ return {
820
+ status: (error as DyFM_Error)?.___status ?? 500,
821
+ message: (error as Error)?.message ??
822
+ (error as DyFM_Error)?._message ??
823
+ `${fnName} was UNSUCCESSFUL (${DyNTS_global_settings.systemShortCodeName})`,
824
+ addECToUserMsg: !(error as DyFM_Error)?.__userMessage,
825
+ userMessage: (error as DyFM_Error)?.__userMessage ?? this.defaultErrorUserMsg,
826
+ issuer: issuer,
827
+ issuerService: this.constructor?.name,
828
+ error: error,
829
+ };
830
+ } */
831
+
832
+ //#endregion
833
+
834
+ //#region Provider Methods (from DyNTS_AI_Provider_ServiceBase)
835
+
836
+ //#endregion
837
+
838
+ //#region Abstract LLM Methods (from DyNTS_AI_LLM_ServiceBase)
839
+
840
+ /**
841
+ * Meghívja az LLM-et system és user message-szel
842
+ *
843
+ * Calls the LLM with system and user messages
844
+ */
845
+ /* async callLLM(
846
+ systemMessage: string,
847
+ userMessage: string,
848
+ settings?: DyFM_AI_CallSettings,
849
+ issuer?: string
850
+ ): Promise<string> {
851
+ const result = await this.resolveMessage({
852
+ conversation: [
853
+ { role: DyFM_AI_MessageRole.system, content: systemMessage },
854
+ { role: DyFM_AI_MessageRole.user, content: userMessage }
855
+ ],
856
+ issuer: issuer ?? 'system',
857
+ settings: this.convertToOAISettings(settings),
858
+ getFullResponse: false
859
+ });
860
+
861
+ if (typeof result === 'string') {
862
+ return result;
863
+ }
864
+ return '';
865
+ } */
866
+
867
+ /**
868
+ * Meghívja az LLM-et üzenet history-val
869
+ *
870
+ * Calls the LLM with message history
871
+ */
872
+ /* async callLLMWithHistory(
873
+ messages: DyFM_AI_Message[],
874
+ settings?: DyFM_AI_CallSettings,
875
+ issuer?: string
876
+ ): Promise<string> {
877
+ const result = await this.resolveMessage({
878
+ conversation: messages,
879
+ issuer: issuer ?? 'system',
880
+ settings: this.convertToOAISettings(settings),
881
+ getFullResponse: false
882
+ });
883
+
884
+ if (typeof result === 'string') {
885
+ return result;
886
+ }
887
+ return '';
888
+ } */
889
+
890
+ /**
891
+ * Meghívja az LLM-et és visszaadja a raw response-t
892
+ *
893
+ * Calls the LLM and returns raw response
894
+ */
895
+ /* async callLLMRaw(
896
+ messages: DyFM_AI_Message[],
897
+ settings?: DyFM_AI_CallSettings,
898
+ issuer?: string
899
+ ): Promise<DyFM_AI_LLM_Response> {
900
+ const result = await this.resolveMessage({
901
+ conversation: messages,
902
+ issuer: issuer ?? 'system',
903
+ settings: this.convertToOAISettings(settings),
904
+ getFullResponse: true
905
+ }) as ChatCompletion;
906
+
907
+ return {
908
+ content: result.choices[0].message.content,
909
+ usage: result.usage ? {
910
+ promptTokens: result.usage.prompt_tokens,
911
+ completionTokens: result.usage.completion_tokens,
912
+ totalTokens: result.usage.total_tokens
913
+ } : undefined,
914
+ model: result.model,
915
+ finishReason: result.choices[0].finish_reason,
916
+ rawResponse: result
917
+ };
918
+ } */
919
+
920
+ /**
921
+ * Küld egy egyszerű üzenetet az LLM-nek
922
+ *
923
+ * Sends a simple message to the LLM
924
+ */
925
+ /* async sendMessage(set: DyFM_AI_Base_Input<DyFM_OAI_CallSettings>): Promise<string> {
926
+ return await this.resolveSimpleUserMessage({
927
+ message: set.message,
928
+ issuer: set.issuer,
929
+ settings: set.settings ? this.convertToOAISettings(set.settings) : undefined
930
+ });
931
+ } */
932
+
933
+ /**
934
+ * Kérdez egy kérdést és JSON választ vár exact keys-szel
935
+ *
936
+ * Asks a question and expects JSON response with exact keys
937
+ */
938
+ /* async requestJSONWithExactKeys<T = any>(
939
+ set: DyFM_AI_JSONExactKeys_Input
940
+ ): Promise<T> {
941
+ const answer = await this.askJSONQuestionWithExactKeys({
942
+ question: set.question,
943
+ keys: set.keys,
944
+ issuer: set.issuer,
945
+ settings: set.settings ? this.convertToOAISettings(set.settings) : undefined,
946
+ debugLog: set.debugLog,
947
+ replaceThisInLog: set.replaceThisInLog
948
+ });
949
+
950
+ if ('unparsableResult' in answer) {
951
+ throw new DyFM_Error({
952
+ message: 'Invalid JSON response from AI',
953
+ additionalContent: { answer },
954
+ userMessage: 'Failed to parse AI response'
955
+ });
956
+ }
957
+
958
+ return answer as T;
959
+ } */
960
+
961
+ /**
962
+ * Kérdez egy kérdést és több választ vár
963
+ *
964
+ * Asks a question and expects multiple responses
965
+ */
966
+ /* async requestMultiselect<T = any>(
967
+ set: DyFM_AI_GenericMultiSelect_Input<T>
968
+ ): Promise<T[]> {
969
+ const answer = await this.requestMultipleSelect({
970
+ question: set.message,
971
+ options: set.options,
972
+ issuer: set.issuer,
973
+ settings: set.settings ? this.convertToOAISettings(set.settings) : undefined,
974
+ debugLog: set.debugLog,
975
+ replaceThisInLog: set.replaceThisInLog
976
+ });
977
+
978
+ return answer as T[];
979
+ } */
980
+
981
+ /**
982
+ * Konvertálja a generic AI settings-et OpenAI specifikus settings-re
983
+ *
984
+ * Converts generic AI settings to OpenAI-specific settings
985
+ */
986
+ /* private convertToOAISettings(settings?: DyFM_AI_CallSettings): DyFM_OAI_CallSettings {
987
+ if (!settings) {
988
+ return undefined;
989
+ }
990
+
991
+ const oaiSettings = new DyFM_OAI_CallSettings();
992
+
993
+ if (settings.systemPrompt !== undefined) {
994
+ oaiSettings.systemPrompt = settings.systemPrompt;
995
+ }
996
+
997
+ if (settings.useModel !== undefined) {
998
+ oaiSettings.useModel = settings.useModel as DyFM_OAI_Model;
999
+ }
1000
+
1001
+ if (settings.project !== undefined) {
1002
+ oaiSettings.project = settings.project;
1003
+ }
1004
+
1005
+ if (settings.debugLog !== undefined) {
1006
+ oaiSettings.debugLog = settings.debugLog;
1007
+ }
1008
+
1009
+ if (settings.temperature !== undefined) {
1010
+ oaiSettings.temperature = settings.temperature;
1011
+ }
1012
+
1013
+ if (settings.maxTokens !== undefined) {
1014
+ oaiSettings.maxTokens = settings.maxTokens;
1015
+ }
1016
+
1017
+ if (settings.topP !== undefined) {
1018
+ oaiSettings.topP = settings.topP;
1019
+ }
1020
+
1021
+ if (settings.frequencyPenalty !== undefined) {
1022
+ oaiSettings.frequencyPenalty = settings.frequencyPenalty;
1023
+ }
1024
+
1025
+ if (settings.presencePenalty !== undefined) {
1026
+ oaiSettings.presencePenalty = settings.presencePenalty;
1027
+ }
1028
+
1029
+ return oaiSettings;
1030
+ } */
1031
+
1032
+ //#endregion
1033
+ }