rez_core 6.1.4 → 6.1.6

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 (427) hide show
  1. package/.prettierrc +3 -3
  2. package/README.md +99 -99
  3. package/dist/migrations/1732612800000-AddEntityJsonGinIndex.js +10 -10
  4. package/dist/module/auth/guards/role.guard.js +3 -3
  5. package/dist/module/enterprise/service/organization.service.d.ts +1 -1
  6. package/dist/module/enterprise/service/organization.service.js +3 -4
  7. package/dist/module/enterprise/service/organization.service.js.map +1 -1
  8. package/dist/module/filter/repository/saved-filter.repository.js +2 -0
  9. package/dist/module/filter/repository/saved-filter.repository.js.map +1 -1
  10. package/dist/module/filter/service/filter.service.js +25 -21
  11. package/dist/module/filter/service/filter.service.js.map +1 -1
  12. package/dist/module/filter/service/flatjson-filter.service.js +5 -5
  13. package/dist/module/integration/examples/usage.example.js +9 -9
  14. package/dist/module/integration/service/integration.service.d.ts +5 -1
  15. package/dist/module/integration/service/integration.service.js +11 -1
  16. package/dist/module/integration/service/integration.service.js.map +1 -1
  17. package/dist/module/integration/service/oauth.service.js +2 -0
  18. package/dist/module/integration/service/oauth.service.js.map +1 -1
  19. package/dist/module/integration/strategies/email/gmail-api.strategy.js +23 -7
  20. package/dist/module/integration/strategies/email/gmail-api.strategy.js.map +1 -1
  21. package/dist/module/integration/strategies/email/sendgrid-api.strategy.js +8 -5
  22. package/dist/module/integration/strategies/email/sendgrid-api.strategy.js.map +1 -1
  23. package/dist/module/meta/controller/media.controller.d.ts +6 -0
  24. package/dist/module/meta/controller/media.controller.js +27 -0
  25. package/dist/module/meta/controller/media.controller.js.map +1 -1
  26. package/dist/module/meta/entity/media-data.entity.d.ts +2 -0
  27. package/dist/module/meta/entity/media-data.entity.js +8 -0
  28. package/dist/module/meta/entity/media-data.entity.js.map +1 -1
  29. package/dist/module/meta/repository/attribute-master.repository.js +8 -8
  30. package/dist/module/meta/service/entity-dynamic.service.js +18 -18
  31. package/dist/module/meta/service/entity-dynamic.service.js.map +1 -1
  32. package/dist/module/meta/service/media-data.service.d.ts +3 -0
  33. package/dist/module/meta/service/media-data.service.js +16 -6
  34. package/dist/module/meta/service/media-data.service.js.map +1 -1
  35. package/dist/module/meta/service/populate-meta.service.js +8 -2
  36. package/dist/module/meta/service/populate-meta.service.js.map +1 -1
  37. package/dist/module/meta/service/resolver.service.js +39 -15
  38. package/dist/module/meta/service/resolver.service.js.map +1 -1
  39. package/dist/module/module/repository/menu.repository.js +4 -4
  40. package/dist/module/user/controller/login.controller.js +19 -19
  41. package/dist/module/user/controller/login.controller.js.map +1 -1
  42. package/dist/module/user/service/user-session.service.js +1 -0
  43. package/dist/module/user/service/user-session.service.js.map +1 -1
  44. package/dist/module/workflow/repository/action.repository.js +2 -2
  45. package/dist/module/workflow/repository/stage.repository.js +8 -8
  46. package/dist/module/workflow/service/action-template-mapping.service.js +2 -2
  47. package/dist/module/workflow/service/action.service.js +5 -5
  48. package/dist/module/workflow/service/entity-modification.service.js +2 -2
  49. package/dist/module/workflow/service/populate-workflow.service.js +5 -0
  50. package/dist/module/workflow/service/populate-workflow.service.js.map +1 -1
  51. package/dist/module/workflow/service/task.service.js +8 -8
  52. package/dist/module/workflow-automation/service/schedule-handler.service.js +9 -9
  53. package/dist/tsconfig.build.tsbuildinfo +1 -1
  54. package/dist/utils/service/reflection-helper.service.js +2 -2
  55. package/docs/modules/event-driven-integration-design.md +91 -91
  56. package/docs/modules/integration.md +250 -250
  57. package/eslint.config.mjs +34 -34
  58. package/nest-cli.json +14 -14
  59. package/package.json +125 -125
  60. package/src/app.controller.ts +13 -13
  61. package/src/app.module.ts +68 -68
  62. package/src/app.service.ts +8 -8
  63. package/src/config/bull.config.ts +69 -69
  64. package/src/config/config.module.ts +17 -17
  65. package/src/config/database.config.ts +48 -48
  66. package/src/constant/global.constant.ts +67 -67
  67. package/src/core.module.ts +94 -94
  68. package/src/decorators/roles.decorator.ts +7 -7
  69. package/src/dtos/response.dto.ts +6 -6
  70. package/src/dtos/response.ts +5 -5
  71. package/src/index.ts +1 -1
  72. package/src/migrations/1732612800000-AddEntityJsonGinIndex.ts +41 -41
  73. package/src/module/auth/auth.module.ts +49 -49
  74. package/src/module/auth/controller/auth.controller.ts +28 -28
  75. package/src/module/auth/guards/google-auth.guard.ts +9 -9
  76. package/src/module/auth/guards/jwt.guard.ts +22 -22
  77. package/src/module/auth/guards/role.guard.ts +68 -68
  78. package/src/module/auth/services/auth.service.ts +56 -56
  79. package/src/module/auth/services/jwt.service.ts +11 -11
  80. package/src/module/auth/strategies/google.strategy.ts +54 -54
  81. package/src/module/auth/strategies/jwt.strategy.ts +58 -58
  82. package/src/module/auth/strategies/local.strategy.ts +13 -13
  83. package/src/module/dashboard/controller/dashboard.controller.ts +38 -38
  84. package/src/module/dashboard/dashboard.module.ts +21 -21
  85. package/src/module/dashboard/entity/dashboard_page_data.entity.ts +27 -27
  86. package/src/module/dashboard/entity/widget_master.entity.ts +18 -18
  87. package/src/module/dashboard/repository/dashboard.repository.ts +49 -49
  88. package/src/module/dashboard/service/dashboard.service.ts +72 -72
  89. package/src/module/enterprise/controller/organization.controller.ts +36 -36
  90. package/src/module/enterprise/enterprise.module.ts +45 -45
  91. package/src/module/enterprise/entity/enterprise.entity.ts +37 -37
  92. package/src/module/enterprise/entity/organization-app-mapping.entity.ts +13 -13
  93. package/src/module/enterprise/entity/organization.entity.ts +92 -92
  94. package/src/module/enterprise/repository/enterprise.repository.ts +31 -31
  95. package/src/module/enterprise/repository/organization.repository.ts +26 -26
  96. package/src/module/enterprise/repository/school.repository.ts +289 -289
  97. package/src/module/enterprise/service/brand.service.ts +5 -5
  98. package/src/module/enterprise/service/enterprise.service.ts +16 -16
  99. package/src/module/enterprise/service/organization-app-mapping.service.ts +4 -4
  100. package/src/module/enterprise/service/organization.service.ts +146 -145
  101. package/src/module/entity_json/controller/entity_json.controller.ts +60 -60
  102. package/src/module/entity_json/docs/FlatJson_Filterin_System.md +2803 -2803
  103. package/src/module/entity_json/entity/entityJson.entity.ts +42 -42
  104. package/src/module/entity_json/entity_json.module.ts +22 -22
  105. package/src/module/entity_json/service/entityJson.repository.ts +37 -37
  106. package/src/module/entity_json/service/entity_json.service.ts +486 -486
  107. package/src/module/export/controller/export.controller.ts +83 -83
  108. package/src/module/export/export.module.ts +14 -14
  109. package/src/module/export/service/export.service.ts +105 -105
  110. package/src/module/filter/controller/filter.controller.ts +87 -87
  111. package/src/module/filter/dto/filter-request.dto.ts +39 -39
  112. package/src/module/filter/entity/saved-filter-detail.entity.ts +41 -41
  113. package/src/module/filter/entity/saved-filter-master.entity.ts +35 -35
  114. package/src/module/filter/filter.module.ts +41 -41
  115. package/src/module/filter/repository/saved-filter.repository.ts +249 -247
  116. package/src/module/filter/repository/saved.filter-detail.repository.ts +19 -19
  117. package/src/module/filter/service/filter-evaluator.service.ts +82 -82
  118. package/src/module/filter/service/filter.service.ts +1406 -1370
  119. package/src/module/filter/service/flatjson-filter.service.ts +903 -903
  120. package/src/module/filter/service/saved-filter.service.ts +154 -154
  121. package/src/module/filter/test/flatjson-filter.service.spec.ts +415 -415
  122. package/src/module/ics/controller/ics.controller.ts +21 -21
  123. package/src/module/ics/dto/ics.dto.ts +55 -55
  124. package/src/module/ics/ics.module.ts +13 -13
  125. package/src/module/ics/service/ics.service.ts +57 -57
  126. package/src/module/integration/controller/calender-event.controller.ts +31 -31
  127. package/src/module/integration/controller/integration.controller.ts +662 -662
  128. package/src/module/integration/controller/wrapper.controller.ts +37 -37
  129. package/src/module/integration/dto/create-config.dto.ts +526 -526
  130. package/src/module/integration/entity/integration-config.entity.ts +112 -112
  131. package/src/module/integration/entity/integration-entity-mapper.entity.ts +14 -14
  132. package/src/module/integration/entity/integration-source.entity.ts +17 -17
  133. package/src/module/integration/entity/user-integration.entity.ts +71 -71
  134. package/src/module/integration/examples/usage.example.ts +338 -338
  135. package/src/module/integration/factories/base.factory.ts +7 -7
  136. package/src/module/integration/factories/email.factory.ts +49 -49
  137. package/src/module/integration/factories/integration.factory.ts +121 -121
  138. package/src/module/integration/factories/sms.factory.ts +51 -51
  139. package/src/module/integration/factories/telephone.factory.ts +41 -41
  140. package/src/module/integration/factories/whatsapp.factory.ts +56 -56
  141. package/src/module/integration/integration.module.ts +110 -110
  142. package/src/module/integration/service/calendar-event.service.ts +118 -118
  143. package/src/module/integration/service/integration-entity-mapper.service.ts +17 -17
  144. package/src/module/integration/service/integration-queue.service.ts +229 -229
  145. package/src/module/integration/service/integration.service.ts +2653 -2639
  146. package/src/module/integration/service/oauth.service.ts +226 -224
  147. package/src/module/integration/service/wrapper.service.ts +754 -754
  148. package/src/module/integration/strategies/email/gmail-api.strategy.ts +307 -281
  149. package/src/module/integration/strategies/email/outlook-api.strategy.ts +44 -44
  150. package/src/module/integration/strategies/email/outlook.strategy.ts +64 -64
  151. package/src/module/integration/strategies/email/sendgrid-api.strategy.ts +263 -260
  152. package/src/module/integration/strategies/integration.strategy.ts +97 -97
  153. package/src/module/integration/strategies/sms/gupshup-sms.strategy.ts +146 -146
  154. package/src/module/integration/strategies/sms/msg91-sms.strategy.ts +164 -164
  155. package/src/module/integration/strategies/sms/tubelight-sms.strategy.ts +163 -163
  156. package/src/module/integration/strategies/telephone/ozonetel-voice.strategy.ts +238 -238
  157. package/src/module/integration/strategies/telephone/tubelight-voice.strategy.ts +210 -210
  158. package/src/module/integration/strategies/whatsapp/gupshup-whatsapp.strategy.ts +359 -359
  159. package/src/module/integration/strategies/whatsapp/tubelight-whatsapp.strategy.ts +372 -372
  160. package/src/module/integration/strategies/whatsapp/whatsapp-cloud.strategy.ts +403 -403
  161. package/src/module/integration/strategies/whatsapp/whatsapp.strategy.ts +57 -57
  162. package/src/module/layout/controller/layout.controller.ts +47 -47
  163. package/src/module/layout/entity/header-items.entity.ts +28 -28
  164. package/src/module/layout/entity/header-section.entity.ts +19 -19
  165. package/src/module/layout/layout.module.ts +21 -21
  166. package/src/module/layout/repository/header-items.repository.ts +18 -18
  167. package/src/module/layout/repository/header-section.repository.ts +22 -22
  168. package/src/module/layout/service/header-section.service.ts +25 -25
  169. package/src/module/layout_preference/controller/layout_preference.controller.ts +76 -76
  170. package/src/module/layout_preference/entity/layout_preference.entity.ts +28 -28
  171. package/src/module/layout_preference/layout_preference.module.ts +22 -22
  172. package/src/module/layout_preference/repository/layout_preference.repository.ts +65 -65
  173. package/src/module/layout_preference/service/layout_preference.service.ts +191 -191
  174. package/src/module/lead/controller/lead.controller.ts +30 -30
  175. package/src/module/lead/lead.module.ts +14 -14
  176. package/src/module/lead/repository/lead.repository.ts +41 -41
  177. package/src/module/lead/service/lead.service.ts +54 -54
  178. package/src/module/linked_attributes/controller/linked_attributes.controller.ts +137 -137
  179. package/src/module/linked_attributes/dto/create-linked-attribute-smart.dto.ts +54 -54
  180. package/src/module/linked_attributes/entity/linked_attribute.entity.ts +51 -51
  181. package/src/module/linked_attributes/linked_attributes.module.ts +23 -23
  182. package/src/module/linked_attributes/repository/linked_attribute.repository.ts +12 -12
  183. package/src/module/linked_attributes/service/linked_attributes.service.ts +648 -648
  184. package/src/module/linked_attributes/test/linked-attributes.service.spec.ts +244 -244
  185. package/src/module/listmaster/controller/list-master.controller.ts +230 -230
  186. package/src/module/listmaster/entity/list-master-items.entity.ts +43 -43
  187. package/src/module/listmaster/entity/list-master.entity.ts +33 -33
  188. package/src/module/listmaster/listmaster.module.ts +46 -46
  189. package/src/module/listmaster/repository/list-master-items.repository.ts +173 -173
  190. package/src/module/listmaster/repository/list-master.repository.ts +56 -56
  191. package/src/module/listmaster/service/list-master-engine.ts +19 -19
  192. package/src/module/listmaster/service/list-master-extension.interface.ts +4 -4
  193. package/src/module/listmaster/service/list-master-item.service.ts +280 -280
  194. package/src/module/listmaster/service/list-master-registry.ts +15 -15
  195. package/src/module/listmaster/service/list-master.service.ts +527 -527
  196. package/src/module/mapper/controller/field-mapper.controller.ts +76 -76
  197. package/src/module/mapper/controller/mapper.controller.ts +20 -20
  198. package/src/module/mapper/dto/field-mapper.dto.ts +14 -14
  199. package/src/module/mapper/entity/field-lovs.entity.ts +19 -19
  200. package/src/module/mapper/entity/field-mapper.entity.ts +53 -53
  201. package/src/module/mapper/entity/mapper.entity.ts +16 -16
  202. package/src/module/mapper/mapper.module.ts +35 -35
  203. package/src/module/mapper/repository/field-lovs.repository.ts +35 -35
  204. package/src/module/mapper/repository/field-mapper.repository.ts +42 -42
  205. package/src/module/mapper/repository/mapper.repository.ts +32 -32
  206. package/src/module/mapper/service/field-mapper.service.ts +269 -269
  207. package/src/module/mapper/service/mapper.service.ts +80 -80
  208. package/src/module/master/controller/master.controller.ts +74 -74
  209. package/src/module/master/service/master.service.ts +484 -484
  210. package/src/module/meta/controller/app-master.controller.ts +38 -38
  211. package/src/module/meta/controller/attribute-master.controller.ts +96 -96
  212. package/src/module/meta/controller/entity-dynamic.controller.ts +125 -125
  213. package/src/module/meta/controller/entity-master.controller.ts +41 -41
  214. package/src/module/meta/controller/entity-relation.controller.ts +36 -36
  215. package/src/module/meta/controller/entity.controller.ts +308 -308
  216. package/src/module/meta/controller/entity.public.controller.ts +75 -75
  217. package/src/module/meta/controller/media.controller.ts +167 -135
  218. package/src/module/meta/controller/meta.controller.ts +101 -101
  219. package/src/module/meta/controller/view-master.controller.ts +79 -79
  220. package/src/module/meta/dto/entity-list-data.dto.ts +6 -6
  221. package/src/module/meta/dto/entity-tab.dto.ts +4 -4
  222. package/src/module/meta/dto/entity-table.dto.ts +12 -12
  223. package/src/module/meta/entity/app-master.entity.ts +37 -37
  224. package/src/module/meta/entity/attribute-master.entity.ts +92 -92
  225. package/src/module/meta/entity/base-entity.entity.ts +75 -75
  226. package/src/module/meta/entity/entity-master.entity.ts +91 -91
  227. package/src/module/meta/entity/entity-relation-data.entity.ts +29 -29
  228. package/src/module/meta/entity/entity-relation.entity.ts +23 -23
  229. package/src/module/meta/entity/entity-table-column.entity.ts +61 -61
  230. package/src/module/meta/entity/entity-table.entity.ts +50 -50
  231. package/src/module/meta/entity/media-data.entity.ts +38 -32
  232. package/src/module/meta/entity/preference.entity.ts +62 -62
  233. package/src/module/meta/entity/view-master.entity.ts +41 -41
  234. package/src/module/meta/entity.module.ts +165 -165
  235. package/src/module/meta/repository/app-master.repository.ts +20 -20
  236. package/src/module/meta/repository/attribute-master.repository.ts +164 -164
  237. package/src/module/meta/repository/entity-attribute-update.repository.ts +48 -48
  238. package/src/module/meta/repository/entity-master.repository.ts +120 -120
  239. package/src/module/meta/repository/entity-relation.repository.ts +22 -22
  240. package/src/module/meta/repository/entity-table-column.repository.ts +39 -39
  241. package/src/module/meta/repository/entity-table.repository.ts +53 -53
  242. package/src/module/meta/repository/media-data.repository.ts +50 -50
  243. package/src/module/meta/repository/preference.repository.ts +20 -20
  244. package/src/module/meta/repository/user-app-mapping.repository.ts +28 -28
  245. package/src/module/meta/repository/view-master.repository.ts +42 -42
  246. package/src/module/meta/service/app-master.service.ts +37 -37
  247. package/src/module/meta/service/attribute-master.service.ts +160 -160
  248. package/src/module/meta/service/common.service.ts +9 -9
  249. package/src/module/meta/service/entity-attribute-update.service.ts +26 -26
  250. package/src/module/meta/service/entity-dynamic.service.ts +835 -824
  251. package/src/module/meta/service/entity-master.service.ts +172 -172
  252. package/src/module/meta/service/entity-realation-data.service.ts +9 -9
  253. package/src/module/meta/service/entity-relation.service.ts +78 -78
  254. package/src/module/meta/service/entity-service-impl.service.ts +389 -389
  255. package/src/module/meta/service/entity-table-column.service.ts +26 -26
  256. package/src/module/meta/service/entity-table.service.ts +171 -171
  257. package/src/module/meta/service/entity-validation.service.ts +188 -188
  258. package/src/module/meta/service/entity.service.ts +48 -48
  259. package/src/module/meta/service/field-group.service.ts +103 -103
  260. package/src/module/meta/service/media-data.service.ts +610 -591
  261. package/src/module/meta/service/populate-meta.service.ts +228 -222
  262. package/src/module/meta/service/preference.service.ts +16 -16
  263. package/src/module/meta/service/resolver.service.ts +347 -319
  264. package/src/module/meta/service/section-master.service.ts +104 -104
  265. package/src/module/meta/service/update-form-json.service.ts +22 -22
  266. package/src/module/meta/service/user-app-mapping.service.ts +17 -17
  267. package/src/module/meta/service/view-master.service.ts +127 -127
  268. package/src/module/microservice-client/microservice-clients.module.ts +13 -13
  269. package/src/module/microservice-client/service/microservice-client-factory.ts +37 -37
  270. package/src/module/microservice-client/service/microservice-clients.ts +4 -4
  271. package/src/module/module/controller/menu.controller.ts +15 -15
  272. package/src/module/module/controller/module-access.controller.ts +133 -133
  273. package/src/module/module/entity/menu.entity.ts +43 -43
  274. package/src/module/module/entity/module-access.entity.ts +25 -25
  275. package/src/module/module/entity/module-action.entity.ts +17 -17
  276. package/src/module/module/entity/module.entity.ts +52 -52
  277. package/src/module/module/module.module.ts +42 -42
  278. package/src/module/module/repository/menu.repository.ts +186 -186
  279. package/src/module/module/repository/module-access.repository.ts +344 -344
  280. package/src/module/module/service/menu.service.ts +82 -82
  281. package/src/module/module/service/module-access.service.ts +189 -189
  282. package/src/module/notification/controller/notification.controller.ts +58 -58
  283. package/src/module/notification/controller/otp.controller.ts +117 -117
  284. package/src/module/notification/entity/notification.entity.ts +26 -26
  285. package/src/module/notification/entity/otp.entity.ts +28 -28
  286. package/src/module/notification/firebase-admin.config.ts +22 -22
  287. package/src/module/notification/notification.module.ts +71 -71
  288. package/src/module/notification/repository/notification.repository.ts +33 -33
  289. package/src/module/notification/repository/otp.repository.ts +27 -27
  290. package/src/module/notification/service/email.service.ts +127 -127
  291. package/src/module/notification/service/notification.service.ts +146 -146
  292. package/src/module/notification/service/otp.service.ts +133 -133
  293. package/src/module/third-party-module/entity/third-party-api-registry.entity.ts +52 -52
  294. package/src/module/third-party-module/repository/third-party-api-registry.repository.ts +20 -20
  295. package/src/module/third-party-module/service/api-registry.service.ts +13 -13
  296. package/src/module/third-party-module/third-party.module.ts +12 -12
  297. package/src/module/user/controller/login.controller.ts +199 -198
  298. package/src/module/user/controller/user.controller.ts +40 -40
  299. package/src/module/user/dto/create-user.dto.ts +62 -62
  300. package/src/module/user/dto/update-user.dto.ts +4 -4
  301. package/src/module/user/entity/role.entity.ts +33 -33
  302. package/src/module/user/entity/user-role-mapping.entity.ts +38 -38
  303. package/src/module/user/entity/user-session.entity.ts +73 -73
  304. package/src/module/user/entity/user.entity.ts +62 -62
  305. package/src/module/user/repository/role.repository.ts +96 -96
  306. package/src/module/user/repository/user-role-mapping.repository.ts +126 -126
  307. package/src/module/user/repository/user.repository.ts +50 -50
  308. package/src/module/user/repository/userSession.repository.ts +33 -33
  309. package/src/module/user/service/login.service.ts +326 -326
  310. package/src/module/user/service/role.service.ts +197 -197
  311. package/src/module/user/service/user-role-mapping.service.ts +98 -98
  312. package/src/module/user/service/user-session.service.ts +201 -200
  313. package/src/module/user/service/user.service.ts +368 -368
  314. package/src/module/user/user.module.ts +65 -65
  315. package/src/module/workflow/controller/action-category.controller.ts +54 -54
  316. package/src/module/workflow/controller/action-resource-mapping.controller.ts +23 -23
  317. package/src/module/workflow/controller/action-template-mapping.controller.ts +35 -35
  318. package/src/module/workflow/controller/action.controller.ts +111 -111
  319. package/src/module/workflow/controller/activity-log.controller.ts +55 -55
  320. package/src/module/workflow/controller/comm-template.controller.ts +43 -43
  321. package/src/module/workflow/controller/entity-modification.controller.ts +35 -35
  322. package/src/module/workflow/controller/form-master.controller.ts +43 -43
  323. package/src/module/workflow/controller/stage-group.controller.ts +49 -49
  324. package/src/module/workflow/controller/stage.controller.ts +51 -51
  325. package/src/module/workflow/controller/task.controller.ts +77 -77
  326. package/src/module/workflow/controller/workflow-list-master.controller.ts +44 -44
  327. package/src/module/workflow/controller/workflow-meta.controller.ts +80 -80
  328. package/src/module/workflow/controller/workflow.controller.ts +67 -67
  329. package/src/module/workflow/entity/action-category.entity.ts +38 -38
  330. package/src/module/workflow/entity/action-data.entity.ts +55 -55
  331. package/src/module/workflow/entity/action-resources-mapping.entity.ts +29 -29
  332. package/src/module/workflow/entity/action-template-mapping.entity.ts +17 -17
  333. package/src/module/workflow/entity/action.entity.ts +53 -53
  334. package/src/module/workflow/entity/activity-log.entity.ts +43 -43
  335. package/src/module/workflow/entity/comm-template.entity.ts +43 -43
  336. package/src/module/workflow/entity/entity-modification.entity.ts +38 -38
  337. package/src/module/workflow/entity/form.entity.ts +25 -25
  338. package/src/module/workflow/entity/stage-action-mapping.entity.ts +17 -17
  339. package/src/module/workflow/entity/stage-group.entity.ts +23 -23
  340. package/src/module/workflow/entity/stage-movement-data.entity.ts +38 -38
  341. package/src/module/workflow/entity/stage.entity.ts +20 -20
  342. package/src/module/workflow/entity/task-data.entity.ts +88 -88
  343. package/src/module/workflow/entity/template-attach-mapper.entity.ts +30 -30
  344. package/src/module/workflow/entity/workflow-data.entity.ts +11 -11
  345. package/src/module/workflow/entity/workflow-level-mapping.entity.ts +18 -18
  346. package/src/module/workflow/entity/workflow.entity.ts +20 -20
  347. package/src/module/workflow/repository/action-category.repository.ts +79 -79
  348. package/src/module/workflow/repository/action-data.repository.ts +354 -354
  349. package/src/module/workflow/repository/action.repository.ts +339 -339
  350. package/src/module/workflow/repository/activity-log.repository.ts +148 -148
  351. package/src/module/workflow/repository/comm-template.repository.ts +157 -157
  352. package/src/module/workflow/repository/form-master.repository.ts +50 -50
  353. package/src/module/workflow/repository/stage-group.repository.ts +186 -186
  354. package/src/module/workflow/repository/stage-movement.repository.ts +217 -217
  355. package/src/module/workflow/repository/stage.repository.ts +160 -160
  356. package/src/module/workflow/repository/task.repository.ts +156 -156
  357. package/src/module/workflow/repository/workflow.repository.ts +42 -42
  358. package/src/module/workflow/service/action-category.service.ts +33 -33
  359. package/src/module/workflow/service/action-data.service.ts +62 -62
  360. package/src/module/workflow/service/action-resources-mapping.service.ts +10 -10
  361. package/src/module/workflow/service/action-template-mapping.service.ts +137 -137
  362. package/src/module/workflow/service/action.service.ts +302 -302
  363. package/src/module/workflow/service/activity-log.service.ts +107 -107
  364. package/src/module/workflow/service/comm-template.service.ts +181 -181
  365. package/src/module/workflow/service/entity-modification.service.ts +61 -61
  366. package/src/module/workflow/service/form-master.service.ts +35 -35
  367. package/src/module/workflow/service/populate-workflow.service.ts +325 -320
  368. package/src/module/workflow/service/stage-action-mapping.service.ts +5 -5
  369. package/src/module/workflow/service/stage-group.service.ts +325 -325
  370. package/src/module/workflow/service/stage.service.ts +197 -197
  371. package/src/module/workflow/service/task.service.ts +551 -551
  372. package/src/module/workflow/service/workflow-list-master.service.ts +68 -68
  373. package/src/module/workflow/service/workflow-meta.service.ts +640 -640
  374. package/src/module/workflow/service/workflow.service.ts +213 -213
  375. package/src/module/workflow/workflow.module.ts +180 -180
  376. package/src/module/workflow-automation/SCHEDULING_GUIDE.md +145 -145
  377. package/src/module/workflow-automation/controller/workflow-automation.controller.ts +43 -43
  378. package/src/module/workflow-automation/entity/workflow-automation-action.entity.ts +26 -26
  379. package/src/module/workflow-automation/entity/workflow-automation.entity.ts +40 -40
  380. package/src/module/workflow-automation/interface/action.decorator.ts +7 -7
  381. package/src/module/workflow-automation/interface/action.interface.ts +5 -5
  382. package/src/module/workflow-automation/service/action-registery.service.ts +35 -35
  383. package/src/module/workflow-automation/service/schedule-handler.service.ts +168 -168
  384. package/src/module/workflow-automation/service/workflow-automation-engine.service.ts +219 -219
  385. package/src/module/workflow-automation/service/workflow-automation.service.ts +474 -474
  386. package/src/module/workflow-automation/workflow-automation.module.ts +54 -54
  387. package/src/module/workflow-schedule/INSTALLATION.md +244 -244
  388. package/src/module/workflow-schedule/MULTI_PROJECT_GUIDE.md +196 -196
  389. package/src/module/workflow-schedule/README.md +422 -422
  390. package/src/module/workflow-schedule/constants/schedule.constants.ts +48 -48
  391. package/src/module/workflow-schedule/controller/workflow-schedule.controller.ts +253 -253
  392. package/src/module/workflow-schedule/docs/CLAUDE_CODE_GUIDE.md +510 -510
  393. package/src/module/workflow-schedule/docs/CLAUDE_CODE_PROMPT.md +362 -362
  394. package/src/module/workflow-schedule/docs/RUN_CLAUDE_CODE.sh +68 -68
  395. package/src/module/workflow-schedule/dto/create-schedule.dto.ts +147 -147
  396. package/src/module/workflow-schedule/dto/get-execution-logs.dto.ts +119 -119
  397. package/src/module/workflow-schedule/dto/update-schedule.dto.ts +96 -96
  398. package/src/module/workflow-schedule/entities/scheduled-workflow.entity.ts +148 -148
  399. package/src/module/workflow-schedule/entities/workflow-execution-log.entity.ts +154 -154
  400. package/src/module/workflow-schedule/interfaces/schedule-job-data.interface.ts +53 -53
  401. package/src/module/workflow-schedule/interfaces/workflow-schedule-options.interface.ts +12 -12
  402. package/src/module/workflow-schedule/processors/schedule.processor.ts +620 -620
  403. package/src/module/workflow-schedule/service/workflow-schedule.service.ts +597 -597
  404. package/src/module/workflow-schedule/workflow-schedule.module.ts +67 -67
  405. package/src/resources/dev.properties.yaml +31 -31
  406. package/src/resources/local.properties.yaml +27 -27
  407. package/src/resources/properties.module.ts +12 -12
  408. package/src/resources/properties.yaml.ts +11 -11
  409. package/src/resources/uat.properties.yaml +31 -31
  410. package/src/table.config.ts +135 -135
  411. package/src/utils/dto/excel-data.dto.ts +14 -14
  412. package/src/utils/dto/excelsheet-data.dto.ts +5 -5
  413. package/src/utils/service/base64util.service.ts +18 -18
  414. package/src/utils/service/clockIDGenUtil.service.ts +21 -21
  415. package/src/utils/service/codeGenerator.service.ts +22 -22
  416. package/src/utils/service/dateUtil.service.ts +17 -17
  417. package/src/utils/service/encryptUtil.service.ts +97 -97
  418. package/src/utils/service/excel-helper.service.ts +72 -72
  419. package/src/utils/service/excelUtil.service.ts +15 -15
  420. package/src/utils/service/file-util.service.ts +11 -11
  421. package/src/utils/service/json-util.service.ts +23 -23
  422. package/src/utils/service/loggingUtil.service.ts +88 -88
  423. package/src/utils/service/reflection-helper.service.ts +62 -62
  424. package/src/utils/service/wbsCodeGen.service.ts +8 -8
  425. package/src/utils/utils.module.ts +27 -27
  426. package/tsconfig.build.json +4 -4
  427. package/tsconfig.json +24 -24
@@ -1,903 +1,903 @@
1
- import { Injectable, BadRequestException, Inject, forwardRef } from '@nestjs/common';
2
- import { EntityManager, SelectQueryBuilder } from 'typeorm';
3
- import { EntityMasterService } from 'src/module/meta/service/entity-master.service';
4
- import { AttributeMasterService } from 'src/module/meta/service/attribute-master.service';
5
- import { ResolverService } from 'src/module/meta/service/resolver.service';
6
- import { LoggingService } from 'src/utils/service/loggingUtil.service';
7
- import { ConfigService } from '@nestjs/config';
8
- import {
9
- FilterRequestDto,
10
- FilterCondition,
11
- SortConfig,
12
- } from '../dto/filter-request.dto';
13
- import { EntityJson } from 'src/module/entity_json/entity/entityJson.entity';
14
- import { SavedFilterRepositoryService } from '../repository/saved-filter.repository';
15
- import { EntityJSONService } from 'src/module/entity_json/service/entity_json.service';
16
-
17
- /**
18
- * JSONB query condition structure
19
- */
20
- interface JsonbCondition {
21
- query: string;
22
- params: Record<string, any>;
23
- }
24
-
25
- @Injectable()
26
- export class FlatjsonFilterService {
27
- constructor(
28
- private readonly entityManager: EntityManager,
29
- @Inject(forwardRef(() => EntityMasterService))
30
- private readonly entityMasterService: EntityMasterService,
31
- @Inject(forwardRef(() => AttributeMasterService))
32
- private readonly attributeMasterService: AttributeMasterService,
33
- @Inject(forwardRef(() => ResolverService))
34
- private readonly resolverService: ResolverService,
35
- @Inject(forwardRef(() => LoggingService))
36
- private readonly loggingService: LoggingService,
37
- private readonly configService: ConfigService,
38
- @Inject(forwardRef(() => SavedFilterRepositoryService))
39
- private readonly savedFilterRepositoryService: SavedFilterRepositoryService,
40
- @Inject(forwardRef(() => EntityJSONService))
41
- private readonly entityJsonService: EntityJSONService,
42
- ) {}
43
-
44
- /**
45
- * Main filtering method - queries frm_entity_json table
46
- */
47
- async applyFlatjsonFilter(dto: FilterRequestDto): Promise<any> {
48
- const {
49
- entity_type,
50
- quickFilter = [],
51
- savedFilterCode,
52
- attributeFilter = [],
53
- sortby = [],
54
- tabs,
55
- page = 1,
56
- size = 20,
57
- loggedInUser,
58
- } = dto;
59
-
60
- await this.loggingService.log(
61
- 'info',
62
- 'FlatjsonFilterService',
63
- 'applyFlatjsonFilter',
64
- `Filtering entity: ${entity_type}`,
65
- );
66
-
67
- // Step 1: Load attribute metadata for this entity
68
- let attributes =
69
- await this.entityJsonService.getAttributeForFlatJSON(
70
- entity_type,
71
- loggedInUser,
72
- 'filter_attribute',
73
- );
74
-
75
- if (!attributes?.length) {
76
- attributes =
77
- await this.attributeMasterService.findAttributesByMappedEntityType(
78
- entity_type,
79
- loggedInUser,
80
- );
81
- }
82
-
83
- attributes = attributes ?? [];
84
-
85
- const attributeMetaMap: Record<string, any> = {};
86
- attributes.forEach((attr) => {
87
- attributeMetaMap[attr.flat_json_key] = attr;
88
- });
89
-
90
- await this.loggingService.log(
91
- 'debug',
92
- 'FlatjsonFilterService',
93
- 'applyFlatjsonFilter',
94
- `Loaded ${attributes.length} attributes`,
95
- );
96
-
97
- // Step 2: Merge filters (quickFilter + savedFilter + attributeFilter)
98
- let allFilters: FilterCondition[] = [...quickFilter, ...attributeFilter];
99
-
100
- if (savedFilterCode) {
101
- const savedFilterDetails =
102
- await this.savedFilterRepositoryService.getDetailsByCode(
103
- savedFilterCode,
104
- );
105
- allFilters = [...allFilters, ...savedFilterDetails];
106
- }
107
-
108
- await this.loggingService.log(
109
- 'debug',
110
- 'FlatjsonFilterService',
111
- 'applyFlatjsonFilter',
112
- `Total filters: ${allFilters.length}`,
113
- );
114
-
115
- // Step 3: Build JSONB conditions
116
- const jsonbConditions = this.buildJsonbConditions(
117
- allFilters,
118
- attributeMetaMap,
119
- );
120
-
121
- // Step 4: Build base query
122
- const qb = this.entityManager
123
- .getRepository(EntityJson)
124
- .createQueryBuilder('ej')
125
- .where('ej.entity_type = :entity_type', { entity_type })
126
- .andWhere('ej.organization_id = :orgId', {
127
- orgId: loggedInUser.organization_id,
128
- });
129
-
130
- // Step 5: Apply JSONB conditions
131
- jsonbConditions.forEach((condition, index) => {
132
- qb.andWhere(condition.query, condition.params);
133
- });
134
-
135
- // Step 6: Apply tab filter if provided
136
- if (tabs?.columnName && tabs?.value) {
137
- const tabMeta = attributeMetaMap[tabs.columnName];
138
- if (tabMeta) {
139
- const flatJsonKey =
140
- tabMeta.flat_json_key ||
141
- `${entity_type}__${tabs.columnName}`;
142
- qb.andWhere(`json_data->>'${flatJsonKey}' = :tabValue`, {
143
- tabValue: tabs.value,
144
- });
145
- }
146
- }
147
-
148
- // Step 7: Get tab counts if needed (before pagination)
149
- let tabCounts: any[] = [];
150
- if (tabs?.columnName && !tabs?.value) {
151
- const tabMeta = attributeMetaMap[tabs.columnName];
152
- if (tabMeta) {
153
- const flatJsonKey =
154
- tabMeta.flat_json_key ||
155
- `${entity_type}__${tabs.columnName}`;
156
- tabCounts = await this.getJsonbTabCounts(
157
- entity_type,
158
- flatJsonKey,
159
- jsonbConditions,
160
- );
161
- }
162
- }
163
-
164
- // Step 8: Apply sorting
165
- if (sortby && sortby.length > 0) {
166
- this.applyJsonbSorting(qb, sortby, attributeMetaMap);
167
- } else {
168
- // Default sort by created_date DESC
169
- qb.orderBy('ej.created_at', 'DESC');
170
- }
171
-
172
- // Step 9: Get total count (before pagination)
173
- const totalCount = await qb.getCount();
174
-
175
- // Step 10: Apply pagination
176
- if (page && size) {
177
- const skip = (page - 1) * size;
178
- qb.skip(skip).take(size);
179
- }
180
-
181
- const results = await qb.getMany();
182
-
183
- await this.loggingService.log(
184
- 'info',
185
- 'FlatjsonFilterService',
186
- 'applyFlatjsonFilter',
187
- `Found ${totalCount} total, returning ${results.length} records`,
188
- );
189
-
190
- // Map only json_data
191
- const jsonDataResults = results.map((r) => r.json_data);
192
-
193
- return {
194
- data: jsonDataResults,
195
- total: totalCount,
196
- page: page || 1,
197
- size: size || 20,
198
- totalPages: size ? Math.ceil(totalCount / size) : 1,
199
- tabCounts: tabCounts.length > 0 ? tabCounts : undefined,
200
- };
201
-
202
- }
203
-
204
- /**
205
- * Build JSONB where clauses from filter conditions
206
- */
207
- private buildJsonbConditions(
208
- filters: FilterCondition[],
209
- attributeMetaMap: Record<string, any>,
210
- ): JsonbCondition[] {
211
- const conditions: JsonbCondition[] = [];
212
-
213
- for (const filter of filters) {
214
- const meta = attributeMetaMap[filter.filter_attribute];
215
- const condition = this.buildJsonbCondition(filter, meta);
216
-
217
- if (condition) {
218
- conditions.push(condition);
219
- } else {
220
- console.warn(
221
- `Could not build condition for filter: ${filter.filter_attribute}`,
222
- );
223
- }
224
- }
225
-
226
- return conditions;
227
- }
228
-
229
- /**
230
- * Resolve entity IDs to full entity data
231
- */
232
- private async resolveEntityData(
233
- entityJsonRecords: EntityJson[],
234
- entity_type: string,
235
- loggedInUser: any,
236
- ): Promise<any[]> {
237
- if (entityJsonRecords.length === 0) return [];
238
-
239
- const entityIds = entityJsonRecords.map((ej) => ej.entity_id);
240
-
241
- try {
242
- // Use resolver service to get full entity data
243
- const fullEntities = await this.resolverService.getResolvedData(
244
- entity_type,
245
- entityIds,
246
- loggedInUser,
247
- );
248
-
249
- // Return with json_data merged
250
- return entityJsonRecords.map((ej) => {
251
- const fullEntity = fullEntities.find((e) => e.id === ej.entity_id);
252
- return {
253
- ...fullEntity,
254
- _flatjson: ej.json_data, // Include flatjson for debugging
255
- };
256
- });
257
- } catch (error) {
258
- console.error('Error resolving entity data:', error);
259
- // Fallback: return just the entity IDs with json_data
260
- return entityJsonRecords.map((ej) => ({
261
- id: ej.entity_id,
262
- entity_type: ej.entity_type,
263
- ...ej.json_data,
264
- }));
265
- }
266
- }
267
-
268
- /**
269
- * Build a single JSONB condition based on data type
270
- */
271
- private buildJsonbCondition(
272
- filter: FilterCondition,
273
- meta: any,
274
- ): JsonbCondition | null {
275
- if (!meta) return null;
276
-
277
- const flatJsonKey =
278
- meta.flat_json_key ||
279
- `${meta.mapped_entity_type}__${filter.filter_attribute}`;
280
- const op = filter.filter_operator;
281
- const val = filter.filter_value;
282
- const key = `param_${filter.filter_attribute}_${Math.random().toString(36).substring(2, 8)}`;
283
-
284
- switch (meta.data_type) {
285
- case 'text':
286
- return this.buildTextCondition(flatJsonKey, op, val, key);
287
- case 'number':
288
- return this.buildNumberCondition(flatJsonKey, op, val, key);
289
- case 'date':
290
- return this.buildDateCondition(flatJsonKey, op, val, key);
291
- case 'select':
292
- case 'radio':
293
- return this.buildSelectCondition(flatJsonKey, op, val, key);
294
- case 'multiselect':
295
- case 'checkbox':
296
- return this.buildMultiSelectCondition(flatJsonKey, op, val, key);
297
- case 'year':
298
- return this.buildYearCondition(flatJsonKey, op, val, key);
299
- default:
300
- return null;
301
- }
302
- }
303
-
304
- /**
305
- * Build text condition (JSONB ->> operator)
306
- */
307
- private buildTextCondition(
308
- flatJsonKey: string,
309
- operator: string,
310
- value: any,
311
- paramKey: string,
312
- ): JsonbCondition | null {
313
- // Text is already stored lowercase in flatjson
314
- const lowerValue = value ? String(value).toLowerCase() : '';
315
-
316
- switch (operator) {
317
- case 'equal':
318
- return {
319
- query: `json_data->>'${flatJsonKey}' = :${paramKey}`,
320
- params: { [paramKey]: lowerValue },
321
- };
322
-
323
- case 'not_equal':
324
- return {
325
- query: `json_data->>'${flatJsonKey}' != :${paramKey}`,
326
- params: { [paramKey]: lowerValue },
327
- };
328
-
329
- case 'contains':
330
- return {
331
- query: `json_data->>'${flatJsonKey}' LIKE :${paramKey}`,
332
- params: { [paramKey]: `%${lowerValue}%` },
333
- };
334
-
335
- case 'not_contains':
336
- return {
337
- query: `json_data->>'${flatJsonKey}' NOT LIKE :${paramKey}`,
338
- params: { [paramKey]: `%${lowerValue}%` },
339
- };
340
-
341
- case 'starts_with':
342
- return {
343
- query: `json_data->>'${flatJsonKey}' LIKE :${paramKey}`,
344
- params: { [paramKey]: `${lowerValue}%` },
345
- };
346
-
347
- case 'ends_with':
348
- return {
349
- query: `json_data->>'${flatJsonKey}' LIKE :${paramKey}`,
350
- params: { [paramKey]: `%${lowerValue}` },
351
- };
352
-
353
- case 'empty':
354
- return {
355
- query: `(json_data->>'${flatJsonKey}' IS NULL OR json_data->>'${flatJsonKey}' = '')`,
356
- params: {},
357
- };
358
-
359
- case 'not_empty':
360
- return {
361
- query: `(json_data->>'${flatJsonKey}' IS NOT NULL AND json_data->>'${flatJsonKey}' != '')`,
362
- params: {},
363
- };
364
-
365
- default:
366
- console.warn(`Unsupported text operator: ${operator}`);
367
- return null;
368
- }
369
- }
370
-
371
- /**
372
- * Build number condition (JSONB ->> with ::numeric cast)
373
- */
374
- private buildNumberCondition(
375
- flatJsonKey: string,
376
- operator: string,
377
- value: any,
378
- paramKey: string,
379
- ): JsonbCondition | null {
380
- // Cast JSONB text to numeric for comparison
381
- const jsonbField = `(json_data->>'${flatJsonKey}')::numeric`;
382
-
383
- switch (operator) {
384
- case 'equal':
385
- return {
386
- query: `${jsonbField} = :${paramKey}`,
387
- params: { [paramKey]: Number(value) },
388
- };
389
-
390
- case 'not_equal':
391
- return {
392
- query: `${jsonbField} != :${paramKey}`,
393
- params: { [paramKey]: Number(value) },
394
- };
395
-
396
- case 'greater_than':
397
- return {
398
- query: `${jsonbField} > :${paramKey}`,
399
- params: { [paramKey]: Number(value) },
400
- };
401
-
402
- case 'less_than':
403
- return {
404
- query: `${jsonbField} < :${paramKey}`,
405
- params: { [paramKey]: Number(value) },
406
- };
407
-
408
- case 'greater_than_equal_to':
409
- return {
410
- query: `${jsonbField} >= :${paramKey}`,
411
- params: { [paramKey]: Number(value) },
412
- };
413
-
414
- case 'less_than_equal_to':
415
- return {
416
- query: `${jsonbField} <= :${paramKey}`,
417
- params: { [paramKey]: Number(value) },
418
- };
419
-
420
- case 'between': {
421
- // Value should be array [min, max] or string "min,max"
422
- let range: number[];
423
- if (typeof value === 'string') {
424
- range = value.split(',').map((v) => Number(v.trim()));
425
- } else if (Array.isArray(value)) {
426
- range = value.map((v) => Number(v));
427
- } else {
428
- return null;
429
- }
430
-
431
- if (range.length !== 2) return null;
432
-
433
- return {
434
- query: `${jsonbField} BETWEEN :${paramKey}_min AND :${paramKey}_max`,
435
- params: {
436
- [`${paramKey}_min`]: range[0],
437
- [`${paramKey}_max`]: range[1],
438
- },
439
- };
440
- }
441
-
442
- case 'empty':
443
- return {
444
- query: `json_data->>'${flatJsonKey}' IS NULL`,
445
- params: {},
446
- };
447
-
448
- case 'not_empty':
449
- return {
450
- query: `json_data->>'${flatJsonKey}' IS NOT NULL`,
451
- params: {},
452
- };
453
-
454
- default:
455
- console.warn(`Unsupported number operator: ${operator}`);
456
- return null;
457
- }
458
- }
459
-
460
- /**
461
- * Build date condition (JSONB ->> with ::bigint cast for epoch ms)
462
- */
463
- private buildDateCondition(
464
- flatJsonKey: string,
465
- operator: string,
466
- value: any,
467
- paramKey: string,
468
- ): JsonbCondition | null {
469
- // Dates stored as epoch milliseconds (bigint)
470
- const jsonbField = `(json_data->>'${flatJsonKey}')::bigint`;
471
-
472
- // Helper: Convert date string to epoch ms
473
- const toEpochMs = (dateStr: string): number => {
474
- return new Date(dateStr).getTime();
475
- };
476
-
477
- // Helper: Get date N days ago
478
- const daysAgo = (days: number): number => {
479
- const d = new Date();
480
- d.setDate(d.getDate() - days);
481
- return d.getTime();
482
- };
483
-
484
- // Helper: Get date N days from now
485
- const daysFromNow = (days: number): number => {
486
- const d = new Date();
487
- d.setDate(d.getDate() + days);
488
- return d.getTime();
489
- };
490
-
491
- // Helper: Subtract business days (skip weekends)
492
- const subtractBusinessDays = (days: number): number => {
493
- let d = new Date();
494
- let count = 0;
495
-
496
- while (count < days) {
497
- d.setDate(d.getDate() - 1);
498
- const day = d.getDay(); // 0=Sun, 6=Sat
499
- if (day !== 0 && day !== 6) {
500
- count++;
501
- }
502
- }
503
-
504
- return d.getTime();
505
- };
506
-
507
- const numVal = Number(value);
508
-
509
- switch (operator) {
510
- // Basic comparisons
511
- case 'equal':
512
- case 'is':
513
- return {
514
- query: `${jsonbField} = :${paramKey}`,
515
- params: { [paramKey]: toEpochMs(value) },
516
- };
517
-
518
- case 'before':
519
- case 'is_before':
520
- return {
521
- query: `${jsonbField} < :${paramKey}`,
522
- params: { [paramKey]: toEpochMs(value) },
523
- };
524
-
525
- case 'after':
526
- case 'is_after':
527
- return {
528
- query: `${jsonbField} > :${paramKey}`,
529
- params: { [paramKey]: toEpochMs(value) },
530
- };
531
-
532
- case 'is_on_or_before':
533
- return {
534
- query: `${jsonbField} <= :${paramKey}`,
535
- params: { [paramKey]: toEpochMs(value) },
536
- };
537
-
538
- case 'is_on_or_after':
539
- return {
540
- query: `${jsonbField} >= :${paramKey}`,
541
- params: { [paramKey]: toEpochMs(value) },
542
- };
543
-
544
- // Day offset logic
545
- case 'is_day_before':
546
- if (isNaN(numVal)) return null;
547
- return {
548
- query: `${jsonbField} <= :${paramKey}`,
549
- params: { [paramKey]: daysAgo(numVal) },
550
- };
551
-
552
- case 'is_day_after':
553
- if (isNaN(numVal)) return null;
554
- return {
555
- query: `${jsonbField} >= :${paramKey}`,
556
- params: { [paramKey]: daysFromNow(numVal) },
557
- };
558
-
559
- // Business days (skip weekends)
560
- case 'is_before_business_days':
561
- if (isNaN(numVal)) return null;
562
- return {
563
- query: `${jsonbField} <= :${paramKey}`,
564
- params: { [paramKey]: subtractBusinessDays(numVal) },
565
- };
566
-
567
- // Range operators
568
- case 'between': {
569
- let range: string[];
570
- if (typeof value === 'string') {
571
- range = value.split(',').map((v) => v.trim());
572
- } else if (Array.isArray(value)) {
573
- range = value;
574
- } else {
575
- return null;
576
- }
577
-
578
- if (range.length !== 2) return null;
579
-
580
- return {
581
- query: `${jsonbField} BETWEEN :${paramKey}_start AND :${paramKey}_end`,
582
- params: {
583
- [`${paramKey}_start`]: toEpochMs(range[0]),
584
- [`${paramKey}_end`]: toEpochMs(range[1]),
585
- },
586
- };
587
- }
588
-
589
- case 'in_last_day':
590
- if (isNaN(numVal)) return null;
591
- return {
592
- query: `${jsonbField} BETWEEN :${paramKey}_start AND :${paramKey}_end`,
593
- params: {
594
- [`${paramKey}_start`]: daysAgo(numVal),
595
- [`${paramKey}_end`]: Date.now(),
596
- },
597
- };
598
-
599
- case 'in_next_day':
600
- if (isNaN(numVal)) return null;
601
- return {
602
- query: `${jsonbField} BETWEEN :${paramKey}_start AND :${paramKey}_end`,
603
- params: {
604
- [`${paramKey}_start`]: Date.now(),
605
- [`${paramKey}_end`]: daysFromNow(numVal),
606
- },
607
- };
608
-
609
- // Special cases
610
- case 'today': {
611
- const todayStart = new Date().setHours(0, 0, 0, 0);
612
- const todayEnd = new Date().setHours(23, 59, 59, 999);
613
- return {
614
- query: `${jsonbField} BETWEEN :${paramKey}_start AND :${paramKey}_end`,
615
- params: {
616
- [`${paramKey}_start`]: todayStart,
617
- [`${paramKey}_end`]: todayEnd,
618
- },
619
- };
620
- }
621
-
622
- case 'empty':
623
- return {
624
- query: `json_data->>'${flatJsonKey}' IS NULL`,
625
- params: {},
626
- };
627
-
628
- case 'not_empty':
629
- return {
630
- query: `json_data->>'${flatJsonKey}' IS NOT NULL`,
631
- params: {},
632
- };
633
-
634
- default:
635
- console.warn(`Unsupported date operator: ${operator}`);
636
- return null;
637
- }
638
- }
639
-
640
- /**
641
- * Build select condition (single value)
642
- */
643
- private buildSelectCondition(
644
- flatJsonKey: string,
645
- operator: string,
646
- value: any,
647
- paramKey: string,
648
- ): JsonbCondition | null {
649
- switch (operator) {
650
- case 'equal':
651
- if (Array.isArray(value)) {
652
- // IN operator for multiple values
653
- return {
654
- query: `json_data->>'${flatJsonKey}' = ANY(:${paramKey})`,
655
- params: { [paramKey]: value.map(String) },
656
- };
657
- }
658
- return {
659
- query: `json_data->>'${flatJsonKey}' = :${paramKey}`,
660
- params: { [paramKey]: String(value) },
661
- };
662
-
663
- case 'not_equal':
664
- if (Array.isArray(value)) {
665
- // NOT IN operator
666
- return {
667
- query: `json_data->>'${flatJsonKey}' != ALL(:${paramKey})`,
668
- params: { [paramKey]: value.map(String) },
669
- };
670
- }
671
- return {
672
- query: `json_data->>'${flatJsonKey}' != :${paramKey}`,
673
- params: { [paramKey]: String(value) },
674
- };
675
-
676
- case 'in': {
677
- const inValues = Array.isArray(value) ? value : [value];
678
- return {
679
- query: `json_data->>'${flatJsonKey}' = ANY(:${paramKey})`,
680
- params: { [paramKey]: inValues.map(String) },
681
- };
682
- }
683
-
684
- case 'not_in': {
685
- const notInValues = Array.isArray(value) ? value : [value];
686
- return {
687
- query: `json_data->>'${flatJsonKey}' != ALL(:${paramKey})`,
688
- params: { [paramKey]: notInValues.map(String) },
689
- };
690
- }
691
-
692
- case 'empty':
693
- return {
694
- query: `json_data->>'${flatJsonKey}' IS NULL`,
695
- params: {},
696
- };
697
-
698
- case 'not_empty':
699
- return {
700
- query: `json_data->>'${flatJsonKey}' IS NOT NULL`,
701
- params: {},
702
- };
703
-
704
- default:
705
- console.warn(`Unsupported select operator: ${operator}`);
706
- return null;
707
- }
708
- }
709
-
710
- /**
711
- * Build multiselect condition (JSONB -> with ? operators for arrays)
712
- */
713
- private buildMultiSelectCondition(
714
- flatJsonKey: string,
715
- operator: string,
716
- value: any,
717
- paramKey: string,
718
- ): JsonbCondition | null {
719
- // Convert value to array if not already
720
- let arr: string[];
721
- if (Array.isArray(value)) {
722
- arr = value.map(String);
723
- } else if (typeof value === 'string') {
724
- arr = value.split(',').map((v) => v.trim());
725
- } else {
726
- arr = [String(value)];
727
- }
728
-
729
- if (arr.length === 0) {
730
- return { query: '1=1', params: {} }; // Always true for empty array
731
- }
732
-
733
- switch (operator) {
734
- case 'contains':
735
- case 'has':
736
- // Check if JSON array contains this element
737
- // json_data->'LEAD__languages' ? 'hindi'
738
- if (arr.length === 1) {
739
- return {
740
- query: `json_data->'${flatJsonKey}' ? :${paramKey}`,
741
- params: { [paramKey]: arr[0] },
742
- };
743
- }
744
- // For multiple values, check if contains ANY
745
- return {
746
- query: `json_data->'${flatJsonKey}' ?| ARRAY[:...${paramKey}]::text[]`,
747
- params: { [paramKey]: arr },
748
- };
749
-
750
- case 'contains_all':
751
- // Check if JSON array contains ALL elements
752
- // json_data->'LEAD__languages' ?& ARRAY['hindi', 'english']
753
- return {
754
- query: `json_data->'${flatJsonKey}' ?& ARRAY[:...${paramKey}]::text[]`,
755
- params: { [paramKey]: arr },
756
- };
757
-
758
- case 'contains_any':
759
- // Check if JSON array contains ANY element
760
- // json_data->'LEAD__languages' ?| ARRAY['hindi', 'english']
761
- return {
762
- query: `json_data->'${flatJsonKey}' ?| ARRAY[:...${paramKey}]::text[]`,
763
- params: { [paramKey]: arr },
764
- };
765
-
766
- case 'not_contains':
767
- // Check if JSON array does NOT contain element
768
- if (arr.length === 1) {
769
- return {
770
- query: `NOT (json_data->'${flatJsonKey}' ? :${paramKey})`,
771
- params: { [paramKey]: arr[0] },
772
- };
773
- }
774
- return {
775
- query: `NOT (json_data->'${flatJsonKey}' ?| ARRAY[:...${paramKey}]::text[])`,
776
- params: { [paramKey]: arr },
777
- };
778
-
779
- case 'equal':
780
- // Exact array match (order matters in PostgreSQL)
781
- return {
782
- query: `json_data->'${flatJsonKey}'::jsonb = :${paramKey}::jsonb`,
783
- params: { [paramKey]: JSON.stringify(arr) },
784
- };
785
-
786
- case 'empty':
787
- return {
788
- query: `(json_data->>'${flatJsonKey}' IS NULL OR json_data->'${flatJsonKey}' = '[]'::jsonb)`,
789
- params: {},
790
- };
791
-
792
- case 'not_empty':
793
- return {
794
- query: `(json_data->>'${flatJsonKey}' IS NOT NULL AND json_data->'${flatJsonKey}' != '[]'::jsonb)`,
795
- params: {},
796
- };
797
-
798
- default:
799
- console.warn(`Unsupported multiselect operator: ${operator}`);
800
- return null;
801
- }
802
- }
803
-
804
- /**
805
- * Build year condition
806
- */
807
- private buildYearCondition(
808
- flatJsonKey: string,
809
- operator: string,
810
- value: any,
811
- paramKey: string,
812
- ): JsonbCondition | null {
813
- // Similar to number condition
814
- return this.buildNumberCondition(flatJsonKey, operator, value, paramKey);
815
- }
816
-
817
- /**
818
- * Get tab aggregation counts from JSONB
819
- */
820
- private async getJsonbTabCounts(
821
- entity_type: string,
822
- flatJsonKey: string,
823
- whereClauses: JsonbCondition[],
824
- ): Promise<Array<{ tab_value: string; tab_value_count: number }>> {
825
- // Build a query that groups by the tab field
826
- const qb = this.entityManager
827
- .getRepository(EntityJson)
828
- .createQueryBuilder('ej')
829
- .select(`json_data->>'${flatJsonKey}'`, 'tab_value')
830
- .addSelect('COUNT(*)', 'tab_value_count')
831
- .where('ej.entity_type = :entity_type', { entity_type })
832
- .groupBy(`json_data->>'${flatJsonKey}'`);
833
-
834
- // Apply the same filter conditions (but not the tab filter itself)
835
- whereClauses.forEach((condition) => {
836
- qb.andWhere(condition.query, condition.params);
837
- });
838
-
839
- const results = await qb.getRawMany();
840
-
841
- // Filter out NULL/undefined tab values and convert count to number
842
- return results
843
- .filter((r) => r.tab_value != null && r.tab_value !== '')
844
- .map((r) => ({
845
- tab_value: r.tab_value,
846
- tab_value_count: parseInt(r.tab_value_count, 10),
847
- }))
848
- .sort((a, b) => b.tab_value_count - a.tab_value_count); // Sort by count DESC
849
- }
850
-
851
- /**
852
- * Apply sorting on JSONB fields
853
- */
854
- private applyJsonbSorting(
855
- qb: SelectQueryBuilder<EntityJson>,
856
- sortby: SortConfig[],
857
- attributeMetaMap: Record<string, any>,
858
- ): void {
859
- sortby.forEach((sort, index) => {
860
- const meta = attributeMetaMap[sort.sortColum];
861
-
862
- if (!meta) {
863
- console.warn(`No metadata found for sort column: ${sort.sortColum}`);
864
- return;
865
- }
866
-
867
- const flatJsonKey =
868
- meta.flat_json_key || `${meta.mapped_entity_type}__${sort.sortColum}`;
869
- const direction = sort.sortType === 'ASC' ? 'ASC' : 'DESC';
870
-
871
- // Apply type-specific casting for proper sorting
872
- let sortExpression: string;
873
-
874
- switch (meta.data_type) {
875
- case 'number':
876
- case 'year':
877
- // Cast to numeric for number sorting
878
- sortExpression = `(json_data->>'${flatJsonKey}')::numeric`;
879
- break;
880
-
881
- case 'date':
882
- // Cast to bigint for date sorting (epoch ms)
883
- sortExpression = `(json_data->>'${flatJsonKey}')::bigint`;
884
- break;
885
-
886
- case 'text':
887
- case 'select':
888
- case 'radio':
889
- default:
890
- // Text sorting (already lowercase)
891
- sortExpression = `json_data->>'${flatJsonKey}'`;
892
- break;
893
- }
894
-
895
- // Add to ORDER BY clause
896
- if (index === 0) {
897
- qb.orderBy(sortExpression, direction);
898
- } else {
899
- qb.addOrderBy(sortExpression, direction);
900
- }
901
- });
902
- }
903
- }
1
+ import { Injectable, BadRequestException, Inject, forwardRef } from '@nestjs/common';
2
+ import { EntityManager, SelectQueryBuilder } from 'typeorm';
3
+ import { EntityMasterService } from 'src/module/meta/service/entity-master.service';
4
+ import { AttributeMasterService } from 'src/module/meta/service/attribute-master.service';
5
+ import { ResolverService } from 'src/module/meta/service/resolver.service';
6
+ import { LoggingService } from 'src/utils/service/loggingUtil.service';
7
+ import { ConfigService } from '@nestjs/config';
8
+ import {
9
+ FilterRequestDto,
10
+ FilterCondition,
11
+ SortConfig,
12
+ } from '../dto/filter-request.dto';
13
+ import { EntityJson } from 'src/module/entity_json/entity/entityJson.entity';
14
+ import { SavedFilterRepositoryService } from '../repository/saved-filter.repository';
15
+ import { EntityJSONService } from 'src/module/entity_json/service/entity_json.service';
16
+
17
+ /**
18
+ * JSONB query condition structure
19
+ */
20
+ interface JsonbCondition {
21
+ query: string;
22
+ params: Record<string, any>;
23
+ }
24
+
25
+ @Injectable()
26
+ export class FlatjsonFilterService {
27
+ constructor(
28
+ private readonly entityManager: EntityManager,
29
+ @Inject(forwardRef(() => EntityMasterService))
30
+ private readonly entityMasterService: EntityMasterService,
31
+ @Inject(forwardRef(() => AttributeMasterService))
32
+ private readonly attributeMasterService: AttributeMasterService,
33
+ @Inject(forwardRef(() => ResolverService))
34
+ private readonly resolverService: ResolverService,
35
+ @Inject(forwardRef(() => LoggingService))
36
+ private readonly loggingService: LoggingService,
37
+ private readonly configService: ConfigService,
38
+ @Inject(forwardRef(() => SavedFilterRepositoryService))
39
+ private readonly savedFilterRepositoryService: SavedFilterRepositoryService,
40
+ @Inject(forwardRef(() => EntityJSONService))
41
+ private readonly entityJsonService: EntityJSONService,
42
+ ) {}
43
+
44
+ /**
45
+ * Main filtering method - queries frm_entity_json table
46
+ */
47
+ async applyFlatjsonFilter(dto: FilterRequestDto): Promise<any> {
48
+ const {
49
+ entity_type,
50
+ quickFilter = [],
51
+ savedFilterCode,
52
+ attributeFilter = [],
53
+ sortby = [],
54
+ tabs,
55
+ page = 1,
56
+ size = 20,
57
+ loggedInUser,
58
+ } = dto;
59
+
60
+ await this.loggingService.log(
61
+ 'info',
62
+ 'FlatjsonFilterService',
63
+ 'applyFlatjsonFilter',
64
+ `Filtering entity: ${entity_type}`,
65
+ );
66
+
67
+ // Step 1: Load attribute metadata for this entity
68
+ let attributes =
69
+ await this.entityJsonService.getAttributeForFlatJSON(
70
+ entity_type,
71
+ loggedInUser,
72
+ 'filter_attribute',
73
+ );
74
+
75
+ if (!attributes?.length) {
76
+ attributes =
77
+ await this.attributeMasterService.findAttributesByMappedEntityType(
78
+ entity_type,
79
+ loggedInUser,
80
+ );
81
+ }
82
+
83
+ attributes = attributes ?? [];
84
+
85
+ const attributeMetaMap: Record<string, any> = {};
86
+ attributes.forEach((attr) => {
87
+ attributeMetaMap[attr.attribute_key] = attr;
88
+ });
89
+
90
+ await this.loggingService.log(
91
+ 'debug',
92
+ 'FlatjsonFilterService',
93
+ 'applyFlatjsonFilter',
94
+ `Loaded ${attributes.length} attributes`,
95
+ );
96
+
97
+ // Step 2: Merge filters (quickFilter + savedFilter + attributeFilter)
98
+ let allFilters: FilterCondition[] = [...quickFilter, ...attributeFilter];
99
+
100
+ if (savedFilterCode) {
101
+ const savedFilterDetails =
102
+ await this.savedFilterRepositoryService.getDetailsByCode(
103
+ savedFilterCode,
104
+ );
105
+ allFilters = [...allFilters, ...savedFilterDetails];
106
+ }
107
+
108
+ await this.loggingService.log(
109
+ 'debug',
110
+ 'FlatjsonFilterService',
111
+ 'applyFlatjsonFilter',
112
+ `Total filters: ${allFilters.length}`,
113
+ );
114
+
115
+ // Step 3: Build JSONB conditions
116
+ const jsonbConditions = this.buildJsonbConditions(
117
+ allFilters,
118
+ attributeMetaMap,
119
+ );
120
+
121
+ // Step 4: Build base query
122
+ const qb = this.entityManager
123
+ .getRepository(EntityJson)
124
+ .createQueryBuilder('ej')
125
+ .where('ej.entity_type = :entity_type', { entity_type })
126
+ .andWhere('ej.organization_id = :orgId', {
127
+ orgId: loggedInUser.organization_id,
128
+ });
129
+
130
+ // Step 5: Apply JSONB conditions
131
+ jsonbConditions.forEach((condition, index) => {
132
+ qb.andWhere(condition.query, condition.params);
133
+ });
134
+
135
+ // Step 6: Apply tab filter if provided
136
+ if (tabs?.columnName && tabs?.value) {
137
+ const tabMeta = attributeMetaMap[tabs.columnName];
138
+ if (tabMeta) {
139
+ const flatJsonKey =
140
+ tabMeta.attribute_key ||
141
+ `${entity_type}__${tabs.columnName}`;
142
+ qb.andWhere(`json_data->>'${flatJsonKey}' = :tabValue`, {
143
+ tabValue: tabs.value,
144
+ });
145
+ }
146
+ }
147
+
148
+ // Step 7: Get tab counts if needed (before pagination)
149
+ let tabCounts: any[] = [];
150
+ if (tabs?.columnName && !tabs?.value) {
151
+ const tabMeta = attributeMetaMap[tabs.columnName];
152
+ if (tabMeta) {
153
+ const flatJsonKey =
154
+ tabMeta.attribute_key ||
155
+ `${entity_type}__${tabs.columnName}`;
156
+ tabCounts = await this.getJsonbTabCounts(
157
+ entity_type,
158
+ flatJsonKey,
159
+ jsonbConditions,
160
+ );
161
+ }
162
+ }
163
+
164
+ // Step 8: Apply sorting
165
+ if (sortby && sortby.length > 0) {
166
+ this.applyJsonbSorting(qb, sortby, attributeMetaMap);
167
+ } else {
168
+ // Default sort by created_date DESC
169
+ qb.orderBy('ej.created_at', 'DESC');
170
+ }
171
+
172
+ // Step 9: Get total count (before pagination)
173
+ const totalCount = await qb.getCount();
174
+
175
+ // Step 10: Apply pagination
176
+ if (page && size) {
177
+ const skip = (page - 1) * size;
178
+ qb.skip(skip).take(size);
179
+ }
180
+
181
+ const results = await qb.getMany();
182
+
183
+ await this.loggingService.log(
184
+ 'info',
185
+ 'FlatjsonFilterService',
186
+ 'applyFlatjsonFilter',
187
+ `Found ${totalCount} total, returning ${results.length} records`,
188
+ );
189
+
190
+ // Map only json_data
191
+ const jsonDataResults = results.map((r) => r.json_data);
192
+
193
+ return {
194
+ data: jsonDataResults,
195
+ total: totalCount,
196
+ page: page || 1,
197
+ size: size || 20,
198
+ totalPages: size ? Math.ceil(totalCount / size) : 1,
199
+ tabCounts: tabCounts.length > 0 ? tabCounts : undefined,
200
+ };
201
+
202
+ }
203
+
204
+ /**
205
+ * Build JSONB where clauses from filter conditions
206
+ */
207
+ private buildJsonbConditions(
208
+ filters: FilterCondition[],
209
+ attributeMetaMap: Record<string, any>,
210
+ ): JsonbCondition[] {
211
+ const conditions: JsonbCondition[] = [];
212
+
213
+ for (const filter of filters) {
214
+ const meta = attributeMetaMap[filter.filter_attribute];
215
+ const condition = this.buildJsonbCondition(filter, meta);
216
+
217
+ if (condition) {
218
+ conditions.push(condition);
219
+ } else {
220
+ console.warn(
221
+ `Could not build condition for filter: ${filter.filter_attribute}`,
222
+ );
223
+ }
224
+ }
225
+
226
+ return conditions;
227
+ }
228
+
229
+ /**
230
+ * Resolve entity IDs to full entity data
231
+ */
232
+ private async resolveEntityData(
233
+ entityJsonRecords: EntityJson[],
234
+ entity_type: string,
235
+ loggedInUser: any,
236
+ ): Promise<any[]> {
237
+ if (entityJsonRecords.length === 0) return [];
238
+
239
+ const entityIds = entityJsonRecords.map((ej) => ej.entity_id);
240
+
241
+ try {
242
+ // Use resolver service to get full entity data
243
+ const fullEntities = await this.resolverService.getResolvedData(
244
+ entity_type,
245
+ entityIds,
246
+ loggedInUser,
247
+ );
248
+
249
+ // Return with json_data merged
250
+ return entityJsonRecords.map((ej) => {
251
+ const fullEntity = fullEntities.find((e) => e.id === ej.entity_id);
252
+ return {
253
+ ...fullEntity,
254
+ _flatjson: ej.json_data, // Include flatjson for debugging
255
+ };
256
+ });
257
+ } catch (error) {
258
+ console.error('Error resolving entity data:', error);
259
+ // Fallback: return just the entity IDs with json_data
260
+ return entityJsonRecords.map((ej) => ({
261
+ id: ej.entity_id,
262
+ entity_type: ej.entity_type,
263
+ ...ej.json_data,
264
+ }));
265
+ }
266
+ }
267
+
268
+ /**
269
+ * Build a single JSONB condition based on data type
270
+ */
271
+ private buildJsonbCondition(
272
+ filter: FilterCondition,
273
+ meta: any,
274
+ ): JsonbCondition | null {
275
+ if (!meta) return null;
276
+
277
+ const flatJsonKey =
278
+ meta.attribute_key ||
279
+ `${meta.mapped_entity_type}__${filter.filter_attribute}`;
280
+ const op = filter.filter_operator;
281
+ const val = filter.filter_value;
282
+ const key = `param_${filter.filter_attribute}_${Math.random().toString(36).substring(2, 8)}`;
283
+
284
+ switch (meta.data_type) {
285
+ case 'text':
286
+ return this.buildTextCondition(flatJsonKey, op, val, key);
287
+ case 'number':
288
+ return this.buildNumberCondition(flatJsonKey, op, val, key);
289
+ case 'date':
290
+ return this.buildDateCondition(flatJsonKey, op, val, key);
291
+ case 'select':
292
+ case 'radio':
293
+ return this.buildSelectCondition(flatJsonKey, op, val, key);
294
+ case 'multiselect':
295
+ case 'checkbox':
296
+ return this.buildMultiSelectCondition(flatJsonKey, op, val, key);
297
+ case 'year':
298
+ return this.buildYearCondition(flatJsonKey, op, val, key);
299
+ default:
300
+ return null;
301
+ }
302
+ }
303
+
304
+ /**
305
+ * Build text condition (JSONB ->> operator)
306
+ */
307
+ private buildTextCondition(
308
+ flatJsonKey: string,
309
+ operator: string,
310
+ value: any,
311
+ paramKey: string,
312
+ ): JsonbCondition | null {
313
+ // Text is already stored lowercase in flatjson
314
+ const lowerValue = value ? String(value).toLowerCase() : '';
315
+
316
+ switch (operator) {
317
+ case 'equal':
318
+ return {
319
+ query: `json_data->>'${flatJsonKey}' = :${paramKey}`,
320
+ params: { [paramKey]: lowerValue },
321
+ };
322
+
323
+ case 'not_equal':
324
+ return {
325
+ query: `json_data->>'${flatJsonKey}' != :${paramKey}`,
326
+ params: { [paramKey]: lowerValue },
327
+ };
328
+
329
+ case 'contains':
330
+ return {
331
+ query: `json_data->>'${flatJsonKey}' LIKE :${paramKey}`,
332
+ params: { [paramKey]: `%${lowerValue}%` },
333
+ };
334
+
335
+ case 'not_contains':
336
+ return {
337
+ query: `json_data->>'${flatJsonKey}' NOT LIKE :${paramKey}`,
338
+ params: { [paramKey]: `%${lowerValue}%` },
339
+ };
340
+
341
+ case 'starts_with':
342
+ return {
343
+ query: `json_data->>'${flatJsonKey}' LIKE :${paramKey}`,
344
+ params: { [paramKey]: `${lowerValue}%` },
345
+ };
346
+
347
+ case 'ends_with':
348
+ return {
349
+ query: `json_data->>'${flatJsonKey}' LIKE :${paramKey}`,
350
+ params: { [paramKey]: `%${lowerValue}` },
351
+ };
352
+
353
+ case 'empty':
354
+ return {
355
+ query: `(json_data->>'${flatJsonKey}' IS NULL OR json_data->>'${flatJsonKey}' = '')`,
356
+ params: {},
357
+ };
358
+
359
+ case 'not_empty':
360
+ return {
361
+ query: `(json_data->>'${flatJsonKey}' IS NOT NULL AND json_data->>'${flatJsonKey}' != '')`,
362
+ params: {},
363
+ };
364
+
365
+ default:
366
+ console.warn(`Unsupported text operator: ${operator}`);
367
+ return null;
368
+ }
369
+ }
370
+
371
+ /**
372
+ * Build number condition (JSONB ->> with ::numeric cast)
373
+ */
374
+ private buildNumberCondition(
375
+ flatJsonKey: string,
376
+ operator: string,
377
+ value: any,
378
+ paramKey: string,
379
+ ): JsonbCondition | null {
380
+ // Cast JSONB text to numeric for comparison
381
+ const jsonbField = `(json_data->>'${flatJsonKey}')::numeric`;
382
+
383
+ switch (operator) {
384
+ case 'equal':
385
+ return {
386
+ query: `${jsonbField} = :${paramKey}`,
387
+ params: { [paramKey]: Number(value) },
388
+ };
389
+
390
+ case 'not_equal':
391
+ return {
392
+ query: `${jsonbField} != :${paramKey}`,
393
+ params: { [paramKey]: Number(value) },
394
+ };
395
+
396
+ case 'greater_than':
397
+ return {
398
+ query: `${jsonbField} > :${paramKey}`,
399
+ params: { [paramKey]: Number(value) },
400
+ };
401
+
402
+ case 'less_than':
403
+ return {
404
+ query: `${jsonbField} < :${paramKey}`,
405
+ params: { [paramKey]: Number(value) },
406
+ };
407
+
408
+ case 'greater_than_equal_to':
409
+ return {
410
+ query: `${jsonbField} >= :${paramKey}`,
411
+ params: { [paramKey]: Number(value) },
412
+ };
413
+
414
+ case 'less_than_equal_to':
415
+ return {
416
+ query: `${jsonbField} <= :${paramKey}`,
417
+ params: { [paramKey]: Number(value) },
418
+ };
419
+
420
+ case 'between': {
421
+ // Value should be array [min, max] or string "min,max"
422
+ let range: number[];
423
+ if (typeof value === 'string') {
424
+ range = value.split(',').map((v) => Number(v.trim()));
425
+ } else if (Array.isArray(value)) {
426
+ range = value.map((v) => Number(v));
427
+ } else {
428
+ return null;
429
+ }
430
+
431
+ if (range.length !== 2) return null;
432
+
433
+ return {
434
+ query: `${jsonbField} BETWEEN :${paramKey}_min AND :${paramKey}_max`,
435
+ params: {
436
+ [`${paramKey}_min`]: range[0],
437
+ [`${paramKey}_max`]: range[1],
438
+ },
439
+ };
440
+ }
441
+
442
+ case 'empty':
443
+ return {
444
+ query: `json_data->>'${flatJsonKey}' IS NULL`,
445
+ params: {},
446
+ };
447
+
448
+ case 'not_empty':
449
+ return {
450
+ query: `json_data->>'${flatJsonKey}' IS NOT NULL`,
451
+ params: {},
452
+ };
453
+
454
+ default:
455
+ console.warn(`Unsupported number operator: ${operator}`);
456
+ return null;
457
+ }
458
+ }
459
+
460
+ /**
461
+ * Build date condition (JSONB ->> with ::bigint cast for epoch ms)
462
+ */
463
+ private buildDateCondition(
464
+ flatJsonKey: string,
465
+ operator: string,
466
+ value: any,
467
+ paramKey: string,
468
+ ): JsonbCondition | null {
469
+ // Dates stored as epoch milliseconds (bigint)
470
+ const jsonbField = `(json_data->>'${flatJsonKey}')::bigint`;
471
+
472
+ // Helper: Convert date string to epoch ms
473
+ const toEpochMs = (dateStr: string): number => {
474
+ return new Date(dateStr).getTime();
475
+ };
476
+
477
+ // Helper: Get date N days ago
478
+ const daysAgo = (days: number): number => {
479
+ const d = new Date();
480
+ d.setDate(d.getDate() - days);
481
+ return d.getTime();
482
+ };
483
+
484
+ // Helper: Get date N days from now
485
+ const daysFromNow = (days: number): number => {
486
+ const d = new Date();
487
+ d.setDate(d.getDate() + days);
488
+ return d.getTime();
489
+ };
490
+
491
+ // Helper: Subtract business days (skip weekends)
492
+ const subtractBusinessDays = (days: number): number => {
493
+ let d = new Date();
494
+ let count = 0;
495
+
496
+ while (count < days) {
497
+ d.setDate(d.getDate() - 1);
498
+ const day = d.getDay(); // 0=Sun, 6=Sat
499
+ if (day !== 0 && day !== 6) {
500
+ count++;
501
+ }
502
+ }
503
+
504
+ return d.getTime();
505
+ };
506
+
507
+ const numVal = Number(value);
508
+
509
+ switch (operator) {
510
+ // Basic comparisons
511
+ case 'equal':
512
+ case 'is':
513
+ return {
514
+ query: `${jsonbField} = :${paramKey}`,
515
+ params: { [paramKey]: toEpochMs(value) },
516
+ };
517
+
518
+ case 'before':
519
+ case 'is_before':
520
+ return {
521
+ query: `${jsonbField} < :${paramKey}`,
522
+ params: { [paramKey]: toEpochMs(value) },
523
+ };
524
+
525
+ case 'after':
526
+ case 'is_after':
527
+ return {
528
+ query: `${jsonbField} > :${paramKey}`,
529
+ params: { [paramKey]: toEpochMs(value) },
530
+ };
531
+
532
+ case 'is_on_or_before':
533
+ return {
534
+ query: `${jsonbField} <= :${paramKey}`,
535
+ params: { [paramKey]: toEpochMs(value) },
536
+ };
537
+
538
+ case 'is_on_or_after':
539
+ return {
540
+ query: `${jsonbField} >= :${paramKey}`,
541
+ params: { [paramKey]: toEpochMs(value) },
542
+ };
543
+
544
+ // Day offset logic
545
+ case 'is_day_before':
546
+ if (isNaN(numVal)) return null;
547
+ return {
548
+ query: `${jsonbField} <= :${paramKey}`,
549
+ params: { [paramKey]: daysAgo(numVal) },
550
+ };
551
+
552
+ case 'is_day_after':
553
+ if (isNaN(numVal)) return null;
554
+ return {
555
+ query: `${jsonbField} >= :${paramKey}`,
556
+ params: { [paramKey]: daysFromNow(numVal) },
557
+ };
558
+
559
+ // Business days (skip weekends)
560
+ case 'is_before_business_days':
561
+ if (isNaN(numVal)) return null;
562
+ return {
563
+ query: `${jsonbField} <= :${paramKey}`,
564
+ params: { [paramKey]: subtractBusinessDays(numVal) },
565
+ };
566
+
567
+ // Range operators
568
+ case 'between': {
569
+ let range: string[];
570
+ if (typeof value === 'string') {
571
+ range = value.split(',').map((v) => v.trim());
572
+ } else if (Array.isArray(value)) {
573
+ range = value;
574
+ } else {
575
+ return null;
576
+ }
577
+
578
+ if (range.length !== 2) return null;
579
+
580
+ return {
581
+ query: `${jsonbField} BETWEEN :${paramKey}_start AND :${paramKey}_end`,
582
+ params: {
583
+ [`${paramKey}_start`]: toEpochMs(range[0]),
584
+ [`${paramKey}_end`]: toEpochMs(range[1]),
585
+ },
586
+ };
587
+ }
588
+
589
+ case 'in_last_day':
590
+ if (isNaN(numVal)) return null;
591
+ return {
592
+ query: `${jsonbField} BETWEEN :${paramKey}_start AND :${paramKey}_end`,
593
+ params: {
594
+ [`${paramKey}_start`]: daysAgo(numVal),
595
+ [`${paramKey}_end`]: Date.now(),
596
+ },
597
+ };
598
+
599
+ case 'in_next_day':
600
+ if (isNaN(numVal)) return null;
601
+ return {
602
+ query: `${jsonbField} BETWEEN :${paramKey}_start AND :${paramKey}_end`,
603
+ params: {
604
+ [`${paramKey}_start`]: Date.now(),
605
+ [`${paramKey}_end`]: daysFromNow(numVal),
606
+ },
607
+ };
608
+
609
+ // Special cases
610
+ case 'today': {
611
+ const todayStart = new Date().setHours(0, 0, 0, 0);
612
+ const todayEnd = new Date().setHours(23, 59, 59, 999);
613
+ return {
614
+ query: `${jsonbField} BETWEEN :${paramKey}_start AND :${paramKey}_end`,
615
+ params: {
616
+ [`${paramKey}_start`]: todayStart,
617
+ [`${paramKey}_end`]: todayEnd,
618
+ },
619
+ };
620
+ }
621
+
622
+ case 'empty':
623
+ return {
624
+ query: `json_data->>'${flatJsonKey}' IS NULL`,
625
+ params: {},
626
+ };
627
+
628
+ case 'not_empty':
629
+ return {
630
+ query: `json_data->>'${flatJsonKey}' IS NOT NULL`,
631
+ params: {},
632
+ };
633
+
634
+ default:
635
+ console.warn(`Unsupported date operator: ${operator}`);
636
+ return null;
637
+ }
638
+ }
639
+
640
+ /**
641
+ * Build select condition (single value)
642
+ */
643
+ private buildSelectCondition(
644
+ flatJsonKey: string,
645
+ operator: string,
646
+ value: any,
647
+ paramKey: string,
648
+ ): JsonbCondition | null {
649
+ switch (operator) {
650
+ case 'equal':
651
+ if (Array.isArray(value)) {
652
+ // IN operator for multiple values
653
+ return {
654
+ query: `json_data->>'${flatJsonKey}' = ANY(:${paramKey})`,
655
+ params: { [paramKey]: value.map(String) },
656
+ };
657
+ }
658
+ return {
659
+ query: `json_data->>'${flatJsonKey}' = :${paramKey}`,
660
+ params: { [paramKey]: String(value) },
661
+ };
662
+
663
+ case 'not_equal':
664
+ if (Array.isArray(value)) {
665
+ // NOT IN operator
666
+ return {
667
+ query: `json_data->>'${flatJsonKey}' != ALL(:${paramKey})`,
668
+ params: { [paramKey]: value.map(String) },
669
+ };
670
+ }
671
+ return {
672
+ query: `json_data->>'${flatJsonKey}' != :${paramKey}`,
673
+ params: { [paramKey]: String(value) },
674
+ };
675
+
676
+ case 'in': {
677
+ const inValues = Array.isArray(value) ? value : [value];
678
+ return {
679
+ query: `json_data->>'${flatJsonKey}' = ANY(:${paramKey})`,
680
+ params: { [paramKey]: inValues.map(String) },
681
+ };
682
+ }
683
+
684
+ case 'not_in': {
685
+ const notInValues = Array.isArray(value) ? value : [value];
686
+ return {
687
+ query: `json_data->>'${flatJsonKey}' != ALL(:${paramKey})`,
688
+ params: { [paramKey]: notInValues.map(String) },
689
+ };
690
+ }
691
+
692
+ case 'empty':
693
+ return {
694
+ query: `json_data->>'${flatJsonKey}' IS NULL`,
695
+ params: {},
696
+ };
697
+
698
+ case 'not_empty':
699
+ return {
700
+ query: `json_data->>'${flatJsonKey}' IS NOT NULL`,
701
+ params: {},
702
+ };
703
+
704
+ default:
705
+ console.warn(`Unsupported select operator: ${operator}`);
706
+ return null;
707
+ }
708
+ }
709
+
710
+ /**
711
+ * Build multiselect condition (JSONB -> with ? operators for arrays)
712
+ */
713
+ private buildMultiSelectCondition(
714
+ flatJsonKey: string,
715
+ operator: string,
716
+ value: any,
717
+ paramKey: string,
718
+ ): JsonbCondition | null {
719
+ // Convert value to array if not already
720
+ let arr: string[];
721
+ if (Array.isArray(value)) {
722
+ arr = value.map(String);
723
+ } else if (typeof value === 'string') {
724
+ arr = value.split(',').map((v) => v.trim());
725
+ } else {
726
+ arr = [String(value)];
727
+ }
728
+
729
+ if (arr.length === 0) {
730
+ return { query: '1=1', params: {} }; // Always true for empty array
731
+ }
732
+
733
+ switch (operator) {
734
+ case 'contains':
735
+ case 'has':
736
+ // Check if JSON array contains this element
737
+ // json_data->'LEAD__languages' ? 'hindi'
738
+ if (arr.length === 1) {
739
+ return {
740
+ query: `json_data->'${flatJsonKey}' ? :${paramKey}`,
741
+ params: { [paramKey]: arr[0] },
742
+ };
743
+ }
744
+ // For multiple values, check if contains ANY
745
+ return {
746
+ query: `json_data->'${flatJsonKey}' ?| ARRAY[:...${paramKey}]::text[]`,
747
+ params: { [paramKey]: arr },
748
+ };
749
+
750
+ case 'contains_all':
751
+ // Check if JSON array contains ALL elements
752
+ // json_data->'LEAD__languages' ?& ARRAY['hindi', 'english']
753
+ return {
754
+ query: `json_data->'${flatJsonKey}' ?& ARRAY[:...${paramKey}]::text[]`,
755
+ params: { [paramKey]: arr },
756
+ };
757
+
758
+ case 'contains_any':
759
+ // Check if JSON array contains ANY element
760
+ // json_data->'LEAD__languages' ?| ARRAY['hindi', 'english']
761
+ return {
762
+ query: `json_data->'${flatJsonKey}' ?| ARRAY[:...${paramKey}]::text[]`,
763
+ params: { [paramKey]: arr },
764
+ };
765
+
766
+ case 'not_contains':
767
+ // Check if JSON array does NOT contain element
768
+ if (arr.length === 1) {
769
+ return {
770
+ query: `NOT (json_data->'${flatJsonKey}' ? :${paramKey})`,
771
+ params: { [paramKey]: arr[0] },
772
+ };
773
+ }
774
+ return {
775
+ query: `NOT (json_data->'${flatJsonKey}' ?| ARRAY[:...${paramKey}]::text[])`,
776
+ params: { [paramKey]: arr },
777
+ };
778
+
779
+ case 'equal':
780
+ // Exact array match (order matters in PostgreSQL)
781
+ return {
782
+ query: `json_data->'${flatJsonKey}'::jsonb = :${paramKey}::jsonb`,
783
+ params: { [paramKey]: JSON.stringify(arr) },
784
+ };
785
+
786
+ case 'empty':
787
+ return {
788
+ query: `(json_data->>'${flatJsonKey}' IS NULL OR json_data->'${flatJsonKey}' = '[]'::jsonb)`,
789
+ params: {},
790
+ };
791
+
792
+ case 'not_empty':
793
+ return {
794
+ query: `(json_data->>'${flatJsonKey}' IS NOT NULL AND json_data->'${flatJsonKey}' != '[]'::jsonb)`,
795
+ params: {},
796
+ };
797
+
798
+ default:
799
+ console.warn(`Unsupported multiselect operator: ${operator}`);
800
+ return null;
801
+ }
802
+ }
803
+
804
+ /**
805
+ * Build year condition
806
+ */
807
+ private buildYearCondition(
808
+ flatJsonKey: string,
809
+ operator: string,
810
+ value: any,
811
+ paramKey: string,
812
+ ): JsonbCondition | null {
813
+ // Similar to number condition
814
+ return this.buildNumberCondition(flatJsonKey, operator, value, paramKey);
815
+ }
816
+
817
+ /**
818
+ * Get tab aggregation counts from JSONB
819
+ */
820
+ private async getJsonbTabCounts(
821
+ entity_type: string,
822
+ flatJsonKey: string,
823
+ whereClauses: JsonbCondition[],
824
+ ): Promise<Array<{ tab_value: string; tab_value_count: number }>> {
825
+ // Build a query that groups by the tab field
826
+ const qb = this.entityManager
827
+ .getRepository(EntityJson)
828
+ .createQueryBuilder('ej')
829
+ .select(`json_data->>'${flatJsonKey}'`, 'tab_value')
830
+ .addSelect('COUNT(*)', 'tab_value_count')
831
+ .where('ej.entity_type = :entity_type', { entity_type })
832
+ .groupBy(`json_data->>'${flatJsonKey}'`);
833
+
834
+ // Apply the same filter conditions (but not the tab filter itself)
835
+ whereClauses.forEach((condition) => {
836
+ qb.andWhere(condition.query, condition.params);
837
+ });
838
+
839
+ const results = await qb.getRawMany();
840
+
841
+ // Filter out NULL/undefined tab values and convert count to number
842
+ return results
843
+ .filter((r) => r.tab_value != null && r.tab_value !== '')
844
+ .map((r) => ({
845
+ tab_value: r.tab_value,
846
+ tab_value_count: parseInt(r.tab_value_count, 10),
847
+ }))
848
+ .sort((a, b) => b.tab_value_count - a.tab_value_count); // Sort by count DESC
849
+ }
850
+
851
+ /**
852
+ * Apply sorting on JSONB fields
853
+ */
854
+ private applyJsonbSorting(
855
+ qb: SelectQueryBuilder<EntityJson>,
856
+ sortby: SortConfig[],
857
+ attributeMetaMap: Record<string, any>,
858
+ ): void {
859
+ sortby.forEach((sort, index) => {
860
+ const meta = attributeMetaMap[sort.sortColum];
861
+
862
+ if (!meta) {
863
+ console.warn(`No metadata found for sort column: ${sort.sortColum}`);
864
+ return;
865
+ }
866
+
867
+ const flatJsonKey =
868
+ meta.attribute_key || `${meta.mapped_entity_type}__${sort.sortColum}`;
869
+ const direction = sort.sortType === 'ASC' ? 'ASC' : 'DESC';
870
+
871
+ // Apply type-specific casting for proper sorting
872
+ let sortExpression: string;
873
+
874
+ switch (meta.data_type) {
875
+ case 'number':
876
+ case 'year':
877
+ // Cast to numeric for number sorting
878
+ sortExpression = `(json_data->>'${flatJsonKey}')::numeric`;
879
+ break;
880
+
881
+ case 'date':
882
+ // Cast to bigint for date sorting (epoch ms)
883
+ sortExpression = `(json_data->>'${flatJsonKey}')::bigint`;
884
+ break;
885
+
886
+ case 'text':
887
+ case 'select':
888
+ case 'radio':
889
+ default:
890
+ // Text sorting (already lowercase)
891
+ sortExpression = `json_data->>'${flatJsonKey}'`;
892
+ break;
893
+ }
894
+
895
+ // Add to ORDER BY clause
896
+ if (index === 0) {
897
+ qb.orderBy(sortExpression, direction);
898
+ } else {
899
+ qb.addOrderBy(sortExpression, direction);
900
+ }
901
+ });
902
+ }
903
+ }