rez_core 5.0.166 → 5.0.168

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 (420) hide show
  1. package/.idea/250218_nodejs_core.iml +11 -8
  2. package/.idea/codeStyles/Project.xml +58 -58
  3. package/.idea/codeStyles/codeStyleConfig.xml +4 -4
  4. package/.idea/inspectionProfiles/Project_Default.xml +1 -1
  5. package/.idea/modules.xml +7 -7
  6. package/.idea/prettier.xml +5 -5
  7. package/.idea/vcs.xml +5 -5
  8. package/.prettierrc +3 -3
  9. package/README.md +99 -99
  10. package/dist/module/auth/guards/role.guard.js +3 -3
  11. package/dist/module/dashboard/service/dashboard.service.js +1 -1
  12. package/dist/module/dashboard/service/dashboard.service.js.map +1 -1
  13. package/dist/module/filter/repository/saved-filter.repository.d.ts +4 -6
  14. package/dist/module/filter/repository/saved-filter.repository.js +43 -9
  15. package/dist/module/filter/repository/saved-filter.repository.js.map +1 -1
  16. package/dist/module/filter/service/filter.service.js +19 -19
  17. package/dist/module/integration/examples/usage.example.js +9 -9
  18. package/dist/module/listmaster/entity/list-master.entity.d.ts +1 -0
  19. package/dist/module/listmaster/entity/list-master.entity.js +4 -0
  20. package/dist/module/listmaster/entity/list-master.entity.js.map +1 -1
  21. package/dist/module/meta/dto/entity-table.dto.d.ts +1 -4
  22. package/dist/module/meta/repository/attribute-master.repository.js +8 -8
  23. package/dist/module/meta/service/entity-dynamic.service.js +17 -17
  24. package/dist/module/meta/service/entity-dynamic.service.js.map +1 -1
  25. package/dist/module/meta/service/entity-list.service.js +2 -2
  26. package/dist/module/meta/service/media-data.service.js +6 -6
  27. package/dist/module/meta/service/resolver.service.js +15 -15
  28. package/dist/module/module/controller/module-access.controller.d.ts +1 -1
  29. package/dist/module/module/controller/module-access.controller.js +6 -8
  30. package/dist/module/module/controller/module-access.controller.js.map +1 -1
  31. package/dist/module/module/repository/menu.repository.js +5 -5
  32. package/dist/module/module/repository/menu.repository.js.map +1 -1
  33. package/dist/module/module/service/module-access.service.d.ts +3 -4
  34. package/dist/module/module/service/module-access.service.js +7 -14
  35. package/dist/module/module/service/module-access.service.js.map +1 -1
  36. package/dist/module/notification/service/notification.service.js +6 -6
  37. package/dist/module/user/controller/login.controller.js +18 -18
  38. package/dist/module/workflow/repository/action.repository.js +2 -2
  39. package/dist/module/workflow/repository/stage.repository.js +8 -8
  40. package/dist/module/workflow/service/action-template-mapping.service.js +2 -2
  41. package/dist/module/workflow/service/action.service.js +5 -5
  42. package/dist/module/workflow/service/entity-modification.service.js +2 -2
  43. package/dist/module/workflow/service/task.service.js +8 -8
  44. package/dist/module/workflow-automation/service/schedule-handler.service.js +9 -9
  45. package/dist/tsconfig.build.tsbuildinfo +1 -1
  46. package/dist/utils/service/reflection-helper.service.js +2 -2
  47. package/docs/modules/event-driven-integration-design.md +91 -91
  48. package/docs/modules/integration.md +250 -250
  49. package/eslint.config.mjs +34 -34
  50. package/nest-cli.json +14 -14
  51. package/package.json +125 -125
  52. package/src/app.controller.ts +12 -12
  53. package/src/app.module.ts +68 -68
  54. package/src/app.service.ts +8 -8
  55. package/src/config/bull.config.ts +69 -69
  56. package/src/config/config.module.ts +17 -17
  57. package/src/config/database.config.ts +48 -48
  58. package/src/constant/global.constant.ts +67 -67
  59. package/src/core.module.ts +94 -94
  60. package/src/decorators/roles.decorator.ts +7 -7
  61. package/src/dtos/response.dto.ts +6 -6
  62. package/src/dtos/response.ts +5 -5
  63. package/src/index.ts +1 -1
  64. package/src/module/auth/auth.module.ts +49 -49
  65. package/src/module/auth/controller/auth.controller.ts +28 -28
  66. package/src/module/auth/guards/google-auth.guard.ts +9 -9
  67. package/src/module/auth/guards/jwt.guard.ts +22 -22
  68. package/src/module/auth/guards/role.guard.ts +68 -68
  69. package/src/module/auth/services/auth.service.ts +56 -56
  70. package/src/module/auth/services/jwt.service.ts +11 -11
  71. package/src/module/auth/strategies/google.strategy.ts +54 -54
  72. package/src/module/auth/strategies/jwt.strategy.ts +58 -58
  73. package/src/module/auth/strategies/local.strategy.ts +13 -13
  74. package/src/module/dashboard/controller/dashboard.controller.ts +36 -36
  75. package/src/module/dashboard/dashboard.module.ts +21 -21
  76. package/src/module/dashboard/entity/dashboard_page_data.entity.ts +27 -27
  77. package/src/module/dashboard/entity/widget_master.entity.ts +18 -18
  78. package/src/module/dashboard/repository/dashboard.repository.ts +51 -51
  79. package/src/module/dashboard/service/dashboard.service.ts +73 -73
  80. package/src/module/enterprise/controller/organization.controller.ts +36 -36
  81. package/src/module/enterprise/enterprise.module.ts +30 -30
  82. package/src/module/enterprise/entity/enterprise.entity.ts +37 -37
  83. package/src/module/enterprise/entity/organization-app-mapping.entity.ts +13 -13
  84. package/src/module/enterprise/entity/organization.entity.ts +92 -92
  85. package/src/module/enterprise/repository/enterprise.repository.ts +31 -31
  86. package/src/module/enterprise/repository/organization.repository.ts +26 -26
  87. package/src/module/enterprise/repository/school.repository.ts +272 -272
  88. package/src/module/enterprise/service/brand.service.ts +5 -5
  89. package/src/module/enterprise/service/enterprise.service.ts +16 -16
  90. package/src/module/enterprise/service/organization-app-mapping.service.ts +4 -4
  91. package/src/module/enterprise/service/organization.service.ts +145 -145
  92. package/src/module/entity_json/controller/entity_json.controller.ts +47 -47
  93. package/src/module/entity_json/entity/entityJson.entity.ts +39 -39
  94. package/src/module/entity_json/entity_json.module.ts +18 -18
  95. package/src/module/entity_json/service/entityJson.repository.ts +37 -37
  96. package/src/module/entity_json/service/entity_json.service.ts +242 -242
  97. package/src/module/export/controller/export.controller.ts +83 -83
  98. package/src/module/export/export.module.ts +14 -14
  99. package/src/module/export/service/export.service.ts +105 -105
  100. package/src/module/filter/controller/filter.controller.ts +84 -84
  101. package/src/module/filter/dto/filter-request.dto.ts +39 -39
  102. package/src/module/filter/entity/saved-filter-detail.entity.ts +41 -41
  103. package/src/module/filter/entity/saved-filter-master.entity.ts +32 -32
  104. package/src/module/filter/filter.module.ts +33 -33
  105. package/src/module/filter/repository/saved-filter.repository.ts +256 -200
  106. package/src/module/filter/repository/saved.filter-detail.repository.ts +19 -19
  107. package/src/module/filter/service/filter-evaluator.service.ts +82 -82
  108. package/src/module/filter/service/filter.service.ts +1319 -1319
  109. package/src/module/filter/service/saved-filter.service.ts +164 -164
  110. package/src/module/ics/controller/ics.controller.ts +21 -21
  111. package/src/module/ics/dto/ics.dto.ts +55 -55
  112. package/src/module/ics/ics.module.ts +13 -13
  113. package/src/module/ics/service/ics.service.ts +57 -57
  114. package/src/module/integration/controller/calender-event.controller.ts +31 -31
  115. package/src/module/integration/controller/integration.controller.ts +662 -662
  116. package/src/module/integration/controller/wrapper.controller.ts +37 -37
  117. package/src/module/integration/dto/create-config.dto.ts +526 -526
  118. package/src/module/integration/entity/integration-config.entity.ts +112 -112
  119. package/src/module/integration/entity/integration-entity-mapper.entity.ts +14 -14
  120. package/src/module/integration/entity/integration-source.entity.ts +17 -17
  121. package/src/module/integration/entity/user-integration.entity.ts +71 -71
  122. package/src/module/integration/examples/usage.example.ts +338 -338
  123. package/src/module/integration/factories/base.factory.ts +7 -7
  124. package/src/module/integration/factories/email.factory.ts +49 -49
  125. package/src/module/integration/factories/integration.factory.ts +121 -121
  126. package/src/module/integration/factories/sms.factory.ts +51 -51
  127. package/src/module/integration/factories/telephone.factory.ts +41 -41
  128. package/src/module/integration/factories/whatsapp.factory.ts +56 -56
  129. package/src/module/integration/integration.module.ts +110 -110
  130. package/src/module/integration/service/calendar-event.service.ts +118 -118
  131. package/src/module/integration/service/integration-entity-mapper.service.ts +17 -17
  132. package/src/module/integration/service/integration-queue.service.ts +229 -229
  133. package/src/module/integration/service/integration.service.ts +2634 -2634
  134. package/src/module/integration/service/oauth.service.ts +224 -224
  135. package/src/module/integration/service/wrapper.service.ts +753 -753
  136. package/src/module/integration/strategies/email/gmail-api.strategy.ts +280 -280
  137. package/src/module/integration/strategies/email/outlook-api.strategy.ts +44 -44
  138. package/src/module/integration/strategies/email/outlook.strategy.ts +64 -64
  139. package/src/module/integration/strategies/email/sendgrid-api.strategy.ts +260 -260
  140. package/src/module/integration/strategies/integration.strategy.ts +97 -97
  141. package/src/module/integration/strategies/sms/gupshup-sms.strategy.ts +146 -146
  142. package/src/module/integration/strategies/sms/msg91-sms.strategy.ts +164 -164
  143. package/src/module/integration/strategies/sms/tubelight-sms.strategy.ts +163 -163
  144. package/src/module/integration/strategies/telephone/ozonetel-voice.strategy.ts +238 -238
  145. package/src/module/integration/strategies/telephone/tubelight-voice.strategy.ts +210 -210
  146. package/src/module/integration/strategies/whatsapp/gupshup-whatsapp.strategy.ts +359 -359
  147. package/src/module/integration/strategies/whatsapp/tubelight-whatsapp.strategy.ts +372 -372
  148. package/src/module/integration/strategies/whatsapp/whatsapp-cloud.strategy.ts +403 -403
  149. package/src/module/integration/strategies/whatsapp/whatsapp.strategy.ts +57 -57
  150. package/src/module/layout/controller/layout.controller.ts +47 -47
  151. package/src/module/layout/entity/header-items.entity.ts +28 -28
  152. package/src/module/layout/entity/header-section.entity.ts +19 -19
  153. package/src/module/layout/layout.module.ts +21 -21
  154. package/src/module/layout/repository/header-items.repository.ts +18 -18
  155. package/src/module/layout/repository/header-section.repository.ts +22 -22
  156. package/src/module/layout/service/header-section.service.ts +25 -25
  157. package/src/module/layout_preference/controller/layout_preference.controller.ts +73 -73
  158. package/src/module/layout_preference/entity/layout_preference.entity.ts +28 -28
  159. package/src/module/layout_preference/layout_preference.module.ts +22 -22
  160. package/src/module/layout_preference/repository/layout_preference.repository.ts +65 -65
  161. package/src/module/layout_preference/service/layout_preference.service.ts +191 -191
  162. package/src/module/lead/controller/lead.controller.ts +30 -30
  163. package/src/module/lead/lead.module.ts +14 -14
  164. package/src/module/lead/repository/lead.repository.ts +41 -41
  165. package/src/module/lead/service/lead.service.ts +54 -54
  166. package/src/module/linked_attributes/controller/linked_attributes.controller.ts +37 -37
  167. package/src/module/linked_attributes/entity/linked_attribute.entity.ts +51 -51
  168. package/src/module/linked_attributes/linked_attributes.module.ts +16 -16
  169. package/src/module/linked_attributes/repository/linked_attribute.repository.ts +12 -12
  170. package/src/module/linked_attributes/service/linked_attributes.service.ts +75 -75
  171. package/src/module/listmaster/controller/list-master.controller.ts +230 -230
  172. package/src/module/listmaster/entity/list-master-items.entity.ts +43 -43
  173. package/src/module/listmaster/entity/list-master.entity.ts +33 -30
  174. package/src/module/listmaster/listmaster.module.ts +46 -46
  175. package/src/module/listmaster/repository/list-master-items.repository.ts +173 -173
  176. package/src/module/listmaster/repository/list-master.repository.ts +56 -56
  177. package/src/module/listmaster/service/list-master-engine.ts +19 -19
  178. package/src/module/listmaster/service/list-master-extension.interface.ts +4 -4
  179. package/src/module/listmaster/service/list-master-item.service.ts +281 -281
  180. package/src/module/listmaster/service/list-master-registry.ts +15 -15
  181. package/src/module/listmaster/service/list-master.service.ts +535 -535
  182. package/src/module/mapper/controller/field-mapper.controller.ts +76 -76
  183. package/src/module/mapper/controller/mapper.controller.ts +20 -20
  184. package/src/module/mapper/dto/field-mapper.dto.ts +14 -14
  185. package/src/module/mapper/entity/field-lovs.entity.ts +19 -19
  186. package/src/module/mapper/entity/field-mapper.entity.ts +53 -53
  187. package/src/module/mapper/entity/mapper.entity.ts +16 -16
  188. package/src/module/mapper/mapper.module.ts +35 -35
  189. package/src/module/mapper/repository/field-lovs.repository.ts +35 -35
  190. package/src/module/mapper/repository/field-mapper.repository.ts +42 -42
  191. package/src/module/mapper/repository/mapper.repository.ts +32 -32
  192. package/src/module/mapper/service/field-mapper.service.ts +269 -269
  193. package/src/module/mapper/service/mapper.service.ts +81 -81
  194. package/src/module/master/controller/master.controller.ts +74 -74
  195. package/src/module/master/service/master.service.ts +483 -483
  196. package/src/module/meta/controller/app-master.controller.ts +38 -38
  197. package/src/module/meta/controller/attribute-master.controller.ts +84 -84
  198. package/src/module/meta/controller/entity-dynamic.controller.ts +125 -125
  199. package/src/module/meta/controller/entity-master.controller.ts +41 -41
  200. package/src/module/meta/controller/entity-relation.controller.ts +36 -36
  201. package/src/module/meta/controller/entity.controller.ts +342 -342
  202. package/src/module/meta/controller/entity.public.controller.ts +75 -75
  203. package/src/module/meta/controller/media.controller.ts +135 -135
  204. package/src/module/meta/controller/meta.controller.ts +96 -96
  205. package/src/module/meta/controller/view-master.controller.ts +86 -86
  206. package/src/module/meta/dto/entity-list-data.dto.ts +6 -6
  207. package/src/module/meta/dto/entity-tab.dto.ts +4 -4
  208. package/src/module/meta/dto/entity-table.dto.ts +12 -12
  209. package/src/module/meta/entity/app-master.entity.ts +37 -37
  210. package/src/module/meta/entity/attribute-master.entity.ts +92 -92
  211. package/src/module/meta/entity/base-entity.entity.ts +75 -75
  212. package/src/module/meta/entity/entity-master.entity.ts +85 -85
  213. package/src/module/meta/entity/entity-relation-data.entity.ts +29 -29
  214. package/src/module/meta/entity/entity-relation.entity.ts +23 -23
  215. package/src/module/meta/entity/entity-table-column.entity.ts +61 -61
  216. package/src/module/meta/entity/entity-table.entity.ts +50 -50
  217. package/src/module/meta/entity/media-data.entity.ts +32 -32
  218. package/src/module/meta/entity/preference.entity.ts +62 -62
  219. package/src/module/meta/entity/view-master.entity.ts +41 -41
  220. package/src/module/meta/entity.module.ts +168 -168
  221. package/src/module/meta/repository/app-master.repository.ts +20 -20
  222. package/src/module/meta/repository/attribute-master.repository.ts +156 -156
  223. package/src/module/meta/repository/entity-attribute-update.repository.ts +48 -48
  224. package/src/module/meta/repository/entity-master.repository.ts +110 -110
  225. package/src/module/meta/repository/entity-relation.repository.ts +22 -22
  226. package/src/module/meta/repository/entity-table-column.repository.ts +39 -39
  227. package/src/module/meta/repository/entity-table.repository.ts +53 -53
  228. package/src/module/meta/repository/media-data.repository.ts +50 -50
  229. package/src/module/meta/repository/preference.repository.ts +20 -20
  230. package/src/module/meta/repository/user-app-mapping.repository.ts +28 -28
  231. package/src/module/meta/repository/view-master.repository.ts +42 -42
  232. package/src/module/meta/service/app-master.service.ts +37 -37
  233. package/src/module/meta/service/attribute-master.service.ts +130 -130
  234. package/src/module/meta/service/common.service.ts +9 -9
  235. package/src/module/meta/service/entity-attribute-update.service.ts +26 -26
  236. package/src/module/meta/service/entity-dynamic.service.ts +822 -822
  237. package/src/module/meta/service/entity-list.service.ts +201 -201
  238. package/src/module/meta/service/entity-master.service.ts +171 -171
  239. package/src/module/meta/service/entity-realation-data.service.ts +9 -9
  240. package/src/module/meta/service/entity-relation.service.ts +74 -74
  241. package/src/module/meta/service/entity-service-impl.service.ts +439 -439
  242. package/src/module/meta/service/entity-table-column.service.ts +39 -39
  243. package/src/module/meta/service/entity-table.service.ts +157 -157
  244. package/src/module/meta/service/entity-validation.service.ts +187 -187
  245. package/src/module/meta/service/entity.service.ts +59 -59
  246. package/src/module/meta/service/field-group.service.ts +103 -103
  247. package/src/module/meta/service/media-data.service.ts +591 -591
  248. package/src/module/meta/service/populate-meta.service.ts +222 -222
  249. package/src/module/meta/service/preference.service.ts +16 -16
  250. package/src/module/meta/service/resolver.service.ts +291 -291
  251. package/src/module/meta/service/section-master.service.ts +104 -104
  252. package/src/module/meta/service/update-form-json.service.ts +22 -22
  253. package/src/module/meta/service/user-app-mapping.service.ts +17 -17
  254. package/src/module/meta/service/view-master.service.ts +127 -127
  255. package/src/module/microservice-client/microservice-clients.module.ts +13 -13
  256. package/src/module/microservice-client/service/microservice-client-factory.ts +37 -37
  257. package/src/module/microservice-client/service/microservice-clients.ts +4 -4
  258. package/src/module/module/controller/menu.controller.ts +15 -15
  259. package/src/module/module/controller/module-access.controller.ts +133 -134
  260. package/src/module/module/entity/menu.entity.ts +43 -43
  261. package/src/module/module/entity/module-access.entity.ts +25 -25
  262. package/src/module/module/entity/module-action.entity.ts +17 -17
  263. package/src/module/module/entity/module.entity.ts +52 -52
  264. package/src/module/module/module.module.ts +42 -42
  265. package/src/module/module/repository/menu.repository.ts +186 -186
  266. package/src/module/module/repository/module-access.repository.ts +344 -344
  267. package/src/module/module/service/menu.service.ts +82 -82
  268. package/src/module/module/service/module-access.service.ts +189 -199
  269. package/src/module/notification/controller/notification.controller.ts +58 -58
  270. package/src/module/notification/controller/otp.controller.ts +117 -117
  271. package/src/module/notification/entity/notification.entity.ts +26 -26
  272. package/src/module/notification/entity/otp.entity.ts +28 -28
  273. package/src/module/notification/firebase-admin.config.ts +22 -22
  274. package/src/module/notification/notification.module.ts +69 -69
  275. package/src/module/notification/repository/otp.repository.ts +27 -27
  276. package/src/module/notification/service/email.service.ts +127 -127
  277. package/src/module/notification/service/notification.service.ts +164 -164
  278. package/src/module/notification/service/otp.service.ts +133 -133
  279. package/src/module/third-party-module/entity/third-party-api-registry.entity.ts +52 -52
  280. package/src/module/third-party-module/repository/third-party-api-registry.repository.ts +20 -20
  281. package/src/module/third-party-module/service/api-registry.service.ts +13 -13
  282. package/src/module/third-party-module/third-party.module.ts +12 -12
  283. package/src/module/user/controller/login.controller.ts +198 -198
  284. package/src/module/user/controller/user.controller.ts +40 -40
  285. package/src/module/user/dto/create-user.dto.ts +62 -62
  286. package/src/module/user/dto/update-user.dto.ts +4 -4
  287. package/src/module/user/entity/role.entity.ts +33 -33
  288. package/src/module/user/entity/user-role-mapping.entity.ts +38 -38
  289. package/src/module/user/entity/user-session.entity.ts +73 -73
  290. package/src/module/user/entity/user.entity.ts +59 -59
  291. package/src/module/user/repository/role.repository.ts +96 -96
  292. package/src/module/user/repository/user-role-mapping.repository.ts +126 -126
  293. package/src/module/user/repository/user.repository.ts +50 -50
  294. package/src/module/user/repository/userSession.repository.ts +33 -33
  295. package/src/module/user/service/login.service.ts +326 -326
  296. package/src/module/user/service/role.service.ts +197 -197
  297. package/src/module/user/service/user-role-mapping.service.ts +98 -98
  298. package/src/module/user/service/user-session.service.ts +200 -200
  299. package/src/module/user/service/user.service.ts +368 -368
  300. package/src/module/user/user.module.ts +65 -65
  301. package/src/module/workflow/controller/action-category.controller.ts +54 -54
  302. package/src/module/workflow/controller/action-resource-mapping.controller.ts +23 -23
  303. package/src/module/workflow/controller/action-template-mapping.controller.ts +35 -35
  304. package/src/module/workflow/controller/action.controller.ts +111 -111
  305. package/src/module/workflow/controller/activity-log.controller.ts +55 -55
  306. package/src/module/workflow/controller/comm-template.controller.ts +43 -43
  307. package/src/module/workflow/controller/entity-modification.controller.ts +35 -35
  308. package/src/module/workflow/controller/form-master.controller.ts +43 -43
  309. package/src/module/workflow/controller/stage-group.controller.ts +48 -48
  310. package/src/module/workflow/controller/stage.controller.ts +50 -50
  311. package/src/module/workflow/controller/task.controller.ts +77 -77
  312. package/src/module/workflow/controller/workflow-list-master.controller.ts +44 -44
  313. package/src/module/workflow/controller/workflow-meta.controller.ts +80 -80
  314. package/src/module/workflow/controller/workflow.controller.ts +67 -67
  315. package/src/module/workflow/entity/action-category.entity.ts +38 -38
  316. package/src/module/workflow/entity/action-data.entity.ts +55 -55
  317. package/src/module/workflow/entity/action-resources-mapping.entity.ts +29 -29
  318. package/src/module/workflow/entity/action-template-mapping.entity.ts +17 -17
  319. package/src/module/workflow/entity/action.entity.ts +53 -53
  320. package/src/module/workflow/entity/activity-log.entity.ts +43 -43
  321. package/src/module/workflow/entity/comm-template.entity.ts +43 -43
  322. package/src/module/workflow/entity/entity-modification.entity.ts +38 -38
  323. package/src/module/workflow/entity/form.entity.ts +25 -25
  324. package/src/module/workflow/entity/stage-action-mapping.entity.ts +17 -17
  325. package/src/module/workflow/entity/stage-group.entity.ts +23 -23
  326. package/src/module/workflow/entity/stage-movement-data.entity.ts +38 -38
  327. package/src/module/workflow/entity/stage.entity.ts +20 -20
  328. package/src/module/workflow/entity/task-data.entity.ts +88 -88
  329. package/src/module/workflow/entity/template-attach-mapper.entity.ts +30 -30
  330. package/src/module/workflow/entity/workflow-data.entity.ts +11 -11
  331. package/src/module/workflow/entity/workflow-level-mapping.entity.ts +18 -18
  332. package/src/module/workflow/entity/workflow.entity.ts +20 -20
  333. package/src/module/workflow/repository/action-category.repository.ts +79 -79
  334. package/src/module/workflow/repository/action-data.repository.ts +346 -346
  335. package/src/module/workflow/repository/action.repository.ts +339 -339
  336. package/src/module/workflow/repository/activity-log.repository.ts +148 -148
  337. package/src/module/workflow/repository/comm-template.repository.ts +157 -157
  338. package/src/module/workflow/repository/form-master.repository.ts +50 -50
  339. package/src/module/workflow/repository/stage-group.repository.ts +186 -186
  340. package/src/module/workflow/repository/stage-movement.repository.ts +257 -257
  341. package/src/module/workflow/repository/stage.repository.ts +160 -160
  342. package/src/module/workflow/repository/task.repository.ts +151 -151
  343. package/src/module/workflow/repository/workflow.repository.ts +42 -42
  344. package/src/module/workflow/service/action-category.service.ts +33 -33
  345. package/src/module/workflow/service/action-data.service.ts +62 -62
  346. package/src/module/workflow/service/action-resources-mapping.service.ts +10 -10
  347. package/src/module/workflow/service/action-template-mapping.service.ts +140 -140
  348. package/src/module/workflow/service/action.service.ts +302 -302
  349. package/src/module/workflow/service/activity-log.service.ts +107 -107
  350. package/src/module/workflow/service/comm-template.service.ts +180 -180
  351. package/src/module/workflow/service/entity-modification.service.ts +61 -61
  352. package/src/module/workflow/service/form-master.service.ts +35 -35
  353. package/src/module/workflow/service/populate-workflow.service.ts +320 -320
  354. package/src/module/workflow/service/stage-action-mapping.service.ts +5 -5
  355. package/src/module/workflow/service/stage-group.service.ts +344 -344
  356. package/src/module/workflow/service/stage.service.ts +207 -207
  357. package/src/module/workflow/service/task.service.ts +550 -550
  358. package/src/module/workflow/service/workflow-list-master.service.ts +68 -68
  359. package/src/module/workflow/service/workflow-meta.service.ts +639 -639
  360. package/src/module/workflow/service/workflow.service.ts +213 -213
  361. package/src/module/workflow/workflow.module.ts +180 -180
  362. package/src/module/workflow-automation/SCHEDULING_GUIDE.md +145 -145
  363. package/src/module/workflow-automation/controller/workflow-automation.controller.ts +43 -43
  364. package/src/module/workflow-automation/entity/workflow-automation-action.entity.ts +26 -26
  365. package/src/module/workflow-automation/entity/workflow-automation.entity.ts +40 -40
  366. package/src/module/workflow-automation/interface/action.decorator.ts +7 -7
  367. package/src/module/workflow-automation/interface/action.interface.ts +5 -5
  368. package/src/module/workflow-automation/service/action-registery.service.ts +35 -35
  369. package/src/module/workflow-automation/service/schedule-handler.service.ts +168 -168
  370. package/src/module/workflow-automation/service/workflow-automation-engine.service.ts +219 -219
  371. package/src/module/workflow-automation/service/workflow-automation.service.ts +515 -515
  372. package/src/module/workflow-automation/workflow-automation.module.ts +54 -54
  373. package/src/module/workflow-schedule/INSTALLATION.md +244 -244
  374. package/src/module/workflow-schedule/MULTI_PROJECT_GUIDE.md +196 -196
  375. package/src/module/workflow-schedule/README.md +422 -422
  376. package/src/module/workflow-schedule/constants/schedule.constants.ts +48 -48
  377. package/src/module/workflow-schedule/controller/workflow-schedule.controller.ts +255 -255
  378. package/src/module/workflow-schedule/docs/CLAUDE_CODE_GUIDE.md +510 -510
  379. package/src/module/workflow-schedule/docs/CLAUDE_CODE_PROMPT.md +362 -362
  380. package/src/module/workflow-schedule/docs/RUN_CLAUDE_CODE.sh +68 -68
  381. package/src/module/workflow-schedule/dto/create-schedule.dto.ts +147 -147
  382. package/src/module/workflow-schedule/dto/get-execution-logs.dto.ts +119 -119
  383. package/src/module/workflow-schedule/dto/update-schedule.dto.ts +96 -96
  384. package/src/module/workflow-schedule/entities/scheduled-workflow.entity.ts +148 -148
  385. package/src/module/workflow-schedule/entities/workflow-execution-log.entity.ts +154 -154
  386. package/src/module/workflow-schedule/interfaces/schedule-job-data.interface.ts +53 -53
  387. package/src/module/workflow-schedule/interfaces/workflow-schedule-options.interface.ts +12 -12
  388. package/src/module/workflow-schedule/processors/schedule.processor.ts +620 -620
  389. package/src/module/workflow-schedule/service/workflow-schedule.service.ts +598 -598
  390. package/src/module/workflow-schedule/workflow-schedule.module.ts +67 -67
  391. package/src/resources/dev.properties.yaml +31 -31
  392. package/src/resources/local.properties.yaml +27 -27
  393. package/src/resources/properties.module.ts +12 -12
  394. package/src/resources/properties.yaml.ts +11 -11
  395. package/src/resources/uat.properties.yaml +31 -31
  396. package/src/table.config.ts +133 -133
  397. package/src/utils/dto/excel-data.dto.ts +14 -14
  398. package/src/utils/dto/excelsheet-data.dto.ts +5 -5
  399. package/src/utils/service/base64util.service.ts +18 -18
  400. package/src/utils/service/clockIDGenUtil.service.ts +21 -21
  401. package/src/utils/service/codeGenerator.service.ts +22 -22
  402. package/src/utils/service/dateUtil.service.ts +17 -17
  403. package/src/utils/service/encryptUtil.service.ts +97 -97
  404. package/src/utils/service/excel-helper.service.ts +72 -72
  405. package/src/utils/service/excelUtil.service.ts +15 -15
  406. package/src/utils/service/file-util.service.ts +11 -11
  407. package/src/utils/service/json-util.service.ts +23 -23
  408. package/src/utils/service/loggingUtil.service.ts +88 -88
  409. package/src/utils/service/reflection-helper.service.ts +62 -62
  410. package/src/utils/service/wbsCodeGen.service.ts +8 -8
  411. package/src/utils/utils.module.ts +27 -27
  412. package/tsconfig.build.json +4 -4
  413. package/tsconfig.json +24 -24
  414. package/.claude/settings.local.json +0 -26
  415. package/.idea/copilot.data.migration.agent.xml +0 -6
  416. package/.idea/copilot.data.migration.ask.xml +0 -6
  417. package/.idea/copilot.data.migration.ask2agent.xml +0 -6
  418. package/.idea/copilot.data.migration.edit.xml +0 -6
  419. package/.idea/misc.xml +0 -6
  420. package/server.log +0 -850
@@ -1,822 +1,822 @@
1
- import { BadRequestException, Injectable } from '@nestjs/common';
2
- import { STATUS_ACTIVE } from 'src/constant/global.constant';
3
- import { EntityManager, In } from 'typeorm';
4
- import { UserData } from 'src/module/user/entity/user.entity';
5
- import { MediaDataService } from './media-data.service';
6
- import { ResolverService } from './resolver.service';
7
- import { EntityMasterRepository } from '../repository/entity-master.repository';
8
- import { ReflectionHelper } from '../../../utils/service/reflection-helper.service';
9
- import { ConfigService } from '@nestjs/config';
10
-
11
- @Injectable()
12
- export class EntityDynamicService {
13
- constructor(
14
- private readonly entityManager: EntityManager,
15
- private readonly mediaDataService: MediaDataService,
16
- private readonly resolverService: ResolverService,
17
- private readonly entityMasterRepo: EntityMasterRepository,
18
- private readonly reflectionHelper: ReflectionHelper,
19
- private readonly configService: ConfigService,
20
- ) {
21
- }
22
- schema = this.configService.get('DB_SCHEMA');
23
-
24
- // -----------------------------
25
- async createEntity(
26
- entityType: string,
27
- entityData: Record<string, any>,
28
- loggedInUser: any,
29
- mainID?: number,
30
- ): Promise<any> {
31
- const organizationId = loggedInUser.organization_id;
32
-
33
- const tableName = await this.getTableName(entityType, organizationId);
34
- const validAttributes = await this.getAttributeCodes(
35
- entityType,
36
- organizationId,
37
- );
38
-
39
- // -------------------------------------------------------
40
- // AUTO fields
41
- // -------------------------------------------------------
42
- entityData.created_date = new Date();
43
-
44
- if (loggedInUser) {
45
- entityData.created_by = loggedInUser.id;
46
-
47
- if (!entityData.organization_id)
48
- entityData.organization_id = loggedInUser.organization_id;
49
-
50
- if (!entityData.enterprise_id)
51
- entityData.enterprise_id = loggedInUser.enterprise_id;
52
-
53
- if (!entityData.level_type)
54
- entityData.level_type = loggedInUser.level_type;
55
-
56
- if (!entityData.level_id) entityData.level_id = loggedInUser.level_id;
57
-
58
- if (!entityData.entity_type) entityData.entity_type = entityType;
59
- }
60
-
61
- // -------------------------------------------------------
62
- // STATUS
63
- // -------------------------------------------------------
64
- const listMasterItemsRepo =
65
- this.reflectionHelper.getRepoService('ListMasterItems');
66
- const statusList = listMasterItemsRepo.find({
67
- where: {
68
- code: STATUS_ACTIVE,
69
- organization_id: organizationId,
70
- },
71
- });
72
-
73
- if (!entityData.status) entityData.status = statusList[0]?.id;
74
-
75
- // -------------------------------------------------------
76
- // AUTO-CODE GENERATION (POSTGRES SAFE)
77
- // -------------------------------------------------------
78
- const entityMaster = await this.entityMasterRepo.getEntityByMappedEntityType(entityType, loggedInUser.organization_id)
79
-
80
- if (!entityData.code && entityData.entity_type && entityMaster) {
81
- // Extract integer suffix
82
- // const repo = this.reflectionHelper.getRepoService(entityMaster.entity_data_class);
83
-
84
- const result = this.entityManager.query(
85
- `SELECT MAX(id) as seq_no FROM ${this.schema}.${entityMaster.db_table_name}`
86
- )
87
-
88
- let maxSeq = Number(result[0]?.max_seq_no) || 0;
89
- maxSeq++;
90
- entityData.code = `${entityData.entity_type}${maxSeq}`;
91
- }
92
-
93
- // -------------------------------------------------------
94
- // Parent ID
95
- // -------------------------------------------------------
96
- if (mainID) {
97
- entityData.parent_id = mainID;
98
- }
99
-
100
- // -------------------------------------------------------
101
- // BYPASS COLUMNS
102
- // -------------------------------------------------------
103
- const bypassColumns = [
104
- 'created_date',
105
- 'created_by',
106
- 'organization_id',
107
- 'enterprise_id',
108
- 'level_type',
109
- 'level_id',
110
- 'status',
111
- 'entity_type',
112
- 'code',
113
- 'parent_id',
114
- ];
115
-
116
- for (const col of bypassColumns) {
117
- if (!validAttributes.some((a) => a.attribute_key === col)) {
118
- validAttributes.push({
119
- attribute_key: col,
120
- is_hidden: false,
121
- db_datatype: 'text',
122
- element_type: 'text',
123
- });
124
- }
125
- }
126
-
127
- // -------------------------------------------------------
128
- // BUILD INSERT QUERY (POSTGRES FORMAT)
129
- // -------------------------------------------------------
130
- const columns: string[] = [];
131
- const values: any[] = [];
132
- let idx = 1;
133
- const placeholders: string[] = [];
134
-
135
- for (const attr of validAttributes) {
136
- if (attr.attribute_key === 'id') continue;
137
-
138
- columns.push(attr.attribute_key);
139
- values.push(entityData[attr.attribute_key] ?? null);
140
- placeholders.push(`$${idx++}`);
141
- }
142
-
143
- const colList = columns.map((c) => `"${c}"`).join(', ');
144
- const placeholderList = placeholders.join(', ');
145
-
146
-
147
- const sql = `
148
- INSERT INTO ${this.schema}.${tableName} (${colList})
149
- VALUES (${placeholderList}) RETURNING id
150
- `;
151
-
152
- const result = await this.entityManager.query(sql, values);
153
- return result[0];
154
- }
155
-
156
- // ----------------------------- get entity with relations
157
-
158
- // ----------------------------- create entity with relations
159
- async createEntityWithRelation(
160
- entityType: string,
161
- data: Record<string, any>,
162
- loggedInUser: any,
163
- ): Promise<any> {
164
- const organizationId = loggedInUser.organization_id;
165
-
166
- const repo = this.reflectionHelper.getRepoService('EntityRelation');
167
-
168
- const getRelation = await repo.find({
169
- where: {
170
- organization_id: organizationId,
171
- source_entity_type: entityType,
172
- },
173
- });
174
-
175
- const { mappedEntities, ...mainData } = data;
176
-
177
- // create main entity
178
- const createdEntityData = await this.createEntity(
179
- entityType,
180
- mainData,
181
- loggedInUser,
182
- );
183
- const mainID = createdEntityData.insertId || createdEntityData.id;
184
-
185
- if (mappedEntities && getRelation.length > 0) {
186
- for (const relation of getRelation) {
187
- const targetEntityType = relation.target_entity_type;
188
- const relationType = relation.relation_type;
189
- const relationId = relation.relation_id;
190
-
191
- if (!mappedEntities[targetEntityType]) continue;
192
-
193
- // normalize: always array
194
- const entityDataArray = Array.isArray(mappedEntities[targetEntityType])
195
- ? mappedEntities[targetEntityType]
196
- : [mappedEntities[targetEntityType]];
197
-
198
- for (const item of entityDataArray) {
199
- const itemWithRef = {
200
- ...item,
201
- entity_type: targetEntityType,
202
- };
203
-
204
- const createdRelatedEntity = await this.createEntity(
205
- targetEntityType,
206
- itemWithRef,
207
- loggedInUser,
208
- mainID, // this will pass for parent_id
209
- );
210
-
211
- const relationRepo =
212
- this.reflectionHelper.getRepoService('EntityRelationData');
213
-
214
- await relationRepo.save({
215
- organizationId: organizationId,
216
- source_entity_id: mainID,
217
- source_entity_type: entityType,
218
- target_entity_id: createdRelatedEntity.id,
219
- target_entity_type: targetEntityType,
220
- relation_type: relationType,
221
- });
222
- }
223
- }
224
- }
225
-
226
- return {
227
- mainEntity: {
228
- id: mainID,
229
- entityType,
230
- data: createdEntityData,
231
- },
232
- };
233
- }
234
-
235
- // ----------------------------- get entity with relations
236
- async getEntityWithRelation(
237
- entityType: string,
238
- id: number | string,
239
- loggedInUser: any,
240
- ): Promise<any> {
241
- const mainEntity = await this.getEntityByDataSource(
242
- entityType,
243
- id,
244
- loggedInUser,
245
- );
246
- const relationRepo = this.reflectionHelper.getRepoService('EntityRelation');
247
- const relatedEntityRepo =
248
- this.reflectionHelper.getRepoService('EntityRelationData');
249
-
250
- const relations = await relationRepo.find({
251
- where: {
252
- source_entity_type: entityType,
253
- organization_id: loggedInUser.organization_id,
254
- },
255
- });
256
-
257
- const targetTypes = relations.map((r) => r.target_entity_type);
258
-
259
- if (targetTypes.length === 0) {
260
- return {
261
- entity_type: entityType,
262
- ...mainEntity,
263
- };
264
- }
265
-
266
- const relatedEntities = await relatedEntityRepo.find({
267
- where: {
268
- source_entity_id: id,
269
- target_entity_type: In(targetTypes),
270
- },
271
- });
272
-
273
- // Format response to match create entity structure
274
- const response: any = {
275
- entity_type: entityType,
276
- ...mainEntity,
277
- };
278
-
279
- if (relatedEntities.length > 0) {
280
- response.mappedEntities = await this.formatMappedEntities(
281
- relatedEntities,
282
- loggedInUser,
283
- );
284
- }
285
-
286
- return response;
287
- }
288
-
289
- // ----------------------------- formatMappedEntities
290
- async formatMappedEntities(
291
- relatedEntities: any[],
292
- loggedInUser: any,
293
- ): Promise<any> {
294
- const mappedEntities: any = {};
295
-
296
- for (const relation of relatedEntities) {
297
- const targetEntityType = relation.target_entity_type;
298
- const targetEntityId = relation.target_entity_id;
299
-
300
- const entityData = await this.getEntity(
301
- targetEntityType,
302
- targetEntityId,
303
- loggedInUser,
304
- );
305
-
306
- if (!mappedEntities[targetEntityType]) {
307
- mappedEntities[targetEntityType] = [];
308
- }
309
-
310
- mappedEntities[targetEntityType].push(entityData);
311
- }
312
-
313
- return mappedEntities;
314
- }
315
-
316
- // ----------------------------- update with relations
317
-
318
- // ----------------------------- update entity with relations
319
- async updateEntityWithRelations(
320
- entityType: string,
321
- id: number | string,
322
- data: Record<string, any>,
323
- loggedInUser: any,
324
- ): Promise<any> {
325
- const organizationId = loggedInUser.organization_id;
326
- const { mappedEntities, ...mainData } = data;
327
-
328
- // Update main entity
329
- const updatedMainEntity = await this.updateEntity(
330
- entityType,
331
- id,
332
- mainData,
333
- loggedInUser,
334
- );
335
-
336
- const updatedRelations: Record<string, any> = {};
337
-
338
- if (mappedEntities) {
339
- const entityRelationRepo =
340
- this.reflectionHelper.getRepoService('EntityRelation');
341
- const getRelationDefs = await entityRelationRepo.find({
342
- where: {
343
- organization_id: organizationId,
344
- source_entity_type: entityType,
345
- },
346
- });
347
-
348
- for (const [targetEntityType, rawEntityData] of Object.entries(
349
- mappedEntities,
350
- )) {
351
- const relationDef = getRelationDefs.find(
352
- (r) => r.target_entity_type === targetEntityType,
353
- );
354
- if (!relationDef) continue;
355
-
356
- const relationType = relationDef.relation_type;
357
- const entityArray = Array.isArray(rawEntityData)
358
- ? rawEntityData
359
- : [rawEntityData];
360
-
361
- // Delete previous relations and related entities
362
- // Create/update new entities & relations
363
- const updatedEntities: any[] = [];
364
-
365
- for (const item of entityArray) {
366
- let targetEntityId;
367
- let entityData;
368
-
369
- if (item.id) {
370
- // Update existing entity
371
- await this.updateEntity(
372
- targetEntityType,
373
- item.id,
374
- item,
375
- loggedInUser,
376
- Number(id), // pass main entity id as parent_id
377
- );
378
- targetEntityId = item.id;
379
- entityData = await this.getEntity(
380
- targetEntityType,
381
- targetEntityId,
382
- loggedInUser,
383
- );
384
- } else {
385
- // Create new entity
386
- const createdEntity = await this.createEntity(
387
- targetEntityType,
388
- item,
389
- loggedInUser,
390
- Number(id), // pass main entity id as parent_id
391
- );
392
- targetEntityId = createdEntity.insertId || createdEntity.id;
393
- entityData = await this.getEntity(
394
- targetEntityType,
395
- targetEntityId,
396
- loggedInUser,
397
- );
398
-
399
- // Insert relation as per new entity created
400
- const entityRelationDataRepo =
401
- this.reflectionHelper.getRepoService('EntityRelationData');
402
- await entityRelationDataRepo.save({
403
- organizationId: organizationId,
404
- source_entity_id: id,
405
- source_entity_type: entityType,
406
- target_entity_id: targetEntityId,
407
- target_entity_type: targetEntityType,
408
- relation_type: relationType,
409
- });
410
- }
411
-
412
- if (relationType === 'ONE_TO_MANY') {
413
- updatedEntities.push(entityData);
414
- } else if (
415
- relationType === 'ONE_TO_ONE' ||
416
- relationType === 'MANY_TO_ONE'
417
- ) {
418
- updatedEntities[0] = entityData; // single object
419
- }
420
- }
421
-
422
- // Assign to response mappedEntities
423
- updatedRelations[targetEntityType] =
424
- relationType === 'ONE_TO_MANY' ? updatedEntities : updatedEntities[0];
425
- }
426
- }
427
-
428
- return {
429
- mainEntity: {
430
- id,
431
- entityType,
432
- data: updatedMainEntity,
433
- },
434
- relatedEntities: updatedRelations,
435
- };
436
- }
437
-
438
- // -----------------------------
439
- async updateEntity(
440
- entityType: string,
441
- id: number | string,
442
- entityData: Record<string, any>,
443
- loggedInUser: any,
444
- mainID?: number,
445
- ): Promise<any> {
446
- const organizationId = loggedInUser.organization_id;
447
-
448
- const tableName = await this.getTableName(entityType, organizationId);
449
- const validAttributes = await this.getAttributeCodes(
450
- entityType,
451
- organizationId,
452
- );
453
-
454
- const updates: string[] = [];
455
- const values: any[] = [];
456
- let idx = 1;
457
-
458
- // Auto fields
459
- entityData.modified_date = new Date();
460
- if (loggedInUser) {
461
- entityData.modified_by = loggedInUser.id;
462
-
463
- if (!entityData.organization_id && entityData.entity_type !== 'ORG')
464
- entityData.organization_id = loggedInUser.organization_id;
465
-
466
- if (!entityData.enterprise_id)
467
- entityData.enterprise_id = loggedInUser.enterprise_id;
468
- }
469
-
470
- if (mainID) {
471
- entityData.parent_id = mainID;
472
- }
473
-
474
- // Add bypass columns if needed
475
- const bypassColumns = [
476
- 'created_date',
477
- 'created_by',
478
- 'modified_date',
479
- 'modified_by',
480
- 'organization_id',
481
- 'enterprise_id',
482
- 'level_type',
483
- 'level_id',
484
- 'status',
485
- 'entity_type',
486
- 'code',
487
- 'parent_id',
488
- ];
489
-
490
- for (const col of bypassColumns) {
491
- if (!validAttributes.some((attr) => attr.attribute_key === col)) {
492
- validAttributes.push({
493
- attribute_key: col,
494
- db_datatype: 'text',
495
- element_type: 'text',
496
- is_hidden: false,
497
- });
498
- }
499
- }
500
-
501
- // Build SET clause
502
- for (const key of Object.keys(entityData)) {
503
- if (validAttributes.some((attr) => attr.attribute_key === key)) {
504
- updates.push(`${key} = $${idx++}`);
505
- values.push(entityData[key]);
506
- }
507
- }
508
-
509
- if (updates.length === 0) {
510
- throw new Error('No valid attributes to update.');
511
- }
512
-
513
- // WHERE clause placeholder
514
- const idPlaceholder = `$${idx}`;
515
- values.push(id);
516
-
517
- const updateQuery = `
518
- UPDATE ${tableName}
519
- SET ${updates.join(', ')}
520
- WHERE id = ${idPlaceholder}
521
- `;
522
-
523
- return await this.entityManager.query(updateQuery, values);
524
- }
525
-
526
- async getEntityByDataSource(
527
- entityType: string,
528
- id: number | string,
529
- loggedInUser: any,
530
- ): Promise<any> {
531
- const organizationId = loggedInUser.organization_id;
532
-
533
- const dataSource = await this.getEntitySourceTableName(
534
- entityType,
535
- organizationId,
536
- );
537
- const validAttributes = await this.getAttributeCodes(
538
- entityType,
539
- organizationId,
540
- false,
541
- );
542
-
543
- const columns = validAttributes
544
- .map((attr) => `${attr.attribute_key}`)
545
- .join(', ');
546
- const selectQuery = `SELECT ${columns}
547
- FROM ${dataSource}
548
- WHERE id = $1`;
549
-
550
- const result = await this.entityManager.query(selectQuery, [id]);
551
- if (!result.length) return null;
552
-
553
- const row = result[0];
554
-
555
- // Convert boolean columns (1/0) into true/false
556
- for (const attr of validAttributes) {
557
- if (
558
- attr.db_datatype == 'boolean' &&
559
- row[attr.attribute_key] !== undefined
560
- ) {
561
- row[attr.attribute_key] = row[attr.attribute_key] == 1 ? true : false;
562
- } else if (
563
- attr.element_type == 'upload' &&
564
- row[attr.attribute_key] !== undefined
565
- ) {
566
- row[attr.attribute_key] =
567
- row[attr.attribute_key] != null
568
- ? await this.mediaDataService.getMediaDownloadUrl(
569
- Number(row[attr.attribute_key]),
570
- loggedInUser,
571
- )
572
- : null;
573
- }
574
- }
575
-
576
- return row;
577
- }
578
-
579
- // -----------------------------
580
- //TODO : make it normal getEntity function make another function if for resolve data
581
- async getEntity(
582
- entityType: string,
583
- id: number | string,
584
- loggedInUser: any,
585
- ): Promise<any> {
586
- const organizationId = loggedInUser.organization_id;
587
-
588
- const entityMaster = await this.entityMasterRepo.getEntityByMappedEntityType(entityType, organizationId);
589
- if(!entityMaster) return null;
590
-
591
- const validAttributes = await this.getAttributeCodes(
592
- entityType,
593
- organizationId,
594
- );
595
-
596
- // const entityRepo = this.reflectionHelper.getRepoService(entityMaster?.entity_data_class);
597
-
598
- const columns = validAttributes.map(attr => `t.${attr.attribute_key}`);
599
-
600
- // const result = await entityRepo.find({
601
- // where: {
602
- // id: id
603
- // },
604
- // select: columns
605
- // });
606
- const selectQuery = `SELECT ${columns}
607
- FROM ${this.schema}.${entityMaster.db_table_name} t
608
- WHERE id = $1`;
609
-
610
- const result = await this.entityManager.query(selectQuery, [id]);
611
- if (!result.length) return null;
612
-
613
- const row = result[0];
614
-
615
- // Convert boolean columns (1/0) into true/false
616
- for (const attr of validAttributes) {
617
- if (
618
- attr.db_datatype == 'boolean' &&
619
- row[attr.attribute_key] !== undefined
620
- ) {
621
- row[attr.attribute_key] = row[attr.attribute_key] == 1 ? true : false;
622
- } else if (
623
- attr.element_type == 'upload' &&
624
- row[attr.attribute_key] !== undefined
625
- ) {
626
- row[attr.attribute_key] =
627
- row[attr.attribute_key] != null
628
- ? await this.mediaDataService.getMediaDownloadUrl(
629
- Number(row[attr.attribute_key]),
630
- loggedInUser,
631
- )
632
- : null;
633
- }
634
- }
635
-
636
- return row;
637
- }
638
-
639
- private async getEntitySourceTableName(
640
- entityType: string,
641
- organizationId: string,
642
- ): Promise<string> {
643
- const result = await this.entityMasterRepo.getEntityByMappedEntityType(
644
- entityType,
645
- organizationId,
646
- );
647
-
648
- if (!result) {
649
- console.log(`Entity type '${entityType}' not found in frm_entity_master`);
650
- throw new BadRequestException();
651
- }
652
-
653
- return result.data_source;
654
- }
655
-
656
- // -----------------------------
657
- private async getTableName(
658
- entityType: string,
659
- organizationId: string,
660
- ): Promise<string> {
661
- let entityMaster = await this.entityMasterRepo.getEntityByMappedEntityType(
662
- entityType,
663
- organizationId,
664
- );
665
-
666
- if (!entityMaster) {
667
- console.log(`Entity type '${entityType}' not found in frm_entity_master`);
668
- throw new BadRequestException();
669
- }
670
-
671
- return entityMaster.db_table_name;
672
- }
673
-
674
- private async getAttributeCodes(
675
- entityType: string,
676
- organizationId: string,
677
- isHidden = true,
678
- ) {
679
- const attributeMasterRepo =
680
- this.reflectionHelper.getRepoService('AttributeMaster');
681
-
682
- const qb = attributeMasterRepo
683
- .createQueryBuilder('fea')
684
- .select('fea.attribute_key', 'attribute_key')
685
- .addSelect('MAX(fea.db_datatype)', 'db_datatype')
686
- .addSelect('MAX(fea.element_type)', 'element_type')
687
- .addSelect('bool_or(fea.is_hidden)', 'is_hidden')
688
- .where('fea.mapped_entity_type = :entityType', { entityType })
689
- .andWhere('fea.organization_id = :organizationId', { organizationId });
690
-
691
- if (isHidden) {
692
- qb.andWhere('(fea.is_hidden IS NULL OR fea.is_hidden = false)');
693
- }
694
-
695
- qb.groupBy('fea.attribute_key');
696
-
697
- const result = await qb.getRawMany();
698
-
699
- return result.map((row: any) => ({
700
- attribute_key: row.attribute_key,
701
- db_datatype: row.db_datatype,
702
- element_type: row.element_type,
703
- is_hidden: row.is_hidden != null ? row.is_hidden === true : undefined,
704
- }));
705
- }
706
-
707
- private async deleteEntity(
708
- entityType: string,
709
- id: number | string,
710
- loggedInUser: any,
711
- ): Promise<any> {
712
- const organizationId = loggedInUser.organization_id;
713
-
714
- const tableName = await this.getTableName(entityType, organizationId);
715
-
716
- const deleteQuery = `DELETE
717
- FROM \`${tableName}\`
718
- WHERE id = $1`;
719
- const result = await this.entityManager.query(deleteQuery, [id]);
720
- return result;
721
- }
722
-
723
- // -----------------------------
724
-
725
- async getEntitiesDropdownList(
726
- loggedInUser: any,
727
- appcode?: string,
728
- ): Promise<any> {
729
- const organizationId = loggedInUser.organization_id;
730
-
731
- let query = `SELECT name as label, mapped_entity_type as value
732
- FROM frm_entity_master
733
- WHERE organization_id = $1`;
734
- const params = [organizationId];
735
-
736
- if (appcode) {
737
- query += ` AND appcode = $2`;
738
- params.push(appcode);
739
- }
740
-
741
- const dropdown = await this.entityManager.query(query, params);
742
- return dropdown;
743
- }
744
-
745
- async getCode(entityType: string, loggedInUser: any): Promise<string> {
746
- const organizationId = loggedInUser.organization_id;
747
-
748
- // 1. Get db_table_name from entity master
749
-
750
- const result = await this.entityMasterRepo.getEntityByMappedEntityType(
751
- entityType,
752
- organizationId,
753
- );
754
-
755
- if (!result) {
756
- throw new Error(
757
- `Entity type '${entityType}' not found in frm_entity_master for org '${organizationId}'`,
758
- );
759
- }
760
-
761
- const tableName = result.db_table_name;
762
-
763
- // 2. Get current max sequence number from that table
764
- const seqResult = await this.entityManager.query(
765
- `SELECT MAX(id) AS max_seq_no
766
- FROM ${tableName}
767
- WHERE entity_type = $1`,
768
- [entityType],
769
- );
770
-
771
- let maxSeqNo = seqResult?.[0]?.max_seq_no
772
- ? Number(seqResult[0].max_seq_no)
773
- : 0;
774
-
775
- maxSeqNo += 1;
776
-
777
- // 3. Return generated code
778
- return `${entityType}${maxSeqNo}`;
779
- }
780
-
781
- async getResolvedEntity(
782
- id: number,
783
- entity: string,
784
- loggedInUser: UserData,
785
- ): Promise<any> {
786
- const leadData = await this.getEntityWithRelation(entity, id, loggedInUser);
787
-
788
- const { mappedEntities, ...data } = leadData as any;
789
-
790
- const resolvedData = await this.resolverService.getResolvedData(
791
- loggedInUser,
792
- data,
793
- entity,
794
- );
795
-
796
- if (mappedEntities) {
797
- resolvedData.mappedEntities = {};
798
- for (const [entityType, entities] of Object.entries(mappedEntities)) {
799
- if (Array.isArray(entities)) {
800
- resolvedData.mappedEntities[entityType] = [];
801
- for (const item of entities) {
802
- const resolvedItem = await this.resolverService.getResolvedData(
803
- loggedInUser,
804
- item,
805
- entityType,
806
- );
807
- resolvedData.mappedEntities[entityType].push(resolvedItem);
808
- }
809
- } else {
810
- resolvedData.mappedEntities[entityType] =
811
- await this.resolverService.getResolvedData(
812
- loggedInUser,
813
- entities,
814
- entityType,
815
- );
816
- }
817
- }
818
- }
819
-
820
- return resolvedData;
821
- }
822
- }
1
+ import { BadRequestException, Injectable } from '@nestjs/common';
2
+ import { STATUS_ACTIVE } from 'src/constant/global.constant';
3
+ import { EntityManager, In } from 'typeorm';
4
+ import { UserData } from 'src/module/user/entity/user.entity';
5
+ import { MediaDataService } from './media-data.service';
6
+ import { ResolverService } from './resolver.service';
7
+ import { EntityMasterRepository } from '../repository/entity-master.repository';
8
+ import { ReflectionHelper } from '../../../utils/service/reflection-helper.service';
9
+ import { ConfigService } from '@nestjs/config';
10
+
11
+ @Injectable()
12
+ export class EntityDynamicService {
13
+ constructor(
14
+ private readonly entityManager: EntityManager,
15
+ private readonly mediaDataService: MediaDataService,
16
+ private readonly resolverService: ResolverService,
17
+ private readonly entityMasterRepo: EntityMasterRepository,
18
+ private readonly reflectionHelper: ReflectionHelper,
19
+ private readonly configService: ConfigService,
20
+ ) {
21
+ }
22
+ schema = this.configService.get('DB_SCHEMA');
23
+
24
+ // -----------------------------
25
+ async createEntity(
26
+ entityType: string,
27
+ entityData: Record<string, any>,
28
+ loggedInUser: any,
29
+ mainID?: number,
30
+ ): Promise<any> {
31
+ const organizationId = loggedInUser.organization_id;
32
+
33
+ const tableName = await this.getTableName(entityType, organizationId);
34
+ const validAttributes = await this.getAttributeCodes(
35
+ entityType,
36
+ organizationId,
37
+ );
38
+
39
+ // -------------------------------------------------------
40
+ // AUTO fields
41
+ // -------------------------------------------------------
42
+ entityData.created_date = new Date();
43
+
44
+ if (loggedInUser) {
45
+ entityData.created_by = loggedInUser.id;
46
+
47
+ if (!entityData.organization_id)
48
+ entityData.organization_id = loggedInUser.organization_id;
49
+
50
+ if (!entityData.enterprise_id)
51
+ entityData.enterprise_id = loggedInUser.enterprise_id;
52
+
53
+ if (!entityData.level_type)
54
+ entityData.level_type = loggedInUser.level_type;
55
+
56
+ if (!entityData.level_id) entityData.level_id = loggedInUser.level_id;
57
+
58
+ if (!entityData.entity_type) entityData.entity_type = entityType;
59
+ }
60
+
61
+ // -------------------------------------------------------
62
+ // STATUS
63
+ // -------------------------------------------------------
64
+ const listMasterItemsRepo =
65
+ this.reflectionHelper.getRepoService('ListMasterItems');
66
+ const statusList = listMasterItemsRepo.find({
67
+ where: {
68
+ code: STATUS_ACTIVE,
69
+ organization_id: organizationId,
70
+ },
71
+ });
72
+
73
+ if (!entityData.status) entityData.status = statusList[0]?.id;
74
+
75
+ // -------------------------------------------------------
76
+ // AUTO-CODE GENERATION (POSTGRES SAFE)
77
+ // -------------------------------------------------------
78
+ const entityMaster = await this.entityMasterRepo.getEntityByMappedEntityType(entityType, loggedInUser.organization_id)
79
+
80
+ if (!entityData.code && entityData.entity_type && entityMaster) {
81
+ // Extract integer suffix
82
+ // const repo = this.reflectionHelper.getRepoService(entityMaster.entity_data_class);
83
+
84
+ const result = this.entityManager.query(
85
+ `SELECT MAX(id) as seq_no FROM ${this.schema}.${entityMaster.db_table_name}`
86
+ )
87
+
88
+ let maxSeq = Number(result[0]?.max_seq_no) || 0;
89
+ maxSeq++;
90
+ entityData.code = `${entityData.entity_type}${maxSeq}`;
91
+ }
92
+
93
+ // -------------------------------------------------------
94
+ // Parent ID
95
+ // -------------------------------------------------------
96
+ if (mainID) {
97
+ entityData.parent_id = mainID;
98
+ }
99
+
100
+ // -------------------------------------------------------
101
+ // BYPASS COLUMNS
102
+ // -------------------------------------------------------
103
+ const bypassColumns = [
104
+ 'created_date',
105
+ 'created_by',
106
+ 'organization_id',
107
+ 'enterprise_id',
108
+ 'level_type',
109
+ 'level_id',
110
+ 'status',
111
+ 'entity_type',
112
+ 'code',
113
+ 'parent_id',
114
+ ];
115
+
116
+ for (const col of bypassColumns) {
117
+ if (!validAttributes.some((a) => a.attribute_key === col)) {
118
+ validAttributes.push({
119
+ attribute_key: col,
120
+ is_hidden: false,
121
+ db_datatype: 'text',
122
+ element_type: 'text',
123
+ });
124
+ }
125
+ }
126
+
127
+ // -------------------------------------------------------
128
+ // BUILD INSERT QUERY (POSTGRES FORMAT)
129
+ // -------------------------------------------------------
130
+ const columns: string[] = [];
131
+ const values: any[] = [];
132
+ let idx = 1;
133
+ const placeholders: string[] = [];
134
+
135
+ for (const attr of validAttributes) {
136
+ if (attr.attribute_key === 'id') continue;
137
+
138
+ columns.push(attr.attribute_key);
139
+ values.push(entityData[attr.attribute_key] ?? null);
140
+ placeholders.push(`$${idx++}`);
141
+ }
142
+
143
+ const colList = columns.map((c) => `"${c}"`).join(', ');
144
+ const placeholderList = placeholders.join(', ');
145
+
146
+
147
+ const sql = `
148
+ INSERT INTO ${this.schema}.${tableName} (${colList})
149
+ VALUES (${placeholderList}) RETURNING id
150
+ `;
151
+
152
+ const result = await this.entityManager.query(sql, values);
153
+ return result[0];
154
+ }
155
+
156
+ // ----------------------------- get entity with relations
157
+
158
+ // ----------------------------- create entity with relations
159
+ async createEntityWithRelation(
160
+ entityType: string,
161
+ data: Record<string, any>,
162
+ loggedInUser: any,
163
+ ): Promise<any> {
164
+ const organizationId = loggedInUser.organization_id;
165
+
166
+ const repo = this.reflectionHelper.getRepoService('EntityRelation');
167
+
168
+ const getRelation = await repo.find({
169
+ where: {
170
+ organization_id: organizationId,
171
+ source_entity_type: entityType,
172
+ },
173
+ });
174
+
175
+ const { mappedEntities, ...mainData } = data;
176
+
177
+ // create main entity
178
+ const createdEntityData = await this.createEntity(
179
+ entityType,
180
+ mainData,
181
+ loggedInUser,
182
+ );
183
+ const mainID = createdEntityData.insertId || createdEntityData.id;
184
+
185
+ if (mappedEntities && getRelation.length > 0) {
186
+ for (const relation of getRelation) {
187
+ const targetEntityType = relation.target_entity_type;
188
+ const relationType = relation.relation_type;
189
+ const relationId = relation.relation_id;
190
+
191
+ if (!mappedEntities[targetEntityType]) continue;
192
+
193
+ // normalize: always array
194
+ const entityDataArray = Array.isArray(mappedEntities[targetEntityType])
195
+ ? mappedEntities[targetEntityType]
196
+ : [mappedEntities[targetEntityType]];
197
+
198
+ for (const item of entityDataArray) {
199
+ const itemWithRef = {
200
+ ...item,
201
+ entity_type: targetEntityType,
202
+ };
203
+
204
+ const createdRelatedEntity = await this.createEntity(
205
+ targetEntityType,
206
+ itemWithRef,
207
+ loggedInUser,
208
+ mainID, // this will pass for parent_id
209
+ );
210
+
211
+ const relationRepo =
212
+ this.reflectionHelper.getRepoService('EntityRelationData');
213
+
214
+ await relationRepo.save({
215
+ organizationId: organizationId,
216
+ source_entity_id: mainID,
217
+ source_entity_type: entityType,
218
+ target_entity_id: createdRelatedEntity.id,
219
+ target_entity_type: targetEntityType,
220
+ relation_type: relationType,
221
+ });
222
+ }
223
+ }
224
+ }
225
+
226
+ return {
227
+ mainEntity: {
228
+ id: mainID,
229
+ entityType,
230
+ data: createdEntityData,
231
+ },
232
+ };
233
+ }
234
+
235
+ // ----------------------------- get entity with relations
236
+ async getEntityWithRelation(
237
+ entityType: string,
238
+ id: number | string,
239
+ loggedInUser: any,
240
+ ): Promise<any> {
241
+ const mainEntity = await this.getEntityByDataSource(
242
+ entityType,
243
+ id,
244
+ loggedInUser,
245
+ );
246
+ const relationRepo = this.reflectionHelper.getRepoService('EntityRelation');
247
+ const relatedEntityRepo =
248
+ this.reflectionHelper.getRepoService('EntityRelationData');
249
+
250
+ const relations = await relationRepo.find({
251
+ where: {
252
+ source_entity_type: entityType,
253
+ organization_id: loggedInUser.organization_id,
254
+ },
255
+ });
256
+
257
+ const targetTypes = relations.map((r) => r.target_entity_type);
258
+
259
+ if (targetTypes.length === 0) {
260
+ return {
261
+ entity_type: entityType,
262
+ ...mainEntity,
263
+ };
264
+ }
265
+
266
+ const relatedEntities = await relatedEntityRepo.find({
267
+ where: {
268
+ source_entity_id: id,
269
+ target_entity_type: In(targetTypes),
270
+ },
271
+ });
272
+
273
+ // Format response to match create entity structure
274
+ const response: any = {
275
+ entity_type: entityType,
276
+ ...mainEntity,
277
+ };
278
+
279
+ if (relatedEntities.length > 0) {
280
+ response.mappedEntities = await this.formatMappedEntities(
281
+ relatedEntities,
282
+ loggedInUser,
283
+ );
284
+ }
285
+
286
+ return response;
287
+ }
288
+
289
+ // ----------------------------- formatMappedEntities
290
+ async formatMappedEntities(
291
+ relatedEntities: any[],
292
+ loggedInUser: any,
293
+ ): Promise<any> {
294
+ const mappedEntities: any = {};
295
+
296
+ for (const relation of relatedEntities) {
297
+ const targetEntityType = relation.target_entity_type;
298
+ const targetEntityId = relation.target_entity_id;
299
+
300
+ const entityData = await this.getEntity(
301
+ targetEntityType,
302
+ targetEntityId,
303
+ loggedInUser,
304
+ );
305
+
306
+ if (!mappedEntities[targetEntityType]) {
307
+ mappedEntities[targetEntityType] = [];
308
+ }
309
+
310
+ mappedEntities[targetEntityType].push(entityData);
311
+ }
312
+
313
+ return mappedEntities;
314
+ }
315
+
316
+ // ----------------------------- update with relations
317
+
318
+ // ----------------------------- update entity with relations
319
+ async updateEntityWithRelations(
320
+ entityType: string,
321
+ id: number | string,
322
+ data: Record<string, any>,
323
+ loggedInUser: any,
324
+ ): Promise<any> {
325
+ const organizationId = loggedInUser.organization_id;
326
+ const { mappedEntities, ...mainData } = data;
327
+
328
+ // Update main entity
329
+ const updatedMainEntity = await this.updateEntity(
330
+ entityType,
331
+ id,
332
+ mainData,
333
+ loggedInUser,
334
+ );
335
+
336
+ const updatedRelations: Record<string, any> = {};
337
+
338
+ if (mappedEntities) {
339
+ const entityRelationRepo =
340
+ this.reflectionHelper.getRepoService('EntityRelation');
341
+ const getRelationDefs = await entityRelationRepo.find({
342
+ where: {
343
+ organization_id: organizationId,
344
+ source_entity_type: entityType,
345
+ },
346
+ });
347
+
348
+ for (const [targetEntityType, rawEntityData] of Object.entries(
349
+ mappedEntities,
350
+ )) {
351
+ const relationDef = getRelationDefs.find(
352
+ (r) => r.target_entity_type === targetEntityType,
353
+ );
354
+ if (!relationDef) continue;
355
+
356
+ const relationType = relationDef.relation_type;
357
+ const entityArray = Array.isArray(rawEntityData)
358
+ ? rawEntityData
359
+ : [rawEntityData];
360
+
361
+ // Delete previous relations and related entities
362
+ // Create/update new entities & relations
363
+ const updatedEntities: any[] = [];
364
+
365
+ for (const item of entityArray) {
366
+ let targetEntityId;
367
+ let entityData;
368
+
369
+ if (item.id) {
370
+ // Update existing entity
371
+ await this.updateEntity(
372
+ targetEntityType,
373
+ item.id,
374
+ item,
375
+ loggedInUser,
376
+ Number(id), // pass main entity id as parent_id
377
+ );
378
+ targetEntityId = item.id;
379
+ entityData = await this.getEntity(
380
+ targetEntityType,
381
+ targetEntityId,
382
+ loggedInUser,
383
+ );
384
+ } else {
385
+ // Create new entity
386
+ const createdEntity = await this.createEntity(
387
+ targetEntityType,
388
+ item,
389
+ loggedInUser,
390
+ Number(id), // pass main entity id as parent_id
391
+ );
392
+ targetEntityId = createdEntity.insertId || createdEntity.id;
393
+ entityData = await this.getEntity(
394
+ targetEntityType,
395
+ targetEntityId,
396
+ loggedInUser,
397
+ );
398
+
399
+ // Insert relation as per new entity created
400
+ const entityRelationDataRepo =
401
+ this.reflectionHelper.getRepoService('EntityRelationData');
402
+ await entityRelationDataRepo.save({
403
+ organizationId: organizationId,
404
+ source_entity_id: id,
405
+ source_entity_type: entityType,
406
+ target_entity_id: targetEntityId,
407
+ target_entity_type: targetEntityType,
408
+ relation_type: relationType,
409
+ });
410
+ }
411
+
412
+ if (relationType === 'ONE_TO_MANY') {
413
+ updatedEntities.push(entityData);
414
+ } else if (
415
+ relationType === 'ONE_TO_ONE' ||
416
+ relationType === 'MANY_TO_ONE'
417
+ ) {
418
+ updatedEntities[0] = entityData; // single object
419
+ }
420
+ }
421
+
422
+ // Assign to response mappedEntities
423
+ updatedRelations[targetEntityType] =
424
+ relationType === 'ONE_TO_MANY' ? updatedEntities : updatedEntities[0];
425
+ }
426
+ }
427
+
428
+ return {
429
+ mainEntity: {
430
+ id,
431
+ entityType,
432
+ data: updatedMainEntity,
433
+ },
434
+ relatedEntities: updatedRelations,
435
+ };
436
+ }
437
+
438
+ // -----------------------------
439
+ async updateEntity(
440
+ entityType: string,
441
+ id: number | string,
442
+ entityData: Record<string, any>,
443
+ loggedInUser: any,
444
+ mainID?: number,
445
+ ): Promise<any> {
446
+ const organizationId = loggedInUser.organization_id;
447
+
448
+ const tableName = await this.getTableName(entityType, organizationId);
449
+ const validAttributes = await this.getAttributeCodes(
450
+ entityType,
451
+ organizationId,
452
+ );
453
+
454
+ const updates: string[] = [];
455
+ const values: any[] = [];
456
+ let idx = 1;
457
+
458
+ // Auto fields
459
+ entityData.modified_date = new Date();
460
+ if (loggedInUser) {
461
+ entityData.modified_by = loggedInUser.id;
462
+
463
+ if (!entityData.organization_id && entityData.entity_type !== 'ORG')
464
+ entityData.organization_id = loggedInUser.organization_id;
465
+
466
+ if (!entityData.enterprise_id)
467
+ entityData.enterprise_id = loggedInUser.enterprise_id;
468
+ }
469
+
470
+ if (mainID) {
471
+ entityData.parent_id = mainID;
472
+ }
473
+
474
+ // Add bypass columns if needed
475
+ const bypassColumns = [
476
+ 'created_date',
477
+ 'created_by',
478
+ 'modified_date',
479
+ 'modified_by',
480
+ 'organization_id',
481
+ 'enterprise_id',
482
+ 'level_type',
483
+ 'level_id',
484
+ 'status',
485
+ 'entity_type',
486
+ 'code',
487
+ 'parent_id',
488
+ ];
489
+
490
+ for (const col of bypassColumns) {
491
+ if (!validAttributes.some((attr) => attr.attribute_key === col)) {
492
+ validAttributes.push({
493
+ attribute_key: col,
494
+ db_datatype: 'text',
495
+ element_type: 'text',
496
+ is_hidden: false,
497
+ });
498
+ }
499
+ }
500
+
501
+ // Build SET clause
502
+ for (const key of Object.keys(entityData)) {
503
+ if (validAttributes.some((attr) => attr.attribute_key === key)) {
504
+ updates.push(`${key} = $${idx++}`);
505
+ values.push(entityData[key]);
506
+ }
507
+ }
508
+
509
+ if (updates.length === 0) {
510
+ throw new Error('No valid attributes to update.');
511
+ }
512
+
513
+ // WHERE clause placeholder
514
+ const idPlaceholder = `$${idx}`;
515
+ values.push(id);
516
+
517
+ const updateQuery = `
518
+ UPDATE ${this.schema}.${tableName}
519
+ SET ${updates.join(', ')}
520
+ WHERE id = ${idPlaceholder}
521
+ `;
522
+
523
+ return await this.entityManager.query(updateQuery, values);
524
+ }
525
+
526
+ async getEntityByDataSource(
527
+ entityType: string,
528
+ id: number | string,
529
+ loggedInUser: any,
530
+ ): Promise<any> {
531
+ const organizationId = loggedInUser.organization_id;
532
+
533
+ const dataSource = await this.getEntitySourceTableName(
534
+ entityType,
535
+ organizationId,
536
+ );
537
+ const validAttributes = await this.getAttributeCodes(
538
+ entityType,
539
+ organizationId,
540
+ false,
541
+ );
542
+
543
+ const columns = validAttributes
544
+ .map((attr) => `${attr.attribute_key}`)
545
+ .join(', ');
546
+ const selectQuery = `SELECT ${columns}
547
+ FROM ${dataSource}
548
+ WHERE id = $1`;
549
+
550
+ const result = await this.entityManager.query(selectQuery, [id]);
551
+ if (!result.length) return null;
552
+
553
+ const row = result[0];
554
+
555
+ // Convert boolean columns (1/0) into true/false
556
+ for (const attr of validAttributes) {
557
+ if (
558
+ attr.db_datatype == 'boolean' &&
559
+ row[attr.attribute_key] !== undefined
560
+ ) {
561
+ row[attr.attribute_key] = row[attr.attribute_key] == 1 ? true : false;
562
+ } else if (
563
+ attr.element_type == 'upload' &&
564
+ row[attr.attribute_key] !== undefined
565
+ ) {
566
+ row[attr.attribute_key] =
567
+ row[attr.attribute_key] != null
568
+ ? await this.mediaDataService.getMediaDownloadUrl(
569
+ Number(row[attr.attribute_key]),
570
+ loggedInUser,
571
+ )
572
+ : null;
573
+ }
574
+ }
575
+
576
+ return row;
577
+ }
578
+
579
+ // -----------------------------
580
+ //TODO : make it normal getEntity function make another function if for resolve data
581
+ async getEntity(
582
+ entityType: string,
583
+ id: number | string,
584
+ loggedInUser: any,
585
+ ): Promise<any> {
586
+ const organizationId = loggedInUser.organization_id;
587
+
588
+ const entityMaster = await this.entityMasterRepo.getEntityByMappedEntityType(entityType, organizationId);
589
+ if(!entityMaster) return null;
590
+
591
+ const validAttributes = await this.getAttributeCodes(
592
+ entityType,
593
+ organizationId,
594
+ );
595
+
596
+ // const entityRepo = this.reflectionHelper.getRepoService(entityMaster?.entity_data_class);
597
+
598
+ const columns = validAttributes.map(attr => `t.${attr.attribute_key}`);
599
+
600
+ // const result = await entityRepo.find({
601
+ // where: {
602
+ // id: id
603
+ // },
604
+ // select: columns
605
+ // });
606
+ const selectQuery = `SELECT ${columns}
607
+ FROM ${this.schema}.${entityMaster.db_table_name} t
608
+ WHERE id = $1`;
609
+
610
+ const result = await this.entityManager.query(selectQuery, [id]);
611
+ if (!result.length) return null;
612
+
613
+ const row = result[0];
614
+
615
+ // Convert boolean columns (1/0) into true/false
616
+ for (const attr of validAttributes) {
617
+ if (
618
+ attr.db_datatype == 'boolean' &&
619
+ row[attr.attribute_key] !== undefined
620
+ ) {
621
+ row[attr.attribute_key] = row[attr.attribute_key] == 1 ? true : false;
622
+ } else if (
623
+ attr.element_type == 'upload' &&
624
+ row[attr.attribute_key] !== undefined
625
+ ) {
626
+ row[attr.attribute_key] =
627
+ row[attr.attribute_key] != null
628
+ ? await this.mediaDataService.getMediaDownloadUrl(
629
+ Number(row[attr.attribute_key]),
630
+ loggedInUser,
631
+ )
632
+ : null;
633
+ }
634
+ }
635
+
636
+ return row;
637
+ }
638
+
639
+ private async getEntitySourceTableName(
640
+ entityType: string,
641
+ organizationId: string,
642
+ ): Promise<string> {
643
+ const result = await this.entityMasterRepo.getEntityByMappedEntityType(
644
+ entityType,
645
+ organizationId,
646
+ );
647
+
648
+ if (!result) {
649
+ console.log(`Entity type '${entityType}' not found in frm_entity_master`);
650
+ throw new BadRequestException();
651
+ }
652
+
653
+ return result.data_source;
654
+ }
655
+
656
+ // -----------------------------
657
+ private async getTableName(
658
+ entityType: string,
659
+ organizationId: string,
660
+ ): Promise<string> {
661
+ let entityMaster = await this.entityMasterRepo.getEntityByMappedEntityType(
662
+ entityType,
663
+ organizationId,
664
+ );
665
+
666
+ if (!entityMaster) {
667
+ console.log(`Entity type '${entityType}' not found in frm_entity_master`);
668
+ throw new BadRequestException();
669
+ }
670
+
671
+ return entityMaster.db_table_name;
672
+ }
673
+
674
+ private async getAttributeCodes(
675
+ entityType: string,
676
+ organizationId: string,
677
+ isHidden = true,
678
+ ) {
679
+ const attributeMasterRepo =
680
+ this.reflectionHelper.getRepoService('AttributeMaster');
681
+
682
+ const qb = attributeMasterRepo
683
+ .createQueryBuilder('fea')
684
+ .select('fea.attribute_key', 'attribute_key')
685
+ .addSelect('MAX(fea.db_datatype)', 'db_datatype')
686
+ .addSelect('MAX(fea.element_type)', 'element_type')
687
+ .addSelect('bool_or(fea.is_hidden)', 'is_hidden')
688
+ .where('fea.mapped_entity_type = :entityType', { entityType })
689
+ .andWhere('fea.organization_id = :organizationId', { organizationId });
690
+
691
+ if (isHidden) {
692
+ qb.andWhere('(fea.is_hidden IS NULL OR fea.is_hidden = false)');
693
+ }
694
+
695
+ qb.groupBy('fea.attribute_key');
696
+
697
+ const result = await qb.getRawMany();
698
+
699
+ return result.map((row: any) => ({
700
+ attribute_key: row.attribute_key,
701
+ db_datatype: row.db_datatype,
702
+ element_type: row.element_type,
703
+ is_hidden: row.is_hidden != null ? row.is_hidden === true : undefined,
704
+ }));
705
+ }
706
+
707
+ private async deleteEntity(
708
+ entityType: string,
709
+ id: number | string,
710
+ loggedInUser: any,
711
+ ): Promise<any> {
712
+ const organizationId = loggedInUser.organization_id;
713
+
714
+ const tableName = await this.getTableName(entityType, organizationId);
715
+
716
+ const deleteQuery = `DELETE
717
+ FROM \`${tableName}\`
718
+ WHERE id = $1`;
719
+ const result = await this.entityManager.query(deleteQuery, [id]);
720
+ return result;
721
+ }
722
+
723
+ // -----------------------------
724
+
725
+ async getEntitiesDropdownList(
726
+ loggedInUser: any,
727
+ appcode?: string,
728
+ ): Promise<any> {
729
+ const organizationId = loggedInUser.organization_id;
730
+
731
+ let query = `SELECT name as label, mapped_entity_type as value
732
+ FROM frm_entity_master
733
+ WHERE organization_id = $1`;
734
+ const params = [organizationId];
735
+
736
+ if (appcode) {
737
+ query += ` AND appcode = $2`;
738
+ params.push(appcode);
739
+ }
740
+
741
+ const dropdown = await this.entityManager.query(query, params);
742
+ return dropdown;
743
+ }
744
+
745
+ async getCode(entityType: string, loggedInUser: any): Promise<string> {
746
+ const organizationId = loggedInUser.organization_id;
747
+
748
+ // 1. Get db_table_name from entity master
749
+
750
+ const result = await this.entityMasterRepo.getEntityByMappedEntityType(
751
+ entityType,
752
+ organizationId,
753
+ );
754
+
755
+ if (!result) {
756
+ throw new Error(
757
+ `Entity type '${entityType}' not found in frm_entity_master for org '${organizationId}'`,
758
+ );
759
+ }
760
+
761
+ const tableName = result.db_table_name;
762
+
763
+ // 2. Get current max sequence number from that table
764
+ const seqResult = await this.entityManager.query(
765
+ `SELECT MAX(id) AS max_seq_no
766
+ FROM ${tableName}
767
+ WHERE entity_type = $1`,
768
+ [entityType],
769
+ );
770
+
771
+ let maxSeqNo = seqResult?.[0]?.max_seq_no
772
+ ? Number(seqResult[0].max_seq_no)
773
+ : 0;
774
+
775
+ maxSeqNo += 1;
776
+
777
+ // 3. Return generated code
778
+ return `${entityType}${maxSeqNo}`;
779
+ }
780
+
781
+ async getResolvedEntity(
782
+ id: number,
783
+ entity: string,
784
+ loggedInUser: UserData,
785
+ ): Promise<any> {
786
+ const leadData = await this.getEntityWithRelation(entity, id, loggedInUser);
787
+
788
+ const { mappedEntities, ...data } = leadData as any;
789
+
790
+ const resolvedData = await this.resolverService.getResolvedData(
791
+ loggedInUser,
792
+ data,
793
+ entity,
794
+ );
795
+
796
+ if (mappedEntities) {
797
+ resolvedData.mappedEntities = {};
798
+ for (const [entityType, entities] of Object.entries(mappedEntities)) {
799
+ if (Array.isArray(entities)) {
800
+ resolvedData.mappedEntities[entityType] = [];
801
+ for (const item of entities) {
802
+ const resolvedItem = await this.resolverService.getResolvedData(
803
+ loggedInUser,
804
+ item,
805
+ entityType,
806
+ );
807
+ resolvedData.mappedEntities[entityType].push(resolvedItem);
808
+ }
809
+ } else {
810
+ resolvedData.mappedEntities[entityType] =
811
+ await this.resolverService.getResolvedData(
812
+ loggedInUser,
813
+ entities,
814
+ entityType,
815
+ );
816
+ }
817
+ }
818
+ }
819
+
820
+ return resolvedData;
821
+ }
822
+ }