rez_core 5.0.154 → 5.0.156

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