DbJobPersistence.cs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. namespace Admin.NET.Core.Service;
  2. /// <summary>
  3. /// 作业持久化(数据库)
  4. /// </summary>
  5. public class DbJobPersistence : IJobPersistence
  6. {
  7. private readonly IServiceProvider _serviceProvider;
  8. public DbJobPersistence(IServiceProvider serviceProvider)
  9. {
  10. _serviceProvider = serviceProvider;
  11. }
  12. /// <summary>
  13. /// 作业调度服务启动时
  14. /// </summary>
  15. /// <returns></returns>
  16. public IEnumerable<SchedulerBuilder> Preload()
  17. {
  18. using var serviceScope = _serviceProvider.CreateScope();
  19. var jobDetailRep = serviceScope.ServiceProvider.GetService<SqlSugarRepository<SysJobDetail>>();
  20. var jobTriggerRep = serviceScope.ServiceProvider.GetService<SqlSugarRepository<SysJobTrigger>>();
  21. var dynamicJobCompiler = serviceScope.ServiceProvider.GetService<DynamicJobCompiler>();
  22. // 获取内存的作业
  23. IEnumerable<SchedulerBuilder> memoryJobs = App.EffectiveTypes.ScanToBuilders();
  24. // 若数据库不存在任何作业,则返回内存作业
  25. if (!jobDetailRep.IsAny(u => true)) return memoryJobs;
  26. var schedulerBuilders = new List<SchedulerBuilder>();
  27. // 获取数据库所有作业
  28. var dbJobs = jobDetailRep.GetList();
  29. foreach (var dbJob in dbJobs)
  30. {
  31. JobBuilder jobDetail;
  32. if (dbJob.CreateFromScript)
  33. {
  34. // 动态创建作业
  35. var jobType = dynamicJobCompiler.BuildJob(dbJob.ScriptCode);
  36. jobDetail = JobBuilder.Create(jobType).LoadFrom(dbJob);
  37. }
  38. else
  39. {
  40. jobDetail = JobBuilder.Create(dbJob.AssemblyName, dbJob.JobType).LoadFrom(dbJob);
  41. }
  42. // 强行设置为不扫描 IJob 实现类 [Trigger] 特性触发器,否则 SchedulerBuilder.Create 会再次扫描,导致重复添加同名触发器
  43. jobDetail.SetIncludeAnnotations(false);
  44. // 加载数据库的触发器
  45. var triggerBuilders = new List<TriggerBuilder>();
  46. var dbTriggers = jobTriggerRep.GetList(u => u.JobId == dbJob.JobId)
  47. .Select(u => Triggers.Create(u.AssemblyName, u.TriggerType).LoadFrom(u)).ToArray();
  48. triggerBuilders.AddRange(dbTriggers);
  49. var memoryTriggers = memoryJobs.Where(u => u.GetJobBuilder().JobId == dbJob.JobId).SelectMany(u => u.GetTriggerBuilders());
  50. foreach (var memTrigger in memoryTriggers)
  51. {
  52. var triggerId = memTrigger.TriggerId;
  53. // 若数据库中已包含这个触发器
  54. if (!string.IsNullOrWhiteSpace(triggerId) && dbTriggers.Any(u => u.TriggerId == triggerId))
  55. continue;
  56. triggerBuilders.Add(memTrigger);
  57. }
  58. schedulerBuilders.Add(SchedulerBuilder.Create(jobDetail, triggerBuilders.ToArray()).Updated());
  59. }
  60. // 合并作业
  61. foreach (var job in memoryJobs)
  62. {
  63. var jobId = job.GetJobBuilder().JobId;
  64. // 若数据库中已包含这个作业
  65. if (!string.IsNullOrWhiteSpace(jobId) && dbJobs.Any(u => u.JobId == jobId))
  66. continue;
  67. schedulerBuilders.Add(job);
  68. }
  69. return schedulerBuilders;
  70. }
  71. /// <summary>
  72. /// 作业计划初始化通知
  73. /// </summary>
  74. /// <param name="builder"></param>
  75. /// <returns></returns>
  76. public SchedulerBuilder OnLoading(SchedulerBuilder builder)
  77. {
  78. using var serviceScope = _serviceProvider.CreateScope();
  79. var jobDetailRep = serviceScope.ServiceProvider.GetService<SqlSugarRepository<SysJobDetail>>();
  80. var jobTriggerRep = serviceScope.ServiceProvider.GetService<SqlSugarRepository<SysJobTrigger>>();
  81. //return rep.IsAny(u => u.JobId != builder.GetJobBuilder().JobId) ? builder.Appended() : builder.Updated();
  82. foreach (var (jobBuilder, triggerBuilder) in builder.GetEnumerable())
  83. {
  84. if (jobTriggerRep.IsAny(u => u.JobId == jobBuilder.JobId && u.TriggerId == triggerBuilder.TriggerId))
  85. triggerBuilder.Updated();
  86. else
  87. triggerBuilder.Appended();
  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. using var serviceScope = _serviceProvider.CreateScope();
  98. var db = serviceScope.ServiceProvider.GetService<ISqlSugarClient>();
  99. var jobDetail = context.JobDetail.Adapt<SysJobDetail>();
  100. if (context.Behavior == PersistenceBehavior.Appended)
  101. {
  102. db.Insertable(jobDetail).ExecuteCommand();
  103. }
  104. else if (context.Behavior == PersistenceBehavior.Updated)
  105. {
  106. db.Updateable(jobDetail).WhereColumns(u => new { u.JobId }).IgnoreColumns(u => new { u.Id, u.CreateFromScript, u.ScriptCode }).ExecuteCommand();
  107. }
  108. else if (context.Behavior == PersistenceBehavior.Removed)
  109. {
  110. db.Deleteable<SysJobDetail>().Where(u => u.JobId == jobDetail.JobId).ExecuteCommand();
  111. }
  112. }
  113. /// <summary>
  114. /// 作业计划Scheduler的触发器Trigger变化时
  115. /// </summary>
  116. /// <param name="context"></param>
  117. public void OnTriggerChanged(PersistenceTriggerContext context)
  118. {
  119. using var serviceScope = _serviceProvider.CreateScope();
  120. var db = serviceScope.ServiceProvider.GetService<ISqlSugarClient>();
  121. var jobTrigger = context.Trigger.Adapt<SysJobTrigger>();
  122. if (context.Behavior == PersistenceBehavior.Appended)
  123. {
  124. db.Insertable(jobTrigger).ExecuteCommand();
  125. }
  126. else if (context.Behavior == PersistenceBehavior.Updated)
  127. {
  128. db.Updateable(jobTrigger).WhereColumns(u => new { u.TriggerId, u.JobId }).IgnoreColumns(u => new { u.Id }).ExecuteCommand();
  129. }
  130. else if (context.Behavior == PersistenceBehavior.Removed)
  131. {
  132. db.Deleteable<SysJobTrigger>().Where(u => u.TriggerId == jobTrigger.TriggerId && u.JobId == jobTrigger.JobId).ExecuteCommand();
  133. }
  134. }
  135. }