SysJobService.cs 13 KB

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