rez_core 5.0.182 → 6.5.1

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 (429) 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-app-mapping.entity.d.ts +4 -2
  29. package/dist/module/enterprise/entity/organization-app-mapping.entity.js +12 -7
  30. package/dist/module/enterprise/entity/organization-app-mapping.entity.js.map +1 -1
  31. package/dist/module/enterprise/entity/organization.entity.d.ts +1 -17
  32. package/dist/module/enterprise/entity/organization.entity.js +4 -72
  33. package/dist/module/enterprise/entity/organization.entity.js.map +1 -1
  34. package/dist/module/enterprise/repository/enterprise.repository.d.ts +2 -2
  35. package/dist/module/enterprise/repository/enterprise.repository.js +9 -4
  36. package/dist/module/enterprise/repository/enterprise.repository.js.map +1 -1
  37. package/dist/module/enterprise/service/enterprise.service.d.ts +2 -2
  38. package/dist/module/enterprise/service/enterprise.service.js +4 -4
  39. package/dist/module/enterprise/service/enterprise.service.js.map +1 -1
  40. package/dist/module/enterprise/service/organization.service.d.ts +2 -2
  41. package/dist/module/enterprise/service/organization.service.js +97 -20
  42. package/dist/module/enterprise/service/organization.service.js.map +1 -1
  43. package/dist/module/filter/service/filter.service.js +19 -19
  44. package/dist/module/integration/examples/usage.example.js +9 -9
  45. package/dist/module/meta/repository/attribute-master.repository.js +8 -8
  46. package/dist/module/meta/service/entity-dynamic.service.js +16 -16
  47. package/dist/module/meta/service/media-data.service.js +6 -6
  48. package/dist/module/meta/service/resolver.service.js +11 -11
  49. package/dist/module/module/repository/menu.repository.js +4 -4
  50. package/dist/module/notification/service/notification.service.js +6 -6
  51. package/dist/module/user/controller/login.controller.js +18 -18
  52. package/dist/module/workflow/repository/action.repository.js +2 -2
  53. package/dist/module/workflow/repository/stage.repository.js +8 -8
  54. package/dist/module/workflow/service/action-template-mapping.service.js +2 -2
  55. package/dist/module/workflow/service/action.service.js +5 -5
  56. package/dist/module/workflow/service/entity-modification.service.js +2 -2
  57. package/dist/module/workflow/service/task.service.js +8 -8
  58. package/dist/module/workflow-automation/service/schedule-handler.service.js +9 -9
  59. package/dist/table.config.d.ts +1 -2
  60. package/dist/tsconfig.build.tsbuildinfo +1 -1
  61. package/dist/utils/service/reflection-helper.service.js +2 -2
  62. package/docs/modules/event-driven-integration-design.md +91 -91
  63. package/docs/modules/integration.md +250 -250
  64. package/eslint.config.mjs +34 -34
  65. package/nest-cli.json +14 -14
  66. package/package.json +125 -125
  67. package/server.log +850 -0
  68. package/src/app.controller.ts +12 -12
  69. package/src/app.module.ts +68 -68
  70. package/src/app.service.ts +8 -8
  71. package/src/config/bull.config.ts +69 -69
  72. package/src/config/config.module.ts +17 -17
  73. package/src/config/database.config.ts +48 -48
  74. package/src/constant/global.constant.ts +67 -67
  75. package/src/core.module.ts +94 -94
  76. package/src/decorators/roles.decorator.ts +7 -7
  77. package/src/dtos/response.dto.ts +6 -6
  78. package/src/dtos/response.ts +5 -5
  79. package/src/index.ts +1 -1
  80. package/src/module/auth/auth.module.ts +49 -49
  81. package/src/module/auth/controller/auth.controller.ts +28 -28
  82. package/src/module/auth/guards/google-auth.guard.ts +9 -9
  83. package/src/module/auth/guards/jwt.guard.ts +22 -22
  84. package/src/module/auth/guards/role.guard.ts +68 -68
  85. package/src/module/auth/services/auth.service.ts +56 -56
  86. package/src/module/auth/services/jwt.service.ts +11 -11
  87. package/src/module/auth/strategies/google.strategy.ts +54 -54
  88. package/src/module/auth/strategies/jwt.strategy.ts +58 -58
  89. package/src/module/auth/strategies/local.strategy.ts +13 -13
  90. package/src/module/dashboard/controller/dashboard.controller.ts +38 -38
  91. package/src/module/dashboard/dashboard.module.ts +21 -21
  92. package/src/module/dashboard/entity/dashboard_page_data.entity.ts +27 -27
  93. package/src/module/dashboard/entity/widget_master.entity.ts +18 -18
  94. package/src/module/dashboard/repository/dashboard.repository.ts +49 -49
  95. package/src/module/dashboard/service/dashboard.service.ts +72 -72
  96. package/src/module/enterprise/controller/enterprise.controller.ts +40 -0
  97. package/src/module/enterprise/controller/organization.controller.ts +93 -36
  98. package/src/module/enterprise/enterprise.module.ts +46 -45
  99. package/src/module/enterprise/entity/enterprise.entity.ts +31 -37
  100. package/src/module/enterprise/entity/organization-app-mapping.entity.ts +18 -13
  101. package/src/module/enterprise/entity/organization.entity.ts +40 -92
  102. package/src/module/enterprise/repository/enterprise.repository.ts +39 -31
  103. package/src/module/enterprise/repository/organization.repository.ts +26 -26
  104. package/src/module/enterprise/repository/school.repository.ts +272 -272
  105. package/src/module/enterprise/service/brand.service.ts +5 -5
  106. package/src/module/enterprise/service/enterprise.service.ts +22 -16
  107. package/src/module/enterprise/service/organization-app-mapping.service.ts +4 -4
  108. package/src/module/enterprise/service/organization.service.ts +262 -145
  109. package/src/module/entity_json/controller/entity_json.controller.ts +47 -47
  110. package/src/module/entity_json/entity/entityJson.entity.ts +39 -39
  111. package/src/module/entity_json/entity_json.module.ts +18 -18
  112. package/src/module/entity_json/service/entityJson.repository.ts +37 -37
  113. package/src/module/entity_json/service/entity_json.service.ts +241 -241
  114. package/src/module/export/controller/export.controller.ts +83 -83
  115. package/src/module/export/export.module.ts +14 -14
  116. package/src/module/export/service/export.service.ts +105 -105
  117. package/src/module/filter/controller/filter.controller.ts +87 -87
  118. package/src/module/filter/dto/filter-request.dto.ts +39 -39
  119. package/src/module/filter/entity/saved-filter-detail.entity.ts +41 -41
  120. package/src/module/filter/entity/saved-filter-master.entity.ts +35 -35
  121. package/src/module/filter/filter.module.ts +33 -33
  122. package/src/module/filter/repository/saved-filter.repository.ts +247 -247
  123. package/src/module/filter/repository/saved.filter-detail.repository.ts +19 -19
  124. package/src/module/filter/service/filter-evaluator.service.ts +82 -82
  125. package/src/module/filter/service/filter.service.ts +1316 -1316
  126. package/src/module/filter/service/saved-filter.service.ts +164 -164
  127. package/src/module/ics/controller/ics.controller.ts +21 -21
  128. package/src/module/ics/dto/ics.dto.ts +55 -55
  129. package/src/module/ics/ics.module.ts +13 -13
  130. package/src/module/ics/service/ics.service.ts +57 -57
  131. package/src/module/integration/controller/calender-event.controller.ts +31 -31
  132. package/src/module/integration/controller/integration.controller.ts +662 -662
  133. package/src/module/integration/controller/wrapper.controller.ts +37 -37
  134. package/src/module/integration/dto/create-config.dto.ts +526 -526
  135. package/src/module/integration/entity/integration-config.entity.ts +112 -112
  136. package/src/module/integration/entity/integration-entity-mapper.entity.ts +14 -14
  137. package/src/module/integration/entity/integration-source.entity.ts +17 -17
  138. package/src/module/integration/entity/user-integration.entity.ts +71 -71
  139. package/src/module/integration/examples/usage.example.ts +338 -338
  140. package/src/module/integration/factories/base.factory.ts +7 -7
  141. package/src/module/integration/factories/email.factory.ts +49 -49
  142. package/src/module/integration/factories/integration.factory.ts +121 -121
  143. package/src/module/integration/factories/sms.factory.ts +51 -51
  144. package/src/module/integration/factories/telephone.factory.ts +41 -41
  145. package/src/module/integration/factories/whatsapp.factory.ts +56 -56
  146. package/src/module/integration/integration.module.ts +110 -110
  147. package/src/module/integration/service/calendar-event.service.ts +118 -118
  148. package/src/module/integration/service/integration-entity-mapper.service.ts +17 -17
  149. package/src/module/integration/service/integration-queue.service.ts +229 -229
  150. package/src/module/integration/service/integration.service.ts +2634 -2634
  151. package/src/module/integration/service/oauth.service.ts +224 -224
  152. package/src/module/integration/service/wrapper.service.ts +753 -753
  153. package/src/module/integration/strategies/email/gmail-api.strategy.ts +280 -280
  154. package/src/module/integration/strategies/email/outlook-api.strategy.ts +44 -44
  155. package/src/module/integration/strategies/email/outlook.strategy.ts +64 -64
  156. package/src/module/integration/strategies/email/sendgrid-api.strategy.ts +260 -260
  157. package/src/module/integration/strategies/integration.strategy.ts +97 -97
  158. package/src/module/integration/strategies/sms/gupshup-sms.strategy.ts +146 -146
  159. package/src/module/integration/strategies/sms/msg91-sms.strategy.ts +164 -164
  160. package/src/module/integration/strategies/sms/tubelight-sms.strategy.ts +163 -163
  161. package/src/module/integration/strategies/telephone/ozonetel-voice.strategy.ts +238 -238
  162. package/src/module/integration/strategies/telephone/tubelight-voice.strategy.ts +210 -210
  163. package/src/module/integration/strategies/whatsapp/gupshup-whatsapp.strategy.ts +359 -359
  164. package/src/module/integration/strategies/whatsapp/tubelight-whatsapp.strategy.ts +372 -372
  165. package/src/module/integration/strategies/whatsapp/whatsapp-cloud.strategy.ts +403 -403
  166. package/src/module/integration/strategies/whatsapp/whatsapp.strategy.ts +57 -57
  167. package/src/module/layout/controller/layout.controller.ts +47 -47
  168. package/src/module/layout/entity/header-items.entity.ts +28 -28
  169. package/src/module/layout/entity/header-section.entity.ts +19 -19
  170. package/src/module/layout/layout.module.ts +21 -21
  171. package/src/module/layout/repository/header-items.repository.ts +18 -18
  172. package/src/module/layout/repository/header-section.repository.ts +22 -22
  173. package/src/module/layout/service/header-section.service.ts +25 -25
  174. package/src/module/layout_preference/controller/layout_preference.controller.ts +76 -76
  175. package/src/module/layout_preference/entity/layout_preference.entity.ts +28 -28
  176. package/src/module/layout_preference/layout_preference.module.ts +22 -22
  177. package/src/module/layout_preference/repository/layout_preference.repository.ts +65 -65
  178. package/src/module/layout_preference/service/layout_preference.service.ts +191 -191
  179. package/src/module/lead/controller/lead.controller.ts +30 -30
  180. package/src/module/lead/lead.module.ts +14 -14
  181. package/src/module/lead/repository/lead.repository.ts +41 -41
  182. package/src/module/lead/service/lead.service.ts +54 -54
  183. package/src/module/linked_attributes/controller/linked_attributes.controller.ts +37 -37
  184. package/src/module/linked_attributes/entity/linked_attribute.entity.ts +51 -51
  185. package/src/module/linked_attributes/linked_attributes.module.ts +16 -16
  186. package/src/module/linked_attributes/repository/linked_attribute.repository.ts +12 -12
  187. package/src/module/linked_attributes/service/linked_attributes.service.ts +73 -73
  188. package/src/module/listmaster/controller/list-master.controller.ts +230 -230
  189. package/src/module/listmaster/entity/list-master-items.entity.ts +43 -43
  190. package/src/module/listmaster/entity/list-master.entity.ts +33 -33
  191. package/src/module/listmaster/listmaster.module.ts +46 -46
  192. package/src/module/listmaster/repository/list-master-items.repository.ts +173 -173
  193. package/src/module/listmaster/repository/list-master.repository.ts +56 -56
  194. package/src/module/listmaster/service/list-master-engine.ts +19 -19
  195. package/src/module/listmaster/service/list-master-extension.interface.ts +4 -4
  196. package/src/module/listmaster/service/list-master-item.service.ts +280 -280
  197. package/src/module/listmaster/service/list-master-registry.ts +15 -15
  198. package/src/module/listmaster/service/list-master.service.ts +527 -527
  199. package/src/module/mapper/controller/field-mapper.controller.ts +76 -76
  200. package/src/module/mapper/controller/mapper.controller.ts +20 -20
  201. package/src/module/mapper/dto/field-mapper.dto.ts +14 -14
  202. package/src/module/mapper/entity/field-lovs.entity.ts +19 -19
  203. package/src/module/mapper/entity/field-mapper.entity.ts +53 -53
  204. package/src/module/mapper/entity/mapper.entity.ts +16 -16
  205. package/src/module/mapper/mapper.module.ts +35 -35
  206. package/src/module/mapper/repository/field-lovs.repository.ts +35 -35
  207. package/src/module/mapper/repository/field-mapper.repository.ts +42 -42
  208. package/src/module/mapper/repository/mapper.repository.ts +32 -32
  209. package/src/module/mapper/service/field-mapper.service.ts +269 -269
  210. package/src/module/mapper/service/mapper.service.ts +80 -80
  211. package/src/module/master/controller/master.controller.ts +74 -74
  212. package/src/module/master/service/master.service.ts +484 -484
  213. package/src/module/meta/controller/app-master.controller.ts +38 -38
  214. package/src/module/meta/controller/attribute-master.controller.ts +84 -84
  215. package/src/module/meta/controller/entity-dynamic.controller.ts +125 -125
  216. package/src/module/meta/controller/entity-master.controller.ts +41 -41
  217. package/src/module/meta/controller/entity-relation.controller.ts +36 -36
  218. package/src/module/meta/controller/entity.controller.ts +308 -308
  219. package/src/module/meta/controller/entity.public.controller.ts +75 -75
  220. package/src/module/meta/controller/media.controller.ts +135 -135
  221. package/src/module/meta/controller/meta.controller.ts +101 -101
  222. package/src/module/meta/controller/view-master.controller.ts +79 -79
  223. package/src/module/meta/dto/entity-list-data.dto.ts +6 -6
  224. package/src/module/meta/dto/entity-tab.dto.ts +4 -4
  225. package/src/module/meta/dto/entity-table.dto.ts +12 -12
  226. package/src/module/meta/entity/app-master.entity.ts +37 -37
  227. package/src/module/meta/entity/attribute-master.entity.ts +92 -92
  228. package/src/module/meta/entity/base-entity.entity.ts +75 -75
  229. package/src/module/meta/entity/entity-master.entity.ts +85 -85
  230. package/src/module/meta/entity/entity-relation-data.entity.ts +29 -29
  231. package/src/module/meta/entity/entity-relation.entity.ts +23 -23
  232. package/src/module/meta/entity/entity-table-column.entity.ts +61 -61
  233. package/src/module/meta/entity/entity-table.entity.ts +50 -50
  234. package/src/module/meta/entity/media-data.entity.ts +32 -32
  235. package/src/module/meta/entity/preference.entity.ts +62 -62
  236. package/src/module/meta/entity/view-master.entity.ts +41 -41
  237. package/src/module/meta/entity.module.ts +165 -165
  238. package/src/module/meta/repository/app-master.repository.ts +20 -20
  239. package/src/module/meta/repository/attribute-master.repository.ts +164 -164
  240. package/src/module/meta/repository/entity-attribute-update.repository.ts +48 -48
  241. package/src/module/meta/repository/entity-master.repository.ts +120 -120
  242. package/src/module/meta/repository/entity-relation.repository.ts +22 -22
  243. package/src/module/meta/repository/entity-table-column.repository.ts +39 -39
  244. package/src/module/meta/repository/entity-table.repository.ts +53 -53
  245. package/src/module/meta/repository/media-data.repository.ts +50 -50
  246. package/src/module/meta/repository/preference.repository.ts +20 -20
  247. package/src/module/meta/repository/user-app-mapping.repository.ts +28 -28
  248. package/src/module/meta/repository/view-master.repository.ts +42 -42
  249. package/src/module/meta/service/app-master.service.ts +37 -37
  250. package/src/module/meta/service/attribute-master.service.ts +132 -132
  251. package/src/module/meta/service/common.service.ts +9 -9
  252. package/src/module/meta/service/entity-attribute-update.service.ts +26 -26
  253. package/src/module/meta/service/entity-dynamic.service.ts +824 -824
  254. package/src/module/meta/service/entity-master.service.ts +171 -171
  255. package/src/module/meta/service/entity-realation-data.service.ts +9 -9
  256. package/src/module/meta/service/entity-relation.service.ts +74 -74
  257. package/src/module/meta/service/entity-service-impl.service.ts +388 -388
  258. package/src/module/meta/service/entity-table-column.service.ts +26 -26
  259. package/src/module/meta/service/entity-table.service.ts +157 -157
  260. package/src/module/meta/service/entity-validation.service.ts +188 -188
  261. package/src/module/meta/service/entity.service.ts +49 -49
  262. package/src/module/meta/service/field-group.service.ts +103 -103
  263. package/src/module/meta/service/media-data.service.ts +591 -591
  264. package/src/module/meta/service/populate-meta.service.ts +222 -222
  265. package/src/module/meta/service/preference.service.ts +16 -16
  266. package/src/module/meta/service/resolver.service.ts +280 -280
  267. package/src/module/meta/service/section-master.service.ts +104 -104
  268. package/src/module/meta/service/update-form-json.service.ts +22 -22
  269. package/src/module/meta/service/user-app-mapping.service.ts +17 -17
  270. package/src/module/meta/service/view-master.service.ts +127 -127
  271. package/src/module/microservice-client/microservice-clients.module.ts +13 -13
  272. package/src/module/microservice-client/service/microservice-client-factory.ts +37 -37
  273. package/src/module/microservice-client/service/microservice-clients.ts +4 -4
  274. package/src/module/module/controller/menu.controller.ts +15 -15
  275. package/src/module/module/controller/module-access.controller.ts +133 -133
  276. package/src/module/module/entity/menu.entity.ts +43 -43
  277. package/src/module/module/entity/module-access.entity.ts +25 -25
  278. package/src/module/module/entity/module-action.entity.ts +17 -17
  279. package/src/module/module/entity/module.entity.ts +52 -52
  280. package/src/module/module/module.module.ts +42 -42
  281. package/src/module/module/repository/menu.repository.ts +186 -186
  282. package/src/module/module/repository/module-access.repository.ts +344 -344
  283. package/src/module/module/service/menu.service.ts +82 -82
  284. package/src/module/module/service/module-access.service.ts +189 -189
  285. package/src/module/notification/controller/notification.controller.ts +58 -58
  286. package/src/module/notification/controller/otp.controller.ts +117 -117
  287. package/src/module/notification/entity/notification.entity.ts +26 -26
  288. package/src/module/notification/entity/otp.entity.ts +28 -28
  289. package/src/module/notification/firebase-admin.config.ts +22 -22
  290. package/src/module/notification/notification.module.ts +69 -69
  291. package/src/module/notification/repository/otp.repository.ts +27 -27
  292. package/src/module/notification/service/email.service.ts +127 -127
  293. package/src/module/notification/service/notification.service.ts +164 -164
  294. package/src/module/notification/service/otp.service.ts +133 -133
  295. package/src/module/third-party-module/entity/third-party-api-registry.entity.ts +52 -52
  296. package/src/module/third-party-module/repository/third-party-api-registry.repository.ts +20 -20
  297. package/src/module/third-party-module/service/api-registry.service.ts +13 -13
  298. package/src/module/third-party-module/third-party.module.ts +12 -12
  299. package/src/module/user/controller/login.controller.ts +198 -198
  300. package/src/module/user/controller/user.controller.ts +40 -40
  301. package/src/module/user/dto/create-user.dto.ts +62 -62
  302. package/src/module/user/dto/update-user.dto.ts +4 -4
  303. package/src/module/user/entity/role.entity.ts +33 -33
  304. package/src/module/user/entity/user-role-mapping.entity.ts +38 -38
  305. package/src/module/user/entity/user-session.entity.ts +73 -73
  306. package/src/module/user/entity/user.entity.ts +62 -62
  307. package/src/module/user/repository/role.repository.ts +96 -96
  308. package/src/module/user/repository/user-role-mapping.repository.ts +126 -126
  309. package/src/module/user/repository/user.repository.ts +50 -50
  310. package/src/module/user/repository/userSession.repository.ts +33 -33
  311. package/src/module/user/service/login.service.ts +326 -326
  312. package/src/module/user/service/role.service.ts +197 -197
  313. package/src/module/user/service/user-role-mapping.service.ts +98 -98
  314. package/src/module/user/service/user-session.service.ts +200 -200
  315. package/src/module/user/service/user.service.ts +368 -368
  316. package/src/module/user/user.module.ts +65 -65
  317. package/src/module/workflow/controller/action-category.controller.ts +54 -54
  318. package/src/module/workflow/controller/action-resource-mapping.controller.ts +23 -23
  319. package/src/module/workflow/controller/action-template-mapping.controller.ts +35 -35
  320. package/src/module/workflow/controller/action.controller.ts +111 -111
  321. package/src/module/workflow/controller/activity-log.controller.ts +55 -55
  322. package/src/module/workflow/controller/comm-template.controller.ts +43 -43
  323. package/src/module/workflow/controller/entity-modification.controller.ts +35 -35
  324. package/src/module/workflow/controller/form-master.controller.ts +43 -43
  325. package/src/module/workflow/controller/stage-group.controller.ts +49 -49
  326. package/src/module/workflow/controller/stage.controller.ts +51 -51
  327. package/src/module/workflow/controller/task.controller.ts +77 -77
  328. package/src/module/workflow/controller/workflow-list-master.controller.ts +44 -44
  329. package/src/module/workflow/controller/workflow-meta.controller.ts +80 -80
  330. package/src/module/workflow/controller/workflow.controller.ts +67 -67
  331. package/src/module/workflow/entity/action-category.entity.ts +38 -38
  332. package/src/module/workflow/entity/action-data.entity.ts +55 -55
  333. package/src/module/workflow/entity/action-resources-mapping.entity.ts +29 -29
  334. package/src/module/workflow/entity/action-template-mapping.entity.ts +17 -17
  335. package/src/module/workflow/entity/action.entity.ts +53 -53
  336. package/src/module/workflow/entity/activity-log.entity.ts +43 -43
  337. package/src/module/workflow/entity/comm-template.entity.ts +43 -43
  338. package/src/module/workflow/entity/entity-modification.entity.ts +38 -38
  339. package/src/module/workflow/entity/form.entity.ts +25 -25
  340. package/src/module/workflow/entity/stage-action-mapping.entity.ts +17 -17
  341. package/src/module/workflow/entity/stage-group.entity.ts +23 -23
  342. package/src/module/workflow/entity/stage-movement-data.entity.ts +38 -38
  343. package/src/module/workflow/entity/stage.entity.ts +20 -20
  344. package/src/module/workflow/entity/task-data.entity.ts +88 -88
  345. package/src/module/workflow/entity/template-attach-mapper.entity.ts +30 -30
  346. package/src/module/workflow/entity/workflow-data.entity.ts +11 -11
  347. package/src/module/workflow/entity/workflow-level-mapping.entity.ts +18 -18
  348. package/src/module/workflow/entity/workflow.entity.ts +20 -20
  349. package/src/module/workflow/repository/action-category.repository.ts +79 -79
  350. package/src/module/workflow/repository/action-data.repository.ts +347 -347
  351. package/src/module/workflow/repository/action.repository.ts +339 -339
  352. package/src/module/workflow/repository/activity-log.repository.ts +148 -148
  353. package/src/module/workflow/repository/comm-template.repository.ts +157 -157
  354. package/src/module/workflow/repository/form-master.repository.ts +50 -50
  355. package/src/module/workflow/repository/stage-group.repository.ts +186 -186
  356. package/src/module/workflow/repository/stage-movement.repository.ts +217 -217
  357. package/src/module/workflow/repository/stage.repository.ts +160 -160
  358. package/src/module/workflow/repository/task.repository.ts +154 -154
  359. package/src/module/workflow/repository/workflow.repository.ts +42 -42
  360. package/src/module/workflow/service/action-category.service.ts +33 -33
  361. package/src/module/workflow/service/action-data.service.ts +62 -62
  362. package/src/module/workflow/service/action-resources-mapping.service.ts +10 -10
  363. package/src/module/workflow/service/action-template-mapping.service.ts +137 -137
  364. package/src/module/workflow/service/action.service.ts +302 -302
  365. package/src/module/workflow/service/activity-log.service.ts +107 -107
  366. package/src/module/workflow/service/comm-template.service.ts +181 -181
  367. package/src/module/workflow/service/entity-modification.service.ts +61 -61
  368. package/src/module/workflow/service/form-master.service.ts +35 -35
  369. package/src/module/workflow/service/populate-workflow.service.ts +320 -320
  370. package/src/module/workflow/service/stage-action-mapping.service.ts +5 -5
  371. package/src/module/workflow/service/stage-group.service.ts +325 -325
  372. package/src/module/workflow/service/stage.service.ts +197 -197
  373. package/src/module/workflow/service/task.service.ts +551 -551
  374. package/src/module/workflow/service/workflow-list-master.service.ts +68 -68
  375. package/src/module/workflow/service/workflow-meta.service.ts +640 -640
  376. package/src/module/workflow/service/workflow.service.ts +213 -213
  377. package/src/module/workflow/workflow.module.ts +180 -180
  378. package/src/module/workflow-automation/SCHEDULING_GUIDE.md +145 -145
  379. package/src/module/workflow-automation/controller/workflow-automation.controller.ts +43 -43
  380. package/src/module/workflow-automation/entity/workflow-automation-action.entity.ts +26 -26
  381. package/src/module/workflow-automation/entity/workflow-automation.entity.ts +40 -40
  382. package/src/module/workflow-automation/interface/action.decorator.ts +7 -7
  383. package/src/module/workflow-automation/interface/action.interface.ts +5 -5
  384. package/src/module/workflow-automation/service/action-registery.service.ts +35 -35
  385. package/src/module/workflow-automation/service/schedule-handler.service.ts +168 -168
  386. package/src/module/workflow-automation/service/workflow-automation-engine.service.ts +219 -219
  387. package/src/module/workflow-automation/service/workflow-automation.service.ts +476 -476
  388. package/src/module/workflow-automation/workflow-automation.module.ts +54 -54
  389. package/src/module/workflow-schedule/INSTALLATION.md +244 -244
  390. package/src/module/workflow-schedule/MULTI_PROJECT_GUIDE.md +196 -196
  391. package/src/module/workflow-schedule/README.md +422 -422
  392. package/src/module/workflow-schedule/constants/schedule.constants.ts +48 -48
  393. package/src/module/workflow-schedule/controller/workflow-schedule.controller.ts +253 -253
  394. package/src/module/workflow-schedule/docs/CLAUDE_CODE_GUIDE.md +510 -510
  395. package/src/module/workflow-schedule/docs/CLAUDE_CODE_PROMPT.md +362 -362
  396. package/src/module/workflow-schedule/docs/RUN_CLAUDE_CODE.sh +68 -68
  397. package/src/module/workflow-schedule/dto/create-schedule.dto.ts +147 -147
  398. package/src/module/workflow-schedule/dto/get-execution-logs.dto.ts +119 -119
  399. package/src/module/workflow-schedule/dto/update-schedule.dto.ts +96 -96
  400. package/src/module/workflow-schedule/entities/scheduled-workflow.entity.ts +148 -148
  401. package/src/module/workflow-schedule/entities/workflow-execution-log.entity.ts +154 -154
  402. package/src/module/workflow-schedule/interfaces/schedule-job-data.interface.ts +53 -53
  403. package/src/module/workflow-schedule/interfaces/workflow-schedule-options.interface.ts +12 -12
  404. package/src/module/workflow-schedule/processors/schedule.processor.ts +620 -620
  405. package/src/module/workflow-schedule/service/workflow-schedule.service.ts +597 -597
  406. package/src/module/workflow-schedule/workflow-schedule.module.ts +67 -67
  407. package/src/resources/dev.properties.yaml +31 -31
  408. package/src/resources/local.properties.yaml +27 -27
  409. package/src/resources/properties.module.ts +12 -12
  410. package/src/resources/properties.yaml.ts +11 -11
  411. package/src/resources/uat.properties.yaml +31 -31
  412. package/src/table.config.ts +133 -133
  413. package/src/utils/dto/excel-data.dto.ts +14 -14
  414. package/src/utils/dto/excelsheet-data.dto.ts +5 -5
  415. package/src/utils/service/base64util.service.ts +18 -18
  416. package/src/utils/service/clockIDGenUtil.service.ts +21 -21
  417. package/src/utils/service/codeGenerator.service.ts +22 -22
  418. package/src/utils/service/dateUtil.service.ts +17 -17
  419. package/src/utils/service/encryptUtil.service.ts +97 -97
  420. package/src/utils/service/excel-helper.service.ts +72 -72
  421. package/src/utils/service/excelUtil.service.ts +15 -15
  422. package/src/utils/service/file-util.service.ts +11 -11
  423. package/src/utils/service/json-util.service.ts +23 -23
  424. package/src/utils/service/loggingUtil.service.ts +88 -88
  425. package/src/utils/service/reflection-helper.service.ts +62 -62
  426. package/src/utils/service/wbsCodeGen.service.ts +8 -8
  427. package/src/utils/utils.module.ts +27 -27
  428. package/tsconfig.build.json +4 -4
  429. 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
+ }