SysJobService.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. namespace Admin.NET.Core.Service;
  7. /// <summary>
  8. /// 系统作业任务服务 🧩
  9. /// </summary>
  10. [ApiDescriptionSettings(Order = 320, Description = "作业任务")]
  11. public class SysJobService : IDynamicApiController, ITransient
  12. {
  13. private readonly SqlSugarRepository<SysJobDetail> _sysJobDetailRep;
  14. private readonly SqlSugarRepository<SysJobTrigger> _sysJobTriggerRep;
  15. private readonly SqlSugarRepository<SysJobTriggerRecord> _sysJobTriggerRecordRep;
  16. private readonly SqlSugarRepository<SysJobCluster> _sysJobClusterRep;
  17. private readonly ISchedulerFactory _schedulerFactory;
  18. private readonly DynamicJobCompiler _dynamicJobCompiler;
  19. public SysJobService(SqlSugarRepository<SysJobDetail> sysJobDetailRep,
  20. SqlSugarRepository<SysJobTrigger> sysJobTriggerRep,
  21. SqlSugarRepository<SysJobTriggerRecord> sysJobTriggerRecordRep,
  22. SqlSugarRepository<SysJobCluster> sysJobClusterRep,
  23. ISchedulerFactory schedulerFactory,
  24. DynamicJobCompiler dynamicJobCompiler)
  25. {
  26. _sysJobDetailRep = sysJobDetailRep;
  27. _sysJobTriggerRep = sysJobTriggerRep;
  28. _sysJobTriggerRecordRep = sysJobTriggerRecordRep;
  29. _sysJobClusterRep = sysJobClusterRep;
  30. _schedulerFactory = schedulerFactory;
  31. _dynamicJobCompiler = dynamicJobCompiler;
  32. }
  33. /// <summary>
  34. /// 获取作业分页列表 ⏰
  35. /// </summary>
  36. [DisplayName("获取作业分页列表")]
  37. public async Task<SqlSugarPagedList<JobDetailOutput>> PageJobDetail(PageJobDetailInput input)
  38. {
  39. var jobDetails = await _sysJobDetailRep.AsQueryable()
  40. .WhereIF(!string.IsNullOrWhiteSpace(input.JobId), u => u.JobId.Contains(input.JobId.Trim()))
  41. .WhereIF(!string.IsNullOrWhiteSpace(input.GroupName), u => u.GroupName.Contains(input.GroupName.Trim()))
  42. .WhereIF(!string.IsNullOrWhiteSpace(input.Description), u => u.Description.Contains(input.Description.Trim()))
  43. .Select(d => new JobDetailOutput
  44. {
  45. JobDetail = d,
  46. }).ToPagedListAsync(input.Page, input.PageSize);
  47. await _sysJobDetailRep.AsSugarClient().ThenMapperAsync(jobDetails.Items, async u =>
  48. {
  49. u.JobTriggers = await _sysJobTriggerRep.GetListAsync(t => t.JobId == u.JobDetail.JobId);
  50. });
  51. // 提取中括号里面的参数值
  52. var rgx = new Regex(@"(?i)(?<=\[)(.*)(?=\])");
  53. foreach (var job in jobDetails.Items)
  54. {
  55. foreach (var jobTrigger in job.JobTriggers)
  56. {
  57. jobTrigger.Args = rgx.Match(jobTrigger.Args ?? "").Value;
  58. }
  59. }
  60. return jobDetails;
  61. }
  62. /// <summary>
  63. /// 获取作业组名称集合 ⏰
  64. /// </summary>
  65. [DisplayName("获取作业组名称集合")]
  66. public async Task<List<string>> ListJobGroup()
  67. {
  68. return await _sysJobDetailRep.AsQueryable().Distinct().Select(e => e.GroupName).ToListAsync();
  69. }
  70. /// <summary>
  71. /// 添加作业 ⏰
  72. /// </summary>
  73. /// <returns></returns>
  74. [ApiDescriptionSettings(Name = "AddJobDetail"), HttpPost]
  75. [DisplayName("添加作业")]
  76. public async Task AddJobDetail(AddJobDetailInput input)
  77. {
  78. var isExist = await _sysJobDetailRep.IsAnyAsync(u => u.JobId == input.JobId && u.Id != input.Id);
  79. if (isExist) throw Oops.Oh(ErrorCodeEnum.D1006);
  80. // 动态创建作业
  81. Type jobType;
  82. switch (input.CreateType)
  83. {
  84. case JobCreateTypeEnum.Script when string.IsNullOrEmpty(input.ScriptCode):
  85. throw Oops.Oh(ErrorCodeEnum.D1701);
  86. case JobCreateTypeEnum.Script:
  87. {
  88. jobType = _dynamicJobCompiler.BuildJob(input.ScriptCode);
  89. if (jobType.GetCustomAttributes(typeof(JobDetailAttribute)).FirstOrDefault() is not JobDetailAttribute jobDetailAttribute)
  90. throw Oops.Oh(ErrorCodeEnum.D1702);
  91. if (jobDetailAttribute.JobId != input.JobId)
  92. throw Oops.Oh(ErrorCodeEnum.D1703);
  93. break;
  94. }
  95. case JobCreateTypeEnum.Http:
  96. jobType = typeof(HttpJob);
  97. break;
  98. default:
  99. throw new NotSupportedException();
  100. }
  101. _schedulerFactory.AddJob(JobBuilder.Create(jobType).LoadFrom(input.Adapt<SysJobDetail>()).SetJobType(jobType));
  102. // 延迟一下等待持久化写入,再执行其他字段的更新
  103. await Task.Delay(500);
  104. await _sysJobDetailRep.AsUpdateable()
  105. .SetColumns(u => new SysJobDetail { CreateType = input.CreateType, ScriptCode = input.ScriptCode })
  106. .Where(u => u.JobId == input.JobId).ExecuteCommandAsync();
  107. }
  108. /// <summary>
  109. /// 更新作业 ⏰
  110. /// </summary>
  111. /// <returns></returns>
  112. [ApiDescriptionSettings(Name = "UpdateJobDetail"), HttpPost]
  113. [DisplayName("更新作业")]
  114. public async Task UpdateJobDetail(UpdateJobDetailInput input)
  115. {
  116. var isExist = await _sysJobDetailRep.IsAnyAsync(u => u.JobId == input.JobId && u.Id != input.Id);
  117. if (isExist) throw Oops.Oh(ErrorCodeEnum.D1006);
  118. var sysJobDetail = await _sysJobDetailRep.GetFirstAsync(u => u.Id == input.Id);
  119. if (sysJobDetail.JobId != input.JobId) throw Oops.Oh(ErrorCodeEnum.D1704);
  120. var scheduler = _schedulerFactory.GetJob(sysJobDetail.JobId);
  121. var oldScriptCode = sysJobDetail.ScriptCode; // 旧脚本代码
  122. input.Adapt(sysJobDetail);
  123. if (input.CreateType == JobCreateTypeEnum.Script)
  124. {
  125. if (string.IsNullOrEmpty(input.ScriptCode)) throw Oops.Oh(ErrorCodeEnum.D1701);
  126. if (input.ScriptCode != oldScriptCode)
  127. {
  128. // 动态创建作业
  129. var jobType = _dynamicJobCompiler.BuildJob(input.ScriptCode);
  130. if (jobType.GetCustomAttributes(typeof(JobDetailAttribute)).FirstOrDefault() is not JobDetailAttribute jobDetailAttribute)
  131. throw Oops.Oh(ErrorCodeEnum.D1702);
  132. if (jobDetailAttribute.JobId != input.JobId) throw Oops.Oh(ErrorCodeEnum.D1703);
  133. scheduler?.UpdateDetail(JobBuilder.Create(jobType).LoadFrom(sysJobDetail).SetJobType(jobType));
  134. }
  135. }
  136. else
  137. {
  138. scheduler?.UpdateDetail(scheduler.GetJobBuilder().LoadFrom(sysJobDetail));
  139. }
  140. // Tip: 假如这次更新有变更了 JobId,变更 JobId 后触发的持久化更新执行,会由于找不到 JobId 而更新不到数据
  141. // 延迟一下等待持久化写入,再执行其他字段的更新
  142. await Task.Delay(500);
  143. await _sysJobDetailRep.UpdateAsync(sysJobDetail);
  144. }
  145. /// <summary>
  146. /// 删除作业 ⏰
  147. /// </summary>
  148. /// <returns></returns>
  149. [ApiDescriptionSettings(Name = "DeleteJobDetail"), HttpPost]
  150. [DisplayName("删除作业")]
  151. public async Task DeleteJobDetail(DeleteJobDetailInput input)
  152. {
  153. _schedulerFactory.RemoveJob(input.JobId);
  154. // 如果 _schedulerFactory 中不存在 JodId,则无法触发持久化,下面的代码确保作业和触发器能被删除
  155. await _sysJobDetailRep.DeleteAsync(u => u.JobId == input.JobId);
  156. await _sysJobTriggerRep.DeleteAsync(u => u.JobId == input.JobId);
  157. }
  158. /// <summary>
  159. /// 获取触发器列表 ⏰
  160. /// </summary>
  161. [DisplayName("获取触发器列表")]
  162. public async Task<List<SysJobTrigger>> GetJobTriggerList([FromQuery] JobDetailInput input)
  163. {
  164. return await _sysJobTriggerRep.AsQueryable()
  165. .WhereIF(!string.IsNullOrWhiteSpace(input.JobId), u => u.JobId.Contains(input.JobId))
  166. .ToListAsync();
  167. }
  168. /// <summary>
  169. /// 添加触发器 ⏰
  170. /// </summary>
  171. /// <returns></returns>
  172. [ApiDescriptionSettings(Name = "AddJobTrigger"), HttpPost]
  173. [DisplayName("添加触发器")]
  174. public async Task AddJobTrigger(AddJobTriggerInput input)
  175. {
  176. var isExist = await _sysJobTriggerRep.IsAnyAsync(u => u.TriggerId == input.TriggerId && u.Id != input.Id);
  177. if (isExist) throw Oops.Oh(ErrorCodeEnum.D1006);
  178. var jobTrigger = input.Adapt<SysJobTrigger>();
  179. jobTrigger.Args = "[" + jobTrigger.Args + "]";
  180. var scheduler = _schedulerFactory.GetJob(input.JobId);
  181. scheduler?.AddTrigger(Triggers.Create(input.AssemblyName, input.TriggerType).LoadFrom(jobTrigger));
  182. }
  183. /// <summary>
  184. /// 更新触发器 ⏰
  185. /// </summary>
  186. /// <returns></returns>
  187. [ApiDescriptionSettings(Name = "UpdateJobTrigger"), HttpPost]
  188. [DisplayName("更新触发器")]
  189. public async Task UpdateJobTrigger(UpdateJobTriggerInput input)
  190. {
  191. var isExist = await _sysJobTriggerRep.IsAnyAsync(u => u.TriggerId == input.TriggerId && u.Id != input.Id);
  192. if (isExist) throw Oops.Oh(ErrorCodeEnum.D1006);
  193. var jobTrigger = input.Adapt<SysJobTrigger>();
  194. if (jobTrigger.EndTime.HasValue && jobTrigger.EndTime.Value.Year < 1901)
  195. {
  196. jobTrigger.EndTime = null;
  197. }
  198. if (jobTrigger.StartTime.HasValue && jobTrigger.StartTime.Value.Year < 1901)
  199. {
  200. jobTrigger.StartTime = null;
  201. }
  202. jobTrigger.Args = "[" + jobTrigger.Args + "]";
  203. var scheduler = _schedulerFactory.GetJob(input.JobId);
  204. scheduler?.UpdateTrigger(Triggers.Create(input.AssemblyName, input.TriggerType).LoadFrom(jobTrigger));
  205. }
  206. /// <summary>
  207. /// 删除触发器 ⏰
  208. /// </summary>
  209. /// <returns></returns>
  210. [ApiDescriptionSettings(Name = "DeleteJobTrigger"), HttpPost]
  211. [DisplayName("删除触发器")]
  212. public async Task DeleteJobTrigger(DeleteJobTriggerInput input)
  213. {
  214. var scheduler = _schedulerFactory.GetJob(input.JobId);
  215. scheduler?.RemoveTrigger(input.TriggerId);
  216. // 如果 _schedulerFactory 中不存在 JodId,则无法触发持久化,下行代码确保触发器能被删除
  217. await _sysJobTriggerRep.DeleteAsync(u => u.JobId == input.JobId && u.TriggerId == input.TriggerId);
  218. }
  219. /// <summary>
  220. /// 暂停所有作业 ⏰
  221. /// </summary>
  222. /// <returns></returns>
  223. [DisplayName("暂停所有作业")]
  224. public void PauseAllJob()
  225. {
  226. _schedulerFactory.PauseAll();
  227. }
  228. /// <summary>
  229. /// 启动所有作业 ⏰
  230. /// </summary>
  231. /// <returns></returns>
  232. [DisplayName("启动所有作业")]
  233. public void StartAllJob()
  234. {
  235. _schedulerFactory.StartAll();
  236. }
  237. /// <summary>
  238. /// 暂停作业 ⏰
  239. /// </summary>
  240. [DisplayName("暂停作业")]
  241. public void PauseJob(JobDetailInput input)
  242. {
  243. _schedulerFactory.TryPauseJob(input.JobId, out _);
  244. }
  245. /// <summary>
  246. /// 启动作业 ⏰
  247. /// </summary>
  248. [DisplayName("启动作业")]
  249. public void StartJob(JobDetailInput input)
  250. {
  251. _schedulerFactory.TryStartJob(input.JobId, out _);
  252. }
  253. /// <summary>
  254. /// 取消作业 ⏰
  255. /// </summary>
  256. [DisplayName("取消作业")]
  257. public void CancelJob(JobDetailInput input)
  258. {
  259. _schedulerFactory.TryCancelJob(input.JobId, out _);
  260. }
  261. /// <summary>
  262. /// 执行作业 ⏰
  263. /// </summary>
  264. /// <param name="input"></param>
  265. [DisplayName("执行作业")]
  266. public void RunJob(JobDetailInput input)
  267. {
  268. if (_schedulerFactory.TryRunJob(input.JobId, out _) != ScheduleResult.Succeed) throw Oops.Oh(ErrorCodeEnum.D1705);
  269. }
  270. /// <summary>
  271. /// 暂停触发器 ⏰
  272. /// </summary>
  273. [DisplayName("暂停触发器")]
  274. public void PauseTrigger(JobTriggerInput input)
  275. {
  276. var scheduler = _schedulerFactory.GetJob(input.JobId);
  277. scheduler?.PauseTrigger(input.TriggerId);
  278. }
  279. /// <summary>
  280. /// 启动触发器 ⏰
  281. /// </summary>
  282. [DisplayName("启动触发器")]
  283. public void StartTrigger(JobTriggerInput input)
  284. {
  285. var scheduler = _schedulerFactory.GetJob(input.JobId);
  286. scheduler?.StartTrigger(input.TriggerId);
  287. }
  288. /// <summary>
  289. /// 强制唤醒作业调度器 ⏰
  290. /// </summary>
  291. [DisplayName("强制唤醒作业调度器")]
  292. public void CancelSleep()
  293. {
  294. _schedulerFactory.CancelSleep();
  295. }
  296. /// <summary>
  297. /// 强制触发所有作业持久化 ⏰
  298. /// </summary>
  299. [DisplayName("强制触发所有作业持久化")]
  300. public void PersistAll()
  301. {
  302. _schedulerFactory.PersistAll();
  303. }
  304. /// <summary>
  305. /// 获取集群列表 ⏰
  306. /// </summary>
  307. [DisplayName("获取集群列表")]
  308. public async Task<List<SysJobCluster>> GetJobClusterList()
  309. {
  310. return await _sysJobClusterRep.GetListAsync();
  311. }
  312. /// <summary>
  313. /// 获取作业触发器运行记录分页列表 ⏰
  314. /// </summary>
  315. [DisplayName("获取作业触发器运行记录分页列表")]
  316. public async Task<SqlSugarPagedList<SysJobTriggerRecord>> PageJobTriggerRecord(PageJobTriggerRecordInput input)
  317. {
  318. return await _sysJobTriggerRecordRep.AsQueryable()
  319. .WhereIF(!string.IsNullOrWhiteSpace(input.JobId), u => u.JobId == input.JobId)
  320. .WhereIF(!string.IsNullOrWhiteSpace(input.TriggerId), u => u.TriggerId == input.TriggerId)
  321. .OrderByDescending(u => u.Id)
  322. .ToPagedListAsync(input.Page, input.PageSize);
  323. }
  324. /// <summary>
  325. /// 清空作业触发器运行记录 🔖
  326. /// </summary>
  327. /// <returns></returns>
  328. [ApiDescriptionSettings(Name = "ClearJobTriggerRecord"), HttpPost]
  329. [DisplayName("清空作业触发器运行记录")]
  330. public void ClearJobTriggerRecord()
  331. {
  332. _sysJobTriggerRecordRep.AsSugarClient().DbMaintenance.TruncateTable<SysJobTriggerRecord>();
  333. }
  334. /// <summary>
  335. /// 清空不保留的作业触发器运行记录 🔖
  336. /// </summary>
  337. /// <returns></returns>
  338. [NonAction]
  339. [DisplayName("清空过期的作业触发器运行记录")]
  340. public async Task ClearExpireJobTriggerRecord(SysJobTriggerRecord input)
  341. {
  342. int keepRecords = 30;//保留记录条数
  343. // 使用CopyNew()创建新的数据库连接实例,避免连接冲突
  344. var db = _sysJobTriggerRecordRep.AsSugarClient().CopyNew();
  345. await db.Deleteable<SysJobTriggerRecord>().In(it => it.Id,
  346. db.Queryable<SysJobTriggerRecord>()
  347. .Skip(keepRecords)
  348. .OrderByDescending(it => it.LastRunTime)
  349. .Where(u => u.JobId == input.JobId && u.TriggerId == input.TriggerId)
  350. .Select(it => it.Id) //注意Select不要ToList(), ToList就2次查询了
  351. ).ExecuteCommandAsync();
  352. }
  353. }