DbJobPersistence.cs 7.0 KB

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