SysJobService.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  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)]
  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)
  80. throw Oops.Oh(ErrorCodeEnum.D1006);
  81. // 动态创建作业
  82. Type jobType;
  83. switch (input.CreateType)
  84. {
  85. case JobCreateTypeEnum.Script when string.IsNullOrEmpty(input.ScriptCode):
  86. throw Oops.Oh(ErrorCodeEnum.D1701);
  87. case JobCreateTypeEnum.Script:
  88. {
  89. jobType = _dynamicJobCompiler.BuildJob(input.ScriptCode);
  90. if (jobType.GetCustomAttributes(typeof(JobDetailAttribute)).FirstOrDefault() is not JobDetailAttribute jobDetailAttribute)
  91. throw Oops.Oh(ErrorCodeEnum.D1702);
  92. if (jobDetailAttribute.JobId != input.JobId)
  93. throw Oops.Oh(ErrorCodeEnum.D1703);
  94. break;
  95. }
  96. case JobCreateTypeEnum.Http:
  97. jobType = typeof(HttpJob);
  98. break;
  99. default:
  100. throw new NotSupportedException();
  101. }
  102. _schedulerFactory.AddJob(
  103. JobBuilder.Create(jobType)
  104. .LoadFrom(input.Adapt<SysJobDetail>()).SetJobType(jobType));
  105. // 延迟一下等待持久化写入,再执行其他字段的更新
  106. await Task.Delay(500);
  107. await _sysJobDetailRep.AsUpdateable()
  108. .SetColumns(u => new SysJobDetail { CreateType = input.CreateType, ScriptCode = input.ScriptCode })
  109. .Where(u => u.JobId == input.JobId).ExecuteCommandAsync();
  110. }
  111. /// <summary>
  112. /// 更新作业 ⏰
  113. /// </summary>
  114. /// <returns></returns>
  115. [ApiDescriptionSettings(Name = "UpdateJobDetail"), HttpPost]
  116. [DisplayName("更新作业")]
  117. public async Task UpdateJobDetail(UpdateJobDetailInput input)
  118. {
  119. var isExist = await _sysJobDetailRep.IsAnyAsync(u => u.JobId == input.JobId && u.Id != input.Id);
  120. if (isExist)
  121. throw Oops.Oh(ErrorCodeEnum.D1006);
  122. var sysJobDetail = await _sysJobDetailRep.GetFirstAsync(u => u.Id == input.Id);
  123. if (sysJobDetail.JobId != input.JobId)
  124. throw Oops.Oh(ErrorCodeEnum.D1704);
  125. var scheduler = _schedulerFactory.GetJob(sysJobDetail.JobId);
  126. var oldScriptCode = sysJobDetail.ScriptCode; // 旧脚本代码
  127. input.Adapt(sysJobDetail);
  128. if (input.CreateType == JobCreateTypeEnum.Script)
  129. {
  130. if (string.IsNullOrEmpty(input.ScriptCode))
  131. throw Oops.Oh(ErrorCodeEnum.D1701);
  132. if (input.ScriptCode != oldScriptCode)
  133. {
  134. // 动态创建作业
  135. var jobType = _dynamicJobCompiler.BuildJob(input.ScriptCode);
  136. if (jobType.GetCustomAttributes(typeof(JobDetailAttribute)).FirstOrDefault() is not JobDetailAttribute jobDetailAttribute)
  137. throw Oops.Oh(ErrorCodeEnum.D1702);
  138. if (jobDetailAttribute.JobId != input.JobId)
  139. throw Oops.Oh(ErrorCodeEnum.D1703);
  140. scheduler?.UpdateDetail(JobBuilder.Create(jobType).LoadFrom(sysJobDetail).SetJobType(jobType));
  141. }
  142. }
  143. else
  144. {
  145. scheduler?.UpdateDetail(scheduler.GetJobBuilder().LoadFrom(sysJobDetail));
  146. }
  147. // Tip: 假如这次更新有变更了 JobId,变更 JobId 后触发的持久化更新执行,会由于找不到 JobId 而更新不到数据
  148. // 延迟一下等待持久化写入,再执行其他字段的更新
  149. await Task.Delay(500);
  150. await _sysJobDetailRep.UpdateAsync(sysJobDetail);
  151. }
  152. /// <summary>
  153. /// 删除作业 ⏰
  154. /// </summary>
  155. /// <returns></returns>
  156. [ApiDescriptionSettings(Name = "DeleteJobDetail"), HttpPost]
  157. [DisplayName("删除作业")]
  158. public async Task DeleteJobDetail(DeleteJobDetailInput input)
  159. {
  160. _schedulerFactory.RemoveJob(input.JobId);
  161. // 如果 _schedulerFactory 中不存在 JodId,则无法触发持久化,下面的代码确保作业和触发器能被删除
  162. await _sysJobDetailRep.DeleteAsync(u => u.JobId == input.JobId);
  163. await _sysJobTriggerRep.DeleteAsync(u => u.JobId == input.JobId);
  164. }
  165. /// <summary>
  166. /// 获取触发器列表 ⏰
  167. /// </summary>
  168. [DisplayName("获取触发器列表")]
  169. public async Task<List<SysJobTrigger>> GetJobTriggerList([FromQuery] JobDetailInput input)
  170. {
  171. return await _sysJobTriggerRep.AsQueryable()
  172. .WhereIF(!string.IsNullOrWhiteSpace(input.JobId), u => u.JobId.Contains(input.JobId))
  173. .ToListAsync();
  174. }
  175. /// <summary>
  176. /// 添加触发器 ⏰
  177. /// </summary>
  178. /// <returns></returns>
  179. [ApiDescriptionSettings(Name = "AddJobTrigger"), HttpPost]
  180. [DisplayName("添加触发器")]
  181. public async Task AddJobTrigger(AddJobTriggerInput input)
  182. {
  183. var isExist = await _sysJobTriggerRep.IsAnyAsync(u => u.TriggerId == input.TriggerId && u.Id != input.Id);
  184. if (isExist)
  185. throw Oops.Oh(ErrorCodeEnum.D1006);
  186. var jobTrigger = input.Adapt<SysJobTrigger>();
  187. jobTrigger.Args = "[" + jobTrigger.Args + "]";
  188. var scheduler = _schedulerFactory.GetJob(input.JobId);
  189. scheduler?.AddTrigger(Triggers.Create(input.AssemblyName, input.TriggerType).LoadFrom(jobTrigger));
  190. }
  191. /// <summary>
  192. /// 更新触发器 ⏰
  193. /// </summary>
  194. /// <returns></returns>
  195. [ApiDescriptionSettings(Name = "UpdateJobTrigger"), HttpPost]
  196. [DisplayName("更新触发器")]
  197. public async Task UpdateJobTrigger(UpdateJobTriggerInput input)
  198. {
  199. var isExist = await _sysJobTriggerRep.IsAnyAsync(u => u.TriggerId == input.TriggerId && u.Id != input.Id);
  200. if (isExist)
  201. throw Oops.Oh(ErrorCodeEnum.D1006);
  202. var jobTrigger = input.Adapt<SysJobTrigger>();
  203. jobTrigger.Args = "[" + jobTrigger.Args + "]";
  204. var scheduler = _schedulerFactory.GetJob(input.JobId);
  205. scheduler?.UpdateTrigger(Triggers.Create(input.AssemblyName, input.TriggerType).LoadFrom(jobTrigger));
  206. }
  207. /// <summary>
  208. /// 删除触发器 ⏰
  209. /// </summary>
  210. /// <returns></returns>
  211. [ApiDescriptionSettings(Name = "DeleteJobTrigger"), HttpPost]
  212. [DisplayName("删除触发器")]
  213. public async Task DeleteJobTrigger(DeleteJobTriggerInput input)
  214. {
  215. var scheduler = _schedulerFactory.GetJob(input.JobId);
  216. scheduler?.RemoveTrigger(input.TriggerId);
  217. // 如果 _schedulerFactory 中不存在 JodId,则无法触发持久化,下行代码确保触发器能被删除
  218. await _sysJobTriggerRep.DeleteAsync(u => u.JobId == input.JobId && u.TriggerId == input.TriggerId);
  219. }
  220. /// <summary>
  221. /// 暂停所有作业 ⏰
  222. /// </summary>
  223. /// <returns></returns>
  224. [DisplayName("暂停所有作业")]
  225. public void PauseAllJob()
  226. {
  227. _schedulerFactory.PauseAll();
  228. }
  229. /// <summary>
  230. /// 启动所有作业 ⏰
  231. /// </summary>
  232. /// <returns></returns>
  233. [DisplayName("启动所有作业")]
  234. public void StartAllJob()
  235. {
  236. _schedulerFactory.StartAll();
  237. }
  238. /// <summary>
  239. /// 暂停作业 ⏰
  240. /// </summary>
  241. [DisplayName("暂停作业")]
  242. public void PauseJob(JobDetailInput input)
  243. {
  244. _schedulerFactory.TryPauseJob(input.JobId, out _);
  245. }
  246. /// <summary>
  247. /// 启动作业 ⏰
  248. /// </summary>
  249. [DisplayName("启动作业")]
  250. public void StartJob(JobDetailInput input)
  251. {
  252. _schedulerFactory.TryStartJob(input.JobId, out _);
  253. }
  254. /// <summary>
  255. /// 取消作业 ⏰
  256. /// </summary>
  257. [DisplayName("取消作业")]
  258. public void CancelJob(JobDetailInput input)
  259. {
  260. _schedulerFactory.TryCancelJob(input.JobId, out _);
  261. }
  262. /// <summary>
  263. /// 执行作业 ⏰
  264. /// </summary>
  265. /// <param name="input"></param>
  266. [DisplayName("执行作业")]
  267. public void RunJob(JobDetailInput input)
  268. {
  269. if (_schedulerFactory.TryRunJob(input.JobId, out _) != ScheduleResult.Succeed)
  270. throw Oops.Oh(ErrorCodeEnum.D1705);
  271. }
  272. /// <summary>
  273. /// 暂停触发器 ⏰
  274. /// </summary>
  275. [DisplayName("暂停触发器")]
  276. public void PauseTrigger(JobTriggerInput input)
  277. {
  278. var scheduler = _schedulerFactory.GetJob(input.JobId);
  279. scheduler?.PauseTrigger(input.TriggerId);
  280. }
  281. /// <summary>
  282. /// 启动触发器 ⏰
  283. /// </summary>
  284. [DisplayName("启动触发器")]
  285. public void StartTrigger(JobTriggerInput input)
  286. {
  287. var scheduler = _schedulerFactory.GetJob(input.JobId);
  288. scheduler?.StartTrigger(input.TriggerId);
  289. }
  290. /// <summary>
  291. /// 强制唤醒作业调度器 ⏰
  292. /// </summary>
  293. [DisplayName("强制唤醒作业调度器")]
  294. public void CancelSleep()
  295. {
  296. _schedulerFactory.CancelSleep();
  297. }
  298. /// <summary>
  299. /// 强制触发所有作业持久化 ⏰
  300. /// </summary>
  301. [DisplayName("强制触发所有作业持久化")]
  302. public void PersistAll()
  303. {
  304. _schedulerFactory.PersistAll();
  305. }
  306. /// <summary>
  307. /// 获取集群列表 ⏰
  308. /// </summary>
  309. [DisplayName("获取集群列表")]
  310. public async Task<List<SysJobCluster>> GetJobClusterList()
  311. {
  312. return await _sysJobClusterRep.GetListAsync();
  313. }
  314. /// <summary>
  315. /// 获取作业触发器运行记录分页列表 ⏰
  316. /// </summary>
  317. [DisplayName("获取作业触发器运行记录分页列表")]
  318. public async Task<SqlSugarPagedList<SysJobTriggerRecord>> PageJobTriggerRecord(PageJobTriggerRecordInput input)
  319. {
  320. return await _sysJobTriggerRecordRep.AsQueryable()
  321. .WhereIF(!string.IsNullOrWhiteSpace(input.JobId), u => u.JobId.Contains(input.JobId))
  322. .WhereIF(!string.IsNullOrWhiteSpace(input.TriggerId), u => u.TriggerId.Contains(input.TriggerId))
  323. .OrderByDescending(u => u.Id)
  324. .ToPagedListAsync(input.Page, input.PageSize);
  325. }
  326. }