// 麻省理工学院许可证 // // 版权所有 (c) 2021-2023 zuohuaijun,大名科技(天津)有限公司 联系电话/微信:18020030720 QQ:515096995 // // 特此免费授予获得本软件的任何人以处理本软件的权利,但须遵守以下条件:在所有副本或重要部分的软件中必须包括上述版权声明和本许可声明。 // // 软件按“原样”提供,不提供任何形式的明示或暗示的保证,包括但不限于对适销性、适用性和非侵权的保证。 // 在任何情况下,作者或版权持有人均不对任何索赔、损害或其他责任负责,无论是因合同、侵权或其他方式引起的,与软件或其使用或其他交易有关。 namespace Admin.NET.Core.Service; /// /// 作业持久化(数据库) /// public class DbJobPersistence : IJobPersistence { private readonly IServiceScopeFactory _serviceScopeFactory; public DbJobPersistence(IServiceScopeFactory serviceScopeFactory) { _serviceScopeFactory = serviceScopeFactory; } /// /// 作业调度服务启动时 /// /// public IEnumerable Preload() { using var scope = _serviceScopeFactory.CreateScope(); var jobDetailRep = scope.ServiceProvider.GetRequiredService>(); var jobTriggerRep = scope.ServiceProvider.GetRequiredService>(); var dynamicJobCompiler = scope.ServiceProvider.GetRequiredService(); // 获取所有定义的作业 var allJobs = App.EffectiveTypes.ScanToBuilders().ToList(); // 若数据库不存在任何作业,则直接返回 if (!jobDetailRep.IsAny(u => true)) return allJobs; // 遍历所有定义的作业 foreach (var schedulerBuilder in allJobs) { // 获取作业信息构建器 var jobBuilder = schedulerBuilder.GetJobBuilder(); // 加载数据库数据 var dbDetail = jobDetailRep.GetFirst(u => u.JobId == jobBuilder.JobId); if (dbDetail == null) continue; // 同步数据库数据 jobBuilder.LoadFrom(dbDetail); // 获取作业的所有数据库的触发器 var dbTriggers = jobTriggerRep.GetList(u => u.JobId == jobBuilder.JobId).ToArray(); // 遍历所有作业触发器 foreach (var (_, triggerBuilder) in schedulerBuilder.GetEnumerable()) { // 加载数据库数据 var dbTrigger = dbTriggers.FirstOrDefault(u => u.JobId == jobBuilder.JobId && u.TriggerId == triggerBuilder.TriggerId); if (dbTrigger == null) continue; triggerBuilder.LoadFrom(dbTrigger).Updated(); // 标记更新 } // 遍历所有非编译时定义的触发器加入到作业中 foreach (var dbTrigger in dbTriggers) { if (schedulerBuilder.GetTriggerBuilder(dbTrigger.TriggerId)?.JobId == jobBuilder.JobId) continue; var triggerBuilder = TriggerBuilder.Create(dbTrigger.TriggerId).LoadFrom(dbTrigger); schedulerBuilder.AddTriggerBuilder(triggerBuilder); // 先添加 triggerBuilder.Updated(); // 再标记更新 } // 标记更新 schedulerBuilder.Updated(); } // 获取数据库所有通过脚本创建的作业 var allDbScriptJobs = jobDetailRep.GetList(u => u.CreateType != JobCreateTypeEnum.BuiltIn); foreach (var dbDetail in allDbScriptJobs) { // 动态创建作业 Type jobType; switch (dbDetail.CreateType) { case JobCreateTypeEnum.Script: jobType = dynamicJobCompiler.BuildJob(dbDetail.ScriptCode); break; case JobCreateTypeEnum.Http: jobType = typeof(HttpJob); break; default: throw new NotSupportedException(); } // 动态构建的 jobType 的程序集名称为随机名称,需重新设置 dbDetail.AssemblyName = jobType.Assembly.FullName!.Split(',')[0]; var jobBuilder = JobBuilder.Create(jobType).LoadFrom(dbDetail); // 强行设置为不扫描 IJob 实现类 [Trigger] 特性触发器,否则 SchedulerBuilder.Create 会再次扫描,导致重复添加同名触发器 jobBuilder.SetIncludeAnnotations(false); // 获取作业的所有数据库的触发器加入到作业中 var dbTriggers = jobTriggerRep.GetList(u => u.JobId == jobBuilder.JobId).ToArray(); var triggerBuilders = dbTriggers.Select(u => TriggerBuilder.Create(u.TriggerId).LoadFrom(u).Updated()); var schedulerBuilder = SchedulerBuilder.Create(jobBuilder, triggerBuilders.ToArray()); // 标记更新 schedulerBuilder.Updated(); allJobs.Add(schedulerBuilder); } return allJobs; } /// /// 作业计划初始化通知 /// /// /// public SchedulerBuilder OnLoading(SchedulerBuilder builder) { return builder; } /// /// 作业计划Scheduler的JobDetail变化时 /// /// public void OnChanged(PersistenceContext context) { using var scope = _serviceScopeFactory.CreateScope(); var jobDetailRep = scope.ServiceProvider.GetRequiredService>(); var jobDetail = context.JobDetail.Adapt(); switch (context.Behavior) { case PersistenceBehavior.Appended: jobDetailRep.AsInsertable(jobDetail).ExecuteCommand(); break; case PersistenceBehavior.Updated: jobDetailRep.AsUpdateable(jobDetail).WhereColumns(u => new { u.JobId }).IgnoreColumns(u => new { u.Id, u.CreateType, u.ScriptCode }).ExecuteCommand(); break; case PersistenceBehavior.Removed: jobDetailRep.AsDeleteable().Where(u => u.JobId == jobDetail.JobId).ExecuteCommand(); break; default: throw new ArgumentOutOfRangeException(); } } /// /// 作业计划Scheduler的触发器Trigger变化时 /// /// public void OnTriggerChanged(PersistenceTriggerContext context) { using var scope = _serviceScopeFactory.CreateScope(); var jobTriggerRep = scope.ServiceProvider.GetRequiredService>(); var jobTrigger = context.Trigger.Adapt(); switch (context.Behavior) { case PersistenceBehavior.Appended: jobTriggerRep.AsInsertable(jobTrigger).ExecuteCommand(); break; case PersistenceBehavior.Updated: jobTriggerRep.AsUpdateable(jobTrigger).WhereColumns(u => new { u.TriggerId, u.JobId }).IgnoreColumns(u => new { u.Id }).ExecuteCommand(); break; case PersistenceBehavior.Removed: jobTriggerRep.AsDeleteable().Where(u => u.TriggerId == jobTrigger.TriggerId && u.JobId == jobTrigger.JobId).ExecuteCommand(); break; default: throw new ArgumentOutOfRangeException(); } } }