@futdevpro/nts-dynamo 1.15.53 → 1.15.55

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 (330) hide show
  1. package/.c8rc.json +26 -26
  2. package/.copilot/patterns.json +7 -7
  3. package/.cursor/rules/__assistant_guide.mdc +30 -30
  4. package/.cursor/rules/_ag_backend-structure.mdc +85 -85
  5. package/.cursor/rules/_ag_backend.mdc +16 -16
  6. package/.cursor/rules/_ag_frontend-structure.mdc +86 -86
  7. package/.cursor/rules/_ag_frontend.mdc +39 -39
  8. package/.cursor/rules/_ag_import-rules.mdc +44 -44
  9. package/.cursor/rules/_ag_naming.mdc +115 -115
  10. package/.cursor/rules/_ag_should-be.mdc +6 -6
  11. package/.cursor/rules/ai_development_guide.md +60 -60
  12. package/.cursor/rules/cursor-rules.md +160 -160
  13. package/.cursor/rules/default-command.mdc +464 -464
  14. package/.cursor/rules/error_code_pattern.md +39 -39
  15. package/.cursor/rules/saved rule mcp server use.md +15 -15
  16. package/.dynamo/pipeline.cicd.config.json +19 -1
  17. package/.github/workflows/main.yml +432 -432
  18. package/.vscode/settings.json +10 -10
  19. package/HOWTO.md +15 -15
  20. package/LICENSE +21 -21
  21. package/__documentations/nts-integration-tests-2026-03-17.md +26 -26
  22. package/_specifications/BACKLOG.md +92 -92
  23. package/_specifications/TODO.md +15 -15
  24. package/_specifications/agent.md +138 -138
  25. package/build/_models/interfaces/global-log-settings.interface.d.ts +26 -0
  26. package/build/_models/interfaces/global-log-settings.interface.d.ts.map +1 -1
  27. package/build/_services/server/app.server.d.ts +16 -0
  28. package/build/_services/server/app.server.d.ts.map +1 -1
  29. package/build/_services/server/app.server.js +24 -0
  30. package/build/_services/server/app.server.js.map +1 -1
  31. package/eslint.config.js +3 -3
  32. package/nodemon.json +24 -24
  33. package/package.json +361 -361
  34. package/pnpm-workspace.yaml +4 -4
  35. package/scripts/run-coverage-tests.js +28 -28
  36. package/spec/support/helpers/spec-reporter-loader.js +359 -359
  37. package/spec/support/helpers/ts-node-helper.js +93 -93
  38. package/spec/support/jasmine.coverage.json +24 -24
  39. package/spec/support/jasmine.json +24 -24
  40. package/src/_collections/archive.util.spec.ts +57 -57
  41. package/src/_collections/archive.util.ts +18 -18
  42. package/src/_collections/atlas-default-db-options.const.ts +9 -9
  43. package/src/_collections/default-fallback-cache-max-age.const.spec.ts +11 -11
  44. package/src/_collections/default-fallback-cache-max-age.const.ts +2 -2
  45. package/src/_collections/default-not-found-page.const.spec.ts +19 -19
  46. package/src/_collections/default-not-found-page.const.ts +22 -22
  47. package/src/_collections/default-socket-path.const.spec.ts +12 -12
  48. package/src/_collections/default-socket-path.const.ts +2 -2
  49. package/src/_collections/get-environment-settings.util.spec.ts +210 -210
  50. package/src/_collections/get-environment-settings.util.ts +48 -48
  51. package/src/_collections/global-settings.const.ts +70 -70
  52. package/src/_collections/sample.env +21 -21
  53. package/src/_collections/star.controller.spec.ts +224 -224
  54. package/src/_collections/star.controller.ts +129 -129
  55. package/src/_enums/data-model-type.enum.ts +14 -14
  56. package/src/_enums/data-service-function.enum.ts +24 -24
  57. package/src/_enums/predefined-data-types.enum.ts +16 -16
  58. package/src/_enums/route-security.enum.ts +12 -12
  59. package/src/_models/control-models/api-call-params.control-model.spec.ts +152 -152
  60. package/src/_models/control-models/api-call-params.control-model.ts +142 -142
  61. package/src/_models/control-models/app-ext-system-controls.control-model.spec.ts +52 -52
  62. package/src/_models/control-models/app-ext-system-controls.control-model.ts +9 -9
  63. package/src/_models/control-models/app-params.control-model.spec.ts +225 -225
  64. package/src/_models/control-models/app-params.control-model.ts +136 -136
  65. package/src/_models/control-models/app-system-controls.control-model.spec.ts +31 -31
  66. package/src/_models/control-models/app-system-controls.control-model.ts +9 -9
  67. package/src/_models/control-models/endpoint-params.control-model.spec.ts +627 -627
  68. package/src/_models/control-models/endpoint-params.control-model.ts +627 -627
  69. package/src/_models/control-models/http-settings.control-model.spec.ts +77 -77
  70. package/src/_models/control-models/http-settings.control-model.ts +37 -37
  71. package/src/_models/control-models/system-control.control-model.spec.ts +27 -27
  72. package/src/_models/control-models/system-control.control-model.ts +12 -12
  73. package/src/_models/interfaces/certification-settings.interface.ts +7 -7
  74. package/src/_models/interfaces/environment-settings.interface.ts +59 -59
  75. package/src/_models/interfaces/global-log-settings.interface.ts +171 -144
  76. package/src/_models/interfaces/global-service-settings.interface.ts +47 -47
  77. package/src/_models/interfaces/routing-module-settings.interface.ts +21 -21
  78. package/src/_models/interfaces/static-client-settings.interface.spec.ts +29 -29
  79. package/src/_models/interfaces/static-client-settings.interface.ts +28 -28
  80. package/src/_models/types/db-update.type.ts +100 -100
  81. package/src/_modules/ai/_models/ai-input-interfaces.ts +117 -117
  82. package/src/_modules/ai/_models/ai-test-generation-result.interface.ts +16 -16
  83. package/src/_modules/ai/_modules/anthropic/_services/aai-user-key.control-service.ts +138 -138
  84. package/src/_modules/ai/_modules/anthropic/index.ts +5 -5
  85. package/src/_modules/ai/_modules/document-ai/_collections/dai-chunking.util.spec.ts +242 -242
  86. package/src/_modules/ai/_modules/document-ai/_collections/dai-chunking.util.ts +639 -639
  87. package/src/_modules/ai/_modules/document-ai/_collections/dai-document.util.spec.ts +209 -209
  88. package/src/_modules/ai/_modules/document-ai/_collections/dai-document.util.ts +85 -85
  89. package/src/_modules/ai/_modules/document-ai/_enums/dai-compare-result-type.enum.ts +7 -7
  90. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-doc-chunk.data-model.ts +146 -146
  91. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-doc-page.data-model.ts +162 -162
  92. package/src/_modules/ai/_modules/document-ai/_models/data-models/dai-document.data-model.ts +99 -99
  93. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-doc-chunk-compare-result.interface.ts +18 -18
  94. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-doc-page-compare-result.interface.ts +19 -19
  95. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-document-compare-result.interface.ts +25 -25
  96. package/src/_modules/ai/_modules/document-ai/index.ts +28 -28
  97. package/src/_modules/ai/_modules/fdp-ai/_services/fdpai-user-key.control-service.ts +189 -189
  98. package/src/_modules/ai/_modules/fdp-ai/index.ts +5 -5
  99. package/src/_modules/ai/_modules/open-ai/_collections/oai-global-settings.const.ts +9 -9
  100. package/src/_modules/ai/_modules/open-ai/_collections/oai-llm-predefined-requests-hu.conts.ts +82 -82
  101. package/src/_modules/ai/_modules/open-ai/_collections/oai-llm-predefined-requests.conts.ts +75 -75
  102. package/src/_modules/ai/_modules/open-ai/_enums/oai-gpt-message-role.enum.ts +45 -45
  103. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-global-settings.interface.ts +7 -7
  104. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-gpt-message.interface.ts +7 -7
  105. package/src/_modules/ai/_modules/open-ai/_models/interfaces/oai-llm-predefined-requests.interface.ts +57 -57
  106. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-doc-chunk-data.service.ts +292 -292
  107. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-document.data-service.spec.ts +342 -342
  108. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-vector-data.service.spec.ts +550 -550
  109. package/src/_modules/ai/_modules/open-ai/_services/data-services/oai-vector-data.service.ts +630 -630
  110. package/src/_modules/ai/_modules/open-ai/_services/oai-embedding.control-service.spec.ts +332 -332
  111. package/src/_modules/ai/_modules/open-ai/_services/oai-llm-chat.service-base.spec.ts +462 -462
  112. package/src/_modules/ai/_modules/open-ai/_services/oai-llm-chat.service-base.ts +634 -634
  113. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.spec.ts +489 -489
  114. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.tools.spec.ts +173 -173
  115. package/src/_modules/ai/_modules/open-ai/_services/oai-llm.service-base.ts +1033 -1033
  116. package/src/_modules/ai/_modules/open-ai/_services/oai-user-key.control-service.ts +157 -157
  117. package/src/_modules/ai/_services/ai-embedding.service-base.spec.ts +98 -98
  118. package/src/_modules/ai/_services/ai-embedding.service-base.ts +48 -48
  119. package/src/_modules/ai/_services/ai-llm-chat.service-base.spec.ts +229 -229
  120. package/src/_modules/ai/_services/ai-llm-chat.service-base.ts +68 -68
  121. package/src/_modules/ai/_services/ai-llm.service-base.spec.ts +250 -250
  122. package/src/_modules/ai/_services/ai-llm.service-base.ts +519 -519
  123. package/src/_modules/ai/_services/ai-provider.service-base.spec.ts +158 -158
  124. package/src/_modules/ai/_services/ai-user-key.service-base.ts +59 -59
  125. package/src/_modules/ai/index.ts +13 -13
  126. package/src/_modules/assistant/_collections/ass-global-settings.const.ts +13 -13
  127. package/src/_modules/assistant/_collections/ass.util.spec.ts +176 -176
  128. package/src/_modules/assistant/_collections/ass.util.ts +50 -50
  129. package/src/_modules/assistant/_models/ass-global-settings.interface.ts +15 -15
  130. package/src/_modules/assistant/_services/ass-io.control-service.spec.ts +140 -140
  131. package/src/_modules/assistant/_services/ass-main.control-service.spec.ts +192 -192
  132. package/src/_modules/assistant/_services/ass-main.control-service.ts +107 -107
  133. package/src/_modules/bot/_collections/bot-default-commands.const.ts +12 -12
  134. package/src/_modules/bot/_collections/bot-global-settings.const.ts +39 -39
  135. package/src/_modules/bot/_models/bot-channel-wrapper.interface.ts +62 -62
  136. package/src/_modules/bot/_models/bot-command.interface.ts +8 -8
  137. package/src/_modules/bot/_models/bot-global-settings.interface.ts +96 -96
  138. package/src/_modules/bot/_models/bot-last-mention-date.interface.ts +6 -6
  139. package/src/_modules/bot/_models/bot-last-message-date.interface.ts +5 -5
  140. package/src/_modules/bot/_models/bot-user-wrapper.interface.ts +41 -41
  141. package/src/_modules/bot/_modules/discord-bot/_models/dib-platform.types.ts +9 -9
  142. package/src/_modules/bot/_modules/discord-bot/_services/dib-messaging-provider.control-service.spec.ts +431 -431
  143. package/src/_modules/bot/_modules/dynamo-bot/_collections/dyb-operations.util.spec.ts +160 -160
  144. package/src/_modules/bot/_modules/dynamo-bot/_collections/dyb-operations.util.ts +55 -55
  145. package/src/_modules/bot/_modules/dynamo-bot/_models/dyb-platform.types.ts +15 -15
  146. package/src/_modules/bot/_modules/dynamo-bot/_services/dyb-messaging-provider.control-service.spec.ts +374 -374
  147. package/src/_modules/bot/_modules/dynamo-bot/_services/dyb-messaging-provider.control-service.ts +447 -447
  148. package/src/_modules/bot/_modules/dynamo-bot/index.ts +15 -15
  149. package/src/_modules/bot/_modules/slack-bot/_models/slb-platform.types.ts +9 -9
  150. package/src/_modules/bot/_modules/slack-bot/_services/slb-messaging-provider.control-service.spec.ts +344 -344
  151. package/src/_modules/bot/_modules/slack-bot/_services/slb-messaging-provider.control-service.ts +197 -197
  152. package/src/_modules/bot/_modules/teams-bot/_models/teb-platform.types.ts +9 -9
  153. package/src/_modules/bot/_modules/teams-bot/_services/teb-messaging-provider.control-service.spec.ts +345 -345
  154. package/src/_modules/bot/_modules/teams-bot/_services/teb-messaging-provider.control-service.ts +197 -197
  155. package/src/_modules/bot/_services/bot-commands.control-service.spec.ts +116 -116
  156. package/src/_modules/bot/_services/bot-io.control-service.spec.ts +285 -285
  157. package/src/_modules/bot/_services/bot-main.control-service.spec.ts +208 -208
  158. package/src/_modules/bot/_services/bot-messaging-provider.service-base.spec.ts +349 -349
  159. package/src/_modules/bot/_services/bot-routines.control-service.spec.ts +111 -111
  160. package/src/_modules/custom-data/custom-data.controller.spec.ts +49 -49
  161. package/src/_modules/custom-data/custom-data.controller.ts +67 -67
  162. package/src/_modules/custom-data/custom-data.data-service.spec.ts +54 -54
  163. package/src/_modules/custom-data/custom-data.data-service.ts +21 -21
  164. package/src/_modules/custom-data/get-custom-data-routing-module.util.spec.ts +28 -28
  165. package/src/_modules/custom-data/get-custom-data-routing-module.util.ts +24 -24
  166. package/src/_modules/custom-data/index.ts +9 -9
  167. package/src/_modules/defaults/_collections/default-endpoints.util.ts +487 -487
  168. package/src/_modules/defaults/_models/default-user.data-model.ts +72 -72
  169. package/src/_modules/defaults/_services/default-auth.service.spec.ts +269 -269
  170. package/src/_modules/defaults/_services/default-auth.service.ts +177 -177
  171. package/src/_modules/defaults/_services/default-socket-events.service.spec.ts +42 -42
  172. package/src/_modules/defaults/_services/default-socket-events.service.ts +61 -61
  173. package/src/_modules/defaults/_services/default-user.data-service.spec.ts +187 -187
  174. package/src/_modules/defaults/_services/default-user.data-service.ts +98 -98
  175. package/src/_modules/defaults/index.ts +17 -17
  176. package/src/_modules/discord-assistant/_collections/dias-global-settings.const.ts +19 -19
  177. package/src/_modules/discord-assistant/_collections/dias.util.spec.ts +366 -366
  178. package/src/_modules/discord-assistant/_collections/dias.util.ts +132 -132
  179. package/src/_modules/discord-assistant/_models/dias-global-settings.interface.ts +19 -19
  180. package/src/_modules/discord-assistant/_models/dias-knowledge.data-model.ts +52 -52
  181. package/src/_modules/discord-assistant/_services/dias-chunk.data-service.ts +177 -177
  182. package/src/_modules/discord-assistant/_services/dias-io.control-service.spec.ts +108 -108
  183. package/src/_modules/discord-assistant/_services/dias-io.control-service.ts +69 -69
  184. package/src/_modules/discord-assistant/_services/dias-main.control-service.spec.ts +22 -22
  185. package/src/_modules/discord-assistant/_services/dias-main.control-service.ts +27 -27
  186. package/src/_modules/discord-assistant/_services/dias.service-base.spec.ts +195 -195
  187. package/src/_modules/discord-assistant/_services/dias.service-base.ts +76 -76
  188. package/src/_modules/discord-assistant/index.ts +38 -38
  189. package/src/_modules/discord-assistant-voiced/_services/dias-discord-bot.control-service.spec.ts +34 -34
  190. package/src/_modules/discord-assistant-voiced/_services/dias-discord-bot.control-service.ts +11 -11
  191. package/src/_modules/discord-assistant-voiced/index.ts +36 -36
  192. package/src/_modules/discord-bot/_collections/dibo-default-commands.const.ts +16 -16
  193. package/src/_modules/discord-bot/_collections/dibo-global-settings.conts.ts +55 -55
  194. package/src/_modules/discord-bot/_collections/dibo-operations.util.spec.ts +214 -214
  195. package/src/_modules/discord-bot/_collections/dibo-operations.util.ts +387 -387
  196. package/src/_modules/discord-bot/_models/dibo-command.interface.ts +12 -12
  197. package/src/_modules/discord-bot/_models/dibo-global-settings.interface.ts +98 -98
  198. package/src/_modules/discord-bot/_models/dibo-last-mention-date.inteface.ts +7 -7
  199. package/src/_modules/discord-bot/_models/dibo-last-message-date.interface.ts +6 -6
  200. package/src/_modules/discord-bot/_services/dibo-commands.control-service.spec.ts +154 -154
  201. package/src/_modules/discord-bot/_services/dibo-commands.control-service.ts +153 -153
  202. package/src/_modules/discord-bot/_services/dibo-io.control-service.spec.ts +264 -264
  203. package/src/_modules/discord-bot/_services/dibo-io.control-service.ts +306 -306
  204. package/src/_modules/discord-bot/_services/dibo-main.control-service.spec.ts +408 -408
  205. package/src/_modules/discord-bot/_services/dibo-main.control-service.ts +487 -487
  206. package/src/_modules/discord-bot/_services/dibo-routines.control-service.spec.ts +105 -105
  207. package/src/_modules/discord-bot/index.ts +36 -36
  208. package/src/_modules/local-vector-search/_enums/lvs-search-mode.enum.ts +35 -35
  209. package/src/_modules/local-vector-search/_models/lvs-search-result.interface.ts +17 -17
  210. package/src/_modules/local-vector-search/_services/lvs-doc-chunk-data.service.spec.ts +418 -418
  211. package/src/_modules/local-vector-search/_services/lvs-doc-chunk-data.service.ts +276 -276
  212. package/src/_modules/local-vector-search/_services/lvs-local-vector-search.data-service.spec.ts +480 -480
  213. package/src/_modules/local-vector-search/_services/lvs-local-vector-search.data-service.ts +416 -416
  214. package/src/_modules/local-vector-search/_services/lvs-vector-pool.control-service.spec.ts +393 -393
  215. package/src/_modules/local-vector-search/_services/lvs-vector-pool.control-service.ts +220 -220
  216. package/src/_modules/local-vector-search/index.ts +11 -11
  217. package/src/_modules/logs/index.ts +11 -11
  218. package/src/_modules/messaging/README.md +354 -354
  219. package/src/_modules/messaging/_collections/get-messaging-routing-module.util.ts +26 -26
  220. package/src/_modules/messaging/_collections/msg-global-settings.const.ts +22 -22
  221. package/src/_modules/messaging/_collections/msg.util.spec.ts +226 -226
  222. package/src/_modules/messaging/_models/msg-global-settings.interface.ts +37 -37
  223. package/src/_modules/messaging/_services/msg-conversation.data-service.ts +146 -146
  224. package/src/_modules/messaging/_services/msg-events.service.spec.ts +219 -219
  225. package/src/_modules/messaging/_services/msg-events.service.ts +267 -267
  226. package/src/_modules/messaging/_services/msg-integration.control-service.ts +179 -179
  227. package/src/_modules/messaging/_services/msg-main.control-service.spec.ts +147 -147
  228. package/src/_modules/messaging/_services/msg-main.control-service.ts +571 -571
  229. package/src/_modules/messaging/_services/msg-message.data-service.ts +129 -129
  230. package/src/_modules/messaging/_services/msg.controller.spec.ts +201 -201
  231. package/src/_modules/messaging/index.ts +30 -30
  232. package/src/_modules/mock/app-extended-server.mock.ts +201 -201
  233. package/src/_modules/mock/app-integration-test.mock.ts +51 -51
  234. package/src/_modules/mock/app-params.mock.spec.ts +21 -21
  235. package/src/_modules/mock/app-params.mock.ts +9 -9
  236. package/src/_modules/mock/app-server.mock.ts +188 -188
  237. package/src/_modules/mock/auth-service.mock.spec.ts +47 -47
  238. package/src/_modules/mock/auth-service.mock.ts +28 -28
  239. package/src/_modules/mock/controller.mock.spec.ts +26 -26
  240. package/src/_modules/mock/controller.mock.ts +16 -16
  241. package/src/_modules/mock/data-model.mock.spec.ts +111 -111
  242. package/src/_modules/mock/data-model.mock.ts +82 -82
  243. package/src/_modules/mock/email-service-collection.mock.spec.ts +24 -24
  244. package/src/_modules/mock/email-service-collection.mock.ts +15 -15
  245. package/src/_modules/mock/email-service.mock.spec.ts +17 -17
  246. package/src/_modules/mock/email-service.mock.ts +20 -20
  247. package/src/_modules/mock/email-template.mock.html +14 -14
  248. package/src/_modules/mock/endpoint.mock.ts +91 -91
  249. package/src/_modules/mock/socket-client.mock.spec.ts +40 -40
  250. package/src/_modules/mock/socket-client.mock.ts +45 -45
  251. package/src/_modules/mock/socket-server.mock.spec.ts +44 -44
  252. package/src/_modules/mock/socket-server.mock.ts +46 -46
  253. package/src/_modules/oauth2/_routes/oauth2.controller.spec.ts +107 -107
  254. package/src/_modules/oauth2/_routes/oauth2.controller.ts +98 -98
  255. package/src/_modules/oauth2/_services/oauth2.auth-service.spec.ts +254 -254
  256. package/src/_modules/oauth2/_services/oauth2.auth-service.ts +232 -232
  257. package/src/_modules/oauth2/_services/oauth2.control-service.spec.ts +585 -585
  258. package/src/_modules/oauth2/_services/oauth2.control-service.ts +653 -653
  259. package/src/_modules/oauth2/index.ts +17 -17
  260. package/src/_modules/server/errors/errors.control-service.spec.ts +238 -238
  261. package/src/_modules/server/errors/errors.control-service.ts +85 -85
  262. package/src/_modules/server/errors/errors.controller.spec.ts +241 -241
  263. package/src/_modules/server/errors/errors.controller.ts +431 -431
  264. package/src/_modules/server/errors/errors.data-service.spec.ts +361 -361
  265. package/src/_modules/server/index.ts +30 -30
  266. package/src/_modules/server/server-status/server-status-snapshot.control-service.spec.ts +70 -70
  267. package/src/_modules/server/server-status/server-status-snapshot.control-service.ts +17 -17
  268. package/src/_modules/server/server-status/server-status-snapshot.data-service.spec.ts +77 -77
  269. package/src/_modules/server/server-status/server-status-snapshot.data-service.ts +37 -37
  270. package/src/_modules/server/server-status/server-status.control-service.spec.ts +576 -576
  271. package/src/_modules/server/server-status/server-status.control-service.ts +396 -396
  272. package/src/_modules/server/server-status/server-status.controller.spec.ts +240 -240
  273. package/src/_modules/server/server-status/server-status.controller.ts +253 -253
  274. package/src/_modules/socket/_enums/socket-security.enum.ts +11 -11
  275. package/src/_modules/socket/_models/socket-client-service-params.control-model.spec.ts +32 -32
  276. package/src/_modules/socket/_models/socket-client-service-params.control-model.ts +22 -22
  277. package/src/_modules/socket/_models/socket-presence.control-model.spec.ts +164 -164
  278. package/src/_modules/socket/_models/socket-presence.control-model.ts +210 -210
  279. package/src/_modules/socket/_models/socket-server-service-params.control-model.spec.ts +46 -46
  280. package/src/_modules/socket/_models/socket-server-service-params.control-model.ts +22 -22
  281. package/src/_modules/socket/_services/socket-client.service.spec.ts +15 -15
  282. package/src/_modules/socket/_services/socket-client.service.ts +260 -260
  283. package/src/_modules/socket/_services/socket-server.service.spec.ts +11 -11
  284. package/src/_modules/socket/app-extended.integration.spec.ts +85 -85
  285. package/src/_modules/socket/app-extended.server.ts +630 -630
  286. package/src/_modules/socket/index.ts +42 -42
  287. package/src/_modules/test/get-test-routing-module.util.spec.ts +28 -28
  288. package/src/_modules/test/get-test-routing-module.util.ts +23 -23
  289. package/src/_modules/test/index.ts +11 -11
  290. package/src/_modules/test/test.controller.spec.ts +72 -72
  291. package/src/_modules/test/test.controller.ts +115 -115
  292. package/src/_modules/usage/get-usage-routing-module.util.ts +22 -22
  293. package/src/_modules/usage/index.ts +15 -15
  294. package/src/_modules/usage/usage.controller.spec.ts +81 -81
  295. package/src/_modules/usage/usage.controller.ts +126 -126
  296. package/src/_modules/usage/usage.data-service.spec.ts +332 -332
  297. package/src/_modules/usage/usage.data-service.ts +185 -185
  298. package/src/_services/base/api.service-base.spec.ts +125 -125
  299. package/src/_services/base/api.service-base.ts +74 -74
  300. package/src/_services/base/archive-data.service.spec.ts +196 -196
  301. package/src/_services/base/archive-data.service.ts +216 -216
  302. package/src/_services/base/data.service.spec.ts +674 -674
  303. package/src/_services/base/data.service.ts +2719 -2719
  304. package/src/_services/base/db.service.spec.ts +73 -73
  305. package/src/_services/base/db.service.ts +1575 -1575
  306. package/src/_services/base/singleton.service-base.spec.ts +28 -28
  307. package/src/_services/base/singleton.service-base.ts +24 -24
  308. package/src/_services/base/singleton.service.spec.ts +114 -114
  309. package/src/_services/base/singleton.service.ts +38 -38
  310. package/src/_services/core/api.service.spec.ts +140 -140
  311. package/src/_services/core/auth.service.spec.ts +159 -159
  312. package/src/_services/core/auth.service.ts +174 -174
  313. package/src/_services/core/email.service.spec.ts +85 -85
  314. package/src/_services/core/email.service.ts +742 -742
  315. package/src/_services/core/global.service.spec.ts +275 -275
  316. package/src/_services/core/global.service.ts +461 -461
  317. package/src/_services/core/service-collection.service.spec.ts +46 -46
  318. package/src/_services/core/service-collection.service.ts +6 -6
  319. package/src/_services/route/controller.service.spec.ts +53 -53
  320. package/src/_services/route/controller.service.ts +148 -148
  321. package/src/_services/route/routing-module.service.spec.ts +98 -98
  322. package/src/_services/route/routing-module.service.ts +330 -330
  323. package/src/_services/server/app.server.ts +1713 -1672
  324. package/src/_services/shared.static-service.spec.ts +99 -99
  325. package/src/_services/shared.static-service.ts +78 -78
  326. package/src/index.ts +95 -95
  327. package/tsconfig.app.json +12 -12
  328. package/tsconfig.json +42 -42
  329. package/.dynamo/logs/cicd-pipeline/output.log +0 -2682
  330. package/.dynamo/logs/cicd-pipeline/status.json +0 -351
@@ -1,639 +1,639 @@
1
- import { DyFM_Log, DyFM_Error, DyFM_AnyError, DyFM_Error_Settings } from '@futdevpro/fsm-dynamo';
2
- import { DyNTS_global_settings } from '../../../../../_collections/global-settings.const';
3
- import { DyNTS_DAI_DocChunk } from '../_models/data-models/dai-doc-chunk.data-model';
4
- import { DyNTS_DAI_DocPage } from '../_models/data-models/dai-doc-page.data-model';
5
-
6
-
7
- interface DyNTS_DAI_DocChunking_Header {
8
- level: number;
9
- header: string;
10
- index: number;
11
- }
12
-
13
- interface DyNTS_DAI_DocChunking_BreakPoint {
14
- index: number;
15
- level: number;
16
- }
17
-
18
- /**
19
- * ClickUp dokumentum chunking szolgáltatás
20
- * @author AI
21
- * @description ClickUp dokumentumok intelligens chunking-ja a leghosszabb lehetséges chunk-ok létrehozásához
22
- */
23
- export class DyNTS_DAI_DocChunking_Util {
24
-
25
- protected static readonly defaultErrorUserMsg: string =
26
- `We encountered an unhandled Control Service Error, ` +
27
- `\nplease contact the responsible development team.`;
28
-
29
- /**
30
- * Unified break points configuration
31
- */
32
- private static readonly breakPoints: { pattern: string, level: number, includeLength: number }[] = [
33
- { pattern: '\n# ', level: 1, includeLength: 0 }, // Header 1 - cut before marker
34
- { pattern: '\n## ', level: 2, includeLength: 0 }, // Header 2 - cut before marker
35
- { pattern: '\n### ', level: 3, includeLength: 0 }, // Header 3 - cut before marker
36
- { pattern: '\n#### ', level: 4, includeLength: 0 }, // Header 4 - cut before marker
37
- { pattern: '\n\n', level: 0, includeLength: 2 }, // Double line break - include
38
- { pattern: '\n', level: 0, includeLength: 1 }, // Single line break - include
39
- { pattern: '. ', level: 0, includeLength: 2 }, // Sentence end - include
40
- { pattern: ' ', level: 0, includeLength: 1 }, // Word break - include
41
- ];
42
-
43
- protected static readonly debugLog: boolean = false;
44
-
45
- /**
46
- * Markdown tartalom chunking-ja a leghosszabb lehetséges chunk-ok létrehozásához
47
- * @param page - ClickUp dokumentum oldal
48
- * @param issuer - Kérés kezdeményezője
49
- * @returns Promise<CU_DocChunk[]> - Létrehozott chunk-ok listája
50
- */
51
- static async chunkMdContent<
52
- T_Chunk extends DyNTS_DAI_DocChunk,
53
- T_Page extends DyNTS_DAI_DocPage<T_Chunk>,
54
- >(
55
- page: T_Page,
56
- getPageLink: (page: T_Page, issuer: string) => string,
57
- issuer: string,
58
- debugLog?: boolean
59
- ): Promise<T_Chunk[]> {
60
- try {
61
- if (debugLog) {
62
- DyFM_Log.info(`Chunking page "${page.name}"...`);
63
- }
64
-
65
- if (page.content.includes('#####')) {
66
- DyFM_Log.warn('lvl5+ headers are not supported yet for page: ' + page.name);
67
- }
68
-
69
- if (!page.documentId) {
70
- throw new DyFM_Error({
71
- ...this.getDefaultErrorSettings('chunkMdContent', new Error('Page documentId is not set'), issuer),
72
- errorCode: 'CCAP-CUC-CC2',
73
- });
74
- }
75
-
76
- if (!page._id) {
77
- throw new DyFM_Error({
78
- ...this.getDefaultErrorSettings('chunkMdContent', new Error('Page _id is not set'), issuer),
79
- errorCode: 'CCAP-CUC-CC3',
80
- });
81
- }
82
-
83
- /* DyFM_Log.log(`Chunking page: ${page.name}\n\n`, JSON.stringify(page.content, null, 2)); */
84
-
85
- let leftovers: string = page.content;
86
- const maxChunkSize: number = DyNTS_global_settings.docChunking.maxChunkSize;
87
- const maxChunkCount: number = DyNTS_global_settings.docChunking.maxChunkCount;
88
- const chunks: T_Chunk[] = [];
89
- let chunkIndex: number = 0;
90
- let currentPosition: number = 0; // Track current position in original content
91
-
92
- // Parse all headers to build hierarchy
93
- const headerHierarchy: DyNTS_DAI_DocChunking_Header[] = this.parseHeaderHierarchy(leftovers);
94
-
95
- // Debug: Log parsed headers
96
- /* console.log('\n=== Parsed Header Hierarchy ===');
97
- headerHierarchy.forEach((header, index) => {
98
- console.log(`${index}: Level ${header.level} at position ${header.index}: "${header.header}"`);
99
- });
100
- console.log('=== End Header Hierarchy ===\n'); */
101
-
102
- // Track the last headers of each level as we progress
103
- const lastHeaders: Map<number, string> = new Map();
104
-
105
- while (leftovers.length > 0 && chunkIndex < maxChunkCount) {
106
- // Ha a teljes tartalom belefér egy chunk-ba, adjuk vissza az egészet
107
- if (leftovers.length <= maxChunkSize) {
108
- const chunkContent = leftovers.trim();
109
-
110
- if (chunkContent.length > 0) {
111
- //const chunkWithHierarchy: string = this.addHeaderHierarchy(chunkContent, headerHierarchy, currentPosition, currentPosition + leftovers.length, lastHeaders);
112
- //const chunkPath: string[] = this.buildChunkPath(page.name, headerHierarchy, currentPosition, currentPosition + leftovers.length, lastHeaders);
113
- const headParents: string[] = this.getRelevantHeaders(headerHierarchy, currentPosition, currentPosition + leftovers.length, lastHeaders); // headerHierarchy.map(header => header.header);
114
- //const allFlaggedParents: string[] = [ ...page.allFlaggedParents, ...headParents ];
115
-
116
- const chunk: T_Chunk = new DyNTS_DAI_DocChunk({
117
- externalDocumentId: page.externalDocumentId,
118
- documentId: page.documentId,
119
- externalPageId: page.externalPageId,
120
- pageId: page._id,
121
-
122
- filePathParents: page.parentedPath,
123
- chunkHeadParents: headParents,
124
-
125
- allFlaggedParents: [ ...page.allFlaggedParents, ...headParents ],
126
- /* path: [ ...page.parentedPath, ...headParents ].join('/'), */
127
- path: page.allFlaggedParentsMerged,
128
-
129
- documentName: page.externalDocumentId,
130
- pageName: page.name,
131
- pageLink: getPageLink(page, issuer),
132
- chunkIndex: chunkIndex,
133
-
134
- chunkContent: chunkContent,
135
- /* chunkOriginalContent: chunkContent,
136
-
137
- chunkContent: chunkWithHierarchy, */
138
- }) as T_Chunk;
139
-
140
- if (this.debugLog || debugLog) {
141
- DyFM_Log.H_info(
142
- 'Full content (before assembleChunkWithHeaders)',
143
- `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
144
- { chunk: chunk.chunkContent }
145
- );
146
- }
147
-
148
- /* chunk.chunkContent = this.assembleChunkWithHeaders(chunk, issuer); */
149
- if (this.debugLog || debugLog) {
150
- DyFM_Log.H_info(
151
- 'Full content (after assembleChunkWithHeaders)',
152
- `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
153
- {
154
- chunk: chunk.chunkContent,
155
- }
156
- );
157
- }
158
-
159
- /* await DyFM_Async.wait(100_000); */
160
- chunks.push(chunk);
161
- }
162
- break;
163
- }
164
-
165
- // Keressük meg a legjobb break point-ot a maxChunkSize-n belül
166
- const breakPoint: DyNTS_DAI_DocChunking_BreakPoint | null = this.findBestBreakPoint(leftovers, maxChunkSize);
167
-
168
- if (breakPoint && breakPoint.index > 0) {
169
- // Ha találtunk break point-ot, vágjunk a marker előtt
170
- const chunkContent: string = leftovers.substring(0, breakPoint.index).trim();
171
-
172
- // Debug: Log current state
173
- /* console.log(`\n--- Creating chunk ${chunkIndex} ---`);
174
- console.log(`Current position: ${currentPosition}`);
175
- console.log(`Last headers:`, Object.fromEntries(lastHeaders));
176
- console.log(`Chunk content preview:`, chunkContent.substring(0, 100) + '...'); */
177
-
178
- // Check if chunk has headers
179
- //const hasHeaders: boolean = headerHierarchy.some(header =>
180
- // header.index >= currentPosition && header.index < currentPosition + breakPoint.index
181
- //);
182
- /* console.log(`Chunk has headers: ${hasHeaders}`); */
183
-
184
- //const chunkWithHierarchy: string = this.addHeaderHierarchy(chunkContent, headerHierarchy, currentPosition, currentPosition + breakPoint.index, lastHeaders);
185
- //const chunkPath: string[] = this.buildChunkPath(page.name, headerHierarchy, currentPosition, currentPosition + breakPoint.index, lastHeaders);
186
- const headParents: string[] = this.getRelevantHeaders(headerHierarchy, currentPosition, currentPosition + breakPoint.index, lastHeaders); // headerHierarchy.map(header => header.header);
187
- //const allFlaggedParents: string[] = [ ...page.allFlaggedParents, ...headParents ];
188
-
189
- /* console.log(`Chunk path:`, chunkPath);
190
- console.log(`Chunk with hierarchy preview:`, chunkWithHierarchy.substring(0, 100) + '...'); */
191
-
192
- const chunk: T_Chunk = new DyNTS_DAI_DocChunk({
193
- externalDocumentId: page.externalDocumentId,
194
- externalPageId: page.externalPageId,
195
- documentId: page.documentId,
196
- pageId: page._id,
197
-
198
- filePathParents: page.parentedPath,
199
- chunkHeadParents: headParents,
200
-
201
- allFlaggedParents: [ ...page.allFlaggedParents, ...headParents ],
202
- path: [ ...page.parentedPath, ...headParents ].join('/'),
203
-
204
- documentName: page.externalDocumentId,
205
- pageName: page.name,
206
- pageLink: getPageLink(page, issuer),
207
- chunkIndex: chunkIndex,
208
-
209
- chunkContent: chunkContent,
210
- /* chunkOriginalContent: chunkContent,
211
-
212
- chunkContent: chunkWithHierarchy, */
213
- }) as T_Chunk;
214
-
215
- if (this.debugLog || debugLog) {
216
- DyFM_Log.H_info(
217
- 'Break point (before assembleChunkWithHeaders)',
218
- `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
219
- { chunk: chunk.chunkContent }
220
- );
221
- }
222
-
223
- /* chunk.chunkContent = this.assembleChunkWithHeaders(chunk, issuer); */
224
- if (this.debugLog || debugLog) {
225
- DyFM_Log.H_info(
226
- 'Break point (after assembleChunkWithHeaders)',
227
- `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
228
- {
229
- chunk: chunk.chunkContent,
230
- }
231
- );
232
- }
233
-
234
- chunks.push(chunk);
235
- chunkIndex++;
236
- currentPosition += breakPoint.index;
237
- leftovers = leftovers.substring(breakPoint.index);
238
-
239
- // Update last headers after creating the chunk
240
- this.updateLastHeaders(headerHierarchy, currentPosition, lastHeaders);
241
-
242
- /* console.log(`Updated last headers:`, Object.fromEntries(lastHeaders));
243
- console.log(`--- End chunk ${chunkIndex - 1} ---\n`); */
244
- } else {
245
- // Ha nem találtunk break point-ot, használjuk a maxChunkSize-t
246
- const chunkContent: string = leftovers.substring(0, maxChunkSize).trim();
247
- //const chunkWithHierarchy: string = this.addHeaderHierarchy(chunkContent, headerHierarchy, currentPosition, currentPosition + maxChunkSize, lastHeaders);
248
- //const chunkPath: string[] = this.buildChunkPath(page.name, headerHierarchy, currentPosition, currentPosition + maxChunkSize, lastHeaders);
249
- const headParents: string[] = this.getRelevantHeaders(headerHierarchy, currentPosition, currentPosition + maxChunkSize, lastHeaders); // headerHierarchy.map(header => header.header);
250
- //const allFlaggedParents: string[] = [ ...page.allFlaggedParents, ...headParents ];
251
-
252
- const chunk: T_Chunk = new DyNTS_DAI_DocChunk({
253
- externalDocumentId: page.externalDocumentId,
254
- externalPageId: page.externalPageId,
255
- documentId: page.documentId,
256
- pageId: page._id,
257
-
258
- filePathParents: page.parentedPath,
259
- chunkHeadParents: headParents,
260
-
261
- allFlaggedParents: [ ...page.allFlaggedParents, ...headParents ],
262
- path: [ ...page.parentedPath, ...headParents ].join('/'),
263
-
264
- documentName: page.externalDocumentId,
265
- pageName: page.name,
266
- pageLink: getPageLink(page, issuer),
267
- chunkIndex: chunkIndex,
268
-
269
- chunkContent: chunkContent,
270
- /* chunkOriginalContent: chunkContent,
271
-
272
- chunkContent: chunkWithHierarchy, */
273
- }) as T_Chunk;
274
-
275
- if (this.debugLog || debugLog) {
276
- DyFM_Log.H_info(
277
- 'Max chunk size',
278
- `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
279
- { chunk: chunk.chunkContent }
280
- );
281
- }
282
-
283
- /* chunk.chunkContent = this.assembleChunkWithHeaders(chunk, issuer); */
284
- if (this.debugLog || debugLog) {
285
- DyFM_Log.H_info(
286
- `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
287
- {
288
- chunk: chunk.chunkContent,
289
- }
290
- );
291
- }
292
-
293
- chunks.push(chunk);
294
- chunkIndex++;
295
- currentPosition += maxChunkSize;
296
- leftovers = leftovers.substring(maxChunkSize);
297
-
298
- // Update last headers after creating the chunk
299
- this.updateLastHeaders(headerHierarchy, currentPosition, lastHeaders);
300
- }
301
-
302
- // Safety check to prevent infinite loops
303
- if (leftovers.length > 0 && leftovers.length >= page.content.length) {
304
- throw new DyFM_Error({
305
- ...this.getDefaultErrorSettings(
306
- 'chunkMdContent',
307
- new Error('Infinite loop detected in chunking'),
308
- issuer
309
- ),
310
- errorCode: 'CCAP-CUC-CC1',
311
- });
312
- }
313
- }
314
-
315
- // Log warning if we hit the chunk count limit
316
- if (chunkIndex >= maxChunkCount && leftovers.length > 0) {
317
- DyFM_Log.warn(
318
- `⚠️ Reached maximum chunk count (${maxChunkCount}). Remaining content: ${leftovers.length} characters`,
319
- {
320
- pageName: page.name,
321
- pageSize: page.content.length,
322
- }
323
- );
324
- }
325
-
326
- /* DyFM_Log.info(`✅ Created ${chunks.length} chunks`); */
327
- return chunks;
328
- } catch (error) {
329
- throw new DyFM_Error({
330
- ...this.getDefaultErrorSettings('chunkMdContent', error, issuer),
331
- errorCode: 'CCAP-CUC-CP0',
332
- });
333
- }
334
- }
335
-
336
- /* private static assembleChunkWithHeaders(chunk: DyNTS_DAI_DocChunk, issuer: string): string {
337
- try {
338
- const fileHeaders = chunk.filePathParents.map(
339
- (header, index) => `${'>'.repeat(index + 1)} **${header}**`
340
- ).join('\n');
341
- const headers = chunk.chunkHeadParents.filter(
342
- path => !chunk.chunkOriginalContent.includes(path)
343
- ).join('\n');
344
- return `${fileHeaders}\n\n${headers}\n\n${chunk.chunkOriginalContent}`;
345
- } catch (error) {
346
- throw new DyFM_Error({
347
- ...this.getDefaultErrorSettings('assembleChunkWithHeaders', error, issuer),
348
- errorCode: 'CCAP-CUC-ACH0',
349
- });
350
- }
351
- } */
352
-
353
- static reassembleChunksToContent(
354
- chunks: DyNTS_DAI_DocChunk[],
355
- issuer: string
356
- ): string {
357
- try {
358
- let pageContent: string = '';
359
-
360
- for (const chunk of chunks) {
361
- if (!chunk) {
362
- throw new DyFM_Error({
363
- ...this.getDefaultErrorSettings(
364
- 'reassembleChunksToContent',
365
- new Error('Chunk is null, you`ve passing null chunks to the function'),
366
- issuer
367
- ),
368
- errorCode: 'CCAP-CUC-RCTC0',
369
-
370
- additionalContent: {
371
- chunks: chunks,
372
- },
373
- });
374
- }
375
-
376
- pageContent += chunk.chunkContent;
377
- }
378
-
379
- return pageContent;
380
- } catch (error) {
381
- throw new DyFM_Error({
382
- ...this.getDefaultErrorSettings('reassembleChunks', error, issuer),
383
- errorCode: 'CCAP-CUC-RC0',
384
- });
385
- }
386
- }
387
-
388
- /**
389
- * Find the best break point within the chunk size
390
- * @param text - The text to search in
391
- * @param maxIndex - Maximum index to search up to
392
- * @returns The best break point or null if not found
393
- */
394
- private static findBestBreakPoint(text: string, maxIndex: number): DyNTS_DAI_DocChunking_BreakPoint | null {
395
- for (const breakPoint of this.breakPoints) {
396
- const index: number = this.findLastOccurrence(text, breakPoint.pattern, maxIndex);
397
-
398
- if (index > 0) {
399
- return {
400
- index: index + breakPoint.includeLength,
401
- level: breakPoint.level
402
- };
403
- }
404
- }
405
- return null;
406
- }
407
-
408
- /**
409
- * Parse header hierarchy from the content
410
- * @param content - The content to parse
411
- * @returns Array of header positions and levels
412
- */
413
- private static parseHeaderHierarchy(content: string): DyNTS_DAI_DocChunking_Header[] {
414
- const headers: DyNTS_DAI_DocChunking_Header[] = [];
415
- const lines: string[] = content.split('\n');
416
- let currentIndex: number = 0;
417
-
418
- for (const line of lines) {
419
- const trimmedLine: string = line.trim();
420
-
421
- // Check if line starts with # and determine the level
422
- if (trimmedLine.startsWith('#')) {
423
- const match: RegExpMatchArray | null = trimmedLine.match(/^(#+)\s+(.+)$/);
424
-
425
- if (match) {
426
- const level: number = match[1].length; // Count the number of # symbols
427
- const headerText: string = trimmedLine;
428
-
429
- if (level >= 1 && level <= 4) { // Only support levels 1-4
430
- headers.push({
431
- index: currentIndex,
432
- level: level,
433
- header: headerText
434
- });
435
- }
436
- }
437
- }
438
-
439
- currentIndex += line.length + 1; // +1 for the newline
440
- }
441
-
442
- return headers;
443
- }
444
-
445
- /**
446
- * Update the last headers map based on current position
447
- * @param headerHierarchy - The parsed header hierarchy
448
- * @param currentPosition - Current position in content
449
- * @param lastHeaders - Map to store last headers of each level
450
- */
451
- private static updateLastHeaders(
452
- headerHierarchy: DyNTS_DAI_DocChunking_Header[],
453
- currentPosition: number,
454
- lastHeaders: Map<number, string>
455
- ): void {
456
- // Find all headers that come before current position (not including current position)
457
- for (const header of headerHierarchy) {
458
- if (header.index < currentPosition) {
459
- // Set this header as the last header for its level
460
- lastHeaders.set(header.level, header.header);
461
-
462
- // Clear all lower level headers (higher level numbers) when any header comes
463
- // For example, if we see a level 2 header, clear levels 3, 4
464
- // If we see a level 1 header, clear levels 2, 3, 4
465
- for (let level: number = header.level + 1; level <= 4; level++) {
466
- lastHeaders.delete(level);
467
- }
468
- }
469
- }
470
- }
471
-
472
- /**
473
- * Visszaadja a chunk-hoz tartozó releváns szülő fejléceket a hierarchia alapján.
474
- * @param headerHierarchy - A feldolgozott fejléc-hierarchia
475
- * @param startIndex - A chunk kezdőindexe az eredeti tartalomban
476
- * @param endIndex - A chunk végindexe az eredeti tartalomban
477
- * @param lastHeaders - Az utolsó ismert fejlécek szintenként
478
- * @returns A releváns szülő fejlécek tömbje
479
- */
480
- private static getRelevantHeaders(
481
- headerHierarchy: DyNTS_DAI_DocChunking_Header[],
482
- startIndex: number,
483
- endIndex: number,
484
- lastHeaders: Map<number, string>
485
- ): string[] {
486
- // Keressük meg az első fejlécet a chunkban
487
- const firstHeaderInChunk: DyNTS_DAI_DocChunking_Header | undefined = headerHierarchy.find(header =>
488
- header.index >= startIndex && header.index < endIndex
489
- );
490
-
491
- if (!firstHeaderInChunk) {
492
- // Nincs fejléc a chunkban, a legmagasabb szintű utolsó fejlécet használjuk
493
- let highestLevel: number = 0;
494
- let highestHeader: string = '';
495
-
496
- for (const [level, header] of lastHeaders.entries()) {
497
- if (level > highestLevel) {
498
- highestLevel = level;
499
- highestHeader = header;
500
- }
501
- }
502
- return highestHeader ? [highestHeader] : [];
503
- }
504
-
505
- // Szülő fejlécek összegyűjtése az első fejléc szintje alapján
506
- const relevantHeaders: string[] = [];
507
- for (let level: number = 1; level < firstHeaderInChunk.level; level++) {
508
- const parentHeader: string | undefined = lastHeaders.get(level);
509
-
510
- if (parentHeader) {
511
- relevantHeaders.push(parentHeader);
512
- }
513
- }
514
- return relevantHeaders;
515
- }
516
-
517
- /**
518
- * Build hierarchical chunk path based on header structure
519
- * @param pageName - The page name
520
- * @param headerHierarchy - The parsed header hierarchy
521
- * @param startIndex - Start index of the chunk in original content
522
- * @param endIndex - End index of the chunk in original content
523
- * @param lastHeaders - Map of last headers for each level
524
- * @returns Hierarchical path array
525
- */
526
- private static buildChunkPath(
527
- pageName: string,
528
- headerHierarchy: DyNTS_DAI_DocChunking_Header[],
529
- startIndex: number,
530
- endIndex: number,
531
- lastHeaders: Map<number, string>
532
- ): string[] {
533
- // Find the first header within this chunk
534
- const firstHeaderInChunk: DyNTS_DAI_DocChunking_Header | undefined = headerHierarchy.find(header =>
535
- header.index >= startIndex && header.index < endIndex
536
- );
537
-
538
- if (!firstHeaderInChunk) {
539
- // No header in chunk, use the highest level from lastHeaders as parent
540
- const pathParts: string[] = [pageName];
541
-
542
- // Find the highest level header from lastHeaders
543
- let highestLevel: number = 0;
544
- let highestHeader: string = '';
545
-
546
- for (const [level, header] of lastHeaders.entries()) {
547
- if (level > highestLevel) {
548
- highestLevel = level;
549
- highestHeader = header;
550
- }
551
- }
552
-
553
- if (highestHeader) {
554
- pathParts.push(highestHeader);
555
- }
556
-
557
- return pathParts;
558
- }
559
-
560
- // Build the path hierarchy
561
- const pathParts: string[] = [pageName];
562
-
563
- // Add parent headers from lastHeaders (these are the actual parent headers)
564
- for (let level = 1; level < firstHeaderInChunk.level; level++) {
565
- const parentHeader: string | undefined = lastHeaders.get(level);
566
-
567
- if (parentHeader) {
568
- pathParts.push(parentHeader); // Include the full header with markdown symbols
569
- }
570
- }
571
-
572
- // Add the first header in chunk
573
- pathParts.push(firstHeaderInChunk.header); // Include the full header with markdown symbols
574
-
575
- return pathParts;
576
- }
577
-
578
- /**
579
- * Add header hierarchy to the beginning of a chunk
580
- * @param chunkContent - The chunk content
581
- * @param headerHierarchy - The parsed header hierarchy
582
- * @param startIndex - Start index of the chunk in original content
583
- * @param endIndex - End index of the chunk in original content
584
- * @param lastHeaders - Map of last headers for each level
585
- * @returns Content with header hierarchy added
586
- */
587
- private static addHeaderHierarchy(
588
- chunkContent: string,
589
- headerHierarchy: DyNTS_DAI_DocChunking_Header[],
590
- startIndex: number,
591
- endIndex: number,
592
- lastHeaders: Map<number, string>
593
- ): string {
594
- const relevantHeaders: string[] = this.getRelevantHeaders(headerHierarchy, startIndex, endIndex, lastHeaders);
595
-
596
- if (relevantHeaders.length > 0) {
597
- return relevantHeaders.join('\n') + '\n\n' + chunkContent;
598
- }
599
-
600
- return chunkContent;
601
- }
602
-
603
- /**
604
- * Helper method to find the last occurrence of a pattern within a specified range
605
- * @param text - The text to search in
606
- * @param pattern - The pattern to search for
607
- * @param maxIndex - Maximum index to search up to
608
- * @returns The index of the last occurrence, or -1 if not found
609
- */
610
- private static findLastOccurrence(text: string, pattern: string, maxIndex: number): number {
611
- const searchText: string = text.substring(0, maxIndex);
612
- const lastIndex: number = searchText.lastIndexOf(pattern);
613
-
614
- return lastIndex;
615
- }
616
-
617
- protected static getDefaultErrorSettings(
618
- fnName: string,
619
- error: DyFM_AnyError,
620
- issuer: string,
621
- /** @deprecated we wont support the separate user message in the future, use the message instead */
622
- useMessageAsUserMessage: boolean = true,
623
- /* errorCode: string, */
624
- ): DyFM_Error_Settings {
625
- return {
626
- status: (error as DyFM_Error)?.___status ?? 500,
627
- message: (error as Error)?.message ??
628
- (error as DyFM_Error)?._message ??
629
- `${fnName} was UNSUCCESSFUL (${DyNTS_global_settings.systemShortCodeName})`,
630
- addECToUserMsg: !(error as DyFM_Error)?.__userMessage,
631
- userMessage: (error as DyFM_Error)?.__userMessage ??
632
- (useMessageAsUserMessage ? (error as Error)?.message : this.defaultErrorUserMsg),
633
- /* errorCode: errorCode, */
634
- issuer: issuer,
635
- issuerService: this.constructor?.name,
636
- error: error,
637
- };
638
- }
639
- }
1
+ import { DyFM_Log, DyFM_Error, DyFM_AnyError, DyFM_Error_Settings } from '@futdevpro/fsm-dynamo';
2
+ import { DyNTS_global_settings } from '../../../../../_collections/global-settings.const';
3
+ import { DyNTS_DAI_DocChunk } from '../_models/data-models/dai-doc-chunk.data-model';
4
+ import { DyNTS_DAI_DocPage } from '../_models/data-models/dai-doc-page.data-model';
5
+
6
+
7
+ interface DyNTS_DAI_DocChunking_Header {
8
+ level: number;
9
+ header: string;
10
+ index: number;
11
+ }
12
+
13
+ interface DyNTS_DAI_DocChunking_BreakPoint {
14
+ index: number;
15
+ level: number;
16
+ }
17
+
18
+ /**
19
+ * ClickUp dokumentum chunking szolgáltatás
20
+ * @author AI
21
+ * @description ClickUp dokumentumok intelligens chunking-ja a leghosszabb lehetséges chunk-ok létrehozásához
22
+ */
23
+ export class DyNTS_DAI_DocChunking_Util {
24
+
25
+ protected static readonly defaultErrorUserMsg: string =
26
+ `We encountered an unhandled Control Service Error, ` +
27
+ `\nplease contact the responsible development team.`;
28
+
29
+ /**
30
+ * Unified break points configuration
31
+ */
32
+ private static readonly breakPoints: { pattern: string, level: number, includeLength: number }[] = [
33
+ { pattern: '\n# ', level: 1, includeLength: 0 }, // Header 1 - cut before marker
34
+ { pattern: '\n## ', level: 2, includeLength: 0 }, // Header 2 - cut before marker
35
+ { pattern: '\n### ', level: 3, includeLength: 0 }, // Header 3 - cut before marker
36
+ { pattern: '\n#### ', level: 4, includeLength: 0 }, // Header 4 - cut before marker
37
+ { pattern: '\n\n', level: 0, includeLength: 2 }, // Double line break - include
38
+ { pattern: '\n', level: 0, includeLength: 1 }, // Single line break - include
39
+ { pattern: '. ', level: 0, includeLength: 2 }, // Sentence end - include
40
+ { pattern: ' ', level: 0, includeLength: 1 }, // Word break - include
41
+ ];
42
+
43
+ protected static readonly debugLog: boolean = false;
44
+
45
+ /**
46
+ * Markdown tartalom chunking-ja a leghosszabb lehetséges chunk-ok létrehozásához
47
+ * @param page - ClickUp dokumentum oldal
48
+ * @param issuer - Kérés kezdeményezője
49
+ * @returns Promise<CU_DocChunk[]> - Létrehozott chunk-ok listája
50
+ */
51
+ static async chunkMdContent<
52
+ T_Chunk extends DyNTS_DAI_DocChunk,
53
+ T_Page extends DyNTS_DAI_DocPage<T_Chunk>,
54
+ >(
55
+ page: T_Page,
56
+ getPageLink: (page: T_Page, issuer: string) => string,
57
+ issuer: string,
58
+ debugLog?: boolean
59
+ ): Promise<T_Chunk[]> {
60
+ try {
61
+ if (debugLog) {
62
+ DyFM_Log.info(`Chunking page "${page.name}"...`);
63
+ }
64
+
65
+ if (page.content.includes('#####')) {
66
+ DyFM_Log.warn('lvl5+ headers are not supported yet for page: ' + page.name);
67
+ }
68
+
69
+ if (!page.documentId) {
70
+ throw new DyFM_Error({
71
+ ...this.getDefaultErrorSettings('chunkMdContent', new Error('Page documentId is not set'), issuer),
72
+ errorCode: 'CCAP-CUC-CC2',
73
+ });
74
+ }
75
+
76
+ if (!page._id) {
77
+ throw new DyFM_Error({
78
+ ...this.getDefaultErrorSettings('chunkMdContent', new Error('Page _id is not set'), issuer),
79
+ errorCode: 'CCAP-CUC-CC3',
80
+ });
81
+ }
82
+
83
+ /* DyFM_Log.log(`Chunking page: ${page.name}\n\n`, JSON.stringify(page.content, null, 2)); */
84
+
85
+ let leftovers: string = page.content;
86
+ const maxChunkSize: number = DyNTS_global_settings.docChunking.maxChunkSize;
87
+ const maxChunkCount: number = DyNTS_global_settings.docChunking.maxChunkCount;
88
+ const chunks: T_Chunk[] = [];
89
+ let chunkIndex: number = 0;
90
+ let currentPosition: number = 0; // Track current position in original content
91
+
92
+ // Parse all headers to build hierarchy
93
+ const headerHierarchy: DyNTS_DAI_DocChunking_Header[] = this.parseHeaderHierarchy(leftovers);
94
+
95
+ // Debug: Log parsed headers
96
+ /* console.log('\n=== Parsed Header Hierarchy ===');
97
+ headerHierarchy.forEach((header, index) => {
98
+ console.log(`${index}: Level ${header.level} at position ${header.index}: "${header.header}"`);
99
+ });
100
+ console.log('=== End Header Hierarchy ===\n'); */
101
+
102
+ // Track the last headers of each level as we progress
103
+ const lastHeaders: Map<number, string> = new Map();
104
+
105
+ while (leftovers.length > 0 && chunkIndex < maxChunkCount) {
106
+ // Ha a teljes tartalom belefér egy chunk-ba, adjuk vissza az egészet
107
+ if (leftovers.length <= maxChunkSize) {
108
+ const chunkContent = leftovers.trim();
109
+
110
+ if (chunkContent.length > 0) {
111
+ //const chunkWithHierarchy: string = this.addHeaderHierarchy(chunkContent, headerHierarchy, currentPosition, currentPosition + leftovers.length, lastHeaders);
112
+ //const chunkPath: string[] = this.buildChunkPath(page.name, headerHierarchy, currentPosition, currentPosition + leftovers.length, lastHeaders);
113
+ const headParents: string[] = this.getRelevantHeaders(headerHierarchy, currentPosition, currentPosition + leftovers.length, lastHeaders); // headerHierarchy.map(header => header.header);
114
+ //const allFlaggedParents: string[] = [ ...page.allFlaggedParents, ...headParents ];
115
+
116
+ const chunk: T_Chunk = new DyNTS_DAI_DocChunk({
117
+ externalDocumentId: page.externalDocumentId,
118
+ documentId: page.documentId,
119
+ externalPageId: page.externalPageId,
120
+ pageId: page._id,
121
+
122
+ filePathParents: page.parentedPath,
123
+ chunkHeadParents: headParents,
124
+
125
+ allFlaggedParents: [ ...page.allFlaggedParents, ...headParents ],
126
+ /* path: [ ...page.parentedPath, ...headParents ].join('/'), */
127
+ path: page.allFlaggedParentsMerged,
128
+
129
+ documentName: page.externalDocumentId,
130
+ pageName: page.name,
131
+ pageLink: getPageLink(page, issuer),
132
+ chunkIndex: chunkIndex,
133
+
134
+ chunkContent: chunkContent,
135
+ /* chunkOriginalContent: chunkContent,
136
+
137
+ chunkContent: chunkWithHierarchy, */
138
+ }) as T_Chunk;
139
+
140
+ if (this.debugLog || debugLog) {
141
+ DyFM_Log.H_info(
142
+ 'Full content (before assembleChunkWithHeaders)',
143
+ `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
144
+ { chunk: chunk.chunkContent }
145
+ );
146
+ }
147
+
148
+ /* chunk.chunkContent = this.assembleChunkWithHeaders(chunk, issuer); */
149
+ if (this.debugLog || debugLog) {
150
+ DyFM_Log.H_info(
151
+ 'Full content (after assembleChunkWithHeaders)',
152
+ `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
153
+ {
154
+ chunk: chunk.chunkContent,
155
+ }
156
+ );
157
+ }
158
+
159
+ /* await DyFM_Async.wait(100_000); */
160
+ chunks.push(chunk);
161
+ }
162
+ break;
163
+ }
164
+
165
+ // Keressük meg a legjobb break point-ot a maxChunkSize-n belül
166
+ const breakPoint: DyNTS_DAI_DocChunking_BreakPoint | null = this.findBestBreakPoint(leftovers, maxChunkSize);
167
+
168
+ if (breakPoint && breakPoint.index > 0) {
169
+ // Ha találtunk break point-ot, vágjunk a marker előtt
170
+ const chunkContent: string = leftovers.substring(0, breakPoint.index).trim();
171
+
172
+ // Debug: Log current state
173
+ /* console.log(`\n--- Creating chunk ${chunkIndex} ---`);
174
+ console.log(`Current position: ${currentPosition}`);
175
+ console.log(`Last headers:`, Object.fromEntries(lastHeaders));
176
+ console.log(`Chunk content preview:`, chunkContent.substring(0, 100) + '...'); */
177
+
178
+ // Check if chunk has headers
179
+ //const hasHeaders: boolean = headerHierarchy.some(header =>
180
+ // header.index >= currentPosition && header.index < currentPosition + breakPoint.index
181
+ //);
182
+ /* console.log(`Chunk has headers: ${hasHeaders}`); */
183
+
184
+ //const chunkWithHierarchy: string = this.addHeaderHierarchy(chunkContent, headerHierarchy, currentPosition, currentPosition + breakPoint.index, lastHeaders);
185
+ //const chunkPath: string[] = this.buildChunkPath(page.name, headerHierarchy, currentPosition, currentPosition + breakPoint.index, lastHeaders);
186
+ const headParents: string[] = this.getRelevantHeaders(headerHierarchy, currentPosition, currentPosition + breakPoint.index, lastHeaders); // headerHierarchy.map(header => header.header);
187
+ //const allFlaggedParents: string[] = [ ...page.allFlaggedParents, ...headParents ];
188
+
189
+ /* console.log(`Chunk path:`, chunkPath);
190
+ console.log(`Chunk with hierarchy preview:`, chunkWithHierarchy.substring(0, 100) + '...'); */
191
+
192
+ const chunk: T_Chunk = new DyNTS_DAI_DocChunk({
193
+ externalDocumentId: page.externalDocumentId,
194
+ externalPageId: page.externalPageId,
195
+ documentId: page.documentId,
196
+ pageId: page._id,
197
+
198
+ filePathParents: page.parentedPath,
199
+ chunkHeadParents: headParents,
200
+
201
+ allFlaggedParents: [ ...page.allFlaggedParents, ...headParents ],
202
+ path: [ ...page.parentedPath, ...headParents ].join('/'),
203
+
204
+ documentName: page.externalDocumentId,
205
+ pageName: page.name,
206
+ pageLink: getPageLink(page, issuer),
207
+ chunkIndex: chunkIndex,
208
+
209
+ chunkContent: chunkContent,
210
+ /* chunkOriginalContent: chunkContent,
211
+
212
+ chunkContent: chunkWithHierarchy, */
213
+ }) as T_Chunk;
214
+
215
+ if (this.debugLog || debugLog) {
216
+ DyFM_Log.H_info(
217
+ 'Break point (before assembleChunkWithHeaders)',
218
+ `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
219
+ { chunk: chunk.chunkContent }
220
+ );
221
+ }
222
+
223
+ /* chunk.chunkContent = this.assembleChunkWithHeaders(chunk, issuer); */
224
+ if (this.debugLog || debugLog) {
225
+ DyFM_Log.H_info(
226
+ 'Break point (after assembleChunkWithHeaders)',
227
+ `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
228
+ {
229
+ chunk: chunk.chunkContent,
230
+ }
231
+ );
232
+ }
233
+
234
+ chunks.push(chunk);
235
+ chunkIndex++;
236
+ currentPosition += breakPoint.index;
237
+ leftovers = leftovers.substring(breakPoint.index);
238
+
239
+ // Update last headers after creating the chunk
240
+ this.updateLastHeaders(headerHierarchy, currentPosition, lastHeaders);
241
+
242
+ /* console.log(`Updated last headers:`, Object.fromEntries(lastHeaders));
243
+ console.log(`--- End chunk ${chunkIndex - 1} ---\n`); */
244
+ } else {
245
+ // Ha nem találtunk break point-ot, használjuk a maxChunkSize-t
246
+ const chunkContent: string = leftovers.substring(0, maxChunkSize).trim();
247
+ //const chunkWithHierarchy: string = this.addHeaderHierarchy(chunkContent, headerHierarchy, currentPosition, currentPosition + maxChunkSize, lastHeaders);
248
+ //const chunkPath: string[] = this.buildChunkPath(page.name, headerHierarchy, currentPosition, currentPosition + maxChunkSize, lastHeaders);
249
+ const headParents: string[] = this.getRelevantHeaders(headerHierarchy, currentPosition, currentPosition + maxChunkSize, lastHeaders); // headerHierarchy.map(header => header.header);
250
+ //const allFlaggedParents: string[] = [ ...page.allFlaggedParents, ...headParents ];
251
+
252
+ const chunk: T_Chunk = new DyNTS_DAI_DocChunk({
253
+ externalDocumentId: page.externalDocumentId,
254
+ externalPageId: page.externalPageId,
255
+ documentId: page.documentId,
256
+ pageId: page._id,
257
+
258
+ filePathParents: page.parentedPath,
259
+ chunkHeadParents: headParents,
260
+
261
+ allFlaggedParents: [ ...page.allFlaggedParents, ...headParents ],
262
+ path: [ ...page.parentedPath, ...headParents ].join('/'),
263
+
264
+ documentName: page.externalDocumentId,
265
+ pageName: page.name,
266
+ pageLink: getPageLink(page, issuer),
267
+ chunkIndex: chunkIndex,
268
+
269
+ chunkContent: chunkContent,
270
+ /* chunkOriginalContent: chunkContent,
271
+
272
+ chunkContent: chunkWithHierarchy, */
273
+ }) as T_Chunk;
274
+
275
+ if (this.debugLog || debugLog) {
276
+ DyFM_Log.H_info(
277
+ 'Max chunk size',
278
+ `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
279
+ { chunk: chunk.chunkContent }
280
+ );
281
+ }
282
+
283
+ /* chunk.chunkContent = this.assembleChunkWithHeaders(chunk, issuer); */
284
+ if (this.debugLog || debugLog) {
285
+ DyFM_Log.H_info(
286
+ `Chunk ${chunkIndex}: ${chunk.path.replaceAll('/', '/\n').split('/')}`,
287
+ {
288
+ chunk: chunk.chunkContent,
289
+ }
290
+ );
291
+ }
292
+
293
+ chunks.push(chunk);
294
+ chunkIndex++;
295
+ currentPosition += maxChunkSize;
296
+ leftovers = leftovers.substring(maxChunkSize);
297
+
298
+ // Update last headers after creating the chunk
299
+ this.updateLastHeaders(headerHierarchy, currentPosition, lastHeaders);
300
+ }
301
+
302
+ // Safety check to prevent infinite loops
303
+ if (leftovers.length > 0 && leftovers.length >= page.content.length) {
304
+ throw new DyFM_Error({
305
+ ...this.getDefaultErrorSettings(
306
+ 'chunkMdContent',
307
+ new Error('Infinite loop detected in chunking'),
308
+ issuer
309
+ ),
310
+ errorCode: 'CCAP-CUC-CC1',
311
+ });
312
+ }
313
+ }
314
+
315
+ // Log warning if we hit the chunk count limit
316
+ if (chunkIndex >= maxChunkCount && leftovers.length > 0) {
317
+ DyFM_Log.warn(
318
+ `⚠️ Reached maximum chunk count (${maxChunkCount}). Remaining content: ${leftovers.length} characters`,
319
+ {
320
+ pageName: page.name,
321
+ pageSize: page.content.length,
322
+ }
323
+ );
324
+ }
325
+
326
+ /* DyFM_Log.info(`✅ Created ${chunks.length} chunks`); */
327
+ return chunks;
328
+ } catch (error) {
329
+ throw new DyFM_Error({
330
+ ...this.getDefaultErrorSettings('chunkMdContent', error, issuer),
331
+ errorCode: 'CCAP-CUC-CP0',
332
+ });
333
+ }
334
+ }
335
+
336
+ /* private static assembleChunkWithHeaders(chunk: DyNTS_DAI_DocChunk, issuer: string): string {
337
+ try {
338
+ const fileHeaders = chunk.filePathParents.map(
339
+ (header, index) => `${'>'.repeat(index + 1)} **${header}**`
340
+ ).join('\n');
341
+ const headers = chunk.chunkHeadParents.filter(
342
+ path => !chunk.chunkOriginalContent.includes(path)
343
+ ).join('\n');
344
+ return `${fileHeaders}\n\n${headers}\n\n${chunk.chunkOriginalContent}`;
345
+ } catch (error) {
346
+ throw new DyFM_Error({
347
+ ...this.getDefaultErrorSettings('assembleChunkWithHeaders', error, issuer),
348
+ errorCode: 'CCAP-CUC-ACH0',
349
+ });
350
+ }
351
+ } */
352
+
353
+ static reassembleChunksToContent(
354
+ chunks: DyNTS_DAI_DocChunk[],
355
+ issuer: string
356
+ ): string {
357
+ try {
358
+ let pageContent: string = '';
359
+
360
+ for (const chunk of chunks) {
361
+ if (!chunk) {
362
+ throw new DyFM_Error({
363
+ ...this.getDefaultErrorSettings(
364
+ 'reassembleChunksToContent',
365
+ new Error('Chunk is null, you`ve passing null chunks to the function'),
366
+ issuer
367
+ ),
368
+ errorCode: 'CCAP-CUC-RCTC0',
369
+
370
+ additionalContent: {
371
+ chunks: chunks,
372
+ },
373
+ });
374
+ }
375
+
376
+ pageContent += chunk.chunkContent;
377
+ }
378
+
379
+ return pageContent;
380
+ } catch (error) {
381
+ throw new DyFM_Error({
382
+ ...this.getDefaultErrorSettings('reassembleChunks', error, issuer),
383
+ errorCode: 'CCAP-CUC-RC0',
384
+ });
385
+ }
386
+ }
387
+
388
+ /**
389
+ * Find the best break point within the chunk size
390
+ * @param text - The text to search in
391
+ * @param maxIndex - Maximum index to search up to
392
+ * @returns The best break point or null if not found
393
+ */
394
+ private static findBestBreakPoint(text: string, maxIndex: number): DyNTS_DAI_DocChunking_BreakPoint | null {
395
+ for (const breakPoint of this.breakPoints) {
396
+ const index: number = this.findLastOccurrence(text, breakPoint.pattern, maxIndex);
397
+
398
+ if (index > 0) {
399
+ return {
400
+ index: index + breakPoint.includeLength,
401
+ level: breakPoint.level
402
+ };
403
+ }
404
+ }
405
+ return null;
406
+ }
407
+
408
+ /**
409
+ * Parse header hierarchy from the content
410
+ * @param content - The content to parse
411
+ * @returns Array of header positions and levels
412
+ */
413
+ private static parseHeaderHierarchy(content: string): DyNTS_DAI_DocChunking_Header[] {
414
+ const headers: DyNTS_DAI_DocChunking_Header[] = [];
415
+ const lines: string[] = content.split('\n');
416
+ let currentIndex: number = 0;
417
+
418
+ for (const line of lines) {
419
+ const trimmedLine: string = line.trim();
420
+
421
+ // Check if line starts with # and determine the level
422
+ if (trimmedLine.startsWith('#')) {
423
+ const match: RegExpMatchArray | null = trimmedLine.match(/^(#+)\s+(.+)$/);
424
+
425
+ if (match) {
426
+ const level: number = match[1].length; // Count the number of # symbols
427
+ const headerText: string = trimmedLine;
428
+
429
+ if (level >= 1 && level <= 4) { // Only support levels 1-4
430
+ headers.push({
431
+ index: currentIndex,
432
+ level: level,
433
+ header: headerText
434
+ });
435
+ }
436
+ }
437
+ }
438
+
439
+ currentIndex += line.length + 1; // +1 for the newline
440
+ }
441
+
442
+ return headers;
443
+ }
444
+
445
+ /**
446
+ * Update the last headers map based on current position
447
+ * @param headerHierarchy - The parsed header hierarchy
448
+ * @param currentPosition - Current position in content
449
+ * @param lastHeaders - Map to store last headers of each level
450
+ */
451
+ private static updateLastHeaders(
452
+ headerHierarchy: DyNTS_DAI_DocChunking_Header[],
453
+ currentPosition: number,
454
+ lastHeaders: Map<number, string>
455
+ ): void {
456
+ // Find all headers that come before current position (not including current position)
457
+ for (const header of headerHierarchy) {
458
+ if (header.index < currentPosition) {
459
+ // Set this header as the last header for its level
460
+ lastHeaders.set(header.level, header.header);
461
+
462
+ // Clear all lower level headers (higher level numbers) when any header comes
463
+ // For example, if we see a level 2 header, clear levels 3, 4
464
+ // If we see a level 1 header, clear levels 2, 3, 4
465
+ for (let level: number = header.level + 1; level <= 4; level++) {
466
+ lastHeaders.delete(level);
467
+ }
468
+ }
469
+ }
470
+ }
471
+
472
+ /**
473
+ * Visszaadja a chunk-hoz tartozó releváns szülő fejléceket a hierarchia alapján.
474
+ * @param headerHierarchy - A feldolgozott fejléc-hierarchia
475
+ * @param startIndex - A chunk kezdőindexe az eredeti tartalomban
476
+ * @param endIndex - A chunk végindexe az eredeti tartalomban
477
+ * @param lastHeaders - Az utolsó ismert fejlécek szintenként
478
+ * @returns A releváns szülő fejlécek tömbje
479
+ */
480
+ private static getRelevantHeaders(
481
+ headerHierarchy: DyNTS_DAI_DocChunking_Header[],
482
+ startIndex: number,
483
+ endIndex: number,
484
+ lastHeaders: Map<number, string>
485
+ ): string[] {
486
+ // Keressük meg az első fejlécet a chunkban
487
+ const firstHeaderInChunk: DyNTS_DAI_DocChunking_Header | undefined = headerHierarchy.find(header =>
488
+ header.index >= startIndex && header.index < endIndex
489
+ );
490
+
491
+ if (!firstHeaderInChunk) {
492
+ // Nincs fejléc a chunkban, a legmagasabb szintű utolsó fejlécet használjuk
493
+ let highestLevel: number = 0;
494
+ let highestHeader: string = '';
495
+
496
+ for (const [level, header] of lastHeaders.entries()) {
497
+ if (level > highestLevel) {
498
+ highestLevel = level;
499
+ highestHeader = header;
500
+ }
501
+ }
502
+ return highestHeader ? [highestHeader] : [];
503
+ }
504
+
505
+ // Szülő fejlécek összegyűjtése az első fejléc szintje alapján
506
+ const relevantHeaders: string[] = [];
507
+ for (let level: number = 1; level < firstHeaderInChunk.level; level++) {
508
+ const parentHeader: string | undefined = lastHeaders.get(level);
509
+
510
+ if (parentHeader) {
511
+ relevantHeaders.push(parentHeader);
512
+ }
513
+ }
514
+ return relevantHeaders;
515
+ }
516
+
517
+ /**
518
+ * Build hierarchical chunk path based on header structure
519
+ * @param pageName - The page name
520
+ * @param headerHierarchy - The parsed header hierarchy
521
+ * @param startIndex - Start index of the chunk in original content
522
+ * @param endIndex - End index of the chunk in original content
523
+ * @param lastHeaders - Map of last headers for each level
524
+ * @returns Hierarchical path array
525
+ */
526
+ private static buildChunkPath(
527
+ pageName: string,
528
+ headerHierarchy: DyNTS_DAI_DocChunking_Header[],
529
+ startIndex: number,
530
+ endIndex: number,
531
+ lastHeaders: Map<number, string>
532
+ ): string[] {
533
+ // Find the first header within this chunk
534
+ const firstHeaderInChunk: DyNTS_DAI_DocChunking_Header | undefined = headerHierarchy.find(header =>
535
+ header.index >= startIndex && header.index < endIndex
536
+ );
537
+
538
+ if (!firstHeaderInChunk) {
539
+ // No header in chunk, use the highest level from lastHeaders as parent
540
+ const pathParts: string[] = [pageName];
541
+
542
+ // Find the highest level header from lastHeaders
543
+ let highestLevel: number = 0;
544
+ let highestHeader: string = '';
545
+
546
+ for (const [level, header] of lastHeaders.entries()) {
547
+ if (level > highestLevel) {
548
+ highestLevel = level;
549
+ highestHeader = header;
550
+ }
551
+ }
552
+
553
+ if (highestHeader) {
554
+ pathParts.push(highestHeader);
555
+ }
556
+
557
+ return pathParts;
558
+ }
559
+
560
+ // Build the path hierarchy
561
+ const pathParts: string[] = [pageName];
562
+
563
+ // Add parent headers from lastHeaders (these are the actual parent headers)
564
+ for (let level = 1; level < firstHeaderInChunk.level; level++) {
565
+ const parentHeader: string | undefined = lastHeaders.get(level);
566
+
567
+ if (parentHeader) {
568
+ pathParts.push(parentHeader); // Include the full header with markdown symbols
569
+ }
570
+ }
571
+
572
+ // Add the first header in chunk
573
+ pathParts.push(firstHeaderInChunk.header); // Include the full header with markdown symbols
574
+
575
+ return pathParts;
576
+ }
577
+
578
+ /**
579
+ * Add header hierarchy to the beginning of a chunk
580
+ * @param chunkContent - The chunk content
581
+ * @param headerHierarchy - The parsed header hierarchy
582
+ * @param startIndex - Start index of the chunk in original content
583
+ * @param endIndex - End index of the chunk in original content
584
+ * @param lastHeaders - Map of last headers for each level
585
+ * @returns Content with header hierarchy added
586
+ */
587
+ private static addHeaderHierarchy(
588
+ chunkContent: string,
589
+ headerHierarchy: DyNTS_DAI_DocChunking_Header[],
590
+ startIndex: number,
591
+ endIndex: number,
592
+ lastHeaders: Map<number, string>
593
+ ): string {
594
+ const relevantHeaders: string[] = this.getRelevantHeaders(headerHierarchy, startIndex, endIndex, lastHeaders);
595
+
596
+ if (relevantHeaders.length > 0) {
597
+ return relevantHeaders.join('\n') + '\n\n' + chunkContent;
598
+ }
599
+
600
+ return chunkContent;
601
+ }
602
+
603
+ /**
604
+ * Helper method to find the last occurrence of a pattern within a specified range
605
+ * @param text - The text to search in
606
+ * @param pattern - The pattern to search for
607
+ * @param maxIndex - Maximum index to search up to
608
+ * @returns The index of the last occurrence, or -1 if not found
609
+ */
610
+ private static findLastOccurrence(text: string, pattern: string, maxIndex: number): number {
611
+ const searchText: string = text.substring(0, maxIndex);
612
+ const lastIndex: number = searchText.lastIndexOf(pattern);
613
+
614
+ return lastIndex;
615
+ }
616
+
617
+ protected static getDefaultErrorSettings(
618
+ fnName: string,
619
+ error: DyFM_AnyError,
620
+ issuer: string,
621
+ /** @deprecated we wont support the separate user message in the future, use the message instead */
622
+ useMessageAsUserMessage: boolean = true,
623
+ /* errorCode: string, */
624
+ ): DyFM_Error_Settings {
625
+ return {
626
+ status: (error as DyFM_Error)?.___status ?? 500,
627
+ message: (error as Error)?.message ??
628
+ (error as DyFM_Error)?._message ??
629
+ `${fnName} was UNSUCCESSFUL (${DyNTS_global_settings.systemShortCodeName})`,
630
+ addECToUserMsg: !(error as DyFM_Error)?.__userMessage,
631
+ userMessage: (error as DyFM_Error)?.__userMessage ??
632
+ (useMessageAsUserMessage ? (error as Error)?.message : this.defaultErrorUserMsg),
633
+ /* errorCode: errorCode, */
634
+ issuer: issuer,
635
+ issuerService: this.constructor?.name,
636
+ error: error,
637
+ };
638
+ }
639
+ }