rez_core 5.0.157 → 5.0.158

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