rez_core 5.0.37 → 5.0.39

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