rez_core 5.0.29 → 5.0.30

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 (427) hide show
  1. package/.claude/settings.local.json +26 -0
  2. package/.idea/250218_nodejs_core.iml +9 -0
  3. package/.idea/codeStyles/Project.xml +59 -0
  4. package/.idea/codeStyles/codeStyleConfig.xml +5 -0
  5. package/.idea/copilot.data.migration.agent.xml +6 -0
  6. package/.idea/copilot.data.migration.ask.xml +6 -0
  7. package/.idea/copilot.data.migration.ask2agent.xml +6 -0
  8. package/.idea/copilot.data.migration.edit.xml +6 -0
  9. package/.idea/inspectionProfiles/Project_Default.xml +6 -0
  10. package/.idea/misc.xml +6 -0
  11. package/.idea/modules.xml +8 -0
  12. package/.idea/prettier.xml +6 -0
  13. package/.idea/vcs.xml +6 -0
  14. package/.prettierrc +3 -3
  15. package/README.md +99 -99
  16. package/dist/module/auth/guards/role.guard.js +3 -3
  17. package/dist/module/auth/services/auth.service.js +2 -2
  18. package/dist/module/filter/repository/saved-filter.repository.js +4 -4
  19. package/dist/module/filter/service/filter-evaluator.service.js +2 -2
  20. package/dist/module/filter/service/filter.service.js +22 -22
  21. package/dist/module/integration/examples/usage.example.js +9 -9
  22. package/dist/module/integration/service/integration.service.js +1 -1
  23. package/dist/module/integration/service/wrapper.service.js +26 -26
  24. package/dist/module/listmaster/service/list-master-item.service.js +2 -2
  25. package/dist/module/listmaster/service/list-master.service.js +0 -9
  26. package/dist/module/listmaster/service/list-master.service.js.map +1 -1
  27. package/dist/module/mapper/service/field-mapper.service.js +4 -4
  28. package/dist/module/mapper/service/mapper.service.js +2 -2
  29. package/dist/module/meta/repository/entity-attribute-update.repository.js +3 -3
  30. package/dist/module/meta/repository/entity-master.repository.js +6 -6
  31. package/dist/module/meta/service/entity-dynamic.service.js +44 -44
  32. package/dist/module/meta/service/entity-list.service.js +3 -3
  33. package/dist/module/meta/service/entity-master.service.js +3 -3
  34. package/dist/module/meta/service/entity-relation.service.js +9 -9
  35. package/dist/module/meta/service/entity-service-impl.service.js +3 -3
  36. package/dist/module/meta/service/resolver.service.js +3 -3
  37. package/dist/module/module/repository/menu.repository.js +12 -12
  38. package/dist/module/notification/service/notification.service.js +10 -10
  39. package/dist/module/user/controller/login.controller.js +18 -18
  40. package/dist/module/user/service/role.service.js +4 -4
  41. package/dist/module/user/service/user-session.service.js +2 -2
  42. package/dist/module/workflow/repository/action.repository.js +21 -20
  43. package/dist/module/workflow/repository/action.repository.js.map +1 -1
  44. package/dist/module/workflow/repository/comm-template.repository.js +6 -6
  45. package/dist/module/workflow/repository/form-master.repository.js +2 -2
  46. package/dist/module/workflow/repository/stage-group.repository.js +20 -20
  47. package/dist/module/workflow/repository/stage-movement.repository.js +17 -17
  48. package/dist/module/workflow/repository/stage.repository.js +8 -8
  49. package/dist/module/workflow/service/action-template-mapping.service.js +31 -31
  50. package/dist/module/workflow/service/action.service.js +7 -7
  51. package/dist/module/workflow/service/entity-modification.service.js +2 -2
  52. package/dist/module/workflow/service/stage-group.service.js +2 -2
  53. package/dist/module/workflow/service/stage.service.js +2 -2
  54. package/dist/module/workflow/service/task.service.js +35 -35
  55. package/dist/module/workflow/service/workflow-list-master.service.js +15 -15
  56. package/dist/module/workflow/service/workflow-meta.service.js +26 -26
  57. package/dist/module/workflow/service/workflow.service.js +2 -2
  58. package/dist/module/workflow-automation/service/schedule-handler.service.js +9 -9
  59. package/dist/module/workflow-schedule/processors/schedule.processor.js +4 -4
  60. package/dist/module/workflow-schedule/service/workflow-schedule.service.js +9 -9
  61. package/dist/tsconfig.build.tsbuildinfo +1 -1
  62. package/dist/utils/service/reflection-helper.service.js +2 -2
  63. package/docs/modules/event-driven-integration-design.md +91 -91
  64. package/docs/modules/integration.md +250 -250
  65. package/eslint.config.mjs +34 -34
  66. package/nest-cli.json +14 -14
  67. package/package.json +124 -124
  68. package/server.log +850 -0
  69. package/src/app.controller.ts +12 -12
  70. package/src/app.module.ts +66 -66
  71. package/src/app.service.ts +8 -8
  72. package/src/config/bull.config.ts +69 -69
  73. package/src/config/config.module.ts +17 -17
  74. package/src/config/database.config.ts +48 -48
  75. package/src/constant/global.constant.ts +67 -67
  76. package/src/core.module.ts +88 -88
  77. package/src/decorators/roles.decorator.ts +7 -7
  78. package/src/dtos/response.dto.ts +6 -6
  79. package/src/dtos/response.ts +5 -5
  80. package/src/index.ts +1 -1
  81. package/src/module/auth/auth.module.ts +49 -49
  82. package/src/module/auth/controller/auth.controller.ts +28 -28
  83. package/src/module/auth/guards/google-auth.guard.ts +9 -9
  84. package/src/module/auth/guards/jwt.guard.ts +22 -22
  85. package/src/module/auth/guards/role.guard.ts +68 -68
  86. package/src/module/auth/services/auth.service.ts +50 -50
  87. package/src/module/auth/services/jwt.service.ts +11 -11
  88. package/src/module/auth/strategies/google.strategy.ts +54 -54
  89. package/src/module/auth/strategies/jwt.strategy.ts +58 -58
  90. package/src/module/auth/strategies/local.strategy.ts +13 -13
  91. package/src/module/dashboard/controller/dashboard.controller.ts +36 -36
  92. package/src/module/dashboard/dashboard.module.ts +21 -21
  93. package/src/module/dashboard/entity/dashboard_page_data.entity.ts +27 -27
  94. package/src/module/dashboard/entity/widget_master.entity.ts +18 -18
  95. package/src/module/dashboard/repository/dashboard.repository.ts +42 -42
  96. package/src/module/dashboard/service/dashboard.service.ts +73 -73
  97. package/src/module/dev/dev.module.ts +12 -12
  98. package/src/module/dev/service/dev.service.ts +7 -7
  99. package/src/module/enterprise/controller/organization.controller.ts +36 -36
  100. package/src/module/enterprise/enterprise.module.ts +30 -30
  101. package/src/module/enterprise/entity/enterprise.entity.ts +37 -37
  102. package/src/module/enterprise/entity/organization-app-mapping.entity.ts +13 -13
  103. package/src/module/enterprise/entity/organization.entity.ts +92 -92
  104. package/src/module/enterprise/repository/enterprise.repository.ts +31 -31
  105. package/src/module/enterprise/repository/organization.repository.ts +26 -26
  106. package/src/module/enterprise/repository/school.repository.ts +278 -278
  107. package/src/module/enterprise/service/brand.service.ts +5 -5
  108. package/src/module/enterprise/service/enterprise.service.ts +16 -16
  109. package/src/module/enterprise/service/organization-app-mapping.service.ts +4 -4
  110. package/src/module/enterprise/service/organization.service.ts +145 -145
  111. package/src/module/entity_json/controller/entity_json.controller.ts +47 -47
  112. package/src/module/entity_json/entity/entityJson.entity.ts +39 -39
  113. package/src/module/entity_json/entity_json.module.ts +18 -18
  114. package/src/module/entity_json/service/entityJson.repository.ts +37 -37
  115. package/src/module/entity_json/service/entity_json.service.ts +234 -234
  116. package/src/module/filter/controller/filter.controller.ts +84 -84
  117. package/src/module/filter/dto/filter-request.dto.ts +38 -38
  118. package/src/module/filter/entity/saved-filter-detail.entity.ts +41 -41
  119. package/src/module/filter/entity/saved-filter-master.entity.ts +23 -23
  120. package/src/module/filter/filter.module.ts +31 -31
  121. package/src/module/filter/repository/saved-filter.repository.ts +168 -168
  122. package/src/module/filter/service/filter-evaluator.service.ts +86 -86
  123. package/src/module/filter/service/filter.service.ts +1144 -1144
  124. package/src/module/filter/service/saved-filter.service.ts +170 -170
  125. package/src/module/ics/controller/ics.controller.ts +21 -21
  126. package/src/module/ics/dto/ics.dto.ts +55 -55
  127. package/src/module/ics/ics.module.ts +13 -13
  128. package/src/module/ics/service/ics.service.ts +57 -57
  129. package/src/module/integration/controller/calender-event.controller.ts +31 -31
  130. package/src/module/integration/controller/integration.controller.ts +662 -662
  131. package/src/module/integration/controller/wrapper.controller.ts +37 -37
  132. package/src/module/integration/dto/create-config.dto.ts +526 -526
  133. package/src/module/integration/entity/integration-config.entity.ts +112 -112
  134. package/src/module/integration/entity/integration-entity-mapper.entity.ts +14 -14
  135. package/src/module/integration/entity/integration-source.entity.ts +17 -17
  136. package/src/module/integration/entity/user-integration.entity.ts +71 -71
  137. package/src/module/integration/examples/usage.example.ts +338 -338
  138. package/src/module/integration/factories/base.factory.ts +7 -7
  139. package/src/module/integration/factories/email.factory.ts +49 -49
  140. package/src/module/integration/factories/integration.factory.ts +121 -121
  141. package/src/module/integration/factories/sms.factory.ts +51 -51
  142. package/src/module/integration/factories/telephone.factory.ts +41 -41
  143. package/src/module/integration/factories/whatsapp.factory.ts +56 -56
  144. package/src/module/integration/integration.module.ts +110 -110
  145. package/src/module/integration/service/calendar-event.service.ts +118 -118
  146. package/src/module/integration/service/integration-entity-mapper.service.ts +17 -17
  147. package/src/module/integration/service/integration-queue.service.ts +229 -229
  148. package/src/module/integration/service/integration.service.ts +2633 -2633
  149. package/src/module/integration/service/oauth.service.ts +224 -224
  150. package/src/module/integration/service/wrapper.service.ts +701 -701
  151. package/src/module/integration/strategies/email/gmail-api.strategy.ts +280 -280
  152. package/src/module/integration/strategies/email/outlook-api.strategy.ts +44 -44
  153. package/src/module/integration/strategies/email/outlook.strategy.ts +64 -64
  154. package/src/module/integration/strategies/email/sendgrid-api.strategy.ts +260 -260
  155. package/src/module/integration/strategies/integration.strategy.ts +97 -97
  156. package/src/module/integration/strategies/sms/gupshup-sms.strategy.ts +146 -146
  157. package/src/module/integration/strategies/sms/msg91-sms.strategy.ts +164 -164
  158. package/src/module/integration/strategies/sms/tubelight-sms.strategy.ts +163 -163
  159. package/src/module/integration/strategies/telephone/ozonetel-voice.strategy.ts +238 -238
  160. package/src/module/integration/strategies/telephone/tubelight-voice.strategy.ts +210 -210
  161. package/src/module/integration/strategies/whatsapp/gupshup-whatsapp.strategy.ts +359 -359
  162. package/src/module/integration/strategies/whatsapp/tubelight-whatsapp.strategy.ts +372 -372
  163. package/src/module/integration/strategies/whatsapp/whatsapp-cloud.strategy.ts +403 -403
  164. package/src/module/integration/strategies/whatsapp/whatsapp.strategy.ts +57 -57
  165. package/src/module/layout/controller/layout.controller.ts +47 -47
  166. package/src/module/layout/entity/header-items.entity.ts +28 -28
  167. package/src/module/layout/entity/header-section.entity.ts +19 -19
  168. package/src/module/layout/layout.module.ts +21 -21
  169. package/src/module/layout/repository/header-items.repository.ts +18 -18
  170. package/src/module/layout/repository/header-section.repository.ts +22 -22
  171. package/src/module/layout/service/header-section.service.ts +25 -25
  172. package/src/module/layout_preference/controller/layout_preference.controller.ts +73 -73
  173. package/src/module/layout_preference/entity/layout_preference.entity.ts +28 -28
  174. package/src/module/layout_preference/layout_preference.module.ts +22 -22
  175. package/src/module/layout_preference/repository/layout_preference.repository.ts +65 -65
  176. package/src/module/layout_preference/service/layout_preference.service.ts +191 -191
  177. package/src/module/lead/controller/lead.controller.ts +30 -30
  178. package/src/module/lead/lead.module.ts +14 -14
  179. package/src/module/lead/repository/lead.repository.ts +41 -41
  180. package/src/module/lead/service/lead.service.ts +54 -54
  181. package/src/module/linked_attributes/controller/linked_attributes.controller.ts +34 -34
  182. package/src/module/linked_attributes/entity/linked_attribute.entity.ts +51 -51
  183. package/src/module/linked_attributes/linked_attributes.module.ts +16 -16
  184. package/src/module/linked_attributes/repository/linked_attribute.repository.ts +12 -12
  185. package/src/module/linked_attributes/service/linked_attributes.service.ts +102 -102
  186. package/src/module/listmaster/controller/list-master.controller.ts +230 -230
  187. package/src/module/listmaster/entity/list-master-items.entity.ts +43 -43
  188. package/src/module/listmaster/entity/list-master.entity.ts +33 -33
  189. package/src/module/listmaster/listmaster.module.ts +46 -46
  190. package/src/module/listmaster/repository/list-master-items.repository.ts +173 -173
  191. package/src/module/listmaster/repository/list-master.repository.ts +46 -46
  192. package/src/module/listmaster/service/list-master-engine.ts +19 -19
  193. package/src/module/listmaster/service/list-master-extension.interface.ts +4 -4
  194. package/src/module/listmaster/service/list-master-item.service.ts +292 -292
  195. package/src/module/listmaster/service/list-master-registry.ts +15 -15
  196. package/src/module/listmaster/service/list-master.service.ts +544 -558
  197. package/src/module/mapper/controller/field-mapper.controller.ts +76 -76
  198. package/src/module/mapper/controller/mapper.controller.ts +20 -20
  199. package/src/module/mapper/dto/field-mapper.dto.ts +14 -14
  200. package/src/module/mapper/entity/field-lovs.entity.ts +19 -19
  201. package/src/module/mapper/entity/field-mapper.entity.ts +53 -53
  202. package/src/module/mapper/entity/mapper.entity.ts +16 -16
  203. package/src/module/mapper/mapper.module.ts +34 -34
  204. package/src/module/mapper/repository/field-lovs.repository.ts +35 -35
  205. package/src/module/mapper/repository/field-mapper.repository.ts +42 -42
  206. package/src/module/mapper/repository/mapper.repository.ts +15 -15
  207. package/src/module/mapper/service/field-mapper.service.ts +271 -271
  208. package/src/module/mapper/service/mapper.service.ts +79 -79
  209. package/src/module/master/controller/master.controller.ts +74 -74
  210. package/src/module/master/service/master.service.ts +483 -483
  211. package/src/module/meta/controller/app-master.controller.ts +38 -38
  212. package/src/module/meta/controller/attribute-master.controller.ts +84 -84
  213. package/src/module/meta/controller/entity-dynamic.controller.ts +125 -125
  214. package/src/module/meta/controller/entity-master.controller.ts +41 -41
  215. package/src/module/meta/controller/entity-relation.controller.ts +36 -36
  216. package/src/module/meta/controller/entity.controller.ts +392 -392
  217. package/src/module/meta/controller/entity.public.controller.ts +75 -75
  218. package/src/module/meta/controller/media.controller.ts +107 -107
  219. package/src/module/meta/controller/meta.controller.ts +96 -96
  220. package/src/module/meta/controller/view-master.controller.ts +86 -86
  221. package/src/module/meta/dto/entity-list-data.dto.ts +6 -6
  222. package/src/module/meta/dto/entity-tab.dto.ts +4 -4
  223. package/src/module/meta/dto/entity-table.dto.ts +9 -9
  224. package/src/module/meta/entity/app-master.entity.ts +34 -34
  225. package/src/module/meta/entity/attribute-master.entity.ts +92 -92
  226. package/src/module/meta/entity/base-entity.entity.ts +75 -75
  227. package/src/module/meta/entity/entity-master.entity.ts +85 -85
  228. package/src/module/meta/entity/entity-relation-data.entity.ts +29 -29
  229. package/src/module/meta/entity/entity-relation.entity.ts +23 -23
  230. package/src/module/meta/entity/entity-table-column.entity.ts +61 -61
  231. package/src/module/meta/entity/entity-table.entity.ts +50 -50
  232. package/src/module/meta/entity/media-data.entity.ts +32 -32
  233. package/src/module/meta/entity/preference.entity.ts +62 -62
  234. package/src/module/meta/entity/view-master.entity.ts +41 -41
  235. package/src/module/meta/entity.module.ts +166 -166
  236. package/src/module/meta/repository/app-master.repository.ts +20 -20
  237. package/src/module/meta/repository/attribute-master.repository.ts +138 -138
  238. package/src/module/meta/repository/entity-attribute-update.repository.ts +44 -44
  239. package/src/module/meta/repository/entity-master.repository.ts +111 -111
  240. package/src/module/meta/repository/entity-table-column.repository.ts +39 -39
  241. package/src/module/meta/repository/entity-table.repository.ts +53 -53
  242. package/src/module/meta/repository/media-data.repository.ts +50 -50
  243. package/src/module/meta/repository/preference.repository.ts +20 -20
  244. package/src/module/meta/repository/user-app-mapping.repository.ts +28 -28
  245. package/src/module/meta/repository/view-master.repository.ts +42 -42
  246. package/src/module/meta/service/app-master.service.ts +37 -37
  247. package/src/module/meta/service/attribute-master.service.ts +130 -130
  248. package/src/module/meta/service/common.service.ts +9 -9
  249. package/src/module/meta/service/entity-attribute-update.service.ts +28 -28
  250. package/src/module/meta/service/entity-dynamic.service.ts +809 -809
  251. package/src/module/meta/service/entity-list.service.ts +205 -205
  252. package/src/module/meta/service/entity-master.service.ts +175 -175
  253. package/src/module/meta/service/entity-realation-data.service.ts +9 -9
  254. package/src/module/meta/service/entity-relation.service.ts +87 -87
  255. package/src/module/meta/service/entity-service-impl.service.ts +525 -525
  256. package/src/module/meta/service/entity-table-column.service.ts +39 -39
  257. package/src/module/meta/service/entity-table.service.ts +150 -150
  258. package/src/module/meta/service/entity-validation.service.ts +187 -187
  259. package/src/module/meta/service/entity.service.ts +67 -67
  260. package/src/module/meta/service/field-group.service.ts +103 -103
  261. package/src/module/meta/service/media-data.service.ts +507 -507
  262. package/src/module/meta/service/populate-meta.service.ts +193 -193
  263. package/src/module/meta/service/preference.service.ts +16 -16
  264. package/src/module/meta/service/resolver.service.ts +271 -271
  265. package/src/module/meta/service/section-master.service.ts +104 -104
  266. package/src/module/meta/service/update-form-json.service.ts +22 -22
  267. package/src/module/meta/service/user-app-mapping.service.ts +17 -17
  268. package/src/module/meta/service/view-master.service.ts +127 -127
  269. package/src/module/microservice-client/microservice-clients.module.ts +13 -13
  270. package/src/module/microservice-client/service/microservice-client-factory.ts +37 -37
  271. package/src/module/microservice-client/service/microservice-clients.ts +4 -4
  272. package/src/module/module/controller/menu.controller.ts +15 -15
  273. package/src/module/module/controller/module-access.controller.ts +134 -134
  274. package/src/module/module/entity/menu.entity.ts +43 -43
  275. package/src/module/module/entity/module-access.entity.ts +25 -25
  276. package/src/module/module/entity/module-action.entity.ts +17 -17
  277. package/src/module/module/entity/module.entity.ts +52 -52
  278. package/src/module/module/module.module.ts +42 -42
  279. package/src/module/module/repository/menu.repository.ts +184 -184
  280. package/src/module/module/repository/module-access.repository.ts +344 -344
  281. package/src/module/module/service/menu.service.ts +82 -82
  282. package/src/module/module/service/module-access.service.ts +194 -194
  283. package/src/module/notification/controller/notification.controller.ts +58 -58
  284. package/src/module/notification/controller/otp.controller.ts +117 -117
  285. package/src/module/notification/entity/notification.entity.ts +26 -26
  286. package/src/module/notification/entity/otp.entity.ts +28 -28
  287. package/src/module/notification/firebase-admin.config.ts +22 -22
  288. package/src/module/notification/notification.module.ts +69 -69
  289. package/src/module/notification/repository/otp.repository.ts +27 -27
  290. package/src/module/notification/service/email.service.ts +127 -127
  291. package/src/module/notification/service/notification.service.ts +160 -160
  292. package/src/module/notification/service/otp.service.ts +133 -133
  293. package/src/module/third-party-module/entity/third-party-api-registry.entity.ts +52 -52
  294. package/src/module/third-party-module/repository/third-party-api-registry.repository.ts +20 -20
  295. package/src/module/third-party-module/service/api-registry.service.ts +13 -13
  296. package/src/module/third-party-module/third-party.module.ts +12 -12
  297. package/src/module/user/controller/login.controller.ts +198 -198
  298. package/src/module/user/controller/user.controller.ts +40 -40
  299. package/src/module/user/dto/create-user.dto.ts +62 -62
  300. package/src/module/user/dto/update-user.dto.ts +4 -4
  301. package/src/module/user/entity/role.entity.ts +33 -33
  302. package/src/module/user/entity/user-role-mapping.entity.ts +38 -38
  303. package/src/module/user/entity/user-session.entity.ts +73 -73
  304. package/src/module/user/entity/user.entity.ts +59 -59
  305. package/src/module/user/repository/role.repository.ts +96 -96
  306. package/src/module/user/repository/user-role-mapping.repository.ts +126 -126
  307. package/src/module/user/repository/user.repository.ts +50 -50
  308. package/src/module/user/repository/userSession.repository.ts +33 -33
  309. package/src/module/user/service/login.service.ts +326 -326
  310. package/src/module/user/service/role.service.ts +189 -189
  311. package/src/module/user/service/user-role-mapping.service.ts +98 -98
  312. package/src/module/user/service/user-session.service.ts +168 -168
  313. package/src/module/user/service/user.service.ts +368 -368
  314. package/src/module/user/user.module.ts +65 -65
  315. package/src/module/workflow/controller/action-category.controller.ts +54 -54
  316. package/src/module/workflow/controller/action-resource-mapping.controller.ts +23 -23
  317. package/src/module/workflow/controller/action-template-mapping.controller.ts +35 -35
  318. package/src/module/workflow/controller/action.controller.ts +111 -111
  319. package/src/module/workflow/controller/activity-log.controller.ts +55 -55
  320. package/src/module/workflow/controller/comm-template.controller.ts +43 -43
  321. package/src/module/workflow/controller/entity-modification.controller.ts +35 -35
  322. package/src/module/workflow/controller/form-master.controller.ts +43 -43
  323. package/src/module/workflow/controller/stage-group.controller.ts +48 -48
  324. package/src/module/workflow/controller/stage.controller.ts +50 -50
  325. package/src/module/workflow/controller/task.controller.ts +77 -77
  326. package/src/module/workflow/controller/workflow-list-master.controller.ts +44 -44
  327. package/src/module/workflow/controller/workflow-meta.controller.ts +80 -80
  328. package/src/module/workflow/controller/workflow.controller.ts +67 -67
  329. package/src/module/workflow/entity/action-category.entity.ts +38 -38
  330. package/src/module/workflow/entity/action-data.entity.ts +55 -55
  331. package/src/module/workflow/entity/action-resources-mapping.entity.ts +29 -29
  332. package/src/module/workflow/entity/action-template-mapping.entity.ts +17 -17
  333. package/src/module/workflow/entity/action.entity.ts +53 -53
  334. package/src/module/workflow/entity/activity-log.entity.ts +43 -43
  335. package/src/module/workflow/entity/comm-template.entity.ts +43 -43
  336. package/src/module/workflow/entity/entity-modification.entity.ts +38 -38
  337. package/src/module/workflow/entity/form.entity.ts +25 -25
  338. package/src/module/workflow/entity/stage-action-mapping.entity.ts +17 -17
  339. package/src/module/workflow/entity/stage-group.entity.ts +23 -23
  340. package/src/module/workflow/entity/stage-movement-data.entity.ts +38 -38
  341. package/src/module/workflow/entity/stage.entity.ts +20 -20
  342. package/src/module/workflow/entity/task-data.entity.ts +88 -88
  343. package/src/module/workflow/entity/template-attach-mapper.entity.ts +30 -30
  344. package/src/module/workflow/entity/workflow-data.entity.ts +11 -11
  345. package/src/module/workflow/entity/workflow-level-mapping.entity.ts +18 -18
  346. package/src/module/workflow/entity/workflow.entity.ts +20 -20
  347. package/src/module/workflow/repository/action-category.repository.ts +79 -79
  348. package/src/module/workflow/repository/action-data.repository.ts +334 -334
  349. package/src/module/workflow/repository/action.repository.ts +324 -323
  350. package/src/module/workflow/repository/activity-log.repository.ts +148 -148
  351. package/src/module/workflow/repository/comm-template.repository.ts +149 -149
  352. package/src/module/workflow/repository/form-master.repository.ts +59 -59
  353. package/src/module/workflow/repository/stage-group.repository.ts +197 -197
  354. package/src/module/workflow/repository/stage-movement.repository.ts +246 -246
  355. package/src/module/workflow/repository/stage.repository.ts +172 -172
  356. package/src/module/workflow/repository/task.repository.ts +134 -134
  357. package/src/module/workflow/repository/workflow.repository.ts +42 -42
  358. package/src/module/workflow/service/action-category.service.ts +33 -33
  359. package/src/module/workflow/service/action-data.service.ts +62 -62
  360. package/src/module/workflow/service/action-resources-mapping.service.ts +10 -10
  361. package/src/module/workflow/service/action-template-mapping.service.ts +115 -115
  362. package/src/module/workflow/service/action.service.ts +279 -279
  363. package/src/module/workflow/service/activity-log.service.ts +107 -107
  364. package/src/module/workflow/service/comm-template.service.ts +180 -180
  365. package/src/module/workflow/service/entity-modification.service.ts +61 -61
  366. package/src/module/workflow/service/form-master.service.ts +35 -35
  367. package/src/module/workflow/service/populate-workflow.service.ts +303 -303
  368. package/src/module/workflow/service/stage-action-mapping.service.ts +5 -5
  369. package/src/module/workflow/service/stage-group.service.ts +342 -342
  370. package/src/module/workflow/service/stage.service.ts +199 -199
  371. package/src/module/workflow/service/task.service.ts +558 -558
  372. package/src/module/workflow/service/workflow-list-master.service.ts +60 -60
  373. package/src/module/workflow/service/workflow-meta.service.ts +654 -654
  374. package/src/module/workflow/service/workflow.service.ts +205 -205
  375. package/src/module/workflow/workflow.module.ts +178 -178
  376. package/src/module/workflow-automation/SCHEDULING_GUIDE.md +145 -145
  377. package/src/module/workflow-automation/controller/workflow-automation.controller.ts +43 -43
  378. package/src/module/workflow-automation/entity/workflow-automation-action.entity.ts +26 -26
  379. package/src/module/workflow-automation/entity/workflow-automation.entity.ts +40 -40
  380. package/src/module/workflow-automation/interface/action.decorator.ts +7 -7
  381. package/src/module/workflow-automation/interface/action.interface.ts +5 -5
  382. package/src/module/workflow-automation/service/action-registery.service.ts +35 -35
  383. package/src/module/workflow-automation/service/schedule-handler.service.ts +168 -168
  384. package/src/module/workflow-automation/service/workflow-automation-engine.service.ts +224 -224
  385. package/src/module/workflow-automation/service/workflow-automation.service.ts +515 -515
  386. package/src/module/workflow-automation/workflow-automation.module.ts +54 -54
  387. package/src/module/workflow-schedule/INSTALLATION.md +244 -244
  388. package/src/module/workflow-schedule/MULTI_PROJECT_GUIDE.md +196 -196
  389. package/src/module/workflow-schedule/README.md +422 -422
  390. package/src/module/workflow-schedule/constants/schedule.constants.ts +48 -48
  391. package/src/module/workflow-schedule/controller/workflow-schedule.controller.ts +255 -255
  392. package/src/module/workflow-schedule/docs/CLAUDE_CODE_GUIDE.md +510 -510
  393. package/src/module/workflow-schedule/docs/CLAUDE_CODE_PROMPT.md +362 -362
  394. package/src/module/workflow-schedule/docs/RUN_CLAUDE_CODE.sh +68 -68
  395. package/src/module/workflow-schedule/dto/create-schedule.dto.ts +147 -147
  396. package/src/module/workflow-schedule/dto/get-execution-logs.dto.ts +119 -119
  397. package/src/module/workflow-schedule/dto/update-schedule.dto.ts +96 -96
  398. package/src/module/workflow-schedule/entities/scheduled-workflow.entity.ts +148 -148
  399. package/src/module/workflow-schedule/entities/workflow-execution-log.entity.ts +154 -154
  400. package/src/module/workflow-schedule/interfaces/schedule-job-data.interface.ts +53 -53
  401. package/src/module/workflow-schedule/interfaces/workflow-schedule-options.interface.ts +12 -12
  402. package/src/module/workflow-schedule/processors/schedule.processor.ts +584 -584
  403. package/src/module/workflow-schedule/service/workflow-schedule.service.ts +600 -600
  404. package/src/module/workflow-schedule/workflow-schedule.module.ts +67 -67
  405. package/src/resources/dev.properties.yaml +31 -31
  406. package/src/resources/local.properties.yaml +27 -27
  407. package/src/resources/properties.module.ts +12 -12
  408. package/src/resources/properties.yaml.ts +11 -11
  409. package/src/resources/uat.properties.yaml +31 -31
  410. package/src/table.config.ts +130 -130
  411. package/src/utils/dto/excel-data.dto.ts +14 -14
  412. package/src/utils/dto/excelsheet-data.dto.ts +5 -5
  413. package/src/utils/service/base64util.service.ts +18 -18
  414. package/src/utils/service/clockIDGenUtil.service.ts +21 -21
  415. package/src/utils/service/codeGenerator.service.ts +22 -22
  416. package/src/utils/service/dateUtil.service.ts +17 -17
  417. package/src/utils/service/encryptUtil.service.ts +97 -97
  418. package/src/utils/service/excel-helper.service.ts +72 -72
  419. package/src/utils/service/excelUtil.service.ts +15 -15
  420. package/src/utils/service/file-util.service.ts +11 -11
  421. package/src/utils/service/json-util.service.ts +23 -23
  422. package/src/utils/service/loggingUtil.service.ts +88 -88
  423. package/src/utils/service/reflection-helper.service.ts +62 -62
  424. package/src/utils/service/wbsCodeGen.service.ts +8 -8
  425. package/src/utils/utils.module.ts +27 -27
  426. package/tsconfig.build.json +4 -4
  427. package/tsconfig.json +24 -24
@@ -1,507 +1,507 @@
1
- import { Injectable } from '@nestjs/common';
2
- import { ConfigService } from '@nestjs/config';
3
- import { S3 } from 'aws-sdk';
4
- import axios from 'axios';
5
- import { DataSource } from 'typeorm';
6
- import { v4 as uuidv4 } from 'uuid';
7
- import {
8
- ENTITYTYPE_MEDIA,
9
- STATUS_PENDING,
10
- } from '../../../constant/global.constant';
11
- import { MediaData } from '../entity/media-data.entity';
12
- import { MediaDataRepository } from '../repository/media-data.repository';
13
- import { EntityServiceImpl } from './entity-service-impl.service';
14
-
15
- @Injectable()
16
- export class MediaDataService extends EntityServiceImpl {
17
- constructor(
18
- private readonly configService: ConfigService,
19
- private readonly mediaRepository: MediaDataRepository,
20
- private dataSource: DataSource,
21
- ) {
22
- super();
23
- }
24
-
25
- s3AccessKeyID = this.configService.get('AWS.S3.AWS_ACCESS_KEY_ID');
26
- s3AccessKeySecret = this.configService.get('AWS.S3.AWS_SECRET_KEY');
27
- s3Region = this.configService.get('AWS.S3.AWS_REGION');
28
- bucketName = this.configService.get<string>('AWS.S3.BUCKET_NAME');
29
-
30
- s3 = new S3({
31
- accessKeyId: this.s3AccessKeyID,
32
- secretAccessKey: this.s3AccessKeySecret,
33
- region: this.s3Region,
34
- signatureVersion: 'v4',
35
- });
36
-
37
- async generateMediaUploadDetails(
38
- fileName: string,
39
- mappedAttributeKey: string,
40
- loggedInUser,
41
- mappedEntityType?: string,
42
- mappedEntityId?: number,
43
- parentId?: number,
44
- parentType?: string,
45
- ) {
46
- if (!fileName || !mappedAttributeKey) {
47
- return null;
48
- }
49
- const ext = fileName.split('.').pop()?.toLowerCase() ?? '';
50
-
51
- const id = uuidv4();
52
-
53
- const s3Path =
54
- (await this.buildUploadPathGeneric(
55
- mappedEntityType || '',
56
- loggedInUser,
57
- mappedEntityId,
58
- parentId,
59
- parentType,
60
- )) || `uploads`;
61
-
62
- const s3Key = `${s3Path}/${id}.${ext}`;
63
-
64
- const uploadUrl = await this.s3.getSignedUrlPromise('putObject', {
65
- Bucket: this.bucketName,
66
- Key: s3Key,
67
- Expires: 60 * 5, // URL valid for 5 mins
68
- ContentType: this.getContentType(ext),
69
- });
70
-
71
- const mediaData = new MediaData();
72
- mediaData.file_name = fileName;
73
- mediaData.mapped_attribute_key = mappedAttributeKey;
74
- mediaData.status = STATUS_PENDING;
75
- if (mappedEntityType) {
76
- mediaData.mapped_entity_type = mappedEntityType;
77
- }
78
- if (mappedEntityId) {
79
- mediaData.mapped_entity_id = mappedEntityId;
80
- }
81
- mediaData.media_url = s3Key;
82
-
83
- //INSERT RECORD IN DOC TABLE
84
- const savedEntity = await super.createEntity(mediaData, loggedInUser);
85
-
86
- if (savedEntity) {
87
- return { id: savedEntity.id, path: s3Key, uploadUrl };
88
- }
89
- return null;
90
- }
91
-
92
- async findByAttributeKeyAndMappedEntityIdAndMappedEntityType(
93
- attributeKey: string,
94
- mappedEntityId: number,
95
- mappedEntityType: string,
96
- ) {
97
- return await this.mediaRepository.findByAttributeKeyAndMappedEntityIdAndMappedEntityType(
98
- attributeKey,
99
- mappedEntityId,
100
- mappedEntityType,
101
- );
102
- }
103
-
104
- async findByMappedEntityIdAndMappedEntityType(
105
- mappedEntityId: number,
106
- mappedEntityType: string,
107
- ) {
108
- return await this.mediaRepository.findByMappedEntityIdAndMappedEntityType(
109
- mappedEntityId,
110
- mappedEntityType,
111
- );
112
- }
113
-
114
- async deleteByAttributeKeyAndMappedEntityIdAndMappedEntityType(
115
- attributeKey: string,
116
- mappedEntityId: number,
117
- mappedEntityType: string,
118
- ) {
119
- return await this.mediaRepository.deleteByAttributeKeyAndMappedEntityIdAndMappedEntityType(
120
- attributeKey,
121
- mappedEntityId,
122
- mappedEntityType,
123
- );
124
- }
125
-
126
- private getContentType(ext: string): string {
127
- const map = {
128
- pdf: 'application/pdf',
129
- jpg: 'image/jpeg',
130
- jpeg: 'image/jpeg',
131
- png: 'image/png',
132
- };
133
- return map[ext] || 'application/octet-stream';
134
- }
135
-
136
- // New method to get the signed URL for the media (download link)
137
- async getMediaDownloadUrl(id: number, loggedInUser, expiresIn?: number) {
138
- try {
139
- const entityData = await this.getEntityData(
140
- ENTITYTYPE_MEDIA,
141
- id,
142
- loggedInUser,
143
- );
144
- const mediaData = entityData as MediaData;
145
- let fileSize: number | undefined;
146
- if (this.bucketName) {
147
- const head = await this.s3
148
- .headObject({
149
- Bucket: this.bucketName,
150
- Key: mediaData.media_url,
151
- })
152
- .promise();
153
- fileSize = head.ContentLength;
154
- }
155
-
156
- const signedUrl = await this.s3.getSignedUrlPromise('getObject', {
157
- Bucket: this.bucketName,
158
- Key: mediaData.media_url,
159
- Expires: Number(expiresIn) || 60 * 5,
160
- });
161
- return {
162
- signedUrl,
163
- fileName: mediaData.file_name,
164
- id: mediaData.id,
165
- size: fileSize,
166
- };
167
- } catch (error) {
168
- console.error('Error generating signed URL for media:', error);
169
- throw new Error('Failed to generate signed URL');
170
- }
171
- }
172
-
173
- public async buildUploadPathGenericdd(
174
- mappedEntityType: string,
175
- loggedInUser,
176
- mappedEntityId?: number,
177
- parentId?: number,
178
- parentType?: string,
179
- ) {
180
- // 1️⃣ Fetch entity metadata
181
- const entityMaster = await this.entityMasterService.getEntityData(
182
- mappedEntityType,
183
- loggedInUser,
184
- );
185
- if (!entityMaster) {
186
- throw new Error(`Entity master not found for ${mappedEntityType}`);
187
- }
188
-
189
- let pathTemplate = entityMaster.doc_upload_path ?? '';
190
- const entityOverwrite = entityMaster.overwrite_path ?? 0;
191
-
192
- if (!pathTemplate) {
193
- throw new Error(`doc_upload_path not defined for ${mappedEntityType}`);
194
- }
195
-
196
- // 2️⃣ Prepare replacement map
197
- const replacements: Record<string, string | null> = {};
198
-
199
- // --- ORG ---
200
- let organizationData = await this.dataSource.query(
201
- 'SELECT * FROM sso_organization WHERE id = $1',
202
- [loggedInUser.organization_id],
203
- );
204
- replacements['org_code'] = organizationData[0].code;
205
-
206
- // --- LEVEL CODE ---
207
- // Priority: If parent is provided, use parent for level_code; otherwise use normal logic
208
- let levelEntityId: number;
209
- let levelEntityType: string;
210
-
211
- if (parentId && parentType) {
212
- // When parent is provided, parent becomes the level_code
213
- levelEntityId = parentId;
214
- levelEntityType = parentType;
215
- } else {
216
- // Normal logic: determine which entity to use for level_code
217
- levelEntityId = entityOverwrite ? mappedEntityId : loggedInUser.level_id;
218
- levelEntityType = entityOverwrite
219
- ? mappedEntityType
220
- : loggedInUser.level_type;
221
- }
222
-
223
- // Get entity data for level_code
224
- const getEntityData = await this.dataSource.query(
225
- `SELECT * FROM cr_entity_master WHERE mapped_entity_type = $1 and organization_id = $2`,
226
- [levelEntityType, loggedInUser.organization_id],
227
- );
228
-
229
- if (getEntityData.length == 0) return null;
230
- const levelEntityDataResult = await this.dataSource.query(
231
- `SELECT * FROM ${getEntityData[0]?.db_table_name} WHERE id = $1`,
232
- [levelEntityId],
233
- );
234
-
235
- const levelEntityData = levelEntityDataResult[0] || null;
236
-
237
- if (levelEntityData?.code) {
238
- replacements['level_code'] = levelEntityData.code; // universal alias
239
- }
240
-
241
- // --- MAPPED ENTITY CODE (when parent is provided) ---
242
- // When parent is provided, the mapped entity becomes an additional level
243
- if (parentId && parentType && mappedEntityId && mappedEntityType) {
244
- const mappedEntityKey = `${mappedEntityType.toLowerCase()}_code`;
245
- const mappedEntityData = await super.getEntityData(
246
- mappedEntityType,
247
- mappedEntityId,
248
- loggedInUser,
249
- );
250
- if (mappedEntityData?.code) {
251
- replacements[mappedEntityKey] = mappedEntityData.code;
252
-
253
- // Modify path template to include mapped entity
254
- // Example: ${org_code}/${level_code} becomes ${org_code}/${level_code}/${tem_code}
255
- const mappedVariable = `\${${mappedEntityKey}}`;
256
- if (!pathTemplate.includes(mappedVariable)) {
257
- pathTemplate = pathTemplate + `/${mappedVariable}`;
258
- }
259
- }
260
- }
261
-
262
- // --- TEMPLATE CODE (if parent is provided) ---
263
- // Special handling for template_code when parent_id and parent_type are provided
264
- if (parentId && parentType) {
265
- // const templateEntityData = await super.getEntityData(
266
- // parentType,
267
- // parentId,
268
- // loggedInUser,
269
- // );
270
-
271
- // Get entity data for level_code
272
- const getEntityData = await this.dataSource.query(
273
- `SELECT * FROM cr_entity_master WHERE mapped_entity_type = $1 and organization_id = $2`,
274
- [mappedEntityType, loggedInUser.organization_id],
275
- );
276
-
277
- if (getEntityData.length == 0) return null;
278
- const levelEntityDataResult = await this.dataSource.query(
279
- `SELECT * FROM ${getEntityData[0]?.db_table_name} WHERE id = $1`,
280
- [mappedEntityId],
281
- );
282
-
283
- if (
284
- levelEntityDataResult?.[0].code &&
285
- parentType != 'SCH' &&
286
- loggedInUser.level_type != 'SCH'
287
- ) {
288
- replacements['template_code'] = levelEntityDataResult[0].code;
289
- // If we have parent template, modify the path to include it
290
- pathTemplate = '${org_code}/template/${template_code}';
291
- } else {
292
- const getEntityData = await this.dataSource.query(
293
- `SELECT * FROM cr_entity_master WHERE mapped_entity_type = $1 and organization_id = $2`,
294
- [mappedEntityType, loggedInUser.organization_id],
295
- );
296
-
297
- if (getEntityData.length == 0) return null;
298
- const levelEntityDataResult = await this.dataSource.query(
299
- `SELECT * FROM ${getEntityData[0]?.db_table_name} WHERE id = $1`,
300
- [mappedEntityId],
301
- );
302
-
303
- replacements['template_code'] = levelEntityDataResult?.[0].code;
304
-
305
- pathTemplate = '${org_code}/${level_code}/template/${template_code}';
306
- }
307
- }
308
-
309
- // --- Dynamic variables inside path ---
310
- const matches = pathTemplate.match(/\$\{(\w+)\}/g) || [];
311
-
312
- for (const variable of matches) {
313
- const key = variable.replace(/\$\{|\}/g, '');
314
- if (replacements[key]) continue; // already resolved
315
-
316
- if (key === 'org_code') continue; // already set
317
- if (key === 'level_code') continue; // already set above
318
- if (key === 'template_code') continue; // handled above if parent is provided
319
-
320
- if (key.endsWith('_code')) {
321
- // derive entity type dynamically
322
- const entityType = key.replace('_code', '').toUpperCase();
323
-
324
- let entityId: number | undefined;
325
-
326
- // Special handling for template_code - use parent if available, otherwise mapped entity
327
- if (key === 'template_code') {
328
- if (parentType?.toUpperCase() === 'TEMPLATE' && parentId) {
329
- entityId = parentId;
330
- } else if (mappedEntityType === 'TEMPLATE') {
331
- entityId = mappedEntityId;
332
- }
333
- } else {
334
- // For other _code variables, check if parent matches the entity type
335
- if (parentType?.toUpperCase() === entityType && parentId) {
336
- entityId = parentId;
337
- } else if (mappedEntityType === entityType) {
338
- entityId = mappedEntityId;
339
- }
340
- }
341
-
342
- // Generic lookup
343
- if (entityId) {
344
- const entityData = await super.getEntityData(
345
- entityType,
346
- entityId,
347
- loggedInUser,
348
- );
349
- replacements[key] = entityData?.code ?? null;
350
- }
351
- }
352
- }
353
-
354
- // --- ORG special case ---
355
- // If ORG level & overwrite = 0 → use org_code, otherwise use the mapped entity's code
356
- if (loggedInUser.level_type === 'ORG' && entityOverwrite === 0) {
357
- // When overwrite is 0 and user is ORG level, but we want to use mapped entity
358
- // Don't set level_code to null - it should already be set above based on entityOverwrite logic
359
- }
360
-
361
- // 3️⃣ Final substitution and cleanup
362
- if (pathTemplate) {
363
- const finalPath = this.resolveUploadPath(pathTemplate, replacements);
364
- return finalPath;
365
- }
366
- return null;
367
- }
368
-
369
- public async buildUploadPathGeneric(
370
- mappedEntityType: string,
371
- loggedInUser,
372
- mappedEntityId?: number,
373
- parentId?: number,
374
- parentType?: string,
375
- ) {
376
- //APPCODE
377
- let appCode = '';
378
-
379
- let org_id = loggedInUser.organization_id;
380
- let level_type = loggedInUser.level_type;
381
- let level_id =
382
- loggedInUser.level_type == 'ORG' && mappedEntityType == 'SCH'
383
- ? mappedEntityId
384
- : loggedInUser.level_id;
385
-
386
- let organizationData = {};
387
- let levelData = {};
388
- let mappedEntityData = {};
389
-
390
- if (loggedInUser && loggedInUser.appcode) {
391
- appCode = loggedInUser.appcode;
392
- }
393
-
394
- if (appCode && appCode != 'ADM') {
395
- const baseUrl = this.configService.get<string>('REDIRECT_BE_URL');
396
-
397
- // Prepare the query string
398
- const queryParams = new URLSearchParams({
399
- loggedInUser: JSON.stringify(loggedInUser),
400
- }).toString();
401
-
402
- organizationData = await axios
403
- .get(
404
- `${baseUrl}/organization/public/${loggedInUser.organization_id}?${queryParams}`,
405
- )
406
- .then((res) => res.data)
407
- .catch((err) => {
408
- console.error('Error fetching organization data:', err.message);
409
- return null;
410
- });
411
-
412
- levelData = await axios
413
- .get(`${baseUrl}/school/public/?${queryParams}`)
414
- .then((res) => res.data)
415
- .catch((err) => {
416
- console.error('Error fetching school data:', err.message);
417
- return null;
418
- });
419
- } else {
420
- organizationData = await this.dataSource.query(
421
- 'SELECT * FROM adm_org_profile WHERE id = $1',
422
- [loggedInUser.organization_id],
423
- );
424
- levelData = await this.dataSource.query(
425
- 'SELECT * FROM adm_school_profile WHERE id = $1',
426
- [loggedInUser.level_id],
427
- );
428
- }
429
-
430
- console.log(levelData);
431
-
432
- // 1️⃣ Fetch entity metadata
433
- // const uploadEntity = await this.entityMasterService.getEntityData(
434
- // mappedEntityType,
435
- // loggedInUser,
436
- // );
437
- // if (!uploadEntity) {
438
- // throw new Error(`Entity master not found for ${mappedEntityType}`);
439
- // }
440
-
441
- let subfolder =
442
- mappedEntityType == level_type
443
- ? 'data'
444
- : parentType
445
- ? `${parentType}/${parentId}/${mappedEntityType}`
446
- : mappedEntityType;
447
- let path =
448
- '${org_code}/${level_code}/' +
449
- (subfolder !== 'data'
450
- ? `${subfolder}/${mappedEntityId}`
451
- : `${subfolder}`);
452
-
453
- // 2️⃣ Prepare replacement map
454
- const replacements: Record<string, string | null> = {};
455
- replacements['org_code'] = organizationData[0]?.code;
456
- replacements['level_code'] = levelData[0]?.code; // universal alias
457
-
458
- // 3️⃣ Final substitution and cleanup
459
- if (path) {
460
- const finalPath = this.resolveUploadPath(path, replacements);
461
- return finalPath;
462
- }
463
- return null;
464
- }
465
-
466
- /**
467
- * Replace placeholders with actual values, remove missing variables
468
- */
469
- private resolveUploadPath(
470
- template: string,
471
- replacements: Record<string, string | null>,
472
- ) {
473
- let path = template;
474
- path = path.replace(/\$\{(\w+)\}/g, (_, key) => replacements[key] || '');
475
- return path.replace(/\/+/g, '/').replace(/\/$/, '');
476
- }
477
-
478
- /**
479
- * Helper method to build upload path for different scenarios
480
- *
481
- * @param mappedEntityType - The main entity type (e.g., 'LEAD', 'SCH')
482
- * @param loggedInUser - Current user context
483
- * @param mappedEntityId - ID of the main entity
484
- * @param parentId - Optional parent entity ID (e.g., template ID)
485
- * @param parentType - Optional parent entity type (e.g., 'TEMPLATE')
486
- * @returns Promise<string | null> - The resolved upload path
487
- *
488
- * Examples:
489
- * - buildUploadPath('LEAD', user, 123) -> "ORG1/template/SCH1"
490
- * - buildUploadPath('LEAD', user, 123, 456, 'TEMPLATE') -> "ORG1/template/SCH1/TEMPLATE1"
491
- */
492
- async buildUploadPath(
493
- mappedEntityType: string,
494
- loggedInUser: any,
495
- mappedEntityId?: number,
496
- parentId?: number,
497
- parentType?: string,
498
- ): Promise<string | null> {
499
- return this.buildUploadPathGeneric(
500
- mappedEntityType,
501
- loggedInUser,
502
- mappedEntityId,
503
- parentId,
504
- parentType,
505
- );
506
- }
507
- }
1
+ import { Injectable } from '@nestjs/common';
2
+ import { ConfigService } from '@nestjs/config';
3
+ import { S3 } from 'aws-sdk';
4
+ import axios from 'axios';
5
+ import { DataSource } from 'typeorm';
6
+ import { v4 as uuidv4 } from 'uuid';
7
+ import {
8
+ ENTITYTYPE_MEDIA,
9
+ STATUS_PENDING,
10
+ } from '../../../constant/global.constant';
11
+ import { MediaData } from '../entity/media-data.entity';
12
+ import { MediaDataRepository } from '../repository/media-data.repository';
13
+ import { EntityServiceImpl } from './entity-service-impl.service';
14
+
15
+ @Injectable()
16
+ export class MediaDataService extends EntityServiceImpl {
17
+ constructor(
18
+ private readonly configService: ConfigService,
19
+ private readonly mediaRepository: MediaDataRepository,
20
+ private dataSource: DataSource,
21
+ ) {
22
+ super();
23
+ }
24
+
25
+ s3AccessKeyID = this.configService.get('AWS.S3.AWS_ACCESS_KEY_ID');
26
+ s3AccessKeySecret = this.configService.get('AWS.S3.AWS_SECRET_KEY');
27
+ s3Region = this.configService.get('AWS.S3.AWS_REGION');
28
+ bucketName = this.configService.get<string>('AWS.S3.BUCKET_NAME');
29
+
30
+ s3 = new S3({
31
+ accessKeyId: this.s3AccessKeyID,
32
+ secretAccessKey: this.s3AccessKeySecret,
33
+ region: this.s3Region,
34
+ signatureVersion: 'v4',
35
+ });
36
+
37
+ async generateMediaUploadDetails(
38
+ fileName: string,
39
+ mappedAttributeKey: string,
40
+ loggedInUser,
41
+ mappedEntityType?: string,
42
+ mappedEntityId?: number,
43
+ parentId?: number,
44
+ parentType?: string,
45
+ ) {
46
+ if (!fileName || !mappedAttributeKey) {
47
+ return null;
48
+ }
49
+ const ext = fileName.split('.').pop()?.toLowerCase() ?? '';
50
+
51
+ const id = uuidv4();
52
+
53
+ const s3Path =
54
+ (await this.buildUploadPathGeneric(
55
+ mappedEntityType || '',
56
+ loggedInUser,
57
+ mappedEntityId,
58
+ parentId,
59
+ parentType,
60
+ )) || `uploads`;
61
+
62
+ const s3Key = `${s3Path}/${id}.${ext}`;
63
+
64
+ const uploadUrl = await this.s3.getSignedUrlPromise('putObject', {
65
+ Bucket: this.bucketName,
66
+ Key: s3Key,
67
+ Expires: 60 * 5, // URL valid for 5 mins
68
+ ContentType: this.getContentType(ext),
69
+ });
70
+
71
+ const mediaData = new MediaData();
72
+ mediaData.file_name = fileName;
73
+ mediaData.mapped_attribute_key = mappedAttributeKey;
74
+ mediaData.status = STATUS_PENDING;
75
+ if (mappedEntityType) {
76
+ mediaData.mapped_entity_type = mappedEntityType;
77
+ }
78
+ if (mappedEntityId) {
79
+ mediaData.mapped_entity_id = mappedEntityId;
80
+ }
81
+ mediaData.media_url = s3Key;
82
+
83
+ //INSERT RECORD IN DOC TABLE
84
+ const savedEntity = await super.createEntity(mediaData, loggedInUser);
85
+
86
+ if (savedEntity) {
87
+ return { id: savedEntity.id, path: s3Key, uploadUrl };
88
+ }
89
+ return null;
90
+ }
91
+
92
+ async findByAttributeKeyAndMappedEntityIdAndMappedEntityType(
93
+ attributeKey: string,
94
+ mappedEntityId: number,
95
+ mappedEntityType: string,
96
+ ) {
97
+ return await this.mediaRepository.findByAttributeKeyAndMappedEntityIdAndMappedEntityType(
98
+ attributeKey,
99
+ mappedEntityId,
100
+ mappedEntityType,
101
+ );
102
+ }
103
+
104
+ async findByMappedEntityIdAndMappedEntityType(
105
+ mappedEntityId: number,
106
+ mappedEntityType: string,
107
+ ) {
108
+ return await this.mediaRepository.findByMappedEntityIdAndMappedEntityType(
109
+ mappedEntityId,
110
+ mappedEntityType,
111
+ );
112
+ }
113
+
114
+ async deleteByAttributeKeyAndMappedEntityIdAndMappedEntityType(
115
+ attributeKey: string,
116
+ mappedEntityId: number,
117
+ mappedEntityType: string,
118
+ ) {
119
+ return await this.mediaRepository.deleteByAttributeKeyAndMappedEntityIdAndMappedEntityType(
120
+ attributeKey,
121
+ mappedEntityId,
122
+ mappedEntityType,
123
+ );
124
+ }
125
+
126
+ private getContentType(ext: string): string {
127
+ const map = {
128
+ pdf: 'application/pdf',
129
+ jpg: 'image/jpeg',
130
+ jpeg: 'image/jpeg',
131
+ png: 'image/png',
132
+ };
133
+ return map[ext] || 'application/octet-stream';
134
+ }
135
+
136
+ // New method to get the signed URL for the media (download link)
137
+ async getMediaDownloadUrl(id: number, loggedInUser, expiresIn?: number) {
138
+ try {
139
+ const entityData = await this.getEntityData(
140
+ ENTITYTYPE_MEDIA,
141
+ id,
142
+ loggedInUser,
143
+ );
144
+ const mediaData = entityData as MediaData;
145
+ let fileSize: number | undefined;
146
+ if (this.bucketName) {
147
+ const head = await this.s3
148
+ .headObject({
149
+ Bucket: this.bucketName,
150
+ Key: mediaData.media_url,
151
+ })
152
+ .promise();
153
+ fileSize = head.ContentLength;
154
+ }
155
+
156
+ const signedUrl = await this.s3.getSignedUrlPromise('getObject', {
157
+ Bucket: this.bucketName,
158
+ Key: mediaData.media_url,
159
+ Expires: Number(expiresIn) || 60 * 5,
160
+ });
161
+ return {
162
+ signedUrl,
163
+ fileName: mediaData.file_name,
164
+ id: mediaData.id,
165
+ size: fileSize,
166
+ };
167
+ } catch (error) {
168
+ console.error('Error generating signed URL for media:', error);
169
+ throw new Error('Failed to generate signed URL');
170
+ }
171
+ }
172
+
173
+ public async buildUploadPathGenericdd(
174
+ mappedEntityType: string,
175
+ loggedInUser,
176
+ mappedEntityId?: number,
177
+ parentId?: number,
178
+ parentType?: string,
179
+ ) {
180
+ // 1️⃣ Fetch entity metadata
181
+ const entityMaster = await this.entityMasterService.getEntityData(
182
+ mappedEntityType,
183
+ loggedInUser,
184
+ );
185
+ if (!entityMaster) {
186
+ throw new Error(`Entity master not found for ${mappedEntityType}`);
187
+ }
188
+
189
+ let pathTemplate = entityMaster.doc_upload_path ?? '';
190
+ const entityOverwrite = entityMaster.overwrite_path ?? 0;
191
+
192
+ if (!pathTemplate) {
193
+ throw new Error(`doc_upload_path not defined for ${mappedEntityType}`);
194
+ }
195
+
196
+ // 2️⃣ Prepare replacement map
197
+ const replacements: Record<string, string | null> = {};
198
+
199
+ // --- ORG ---
200
+ let organizationData = await this.dataSource.query(
201
+ 'SELECT * FROM sso_organization WHERE id = $1',
202
+ [loggedInUser.organization_id],
203
+ );
204
+ replacements['org_code'] = organizationData[0].code;
205
+
206
+ // --- LEVEL CODE ---
207
+ // Priority: If parent is provided, use parent for level_code; otherwise use normal logic
208
+ let levelEntityId: number;
209
+ let levelEntityType: string;
210
+
211
+ if (parentId && parentType) {
212
+ // When parent is provided, parent becomes the level_code
213
+ levelEntityId = parentId;
214
+ levelEntityType = parentType;
215
+ } else {
216
+ // Normal logic: determine which entity to use for level_code
217
+ levelEntityId = entityOverwrite ? mappedEntityId : loggedInUser.level_id;
218
+ levelEntityType = entityOverwrite
219
+ ? mappedEntityType
220
+ : loggedInUser.level_type;
221
+ }
222
+
223
+ // Get entity data for level_code
224
+ const getEntityData = await this.dataSource.query(
225
+ `SELECT * FROM cr_entity_master WHERE mapped_entity_type = $1 and organization_id = $2`,
226
+ [levelEntityType, loggedInUser.organization_id],
227
+ );
228
+
229
+ if (getEntityData.length == 0) return null;
230
+ const levelEntityDataResult = await this.dataSource.query(
231
+ `SELECT * FROM ${getEntityData[0]?.db_table_name} WHERE id = $1`,
232
+ [levelEntityId],
233
+ );
234
+
235
+ const levelEntityData = levelEntityDataResult[0] || null;
236
+
237
+ if (levelEntityData?.code) {
238
+ replacements['level_code'] = levelEntityData.code; // universal alias
239
+ }
240
+
241
+ // --- MAPPED ENTITY CODE (when parent is provided) ---
242
+ // When parent is provided, the mapped entity becomes an additional level
243
+ if (parentId && parentType && mappedEntityId && mappedEntityType) {
244
+ const mappedEntityKey = `${mappedEntityType.toLowerCase()}_code`;
245
+ const mappedEntityData = await super.getEntityData(
246
+ mappedEntityType,
247
+ mappedEntityId,
248
+ loggedInUser,
249
+ );
250
+ if (mappedEntityData?.code) {
251
+ replacements[mappedEntityKey] = mappedEntityData.code;
252
+
253
+ // Modify path template to include mapped entity
254
+ // Example: ${org_code}/${level_code} becomes ${org_code}/${level_code}/${tem_code}
255
+ const mappedVariable = `\${${mappedEntityKey}}`;
256
+ if (!pathTemplate.includes(mappedVariable)) {
257
+ pathTemplate = pathTemplate + `/${mappedVariable}`;
258
+ }
259
+ }
260
+ }
261
+
262
+ // --- TEMPLATE CODE (if parent is provided) ---
263
+ // Special handling for template_code when parent_id and parent_type are provided
264
+ if (parentId && parentType) {
265
+ // const templateEntityData = await super.getEntityData(
266
+ // parentType,
267
+ // parentId,
268
+ // loggedInUser,
269
+ // );
270
+
271
+ // Get entity data for level_code
272
+ const getEntityData = await this.dataSource.query(
273
+ `SELECT * FROM cr_entity_master WHERE mapped_entity_type = $1 and organization_id = $2`,
274
+ [mappedEntityType, loggedInUser.organization_id],
275
+ );
276
+
277
+ if (getEntityData.length == 0) return null;
278
+ const levelEntityDataResult = await this.dataSource.query(
279
+ `SELECT * FROM ${getEntityData[0]?.db_table_name} WHERE id = $1`,
280
+ [mappedEntityId],
281
+ );
282
+
283
+ if (
284
+ levelEntityDataResult?.[0].code &&
285
+ parentType != 'SCH' &&
286
+ loggedInUser.level_type != 'SCH'
287
+ ) {
288
+ replacements['template_code'] = levelEntityDataResult[0].code;
289
+ // If we have parent template, modify the path to include it
290
+ pathTemplate = '${org_code}/template/${template_code}';
291
+ } else {
292
+ const getEntityData = await this.dataSource.query(
293
+ `SELECT * FROM cr_entity_master WHERE mapped_entity_type = $1 and organization_id = $2`,
294
+ [mappedEntityType, loggedInUser.organization_id],
295
+ );
296
+
297
+ if (getEntityData.length == 0) return null;
298
+ const levelEntityDataResult = await this.dataSource.query(
299
+ `SELECT * FROM ${getEntityData[0]?.db_table_name} WHERE id = $1`,
300
+ [mappedEntityId],
301
+ );
302
+
303
+ replacements['template_code'] = levelEntityDataResult?.[0].code;
304
+
305
+ pathTemplate = '${org_code}/${level_code}/template/${template_code}';
306
+ }
307
+ }
308
+
309
+ // --- Dynamic variables inside path ---
310
+ const matches = pathTemplate.match(/\$\{(\w+)\}/g) || [];
311
+
312
+ for (const variable of matches) {
313
+ const key = variable.replace(/\$\{|\}/g, '');
314
+ if (replacements[key]) continue; // already resolved
315
+
316
+ if (key === 'org_code') continue; // already set
317
+ if (key === 'level_code') continue; // already set above
318
+ if (key === 'template_code') continue; // handled above if parent is provided
319
+
320
+ if (key.endsWith('_code')) {
321
+ // derive entity type dynamically
322
+ const entityType = key.replace('_code', '').toUpperCase();
323
+
324
+ let entityId: number | undefined;
325
+
326
+ // Special handling for template_code - use parent if available, otherwise mapped entity
327
+ if (key === 'template_code') {
328
+ if (parentType?.toUpperCase() === 'TEMPLATE' && parentId) {
329
+ entityId = parentId;
330
+ } else if (mappedEntityType === 'TEMPLATE') {
331
+ entityId = mappedEntityId;
332
+ }
333
+ } else {
334
+ // For other _code variables, check if parent matches the entity type
335
+ if (parentType?.toUpperCase() === entityType && parentId) {
336
+ entityId = parentId;
337
+ } else if (mappedEntityType === entityType) {
338
+ entityId = mappedEntityId;
339
+ }
340
+ }
341
+
342
+ // Generic lookup
343
+ if (entityId) {
344
+ const entityData = await super.getEntityData(
345
+ entityType,
346
+ entityId,
347
+ loggedInUser,
348
+ );
349
+ replacements[key] = entityData?.code ?? null;
350
+ }
351
+ }
352
+ }
353
+
354
+ // --- ORG special case ---
355
+ // If ORG level & overwrite = 0 → use org_code, otherwise use the mapped entity's code
356
+ if (loggedInUser.level_type === 'ORG' && entityOverwrite === 0) {
357
+ // When overwrite is 0 and user is ORG level, but we want to use mapped entity
358
+ // Don't set level_code to null - it should already be set above based on entityOverwrite logic
359
+ }
360
+
361
+ // 3️⃣ Final substitution and cleanup
362
+ if (pathTemplate) {
363
+ const finalPath = this.resolveUploadPath(pathTemplate, replacements);
364
+ return finalPath;
365
+ }
366
+ return null;
367
+ }
368
+
369
+ public async buildUploadPathGeneric(
370
+ mappedEntityType: string,
371
+ loggedInUser,
372
+ mappedEntityId?: number,
373
+ parentId?: number,
374
+ parentType?: string,
375
+ ) {
376
+ //APPCODE
377
+ let appCode = '';
378
+
379
+ let org_id = loggedInUser.organization_id;
380
+ let level_type = loggedInUser.level_type;
381
+ let level_id =
382
+ loggedInUser.level_type == 'ORG' && mappedEntityType == 'SCH'
383
+ ? mappedEntityId
384
+ : loggedInUser.level_id;
385
+
386
+ let organizationData = {};
387
+ let levelData = {};
388
+ let mappedEntityData = {};
389
+
390
+ if (loggedInUser && loggedInUser.appcode) {
391
+ appCode = loggedInUser.appcode;
392
+ }
393
+
394
+ if (appCode && appCode != 'ADM') {
395
+ const baseUrl = this.configService.get<string>('REDIRECT_BE_URL');
396
+
397
+ // Prepare the query string
398
+ const queryParams = new URLSearchParams({
399
+ loggedInUser: JSON.stringify(loggedInUser),
400
+ }).toString();
401
+
402
+ organizationData = await axios
403
+ .get(
404
+ `${baseUrl}/organization/public/${loggedInUser.organization_id}?${queryParams}`,
405
+ )
406
+ .then((res) => res.data)
407
+ .catch((err) => {
408
+ console.error('Error fetching organization data:', err.message);
409
+ return null;
410
+ });
411
+
412
+ levelData = await axios
413
+ .get(`${baseUrl}/school/public/?${queryParams}`)
414
+ .then((res) => res.data)
415
+ .catch((err) => {
416
+ console.error('Error fetching school data:', err.message);
417
+ return null;
418
+ });
419
+ } else {
420
+ organizationData = await this.dataSource.query(
421
+ 'SELECT * FROM adm_org_profile WHERE id = $1',
422
+ [loggedInUser.organization_id],
423
+ );
424
+ levelData = await this.dataSource.query(
425
+ 'SELECT * FROM adm_school_profile WHERE id = $1',
426
+ [loggedInUser.level_id],
427
+ );
428
+ }
429
+
430
+ console.log(levelData);
431
+
432
+ // 1️⃣ Fetch entity metadata
433
+ // const uploadEntity = await this.entityMasterService.getEntityData(
434
+ // mappedEntityType,
435
+ // loggedInUser,
436
+ // );
437
+ // if (!uploadEntity) {
438
+ // throw new Error(`Entity master not found for ${mappedEntityType}`);
439
+ // }
440
+
441
+ let subfolder =
442
+ mappedEntityType == level_type
443
+ ? 'data'
444
+ : parentType
445
+ ? `${parentType}/${parentId}/${mappedEntityType}`
446
+ : mappedEntityType;
447
+ let path =
448
+ '${org_code}/${level_code}/' +
449
+ (subfolder !== 'data'
450
+ ? `${subfolder}/${mappedEntityId}`
451
+ : `${subfolder}`);
452
+
453
+ // 2️⃣ Prepare replacement map
454
+ const replacements: Record<string, string | null> = {};
455
+ replacements['org_code'] = organizationData[0]?.code;
456
+ replacements['level_code'] = levelData[0]?.code; // universal alias
457
+
458
+ // 3️⃣ Final substitution and cleanup
459
+ if (path) {
460
+ const finalPath = this.resolveUploadPath(path, replacements);
461
+ return finalPath;
462
+ }
463
+ return null;
464
+ }
465
+
466
+ /**
467
+ * Replace placeholders with actual values, remove missing variables
468
+ */
469
+ private resolveUploadPath(
470
+ template: string,
471
+ replacements: Record<string, string | null>,
472
+ ) {
473
+ let path = template;
474
+ path = path.replace(/\$\{(\w+)\}/g, (_, key) => replacements[key] || '');
475
+ return path.replace(/\/+/g, '/').replace(/\/$/, '');
476
+ }
477
+
478
+ /**
479
+ * Helper method to build upload path for different scenarios
480
+ *
481
+ * @param mappedEntityType - The main entity type (e.g., 'LEAD', 'SCH')
482
+ * @param loggedInUser - Current user context
483
+ * @param mappedEntityId - ID of the main entity
484
+ * @param parentId - Optional parent entity ID (e.g., template ID)
485
+ * @param parentType - Optional parent entity type (e.g., 'TEMPLATE')
486
+ * @returns Promise<string | null> - The resolved upload path
487
+ *
488
+ * Examples:
489
+ * - buildUploadPath('LEAD', user, 123) -> "ORG1/template/SCH1"
490
+ * - buildUploadPath('LEAD', user, 123, 456, 'TEMPLATE') -> "ORG1/template/SCH1/TEMPLATE1"
491
+ */
492
+ async buildUploadPath(
493
+ mappedEntityType: string,
494
+ loggedInUser: any,
495
+ mappedEntityId?: number,
496
+ parentId?: number,
497
+ parentType?: string,
498
+ ): Promise<string | null> {
499
+ return this.buildUploadPathGeneric(
500
+ mappedEntityType,
501
+ loggedInUser,
502
+ mappedEntityId,
503
+ parentId,
504
+ parentType,
505
+ );
506
+ }
507
+ }