rez_core 5.0.182 → 6.5.0

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 (426) 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 +6 -0
  10. package/.idea/misc.xml +6 -0
  11. package/.idea/modules.xml +7 -7
  12. package/.idea/prettier.xml +6 -0
  13. package/.idea/vcs.xml +5 -5
  14. package/.prettierrc +3 -3
  15. package/README.md +99 -99
  16. package/dist/module/auth/guards/role.guard.js +3 -3
  17. package/dist/module/enterprise/controller/enterprise.controller.d.ts +12 -0
  18. package/dist/module/enterprise/controller/enterprise.controller.js +57 -0
  19. package/dist/module/enterprise/controller/enterprise.controller.js.map +1 -0
  20. package/dist/module/enterprise/controller/organization.controller.d.ts +11 -1
  21. package/dist/module/enterprise/controller/organization.controller.js +62 -2
  22. package/dist/module/enterprise/controller/organization.controller.js.map +1 -1
  23. package/dist/module/enterprise/enterprise.module.js +2 -1
  24. package/dist/module/enterprise/enterprise.module.js.map +1 -1
  25. package/dist/module/enterprise/entity/enterprise.entity.d.ts +1 -3
  26. package/dist/module/enterprise/entity/enterprise.entity.js +4 -12
  27. package/dist/module/enterprise/entity/enterprise.entity.js.map +1 -1
  28. package/dist/module/enterprise/entity/organization.entity.d.ts +1 -17
  29. package/dist/module/enterprise/entity/organization.entity.js +4 -72
  30. package/dist/module/enterprise/entity/organization.entity.js.map +1 -1
  31. package/dist/module/enterprise/repository/enterprise.repository.d.ts +2 -2
  32. package/dist/module/enterprise/repository/enterprise.repository.js +9 -4
  33. package/dist/module/enterprise/repository/enterprise.repository.js.map +1 -1
  34. package/dist/module/enterprise/service/enterprise.service.d.ts +2 -2
  35. package/dist/module/enterprise/service/enterprise.service.js +4 -4
  36. package/dist/module/enterprise/service/enterprise.service.js.map +1 -1
  37. package/dist/module/enterprise/service/organization.service.d.ts +2 -2
  38. package/dist/module/enterprise/service/organization.service.js +97 -20
  39. package/dist/module/enterprise/service/organization.service.js.map +1 -1
  40. package/dist/module/filter/service/filter.service.js +19 -19
  41. package/dist/module/integration/examples/usage.example.js +9 -9
  42. package/dist/module/meta/repository/attribute-master.repository.js +8 -8
  43. package/dist/module/meta/service/entity-dynamic.service.js +16 -16
  44. package/dist/module/meta/service/media-data.service.js +6 -6
  45. package/dist/module/meta/service/resolver.service.js +11 -11
  46. package/dist/module/module/repository/menu.repository.js +4 -4
  47. package/dist/module/notification/service/notification.service.js +6 -6
  48. package/dist/module/user/controller/login.controller.js +18 -18
  49. package/dist/module/workflow/repository/action.repository.js +2 -2
  50. package/dist/module/workflow/repository/stage.repository.js +8 -8
  51. package/dist/module/workflow/service/action-template-mapping.service.js +2 -2
  52. package/dist/module/workflow/service/action.service.js +5 -5
  53. package/dist/module/workflow/service/entity-modification.service.js +2 -2
  54. package/dist/module/workflow/service/task.service.js +8 -8
  55. package/dist/module/workflow-automation/service/schedule-handler.service.js +9 -9
  56. package/dist/table.config.d.ts +1 -2
  57. package/dist/tsconfig.build.tsbuildinfo +1 -1
  58. package/dist/utils/service/reflection-helper.service.js +2 -2
  59. package/docs/modules/event-driven-integration-design.md +91 -91
  60. package/docs/modules/integration.md +250 -250
  61. package/eslint.config.mjs +34 -34
  62. package/nest-cli.json +14 -14
  63. package/package.json +125 -125
  64. package/server.log +850 -0
  65. package/src/app.controller.ts +12 -12
  66. package/src/app.module.ts +68 -68
  67. package/src/app.service.ts +8 -8
  68. package/src/config/bull.config.ts +69 -69
  69. package/src/config/config.module.ts +17 -17
  70. package/src/config/database.config.ts +48 -48
  71. package/src/constant/global.constant.ts +67 -67
  72. package/src/core.module.ts +94 -94
  73. package/src/decorators/roles.decorator.ts +7 -7
  74. package/src/dtos/response.dto.ts +6 -6
  75. package/src/dtos/response.ts +5 -5
  76. package/src/index.ts +1 -1
  77. package/src/module/auth/auth.module.ts +49 -49
  78. package/src/module/auth/controller/auth.controller.ts +28 -28
  79. package/src/module/auth/guards/google-auth.guard.ts +9 -9
  80. package/src/module/auth/guards/jwt.guard.ts +22 -22
  81. package/src/module/auth/guards/role.guard.ts +68 -68
  82. package/src/module/auth/services/auth.service.ts +56 -56
  83. package/src/module/auth/services/jwt.service.ts +11 -11
  84. package/src/module/auth/strategies/google.strategy.ts +54 -54
  85. package/src/module/auth/strategies/jwt.strategy.ts +58 -58
  86. package/src/module/auth/strategies/local.strategy.ts +13 -13
  87. package/src/module/dashboard/controller/dashboard.controller.ts +38 -38
  88. package/src/module/dashboard/dashboard.module.ts +21 -21
  89. package/src/module/dashboard/entity/dashboard_page_data.entity.ts +27 -27
  90. package/src/module/dashboard/entity/widget_master.entity.ts +18 -18
  91. package/src/module/dashboard/repository/dashboard.repository.ts +49 -49
  92. package/src/module/dashboard/service/dashboard.service.ts +72 -72
  93. package/src/module/enterprise/controller/enterprise.controller.ts +40 -0
  94. package/src/module/enterprise/controller/organization.controller.ts +93 -36
  95. package/src/module/enterprise/enterprise.module.ts +46 -45
  96. package/src/module/enterprise/entity/enterprise.entity.ts +31 -37
  97. package/src/module/enterprise/entity/organization-app-mapping.entity.ts +13 -13
  98. package/src/module/enterprise/entity/organization.entity.ts +40 -92
  99. package/src/module/enterprise/repository/enterprise.repository.ts +39 -31
  100. package/src/module/enterprise/repository/organization.repository.ts +26 -26
  101. package/src/module/enterprise/repository/school.repository.ts +272 -272
  102. package/src/module/enterprise/service/brand.service.ts +5 -5
  103. package/src/module/enterprise/service/enterprise.service.ts +22 -16
  104. package/src/module/enterprise/service/organization-app-mapping.service.ts +4 -4
  105. package/src/module/enterprise/service/organization.service.ts +262 -145
  106. package/src/module/entity_json/controller/entity_json.controller.ts +47 -47
  107. package/src/module/entity_json/entity/entityJson.entity.ts +39 -39
  108. package/src/module/entity_json/entity_json.module.ts +18 -18
  109. package/src/module/entity_json/service/entityJson.repository.ts +37 -37
  110. package/src/module/entity_json/service/entity_json.service.ts +241 -241
  111. package/src/module/export/controller/export.controller.ts +83 -83
  112. package/src/module/export/export.module.ts +14 -14
  113. package/src/module/export/service/export.service.ts +105 -105
  114. package/src/module/filter/controller/filter.controller.ts +87 -87
  115. package/src/module/filter/dto/filter-request.dto.ts +39 -39
  116. package/src/module/filter/entity/saved-filter-detail.entity.ts +41 -41
  117. package/src/module/filter/entity/saved-filter-master.entity.ts +35 -35
  118. package/src/module/filter/filter.module.ts +33 -33
  119. package/src/module/filter/repository/saved-filter.repository.ts +247 -247
  120. package/src/module/filter/repository/saved.filter-detail.repository.ts +19 -19
  121. package/src/module/filter/service/filter-evaluator.service.ts +82 -82
  122. package/src/module/filter/service/filter.service.ts +1316 -1316
  123. package/src/module/filter/service/saved-filter.service.ts +164 -164
  124. package/src/module/ics/controller/ics.controller.ts +21 -21
  125. package/src/module/ics/dto/ics.dto.ts +55 -55
  126. package/src/module/ics/ics.module.ts +13 -13
  127. package/src/module/ics/service/ics.service.ts +57 -57
  128. package/src/module/integration/controller/calender-event.controller.ts +31 -31
  129. package/src/module/integration/controller/integration.controller.ts +662 -662
  130. package/src/module/integration/controller/wrapper.controller.ts +37 -37
  131. package/src/module/integration/dto/create-config.dto.ts +526 -526
  132. package/src/module/integration/entity/integration-config.entity.ts +112 -112
  133. package/src/module/integration/entity/integration-entity-mapper.entity.ts +14 -14
  134. package/src/module/integration/entity/integration-source.entity.ts +17 -17
  135. package/src/module/integration/entity/user-integration.entity.ts +71 -71
  136. package/src/module/integration/examples/usage.example.ts +338 -338
  137. package/src/module/integration/factories/base.factory.ts +7 -7
  138. package/src/module/integration/factories/email.factory.ts +49 -49
  139. package/src/module/integration/factories/integration.factory.ts +121 -121
  140. package/src/module/integration/factories/sms.factory.ts +51 -51
  141. package/src/module/integration/factories/telephone.factory.ts +41 -41
  142. package/src/module/integration/factories/whatsapp.factory.ts +56 -56
  143. package/src/module/integration/integration.module.ts +110 -110
  144. package/src/module/integration/service/calendar-event.service.ts +118 -118
  145. package/src/module/integration/service/integration-entity-mapper.service.ts +17 -17
  146. package/src/module/integration/service/integration-queue.service.ts +229 -229
  147. package/src/module/integration/service/integration.service.ts +2634 -2634
  148. package/src/module/integration/service/oauth.service.ts +224 -224
  149. package/src/module/integration/service/wrapper.service.ts +753 -753
  150. package/src/module/integration/strategies/email/gmail-api.strategy.ts +280 -280
  151. package/src/module/integration/strategies/email/outlook-api.strategy.ts +44 -44
  152. package/src/module/integration/strategies/email/outlook.strategy.ts +64 -64
  153. package/src/module/integration/strategies/email/sendgrid-api.strategy.ts +260 -260
  154. package/src/module/integration/strategies/integration.strategy.ts +97 -97
  155. package/src/module/integration/strategies/sms/gupshup-sms.strategy.ts +146 -146
  156. package/src/module/integration/strategies/sms/msg91-sms.strategy.ts +164 -164
  157. package/src/module/integration/strategies/sms/tubelight-sms.strategy.ts +163 -163
  158. package/src/module/integration/strategies/telephone/ozonetel-voice.strategy.ts +238 -238
  159. package/src/module/integration/strategies/telephone/tubelight-voice.strategy.ts +210 -210
  160. package/src/module/integration/strategies/whatsapp/gupshup-whatsapp.strategy.ts +359 -359
  161. package/src/module/integration/strategies/whatsapp/tubelight-whatsapp.strategy.ts +372 -372
  162. package/src/module/integration/strategies/whatsapp/whatsapp-cloud.strategy.ts +403 -403
  163. package/src/module/integration/strategies/whatsapp/whatsapp.strategy.ts +57 -57
  164. package/src/module/layout/controller/layout.controller.ts +47 -47
  165. package/src/module/layout/entity/header-items.entity.ts +28 -28
  166. package/src/module/layout/entity/header-section.entity.ts +19 -19
  167. package/src/module/layout/layout.module.ts +21 -21
  168. package/src/module/layout/repository/header-items.repository.ts +18 -18
  169. package/src/module/layout/repository/header-section.repository.ts +22 -22
  170. package/src/module/layout/service/header-section.service.ts +25 -25
  171. package/src/module/layout_preference/controller/layout_preference.controller.ts +76 -76
  172. package/src/module/layout_preference/entity/layout_preference.entity.ts +28 -28
  173. package/src/module/layout_preference/layout_preference.module.ts +22 -22
  174. package/src/module/layout_preference/repository/layout_preference.repository.ts +65 -65
  175. package/src/module/layout_preference/service/layout_preference.service.ts +191 -191
  176. package/src/module/lead/controller/lead.controller.ts +30 -30
  177. package/src/module/lead/lead.module.ts +14 -14
  178. package/src/module/lead/repository/lead.repository.ts +41 -41
  179. package/src/module/lead/service/lead.service.ts +54 -54
  180. package/src/module/linked_attributes/controller/linked_attributes.controller.ts +37 -37
  181. package/src/module/linked_attributes/entity/linked_attribute.entity.ts +51 -51
  182. package/src/module/linked_attributes/linked_attributes.module.ts +16 -16
  183. package/src/module/linked_attributes/repository/linked_attribute.repository.ts +12 -12
  184. package/src/module/linked_attributes/service/linked_attributes.service.ts +73 -73
  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 +84 -84
  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 +135 -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 +85 -85
  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 +32 -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 +132 -132
  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 +824 -824
  251. package/src/module/meta/service/entity-master.service.ts +171 -171
  252. package/src/module/meta/service/entity-realation-data.service.ts +9 -9
  253. package/src/module/meta/service/entity-relation.service.ts +74 -74
  254. package/src/module/meta/service/entity-service-impl.service.ts +388 -388
  255. package/src/module/meta/service/entity-table-column.service.ts +26 -26
  256. package/src/module/meta/service/entity-table.service.ts +157 -157
  257. package/src/module/meta/service/entity-validation.service.ts +188 -188
  258. package/src/module/meta/service/entity.service.ts +49 -49
  259. package/src/module/meta/service/field-group.service.ts +103 -103
  260. package/src/module/meta/service/media-data.service.ts +591 -591
  261. package/src/module/meta/service/populate-meta.service.ts +222 -222
  262. package/src/module/meta/service/preference.service.ts +16 -16
  263. package/src/module/meta/service/resolver.service.ts +280 -280
  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 +69 -69
  288. package/src/module/notification/repository/otp.repository.ts +27 -27
  289. package/src/module/notification/service/email.service.ts +127 -127
  290. package/src/module/notification/service/notification.service.ts +164 -164
  291. package/src/module/notification/service/otp.service.ts +133 -133
  292. package/src/module/third-party-module/entity/third-party-api-registry.entity.ts +52 -52
  293. package/src/module/third-party-module/repository/third-party-api-registry.repository.ts +20 -20
  294. package/src/module/third-party-module/service/api-registry.service.ts +13 -13
  295. package/src/module/third-party-module/third-party.module.ts +12 -12
  296. package/src/module/user/controller/login.controller.ts +198 -198
  297. package/src/module/user/controller/user.controller.ts +40 -40
  298. package/src/module/user/dto/create-user.dto.ts +62 -62
  299. package/src/module/user/dto/update-user.dto.ts +4 -4
  300. package/src/module/user/entity/role.entity.ts +33 -33
  301. package/src/module/user/entity/user-role-mapping.entity.ts +38 -38
  302. package/src/module/user/entity/user-session.entity.ts +73 -73
  303. package/src/module/user/entity/user.entity.ts +62 -62
  304. package/src/module/user/repository/role.repository.ts +96 -96
  305. package/src/module/user/repository/user-role-mapping.repository.ts +126 -126
  306. package/src/module/user/repository/user.repository.ts +50 -50
  307. package/src/module/user/repository/userSession.repository.ts +33 -33
  308. package/src/module/user/service/login.service.ts +326 -326
  309. package/src/module/user/service/role.service.ts +197 -197
  310. package/src/module/user/service/user-role-mapping.service.ts +98 -98
  311. package/src/module/user/service/user-session.service.ts +200 -200
  312. package/src/module/user/service/user.service.ts +368 -368
  313. package/src/module/user/user.module.ts +65 -65
  314. package/src/module/workflow/controller/action-category.controller.ts +54 -54
  315. package/src/module/workflow/controller/action-resource-mapping.controller.ts +23 -23
  316. package/src/module/workflow/controller/action-template-mapping.controller.ts +35 -35
  317. package/src/module/workflow/controller/action.controller.ts +111 -111
  318. package/src/module/workflow/controller/activity-log.controller.ts +55 -55
  319. package/src/module/workflow/controller/comm-template.controller.ts +43 -43
  320. package/src/module/workflow/controller/entity-modification.controller.ts +35 -35
  321. package/src/module/workflow/controller/form-master.controller.ts +43 -43
  322. package/src/module/workflow/controller/stage-group.controller.ts +49 -49
  323. package/src/module/workflow/controller/stage.controller.ts +51 -51
  324. package/src/module/workflow/controller/task.controller.ts +77 -77
  325. package/src/module/workflow/controller/workflow-list-master.controller.ts +44 -44
  326. package/src/module/workflow/controller/workflow-meta.controller.ts +80 -80
  327. package/src/module/workflow/controller/workflow.controller.ts +67 -67
  328. package/src/module/workflow/entity/action-category.entity.ts +38 -38
  329. package/src/module/workflow/entity/action-data.entity.ts +55 -55
  330. package/src/module/workflow/entity/action-resources-mapping.entity.ts +29 -29
  331. package/src/module/workflow/entity/action-template-mapping.entity.ts +17 -17
  332. package/src/module/workflow/entity/action.entity.ts +53 -53
  333. package/src/module/workflow/entity/activity-log.entity.ts +43 -43
  334. package/src/module/workflow/entity/comm-template.entity.ts +43 -43
  335. package/src/module/workflow/entity/entity-modification.entity.ts +38 -38
  336. package/src/module/workflow/entity/form.entity.ts +25 -25
  337. package/src/module/workflow/entity/stage-action-mapping.entity.ts +17 -17
  338. package/src/module/workflow/entity/stage-group.entity.ts +23 -23
  339. package/src/module/workflow/entity/stage-movement-data.entity.ts +38 -38
  340. package/src/module/workflow/entity/stage.entity.ts +20 -20
  341. package/src/module/workflow/entity/task-data.entity.ts +88 -88
  342. package/src/module/workflow/entity/template-attach-mapper.entity.ts +30 -30
  343. package/src/module/workflow/entity/workflow-data.entity.ts +11 -11
  344. package/src/module/workflow/entity/workflow-level-mapping.entity.ts +18 -18
  345. package/src/module/workflow/entity/workflow.entity.ts +20 -20
  346. package/src/module/workflow/repository/action-category.repository.ts +79 -79
  347. package/src/module/workflow/repository/action-data.repository.ts +347 -347
  348. package/src/module/workflow/repository/action.repository.ts +339 -339
  349. package/src/module/workflow/repository/activity-log.repository.ts +148 -148
  350. package/src/module/workflow/repository/comm-template.repository.ts +157 -157
  351. package/src/module/workflow/repository/form-master.repository.ts +50 -50
  352. package/src/module/workflow/repository/stage-group.repository.ts +186 -186
  353. package/src/module/workflow/repository/stage-movement.repository.ts +217 -217
  354. package/src/module/workflow/repository/stage.repository.ts +160 -160
  355. package/src/module/workflow/repository/task.repository.ts +154 -154
  356. package/src/module/workflow/repository/workflow.repository.ts +42 -42
  357. package/src/module/workflow/service/action-category.service.ts +33 -33
  358. package/src/module/workflow/service/action-data.service.ts +62 -62
  359. package/src/module/workflow/service/action-resources-mapping.service.ts +10 -10
  360. package/src/module/workflow/service/action-template-mapping.service.ts +137 -137
  361. package/src/module/workflow/service/action.service.ts +302 -302
  362. package/src/module/workflow/service/activity-log.service.ts +107 -107
  363. package/src/module/workflow/service/comm-template.service.ts +181 -181
  364. package/src/module/workflow/service/entity-modification.service.ts +61 -61
  365. package/src/module/workflow/service/form-master.service.ts +35 -35
  366. package/src/module/workflow/service/populate-workflow.service.ts +320 -320
  367. package/src/module/workflow/service/stage-action-mapping.service.ts +5 -5
  368. package/src/module/workflow/service/stage-group.service.ts +325 -325
  369. package/src/module/workflow/service/stage.service.ts +197 -197
  370. package/src/module/workflow/service/task.service.ts +551 -551
  371. package/src/module/workflow/service/workflow-list-master.service.ts +68 -68
  372. package/src/module/workflow/service/workflow-meta.service.ts +640 -640
  373. package/src/module/workflow/service/workflow.service.ts +213 -213
  374. package/src/module/workflow/workflow.module.ts +180 -180
  375. package/src/module/workflow-automation/SCHEDULING_GUIDE.md +145 -145
  376. package/src/module/workflow-automation/controller/workflow-automation.controller.ts +43 -43
  377. package/src/module/workflow-automation/entity/workflow-automation-action.entity.ts +26 -26
  378. package/src/module/workflow-automation/entity/workflow-automation.entity.ts +40 -40
  379. package/src/module/workflow-automation/interface/action.decorator.ts +7 -7
  380. package/src/module/workflow-automation/interface/action.interface.ts +5 -5
  381. package/src/module/workflow-automation/service/action-registery.service.ts +35 -35
  382. package/src/module/workflow-automation/service/schedule-handler.service.ts +168 -168
  383. package/src/module/workflow-automation/service/workflow-automation-engine.service.ts +219 -219
  384. package/src/module/workflow-automation/service/workflow-automation.service.ts +476 -476
  385. package/src/module/workflow-automation/workflow-automation.module.ts +54 -54
  386. package/src/module/workflow-schedule/INSTALLATION.md +244 -244
  387. package/src/module/workflow-schedule/MULTI_PROJECT_GUIDE.md +196 -196
  388. package/src/module/workflow-schedule/README.md +422 -422
  389. package/src/module/workflow-schedule/constants/schedule.constants.ts +48 -48
  390. package/src/module/workflow-schedule/controller/workflow-schedule.controller.ts +253 -253
  391. package/src/module/workflow-schedule/docs/CLAUDE_CODE_GUIDE.md +510 -510
  392. package/src/module/workflow-schedule/docs/CLAUDE_CODE_PROMPT.md +362 -362
  393. package/src/module/workflow-schedule/docs/RUN_CLAUDE_CODE.sh +68 -68
  394. package/src/module/workflow-schedule/dto/create-schedule.dto.ts +147 -147
  395. package/src/module/workflow-schedule/dto/get-execution-logs.dto.ts +119 -119
  396. package/src/module/workflow-schedule/dto/update-schedule.dto.ts +96 -96
  397. package/src/module/workflow-schedule/entities/scheduled-workflow.entity.ts +148 -148
  398. package/src/module/workflow-schedule/entities/workflow-execution-log.entity.ts +154 -154
  399. package/src/module/workflow-schedule/interfaces/schedule-job-data.interface.ts +53 -53
  400. package/src/module/workflow-schedule/interfaces/workflow-schedule-options.interface.ts +12 -12
  401. package/src/module/workflow-schedule/processors/schedule.processor.ts +620 -620
  402. package/src/module/workflow-schedule/service/workflow-schedule.service.ts +597 -597
  403. package/src/module/workflow-schedule/workflow-schedule.module.ts +67 -67
  404. package/src/resources/dev.properties.yaml +31 -31
  405. package/src/resources/local.properties.yaml +27 -27
  406. package/src/resources/properties.module.ts +12 -12
  407. package/src/resources/properties.yaml.ts +11 -11
  408. package/src/resources/uat.properties.yaml +31 -31
  409. package/src/table.config.ts +133 -133
  410. package/src/utils/dto/excel-data.dto.ts +14 -14
  411. package/src/utils/dto/excelsheet-data.dto.ts +5 -5
  412. package/src/utils/service/base64util.service.ts +18 -18
  413. package/src/utils/service/clockIDGenUtil.service.ts +21 -21
  414. package/src/utils/service/codeGenerator.service.ts +22 -22
  415. package/src/utils/service/dateUtil.service.ts +17 -17
  416. package/src/utils/service/encryptUtil.service.ts +97 -97
  417. package/src/utils/service/excel-helper.service.ts +72 -72
  418. package/src/utils/service/excelUtil.service.ts +15 -15
  419. package/src/utils/service/file-util.service.ts +11 -11
  420. package/src/utils/service/json-util.service.ts +23 -23
  421. package/src/utils/service/loggingUtil.service.ts +88 -88
  422. package/src/utils/service/reflection-helper.service.ts +62 -62
  423. package/src/utils/service/wbsCodeGen.service.ts +8 -8
  424. package/src/utils/utils.module.ts +27 -27
  425. package/tsconfig.build.json +4 -4
  426. package/tsconfig.json +24 -24
@@ -1,597 +1,597 @@
1
- import { BadRequestException, Injectable, Logger, NotFoundException } from '@nestjs/common';
2
- import { InjectRepository } from '@nestjs/typeorm';
3
- import { Between, In, Repository } from 'typeorm';
4
- import { InjectQueue } from '@nestjs/bull';
5
- import { Queue } from 'bull';
6
- import * as cronParser from 'cron-parser';
7
-
8
- import { ScheduledWorkflow } from '../entities/scheduled-workflow.entity';
9
- import { WorkflowExecutionLog } from '../entities/workflow-execution-log.entity';
10
- import { CreateScheduleDto } from '../dto/create-schedule.dto';
11
- import { UpdateScheduleDto } from '../dto/update-schedule.dto';
12
- import { GetExecutionLogsDto } from '../dto/get-execution-logs.dto';
13
- import { UserData } from '../../user/entity/user.entity';
14
- import {
15
- DEFAULT_BACKOFF_MULTIPLIER,
16
- DEFAULT_MAX_RETRIES,
17
- DEFAULT_RETRY_DELAY,
18
- DEFAULT_TIMEZONE,
19
- EXECUTE_SCHEDULED_WORKFLOW_JOB,
20
- SCHEDULE_STATUS_ACTIVE,
21
- SCHEDULE_STATUS_INACTIVE,
22
- SCHEDULE_STATUS_PAUSED,
23
- WORKFLOW_SCHEDULE_QUEUE,
24
- } from '../constants/schedule.constants';
25
- import { ScheduleJobData } from '../interfaces/schedule-job-data.interface';
26
-
27
- /**
28
- * Workflow Schedule Service
29
- * Manages scheduled workflows and their execution
30
- */
31
- @Injectable()
32
- export class WorkflowScheduleService {
33
- private readonly logger = new Logger(WorkflowScheduleService.name);
34
-
35
- constructor(
36
- @InjectRepository(ScheduledWorkflow)
37
- private readonly scheduledWorkflowRepository: Repository<ScheduledWorkflow>,
38
- @InjectRepository(WorkflowExecutionLog)
39
- private readonly executionLogRepository: Repository<WorkflowExecutionLog>,
40
- @InjectQueue(WORKFLOW_SCHEDULE_QUEUE)
41
- private readonly scheduleQueue: Queue<ScheduleJobData>,
42
- ) {
43
- }
44
-
45
- /**
46
- * 1. Create a new scheduled workflow
47
- */
48
- async createSchedule(
49
- createScheduleDto: CreateScheduleDto,
50
- loggedInUser: UserData,
51
- ): Promise<ScheduledWorkflow> {
52
- this.logger.log(
53
- `Creating schedule for workflow ${createScheduleDto.workflow_id} by user ${loggedInUser.id}`,
54
- );
55
-
56
- try {
57
- // Validate cron expression
58
- this.validateCronExpression(createScheduleDto.cron_expression, createScheduleDto.timezone);
59
-
60
- // Calculate next execution time
61
- const nextExecutionAt = this.calculateNextExecution(
62
- createScheduleDto.cron_expression,
63
- createScheduleDto.timezone || DEFAULT_TIMEZONE,
64
- );
65
-
66
- // Create scheduled workflow entity
67
- const schedule = this.scheduledWorkflowRepository.create({
68
- workflow_id: createScheduleDto.workflow_id,
69
- workflow_name: createScheduleDto.workflow_name,
70
- name: createScheduleDto.name,
71
- description: createScheduleDto.description,
72
- cron_expression: createScheduleDto.cron_expression,
73
- timezone: createScheduleDto.timezone || DEFAULT_TIMEZONE,
74
- start_date: createScheduleDto.start_date ? new Date(createScheduleDto.start_date) : null,
75
- end_date: createScheduleDto.end_date ? new Date(createScheduleDto.end_date) : null,
76
- max_executions: createScheduleDto.max_executions,
77
- execution_count: 0,
78
- next_execution_at: nextExecutionAt,
79
- schedule_status: SCHEDULE_STATUS_ACTIVE,
80
- retry_config: createScheduleDto.retry_config || {
81
- maxRetries: DEFAULT_MAX_RETRIES,
82
- retryDelay: DEFAULT_RETRY_DELAY,
83
- backoffMultiplier: DEFAULT_BACKOFF_MULTIPLIER,
84
- },
85
- metadata: createScheduleDto.metadata || {},
86
- is_enabled: createScheduleDto.is_enabled !== false,
87
- organization_id: createScheduleDto.organization_id || loggedInUser.organization_id,
88
- enterprise_id: createScheduleDto.enterprise_id || loggedInUser.enterprise_id,
89
- level_id: createScheduleDto.level_id || loggedInUser.level_id,
90
- level_type: createScheduleDto.level_type || loggedInUser.level_type,
91
- appcode: createScheduleDto.appcode || loggedInUser.appcode,
92
- created_by: loggedInUser.id,
93
- created_date: new Date(),
94
- status: createScheduleDto.status || 'ACTIVE',
95
- entity_type: 'WFSC',
96
- });
97
-
98
- const savedSchedule = await this.scheduledWorkflowRepository.save(schedule);
99
-
100
- // Add job to Bull queue with cron schedule
101
- if (savedSchedule.is_enabled && savedSchedule.schedule_status === SCHEDULE_STATUS_ACTIVE) {
102
- await this.scheduleJob(savedSchedule, loggedInUser);
103
- }
104
-
105
- this.logger.log(`Schedule created successfully with ID: ${savedSchedule.id}`);
106
- return savedSchedule;
107
- } catch (error) {
108
- this.logger.error(`Failed to create schedule: ${error.message}`, error.stack);
109
- throw error;
110
- }
111
- }
112
-
113
- /**
114
- * 2. Update an existing scheduled workflow
115
- */
116
- async updateSchedule(
117
- updateScheduleDto: UpdateScheduleDto,
118
- loggedInUser: UserData,
119
- ): Promise<ScheduledWorkflow> {
120
- this.logger.log(`Updating schedule ${updateScheduleDto.id} by user ${loggedInUser.id}`);
121
-
122
- try {
123
- const schedule = await this.scheduledWorkflowRepository.findOne({
124
- where: { id: updateScheduleDto.id },
125
- });
126
-
127
- if (!schedule) {
128
- throw new NotFoundException(`Scheduled workflow not found with ID: ${updateScheduleDto.id}`);
129
- }
130
-
131
- // Validate cron expression if updated
132
- if (updateScheduleDto.cron_expression) {
133
- this.validateCronExpression(
134
- updateScheduleDto.cron_expression,
135
- updateScheduleDto.timezone || schedule.timezone,
136
- );
137
- }
138
-
139
- // Update fields
140
- if (updateScheduleDto.workflow_id !== undefined) {
141
- schedule.workflow_id = updateScheduleDto.workflow_id;
142
- }
143
- if (updateScheduleDto.workflow_name !== undefined) {
144
- schedule.workflow_name = updateScheduleDto.workflow_name;
145
- }
146
- if (updateScheduleDto.name !== undefined) {
147
- schedule.name = updateScheduleDto.name;
148
- }
149
- if (updateScheduleDto.description !== undefined) {
150
- schedule.description = updateScheduleDto.description;
151
- }
152
- if (updateScheduleDto.cron_expression !== undefined) {
153
- schedule.cron_expression = updateScheduleDto.cron_expression;
154
- schedule.next_execution_at = this.calculateNextExecution(
155
- updateScheduleDto.cron_expression,
156
- updateScheduleDto.timezone || schedule.timezone,
157
- );
158
- }
159
- if (updateScheduleDto.timezone !== undefined) {
160
- schedule.timezone = updateScheduleDto.timezone;
161
- }
162
- if (updateScheduleDto.start_date !== undefined) {
163
- schedule.start_date = updateScheduleDto.start_date
164
- ? new Date(updateScheduleDto.start_date)
165
- : null;
166
- }
167
- if (updateScheduleDto.end_date !== undefined) {
168
- schedule.end_date = updateScheduleDto.end_date ? new Date(updateScheduleDto.end_date) : null;
169
- }
170
- if (updateScheduleDto.max_executions !== undefined) {
171
- schedule.max_executions = updateScheduleDto.max_executions;
172
- }
173
- if (updateScheduleDto.schedule_status !== undefined) {
174
- schedule.schedule_status = updateScheduleDto.schedule_status;
175
- }
176
- if (updateScheduleDto.retry_config !== undefined) {
177
- schedule.retry_config = updateScheduleDto.retry_config;
178
- }
179
- if (updateScheduleDto.actions !== undefined) {
180
- schedule.actions = updateScheduleDto.actions;
181
- }
182
- if (updateScheduleDto.metadata !== undefined) {
183
- schedule.metadata = updateScheduleDto.metadata;
184
- }
185
- if (updateScheduleDto.is_enabled !== undefined) {
186
- schedule.is_enabled = updateScheduleDto.is_enabled;
187
- }
188
- if (updateScheduleDto.status !== undefined) {
189
- schedule.status = updateScheduleDto.status;
190
- }
191
-
192
- schedule.modified_by = loggedInUser.id;
193
- schedule.modified_date = new Date();
194
-
195
- const updatedSchedule = await this.scheduledWorkflowRepository.save(schedule);
196
-
197
- // Remove old job and create new one if schedule changed
198
- if (schedule.job_id) {
199
- await this.removeJob(schedule.job_id);
200
- }
201
-
202
- if (updatedSchedule.is_enabled && updatedSchedule.schedule_status === SCHEDULE_STATUS_ACTIVE) {
203
- await this.scheduleJob(updatedSchedule, loggedInUser);
204
- }
205
-
206
- this.logger.log(`Schedule updated successfully: ${updatedSchedule.id}`);
207
- return updatedSchedule;
208
- } catch (error) {
209
- this.logger.error(`Failed to update schedule: ${error.message}`, error.stack);
210
- throw error;
211
- }
212
- }
213
-
214
- /**
215
- * 3. Get a scheduled workflow by ID
216
- */
217
- async getScheduleById(scheduleId: number): Promise<ScheduledWorkflow> {
218
- const schedule = await this.scheduledWorkflowRepository.findOne({
219
- where: {
220
- id: scheduleId,
221
- },
222
- });
223
-
224
- if (!schedule) {
225
- throw new NotFoundException(`Scheduled workflow not found with ID: ${scheduleId}`);
226
- }
227
-
228
- return schedule;
229
- }
230
-
231
- /**
232
- * 4. Get all scheduled workflows with pagination and filters
233
- */
234
- async getAllSchedules(
235
- page: number = 1,
236
- size: number = 10,
237
- filters: any = {},
238
- loggedInUser: UserData,
239
- ): Promise<{ data: ScheduledWorkflow[]; total: number; page: number; size: number }> {
240
- const skip = (page - 1) * size;
241
-
242
- const whereConditions: any = {
243
- enterprise_id: loggedInUser.enterprise_id,
244
- };
245
-
246
- if (filters.workflow_id) {
247
- whereConditions.workflow_id = filters.workflow_id;
248
- }
249
- if (filters.schedule_status) {
250
- whereConditions.schedule_status = filters.schedule_status;
251
- }
252
- if (filters.is_enabled !== undefined) {
253
- whereConditions.is_enabled = filters.is_enabled;
254
- }
255
-
256
- const [data, total] = await this.scheduledWorkflowRepository.findAndCount({
257
- where: whereConditions,
258
- skip,
259
- take: size,
260
- order: { created_date: 'DESC' },
261
- });
262
-
263
- return {
264
- data,
265
- total,
266
- page,
267
- size,
268
- };
269
- }
270
-
271
- /**
272
- * 5. Pause a scheduled workflow
273
- */
274
- async pauseSchedule(scheduleId: number, loggedInUser: UserData): Promise<ScheduledWorkflow> {
275
- this.logger.log(`Pausing schedule ${scheduleId} by user ${loggedInUser.id}`);
276
-
277
- const schedule = await this.getScheduleById(scheduleId);
278
-
279
- if (schedule.schedule_status === SCHEDULE_STATUS_PAUSED) {
280
- throw new BadRequestException('Schedule is already paused');
281
- }
282
-
283
- schedule.schedule_status = SCHEDULE_STATUS_PAUSED;
284
- schedule.modified_by = loggedInUser.id;
285
- schedule.modified_date = new Date();
286
-
287
- const updatedSchedule = await this.scheduledWorkflowRepository.save(schedule);
288
-
289
- // Remove job from queue
290
- if (schedule.job_id) {
291
- await this.removeJob(schedule.job_id);
292
- updatedSchedule.job_id = null;
293
- await this.scheduledWorkflowRepository.save(updatedSchedule);
294
- }
295
-
296
- this.logger.log(`Schedule paused successfully: ${scheduleId}`);
297
- return updatedSchedule;
298
- }
299
-
300
- /**
301
- * 6. Resume a paused scheduled workflow
302
- */
303
- async resumeSchedule(scheduleId: number, loggedInUser: UserData): Promise<ScheduledWorkflow> {
304
- this.logger.log(`Resuming schedule ${scheduleId} by user ${loggedInUser.id}`);
305
-
306
- const schedule = await this.getScheduleById(scheduleId);
307
-
308
- if (schedule.schedule_status !== SCHEDULE_STATUS_PAUSED) {
309
- throw new BadRequestException('Schedule is not paused');
310
- }
311
-
312
- schedule.schedule_status = SCHEDULE_STATUS_ACTIVE;
313
- schedule.modified_by = loggedInUser.id;
314
- schedule.modified_date = new Date();
315
-
316
- // Recalculate next execution time
317
- schedule.next_execution_at = this.calculateNextExecution(
318
- schedule.cron_expression,
319
- schedule.timezone,
320
- );
321
-
322
- const updatedSchedule = await this.scheduledWorkflowRepository.save(schedule);
323
-
324
- // Re-add job to queue
325
- if (updatedSchedule.is_enabled) {
326
- await this.scheduleJob(updatedSchedule, loggedInUser);
327
- }
328
-
329
- this.logger.log(`Schedule resumed successfully: ${scheduleId}`);
330
- return updatedSchedule;
331
- }
332
-
333
- /**
334
- * 7. Delete a scheduled workflow
335
- */
336
- async deleteSchedule(scheduleId: number, loggedInUser: UserData): Promise<void> {
337
- this.logger.log(`Deleting schedule ${scheduleId} by user ${loggedInUser.id}`);
338
-
339
- const schedule = await this.getScheduleById(scheduleId);
340
-
341
- // Remove job from queue
342
- if (schedule.job_id) {
343
- await this.removeJob(schedule.job_id);
344
- }
345
-
346
- // Soft delete by setting status to inactive
347
- schedule.schedule_status = SCHEDULE_STATUS_INACTIVE;
348
- schedule.status = 'INACTIVE';
349
- schedule.is_enabled = false;
350
- schedule.modified_by = loggedInUser.id;
351
- schedule.modified_date = new Date();
352
-
353
- await this.scheduledWorkflowRepository.save(schedule);
354
-
355
- this.logger.log(`Schedule deleted successfully: ${scheduleId}`);
356
- }
357
-
358
- /**
359
- * 8. Manually trigger a scheduled workflow execution
360
- */
361
- async triggerManualExecution(
362
- scheduleId: number,
363
- loggedInUser: UserData,
364
- metadata?: Record<string, any>,
365
- ): Promise<{ executionLogId: number; jobId: string }> {
366
- this.logger.log(`Manual trigger for schedule ${scheduleId} by user ${loggedInUser.id}`);
367
-
368
- const schedule = await this.getScheduleById(scheduleId);
369
-
370
- const jobData: ScheduleJobData = {
371
- scheduleId: schedule.id,
372
- workflowId: schedule.workflow_id,
373
- workflowName: schedule.workflow_name,
374
- organizationId: schedule.organization_id,
375
- enterpriseId: schedule.enterprise_id,
376
- levelId: schedule.level_id,
377
- levelType: schedule.level_type,
378
- appcode: schedule.appcode,
379
- createdBy: loggedInUser.id,
380
- triggeredBy: 'MANUAL',
381
- triggeredAt: new Date(),
382
- metadata: metadata || {},
383
- };
384
-
385
- // Add job to queue immediately (not scheduled)
386
- const job = await this.scheduleQueue.add(EXECUTE_SCHEDULED_WORKFLOW_JOB, jobData, {
387
- attempts: schedule.retry_config?.maxRetries || DEFAULT_MAX_RETRIES,
388
- backoff: {
389
- type: 'exponential',
390
- delay: schedule.retry_config?.retryDelay || DEFAULT_RETRY_DELAY,
391
- },
392
- });
393
-
394
- // Create execution log entry
395
- const executionLog = this.executionLogRepository.create({
396
- schedule_id: schedule.id,
397
- workflow_id: schedule.workflow_id,
398
- job_id: job.id.toString(),
399
- execution_status: 'PENDING',
400
- triggered_by: 'MANUAL',
401
- triggered_by_user_id: loggedInUser.id,
402
- organization_id: schedule.organization_id,
403
- enterprise_id: schedule.enterprise_id,
404
- created_by: loggedInUser.id,
405
- created_date: new Date(),
406
- entity_type: 'WFEL',
407
- status: 'ACTIVE',
408
- });
409
-
410
- const savedLog = await this.executionLogRepository.save(executionLog);
411
-
412
- this.logger.log(`Manual execution triggered for schedule ${scheduleId}, job ID: ${job.id}`);
413
-
414
- return {
415
- executionLogId: savedLog.id,
416
- jobId: job.id.toString(),
417
- };
418
- }
419
-
420
- /**
421
- * 9. Get execution logs with filters and pagination
422
- */
423
- async getExecutionLogs(
424
- getExecutionLogsDto: GetExecutionLogsDto,
425
- loggedInUser: UserData,
426
- ): Promise<{ data: WorkflowExecutionLog[]; total: number; page: number; size: number }> {
427
- const page = getExecutionLogsDto.page || 1;
428
- const size = getExecutionLogsDto.size || 10;
429
- const skip = (page - 1) * size;
430
-
431
- const whereConditions: any = {
432
- enterprise_id: getExecutionLogsDto.enterprise_id || loggedInUser.enterprise_id,
433
- };
434
-
435
- if (getExecutionLogsDto.schedule_id) {
436
- whereConditions.schedule_id = getExecutionLogsDto.schedule_id;
437
- }
438
- if (getExecutionLogsDto.workflow_id) {
439
- whereConditions.workflow_id = getExecutionLogsDto.workflow_id;
440
- }
441
- if (getExecutionLogsDto.execution_status) {
442
- whereConditions.execution_status = getExecutionLogsDto.execution_status;
443
- }
444
- if (getExecutionLogsDto.execution_statuses && getExecutionLogsDto.execution_statuses.length > 0) {
445
- whereConditions.execution_status = In(getExecutionLogsDto.execution_statuses);
446
- }
447
- if (getExecutionLogsDto.triggered_by) {
448
- whereConditions.triggered_by = getExecutionLogsDto.triggered_by;
449
- }
450
- if (getExecutionLogsDto.triggered_by_user_id) {
451
- whereConditions.triggered_by_user_id = getExecutionLogsDto.triggered_by_user_id;
452
- }
453
-
454
- // Date range filter
455
- if (getExecutionLogsDto.start_date && getExecutionLogsDto.end_date) {
456
- whereConditions.created_date = Between(
457
- new Date(getExecutionLogsDto.start_date),
458
- new Date(getExecutionLogsDto.end_date),
459
- );
460
- }
461
-
462
- const [data, total] = await this.executionLogRepository.findAndCount({
463
- where: whereConditions,
464
- skip,
465
- take: size,
466
- order: {
467
- [getExecutionLogsDto.sortBy || 'created_date']:
468
- getExecutionLogsDto.sortOrder || 'DESC',
469
- },
470
- });
471
-
472
- return {
473
- data,
474
- total,
475
- page,
476
- size,
477
- };
478
- }
479
-
480
- /**
481
- * Get execution statistics for a schedule
482
- */
483
- async getExecutionStats(scheduleId: number): Promise<any> {
484
- const schedule = await this.getScheduleById(scheduleId);
485
-
486
- const result = await this.executionLogRepository
487
- .createQueryBuilder('log')
488
- .select([
489
- 'COUNT(*) AS total_executions',
490
- 'SUM(CASE WHEN log.execution_status = \'COMPLETED\' THEN 1 ELSE 0 END) AS successful_executions',
491
- 'SUM(CASE WHEN log.execution_status = \'FAILED\' THEN 1 ELSE 0 END) AS failed_executions',
492
- 'SUM(CASE WHEN log.execution_status = \'PENDING\' THEN 1 ELSE 0 END) AS pending_executions',
493
- 'SUM(CASE WHEN log.execution_status = \'RUNNING\' THEN 1 ELSE 0 END) AS running_executions',
494
- 'AVG(log.duration_ms) AS average_duration_ms',
495
- 'SUM(log.total_records) AS total_records_processed',
496
- ])
497
- .where('log.schedule_id = :scheduleId', { scheduleId })
498
- .getRawOne();
499
- const successRate =
500
- result.total_executions > 0
501
- ? (result.successful_executions / result.total_executions) * 100
502
- : 0;
503
-
504
- return {
505
- schedule_id: scheduleId,
506
- schedule_name: schedule.name,
507
- total_executions: parseInt(result.total_executions),
508
- successful_executions: parseInt(result.successful_executions),
509
- failed_executions: parseInt(result.failed_executions),
510
- pending_executions: parseInt(result.pending_executions),
511
- running_executions: parseInt(result.running_executions),
512
- average_duration_ms: parseFloat(result.average_duration_ms) || 0,
513
- total_records_processed: parseInt(result.total_records_processed) || 0,
514
- success_rate_percentage: parseFloat(successRate.toFixed(2)),
515
- };
516
- }
517
-
518
- /**
519
- * Private helper: Schedule a job in Bull queue
520
- */
521
- private async scheduleJob(schedule: ScheduledWorkflow, loggedInUser: UserData): Promise<void> {
522
- const jobData: ScheduleJobData = {
523
- scheduleId: schedule.id,
524
- workflowId: schedule.workflow_id,
525
- workflowName: schedule.workflow_name,
526
- organizationId: schedule.organization_id,
527
- enterpriseId: schedule.enterprise_id,
528
- levelId: schedule.level_id,
529
- levelType: schedule.level_type,
530
- appcode: schedule.appcode,
531
- createdBy: loggedInUser.id,
532
- triggeredBy: 'SCHEDULE',
533
- triggeredAt: new Date(),
534
- metadata: schedule.metadata || {},
535
- };
536
-
537
- const job = await this.scheduleQueue.add(EXECUTE_SCHEDULED_WORKFLOW_JOB, jobData, {
538
- repeat: {
539
- cron: schedule.cron_expression,
540
- tz: schedule.timezone,
541
- startDate: schedule.start_date || undefined,
542
- endDate: schedule.end_date || undefined,
543
- },
544
- attempts: schedule.retry_config?.maxRetries || DEFAULT_MAX_RETRIES,
545
- backoff: {
546
- type: 'exponential',
547
- delay: schedule.retry_config?.retryDelay || DEFAULT_RETRY_DELAY,
548
- },
549
- });
550
-
551
- schedule.job_id = job.id.toString();
552
- await this.scheduledWorkflowRepository.save(schedule);
553
-
554
- this.logger.log(`Job scheduled with ID: ${job.id} for schedule ${schedule.id}`);
555
- }
556
-
557
- /**
558
- * Private helper: Remove a job from Bull queue
559
- */
560
- private async removeJob(jobId: string): Promise<void> {
561
- try {
562
- const job = await this.scheduleQueue.getJob(jobId);
563
- if (job) {
564
- await job.remove();
565
- this.logger.log(`Job removed: ${jobId}`);
566
- }
567
- } catch (error) {
568
- this.logger.warn(`Failed to remove job ${jobId}: ${error.message}`);
569
- }
570
- }
571
-
572
- /**
573
- * Private helper: Validate cron expression
574
- */
575
- private validateCronExpression(cronExpression: string, timezone: string): void {
576
- try {
577
- cronParser.CronExpressionParser.parse(cronExpression, { tz: timezone });
578
- } catch (error: any) {
579
- throw new BadRequestException(
580
- `Invalid cron expression: ${cronExpression}. Error: ${error.message}`,
581
- );
582
- }
583
- }
584
-
585
- /**
586
- * Private helper: Calculate next execution time
587
- */
588
- private calculateNextExecution(cronExpression: string, timezone: string): Date | null {
589
- try {
590
- const interval = cronParser.CronExpressionParser.parse(cronExpression, { tz: timezone });
591
- return interval.next().toDate();
592
- } catch (error: any) {
593
- this.logger.error(`Failed to calculate next execution: ${error.message}`);
594
- return null;
595
- }
596
- }
597
- }
1
+ import { BadRequestException, Injectable, Logger, NotFoundException } from '@nestjs/common';
2
+ import { InjectRepository } from '@nestjs/typeorm';
3
+ import { Between, In, Repository } from 'typeorm';
4
+ import { InjectQueue } from '@nestjs/bull';
5
+ import { Queue } from 'bull';
6
+ import * as cronParser from 'cron-parser';
7
+
8
+ import { ScheduledWorkflow } from '../entities/scheduled-workflow.entity';
9
+ import { WorkflowExecutionLog } from '../entities/workflow-execution-log.entity';
10
+ import { CreateScheduleDto } from '../dto/create-schedule.dto';
11
+ import { UpdateScheduleDto } from '../dto/update-schedule.dto';
12
+ import { GetExecutionLogsDto } from '../dto/get-execution-logs.dto';
13
+ import { UserData } from '../../user/entity/user.entity';
14
+ import {
15
+ DEFAULT_BACKOFF_MULTIPLIER,
16
+ DEFAULT_MAX_RETRIES,
17
+ DEFAULT_RETRY_DELAY,
18
+ DEFAULT_TIMEZONE,
19
+ EXECUTE_SCHEDULED_WORKFLOW_JOB,
20
+ SCHEDULE_STATUS_ACTIVE,
21
+ SCHEDULE_STATUS_INACTIVE,
22
+ SCHEDULE_STATUS_PAUSED,
23
+ WORKFLOW_SCHEDULE_QUEUE,
24
+ } from '../constants/schedule.constants';
25
+ import { ScheduleJobData } from '../interfaces/schedule-job-data.interface';
26
+
27
+ /**
28
+ * Workflow Schedule Service
29
+ * Manages scheduled workflows and their execution
30
+ */
31
+ @Injectable()
32
+ export class WorkflowScheduleService {
33
+ private readonly logger = new Logger(WorkflowScheduleService.name);
34
+
35
+ constructor(
36
+ @InjectRepository(ScheduledWorkflow)
37
+ private readonly scheduledWorkflowRepository: Repository<ScheduledWorkflow>,
38
+ @InjectRepository(WorkflowExecutionLog)
39
+ private readonly executionLogRepository: Repository<WorkflowExecutionLog>,
40
+ @InjectQueue(WORKFLOW_SCHEDULE_QUEUE)
41
+ private readonly scheduleQueue: Queue<ScheduleJobData>,
42
+ ) {
43
+ }
44
+
45
+ /**
46
+ * 1. Create a new scheduled workflow
47
+ */
48
+ async createSchedule(
49
+ createScheduleDto: CreateScheduleDto,
50
+ loggedInUser: UserData,
51
+ ): Promise<ScheduledWorkflow> {
52
+ this.logger.log(
53
+ `Creating schedule for workflow ${createScheduleDto.workflow_id} by user ${loggedInUser.id}`,
54
+ );
55
+
56
+ try {
57
+ // Validate cron expression
58
+ this.validateCronExpression(createScheduleDto.cron_expression, createScheduleDto.timezone);
59
+
60
+ // Calculate next execution time
61
+ const nextExecutionAt = this.calculateNextExecution(
62
+ createScheduleDto.cron_expression,
63
+ createScheduleDto.timezone || DEFAULT_TIMEZONE,
64
+ );
65
+
66
+ // Create scheduled workflow entity
67
+ const schedule = this.scheduledWorkflowRepository.create({
68
+ workflow_id: createScheduleDto.workflow_id,
69
+ workflow_name: createScheduleDto.workflow_name,
70
+ name: createScheduleDto.name,
71
+ description: createScheduleDto.description,
72
+ cron_expression: createScheduleDto.cron_expression,
73
+ timezone: createScheduleDto.timezone || DEFAULT_TIMEZONE,
74
+ start_date: createScheduleDto.start_date ? new Date(createScheduleDto.start_date) : null,
75
+ end_date: createScheduleDto.end_date ? new Date(createScheduleDto.end_date) : null,
76
+ max_executions: createScheduleDto.max_executions,
77
+ execution_count: 0,
78
+ next_execution_at: nextExecutionAt,
79
+ schedule_status: SCHEDULE_STATUS_ACTIVE,
80
+ retry_config: createScheduleDto.retry_config || {
81
+ maxRetries: DEFAULT_MAX_RETRIES,
82
+ retryDelay: DEFAULT_RETRY_DELAY,
83
+ backoffMultiplier: DEFAULT_BACKOFF_MULTIPLIER,
84
+ },
85
+ metadata: createScheduleDto.metadata || {},
86
+ is_enabled: createScheduleDto.is_enabled !== false,
87
+ organization_id: createScheduleDto.organization_id || loggedInUser.organization_id,
88
+ enterprise_id: createScheduleDto.enterprise_id || loggedInUser.enterprise_id,
89
+ level_id: createScheduleDto.level_id || loggedInUser.level_id,
90
+ level_type: createScheduleDto.level_type || loggedInUser.level_type,
91
+ appcode: createScheduleDto.appcode || loggedInUser.appcode,
92
+ created_by: loggedInUser.id,
93
+ created_date: new Date(),
94
+ status: createScheduleDto.status || 'ACTIVE',
95
+ entity_type: 'WFSC',
96
+ });
97
+
98
+ const savedSchedule = await this.scheduledWorkflowRepository.save(schedule);
99
+
100
+ // Add job to Bull queue with cron schedule
101
+ if (savedSchedule.is_enabled && savedSchedule.schedule_status === SCHEDULE_STATUS_ACTIVE) {
102
+ await this.scheduleJob(savedSchedule, loggedInUser);
103
+ }
104
+
105
+ this.logger.log(`Schedule created successfully with ID: ${savedSchedule.id}`);
106
+ return savedSchedule;
107
+ } catch (error) {
108
+ this.logger.error(`Failed to create schedule: ${error.message}`, error.stack);
109
+ throw error;
110
+ }
111
+ }
112
+
113
+ /**
114
+ * 2. Update an existing scheduled workflow
115
+ */
116
+ async updateSchedule(
117
+ updateScheduleDto: UpdateScheduleDto,
118
+ loggedInUser: UserData,
119
+ ): Promise<ScheduledWorkflow> {
120
+ this.logger.log(`Updating schedule ${updateScheduleDto.id} by user ${loggedInUser.id}`);
121
+
122
+ try {
123
+ const schedule = await this.scheduledWorkflowRepository.findOne({
124
+ where: { id: updateScheduleDto.id },
125
+ });
126
+
127
+ if (!schedule) {
128
+ throw new NotFoundException(`Scheduled workflow not found with ID: ${updateScheduleDto.id}`);
129
+ }
130
+
131
+ // Validate cron expression if updated
132
+ if (updateScheduleDto.cron_expression) {
133
+ this.validateCronExpression(
134
+ updateScheduleDto.cron_expression,
135
+ updateScheduleDto.timezone || schedule.timezone,
136
+ );
137
+ }
138
+
139
+ // Update fields
140
+ if (updateScheduleDto.workflow_id !== undefined) {
141
+ schedule.workflow_id = updateScheduleDto.workflow_id;
142
+ }
143
+ if (updateScheduleDto.workflow_name !== undefined) {
144
+ schedule.workflow_name = updateScheduleDto.workflow_name;
145
+ }
146
+ if (updateScheduleDto.name !== undefined) {
147
+ schedule.name = updateScheduleDto.name;
148
+ }
149
+ if (updateScheduleDto.description !== undefined) {
150
+ schedule.description = updateScheduleDto.description;
151
+ }
152
+ if (updateScheduleDto.cron_expression !== undefined) {
153
+ schedule.cron_expression = updateScheduleDto.cron_expression;
154
+ schedule.next_execution_at = this.calculateNextExecution(
155
+ updateScheduleDto.cron_expression,
156
+ updateScheduleDto.timezone || schedule.timezone,
157
+ );
158
+ }
159
+ if (updateScheduleDto.timezone !== undefined) {
160
+ schedule.timezone = updateScheduleDto.timezone;
161
+ }
162
+ if (updateScheduleDto.start_date !== undefined) {
163
+ schedule.start_date = updateScheduleDto.start_date
164
+ ? new Date(updateScheduleDto.start_date)
165
+ : null;
166
+ }
167
+ if (updateScheduleDto.end_date !== undefined) {
168
+ schedule.end_date = updateScheduleDto.end_date ? new Date(updateScheduleDto.end_date) : null;
169
+ }
170
+ if (updateScheduleDto.max_executions !== undefined) {
171
+ schedule.max_executions = updateScheduleDto.max_executions;
172
+ }
173
+ if (updateScheduleDto.schedule_status !== undefined) {
174
+ schedule.schedule_status = updateScheduleDto.schedule_status;
175
+ }
176
+ if (updateScheduleDto.retry_config !== undefined) {
177
+ schedule.retry_config = updateScheduleDto.retry_config;
178
+ }
179
+ if (updateScheduleDto.actions !== undefined) {
180
+ schedule.actions = updateScheduleDto.actions;
181
+ }
182
+ if (updateScheduleDto.metadata !== undefined) {
183
+ schedule.metadata = updateScheduleDto.metadata;
184
+ }
185
+ if (updateScheduleDto.is_enabled !== undefined) {
186
+ schedule.is_enabled = updateScheduleDto.is_enabled;
187
+ }
188
+ if (updateScheduleDto.status !== undefined) {
189
+ schedule.status = updateScheduleDto.status;
190
+ }
191
+
192
+ schedule.modified_by = loggedInUser.id;
193
+ schedule.modified_date = new Date();
194
+
195
+ const updatedSchedule = await this.scheduledWorkflowRepository.save(schedule);
196
+
197
+ // Remove old job and create new one if schedule changed
198
+ if (schedule.job_id) {
199
+ await this.removeJob(schedule.job_id);
200
+ }
201
+
202
+ if (updatedSchedule.is_enabled && updatedSchedule.schedule_status === SCHEDULE_STATUS_ACTIVE) {
203
+ await this.scheduleJob(updatedSchedule, loggedInUser);
204
+ }
205
+
206
+ this.logger.log(`Schedule updated successfully: ${updatedSchedule.id}`);
207
+ return updatedSchedule;
208
+ } catch (error) {
209
+ this.logger.error(`Failed to update schedule: ${error.message}`, error.stack);
210
+ throw error;
211
+ }
212
+ }
213
+
214
+ /**
215
+ * 3. Get a scheduled workflow by ID
216
+ */
217
+ async getScheduleById(scheduleId: number): Promise<ScheduledWorkflow> {
218
+ const schedule = await this.scheduledWorkflowRepository.findOne({
219
+ where: {
220
+ id: scheduleId,
221
+ },
222
+ });
223
+
224
+ if (!schedule) {
225
+ throw new NotFoundException(`Scheduled workflow not found with ID: ${scheduleId}`);
226
+ }
227
+
228
+ return schedule;
229
+ }
230
+
231
+ /**
232
+ * 4. Get all scheduled workflows with pagination and filters
233
+ */
234
+ async getAllSchedules(
235
+ page: number = 1,
236
+ size: number = 10,
237
+ filters: any = {},
238
+ loggedInUser: UserData,
239
+ ): Promise<{ data: ScheduledWorkflow[]; total: number; page: number; size: number }> {
240
+ const skip = (page - 1) * size;
241
+
242
+ const whereConditions: any = {
243
+ enterprise_id: loggedInUser.enterprise_id,
244
+ };
245
+
246
+ if (filters.workflow_id) {
247
+ whereConditions.workflow_id = filters.workflow_id;
248
+ }
249
+ if (filters.schedule_status) {
250
+ whereConditions.schedule_status = filters.schedule_status;
251
+ }
252
+ if (filters.is_enabled !== undefined) {
253
+ whereConditions.is_enabled = filters.is_enabled;
254
+ }
255
+
256
+ const [data, total] = await this.scheduledWorkflowRepository.findAndCount({
257
+ where: whereConditions,
258
+ skip,
259
+ take: size,
260
+ order: { created_date: 'DESC' },
261
+ });
262
+
263
+ return {
264
+ data,
265
+ total,
266
+ page,
267
+ size,
268
+ };
269
+ }
270
+
271
+ /**
272
+ * 5. Pause a scheduled workflow
273
+ */
274
+ async pauseSchedule(scheduleId: number, loggedInUser: UserData): Promise<ScheduledWorkflow> {
275
+ this.logger.log(`Pausing schedule ${scheduleId} by user ${loggedInUser.id}`);
276
+
277
+ const schedule = await this.getScheduleById(scheduleId);
278
+
279
+ if (schedule.schedule_status === SCHEDULE_STATUS_PAUSED) {
280
+ throw new BadRequestException('Schedule is already paused');
281
+ }
282
+
283
+ schedule.schedule_status = SCHEDULE_STATUS_PAUSED;
284
+ schedule.modified_by = loggedInUser.id;
285
+ schedule.modified_date = new Date();
286
+
287
+ const updatedSchedule = await this.scheduledWorkflowRepository.save(schedule);
288
+
289
+ // Remove job from queue
290
+ if (schedule.job_id) {
291
+ await this.removeJob(schedule.job_id);
292
+ updatedSchedule.job_id = null;
293
+ await this.scheduledWorkflowRepository.save(updatedSchedule);
294
+ }
295
+
296
+ this.logger.log(`Schedule paused successfully: ${scheduleId}`);
297
+ return updatedSchedule;
298
+ }
299
+
300
+ /**
301
+ * 6. Resume a paused scheduled workflow
302
+ */
303
+ async resumeSchedule(scheduleId: number, loggedInUser: UserData): Promise<ScheduledWorkflow> {
304
+ this.logger.log(`Resuming schedule ${scheduleId} by user ${loggedInUser.id}`);
305
+
306
+ const schedule = await this.getScheduleById(scheduleId);
307
+
308
+ if (schedule.schedule_status !== SCHEDULE_STATUS_PAUSED) {
309
+ throw new BadRequestException('Schedule is not paused');
310
+ }
311
+
312
+ schedule.schedule_status = SCHEDULE_STATUS_ACTIVE;
313
+ schedule.modified_by = loggedInUser.id;
314
+ schedule.modified_date = new Date();
315
+
316
+ // Recalculate next execution time
317
+ schedule.next_execution_at = this.calculateNextExecution(
318
+ schedule.cron_expression,
319
+ schedule.timezone,
320
+ );
321
+
322
+ const updatedSchedule = await this.scheduledWorkflowRepository.save(schedule);
323
+
324
+ // Re-add job to queue
325
+ if (updatedSchedule.is_enabled) {
326
+ await this.scheduleJob(updatedSchedule, loggedInUser);
327
+ }
328
+
329
+ this.logger.log(`Schedule resumed successfully: ${scheduleId}`);
330
+ return updatedSchedule;
331
+ }
332
+
333
+ /**
334
+ * 7. Delete a scheduled workflow
335
+ */
336
+ async deleteSchedule(scheduleId: number, loggedInUser: UserData): Promise<void> {
337
+ this.logger.log(`Deleting schedule ${scheduleId} by user ${loggedInUser.id}`);
338
+
339
+ const schedule = await this.getScheduleById(scheduleId);
340
+
341
+ // Remove job from queue
342
+ if (schedule.job_id) {
343
+ await this.removeJob(schedule.job_id);
344
+ }
345
+
346
+ // Soft delete by setting status to inactive
347
+ schedule.schedule_status = SCHEDULE_STATUS_INACTIVE;
348
+ schedule.status = 'INACTIVE';
349
+ schedule.is_enabled = false;
350
+ schedule.modified_by = loggedInUser.id;
351
+ schedule.modified_date = new Date();
352
+
353
+ await this.scheduledWorkflowRepository.save(schedule);
354
+
355
+ this.logger.log(`Schedule deleted successfully: ${scheduleId}`);
356
+ }
357
+
358
+ /**
359
+ * 8. Manually trigger a scheduled workflow execution
360
+ */
361
+ async triggerManualExecution(
362
+ scheduleId: number,
363
+ loggedInUser: UserData,
364
+ metadata?: Record<string, any>,
365
+ ): Promise<{ executionLogId: number; jobId: string }> {
366
+ this.logger.log(`Manual trigger for schedule ${scheduleId} by user ${loggedInUser.id}`);
367
+
368
+ const schedule = await this.getScheduleById(scheduleId);
369
+
370
+ const jobData: ScheduleJobData = {
371
+ scheduleId: schedule.id,
372
+ workflowId: schedule.workflow_id,
373
+ workflowName: schedule.workflow_name,
374
+ organizationId: schedule.organization_id,
375
+ enterpriseId: schedule.enterprise_id,
376
+ levelId: schedule.level_id,
377
+ levelType: schedule.level_type,
378
+ appcode: schedule.appcode,
379
+ createdBy: loggedInUser.id,
380
+ triggeredBy: 'MANUAL',
381
+ triggeredAt: new Date(),
382
+ metadata: metadata || {},
383
+ };
384
+
385
+ // Add job to queue immediately (not scheduled)
386
+ const job = await this.scheduleQueue.add(EXECUTE_SCHEDULED_WORKFLOW_JOB, jobData, {
387
+ attempts: schedule.retry_config?.maxRetries || DEFAULT_MAX_RETRIES,
388
+ backoff: {
389
+ type: 'exponential',
390
+ delay: schedule.retry_config?.retryDelay || DEFAULT_RETRY_DELAY,
391
+ },
392
+ });
393
+
394
+ // Create execution log entry
395
+ const executionLog = this.executionLogRepository.create({
396
+ schedule_id: schedule.id,
397
+ workflow_id: schedule.workflow_id,
398
+ job_id: job.id.toString(),
399
+ execution_status: 'PENDING',
400
+ triggered_by: 'MANUAL',
401
+ triggered_by_user_id: loggedInUser.id,
402
+ organization_id: schedule.organization_id,
403
+ enterprise_id: schedule.enterprise_id,
404
+ created_by: loggedInUser.id,
405
+ created_date: new Date(),
406
+ entity_type: 'WFEL',
407
+ status: 'ACTIVE',
408
+ });
409
+
410
+ const savedLog = await this.executionLogRepository.save(executionLog);
411
+
412
+ this.logger.log(`Manual execution triggered for schedule ${scheduleId}, job ID: ${job.id}`);
413
+
414
+ return {
415
+ executionLogId: savedLog.id,
416
+ jobId: job.id.toString(),
417
+ };
418
+ }
419
+
420
+ /**
421
+ * 9. Get execution logs with filters and pagination
422
+ */
423
+ async getExecutionLogs(
424
+ getExecutionLogsDto: GetExecutionLogsDto,
425
+ loggedInUser: UserData,
426
+ ): Promise<{ data: WorkflowExecutionLog[]; total: number; page: number; size: number }> {
427
+ const page = getExecutionLogsDto.page || 1;
428
+ const size = getExecutionLogsDto.size || 10;
429
+ const skip = (page - 1) * size;
430
+
431
+ const whereConditions: any = {
432
+ enterprise_id: getExecutionLogsDto.enterprise_id || loggedInUser.enterprise_id,
433
+ };
434
+
435
+ if (getExecutionLogsDto.schedule_id) {
436
+ whereConditions.schedule_id = getExecutionLogsDto.schedule_id;
437
+ }
438
+ if (getExecutionLogsDto.workflow_id) {
439
+ whereConditions.workflow_id = getExecutionLogsDto.workflow_id;
440
+ }
441
+ if (getExecutionLogsDto.execution_status) {
442
+ whereConditions.execution_status = getExecutionLogsDto.execution_status;
443
+ }
444
+ if (getExecutionLogsDto.execution_statuses && getExecutionLogsDto.execution_statuses.length > 0) {
445
+ whereConditions.execution_status = In(getExecutionLogsDto.execution_statuses);
446
+ }
447
+ if (getExecutionLogsDto.triggered_by) {
448
+ whereConditions.triggered_by = getExecutionLogsDto.triggered_by;
449
+ }
450
+ if (getExecutionLogsDto.triggered_by_user_id) {
451
+ whereConditions.triggered_by_user_id = getExecutionLogsDto.triggered_by_user_id;
452
+ }
453
+
454
+ // Date range filter
455
+ if (getExecutionLogsDto.start_date && getExecutionLogsDto.end_date) {
456
+ whereConditions.created_date = Between(
457
+ new Date(getExecutionLogsDto.start_date),
458
+ new Date(getExecutionLogsDto.end_date),
459
+ );
460
+ }
461
+
462
+ const [data, total] = await this.executionLogRepository.findAndCount({
463
+ where: whereConditions,
464
+ skip,
465
+ take: size,
466
+ order: {
467
+ [getExecutionLogsDto.sortBy || 'created_date']:
468
+ getExecutionLogsDto.sortOrder || 'DESC',
469
+ },
470
+ });
471
+
472
+ return {
473
+ data,
474
+ total,
475
+ page,
476
+ size,
477
+ };
478
+ }
479
+
480
+ /**
481
+ * Get execution statistics for a schedule
482
+ */
483
+ async getExecutionStats(scheduleId: number): Promise<any> {
484
+ const schedule = await this.getScheduleById(scheduleId);
485
+
486
+ const result = await this.executionLogRepository
487
+ .createQueryBuilder('log')
488
+ .select([
489
+ 'COUNT(*) AS total_executions',
490
+ 'SUM(CASE WHEN log.execution_status = \'COMPLETED\' THEN 1 ELSE 0 END) AS successful_executions',
491
+ 'SUM(CASE WHEN log.execution_status = \'FAILED\' THEN 1 ELSE 0 END) AS failed_executions',
492
+ 'SUM(CASE WHEN log.execution_status = \'PENDING\' THEN 1 ELSE 0 END) AS pending_executions',
493
+ 'SUM(CASE WHEN log.execution_status = \'RUNNING\' THEN 1 ELSE 0 END) AS running_executions',
494
+ 'AVG(log.duration_ms) AS average_duration_ms',
495
+ 'SUM(log.total_records) AS total_records_processed',
496
+ ])
497
+ .where('log.schedule_id = :scheduleId', { scheduleId })
498
+ .getRawOne();
499
+ const successRate =
500
+ result.total_executions > 0
501
+ ? (result.successful_executions / result.total_executions) * 100
502
+ : 0;
503
+
504
+ return {
505
+ schedule_id: scheduleId,
506
+ schedule_name: schedule.name,
507
+ total_executions: parseInt(result.total_executions),
508
+ successful_executions: parseInt(result.successful_executions),
509
+ failed_executions: parseInt(result.failed_executions),
510
+ pending_executions: parseInt(result.pending_executions),
511
+ running_executions: parseInt(result.running_executions),
512
+ average_duration_ms: parseFloat(result.average_duration_ms) || 0,
513
+ total_records_processed: parseInt(result.total_records_processed) || 0,
514
+ success_rate_percentage: parseFloat(successRate.toFixed(2)),
515
+ };
516
+ }
517
+
518
+ /**
519
+ * Private helper: Schedule a job in Bull queue
520
+ */
521
+ private async scheduleJob(schedule: ScheduledWorkflow, loggedInUser: UserData): Promise<void> {
522
+ const jobData: ScheduleJobData = {
523
+ scheduleId: schedule.id,
524
+ workflowId: schedule.workflow_id,
525
+ workflowName: schedule.workflow_name,
526
+ organizationId: schedule.organization_id,
527
+ enterpriseId: schedule.enterprise_id,
528
+ levelId: schedule.level_id,
529
+ levelType: schedule.level_type,
530
+ appcode: schedule.appcode,
531
+ createdBy: loggedInUser.id,
532
+ triggeredBy: 'SCHEDULE',
533
+ triggeredAt: new Date(),
534
+ metadata: schedule.metadata || {},
535
+ };
536
+
537
+ const job = await this.scheduleQueue.add(EXECUTE_SCHEDULED_WORKFLOW_JOB, jobData, {
538
+ repeat: {
539
+ cron: schedule.cron_expression,
540
+ tz: schedule.timezone,
541
+ startDate: schedule.start_date || undefined,
542
+ endDate: schedule.end_date || undefined,
543
+ },
544
+ attempts: schedule.retry_config?.maxRetries || DEFAULT_MAX_RETRIES,
545
+ backoff: {
546
+ type: 'exponential',
547
+ delay: schedule.retry_config?.retryDelay || DEFAULT_RETRY_DELAY,
548
+ },
549
+ });
550
+
551
+ schedule.job_id = job.id.toString();
552
+ await this.scheduledWorkflowRepository.save(schedule);
553
+
554
+ this.logger.log(`Job scheduled with ID: ${job.id} for schedule ${schedule.id}`);
555
+ }
556
+
557
+ /**
558
+ * Private helper: Remove a job from Bull queue
559
+ */
560
+ private async removeJob(jobId: string): Promise<void> {
561
+ try {
562
+ const job = await this.scheduleQueue.getJob(jobId);
563
+ if (job) {
564
+ await job.remove();
565
+ this.logger.log(`Job removed: ${jobId}`);
566
+ }
567
+ } catch (error) {
568
+ this.logger.warn(`Failed to remove job ${jobId}: ${error.message}`);
569
+ }
570
+ }
571
+
572
+ /**
573
+ * Private helper: Validate cron expression
574
+ */
575
+ private validateCronExpression(cronExpression: string, timezone: string): void {
576
+ try {
577
+ cronParser.CronExpressionParser.parse(cronExpression, { tz: timezone });
578
+ } catch (error: any) {
579
+ throw new BadRequestException(
580
+ `Invalid cron expression: ${cronExpression}. Error: ${error.message}`,
581
+ );
582
+ }
583
+ }
584
+
585
+ /**
586
+ * Private helper: Calculate next execution time
587
+ */
588
+ private calculateNextExecution(cronExpression: string, timezone: string): Date | null {
589
+ try {
590
+ const interval = cronParser.CronExpressionParser.parse(cronExpression, { tz: timezone });
591
+ return interval.next().toDate();
592
+ } catch (error: any) {
593
+ this.logger.error(`Failed to calculate next execution: ${error.message}`);
594
+ return null;
595
+ }
596
+ }
597
+ }