DbJobPersistence.cs 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. namespace Admin.NET.Core.Service;
  2. /// <summary>
  3. /// 作业持久化(数据库)
  4. /// </summary>
  5. public class DbJobPersistence : IJobPersistence
  6. {
  7. private readonly IServiceScope _serviceScope;
  8. private readonly ISqlSugarClient _sqlSugarClient;
  9. private readonly SqlSugarRepository<SysJobDetail> _jobRepository;
  10. private readonly SqlSugarRepository<SysJobTrigger> _triggerRepository;
  11. private readonly DynamicJobCompiler _dynamicJobCompiler;
  12. public DbJobPersistence(IServiceProvider serviceProvider)
  13. {
  14. _serviceScope = serviceProvider.CreateScope();
  15. var services = _serviceScope.ServiceProvider;
  16. _dynamicJobCompiler = services.GetService<DynamicJobCompiler>();
  17. _sqlSugarClient = services.GetService<ISqlSugarClient>();
  18. _jobRepository = services.GetService<SqlSugarRepository<SysJobDetail>>();
  19. _triggerRepository = services.GetService<SqlSugarRepository<SysJobTrigger>>();
  20. }
  21. /// <summary>
  22. /// 作业调度服务启动时
  23. /// </summary>
  24. /// <returns></returns>
  25. public IEnumerable<SchedulerBuilder> Preload()
  26. {
  27. // 获取所有定义的作业
  28. var allJobs = App.EffectiveTypes.ScanToBuilders().ToList();
  29. // 若数据库不存在任何作业,则直接返回
  30. if (!_jobRepository.IsAny(u => true)) return allJobs;
  31. // 遍历所有定义的作业
  32. foreach (var schedulerBuilder in allJobs)
  33. {
  34. // 获取作业信息构建器
  35. var jobBuilder = schedulerBuilder.GetJobBuilder();
  36. // 加载数据库数据
  37. var dbDetail = _jobRepository.GetFirst(u => u.JobId == jobBuilder.JobId);
  38. if (dbDetail == null) continue;
  39. // 同步数据库数据
  40. jobBuilder.LoadFrom(dbDetail);
  41. // 获取作业的所有数据库的触发器
  42. var dbTriggers = _triggerRepository.GetList(u => u.JobId == jobBuilder.JobId).ToArray();
  43. // 遍历所有作业触发器
  44. foreach (var (_, triggerBuilder) in schedulerBuilder.GetEnumerable())
  45. {
  46. // 加载数据库数据
  47. var dbTrigger = dbTriggers.FirstOrDefault(u => u.JobId == jobBuilder.JobId && u.TriggerId == triggerBuilder.TriggerId);
  48. if (dbTrigger == null) continue;
  49. triggerBuilder.LoadFrom(dbTrigger)
  50. .Updated(); // 标记更新
  51. }
  52. // 遍历所有非编译时定义的触发器加入到作业中
  53. foreach (var dbTrigger in dbTriggers)
  54. {
  55. if (schedulerBuilder.GetTriggerBuilder(dbTrigger.TriggerId)?.JobId == jobBuilder.JobId) continue;
  56. var triggerBuilder = TriggerBuilder.Create(dbTrigger.TriggerId).LoadFrom(dbTrigger);
  57. schedulerBuilder.AddTriggerBuilder(triggerBuilder); // 先添加
  58. triggerBuilder.Updated(); // 再标记更新
  59. }
  60. // 标记更新
  61. schedulerBuilder.Updated();
  62. }
  63. // 获取数据库所有通过脚本创建的作业
  64. var allDbScriptJobs = _jobRepository.GetList(u => u.CreateFromScript);
  65. foreach (var dbDetail in allDbScriptJobs)
  66. {
  67. // 动态创建作业
  68. var jobType = _dynamicJobCompiler.BuildJob(dbDetail.ScriptCode);
  69. var jobBuilder = JobBuilder.Create(jobType).LoadFrom(dbDetail);
  70. // 强行设置为不扫描 IJob 实现类 [Trigger] 特性触发器,否则 SchedulerBuilder.Create 会再次扫描,导致重复添加同名触发器
  71. jobBuilder.SetIncludeAnnotations(false);
  72. // 获取作业的所有数据库的触发器加入到作业中
  73. var dbTriggers = _triggerRepository.GetList(u => u.JobId == jobBuilder.JobId).ToArray();
  74. var triggerBuilders = dbTriggers.Select(u => TriggerBuilder.Create(u.TriggerId).LoadFrom(u).Updated());
  75. var schedulerBuilder = SchedulerBuilder.Create(jobBuilder, triggerBuilders.ToArray());
  76. // 标记更新
  77. schedulerBuilder.Updated();
  78. allJobs.Add(schedulerBuilder);
  79. }
  80. return allJobs;
  81. }
  82. /// <summary>
  83. /// 作业计划初始化通知
  84. /// </summary>
  85. /// <param name="builder"></param>
  86. /// <returns></returns>
  87. public SchedulerBuilder OnLoading(SchedulerBuilder builder)
  88. {
  89. return builder;
  90. }
  91. /// <summary>
  92. /// 作业计划Scheduler的JobDetail变化时
  93. /// </summary>
  94. /// <param name="context"></param>
  95. public void OnChanged(PersistenceContext context)
  96. {
  97. var jobDetail = context.JobDetail.Adapt<SysJobDetail>();
  98. if (context.Behavior == PersistenceBehavior.Appended)
  99. {
  100. _sqlSugarClient.Insertable(jobDetail).ExecuteCommand();
  101. }
  102. else if (context.Behavior == PersistenceBehavior.Updated)
  103. {
  104. _sqlSugarClient.Updateable(jobDetail).WhereColumns(u => new { u.JobId }).IgnoreColumns(u => new { u.Id, u.CreateFromScript, u.ScriptCode }).ExecuteCommand();
  105. }
  106. else if (context.Behavior == PersistenceBehavior.Removed)
  107. {
  108. _sqlSugarClient.Deleteable<SysJobDetail>().Where(u => u.JobId == jobDetail.JobId).ExecuteCommand();
  109. }
  110. }
  111. /// <summary>
  112. /// 作业计划Scheduler的触发器Trigger变化时
  113. /// </summary>
  114. /// <param name="context"></param>
  115. public void OnTriggerChanged(PersistenceTriggerContext context)
  116. {
  117. var jobTrigger = context.Trigger.Adapt<SysJobTrigger>();
  118. if (context.Behavior == PersistenceBehavior.Appended)
  119. {
  120. _sqlSugarClient.Insertable(jobTrigger).ExecuteCommand();
  121. }
  122. else if (context.Behavior == PersistenceBehavior.Updated)
  123. {
  124. _sqlSugarClient.Updateable(jobTrigger).WhereColumns(u => new { u.TriggerId, u.JobId }).IgnoreColumns(u => new { u.Id }).ExecuteCommand();
  125. }
  126. else if (context.Behavior == PersistenceBehavior.Removed)
  127. {
  128. _sqlSugarClient.Deleteable<SysJobTrigger>().Where(u => u.TriggerId == jobTrigger.TriggerId && u.JobId == jobTrigger.JobId).ExecuteCommand();
  129. }
  130. }
  131. public void Dispose()
  132. {
  133. _serviceScope?.Dispose();
  134. }
  135. }