TaskSchedulerServer.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366
  1. using Infrastructure.Model;
  2. using NLog;
  3. using Quartz;
  4. using Quartz.Impl;
  5. using Quartz.Impl.Matchers;
  6. using Quartz.Impl.Triggers;
  7. using Quartz.Spi;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Collections.Specialized;
  11. using System.Linq;
  12. using System.Reflection;
  13. using System.Threading.Tasks;
  14. using ZR.Model.System;
  15. namespace ZR.Tasks
  16. {
  17. /// <summary>
  18. /// 计划任务中心
  19. /// </summary>
  20. //[AppService]
  21. public class TaskSchedulerServer : ITaskSchedulerServer
  22. {
  23. private Task<IScheduler> _scheduler;
  24. private readonly IJobFactory _jobFactory;
  25. /// <summary>
  26. /// 日志接口
  27. /// </summary>
  28. private static Logger logger = LogManager.GetCurrentClassLogger();
  29. public TaskSchedulerServer(IJobFactory jobFactory)
  30. {
  31. _scheduler = GetTaskSchedulerAsync();
  32. _jobFactory = jobFactory;
  33. }
  34. /// <summary>
  35. /// 开启计划任务
  36. /// 参考文章:https://www.quartz-scheduler.net/documentation/quartz-3.x/configuration/reference.html#datasources-ado-net-jobstores
  37. /// </summary>
  38. /// <returns></returns>
  39. private Task<IScheduler> GetTaskSchedulerAsync()
  40. {
  41. if (_scheduler != null)
  42. {
  43. return _scheduler;
  44. }
  45. NameValueCollection collection = new NameValueCollection
  46. {
  47. { "quartz.serializer.type","binary" },
  48. //quartz参数
  49. //{ "quartz.scheduler.instanceId", "AUTO" },
  50. //{ "quartz.serializer.type","json" },
  51. ////下面为指定quartz持久化数据库的配置
  52. //{ "quartz.jobStore.type","Quartz.Impl.AdoJobStore.JobStoreTX, Quartz" },
  53. //{ "quartz.jobStore.tablePrefix","QRTZ_"},
  54. //{ "quartz.jobStore.driverDelegateType", "Quartz.Impl.AdoJobStore.MySQLDelegate, Quartz"},
  55. //{ "quartz.jobStore.useProperties", "true"},
  56. //{ "quartz.jobStore.dataSource", "myDS" },
  57. //{ "quartz.dataSource.myDS.connectionString", @"server=xxx.xxx.xxx.xxx;port=3306;database=Admin;uid=zrry;pwd=********;Charset=utf8;"},
  58. //{ "quartz.dataSource.myDS.provider", "MySql" },
  59. };
  60. StdSchedulerFactory factory = new StdSchedulerFactory(collection);
  61. return _scheduler = factory.GetScheduler();
  62. }
  63. public async Task<ApiResult> StartTaskScheduleAsync()
  64. {
  65. try
  66. {
  67. _scheduler.Result.JobFactory = _jobFactory;
  68. if (_scheduler.Result.IsStarted)
  69. {
  70. return ApiResult.Error(500, $"计划任务已经开启");
  71. }
  72. //等待任务运行完成
  73. await _scheduler.Result.Start();
  74. return ApiResult.Success("计划任务开启成功");
  75. }
  76. catch (Exception)
  77. {
  78. throw;
  79. }
  80. }
  81. /// <summary>
  82. /// 停止计划任务
  83. /// </summary>
  84. /// <returns></returns>
  85. public async Task<ApiResult> StopTaskScheduleAsync()
  86. {
  87. try
  88. {
  89. if (_scheduler.Result.IsShutdown)
  90. {
  91. return ApiResult.Error(500, $"计划任务已经停止");
  92. }
  93. await _scheduler.Result.Shutdown();
  94. return ApiResult.Success("计划任务已经停止");
  95. }
  96. catch (Exception)
  97. {
  98. throw;
  99. }
  100. }
  101. /// <summary>
  102. /// 添加一个计划任务
  103. /// </summary>
  104. /// <param name="tasksQz"></param>
  105. /// <returns></returns>
  106. public async Task<ApiResult> AddTaskScheduleAsync(SysTasks tasksQz)
  107. {
  108. try
  109. {
  110. JobKey jobKey = new JobKey(tasksQz.ID, tasksQz.JobGroup);
  111. if (await _scheduler.Result.CheckExists(jobKey))
  112. {
  113. return ApiResult.Error(500, $"该计划任务已经在执行:【{tasksQz.Name}】,请勿重复添加!");
  114. }
  115. if (tasksQz?.EndTime <= DateTime.Now)
  116. {
  117. return ApiResult.Error(500, $"结束时间小于当前时间计划将不会被执行");
  118. }
  119. #region 设置开始时间和结束时间
  120. tasksQz.BeginTime = tasksQz.BeginTime == null ? DateTime.Now : tasksQz.BeginTime;
  121. tasksQz.EndTime = tasksQz.EndTime == null ? DateTime.MaxValue.AddDays(-1) : tasksQz.EndTime;
  122. DateTimeOffset starRunTime = DateBuilder.NextGivenSecondDate(tasksQz.BeginTime, 1);//设置开始时间
  123. DateTimeOffset endRunTime = DateBuilder.NextGivenSecondDate(tasksQz.EndTime, 1);//设置暂停时间
  124. #endregion
  125. #region 通过反射获取程序集类型和类
  126. Assembly assembly = Assembly.Load(new AssemblyName(tasksQz.AssemblyName));
  127. Type jobType = assembly.GetType(tasksQz.AssemblyName + "." + tasksQz.ClassName);
  128. #endregion
  129. //2、开启调度器。判断任务调度是否开启
  130. if (!_scheduler.Result.IsStarted)
  131. {
  132. await StartTaskScheduleAsync();
  133. }
  134. //3、创建任务。传入反射出来的执行程序集
  135. IJobDetail job = new JobDetailImpl(tasksQz.ID, tasksQz.JobGroup, jobType);
  136. job.JobDataMap.Add("JobParam", tasksQz.JobParams);
  137. ITrigger trigger;
  138. //4、创建一个触发器
  139. if (tasksQz.Cron != null && CronExpression.IsValidExpression(tasksQz.Cron))
  140. {
  141. trigger = CreateCronTrigger(tasksQz);
  142. //解决Quartz启动后第一次会立即执行问题解决办法
  143. ((CronTriggerImpl)trigger).MisfireInstruction = MisfireInstruction.CronTrigger.DoNothing;
  144. }
  145. else
  146. {
  147. trigger = CreateSimpleTrigger(tasksQz);
  148. ((SimpleTriggerImpl)trigger).MisfireInstruction = MisfireInstruction.CronTrigger.DoNothing;
  149. }
  150. // 5、将触发器和任务器绑定到调度器中
  151. await _scheduler.Result.ScheduleJob(job, trigger);
  152. //任务没有启动、暂停任务
  153. //if (!tasksQz.IsStart)
  154. //{
  155. // _scheduler.Result.PauseJob(jobKey).Wait();
  156. //}
  157. //按新的trigger重新设置job执行
  158. await _scheduler.Result.ResumeTrigger(trigger.Key);
  159. return ApiResult.Success($"启动计划任务:【{tasksQz.Name}】成功!");
  160. }
  161. catch (Exception ex)
  162. {
  163. logger.Error(ex, $"启动计划任务失败,分组:{tasksQz.JobGroup},作业:【{tasksQz.Name}】,error:{ex.Message}");
  164. return ApiResult.Error(500, $"启动计划任务:【{tasksQz.Name}】失败!");
  165. }
  166. }
  167. /// <summary>
  168. /// 暂停指定的计划任务
  169. /// </summary>
  170. /// <param name="tasksQz"></param>
  171. /// <returns></returns>
  172. public async Task<ApiResult> PauseTaskScheduleAsync(SysTasks tasksQz)
  173. {
  174. try
  175. {
  176. JobKey jobKey = new JobKey(tasksQz.ID, tasksQz.JobGroup);
  177. if (await _scheduler.Result.CheckExists(jobKey))
  178. {
  179. // 防止创建时存在数据问题 先移除,然后在执行创建操作
  180. await _scheduler.Result.PauseJob(jobKey);
  181. }
  182. return ApiResult.Success($"暂停计划任务:【{tasksQz.Name}】成功");
  183. }
  184. catch (Exception ex)
  185. {
  186. logger.Error(ex);
  187. return new ApiResult(500, $"暂停计划任务:【{tasksQz.Name}】失败,{ex.Message}");
  188. }
  189. }
  190. /// <summary>
  191. /// 恢复指定计划任务
  192. /// </summary>
  193. /// <param name="tasksQz"></param>
  194. /// <returns></returns>
  195. public async Task<ApiResult> ResumeTaskScheduleAsync(SysTasks tasksQz)
  196. {
  197. try
  198. {
  199. JobKey jobKey = new JobKey(tasksQz.ID, tasksQz.JobGroup);
  200. if (!await _scheduler.Result.CheckExists(jobKey))
  201. {
  202. return ApiResult.Error(500, $"未找到计划任务:【{tasksQz.Name}】");
  203. }
  204. await _scheduler.Result.ResumeJob(jobKey);
  205. return ApiResult.Success($"恢复计划任务:【{tasksQz.Name}】成功");
  206. }
  207. catch (Exception)
  208. {
  209. throw;
  210. }
  211. }
  212. /// <summary>
  213. /// 删除指定计划任务
  214. /// </summary>
  215. /// <param name="tasksQz"></param>
  216. /// <returns></returns>
  217. public async Task<ApiResult> DeleteTaskScheduleAsync(SysTasks tasksQz)
  218. {
  219. try
  220. {
  221. JobKey jobKey = new JobKey(tasksQz.ID, tasksQz.JobGroup);
  222. //await _scheduler.Result.ScheduleJob()
  223. await _scheduler.Result.DeleteJob(jobKey);
  224. return ApiResult.Success($"删除计划任务:【{tasksQz.Name}】成功");
  225. }
  226. catch (Exception ex)
  227. {
  228. logger.Error(ex);
  229. return ApiResult.Error($"删除计划任务:【{tasksQz.Name}】失败, error={ex.Message}");
  230. }
  231. }
  232. /// <summary>
  233. /// 立即运行
  234. /// </summary>
  235. /// <param name="tasksQz"></param>
  236. /// <returns></returns>
  237. public async Task<ApiResult> RunTaskScheduleAsync(SysTasks tasksQz)
  238. {
  239. try
  240. {
  241. JobKey jobKey = new JobKey(tasksQz.ID, tasksQz.JobGroup);
  242. List<JobKey> jobKeys = _scheduler.Result.GetJobKeys(GroupMatcher<JobKey>.GroupEquals(tasksQz.JobGroup)).Result.ToList();
  243. if (jobKeys == null || jobKeys.Count == 0)
  244. {
  245. await AddTaskScheduleAsync(tasksQz);
  246. }
  247. var triggers = await _scheduler.Result.GetTriggersOfJob(jobKey);
  248. if (triggers.Count <= 0)
  249. {
  250. return new ApiResult(110, $"未找到触发器[{jobKey.Name}]");
  251. }
  252. await _scheduler.Result.TriggerJob(jobKey);
  253. return ApiResult.Success($"运行计划任务:【{tasksQz.Name}】成功");
  254. }
  255. catch (Exception ex)
  256. {
  257. return new ApiResult(500, $"执行计划任务:【{tasksQz.Name}】失败,{ex.Message}");
  258. }
  259. }
  260. /// <summary>
  261. /// 更新计划任务
  262. /// </summary>
  263. /// <param name="tasksQz"></param>
  264. /// <returns></returns>
  265. public async Task<ApiResult> UpdateTaskScheduleAsync(SysTasks tasksQz)
  266. {
  267. try
  268. {
  269. JobKey jobKey = new JobKey(tasksQz.ID, tasksQz.JobGroup);
  270. if (await _scheduler.Result.CheckExists(jobKey))
  271. {
  272. //防止创建时存在数据问题 先移除,然后在执行创建操作
  273. await _scheduler.Result.DeleteJob(jobKey);
  274. }
  275. //await AddTaskScheduleAsync(tasksQz);
  276. return ApiResult.Success("修改计划成功");
  277. }
  278. catch (Exception ex)
  279. {
  280. logger.Error(ex);
  281. return ApiResult.Error($"修改计划失败,error={ex.Message}");
  282. }
  283. }
  284. #region 创建触发器帮助方法
  285. /// <summary>
  286. /// 创建SimpleTrigger触发器(简单触发器)
  287. /// </summary>
  288. /// <param name="tasksQz"></param>
  289. /// <returns></returns>
  290. private ITrigger CreateSimpleTrigger(SysTasks tasksQz)
  291. {
  292. if (tasksQz.RunTimes > 0)
  293. {
  294. ITrigger trigger = TriggerBuilder.Create()
  295. .WithIdentity(tasksQz.ID, tasksQz.JobGroup)
  296. .StartAt(tasksQz.BeginTime.Value)
  297. .EndAt(tasksQz.EndTime.Value)
  298. .WithSimpleSchedule(x => x.WithIntervalInSeconds(tasksQz.IntervalSecond)
  299. .WithRepeatCount(tasksQz.RunTimes)).ForJob(tasksQz.ID, tasksQz.JobGroup).Build();
  300. return trigger;
  301. }
  302. else
  303. {
  304. ITrigger trigger = TriggerBuilder.Create()
  305. .WithIdentity(tasksQz.ID, tasksQz.JobGroup)
  306. .StartAt(tasksQz.BeginTime.Value)
  307. .EndAt(tasksQz.EndTime.Value)
  308. .WithSimpleSchedule(x => x.WithIntervalInSeconds(tasksQz.IntervalSecond)
  309. .RepeatForever()).ForJob(tasksQz.ID, tasksQz.JobGroup).Build();
  310. return trigger;
  311. }
  312. // 触发作业立即运行,然后每10秒重复一次,无限循环
  313. }
  314. /// <summary>
  315. /// 创建类型Cron的触发器
  316. /// </summary>
  317. /// <param name="tasksQz"></param>
  318. /// <returns></returns>
  319. private ITrigger CreateCronTrigger(SysTasks tasksQz)
  320. {
  321. // 作业触发器
  322. return TriggerBuilder.Create()
  323. .WithIdentity(tasksQz.ID, tasksQz.JobGroup)
  324. .StartAt(tasksQz.BeginTime.Value)//开始时间
  325. .EndAt(tasksQz.EndTime.Value)//结束数据
  326. .WithCronSchedule(tasksQz.Cron)//指定cron表达式
  327. .ForJob(tasksQz.ID, tasksQz.JobGroup)//作业名称
  328. .Build();
  329. }
  330. #endregion
  331. }
  332. }