SchedulerCenter.cs 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617
  1. using Host.Common;
  2. using Host.Common.Enums;
  3. using Host.Entity;
  4. using Host.IJobs;
  5. using Host.Repositories;
  6. using Microsoft.Extensions.Configuration;
  7. using Newtonsoft.Json;
  8. using Quartz;
  9. using Quartz.Impl;
  10. using Quartz.Impl.AdoJobStore;
  11. using Quartz.Impl.AdoJobStore.Common;
  12. using Quartz.Impl.Matchers;
  13. using Quartz.Impl.Triggers;
  14. using Quartz.Simpl;
  15. using Quartz.Util;
  16. using Serilog;
  17. using System;
  18. using System.Collections.Generic;
  19. using System.Linq;
  20. using System.Threading.Tasks;
  21. using Talk.Extensions;
  22. namespace Host
  23. {
  24. /// <summary>
  25. /// 调度中心 [单例]
  26. /// </summary>
  27. public class SchedulerCenter
  28. {
  29. /// <summary>
  30. /// 数据连接
  31. /// </summary>
  32. private IDbProvider dbProvider;
  33. /// <summary>
  34. /// ADO 数据类型
  35. /// </summary>
  36. private string driverDelegateType;
  37. public SchedulerCenter()
  38. {
  39. InitDriverDelegateType();
  40. dbProvider = new DbProvider(AppConfig.DbProviderName, AppConfig.ConnectionString);
  41. }
  42. /// <summary>
  43. /// 初始化DriverDelegateType
  44. /// </summary>
  45. private void InitDriverDelegateType()
  46. {
  47. switch (AppConfig.DbProviderName)
  48. {
  49. case "SQLite-Microsoft":
  50. case "SQLite":
  51. driverDelegateType = typeof(SQLiteDelegate).AssemblyQualifiedName;
  52. break;
  53. case "MySql":
  54. driverDelegateType = typeof(MySQLDelegate).AssemblyQualifiedName;
  55. break;
  56. case "OracleODPManaged":
  57. driverDelegateType = typeof(OracleDelegate).AssemblyQualifiedName;
  58. break;
  59. case "SqlServer":
  60. case "SQLServerMOT":
  61. driverDelegateType = typeof(SqlServerDelegate).AssemblyQualifiedName;
  62. break;
  63. case "Npgsql":
  64. driverDelegateType = typeof(PostgreSQLDelegate).AssemblyQualifiedName;
  65. break;
  66. case "Firebird":
  67. driverDelegateType = typeof(FirebirdDelegate).AssemblyQualifiedName;
  68. break;
  69. default:
  70. throw new Exception("dbProviderName unreasonable");
  71. }
  72. }
  73. /// <summary>
  74. /// 初始化数据库表
  75. /// </summary>
  76. /// <returns></returns>
  77. private async Task InitDBTableAsync()
  78. {
  79. //如果不存在sqlite数据库,则创建
  80. //TODO 其他数据源...
  81. if (driverDelegateType.Equals(typeof(SQLiteDelegate).AssemblyQualifiedName) ||
  82. driverDelegateType.Equals(typeof(MySQLDelegate).AssemblyQualifiedName) ||
  83. driverDelegateType.Equals(typeof(SqlServerDelegate).AssemblyQualifiedName) ||
  84. driverDelegateType.Equals(typeof(PostgreSQLDelegate).AssemblyQualifiedName))
  85. {
  86. IRepositorie repositorie = RepositorieFactory.CreateRepositorie(driverDelegateType, dbProvider);
  87. await repositorie?.InitTable();
  88. }
  89. }
  90. /// <summary>
  91. /// 调度器
  92. /// </summary>
  93. private IScheduler scheduler;
  94. /// <summary>
  95. /// 初始化Scheduler
  96. /// </summary>
  97. private async Task InitSchedulerAsync()
  98. {
  99. if (scheduler == null)
  100. {
  101. DBConnectionManager.Instance.AddConnectionProvider("default", dbProvider);
  102. var serializer = new JsonObjectSerializer();
  103. serializer.Initialize();
  104. var jobStore = new JobStoreTX
  105. {
  106. DataSource = "default",
  107. TablePrefix = "QRTZ_",
  108. InstanceId = "AUTO",
  109. DriverDelegateType = driverDelegateType,
  110. ObjectSerializer = serializer,
  111. };
  112. DirectSchedulerFactory.Instance.CreateScheduler("bennyScheduler", "AUTO", new DefaultThreadPool(), jobStore);
  113. scheduler = await SchedulerRepository.Instance.Lookup("bennyScheduler");
  114. }
  115. }
  116. /// <summary>
  117. /// 开启调度器
  118. /// </summary>
  119. /// <returns></returns>
  120. public async Task<bool> StartScheduleAsync()
  121. {
  122. //初始化数据库表结构
  123. await InitDBTableAsync();
  124. //初始化Scheduler
  125. await InitSchedulerAsync();
  126. //开启调度器
  127. if (scheduler.InStandbyMode)
  128. {
  129. await scheduler.Start();
  130. Log.Information("任务调度启动!");
  131. }
  132. return scheduler.InStandbyMode;
  133. }
  134. /// <summary>
  135. /// 添加一个工作调度
  136. /// </summary>
  137. /// <param name="entity"></param>
  138. /// <param name="runNumber"></param>
  139. /// <returns></returns>
  140. public async Task<BaseResult> AddScheduleJobAsync(ScheduleEntity entity, long? runNumber = null)
  141. {
  142. var result = new BaseResult();
  143. try
  144. {
  145. //检查任务是否已存在
  146. var jobKey = new JobKey(entity.JobName, entity.JobGroup);
  147. if (await scheduler.CheckExists(jobKey))
  148. {
  149. result.Code = 500;
  150. result.Msg = "任务已存在";
  151. return result;
  152. }
  153. //http请求配置
  154. var httpDir = new Dictionary<string, string>()
  155. {
  156. { Constant.EndAt, entity.EndTime.ToString()},
  157. { Constant.JobTypeEnum, ((int)entity.JobType).ToString()},
  158. { Constant.MAILMESSAGE, ((int)entity.MailMessage).ToString()},
  159. };
  160. if (runNumber.HasValue)
  161. httpDir.Add(Constant.RUNNUMBER, runNumber.ToString());
  162. IJobConfigurator jobConfigurator = null;
  163. if (entity.JobType == JobTypeEnum.Url)
  164. {
  165. jobConfigurator = JobBuilder.Create<HttpJob>();
  166. httpDir.Add(Constant.REQUESTURL, entity.RequestUrl);
  167. httpDir.Add(Constant.HEADERS, entity.Headers);
  168. httpDir.Add(Constant.REQUESTPARAMETERS, entity.RequestParameters);
  169. httpDir.Add(Constant.REQUESTTYPE, ((int)entity.RequestType).ToString());
  170. }
  171. else if (entity.JobType == JobTypeEnum.Emial)
  172. {
  173. jobConfigurator = JobBuilder.Create<MailJob>();
  174. httpDir.Add(Constant.MailTitle, entity.MailTitle);
  175. httpDir.Add(Constant.MailContent, entity.MailContent);
  176. httpDir.Add(Constant.MailTo, entity.MailTo);
  177. }
  178. else if (entity.JobType == JobTypeEnum.Mqtt)
  179. {
  180. jobConfigurator = JobBuilder.Create<MqttJob>();
  181. httpDir.Add(Constant.Topic, entity.Topic);
  182. httpDir.Add(Constant.Payload, entity.Payload);
  183. }
  184. else if (entity.JobType == JobTypeEnum.RabbitMQ)
  185. {
  186. jobConfigurator = JobBuilder.Create<RabbitJob>();
  187. httpDir.Add(Constant.RabbitQueue, entity.RabbitQueue);
  188. httpDir.Add(Constant.RabbitBody, entity.RabbitBody);
  189. }
  190. // 定义这个工作,并将其绑定到我们的IJob实现类
  191. IJobDetail job = jobConfigurator
  192. .SetJobData(new JobDataMap(httpDir))
  193. .WithDescription(entity.Description)
  194. .WithIdentity(entity.JobName, entity.JobGroup)
  195. .Build();
  196. // 创建触发器
  197. ITrigger trigger;
  198. //校验是否正确的执行周期表达式
  199. if (entity.TriggerType == TriggerTypeEnum.Cron)//CronExpression.IsValidExpression(entity.Cron))
  200. {
  201. trigger = CreateCronTrigger(entity);
  202. }
  203. else
  204. {
  205. trigger = CreateSimpleTrigger(entity);
  206. }
  207. // 告诉Quartz使用我们的触发器来安排作业
  208. await scheduler.ScheduleJob(job, trigger);
  209. result.Code = 200;
  210. }
  211. catch (Exception ex)
  212. {
  213. result.Code = 505;
  214. result.Msg = ex.Message;
  215. }
  216. return result;
  217. }
  218. /// <summary>
  219. /// 暂停/删除 指定的计划
  220. /// </summary>
  221. /// <param name="jobGroup">任务分组</param>
  222. /// <param name="jobName">任务名称</param>
  223. /// <param name="isDelete">停止并删除任务</param>
  224. /// <returns></returns>
  225. public async Task<BaseResult> StopOrDelScheduleJobAsync(string jobGroup, string jobName, bool isDelete = false)
  226. {
  227. BaseResult result;
  228. try
  229. {
  230. await scheduler.PauseJob(new JobKey(jobName, jobGroup));
  231. if (isDelete)
  232. {
  233. await scheduler.DeleteJob(new JobKey(jobName, jobGroup));
  234. result = new BaseResult
  235. {
  236. Code = 200,
  237. Msg = "删除任务计划成功!"
  238. };
  239. }
  240. else
  241. {
  242. result = new BaseResult
  243. {
  244. Code = 200,
  245. Msg = "停止任务计划成功!"
  246. };
  247. }
  248. }
  249. catch (Exception ex)
  250. {
  251. result = new BaseResult
  252. {
  253. Code = 505,
  254. Msg = "停止任务计划失败" + ex.Message
  255. };
  256. }
  257. return result;
  258. }
  259. /// <summary>
  260. /// 恢复运行暂停的任务
  261. /// </summary>
  262. /// <param name="jobName">任务名称</param>
  263. /// <param name="jobGroup">任务分组</param>
  264. public async Task<BaseResult> ResumeJobAsync(string jobGroup, string jobName)
  265. {
  266. BaseResult result = new BaseResult();
  267. try
  268. {
  269. //检查任务是否存在
  270. var jobKey = new JobKey(jobName, jobGroup);
  271. if (await scheduler.CheckExists(jobKey))
  272. {
  273. var jobDetail = await scheduler.GetJobDetail(jobKey);
  274. var endTime = jobDetail.JobDataMap.GetString("EndAt");
  275. if (!string.IsNullOrWhiteSpace(endTime) && DateTime.Parse(endTime) <= DateTime.Now)
  276. {
  277. result.Code = 500;
  278. result.Msg = "Job的结束时间已过期。";
  279. }
  280. else
  281. {
  282. //任务已经存在则暂停任务
  283. await scheduler.ResumeJob(jobKey);
  284. result.Msg = "恢复任务计划成功!";
  285. Log.Information(string.Format("任务“{0}”恢复运行", jobName));
  286. }
  287. }
  288. else
  289. {
  290. result.Code = 500;
  291. result.Msg = "任务不存在";
  292. }
  293. }
  294. catch (Exception ex)
  295. {
  296. result.Msg = "恢复任务计划失败!";
  297. result.Code = 500;
  298. Log.Error(string.Format("恢复任务失败!{0}", ex));
  299. }
  300. return result;
  301. }
  302. /// <summary>
  303. /// 查询任务
  304. /// </summary>
  305. /// <param name="jobGroup"></param>
  306. /// <param name="jobName"></param>
  307. /// <returns></returns>
  308. public async Task<ScheduleEntity> QueryJobAsync(string jobGroup, string jobName)
  309. {
  310. var entity = new ScheduleEntity();
  311. var jobKey = new JobKey(jobName, jobGroup);
  312. var jobDetail = await scheduler.GetJobDetail(jobKey);
  313. var triggersList = await scheduler.GetTriggersOfJob(jobKey);
  314. var triggers = triggersList.AsEnumerable().FirstOrDefault();
  315. var intervalSeconds = (triggers as SimpleTriggerImpl)?.RepeatInterval.TotalSeconds;
  316. var endTime = jobDetail.JobDataMap.GetString("EndAt");
  317. entity.BeginTime = triggers.StartTimeUtc.LocalDateTime;
  318. if (!string.IsNullOrWhiteSpace(endTime)) entity.EndTime = DateTime.Parse(endTime);
  319. if (intervalSeconds.HasValue) entity.IntervalSecond = Convert.ToInt32(intervalSeconds.Value);
  320. entity.JobGroup = jobGroup;
  321. entity.JobName = jobName;
  322. entity.Cron = (triggers as CronTriggerImpl)?.CronExpressionString;
  323. entity.RunTimes = (triggers as SimpleTriggerImpl)?.RepeatCount;
  324. entity.TriggerType = triggers is SimpleTriggerImpl ? TriggerTypeEnum.Simple : TriggerTypeEnum.Cron;
  325. entity.MailMessage = (MailMessageEnum)int.Parse(jobDetail.JobDataMap.GetString(Constant.MAILMESSAGE) ?? "0");
  326. entity.Description = jobDetail.Description;
  327. //旧代码没有保存JobTypeEnum,所以None可以默认为Url。
  328. entity.JobType = (JobTypeEnum)int.Parse(jobDetail.JobDataMap.GetString(Constant.JobTypeEnum) ?? "1");
  329. switch (entity.JobType)
  330. {
  331. case JobTypeEnum.None:
  332. break;
  333. case JobTypeEnum.Url:
  334. entity.RequestUrl = jobDetail.JobDataMap.GetString(Constant.REQUESTURL);
  335. entity.RequestType = (RequestTypeEnum)int.Parse(jobDetail.JobDataMap.GetString(Constant.REQUESTTYPE));
  336. entity.RequestParameters = jobDetail.JobDataMap.GetString(Constant.REQUESTPARAMETERS);
  337. entity.Headers = jobDetail.JobDataMap.GetString(Constant.HEADERS);
  338. break;
  339. case JobTypeEnum.Emial:
  340. entity.MailTitle = jobDetail.JobDataMap.GetString(Constant.MailTitle);
  341. entity.MailContent = jobDetail.JobDataMap.GetString(Constant.MailContent);
  342. entity.MailTo = jobDetail.JobDataMap.GetString(Constant.MailTo);
  343. break;
  344. case JobTypeEnum.Mqtt:
  345. entity.Payload = jobDetail.JobDataMap.GetString(Constant.Payload);
  346. entity.Topic = jobDetail.JobDataMap.GetString(Constant.Topic);
  347. break;
  348. case JobTypeEnum.RabbitMQ:
  349. entity.RabbitQueue = jobDetail.JobDataMap.GetString(Constant.RabbitQueue);
  350. entity.RabbitBody = jobDetail.JobDataMap.GetString(Constant.RabbitBody);
  351. break;
  352. case JobTypeEnum.Hotreload:
  353. break;
  354. default:
  355. break;
  356. }
  357. return entity;
  358. }
  359. /// <summary>
  360. /// 立即执行
  361. /// </summary>
  362. /// <param name="jobKey"></param>
  363. /// <returns></returns>
  364. public async Task<bool> TriggerJobAsync(JobKey jobKey)
  365. {
  366. await scheduler.TriggerJob(jobKey);
  367. return true;
  368. }
  369. /// <summary>
  370. /// 获取job日志
  371. /// </summary>
  372. /// <param name="jobKey"></param>
  373. /// <returns></returns>
  374. public async Task<List<string>> GetJobLogsAsync(JobKey jobKey)
  375. {
  376. var jobDetail = await scheduler.GetJobDetail(jobKey);
  377. return jobDetail.JobDataMap[Constant.LOGLIST] as List<string>;
  378. }
  379. /// <summary>
  380. /// 获取运行次数
  381. /// </summary>
  382. /// <param name="jobKey"></param>
  383. /// <returns></returns>
  384. public async Task<long> GetRunNumberAsync(JobKey jobKey)
  385. {
  386. var jobDetail = await scheduler.GetJobDetail(jobKey);
  387. return jobDetail.JobDataMap.GetLong(Constant.RUNNUMBER);
  388. }
  389. /// <summary>
  390. /// 获取所有Job(详情信息 - 初始化页面调用)
  391. /// </summary>
  392. /// <returns></returns>
  393. public async Task<List<JobInfoEntity>> GetAllJobAsync()
  394. {
  395. List<JobKey> jboKeyList = new List<JobKey>();
  396. List<JobInfoEntity> jobInfoList = new List<JobInfoEntity>();
  397. var groupNames = await scheduler.GetJobGroupNames();
  398. foreach (var groupName in groupNames.OrderBy(t => t))
  399. {
  400. jboKeyList.AddRange(await scheduler.GetJobKeys(GroupMatcher<JobKey>.GroupEquals(groupName)));
  401. jobInfoList.Add(new JobInfoEntity() { GroupName = groupName });
  402. }
  403. foreach (var jobKey in jboKeyList.OrderBy(t => t.Name))
  404. {
  405. var jobDetail = await scheduler.GetJobDetail(jobKey);
  406. var triggersList = await scheduler.GetTriggersOfJob(jobKey);
  407. var triggers = triggersList.AsEnumerable().FirstOrDefault();
  408. var interval = string.Empty;
  409. if (triggers is SimpleTriggerImpl)
  410. interval = (triggers as SimpleTriggerImpl)?.RepeatInterval.ToString();
  411. else
  412. interval = (triggers as CronTriggerImpl)?.CronExpressionString;
  413. foreach (var jobInfo in jobInfoList)
  414. {
  415. if (jobInfo.GroupName == jobKey.Group)
  416. {
  417. //旧代码没有保存JobTypeEnum,所以None可以默认为Url。
  418. var jobType = (JobTypeEnum)jobDetail.JobDataMap.GetLong(Constant.JobTypeEnum);
  419. jobType = jobType == JobTypeEnum.None ? JobTypeEnum.Url : jobType;
  420. var triggerAddress = string.Empty;
  421. if (jobType == JobTypeEnum.Url)
  422. triggerAddress = jobDetail.JobDataMap.GetString(Constant.REQUESTURL);
  423. else if (jobType == JobTypeEnum.Emial)
  424. triggerAddress = jobDetail.JobDataMap.GetString(Constant.MailTo);
  425. else if (jobType == JobTypeEnum.Mqtt)
  426. triggerAddress = jobDetail.JobDataMap.GetString(Constant.Topic);
  427. else if (jobType == JobTypeEnum.RabbitMQ)
  428. triggerAddress = jobDetail.JobDataMap.GetString(Constant.RabbitQueue);
  429. //Constant.MailTo
  430. jobInfo.JobInfoList.Add(new JobInfo()
  431. {
  432. Name = jobKey.Name,
  433. LastErrMsg = jobDetail.JobDataMap.GetString(Constant.EXCEPTION),
  434. TriggerAddress = triggerAddress,
  435. TriggerState = await scheduler.GetTriggerState(triggers.Key),
  436. PreviousFireTime = triggers.GetPreviousFireTimeUtc()?.LocalDateTime,
  437. NextFireTime = triggers.GetNextFireTimeUtc()?.LocalDateTime,
  438. BeginTime = triggers.StartTimeUtc.LocalDateTime,
  439. Interval = interval,
  440. EndTime = triggers.EndTimeUtc?.LocalDateTime,
  441. Description = jobDetail.Description,
  442. RequestType = jobDetail.JobDataMap.GetString(Constant.REQUESTTYPE),
  443. RunNumber = jobDetail.JobDataMap.GetLong(Constant.RUNNUMBER),
  444. JobType = (long)jobType
  445. //(triggers as SimpleTriggerImpl)?.TimesTriggered
  446. //CronTriggerImpl 中没有 TimesTriggered 所以自己RUNNUMBER记录
  447. });
  448. continue;
  449. }
  450. }
  451. }
  452. return jobInfoList;
  453. }
  454. /// <summary>
  455. /// 移除异常信息
  456. /// 因为只能在IJob持久化操作JobDataMap,所有这里直接暴力操作数据库。
  457. /// </summary>
  458. /// <param name="jobGroup"></param>
  459. /// <param name="jobName"></param>
  460. /// <returns></returns>
  461. public async Task<bool> RemoveErrLog(string jobGroup, string jobName)
  462. {
  463. IRepositorie logRepositorie = RepositorieFactory.CreateRepositorie(driverDelegateType, dbProvider);
  464. if (logRepositorie == null) return false;
  465. await logRepositorie.RemoveErrLogAsync(jobGroup, jobName);
  466. var jobKey = new JobKey(jobName, jobGroup);
  467. var jobDetail = await scheduler.GetJobDetail(jobKey);
  468. jobDetail.JobDataMap[Constant.EXCEPTION] = string.Empty;
  469. return true;
  470. }
  471. /// <summary>
  472. /// 获取所有Job信息(简要信息 - 刷新数据的时候使用)
  473. /// </summary>
  474. /// <returns></returns>
  475. public async Task<List<JobBriefInfoEntity>> GetAllJobBriefInfoAsync()
  476. {
  477. List<JobKey> jboKeyList = new List<JobKey>();
  478. List<JobBriefInfoEntity> jobInfoList = new List<JobBriefInfoEntity>();
  479. var groupNames = await scheduler.GetJobGroupNames();
  480. foreach (var groupName in groupNames.OrderBy(t => t))
  481. {
  482. jboKeyList.AddRange(await scheduler.GetJobKeys(GroupMatcher<JobKey>.GroupEquals(groupName)));
  483. jobInfoList.Add(new JobBriefInfoEntity() { GroupName = groupName });
  484. }
  485. foreach (var jobKey in jboKeyList.OrderBy(t => t.Name))
  486. {
  487. var jobDetail = await scheduler.GetJobDetail(jobKey);
  488. var triggersList = await scheduler.GetTriggersOfJob(jobKey);
  489. var triggers = triggersList.AsEnumerable().FirstOrDefault();
  490. foreach (var jobInfo in jobInfoList)
  491. {
  492. if (jobInfo.GroupName == jobKey.Group)
  493. {
  494. jobInfo.JobInfoList.Add(new JobBriefInfo()
  495. {
  496. Name = jobKey.Name,
  497. LastErrMsg = jobDetail?.JobDataMap.GetString(Constant.EXCEPTION),
  498. TriggerState = await scheduler.GetTriggerState(triggers.Key),
  499. PreviousFireTime = triggers.GetPreviousFireTimeUtc()?.LocalDateTime,
  500. NextFireTime = triggers.GetNextFireTimeUtc()?.LocalDateTime,
  501. RunNumber = jobDetail?.JobDataMap.GetLong(Constant.RUNNUMBER) ?? 0
  502. });
  503. continue;
  504. }
  505. }
  506. }
  507. return jobInfoList;
  508. }
  509. /// <summary>
  510. /// 停止任务调度
  511. /// </summary>
  512. public async Task<bool> StopScheduleAsync()
  513. {
  514. //判断调度是否已经关闭
  515. if (!scheduler.InStandbyMode)
  516. {
  517. //等待任务运行完成
  518. await scheduler.Standby(); //TODO 注意:Shutdown后Start会报错,所以这里使用暂停。
  519. Log.Information("任务调度暂停!");
  520. }
  521. return !scheduler.InStandbyMode;
  522. }
  523. /// <summary>
  524. /// 创建类型Simple的触发器
  525. /// </summary>
  526. /// <param name="entity"></param>
  527. /// <returns></returns>
  528. private ITrigger CreateSimpleTrigger(ScheduleEntity entity)
  529. {
  530. //作业触发器
  531. if (entity.RunTimes.HasValue && entity.RunTimes > 0)
  532. {
  533. return TriggerBuilder.Create()
  534. .WithIdentity(entity.JobName, entity.JobGroup)
  535. .StartAt(entity.BeginTime)//开始时间
  536. //.EndAt(entity.EndTime)//结束数据
  537. .WithSimpleSchedule(x =>
  538. {
  539. x.WithIntervalInSeconds(entity.IntervalSecond.Value)//执行时间间隔,单位秒
  540. .WithRepeatCount(entity.RunTimes.Value)//执行次数、默认从0开始
  541. .WithMisfireHandlingInstructionFireNow();
  542. })
  543. .ForJob(entity.JobName, entity.JobGroup)//作业名称
  544. .Build();
  545. }
  546. else
  547. {
  548. return TriggerBuilder.Create()
  549. .WithIdentity(entity.JobName, entity.JobGroup)
  550. .StartAt(entity.BeginTime)//开始时间
  551. //.EndAt(entity.EndTime)//结束数据
  552. .WithSimpleSchedule(x =>
  553. {
  554. x.WithIntervalInSeconds(entity.IntervalSecond.Value)//执行时间间隔,单位秒
  555. .RepeatForever()//无限循环
  556. .WithMisfireHandlingInstructionFireNow();
  557. })
  558. .ForJob(entity.JobName, entity.JobGroup)//作业名称
  559. .Build();
  560. }
  561. }
  562. /// <summary>
  563. /// 创建类型Cron的触发器
  564. /// </summary>
  565. /// <param name="entity"></param>
  566. /// <returns></returns>
  567. private ITrigger CreateCronTrigger(ScheduleEntity entity)
  568. {
  569. // 作业触发器
  570. return TriggerBuilder.Create()
  571. .WithIdentity(entity.JobName, entity.JobGroup)
  572. .StartAt(entity.BeginTime)//开始时间
  573. //.EndAt(entity.EndTime)//结束时间
  574. .WithCronSchedule(entity.Cron, cronScheduleBuilder => cronScheduleBuilder.WithMisfireHandlingInstructionFireAndProceed())//指定cron表达式
  575. .ForJob(entity.JobName, entity.JobGroup)//作业名称
  576. .Build();
  577. }
  578. }
  579. }