rez_core 5.0.52 → 5.0.53

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 (426) hide show
  1. package/.claude/settings.local.json +26 -0
  2. package/.idea/250218_nodejs_core.iml +8 -11
  3. package/.idea/codeStyles/Project.xml +58 -58
  4. package/.idea/codeStyles/codeStyleConfig.xml +4 -4
  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 +1 -1
  10. package/.idea/misc.xml +6 -0
  11. package/.idea/modules.xml +7 -7
  12. package/.idea/prettier.xml +5 -5
  13. package/.idea/vcs.xml +5 -5
  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/mapper/service/field-mapper.service.js +4 -4
  26. package/dist/module/meta/repository/entity-attribute-update.repository.js +3 -3
  27. package/dist/module/meta/repository/entity-master.repository.js +6 -6
  28. package/dist/module/meta/service/entity-dynamic.service.js +44 -44
  29. package/dist/module/meta/service/entity-list.service.js +3 -3
  30. package/dist/module/meta/service/entity-relation.service.js +9 -9
  31. package/dist/module/meta/service/entity-service-impl.service.js +3 -3
  32. package/dist/module/meta/service/resolver.service.js +3 -3
  33. package/dist/module/module/repository/menu.repository.js +17 -17
  34. package/dist/module/notification/service/notification.service.js +10 -10
  35. package/dist/module/user/controller/login.controller.js +18 -18
  36. package/dist/module/user/service/role.service.js +4 -4
  37. package/dist/module/user/service/user-session.service.js +2 -2
  38. package/dist/module/workflow/controller/action.controller.d.ts +1 -1
  39. package/dist/module/workflow/repository/action.repository.d.ts +4 -2
  40. package/dist/module/workflow/repository/action.repository.js +34 -26
  41. package/dist/module/workflow/repository/action.repository.js.map +1 -1
  42. package/dist/module/workflow/repository/comm-template.repository.js +6 -6
  43. package/dist/module/workflow/repository/form-master.repository.js +2 -2
  44. package/dist/module/workflow/repository/stage-group.repository.js +20 -20
  45. package/dist/module/workflow/repository/stage-movement.repository.js +17 -17
  46. package/dist/module/workflow/repository/stage.repository.js +8 -8
  47. package/dist/module/workflow/service/action-template-mapping.service.js +13 -13
  48. package/dist/module/workflow/service/action.service.d.ts +1 -1
  49. package/dist/module/workflow/service/action.service.js +7 -7
  50. package/dist/module/workflow/service/entity-modification.service.js +2 -2
  51. package/dist/module/workflow/service/stage-group.service.js +2 -2
  52. package/dist/module/workflow/service/stage.service.js +2 -2
  53. package/dist/module/workflow/service/task.service.js +35 -35
  54. package/dist/module/workflow/service/workflow-list-master.service.js +15 -15
  55. package/dist/module/workflow/service/workflow-meta.service.js +26 -26
  56. package/dist/module/workflow/service/workflow.service.js +2 -2
  57. package/dist/module/workflow-automation/service/schedule-handler.service.js +9 -9
  58. package/dist/module/workflow-schedule/processors/schedule.processor.js +4 -4
  59. package/dist/module/workflow-schedule/service/workflow-schedule.service.js +9 -9
  60. package/dist/tsconfig.build.tsbuildinfo +1 -1
  61. package/dist/utils/service/reflection-helper.service.js +2 -2
  62. package/docs/modules/event-driven-integration-design.md +91 -91
  63. package/docs/modules/integration.md +250 -250
  64. package/eslint.config.mjs +34 -34
  65. package/nest-cli.json +14 -14
  66. package/package.json +124 -124
  67. package/server.log +850 -0
  68. package/src/app.controller.ts +12 -12
  69. package/src/app.module.ts +66 -66
  70. package/src/app.service.ts +8 -8
  71. package/src/config/bull.config.ts +69 -69
  72. package/src/config/config.module.ts +17 -17
  73. package/src/config/database.config.ts +48 -48
  74. package/src/constant/global.constant.ts +67 -67
  75. package/src/core.module.ts +91 -91
  76. package/src/decorators/roles.decorator.ts +7 -7
  77. package/src/dtos/response.dto.ts +6 -6
  78. package/src/dtos/response.ts +5 -5
  79. package/src/index.ts +1 -1
  80. package/src/module/auth/auth.module.ts +49 -49
  81. package/src/module/auth/controller/auth.controller.ts +28 -28
  82. package/src/module/auth/guards/google-auth.guard.ts +9 -9
  83. package/src/module/auth/guards/jwt.guard.ts +22 -22
  84. package/src/module/auth/guards/role.guard.ts +68 -68
  85. package/src/module/auth/services/auth.service.ts +50 -50
  86. package/src/module/auth/services/jwt.service.ts +11 -11
  87. package/src/module/auth/strategies/google.strategy.ts +54 -54
  88. package/src/module/auth/strategies/jwt.strategy.ts +58 -58
  89. package/src/module/auth/strategies/local.strategy.ts +13 -13
  90. package/src/module/dashboard/controller/dashboard.controller.ts +36 -36
  91. package/src/module/dashboard/dashboard.module.ts +21 -21
  92. package/src/module/dashboard/entity/dashboard_page_data.entity.ts +27 -27
  93. package/src/module/dashboard/entity/widget_master.entity.ts +18 -18
  94. package/src/module/dashboard/repository/dashboard.repository.ts +42 -42
  95. package/src/module/dashboard/service/dashboard.service.ts +73 -73
  96. package/src/module/dev/dev.module.ts +12 -12
  97. package/src/module/dev/service/dev.service.ts +7 -7
  98. package/src/module/enterprise/controller/organization.controller.ts +36 -36
  99. package/src/module/enterprise/enterprise.module.ts +30 -30
  100. package/src/module/enterprise/entity/enterprise.entity.ts +37 -37
  101. package/src/module/enterprise/entity/organization-app-mapping.entity.ts +13 -13
  102. package/src/module/enterprise/entity/organization.entity.ts +92 -92
  103. package/src/module/enterprise/repository/enterprise.repository.ts +31 -31
  104. package/src/module/enterprise/repository/organization.repository.ts +26 -26
  105. package/src/module/enterprise/repository/school.repository.ts +278 -278
  106. package/src/module/enterprise/service/brand.service.ts +5 -5
  107. package/src/module/enterprise/service/enterprise.service.ts +16 -16
  108. package/src/module/enterprise/service/organization-app-mapping.service.ts +4 -4
  109. package/src/module/enterprise/service/organization.service.ts +145 -145
  110. package/src/module/entity_json/controller/entity_json.controller.ts +47 -47
  111. package/src/module/entity_json/entity/entityJson.entity.ts +39 -39
  112. package/src/module/entity_json/entity_json.module.ts +18 -18
  113. package/src/module/entity_json/service/entityJson.repository.ts +37 -37
  114. package/src/module/entity_json/service/entity_json.service.ts +234 -234
  115. package/src/module/filter/controller/filter.controller.ts +84 -84
  116. package/src/module/filter/dto/filter-request.dto.ts +38 -38
  117. package/src/module/filter/entity/saved-filter-detail.entity.ts +41 -41
  118. package/src/module/filter/entity/saved-filter-master.entity.ts +23 -23
  119. package/src/module/filter/filter.module.ts +31 -31
  120. package/src/module/filter/repository/saved-filter.repository.ts +168 -168
  121. package/src/module/filter/service/filter-evaluator.service.ts +86 -86
  122. package/src/module/filter/service/filter.service.ts +1237 -1237
  123. package/src/module/filter/service/saved-filter.service.ts +170 -170
  124. package/src/module/ics/controller/ics.controller.ts +21 -21
  125. package/src/module/ics/dto/ics.dto.ts +55 -55
  126. package/src/module/ics/ics.module.ts +13 -13
  127. package/src/module/ics/service/ics.service.ts +57 -57
  128. package/src/module/integration/controller/calender-event.controller.ts +31 -31
  129. package/src/module/integration/controller/integration.controller.ts +662 -662
  130. package/src/module/integration/controller/wrapper.controller.ts +37 -37
  131. package/src/module/integration/dto/create-config.dto.ts +526 -526
  132. package/src/module/integration/entity/integration-config.entity.ts +112 -112
  133. package/src/module/integration/entity/integration-entity-mapper.entity.ts +14 -14
  134. package/src/module/integration/entity/integration-source.entity.ts +17 -17
  135. package/src/module/integration/entity/user-integration.entity.ts +71 -71
  136. package/src/module/integration/examples/usage.example.ts +338 -338
  137. package/src/module/integration/factories/base.factory.ts +7 -7
  138. package/src/module/integration/factories/email.factory.ts +49 -49
  139. package/src/module/integration/factories/integration.factory.ts +121 -121
  140. package/src/module/integration/factories/sms.factory.ts +51 -51
  141. package/src/module/integration/factories/telephone.factory.ts +41 -41
  142. package/src/module/integration/factories/whatsapp.factory.ts +56 -56
  143. package/src/module/integration/integration.module.ts +110 -110
  144. package/src/module/integration/service/calendar-event.service.ts +118 -118
  145. package/src/module/integration/service/integration-entity-mapper.service.ts +17 -17
  146. package/src/module/integration/service/integration-queue.service.ts +229 -229
  147. package/src/module/integration/service/integration.service.ts +2633 -2633
  148. package/src/module/integration/service/oauth.service.ts +224 -224
  149. package/src/module/integration/service/wrapper.service.ts +701 -701
  150. package/src/module/integration/strategies/email/gmail-api.strategy.ts +280 -280
  151. package/src/module/integration/strategies/email/outlook-api.strategy.ts +44 -44
  152. package/src/module/integration/strategies/email/outlook.strategy.ts +64 -64
  153. package/src/module/integration/strategies/email/sendgrid-api.strategy.ts +260 -260
  154. package/src/module/integration/strategies/integration.strategy.ts +97 -97
  155. package/src/module/integration/strategies/sms/gupshup-sms.strategy.ts +146 -146
  156. package/src/module/integration/strategies/sms/msg91-sms.strategy.ts +164 -164
  157. package/src/module/integration/strategies/sms/tubelight-sms.strategy.ts +163 -163
  158. package/src/module/integration/strategies/telephone/ozonetel-voice.strategy.ts +238 -238
  159. package/src/module/integration/strategies/telephone/tubelight-voice.strategy.ts +210 -210
  160. package/src/module/integration/strategies/whatsapp/gupshup-whatsapp.strategy.ts +359 -359
  161. package/src/module/integration/strategies/whatsapp/tubelight-whatsapp.strategy.ts +372 -372
  162. package/src/module/integration/strategies/whatsapp/whatsapp-cloud.strategy.ts +403 -403
  163. package/src/module/integration/strategies/whatsapp/whatsapp.strategy.ts +57 -57
  164. package/src/module/layout/controller/layout.controller.ts +47 -47
  165. package/src/module/layout/entity/header-items.entity.ts +28 -28
  166. package/src/module/layout/entity/header-section.entity.ts +19 -19
  167. package/src/module/layout/layout.module.ts +21 -21
  168. package/src/module/layout/repository/header-items.repository.ts +18 -18
  169. package/src/module/layout/repository/header-section.repository.ts +22 -22
  170. package/src/module/layout/service/header-section.service.ts +25 -25
  171. package/src/module/layout_preference/controller/layout_preference.controller.ts +73 -73
  172. package/src/module/layout_preference/entity/layout_preference.entity.ts +28 -28
  173. package/src/module/layout_preference/layout_preference.module.ts +22 -22
  174. package/src/module/layout_preference/repository/layout_preference.repository.ts +65 -65
  175. package/src/module/layout_preference/service/layout_preference.service.ts +191 -191
  176. package/src/module/lead/controller/lead.controller.ts +30 -30
  177. package/src/module/lead/lead.module.ts +14 -14
  178. package/src/module/lead/repository/lead.repository.ts +41 -41
  179. package/src/module/lead/service/lead.service.ts +54 -54
  180. package/src/module/linked_attributes/controller/linked_attributes.controller.ts +37 -37
  181. package/src/module/linked_attributes/entity/linked_attribute.entity.ts +51 -51
  182. package/src/module/linked_attributes/linked_attributes.module.ts +16 -16
  183. package/src/module/linked_attributes/repository/linked_attribute.repository.ts +12 -12
  184. package/src/module/linked_attributes/service/linked_attributes.service.ts +75 -75
  185. package/src/module/listmaster/controller/list-master.controller.ts +230 -230
  186. package/src/module/listmaster/entity/list-master-items.entity.ts +43 -43
  187. package/src/module/listmaster/entity/list-master.entity.ts +33 -33
  188. package/src/module/listmaster/listmaster.module.ts +46 -46
  189. package/src/module/listmaster/repository/list-master-items.repository.ts +173 -173
  190. package/src/module/listmaster/repository/list-master.repository.ts +46 -46
  191. package/src/module/listmaster/service/list-master-engine.ts +19 -19
  192. package/src/module/listmaster/service/list-master-extension.interface.ts +4 -4
  193. package/src/module/listmaster/service/list-master-item.service.ts +292 -292
  194. package/src/module/listmaster/service/list-master-registry.ts +15 -15
  195. package/src/module/listmaster/service/list-master.service.ts +535 -535
  196. package/src/module/mapper/controller/field-mapper.controller.ts +76 -76
  197. package/src/module/mapper/controller/mapper.controller.ts +20 -20
  198. package/src/module/mapper/dto/field-mapper.dto.ts +14 -14
  199. package/src/module/mapper/entity/field-lovs.entity.ts +19 -19
  200. package/src/module/mapper/entity/field-mapper.entity.ts +53 -53
  201. package/src/module/mapper/entity/mapper.entity.ts +16 -16
  202. package/src/module/mapper/mapper.module.ts +35 -35
  203. package/src/module/mapper/repository/field-lovs.repository.ts +35 -35
  204. package/src/module/mapper/repository/field-mapper.repository.ts +42 -42
  205. package/src/module/mapper/repository/mapper.repository.ts +32 -32
  206. package/src/module/mapper/service/field-mapper.service.ts +271 -271
  207. package/src/module/mapper/service/mapper.service.ts +81 -81
  208. package/src/module/master/controller/master.controller.ts +74 -74
  209. package/src/module/master/service/master.service.ts +483 -483
  210. package/src/module/meta/controller/app-master.controller.ts +38 -38
  211. package/src/module/meta/controller/attribute-master.controller.ts +84 -84
  212. package/src/module/meta/controller/entity-dynamic.controller.ts +125 -125
  213. package/src/module/meta/controller/entity-master.controller.ts +41 -41
  214. package/src/module/meta/controller/entity-relation.controller.ts +36 -36
  215. package/src/module/meta/controller/entity.controller.ts +392 -392
  216. package/src/module/meta/controller/entity.public.controller.ts +75 -75
  217. package/src/module/meta/controller/media.controller.ts +107 -107
  218. package/src/module/meta/controller/meta.controller.ts +96 -96
  219. package/src/module/meta/controller/view-master.controller.ts +86 -86
  220. package/src/module/meta/dto/entity-list-data.dto.ts +6 -6
  221. package/src/module/meta/dto/entity-tab.dto.ts +4 -4
  222. package/src/module/meta/dto/entity-table.dto.ts +9 -9
  223. package/src/module/meta/entity/app-master.entity.ts +34 -34
  224. package/src/module/meta/entity/attribute-master.entity.ts +92 -92
  225. package/src/module/meta/entity/base-entity.entity.ts +75 -75
  226. package/src/module/meta/entity/entity-master.entity.ts +85 -85
  227. package/src/module/meta/entity/entity-relation-data.entity.ts +29 -29
  228. package/src/module/meta/entity/entity-relation.entity.ts +23 -23
  229. package/src/module/meta/entity/entity-table-column.entity.ts +61 -61
  230. package/src/module/meta/entity/entity-table.entity.ts +50 -50
  231. package/src/module/meta/entity/media-data.entity.ts +32 -32
  232. package/src/module/meta/entity/preference.entity.ts +62 -62
  233. package/src/module/meta/entity/view-master.entity.ts +41 -41
  234. package/src/module/meta/entity.module.ts +166 -166
  235. package/src/module/meta/repository/app-master.repository.ts +20 -20
  236. package/src/module/meta/repository/attribute-master.repository.ts +138 -138
  237. package/src/module/meta/repository/entity-attribute-update.repository.ts +44 -44
  238. package/src/module/meta/repository/entity-master.repository.ts +111 -111
  239. package/src/module/meta/repository/entity-table-column.repository.ts +39 -39
  240. package/src/module/meta/repository/entity-table.repository.ts +53 -53
  241. package/src/module/meta/repository/media-data.repository.ts +50 -50
  242. package/src/module/meta/repository/preference.repository.ts +20 -20
  243. package/src/module/meta/repository/user-app-mapping.repository.ts +28 -28
  244. package/src/module/meta/repository/view-master.repository.ts +42 -42
  245. package/src/module/meta/service/app-master.service.ts +37 -37
  246. package/src/module/meta/service/attribute-master.service.ts +130 -130
  247. package/src/module/meta/service/common.service.ts +9 -9
  248. package/src/module/meta/service/entity-attribute-update.service.ts +28 -28
  249. package/src/module/meta/service/entity-dynamic.service.ts +821 -821
  250. package/src/module/meta/service/entity-list.service.ts +205 -205
  251. package/src/module/meta/service/entity-master.service.ts +171 -171
  252. package/src/module/meta/service/entity-realation-data.service.ts +9 -9
  253. package/src/module/meta/service/entity-relation.service.ts +87 -87
  254. package/src/module/meta/service/entity-service-impl.service.ts +525 -525
  255. package/src/module/meta/service/entity-table-column.service.ts +39 -39
  256. package/src/module/meta/service/entity-table.service.ts +150 -150
  257. package/src/module/meta/service/entity-validation.service.ts +187 -187
  258. package/src/module/meta/service/entity.service.ts +67 -67
  259. package/src/module/meta/service/field-group.service.ts +103 -103
  260. package/src/module/meta/service/media-data.service.ts +507 -507
  261. package/src/module/meta/service/populate-meta.service.ts +222 -222
  262. package/src/module/meta/service/preference.service.ts +16 -16
  263. package/src/module/meta/service/resolver.service.ts +291 -291
  264. package/src/module/meta/service/section-master.service.ts +104 -104
  265. package/src/module/meta/service/update-form-json.service.ts +22 -22
  266. package/src/module/meta/service/user-app-mapping.service.ts +17 -17
  267. package/src/module/meta/service/view-master.service.ts +127 -127
  268. package/src/module/microservice-client/microservice-clients.module.ts +13 -13
  269. package/src/module/microservice-client/service/microservice-client-factory.ts +37 -37
  270. package/src/module/microservice-client/service/microservice-clients.ts +4 -4
  271. package/src/module/module/controller/menu.controller.ts +15 -15
  272. package/src/module/module/controller/module-access.controller.ts +134 -134
  273. package/src/module/module/entity/menu.entity.ts +43 -43
  274. package/src/module/module/entity/module-access.entity.ts +25 -25
  275. package/src/module/module/entity/module-action.entity.ts +17 -17
  276. package/src/module/module/entity/module.entity.ts +52 -52
  277. package/src/module/module/module.module.ts +42 -42
  278. package/src/module/module/repository/menu.repository.ts +191 -191
  279. package/src/module/module/repository/module-access.repository.ts +344 -344
  280. package/src/module/module/service/menu.service.ts +82 -82
  281. package/src/module/module/service/module-access.service.ts +195 -195
  282. package/src/module/notification/controller/notification.controller.ts +58 -58
  283. package/src/module/notification/controller/otp.controller.ts +117 -117
  284. package/src/module/notification/entity/notification.entity.ts +26 -26
  285. package/src/module/notification/entity/otp.entity.ts +28 -28
  286. package/src/module/notification/firebase-admin.config.ts +22 -22
  287. package/src/module/notification/notification.module.ts +69 -69
  288. package/src/module/notification/repository/otp.repository.ts +27 -27
  289. package/src/module/notification/service/email.service.ts +127 -127
  290. package/src/module/notification/service/notification.service.ts +160 -160
  291. package/src/module/notification/service/otp.service.ts +133 -133
  292. package/src/module/third-party-module/entity/third-party-api-registry.entity.ts +52 -52
  293. package/src/module/third-party-module/repository/third-party-api-registry.repository.ts +20 -20
  294. package/src/module/third-party-module/service/api-registry.service.ts +13 -13
  295. package/src/module/third-party-module/third-party.module.ts +12 -12
  296. package/src/module/user/controller/login.controller.ts +198 -198
  297. package/src/module/user/controller/user.controller.ts +40 -40
  298. package/src/module/user/dto/create-user.dto.ts +62 -62
  299. package/src/module/user/dto/update-user.dto.ts +4 -4
  300. package/src/module/user/entity/role.entity.ts +33 -33
  301. package/src/module/user/entity/user-role-mapping.entity.ts +38 -38
  302. package/src/module/user/entity/user-session.entity.ts +73 -73
  303. package/src/module/user/entity/user.entity.ts +59 -59
  304. package/src/module/user/repository/role.repository.ts +96 -96
  305. package/src/module/user/repository/user-role-mapping.repository.ts +126 -126
  306. package/src/module/user/repository/user.repository.ts +50 -50
  307. package/src/module/user/repository/userSession.repository.ts +33 -33
  308. package/src/module/user/service/login.service.ts +326 -326
  309. package/src/module/user/service/role.service.ts +189 -189
  310. package/src/module/user/service/user-role-mapping.service.ts +98 -98
  311. package/src/module/user/service/user-session.service.ts +168 -168
  312. package/src/module/user/service/user.service.ts +368 -368
  313. package/src/module/user/user.module.ts +65 -65
  314. package/src/module/workflow/controller/action-category.controller.ts +54 -54
  315. package/src/module/workflow/controller/action-resource-mapping.controller.ts +23 -23
  316. package/src/module/workflow/controller/action-template-mapping.controller.ts +35 -35
  317. package/src/module/workflow/controller/action.controller.ts +111 -111
  318. package/src/module/workflow/controller/activity-log.controller.ts +55 -55
  319. package/src/module/workflow/controller/comm-template.controller.ts +43 -43
  320. package/src/module/workflow/controller/entity-modification.controller.ts +35 -35
  321. package/src/module/workflow/controller/form-master.controller.ts +43 -43
  322. package/src/module/workflow/controller/stage-group.controller.ts +48 -48
  323. package/src/module/workflow/controller/stage.controller.ts +50 -50
  324. package/src/module/workflow/controller/task.controller.ts +77 -77
  325. package/src/module/workflow/controller/workflow-list-master.controller.ts +44 -44
  326. package/src/module/workflow/controller/workflow-meta.controller.ts +80 -80
  327. package/src/module/workflow/controller/workflow.controller.ts +67 -67
  328. package/src/module/workflow/entity/action-category.entity.ts +38 -38
  329. package/src/module/workflow/entity/action-data.entity.ts +55 -55
  330. package/src/module/workflow/entity/action-resources-mapping.entity.ts +29 -29
  331. package/src/module/workflow/entity/action-template-mapping.entity.ts +17 -17
  332. package/src/module/workflow/entity/action.entity.ts +53 -53
  333. package/src/module/workflow/entity/activity-log.entity.ts +43 -43
  334. package/src/module/workflow/entity/comm-template.entity.ts +43 -43
  335. package/src/module/workflow/entity/entity-modification.entity.ts +38 -38
  336. package/src/module/workflow/entity/form.entity.ts +25 -25
  337. package/src/module/workflow/entity/stage-action-mapping.entity.ts +17 -17
  338. package/src/module/workflow/entity/stage-group.entity.ts +23 -23
  339. package/src/module/workflow/entity/stage-movement-data.entity.ts +38 -38
  340. package/src/module/workflow/entity/stage.entity.ts +20 -20
  341. package/src/module/workflow/entity/task-data.entity.ts +88 -88
  342. package/src/module/workflow/entity/template-attach-mapper.entity.ts +30 -30
  343. package/src/module/workflow/entity/workflow-data.entity.ts +11 -11
  344. package/src/module/workflow/entity/workflow-level-mapping.entity.ts +18 -18
  345. package/src/module/workflow/entity/workflow.entity.ts +20 -20
  346. package/src/module/workflow/repository/action-category.repository.ts +79 -79
  347. package/src/module/workflow/repository/action-data.repository.ts +334 -334
  348. package/src/module/workflow/repository/action.repository.ts +329 -324
  349. package/src/module/workflow/repository/activity-log.repository.ts +148 -148
  350. package/src/module/workflow/repository/comm-template.repository.ts +149 -149
  351. package/src/module/workflow/repository/form-master.repository.ts +59 -59
  352. package/src/module/workflow/repository/stage-group.repository.ts +197 -197
  353. package/src/module/workflow/repository/stage-movement.repository.ts +246 -246
  354. package/src/module/workflow/repository/stage.repository.ts +172 -172
  355. package/src/module/workflow/repository/task.repository.ts +134 -134
  356. package/src/module/workflow/repository/workflow.repository.ts +42 -42
  357. package/src/module/workflow/service/action-category.service.ts +33 -33
  358. package/src/module/workflow/service/action-data.service.ts +62 -62
  359. package/src/module/workflow/service/action-resources-mapping.service.ts +10 -10
  360. package/src/module/workflow/service/action-template-mapping.service.ts +131 -131
  361. package/src/module/workflow/service/action.service.ts +279 -279
  362. package/src/module/workflow/service/activity-log.service.ts +107 -107
  363. package/src/module/workflow/service/comm-template.service.ts +180 -180
  364. package/src/module/workflow/service/entity-modification.service.ts +61 -61
  365. package/src/module/workflow/service/form-master.service.ts +35 -35
  366. package/src/module/workflow/service/populate-workflow.service.ts +331 -331
  367. package/src/module/workflow/service/stage-action-mapping.service.ts +5 -5
  368. package/src/module/workflow/service/stage-group.service.ts +342 -342
  369. package/src/module/workflow/service/stage.service.ts +199 -199
  370. package/src/module/workflow/service/task.service.ts +558 -558
  371. package/src/module/workflow/service/workflow-list-master.service.ts +60 -60
  372. package/src/module/workflow/service/workflow-meta.service.ts +654 -654
  373. package/src/module/workflow/service/workflow.service.ts +205 -205
  374. package/src/module/workflow/workflow.module.ts +180 -180
  375. package/src/module/workflow-automation/SCHEDULING_GUIDE.md +145 -145
  376. package/src/module/workflow-automation/controller/workflow-automation.controller.ts +43 -43
  377. package/src/module/workflow-automation/entity/workflow-automation-action.entity.ts +26 -26
  378. package/src/module/workflow-automation/entity/workflow-automation.entity.ts +40 -40
  379. package/src/module/workflow-automation/interface/action.decorator.ts +7 -7
  380. package/src/module/workflow-automation/interface/action.interface.ts +5 -5
  381. package/src/module/workflow-automation/service/action-registery.service.ts +35 -35
  382. package/src/module/workflow-automation/service/schedule-handler.service.ts +168 -168
  383. package/src/module/workflow-automation/service/workflow-automation-engine.service.ts +224 -224
  384. package/src/module/workflow-automation/service/workflow-automation.service.ts +515 -515
  385. package/src/module/workflow-automation/workflow-automation.module.ts +54 -54
  386. package/src/module/workflow-schedule/INSTALLATION.md +244 -244
  387. package/src/module/workflow-schedule/MULTI_PROJECT_GUIDE.md +196 -196
  388. package/src/module/workflow-schedule/README.md +422 -422
  389. package/src/module/workflow-schedule/constants/schedule.constants.ts +48 -48
  390. package/src/module/workflow-schedule/controller/workflow-schedule.controller.ts +255 -255
  391. package/src/module/workflow-schedule/docs/CLAUDE_CODE_GUIDE.md +510 -510
  392. package/src/module/workflow-schedule/docs/CLAUDE_CODE_PROMPT.md +362 -362
  393. package/src/module/workflow-schedule/docs/RUN_CLAUDE_CODE.sh +68 -68
  394. package/src/module/workflow-schedule/dto/create-schedule.dto.ts +147 -147
  395. package/src/module/workflow-schedule/dto/get-execution-logs.dto.ts +119 -119
  396. package/src/module/workflow-schedule/dto/update-schedule.dto.ts +96 -96
  397. package/src/module/workflow-schedule/entities/scheduled-workflow.entity.ts +148 -148
  398. package/src/module/workflow-schedule/entities/workflow-execution-log.entity.ts +154 -154
  399. package/src/module/workflow-schedule/interfaces/schedule-job-data.interface.ts +53 -53
  400. package/src/module/workflow-schedule/interfaces/workflow-schedule-options.interface.ts +12 -12
  401. package/src/module/workflow-schedule/processors/schedule.processor.ts +584 -584
  402. package/src/module/workflow-schedule/service/workflow-schedule.service.ts +600 -600
  403. package/src/module/workflow-schedule/workflow-schedule.module.ts +67 -67
  404. package/src/resources/dev.properties.yaml +31 -31
  405. package/src/resources/local.properties.yaml +27 -27
  406. package/src/resources/properties.module.ts +12 -12
  407. package/src/resources/properties.yaml.ts +11 -11
  408. package/src/resources/uat.properties.yaml +31 -31
  409. package/src/table.config.ts +133 -133
  410. package/src/utils/dto/excel-data.dto.ts +14 -14
  411. package/src/utils/dto/excelsheet-data.dto.ts +5 -5
  412. package/src/utils/service/base64util.service.ts +18 -18
  413. package/src/utils/service/clockIDGenUtil.service.ts +21 -21
  414. package/src/utils/service/codeGenerator.service.ts +22 -22
  415. package/src/utils/service/dateUtil.service.ts +17 -17
  416. package/src/utils/service/encryptUtil.service.ts +97 -97
  417. package/src/utils/service/excel-helper.service.ts +72 -72
  418. package/src/utils/service/excelUtil.service.ts +15 -15
  419. package/src/utils/service/file-util.service.ts +11 -11
  420. package/src/utils/service/json-util.service.ts +23 -23
  421. package/src/utils/service/loggingUtil.service.ts +88 -88
  422. package/src/utils/service/reflection-helper.service.ts +62 -62
  423. package/src/utils/service/wbsCodeGen.service.ts +8 -8
  424. package/src/utils/utils.module.ts +27 -27
  425. package/tsconfig.build.json +4 -4
  426. 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
+ }