rez_core 5.0.211 → 6.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (458) hide show
  1. package/.prettierrc +3 -3
  2. package/README.md +99 -99
  3. package/dist/config/database.config.js +1 -1
  4. package/dist/config/database.config.js.map +1 -1
  5. package/dist/migrations/1732612800000-AddEntityJsonGinIndex.d.ts +6 -0
  6. package/dist/migrations/1732612800000-AddEntityJsonGinIndex.js +32 -0
  7. package/dist/migrations/1732612800000-AddEntityJsonGinIndex.js.map +1 -0
  8. package/dist/module/auth/guards/role.guard.js +3 -3
  9. package/dist/module/entity_json/controller/entity_json.controller.d.ts +2 -9
  10. package/dist/module/entity_json/controller/entity_json.controller.js.map +1 -1
  11. package/dist/module/entity_json/entity/entityJson.entity.d.ts +2 -1
  12. package/dist/module/entity_json/entity/entityJson.entity.js +5 -1
  13. package/dist/module/entity_json/entity/entityJson.entity.js.map +1 -1
  14. package/dist/module/entity_json/entity_json.module.js +7 -2
  15. package/dist/module/entity_json/entity_json.module.js.map +1 -1
  16. package/dist/module/entity_json/service/entity_json.service.d.ts +5 -12
  17. package/dist/module/entity_json/service/entity_json.service.js +111 -29
  18. package/dist/module/entity_json/service/entity_json.service.js.map +1 -1
  19. package/dist/module/filter/controller/filter.controller.d.ts +12 -0
  20. package/dist/module/filter/controller/filter.controller.js +1 -1
  21. package/dist/module/filter/controller/filter.controller.js.map +1 -1
  22. package/dist/module/filter/filter.module.js +11 -2
  23. package/dist/module/filter/filter.module.js.map +1 -1
  24. package/dist/module/filter/service/filter.service.d.ts +38 -2
  25. package/dist/module/filter/service/filter.service.js +61 -68
  26. package/dist/module/filter/service/filter.service.js.map +1 -1
  27. package/dist/module/filter/service/flatjson-filter.service.d.ts +32 -0
  28. package/dist/module/filter/service/flatjson-filter.service.js +632 -0
  29. package/dist/module/filter/service/flatjson-filter.service.js.map +1 -0
  30. package/dist/module/filter/service/saved-filter.service.d.ts +3 -2
  31. package/dist/module/filter/service/saved-filter.service.js +14 -18
  32. package/dist/module/filter/service/saved-filter.service.js.map +1 -1
  33. package/dist/module/integration/examples/usage.example.js +9 -9
  34. package/dist/module/linked_attributes/controller/linked_attributes.controller.d.ts +19 -0
  35. package/dist/module/linked_attributes/controller/linked_attributes.controller.js +77 -0
  36. package/dist/module/linked_attributes/controller/linked_attributes.controller.js.map +1 -1
  37. package/dist/module/linked_attributes/dto/create-linked-attribute-smart.dto.d.ts +13 -0
  38. package/dist/module/linked_attributes/dto/create-linked-attribute-smart.dto.js +64 -0
  39. package/dist/module/linked_attributes/dto/create-linked-attribute-smart.dto.js.map +1 -0
  40. package/dist/module/linked_attributes/linked_attributes.module.js +4 -2
  41. package/dist/module/linked_attributes/linked_attributes.module.js.map +1 -1
  42. package/dist/module/linked_attributes/service/linked_attributes.service.d.ts +41 -1
  43. package/dist/module/linked_attributes/service/linked_attributes.service.js +265 -1
  44. package/dist/module/linked_attributes/service/linked_attributes.service.js.map +1 -1
  45. package/dist/module/meta/controller/attribute-master.controller.d.ts +3 -0
  46. package/dist/module/meta/controller/attribute-master.controller.js +12 -0
  47. package/dist/module/meta/controller/attribute-master.controller.js.map +1 -1
  48. package/dist/module/meta/entity.module.js +2 -2
  49. package/dist/module/meta/entity.module.js.map +1 -1
  50. package/dist/module/meta/repository/attribute-master.repository.js +8 -8
  51. package/dist/module/meta/service/attribute-master.service.d.ts +6 -1
  52. package/dist/module/meta/service/attribute-master.service.js +20 -2
  53. package/dist/module/meta/service/attribute-master.service.js.map +1 -1
  54. package/dist/module/meta/service/entity-dynamic.service.js +16 -16
  55. package/dist/module/meta/service/entity-master.service.js +1 -0
  56. package/dist/module/meta/service/entity-master.service.js.map +1 -1
  57. package/dist/module/meta/service/entity-relation.service.d.ts +4 -3
  58. package/dist/module/meta/service/entity-relation.service.js +10 -4
  59. package/dist/module/meta/service/entity-relation.service.js.map +1 -1
  60. package/dist/module/meta/service/entity-service-impl.service.d.ts +1 -1
  61. package/dist/module/meta/service/entity-service-impl.service.js +2 -2
  62. package/dist/module/meta/service/entity-service-impl.service.js.map +1 -1
  63. package/dist/module/meta/service/entity-table.service.d.ts +5 -4
  64. package/dist/module/meta/service/entity-table.service.js +45 -24
  65. package/dist/module/meta/service/entity-table.service.js.map +1 -1
  66. package/dist/module/meta/service/media-data.service.js +6 -6
  67. package/dist/module/meta/service/resolver.service.d.ts +1 -1
  68. package/dist/module/meta/service/resolver.service.js +11 -8
  69. package/dist/module/meta/service/resolver.service.js.map +1 -1
  70. package/dist/module/module/repository/menu.repository.js +4 -4
  71. package/dist/module/user/controller/login.controller.js +18 -18
  72. package/dist/module/workflow/repository/action.repository.js +2 -2
  73. package/dist/module/workflow/repository/stage.repository.js +8 -8
  74. package/dist/module/workflow/service/action-template-mapping.service.js +2 -2
  75. package/dist/module/workflow/service/action.service.js +5 -5
  76. package/dist/module/workflow/service/entity-modification.service.js +2 -2
  77. package/dist/module/workflow/service/task.service.js +8 -8
  78. package/dist/module/workflow-automation/service/schedule-handler.service.js +9 -9
  79. package/dist/module/workflow-automation/service/workflow-automation.service.js +2 -3
  80. package/dist/module/workflow-automation/service/workflow-automation.service.js.map +1 -1
  81. package/dist/table.config.d.ts +2 -1
  82. package/dist/table.config.js +2 -0
  83. package/dist/table.config.js.map +1 -1
  84. package/dist/tsconfig.build.tsbuildinfo +1 -1
  85. package/dist/utils/service/reflection-helper.service.js +2 -2
  86. package/docs/modules/event-driven-integration-design.md +91 -91
  87. package/docs/modules/integration.md +250 -250
  88. package/eslint.config.mjs +34 -34
  89. package/nest-cli.json +14 -14
  90. package/package.json +125 -125
  91. package/src/app.controller.ts +12 -12
  92. package/src/app.module.ts +68 -68
  93. package/src/app.service.ts +8 -8
  94. package/src/config/bull.config.ts +69 -69
  95. package/src/config/config.module.ts +17 -17
  96. package/src/config/database.config.ts +48 -48
  97. package/src/constant/global.constant.ts +67 -67
  98. package/src/core.module.ts +94 -94
  99. package/src/decorators/roles.decorator.ts +7 -7
  100. package/src/dtos/response.dto.ts +6 -6
  101. package/src/dtos/response.ts +5 -5
  102. package/src/index.ts +1 -1
  103. package/src/migrations/1732612800000-AddEntityJsonGinIndex.ts +41 -0
  104. package/src/module/auth/auth.module.ts +49 -49
  105. package/src/module/auth/controller/auth.controller.ts +28 -28
  106. package/src/module/auth/guards/google-auth.guard.ts +9 -9
  107. package/src/module/auth/guards/jwt.guard.ts +22 -22
  108. package/src/module/auth/guards/role.guard.ts +68 -68
  109. package/src/module/auth/services/auth.service.ts +56 -56
  110. package/src/module/auth/services/jwt.service.ts +11 -11
  111. package/src/module/auth/strategies/google.strategy.ts +54 -54
  112. package/src/module/auth/strategies/jwt.strategy.ts +58 -58
  113. package/src/module/auth/strategies/local.strategy.ts +13 -13
  114. package/src/module/dashboard/controller/dashboard.controller.ts +38 -38
  115. package/src/module/dashboard/dashboard.module.ts +21 -21
  116. package/src/module/dashboard/entity/dashboard_page_data.entity.ts +27 -27
  117. package/src/module/dashboard/entity/widget_master.entity.ts +18 -18
  118. package/src/module/dashboard/repository/dashboard.repository.ts +49 -49
  119. package/src/module/dashboard/service/dashboard.service.ts +72 -72
  120. package/src/module/enterprise/controller/organization.controller.ts +36 -36
  121. package/src/module/enterprise/enterprise.module.ts +45 -45
  122. package/src/module/enterprise/entity/enterprise.entity.ts +37 -37
  123. package/src/module/enterprise/entity/organization-app-mapping.entity.ts +13 -13
  124. package/src/module/enterprise/entity/organization.entity.ts +92 -92
  125. package/src/module/enterprise/repository/enterprise.repository.ts +31 -31
  126. package/src/module/enterprise/repository/organization.repository.ts +26 -26
  127. package/src/module/enterprise/repository/school.repository.ts +289 -289
  128. package/src/module/enterprise/service/brand.service.ts +5 -5
  129. package/src/module/enterprise/service/enterprise.service.ts +16 -16
  130. package/src/module/enterprise/service/organization-app-mapping.service.ts +4 -4
  131. package/src/module/enterprise/service/organization.service.ts +145 -145
  132. package/src/module/entity_json/controller/entity_json.controller.ts +60 -47
  133. package/src/module/entity_json/docs/FlatJson_Filterin_System.md +2804 -0
  134. package/src/module/entity_json/entity/entityJson.entity.ts +42 -39
  135. package/src/module/entity_json/entity_json.module.ts +22 -18
  136. package/src/module/entity_json/service/entityJson.repository.ts +37 -37
  137. package/src/module/entity_json/service/entity_json.service.ts +428 -241
  138. package/src/module/export/controller/export.controller.ts +83 -83
  139. package/src/module/export/export.module.ts +14 -14
  140. package/src/module/export/service/export.service.ts +105 -105
  141. package/src/module/filter/controller/filter.controller.ts +87 -87
  142. package/src/module/filter/dto/filter-request.dto.ts +39 -39
  143. package/src/module/filter/entity/saved-filter-detail.entity.ts +41 -41
  144. package/src/module/filter/entity/saved-filter-master.entity.ts +35 -35
  145. package/src/module/filter/filter.module.ts +42 -33
  146. package/src/module/filter/repository/saved-filter.repository.ts +247 -247
  147. package/src/module/filter/repository/saved.filter-detail.repository.ts +19 -19
  148. package/src/module/filter/service/filter-evaluator.service.ts +82 -82
  149. package/src/module/filter/service/filter.service.ts +1369 -1310
  150. package/src/module/filter/service/flatjson-filter.service.ts +903 -0
  151. package/src/module/filter/service/saved-filter.service.ts +154 -164
  152. package/src/module/filter/test/flatjson-filter.service.spec.ts +415 -0
  153. package/src/module/ics/controller/ics.controller.ts +21 -21
  154. package/src/module/ics/dto/ics.dto.ts +55 -55
  155. package/src/module/ics/ics.module.ts +13 -13
  156. package/src/module/ics/service/ics.service.ts +57 -57
  157. package/src/module/integration/controller/calender-event.controller.ts +31 -31
  158. package/src/module/integration/controller/integration.controller.ts +662 -662
  159. package/src/module/integration/controller/wrapper.controller.ts +37 -37
  160. package/src/module/integration/dto/create-config.dto.ts +526 -526
  161. package/src/module/integration/entity/integration-config.entity.ts +112 -112
  162. package/src/module/integration/entity/integration-entity-mapper.entity.ts +14 -14
  163. package/src/module/integration/entity/integration-source.entity.ts +17 -17
  164. package/src/module/integration/entity/user-integration.entity.ts +71 -71
  165. package/src/module/integration/examples/usage.example.ts +338 -338
  166. package/src/module/integration/factories/base.factory.ts +7 -7
  167. package/src/module/integration/factories/email.factory.ts +49 -49
  168. package/src/module/integration/factories/integration.factory.ts +121 -121
  169. package/src/module/integration/factories/sms.factory.ts +51 -51
  170. package/src/module/integration/factories/telephone.factory.ts +41 -41
  171. package/src/module/integration/factories/whatsapp.factory.ts +56 -56
  172. package/src/module/integration/integration.module.ts +110 -110
  173. package/src/module/integration/service/calendar-event.service.ts +118 -118
  174. package/src/module/integration/service/integration-entity-mapper.service.ts +17 -17
  175. package/src/module/integration/service/integration-queue.service.ts +229 -229
  176. package/src/module/integration/service/integration.service.ts +2639 -2639
  177. package/src/module/integration/service/oauth.service.ts +224 -224
  178. package/src/module/integration/service/wrapper.service.ts +754 -754
  179. package/src/module/integration/strategies/email/gmail-api.strategy.ts +280 -280
  180. package/src/module/integration/strategies/email/outlook-api.strategy.ts +44 -44
  181. package/src/module/integration/strategies/email/outlook.strategy.ts +64 -64
  182. package/src/module/integration/strategies/email/sendgrid-api.strategy.ts +260 -260
  183. package/src/module/integration/strategies/integration.strategy.ts +97 -97
  184. package/src/module/integration/strategies/sms/gupshup-sms.strategy.ts +146 -146
  185. package/src/module/integration/strategies/sms/msg91-sms.strategy.ts +164 -164
  186. package/src/module/integration/strategies/sms/tubelight-sms.strategy.ts +163 -163
  187. package/src/module/integration/strategies/telephone/ozonetel-voice.strategy.ts +238 -238
  188. package/src/module/integration/strategies/telephone/tubelight-voice.strategy.ts +210 -210
  189. package/src/module/integration/strategies/whatsapp/gupshup-whatsapp.strategy.ts +359 -359
  190. package/src/module/integration/strategies/whatsapp/tubelight-whatsapp.strategy.ts +372 -372
  191. package/src/module/integration/strategies/whatsapp/whatsapp-cloud.strategy.ts +403 -403
  192. package/src/module/integration/strategies/whatsapp/whatsapp.strategy.ts +57 -57
  193. package/src/module/layout/controller/layout.controller.ts +47 -47
  194. package/src/module/layout/entity/header-items.entity.ts +28 -28
  195. package/src/module/layout/entity/header-section.entity.ts +19 -19
  196. package/src/module/layout/layout.module.ts +21 -21
  197. package/src/module/layout/repository/header-items.repository.ts +18 -18
  198. package/src/module/layout/repository/header-section.repository.ts +22 -22
  199. package/src/module/layout/service/header-section.service.ts +25 -25
  200. package/src/module/layout_preference/controller/layout_preference.controller.ts +76 -76
  201. package/src/module/layout_preference/entity/layout_preference.entity.ts +28 -28
  202. package/src/module/layout_preference/layout_preference.module.ts +22 -22
  203. package/src/module/layout_preference/repository/layout_preference.repository.ts +65 -65
  204. package/src/module/layout_preference/service/layout_preference.service.ts +191 -191
  205. package/src/module/lead/controller/lead.controller.ts +30 -30
  206. package/src/module/lead/lead.module.ts +14 -14
  207. package/src/module/lead/repository/lead.repository.ts +41 -41
  208. package/src/module/lead/service/lead.service.ts +54 -54
  209. package/src/module/linked_attributes/controller/linked_attributes.controller.ts +137 -52
  210. package/src/module/linked_attributes/dto/create-linked-attribute-smart.dto.ts +54 -0
  211. package/src/module/linked_attributes/entity/linked_attribute.entity.ts +51 -51
  212. package/src/module/linked_attributes/linked_attributes.module.ts +23 -21
  213. package/src/module/linked_attributes/repository/linked_attribute.repository.ts +12 -12
  214. package/src/module/linked_attributes/service/linked_attributes.service.ts +648 -105
  215. package/src/module/linked_attributes/test/linked-attributes.service.spec.ts +244 -0
  216. package/src/module/listmaster/controller/list-master.controller.ts +230 -230
  217. package/src/module/listmaster/entity/list-master-items.entity.ts +43 -43
  218. package/src/module/listmaster/entity/list-master.entity.ts +33 -33
  219. package/src/module/listmaster/listmaster.module.ts +46 -46
  220. package/src/module/listmaster/repository/list-master-items.repository.ts +173 -173
  221. package/src/module/listmaster/repository/list-master.repository.ts +56 -56
  222. package/src/module/listmaster/service/list-master-engine.ts +19 -19
  223. package/src/module/listmaster/service/list-master-extension.interface.ts +4 -4
  224. package/src/module/listmaster/service/list-master-item.service.ts +280 -280
  225. package/src/module/listmaster/service/list-master-registry.ts +15 -15
  226. package/src/module/listmaster/service/list-master.service.ts +527 -527
  227. package/src/module/mapper/controller/field-mapper.controller.ts +76 -76
  228. package/src/module/mapper/controller/mapper.controller.ts +20 -20
  229. package/src/module/mapper/dto/field-mapper.dto.ts +14 -14
  230. package/src/module/mapper/entity/field-lovs.entity.ts +19 -19
  231. package/src/module/mapper/entity/field-mapper.entity.ts +53 -53
  232. package/src/module/mapper/entity/mapper.entity.ts +16 -16
  233. package/src/module/mapper/mapper.module.ts +35 -35
  234. package/src/module/mapper/repository/field-lovs.repository.ts +35 -35
  235. package/src/module/mapper/repository/field-mapper.repository.ts +42 -42
  236. package/src/module/mapper/repository/mapper.repository.ts +32 -32
  237. package/src/module/mapper/service/field-mapper.service.ts +269 -269
  238. package/src/module/mapper/service/mapper.service.ts +80 -80
  239. package/src/module/master/controller/master.controller.ts +74 -74
  240. package/src/module/master/service/master.service.ts +484 -484
  241. package/src/module/meta/controller/app-master.controller.ts +38 -38
  242. package/src/module/meta/controller/attribute-master.controller.ts +96 -84
  243. package/src/module/meta/controller/entity-dynamic.controller.ts +125 -125
  244. package/src/module/meta/controller/entity-master.controller.ts +41 -41
  245. package/src/module/meta/controller/entity-relation.controller.ts +36 -36
  246. package/src/module/meta/controller/entity.controller.ts +308 -308
  247. package/src/module/meta/controller/entity.public.controller.ts +75 -75
  248. package/src/module/meta/controller/media.controller.ts +135 -135
  249. package/src/module/meta/controller/meta.controller.ts +101 -101
  250. package/src/module/meta/controller/view-master.controller.ts +79 -79
  251. package/src/module/meta/dto/entity-list-data.dto.ts +6 -6
  252. package/src/module/meta/dto/entity-tab.dto.ts +4 -4
  253. package/src/module/meta/dto/entity-table.dto.ts +12 -12
  254. package/src/module/meta/entity/app-master.entity.ts +37 -37
  255. package/src/module/meta/entity/attribute-master.entity.ts +92 -92
  256. package/src/module/meta/entity/base-entity.entity.ts +75 -75
  257. package/src/module/meta/entity/entity-master.entity.ts +91 -91
  258. package/src/module/meta/entity/entity-relation-data.entity.ts +29 -29
  259. package/src/module/meta/entity/entity-relation.entity.ts +23 -23
  260. package/src/module/meta/entity/entity-table-column.entity.ts +61 -61
  261. package/src/module/meta/entity/entity-table.entity.ts +50 -50
  262. package/src/module/meta/entity/media-data.entity.ts +32 -32
  263. package/src/module/meta/entity/preference.entity.ts +62 -62
  264. package/src/module/meta/entity/view-master.entity.ts +41 -41
  265. package/src/module/meta/entity.module.ts +165 -165
  266. package/src/module/meta/repository/app-master.repository.ts +20 -20
  267. package/src/module/meta/repository/attribute-master.repository.ts +164 -164
  268. package/src/module/meta/repository/entity-attribute-update.repository.ts +48 -48
  269. package/src/module/meta/repository/entity-master.repository.ts +120 -120
  270. package/src/module/meta/repository/entity-relation.repository.ts +22 -22
  271. package/src/module/meta/repository/entity-table-column.repository.ts +39 -39
  272. package/src/module/meta/repository/entity-table.repository.ts +53 -53
  273. package/src/module/meta/repository/media-data.repository.ts +50 -50
  274. package/src/module/meta/repository/preference.repository.ts +20 -20
  275. package/src/module/meta/repository/user-app-mapping.repository.ts +28 -28
  276. package/src/module/meta/repository/view-master.repository.ts +42 -42
  277. package/src/module/meta/service/app-master.service.ts +37 -37
  278. package/src/module/meta/service/attribute-master.service.ts +160 -132
  279. package/src/module/meta/service/common.service.ts +9 -9
  280. package/src/module/meta/service/entity-attribute-update.service.ts +26 -26
  281. package/src/module/meta/service/entity-dynamic.service.ts +824 -824
  282. package/src/module/meta/service/entity-master.service.ts +172 -171
  283. package/src/module/meta/service/entity-realation-data.service.ts +9 -9
  284. package/src/module/meta/service/entity-relation.service.ts +78 -74
  285. package/src/module/meta/service/entity-service-impl.service.ts +389 -388
  286. package/src/module/meta/service/entity-table-column.service.ts +26 -26
  287. package/src/module/meta/service/entity-table.service.ts +171 -157
  288. package/src/module/meta/service/entity-validation.service.ts +188 -188
  289. package/src/module/meta/service/entity.service.ts +48 -49
  290. package/src/module/meta/service/field-group.service.ts +103 -103
  291. package/src/module/meta/service/media-data.service.ts +591 -591
  292. package/src/module/meta/service/populate-meta.service.ts +222 -222
  293. package/src/module/meta/service/preference.service.ts +16 -16
  294. package/src/module/meta/service/resolver.service.ts +319 -315
  295. package/src/module/meta/service/section-master.service.ts +104 -104
  296. package/src/module/meta/service/update-form-json.service.ts +22 -22
  297. package/src/module/meta/service/user-app-mapping.service.ts +17 -17
  298. package/src/module/meta/service/view-master.service.ts +127 -127
  299. package/src/module/microservice-client/microservice-clients.module.ts +13 -13
  300. package/src/module/microservice-client/service/microservice-client-factory.ts +37 -37
  301. package/src/module/microservice-client/service/microservice-clients.ts +4 -4
  302. package/src/module/module/controller/menu.controller.ts +15 -15
  303. package/src/module/module/controller/module-access.controller.ts +133 -133
  304. package/src/module/module/entity/menu.entity.ts +43 -43
  305. package/src/module/module/entity/module-access.entity.ts +25 -25
  306. package/src/module/module/entity/module-action.entity.ts +17 -17
  307. package/src/module/module/entity/module.entity.ts +52 -52
  308. package/src/module/module/module.module.ts +42 -42
  309. package/src/module/module/repository/menu.repository.ts +186 -186
  310. package/src/module/module/repository/module-access.repository.ts +344 -344
  311. package/src/module/module/service/menu.service.ts +82 -82
  312. package/src/module/module/service/module-access.service.ts +189 -189
  313. package/src/module/notification/controller/notification.controller.ts +58 -58
  314. package/src/module/notification/controller/otp.controller.ts +117 -117
  315. package/src/module/notification/entity/notification.entity.ts +26 -26
  316. package/src/module/notification/entity/otp.entity.ts +28 -28
  317. package/src/module/notification/firebase-admin.config.ts +22 -22
  318. package/src/module/notification/notification.module.ts +71 -71
  319. package/src/module/notification/repository/notification.repository.ts +33 -33
  320. package/src/module/notification/repository/otp.repository.ts +27 -27
  321. package/src/module/notification/service/email.service.ts +127 -127
  322. package/src/module/notification/service/notification.service.ts +146 -146
  323. package/src/module/notification/service/otp.service.ts +133 -133
  324. package/src/module/third-party-module/entity/third-party-api-registry.entity.ts +52 -52
  325. package/src/module/third-party-module/repository/third-party-api-registry.repository.ts +20 -20
  326. package/src/module/third-party-module/service/api-registry.service.ts +13 -13
  327. package/src/module/third-party-module/third-party.module.ts +12 -12
  328. package/src/module/user/controller/login.controller.ts +198 -198
  329. package/src/module/user/controller/user.controller.ts +40 -40
  330. package/src/module/user/dto/create-user.dto.ts +62 -62
  331. package/src/module/user/dto/update-user.dto.ts +4 -4
  332. package/src/module/user/entity/role.entity.ts +33 -33
  333. package/src/module/user/entity/user-role-mapping.entity.ts +38 -38
  334. package/src/module/user/entity/user-session.entity.ts +73 -73
  335. package/src/module/user/entity/user.entity.ts +62 -62
  336. package/src/module/user/repository/role.repository.ts +96 -96
  337. package/src/module/user/repository/user-role-mapping.repository.ts +126 -126
  338. package/src/module/user/repository/user.repository.ts +50 -50
  339. package/src/module/user/repository/userSession.repository.ts +33 -33
  340. package/src/module/user/service/login.service.ts +326 -326
  341. package/src/module/user/service/role.service.ts +197 -197
  342. package/src/module/user/service/user-role-mapping.service.ts +98 -98
  343. package/src/module/user/service/user-session.service.ts +200 -200
  344. package/src/module/user/service/user.service.ts +368 -368
  345. package/src/module/user/user.module.ts +65 -65
  346. package/src/module/workflow/controller/action-category.controller.ts +54 -54
  347. package/src/module/workflow/controller/action-resource-mapping.controller.ts +23 -23
  348. package/src/module/workflow/controller/action-template-mapping.controller.ts +35 -35
  349. package/src/module/workflow/controller/action.controller.ts +111 -111
  350. package/src/module/workflow/controller/activity-log.controller.ts +55 -55
  351. package/src/module/workflow/controller/comm-template.controller.ts +43 -43
  352. package/src/module/workflow/controller/entity-modification.controller.ts +35 -35
  353. package/src/module/workflow/controller/form-master.controller.ts +43 -43
  354. package/src/module/workflow/controller/stage-group.controller.ts +49 -49
  355. package/src/module/workflow/controller/stage.controller.ts +51 -51
  356. package/src/module/workflow/controller/task.controller.ts +77 -77
  357. package/src/module/workflow/controller/workflow-list-master.controller.ts +44 -44
  358. package/src/module/workflow/controller/workflow-meta.controller.ts +80 -80
  359. package/src/module/workflow/controller/workflow.controller.ts +67 -67
  360. package/src/module/workflow/entity/action-category.entity.ts +38 -38
  361. package/src/module/workflow/entity/action-data.entity.ts +55 -55
  362. package/src/module/workflow/entity/action-resources-mapping.entity.ts +29 -29
  363. package/src/module/workflow/entity/action-template-mapping.entity.ts +17 -17
  364. package/src/module/workflow/entity/action.entity.ts +53 -53
  365. package/src/module/workflow/entity/activity-log.entity.ts +43 -43
  366. package/src/module/workflow/entity/comm-template.entity.ts +43 -43
  367. package/src/module/workflow/entity/entity-modification.entity.ts +38 -38
  368. package/src/module/workflow/entity/form.entity.ts +25 -25
  369. package/src/module/workflow/entity/stage-action-mapping.entity.ts +17 -17
  370. package/src/module/workflow/entity/stage-group.entity.ts +23 -23
  371. package/src/module/workflow/entity/stage-movement-data.entity.ts +38 -38
  372. package/src/module/workflow/entity/stage.entity.ts +20 -20
  373. package/src/module/workflow/entity/task-data.entity.ts +88 -88
  374. package/src/module/workflow/entity/template-attach-mapper.entity.ts +30 -30
  375. package/src/module/workflow/entity/workflow-data.entity.ts +11 -11
  376. package/src/module/workflow/entity/workflow-level-mapping.entity.ts +18 -18
  377. package/src/module/workflow/entity/workflow.entity.ts +20 -20
  378. package/src/module/workflow/repository/action-category.repository.ts +79 -79
  379. package/src/module/workflow/repository/action-data.repository.ts +354 -354
  380. package/src/module/workflow/repository/action.repository.ts +339 -339
  381. package/src/module/workflow/repository/activity-log.repository.ts +148 -148
  382. package/src/module/workflow/repository/comm-template.repository.ts +157 -157
  383. package/src/module/workflow/repository/form-master.repository.ts +50 -50
  384. package/src/module/workflow/repository/stage-group.repository.ts +186 -186
  385. package/src/module/workflow/repository/stage-movement.repository.ts +217 -217
  386. package/src/module/workflow/repository/stage.repository.ts +160 -160
  387. package/src/module/workflow/repository/task.repository.ts +156 -156
  388. package/src/module/workflow/repository/workflow.repository.ts +42 -42
  389. package/src/module/workflow/service/action-category.service.ts +33 -33
  390. package/src/module/workflow/service/action-data.service.ts +62 -62
  391. package/src/module/workflow/service/action-resources-mapping.service.ts +10 -10
  392. package/src/module/workflow/service/action-template-mapping.service.ts +137 -137
  393. package/src/module/workflow/service/action.service.ts +302 -302
  394. package/src/module/workflow/service/activity-log.service.ts +107 -107
  395. package/src/module/workflow/service/comm-template.service.ts +181 -181
  396. package/src/module/workflow/service/entity-modification.service.ts +61 -61
  397. package/src/module/workflow/service/form-master.service.ts +35 -35
  398. package/src/module/workflow/service/populate-workflow.service.ts +320 -320
  399. package/src/module/workflow/service/stage-action-mapping.service.ts +5 -5
  400. package/src/module/workflow/service/stage-group.service.ts +325 -325
  401. package/src/module/workflow/service/stage.service.ts +197 -197
  402. package/src/module/workflow/service/task.service.ts +551 -551
  403. package/src/module/workflow/service/workflow-list-master.service.ts +68 -68
  404. package/src/module/workflow/service/workflow-meta.service.ts +640 -640
  405. package/src/module/workflow/service/workflow.service.ts +213 -213
  406. package/src/module/workflow/workflow.module.ts +180 -180
  407. package/src/module/workflow-automation/SCHEDULING_GUIDE.md +145 -145
  408. package/src/module/workflow-automation/controller/workflow-automation.controller.ts +43 -43
  409. package/src/module/workflow-automation/entity/workflow-automation-action.entity.ts +26 -26
  410. package/src/module/workflow-automation/entity/workflow-automation.entity.ts +40 -40
  411. package/src/module/workflow-automation/interface/action.decorator.ts +7 -7
  412. package/src/module/workflow-automation/interface/action.interface.ts +5 -5
  413. package/src/module/workflow-automation/service/action-registery.service.ts +35 -35
  414. package/src/module/workflow-automation/service/schedule-handler.service.ts +168 -168
  415. package/src/module/workflow-automation/service/workflow-automation-engine.service.ts +219 -219
  416. package/src/module/workflow-automation/service/workflow-automation.service.ts +474 -476
  417. package/src/module/workflow-automation/workflow-automation.module.ts +54 -54
  418. package/src/module/workflow-schedule/INSTALLATION.md +244 -244
  419. package/src/module/workflow-schedule/MULTI_PROJECT_GUIDE.md +196 -196
  420. package/src/module/workflow-schedule/README.md +422 -422
  421. package/src/module/workflow-schedule/constants/schedule.constants.ts +48 -48
  422. package/src/module/workflow-schedule/controller/workflow-schedule.controller.ts +253 -253
  423. package/src/module/workflow-schedule/docs/CLAUDE_CODE_GUIDE.md +510 -510
  424. package/src/module/workflow-schedule/docs/CLAUDE_CODE_PROMPT.md +362 -362
  425. package/src/module/workflow-schedule/docs/RUN_CLAUDE_CODE.sh +68 -68
  426. package/src/module/workflow-schedule/dto/create-schedule.dto.ts +147 -147
  427. package/src/module/workflow-schedule/dto/get-execution-logs.dto.ts +119 -119
  428. package/src/module/workflow-schedule/dto/update-schedule.dto.ts +96 -96
  429. package/src/module/workflow-schedule/entities/scheduled-workflow.entity.ts +148 -148
  430. package/src/module/workflow-schedule/entities/workflow-execution-log.entity.ts +154 -154
  431. package/src/module/workflow-schedule/interfaces/schedule-job-data.interface.ts +53 -53
  432. package/src/module/workflow-schedule/interfaces/workflow-schedule-options.interface.ts +12 -12
  433. package/src/module/workflow-schedule/processors/schedule.processor.ts +620 -620
  434. package/src/module/workflow-schedule/service/workflow-schedule.service.ts +597 -597
  435. package/src/module/workflow-schedule/workflow-schedule.module.ts +67 -67
  436. package/src/resources/dev.properties.yaml +31 -31
  437. package/src/resources/local.properties.yaml +27 -27
  438. package/src/resources/properties.module.ts +12 -12
  439. package/src/resources/properties.yaml.ts +11 -11
  440. package/src/resources/uat.properties.yaml +31 -31
  441. package/src/table.config.ts +135 -133
  442. package/src/utils/dto/excel-data.dto.ts +14 -14
  443. package/src/utils/dto/excelsheet-data.dto.ts +5 -5
  444. package/src/utils/service/base64util.service.ts +18 -18
  445. package/src/utils/service/clockIDGenUtil.service.ts +21 -21
  446. package/src/utils/service/codeGenerator.service.ts +22 -22
  447. package/src/utils/service/dateUtil.service.ts +17 -17
  448. package/src/utils/service/encryptUtil.service.ts +97 -97
  449. package/src/utils/service/excel-helper.service.ts +72 -72
  450. package/src/utils/service/excelUtil.service.ts +15 -15
  451. package/src/utils/service/file-util.service.ts +11 -11
  452. package/src/utils/service/json-util.service.ts +23 -23
  453. package/src/utils/service/loggingUtil.service.ts +88 -88
  454. package/src/utils/service/reflection-helper.service.ts +62 -62
  455. package/src/utils/service/wbsCodeGen.service.ts +8 -8
  456. package/src/utils/utils.module.ts +27 -27
  457. package/tsconfig.build.json +4 -4
  458. package/tsconfig.json +24 -24
@@ -1,640 +1,640 @@
1
- import { Inject, Injectable } from '@nestjs/common';
2
- import { ConfigService } from '@nestjs/config';
3
- import { InjectRepository } from '@nestjs/typeorm';
4
- import axios from 'axios';
5
- import { EntityServiceImpl } from 'src/module/meta/service/entity-service-impl.service';
6
- import { UserData } from 'src/module/user/entity/user.entity';
7
- import { In, Repository } from 'typeorm';
8
- import { StageMovementData } from '../entity/stage-movement-data.entity';
9
- import { ACTIVITY_CATEGORIES } from '../repository/activity-log.repository';
10
- import { StageMovementRepository } from '../repository/stage-movement.repository';
11
- import { TaskRepository } from '../repository/task.repository';
12
- import { ActionDataService } from './action-data.service';
13
- import { ActivityLogService } from './activity-log.service';
14
- import { EntityModificationService } from './entity-modification.service';
15
- import { TaskService } from './task.service';
16
- import { ActionCategory } from '../entity/action-category.entity';
17
- import { StageGroup } from '../entity/stage-group.entity';
18
- import { ActionDataEntity } from '../entity/action-data.entity';
19
- import { TaskDataEntity } from '../entity/task-data.entity';
20
-
21
- @Injectable()
22
- export class WorkflowMetaService extends EntityServiceImpl {
23
- constructor(
24
- @InjectRepository(StageMovementData)
25
- private readonly stageMovementRepo: Repository<StageMovementData>,
26
- private readonly stageMovementRepository: StageMovementRepository,
27
- private readonly taskRepository: TaskRepository,
28
- private readonly actionDataService: ActionDataService,
29
- @Inject('TaskService')
30
- private readonly taskService: TaskService,
31
- @Inject('ActivityLogService')
32
- private readonly activityLogService: ActivityLogService,
33
- @Inject('EntityModificationService')
34
- private readonly modificationService: EntityModificationService,
35
- private readonly configService: ConfigService,
36
- @InjectRepository(ActionCategory)
37
- private readonly actionCategoryRepo: Repository<ActionCategory>,
38
- @InjectRepository(StageGroup)
39
- private readonly stageGroupRepo: Repository<StageGroup>,
40
- @InjectRepository(ActionDataEntity)
41
- private readonly actionDataEntityRepository:Repository<ActionDataEntity>,
42
- @InjectRepository(TaskDataEntity)
43
- private readonly taskDataEntityRepository:Repository<TaskDataEntity>,
44
- ) {
45
- super();
46
- }
47
-
48
- // get first stage from stage table
49
- async getFirstStage(
50
- loggedInUser: UserData,
51
- mapped_entity_type: string,
52
- mapped_entity_id: number,
53
- ): Promise<any | null> {
54
- // step1 find in workflow_level_mapping if not found with current level_id and level_type search in for ORG
55
- // step2 find in stage_group => workflow_id
56
- // step3 find in stage => stage_group_id
57
- // step4 get first stage from stage table
58
- return this.stageMovementRepository.getFirstStage({
59
- loggedInUser,
60
- mapped_entity_type,
61
- mapped_entity_id,
62
- });
63
- }
64
-
65
- /**
66
- * Get the current active stage for a given mapped entity
67
- */
68
- async getCurrentStage(
69
- mapped_entity_type: string,
70
- mapped_entity_id: number,
71
- loggedInUser: UserData,
72
- ): Promise<any | null> {
73
- // Try to find the current stage movement entry
74
- const currentStage = await this.stageMovementRepo.findOne({
75
- where: {
76
- mapped_entity_type,
77
- mapped_entity_id,
78
- is_current: 'Y',
79
- },
80
- });
81
-
82
- if (!currentStage) {
83
- // Fetch the latest stage movement for the given entity
84
- const latestMovement = await this.stageMovementRepo.findOne({
85
- where: {
86
- mapped_entity_type,
87
- mapped_entity_id,
88
- },
89
- order: { id: 'DESC' }, // Get the most recent movement
90
- });
91
-
92
- if (!latestMovement) return null;
93
-
94
- let stageGroup = await this.stageGroupRepo.findOne({
95
- where: { id: latestMovement?.stage_group_id },
96
- });
97
-
98
- return {
99
- ...latestMovement,
100
- stage_group_name: stageGroup?.name,
101
- };
102
- }
103
-
104
- // get group name for the current stage
105
- const stageGroup = await this.stageGroupRepo.findOne({
106
- where: { id: currentStage?.stage_group_id },
107
- });
108
-
109
- // Return the found or newly created current stage movement
110
- return {
111
- ...currentStage,
112
- stage_group_name: stageGroup?.name,
113
- };
114
- }
115
-
116
- /**
117
- * Get next stage ID based on current stage
118
- * (Stub logic – replace with real stage transition resolution)
119
- */
120
- async getNextStage(currentStage: any): Promise<any | null> {
121
- // get all stage for that stage group
122
- const nextStage =
123
- await this.stageMovementRepository.getNextStageOrFirstOfNextGroup(
124
- currentStage?.stage_group_id,
125
- currentStage?.stage_id,
126
- );
127
-
128
- if (!nextStage) {
129
- return { hasNextStage: false, nextStage: null };
130
- }
131
-
132
- const stageGroup = await this.stageGroupRepo.findOne({
133
- where: {
134
- id: nextStage?.stage_group_id,
135
- },
136
- });
137
-
138
- return {
139
- hasNextStage: !!nextStage,
140
- nextStage: { ...nextStage, stage_group_name: stageGroup?.name },
141
- };
142
- }
143
-
144
- /**
145
- * Move mapped entity to the next stage in the workflow
146
- */
147
- async moveToNextStage(
148
- mapped_entity_type: string,
149
- mapped_entity_id: number,
150
- loggedInUser: UserData,
151
- reason_code?: string,
152
- remark?: string,
153
- ): Promise<string> {
154
- const now = new Date();
155
- let currentStage = await this.getCurrentStage(
156
- mapped_entity_type,
157
- mapped_entity_id,
158
- loggedInUser,
159
- );
160
-
161
- if (currentStage) {
162
- await this.taskService.createSystemNote(
163
- {
164
- reason_code: reason_code || '',
165
- remark: remark || '',
166
- mapped_entity_id,
167
- stage_id: currentStage.stage_id,
168
- action_id: currentStage.action_id,
169
- stage_group_id: currentStage.stage_group_id,
170
- },
171
- loggedInUser,
172
- );
173
- }
174
-
175
- // Case 1: First stage – initialize
176
- if (!currentStage) {
177
- const getAllResult = await this.getFirstStage(
178
- loggedInUser,
179
- mapped_entity_type,
180
- mapped_entity_id,
181
- );
182
-
183
- const { mappingUsed, stageGroup, firstStage } = getAllResult;
184
-
185
- currentStage = await this.stageMovementRepo.save({
186
- entity_type: 'WFSA',
187
- mapped_entity_type,
188
- mapped_entity_id,
189
- name: firstStage.name,
190
- status: firstStage.status,
191
- current_user_id: loggedInUser.id,
192
- stage_action_mapping_id: firstStage.id,
193
- organization_id: loggedInUser.organization_id,
194
- enterprise_id: loggedInUser.enterprise_id,
195
- stage_group_id: stageGroup.id,
196
- stage_id: firstStage.id,
197
- start_date: new Date(),
198
- is_current: 'Y',
199
- });
200
-
201
- await this.populateActionService(
202
- currentStage.stage_id,
203
- loggedInUser,
204
- mapped_entity_id,
205
- mapped_entity_type,
206
- );
207
- return `Initialized workflow with first stage (Stage ID: ${firstStage.id}).`;
208
- }
209
-
210
- // Case 2: Has next stage – move forward
211
- const stageData = await this.getNextStage(currentStage);
212
-
213
- if (stageData.hasNextStage) {
214
- const { nextStage } = stageData;
215
-
216
- // ✅ NEW: Log stage group completion if group changes
217
- if (nextStage.stage_group_id !== currentStage.stage_group_id) {
218
- try {
219
- // stage action category SGCP
220
- // if current stage actions contain any action with action_category as SGCP then set the status of that action to completed
221
- const stageActions =
222
- await this.stageMovementRepository.getAllActionByStageId(
223
- currentStage.stage_id,
224
- );
225
-
226
- for (const action of stageActions) {
227
- const actionCategoryRepo = this.reflectionHelper.getRepoService('ActionCategory');
228
- const actionCategory = await actionCategoryRepo.findOne({
229
- where: {
230
- id: Number(action.action_category),
231
- },
232
- });
233
-
234
- if (actionCategory?.code == 'SGCP') {
235
- await this.taskRepository.updateTaskStatus(
236
- loggedInUser,
237
- mapped_entity_type,
238
- mapped_entity_id,
239
- currentStage.stage_id,
240
- action.id,
241
- );
242
- }
243
- }
244
-
245
- await this.activityLogService.logActivity(
246
- {
247
- mapped_entity_id,
248
- mapped_entity_type,
249
- title: 'Stage Group Completed',
250
- description: `Stage Group changed from ${currentStage.stage_group_name} to ${nextStage.stage_group_name}.`,
251
- action: 'completed',
252
- category: ACTIVITY_CATEGORIES.STAGEGROUP,
253
- appcode: loggedInUser.appcode,
254
- },
255
- loggedInUser,
256
- );
257
- } catch (error) {
258
- console.error(
259
- 'Failed to log stage group completion:',
260
- error?.message || error,
261
- );
262
- }
263
- }
264
-
265
- // Close current stage
266
- currentStage.end_date = now;
267
- currentStage.is_current = 'N';
268
- await this.stageMovementRepo.save(currentStage);
269
-
270
- // Insert next stage
271
- await this.stageMovementRepo.save({
272
- entity_type: 'WFSA',
273
- mapped_entity_type,
274
- mapped_entity_id,
275
- name: nextStage.name,
276
- status: nextStage.status,
277
- current_user_id: loggedInUser.id,
278
- stage_action_mapping_id: nextStage.id,
279
- organization_id: loggedInUser.organization_id,
280
- enterprise_id: loggedInUser.enterprise_id,
281
- start_date: now,
282
- stage_group_id: nextStage?.stage_group_id,
283
- stage_id: nextStage.id,
284
- is_current: 'Y',
285
- });
286
-
287
- await this.populateActionService(
288
- stageData.nextStage.id,
289
- loggedInUser,
290
- mapped_entity_id,
291
- mapped_entity_type,
292
- );
293
-
294
- // Stage completion log
295
- try {
296
- await this.activityLogService.logActivity(
297
- {
298
- mapped_entity_id: mapped_entity_id,
299
- mapped_entity_type: mapped_entity_type,
300
- title: 'Stage Completed',
301
- description: `Stage changed from ${currentStage.name} to ${nextStage.name}.`,
302
- action: 'status',
303
- category: ACTIVITY_CATEGORIES.STAGE,
304
- appcode: loggedInUser.appcode,
305
- },
306
- loggedInUser,
307
- );
308
- } catch (error) {
309
- console.error(
310
- 'Failed to log activity for stage completion:',
311
- error?.message || error,
312
- );
313
- }
314
- return `Moved to next stage (Stage ID: ${nextStage.id}).`;
315
- }
316
-
317
- // Case 3: No next stage – mark workflow as done
318
- currentStage.end_date = now;
319
- currentStage.is_current = 'N';
320
- await this.stageMovementRepo.save(currentStage);
321
-
322
- return 'Workflow completed. No next stage available.';
323
- }
324
-
325
- async populateActionService(
326
- stage_id: number,
327
- loggedInUser: UserData,
328
- mapped_entity_id: number,
329
- mapped_entity_type: string,
330
- ): Promise<any> {
331
- const actions =
332
- await this.stageMovementRepository.getAllActionByStageId(stage_id);
333
- // Populate the action service with the retrieved actions
334
-
335
- if (!actions || actions.length === 0) {
336
- return 'No actions found for this stage.';
337
- }
338
-
339
- // check whether first action's action_category is owner_assignment and assignment_type is AUTO_ASSIGN
340
- const firstAction = actions[0];
341
-
342
- const actionCategory = await this.actionCategoryRepo.findOne({
343
- where: {
344
- id: Number(firstAction.action_category),
345
- },
346
- });
347
-
348
- const listMasterItemsRepo = this.reflectionHelper.getRepoService('ListMasterItems');
349
-
350
- const assignmentType = await listMasterItemsRepo.findOne({
351
- where: {
352
- id: Number(firstAction.assignment_type),
353
- },
354
- });
355
-
356
- // save in action data
357
- await this.actionDataService.saveActionData(
358
- actions,
359
- loggedInUser,
360
- mapped_entity_id,
361
- mapped_entity_type,
362
- );
363
-
364
- // save in task data
365
- await this.taskService.saveActionData(
366
- actions,
367
- loggedInUser,
368
- mapped_entity_id,
369
- mapped_entity_type,
370
- );
371
-
372
- if (
373
- actionCategory?.code == 'OWAS' &&
374
- assignmentType[0]?.value == 'round_robin'
375
- ) {
376
- console.log('Auto-assigning owner based on round-robin assignment type');
377
- await this.assignLead(
378
- loggedInUser,
379
- mapped_entity_id,
380
- mapped_entity_type,
381
- stage_id,
382
- actions,
383
- );
384
- }
385
- }
386
-
387
- async updateLeadOwner(entityData, loggedInUser) {
388
- const { lead_id, lead_owner, stage_id, entity_type } = entityData;
389
- const updatedData = {
390
- id: lead_id,
391
- lead_owner: lead_owner,
392
- entity_type: entity_type,
393
- };
394
- entityData = updatedData;
395
-
396
- this.modificationService.logModification(
397
- {
398
- entity_type: 'ENMD',
399
- mapped_entity_type: entity_type,
400
- mapped_entity_id: lead_id,
401
- attribute_key: 'lead_owner',
402
- new_value: lead_owner,
403
- old_value: entityData.lead_owner,
404
- remarks: entityData.remarks,
405
- reason_code: entityData.reason_code,
406
- stage_id: stage_id,
407
- action_id: entityData.action_id,
408
- },
409
- loggedInUser,
410
- );
411
-
412
- const leadData: any = await this.getEntityData(
413
- 'LEAD',
414
- lead_id,
415
- loggedInUser,
416
- );
417
-
418
- const listMasterItemsRepo = this.reflectionHelper.getRepoService('ListMasterItems');
419
- const unassignedListMasterItemData = await listMasterItemsRepo.find({
420
- where: {
421
- listtype: 'LEST',
422
- enterprise_id: loggedInUser.enterprise_id,
423
- value: In(['unassigned', 'active'])
424
- }
425
- });
426
-
427
- // Find the IDs explicitly
428
- const unassignedId = unassignedListMasterItemData.find(
429
- (item) => item.name.toLowerCase() === 'unassigned',
430
- )?.id;
431
-
432
- const activeId = unassignedListMasterItemData.find(
433
- (item) => item.name.toLowerCase() === 'active',
434
- )?.id;
435
-
436
- // Use unassignedId for comparison
437
- if (leadData?.lead_status === unassignedId) {
438
- leadData.lead_status = activeId;
439
- }
440
-
441
- const result = await super.updateEntity(
442
- {
443
- ...entityData,
444
- status: leadData?.lead_status,
445
- lead_status: leadData?.lead_status,
446
- },
447
- loggedInUser,
448
- );
449
-
450
- await this.actionDataEntityRepository
451
- .createQueryBuilder()
452
- .update()
453
- .set({ user_id: lead_owner })
454
- .where("mapped_entity_id = :leadId", { leadId: lead_id })
455
- .andWhere("mapped_entity_type = :entityType", { entityType: entity_type })
456
- .andWhere("stage_id = :stageId", { stageId: stage_id })
457
- .andWhere("(is_current = 'Y' OR is_current IS NULL)")
458
- .execute();
459
-
460
- const leadMeetingRepo = this.reflectionHelper.getRepoService('LeadScheduleMeet');
461
-
462
- await leadMeetingRepo
463
- .createQueryBuilder()
464
- .update()
465
- .set({ user_id: lead_owner })
466
- .where("stage_id = :stageId", { stageId: stage_id })
467
- .andWhere("mapped_entity_id = :leadId", { leadId: lead_id })
468
- .andWhere("mapped_entity_type = :entityType", { entityType: "LEAD" })
469
- .andWhere("(status = 'scheduled' OR status = 'rescheduled')")
470
- .execute();
471
-
472
- const taskRows = await this.taskDataEntityRepository.find({
473
- where: {
474
- mapped_entity_id: lead_id,
475
- mapped_entity_type: entity_type,
476
- stage_id
477
- }
478
- });
479
-
480
- for (const task of taskRows) {
481
- const statusRows = await listMasterItemsRepo.findOne({
482
- where: {
483
- id: task.status
484
- }
485
- });
486
-
487
- const statusName = statusRows?.value?.toLowerCase() || '';
488
- if (
489
- ['todo', 'in_progress'].includes(statusName) ||
490
- statusName === 'todo'
491
- ) {
492
-
493
- await this.taskDataEntityRepository.update(task.id,{
494
- user_id: lead_owner,
495
- task_owner: lead_owner
496
- });
497
- }
498
- }
499
-
500
- let leadOwnerName;
501
-
502
- try {
503
- const baseUrl = this.configService.get<string>('REDIRECT_BE_URL');
504
-
505
- // Prepare query params
506
- const queryParams = new URLSearchParams({
507
- loggedInUser: JSON.stringify(loggedInUser),
508
- }).toString();
509
-
510
- const url = `${baseUrl}/entity/public/getById/${entityData?.lead_owner}?entity_type=USR&${queryParams}`;
511
-
512
- const response = await axios.get(url);
513
-
514
- leadOwnerName = response.data;
515
- } catch (error) {
516
- console.error('Internal Entity API call failed:', error.message);
517
- }
518
-
519
- try {
520
- const logData = {
521
- mapped_entity_id: lead_id,
522
- mapped_entity_type: 'LEAD',
523
- title: 'Owner Assigned',
524
- description: leadData?.lead_owner
525
- ? `${leadOwnerName?.name} reassigned as Lead owner.`
526
- : `${leadOwnerName?.name} assigned as Lead owner.`,
527
- action: 'assign',
528
- category: ACTIVITY_CATEGORIES.ASSIGN,
529
- appcode: loggedInUser.appcode,
530
- };
531
-
532
- await this.activityLogService.logActivity(logData, loggedInUser);
533
- } catch (error) {
534
- console.error(
535
- 'Failed to log activity for meeting:',
536
- error?.message || error,
537
- );
538
- // Logging should not block main flow
539
- }
540
- return result;
541
- }
542
-
543
- async assignLead(
544
- loggedInUser: UserData,
545
- mapped_entity_id: number,
546
- mapped_entity_type: string,
547
- stage_id: number,
548
- actions: any[],
549
- ): Promise<number> {
550
- const { organization_id, level_id, level_type } = loggedInUser;
551
-
552
- let owners = [] as any;
553
-
554
- try {
555
- const baseUrl = this.configService.get<string>('REDIRECT_BE_URL');
556
-
557
- // Prepare query params
558
- const queryParams = new URLSearchParams({
559
- loggedInUser: JSON.stringify(loggedInUser),
560
- }).toString();
561
-
562
- const { level_id } = loggedInUser;
563
-
564
- const url = `${baseUrl}/users/public/lead-owners/?entity_type=USR&${queryParams}`;
565
-
566
- const response = await axios.get(url);
567
-
568
- owners = response.data;
569
-
570
- // The API response is likely a JSON object, not an array
571
- } catch (error) {
572
- console.error('Internal Entity API call failed:', error.message);
573
- }
574
-
575
- if (!owners?.length)
576
- console.log('No eligible owners found for lead assignment.');
577
- const userIds = owners?.map((o) => Number(o.id)); // normalize to numbers
578
-
579
- // 2) Find the last assigned *eligible* owner (use IN (...))
580
- const placeholders = userIds.map(() => '?').join(',');
581
-
582
- const leadRepo = this.reflectionHelper.getRepoService('CRMLead');
583
- const lastRow = await leadRepo
584
- .createQueryBuilder("cl")
585
- .select("cl.lead_owner", "lead_owner")
586
- .where("cl.organization_id = :orgId", { orgId: organization_id })
587
- .andWhere("cl.level_id = :levelId", { levelId: Number(level_id) })
588
- .andWhere("cl.level_type = :levelType", { levelType: level_type })
589
- .andWhere("cl.lead_owner IN (:...owners)", { owners: userIds })
590
- .orderBy("cl.created_date", "DESC")
591
- .limit(1)
592
- .getRawOne();
593
-
594
- const lastAssigned = lastRow.length ? Number(lastRow[0].lead_owner) : null;
595
-
596
- // 3) Compute next user in round-robin
597
- const lastIdx = lastAssigned != null ? userIds.indexOf(lastAssigned) : -1;
598
- const nextIdx = (lastIdx + 1) % userIds.length;
599
- const nextUser = userIds[nextIdx];
600
-
601
- //4) Update lead owner in the lead table no need because we are updating lead owner in updateLeadOwner method
602
- console.log(`Assigning lead to user ID: ${nextUser}`);
603
-
604
- const firstAction = actions[0];
605
-
606
- // update lead owner status
607
- await this.updateLeadOwner(
608
- {
609
- lead_id: mapped_entity_id,
610
- lead_owner: nextUser,
611
- stage_id: stage_id,
612
- entity_type: mapped_entity_type,
613
- action_id: firstAction.id,
614
- },
615
- loggedInUser,
616
- );
617
-
618
- // move task
619
- await this.taskService.moveTask(loggedInUser, {
620
- mapped_entity_type,
621
- mapped_entity_id,
622
- stage_id,
623
- action_id: firstAction.id,
624
- });
625
-
626
- // if it has only one action then move next stage
627
- if (actions.length == 1) {
628
- console.log(
629
- 'Only one action present and it is owner assignment. Moving to next stage.',
630
- );
631
- await this.moveToNextStage(
632
- mapped_entity_type,
633
- mapped_entity_id,
634
- loggedInUser,
635
- );
636
- }
637
-
638
- return nextUser;
639
- }
640
- }
1
+ import { Inject, Injectable } from '@nestjs/common';
2
+ import { ConfigService } from '@nestjs/config';
3
+ import { InjectRepository } from '@nestjs/typeorm';
4
+ import axios from 'axios';
5
+ import { EntityServiceImpl } from 'src/module/meta/service/entity-service-impl.service';
6
+ import { UserData } from 'src/module/user/entity/user.entity';
7
+ import { In, Repository } from 'typeorm';
8
+ import { StageMovementData } from '../entity/stage-movement-data.entity';
9
+ import { ACTIVITY_CATEGORIES } from '../repository/activity-log.repository';
10
+ import { StageMovementRepository } from '../repository/stage-movement.repository';
11
+ import { TaskRepository } from '../repository/task.repository';
12
+ import { ActionDataService } from './action-data.service';
13
+ import { ActivityLogService } from './activity-log.service';
14
+ import { EntityModificationService } from './entity-modification.service';
15
+ import { TaskService } from './task.service';
16
+ import { ActionCategory } from '../entity/action-category.entity';
17
+ import { StageGroup } from '../entity/stage-group.entity';
18
+ import { ActionDataEntity } from '../entity/action-data.entity';
19
+ import { TaskDataEntity } from '../entity/task-data.entity';
20
+
21
+ @Injectable()
22
+ export class WorkflowMetaService extends EntityServiceImpl {
23
+ constructor(
24
+ @InjectRepository(StageMovementData)
25
+ private readonly stageMovementRepo: Repository<StageMovementData>,
26
+ private readonly stageMovementRepository: StageMovementRepository,
27
+ private readonly taskRepository: TaskRepository,
28
+ private readonly actionDataService: ActionDataService,
29
+ @Inject('TaskService')
30
+ private readonly taskService: TaskService,
31
+ @Inject('ActivityLogService')
32
+ private readonly activityLogService: ActivityLogService,
33
+ @Inject('EntityModificationService')
34
+ private readonly modificationService: EntityModificationService,
35
+ private readonly configService: ConfigService,
36
+ @InjectRepository(ActionCategory)
37
+ private readonly actionCategoryRepo: Repository<ActionCategory>,
38
+ @InjectRepository(StageGroup)
39
+ private readonly stageGroupRepo: Repository<StageGroup>,
40
+ @InjectRepository(ActionDataEntity)
41
+ private readonly actionDataEntityRepository:Repository<ActionDataEntity>,
42
+ @InjectRepository(TaskDataEntity)
43
+ private readonly taskDataEntityRepository:Repository<TaskDataEntity>,
44
+ ) {
45
+ super();
46
+ }
47
+
48
+ // get first stage from stage table
49
+ async getFirstStage(
50
+ loggedInUser: UserData,
51
+ mapped_entity_type: string,
52
+ mapped_entity_id: number,
53
+ ): Promise<any | null> {
54
+ // step1 find in workflow_level_mapping if not found with current level_id and level_type search in for ORG
55
+ // step2 find in stage_group => workflow_id
56
+ // step3 find in stage => stage_group_id
57
+ // step4 get first stage from stage table
58
+ return this.stageMovementRepository.getFirstStage({
59
+ loggedInUser,
60
+ mapped_entity_type,
61
+ mapped_entity_id,
62
+ });
63
+ }
64
+
65
+ /**
66
+ * Get the current active stage for a given mapped entity
67
+ */
68
+ async getCurrentStage(
69
+ mapped_entity_type: string,
70
+ mapped_entity_id: number,
71
+ loggedInUser: UserData,
72
+ ): Promise<any | null> {
73
+ // Try to find the current stage movement entry
74
+ const currentStage = await this.stageMovementRepo.findOne({
75
+ where: {
76
+ mapped_entity_type,
77
+ mapped_entity_id,
78
+ is_current: 'Y',
79
+ },
80
+ });
81
+
82
+ if (!currentStage) {
83
+ // Fetch the latest stage movement for the given entity
84
+ const latestMovement = await this.stageMovementRepo.findOne({
85
+ where: {
86
+ mapped_entity_type,
87
+ mapped_entity_id,
88
+ },
89
+ order: { id: 'DESC' }, // Get the most recent movement
90
+ });
91
+
92
+ if (!latestMovement) return null;
93
+
94
+ let stageGroup = await this.stageGroupRepo.findOne({
95
+ where: { id: latestMovement?.stage_group_id },
96
+ });
97
+
98
+ return {
99
+ ...latestMovement,
100
+ stage_group_name: stageGroup?.name,
101
+ };
102
+ }
103
+
104
+ // get group name for the current stage
105
+ const stageGroup = await this.stageGroupRepo.findOne({
106
+ where: { id: currentStage?.stage_group_id },
107
+ });
108
+
109
+ // Return the found or newly created current stage movement
110
+ return {
111
+ ...currentStage,
112
+ stage_group_name: stageGroup?.name,
113
+ };
114
+ }
115
+
116
+ /**
117
+ * Get next stage ID based on current stage
118
+ * (Stub logic – replace with real stage transition resolution)
119
+ */
120
+ async getNextStage(currentStage: any): Promise<any | null> {
121
+ // get all stage for that stage group
122
+ const nextStage =
123
+ await this.stageMovementRepository.getNextStageOrFirstOfNextGroup(
124
+ currentStage?.stage_group_id,
125
+ currentStage?.stage_id,
126
+ );
127
+
128
+ if (!nextStage) {
129
+ return { hasNextStage: false, nextStage: null };
130
+ }
131
+
132
+ const stageGroup = await this.stageGroupRepo.findOne({
133
+ where: {
134
+ id: nextStage?.stage_group_id,
135
+ },
136
+ });
137
+
138
+ return {
139
+ hasNextStage: !!nextStage,
140
+ nextStage: { ...nextStage, stage_group_name: stageGroup?.name },
141
+ };
142
+ }
143
+
144
+ /**
145
+ * Move mapped entity to the next stage in the workflow
146
+ */
147
+ async moveToNextStage(
148
+ mapped_entity_type: string,
149
+ mapped_entity_id: number,
150
+ loggedInUser: UserData,
151
+ reason_code?: string,
152
+ remark?: string,
153
+ ): Promise<string> {
154
+ const now = new Date();
155
+ let currentStage = await this.getCurrentStage(
156
+ mapped_entity_type,
157
+ mapped_entity_id,
158
+ loggedInUser,
159
+ );
160
+
161
+ if (currentStage) {
162
+ await this.taskService.createSystemNote(
163
+ {
164
+ reason_code: reason_code || '',
165
+ remark: remark || '',
166
+ mapped_entity_id,
167
+ stage_id: currentStage.stage_id,
168
+ action_id: currentStage.action_id,
169
+ stage_group_id: currentStage.stage_group_id,
170
+ },
171
+ loggedInUser,
172
+ );
173
+ }
174
+
175
+ // Case 1: First stage – initialize
176
+ if (!currentStage) {
177
+ const getAllResult = await this.getFirstStage(
178
+ loggedInUser,
179
+ mapped_entity_type,
180
+ mapped_entity_id,
181
+ );
182
+
183
+ const { mappingUsed, stageGroup, firstStage } = getAllResult;
184
+
185
+ currentStage = await this.stageMovementRepo.save({
186
+ entity_type: 'WFSA',
187
+ mapped_entity_type,
188
+ mapped_entity_id,
189
+ name: firstStage.name,
190
+ status: firstStage.status,
191
+ current_user_id: loggedInUser.id,
192
+ stage_action_mapping_id: firstStage.id,
193
+ organization_id: loggedInUser.organization_id,
194
+ enterprise_id: loggedInUser.enterprise_id,
195
+ stage_group_id: stageGroup.id,
196
+ stage_id: firstStage.id,
197
+ start_date: new Date(),
198
+ is_current: 'Y',
199
+ });
200
+
201
+ await this.populateActionService(
202
+ currentStage.stage_id,
203
+ loggedInUser,
204
+ mapped_entity_id,
205
+ mapped_entity_type,
206
+ );
207
+ return `Initialized workflow with first stage (Stage ID: ${firstStage.id}).`;
208
+ }
209
+
210
+ // Case 2: Has next stage – move forward
211
+ const stageData = await this.getNextStage(currentStage);
212
+
213
+ if (stageData.hasNextStage) {
214
+ const { nextStage } = stageData;
215
+
216
+ // ✅ NEW: Log stage group completion if group changes
217
+ if (nextStage.stage_group_id !== currentStage.stage_group_id) {
218
+ try {
219
+ // stage action category SGCP
220
+ // if current stage actions contain any action with action_category as SGCP then set the status of that action to completed
221
+ const stageActions =
222
+ await this.stageMovementRepository.getAllActionByStageId(
223
+ currentStage.stage_id,
224
+ );
225
+
226
+ for (const action of stageActions) {
227
+ const actionCategoryRepo = this.reflectionHelper.getRepoService('ActionCategory');
228
+ const actionCategory = await actionCategoryRepo.findOne({
229
+ where: {
230
+ id: Number(action.action_category),
231
+ },
232
+ });
233
+
234
+ if (actionCategory?.code == 'SGCP') {
235
+ await this.taskRepository.updateTaskStatus(
236
+ loggedInUser,
237
+ mapped_entity_type,
238
+ mapped_entity_id,
239
+ currentStage.stage_id,
240
+ action.id,
241
+ );
242
+ }
243
+ }
244
+
245
+ await this.activityLogService.logActivity(
246
+ {
247
+ mapped_entity_id,
248
+ mapped_entity_type,
249
+ title: 'Stage Group Completed',
250
+ description: `Stage Group changed from ${currentStage.stage_group_name} to ${nextStage.stage_group_name}.`,
251
+ action: 'completed',
252
+ category: ACTIVITY_CATEGORIES.STAGEGROUP,
253
+ appcode: loggedInUser.appcode,
254
+ },
255
+ loggedInUser,
256
+ );
257
+ } catch (error) {
258
+ console.error(
259
+ 'Failed to log stage group completion:',
260
+ error?.message || error,
261
+ );
262
+ }
263
+ }
264
+
265
+ // Close current stage
266
+ currentStage.end_date = now;
267
+ currentStage.is_current = 'N';
268
+ await this.stageMovementRepo.save(currentStage);
269
+
270
+ // Insert next stage
271
+ await this.stageMovementRepo.save({
272
+ entity_type: 'WFSA',
273
+ mapped_entity_type,
274
+ mapped_entity_id,
275
+ name: nextStage.name,
276
+ status: nextStage.status,
277
+ current_user_id: loggedInUser.id,
278
+ stage_action_mapping_id: nextStage.id,
279
+ organization_id: loggedInUser.organization_id,
280
+ enterprise_id: loggedInUser.enterprise_id,
281
+ start_date: now,
282
+ stage_group_id: nextStage?.stage_group_id,
283
+ stage_id: nextStage.id,
284
+ is_current: 'Y',
285
+ });
286
+
287
+ await this.populateActionService(
288
+ stageData.nextStage.id,
289
+ loggedInUser,
290
+ mapped_entity_id,
291
+ mapped_entity_type,
292
+ );
293
+
294
+ // Stage completion log
295
+ try {
296
+ await this.activityLogService.logActivity(
297
+ {
298
+ mapped_entity_id: mapped_entity_id,
299
+ mapped_entity_type: mapped_entity_type,
300
+ title: 'Stage Completed',
301
+ description: `Stage changed from ${currentStage.name} to ${nextStage.name}.`,
302
+ action: 'status',
303
+ category: ACTIVITY_CATEGORIES.STAGE,
304
+ appcode: loggedInUser.appcode,
305
+ },
306
+ loggedInUser,
307
+ );
308
+ } catch (error) {
309
+ console.error(
310
+ 'Failed to log activity for stage completion:',
311
+ error?.message || error,
312
+ );
313
+ }
314
+ return `Moved to next stage (Stage ID: ${nextStage.id}).`;
315
+ }
316
+
317
+ // Case 3: No next stage – mark workflow as done
318
+ currentStage.end_date = now;
319
+ currentStage.is_current = 'N';
320
+ await this.stageMovementRepo.save(currentStage);
321
+
322
+ return 'Workflow completed. No next stage available.';
323
+ }
324
+
325
+ async populateActionService(
326
+ stage_id: number,
327
+ loggedInUser: UserData,
328
+ mapped_entity_id: number,
329
+ mapped_entity_type: string,
330
+ ): Promise<any> {
331
+ const actions =
332
+ await this.stageMovementRepository.getAllActionByStageId(stage_id);
333
+ // Populate the action service with the retrieved actions
334
+
335
+ if (!actions || actions.length === 0) {
336
+ return 'No actions found for this stage.';
337
+ }
338
+
339
+ // check whether first action's action_category is owner_assignment and assignment_type is AUTO_ASSIGN
340
+ const firstAction = actions[0];
341
+
342
+ const actionCategory = await this.actionCategoryRepo.findOne({
343
+ where: {
344
+ id: Number(firstAction.action_category),
345
+ },
346
+ });
347
+
348
+ const listMasterItemsRepo = this.reflectionHelper.getRepoService('ListMasterItems');
349
+
350
+ const assignmentType = await listMasterItemsRepo.findOne({
351
+ where: {
352
+ id: Number(firstAction.assignment_type),
353
+ },
354
+ });
355
+
356
+ // save in action data
357
+ await this.actionDataService.saveActionData(
358
+ actions,
359
+ loggedInUser,
360
+ mapped_entity_id,
361
+ mapped_entity_type,
362
+ );
363
+
364
+ // save in task data
365
+ await this.taskService.saveActionData(
366
+ actions,
367
+ loggedInUser,
368
+ mapped_entity_id,
369
+ mapped_entity_type,
370
+ );
371
+
372
+ if (
373
+ actionCategory?.code == 'OWAS' &&
374
+ assignmentType[0]?.value == 'round_robin'
375
+ ) {
376
+ console.log('Auto-assigning owner based on round-robin assignment type');
377
+ await this.assignLead(
378
+ loggedInUser,
379
+ mapped_entity_id,
380
+ mapped_entity_type,
381
+ stage_id,
382
+ actions,
383
+ );
384
+ }
385
+ }
386
+
387
+ async updateLeadOwner(entityData, loggedInUser) {
388
+ const { lead_id, lead_owner, stage_id, entity_type } = entityData;
389
+ const updatedData = {
390
+ id: lead_id,
391
+ lead_owner: lead_owner,
392
+ entity_type: entity_type,
393
+ };
394
+ entityData = updatedData;
395
+
396
+ this.modificationService.logModification(
397
+ {
398
+ entity_type: 'ENMD',
399
+ mapped_entity_type: entity_type,
400
+ mapped_entity_id: lead_id,
401
+ attribute_key: 'lead_owner',
402
+ new_value: lead_owner,
403
+ old_value: entityData.lead_owner,
404
+ remarks: entityData.remarks,
405
+ reason_code: entityData.reason_code,
406
+ stage_id: stage_id,
407
+ action_id: entityData.action_id,
408
+ },
409
+ loggedInUser,
410
+ );
411
+
412
+ const leadData: any = await this.getEntityData(
413
+ 'LEAD',
414
+ lead_id,
415
+ loggedInUser,
416
+ );
417
+
418
+ const listMasterItemsRepo = this.reflectionHelper.getRepoService('ListMasterItems');
419
+ const unassignedListMasterItemData = await listMasterItemsRepo.find({
420
+ where: {
421
+ listtype: 'LEST',
422
+ enterprise_id: loggedInUser.enterprise_id,
423
+ value: In(['unassigned', 'active'])
424
+ }
425
+ });
426
+
427
+ // Find the IDs explicitly
428
+ const unassignedId = unassignedListMasterItemData.find(
429
+ (item) => item.name.toLowerCase() === 'unassigned',
430
+ )?.id;
431
+
432
+ const activeId = unassignedListMasterItemData.find(
433
+ (item) => item.name.toLowerCase() === 'active',
434
+ )?.id;
435
+
436
+ // Use unassignedId for comparison
437
+ if (leadData?.lead_status === unassignedId) {
438
+ leadData.lead_status = activeId;
439
+ }
440
+
441
+ const result = await super.updateEntity(
442
+ {
443
+ ...entityData,
444
+ status: leadData?.lead_status,
445
+ lead_status: leadData?.lead_status,
446
+ },
447
+ loggedInUser,
448
+ );
449
+
450
+ await this.actionDataEntityRepository
451
+ .createQueryBuilder()
452
+ .update()
453
+ .set({ user_id: lead_owner })
454
+ .where("mapped_entity_id = :leadId", { leadId: lead_id })
455
+ .andWhere("mapped_entity_type = :entityType", { entityType: entity_type })
456
+ .andWhere("stage_id = :stageId", { stageId: stage_id })
457
+ .andWhere("(is_current = 'Y' OR is_current IS NULL)")
458
+ .execute();
459
+
460
+ const leadMeetingRepo = this.reflectionHelper.getRepoService('LeadScheduleMeet');
461
+
462
+ await leadMeetingRepo
463
+ .createQueryBuilder()
464
+ .update()
465
+ .set({ user_id: lead_owner })
466
+ .where("stage_id = :stageId", { stageId: stage_id })
467
+ .andWhere("mapped_entity_id = :leadId", { leadId: lead_id })
468
+ .andWhere("mapped_entity_type = :entityType", { entityType: "LEAD" })
469
+ .andWhere("(status = 'scheduled' OR status = 'rescheduled')")
470
+ .execute();
471
+
472
+ const taskRows = await this.taskDataEntityRepository.find({
473
+ where: {
474
+ mapped_entity_id: lead_id,
475
+ mapped_entity_type: entity_type,
476
+ stage_id
477
+ }
478
+ });
479
+
480
+ for (const task of taskRows) {
481
+ const statusRows = await listMasterItemsRepo.findOne({
482
+ where: {
483
+ id: task.status
484
+ }
485
+ });
486
+
487
+ const statusName = statusRows?.value?.toLowerCase() || '';
488
+ if (
489
+ ['todo', 'in_progress'].includes(statusName) ||
490
+ statusName === 'todo'
491
+ ) {
492
+
493
+ await this.taskDataEntityRepository.update(task.id,{
494
+ user_id: lead_owner,
495
+ task_owner: lead_owner
496
+ });
497
+ }
498
+ }
499
+
500
+ let leadOwnerName;
501
+
502
+ try {
503
+ const baseUrl = this.configService.get<string>('REDIRECT_BE_URL');
504
+
505
+ // Prepare query params
506
+ const queryParams = new URLSearchParams({
507
+ loggedInUser: JSON.stringify(loggedInUser),
508
+ }).toString();
509
+
510
+ const url = `${baseUrl}/entity/public/getById/${entityData?.lead_owner}?entity_type=USR&${queryParams}`;
511
+
512
+ const response = await axios.get(url);
513
+
514
+ leadOwnerName = response.data;
515
+ } catch (error) {
516
+ console.error('Internal Entity API call failed:', error.message);
517
+ }
518
+
519
+ try {
520
+ const logData = {
521
+ mapped_entity_id: lead_id,
522
+ mapped_entity_type: 'LEAD',
523
+ title: 'Owner Assigned',
524
+ description: leadData?.lead_owner
525
+ ? `${leadOwnerName?.name} reassigned as Lead owner.`
526
+ : `${leadOwnerName?.name} assigned as Lead owner.`,
527
+ action: 'assign',
528
+ category: ACTIVITY_CATEGORIES.ASSIGN,
529
+ appcode: loggedInUser.appcode,
530
+ };
531
+
532
+ await this.activityLogService.logActivity(logData, loggedInUser);
533
+ } catch (error) {
534
+ console.error(
535
+ 'Failed to log activity for meeting:',
536
+ error?.message || error,
537
+ );
538
+ // Logging should not block main flow
539
+ }
540
+ return result;
541
+ }
542
+
543
+ async assignLead(
544
+ loggedInUser: UserData,
545
+ mapped_entity_id: number,
546
+ mapped_entity_type: string,
547
+ stage_id: number,
548
+ actions: any[],
549
+ ): Promise<number> {
550
+ const { organization_id, level_id, level_type } = loggedInUser;
551
+
552
+ let owners = [] as any;
553
+
554
+ try {
555
+ const baseUrl = this.configService.get<string>('REDIRECT_BE_URL');
556
+
557
+ // Prepare query params
558
+ const queryParams = new URLSearchParams({
559
+ loggedInUser: JSON.stringify(loggedInUser),
560
+ }).toString();
561
+
562
+ const { level_id } = loggedInUser;
563
+
564
+ const url = `${baseUrl}/users/public/lead-owners/?entity_type=USR&${queryParams}`;
565
+
566
+ const response = await axios.get(url);
567
+
568
+ owners = response.data;
569
+
570
+ // The API response is likely a JSON object, not an array
571
+ } catch (error) {
572
+ console.error('Internal Entity API call failed:', error.message);
573
+ }
574
+
575
+ if (!owners?.length)
576
+ console.log('No eligible owners found for lead assignment.');
577
+ const userIds = owners?.map((o) => Number(o.id)); // normalize to numbers
578
+
579
+ // 2) Find the last assigned *eligible* owner (use IN (...))
580
+ const placeholders = userIds.map(() => '?').join(',');
581
+
582
+ const leadRepo = this.reflectionHelper.getRepoService('CRMLead');
583
+ const lastRow = await leadRepo
584
+ .createQueryBuilder("cl")
585
+ .select("cl.lead_owner", "lead_owner")
586
+ .where("cl.organization_id = :orgId", { orgId: organization_id })
587
+ .andWhere("cl.level_id = :levelId", { levelId: Number(level_id) })
588
+ .andWhere("cl.level_type = :levelType", { levelType: level_type })
589
+ .andWhere("cl.lead_owner IN (:...owners)", { owners: userIds })
590
+ .orderBy("cl.created_date", "DESC")
591
+ .limit(1)
592
+ .getRawOne();
593
+
594
+ const lastAssigned = lastRow.length ? Number(lastRow[0].lead_owner) : null;
595
+
596
+ // 3) Compute next user in round-robin
597
+ const lastIdx = lastAssigned != null ? userIds.indexOf(lastAssigned) : -1;
598
+ const nextIdx = (lastIdx + 1) % userIds.length;
599
+ const nextUser = userIds[nextIdx];
600
+
601
+ //4) Update lead owner in the lead table no need because we are updating lead owner in updateLeadOwner method
602
+ console.log(`Assigning lead to user ID: ${nextUser}`);
603
+
604
+ const firstAction = actions[0];
605
+
606
+ // update lead owner status
607
+ await this.updateLeadOwner(
608
+ {
609
+ lead_id: mapped_entity_id,
610
+ lead_owner: nextUser,
611
+ stage_id: stage_id,
612
+ entity_type: mapped_entity_type,
613
+ action_id: firstAction.id,
614
+ },
615
+ loggedInUser,
616
+ );
617
+
618
+ // move task
619
+ await this.taskService.moveTask(loggedInUser, {
620
+ mapped_entity_type,
621
+ mapped_entity_id,
622
+ stage_id,
623
+ action_id: firstAction.id,
624
+ });
625
+
626
+ // if it has only one action then move next stage
627
+ if (actions.length == 1) {
628
+ console.log(
629
+ 'Only one action present and it is owner assignment. Moving to next stage.',
630
+ );
631
+ await this.moveToNextStage(
632
+ mapped_entity_type,
633
+ mapped_entity_id,
634
+ loggedInUser,
635
+ );
636
+ }
637
+
638
+ return nextUser;
639
+ }
640
+ }