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