rez_core 5.0.176 → 5.0.178

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