| | |
| | | using FluentScheduler; |
| | | using Newtonsoft.Json; |
| | | using Rhea.Common; |
| | | using Sundial; |
| | | using System; |
| | | using System.Collections; |
| | | using System.Collections.Generic; |
| | | using System.Linq; |
| | | using System.Security.Cryptography; |
| | | using System.Text; |
| | | using System.Threading; |
| | | using System.Threading.Tasks; |
| | | using Tiger.IBusiness; |
| | | using Tiger.Model; |
| | | using Tiger.Model.Entitys.MES.Position; |
| | | using Tiger.Model.Minsun; |
| | | using static Microsoft.CodeAnalysis.CSharp.SyntaxTokenParser; |
| | | using static Tiger.Model.TrigArgs; |
| | | |
| | | namespace Tiger.Business |
| | | { |
| | |
| | | JobManager.JobException += info => Logger.Scheduler.Error("An error just happened with a scheduled job: " + info.Exception); |
| | | JobManager.JobStart += info => Logger.Scheduler.Info($"{info.Name}: started\r\n"); |
| | | JobManager.JobEnd += info => Logger.Scheduler.Info($"{info.Name}: ended ({info.Duration})\r\n"); |
| | | SaveRunningStatus(); //保存状态 |
| | | } |
| | | |
| | | private void SaveRunningStatus() |
| | | { |
| | | List<TSK_TRIG> listTrig = new List<TSK_TRIG>(); |
| | | foreach (var schedule in JobManager.RunningSchedules) |
| | | { |
| | | var job = Biz.Db.Queryable<TSK_JOB>().Where(q => q.JobName == schedule.Name).First(); |
| | | if (job != null) |
| | | { |
| | | TSK_TRIG trig = new TSK_TRIG(); |
| | | trig = Biz.Db.Queryable<TSK_TRIG>().Where(q => q.JobId == job.ID).First(); |
| | | trig.Status = TSK_TRIG.Statuss.Running.GetValue(); |
| | | trig.StartTime = DateTime.Now; |
| | | listTrig.Add(trig); |
| | | } |
| | | } |
| | | var db = Biz.Db; |
| | | //保存到数据库 |
| | | var dbTran = db.UseTran(() => |
| | | { |
| | | if (listTrig.Count > 0) |
| | | { |
| | | db.Updateable(listTrig, "system").UpdateColumns(q => new { q.Status, q.UPDATE_USER, q.UPDATE_TIME }).ExecuteCommand(); |
| | | } |
| | | }); |
| | | if (!dbTran.IsSuccess) |
| | | { |
| | | Logger.Scheduler.Trace(dbTran.ErrorException, $"作业保存到数据库异常"); |
| | | } |
| | | } |
| | | |
| | | /// <summary> |
| | |
| | | /// <param name="newEntity"></param> |
| | | /// <param name="jobParam"></param> |
| | | /// <returns></returns> |
| | | public ApiAction AddJob<TJob>(TJob newEntity, FluentJobParam jobParam) where TJob : class, IJob |
| | | public ApiAction AddTskJob<TJob>(TJob newEntity, FluentJobParam jobParam) where TJob : class, ITJob |
| | | { |
| | | ApiAction apiAction = new(); |
| | | try |
| | | { |
| | | JobManager.AddJob<TJob>((s) => GenerateSchedule(jobParam.Args, s.WithName(jobParam.JobName))); |
| | | if (!JobManager.AllSchedules.Any(q => q.Name == jobParam.JobName)) |
| | | { |
| | | JobManager.AddJob<TJob>((s) => GenerateSchedule(jobParam.Args, s.WithName(jobParam.JobName))); |
| | | string guid = Guid.NewGuid().ToString("N"); |
| | | TSK_JOB job = Biz.Db.Queryable<TSK_JOB>().Where(x => x.JobName == jobParam.JobName).First() ?? new() |
| | | { |
| | | ID = guid, |
| | | Remark = jobParam.Remark, |
| | | JobType = jobParam.JobName, |
| | | AssemblyName = jobParam.AssemblyName, |
| | | JobName = jobParam.JobName, |
| | | }; |
| | | var trigger = Biz.Db.Queryable<TSK_TRIG>().Where(x => x.JobId == job.ID).First() ?? new TSK_TRIG() |
| | | { |
| | | JobId = guid, |
| | | StartTime = DateTime.Now, |
| | | NumberOfRuns = 1, |
| | | }; |
| | | trigger.Args = JsonConvert.SerializeObject(jobParam.Args); |
| | | trigger.Status = TSK_TRIG.Statuss.Running.GetValue(); |
| | | var db = Biz.Db; |
| | | //保存到数据库 |
| | | var dbTran = db.UseTran(() => |
| | | { |
| | | var y = db.Storageable(job) |
| | | .WhereColumns(t => new { t.JobName, t.GHOST_ROW }) |
| | | .ToStorage(); |
| | | y.AsInsertable.ExecuteCommand(); |
| | | y.AsUpdateable.IgnoreColumns(x => x.ID).ExecuteCommand(); |
| | | |
| | | //Sundial.TriggerBuilder triggerBuilder = Sundial.Triggers.Period(jobParam.Period * 60 * 1000).LoadFrom(new |
| | | //{ |
| | | // TriggerId = jobParam.JobName + "_trigger1", |
| | | //}); |
| | | //_schedulerFactory.AddJob<TJob>(jobParam.JobName, triggerBuilder); |
| | | //var trigger = GetJob(jobParam.JobName).GetTrigger($"{jobParam.JobName}_trigger1"); |
| | | //TSK_JOB entity = ReturnDetail(jobParam.JobName); |
| | | //entity.Remark = jobParam.Remark; |
| | | //entity.JobType = typeof(TJob).Name; |
| | | //entity.JobName = jobParam.JobName; |
| | | //entity.TriggersWithGhost = new List<TSK_TRIG> { JsonConvert.DeserializeObject<TSK_TRIG>(JsonConvert.SerializeObject(trigger)) }; |
| | | //entity.Triggers[0].Args = jobParam.Period.ToString(); |
| | | //apiAction = SaveJob(entity); |
| | | //apiAction.Data = entity; |
| | | var z = db.Storageable(trigger) |
| | | .WhereColumns(t => new { t.JobId, t.GHOST_ROW }) |
| | | .ToStorage(); |
| | | z.AsInsertable.ExecuteCommand(); |
| | | z.AsUpdateable.IgnoreColumns(x => x.ID).ExecuteCommand(); |
| | | }); |
| | | if (!dbTran.IsSuccess) |
| | | { |
| | | apiAction.CatchExceptionWithLog(dbTran.ErrorException, $"添加工作任务时保存到数据库异常"); |
| | | } |
| | | } |
| | | |
| | | } |
| | | catch (System.Exception ex) |
| | | { |
| | | Logger.Console.Fatal(ex, "Add Job From DB Exception"); |
| | | apiAction.CatchExceptionWithLog(ex); |
| | | } |
| | | if (!apiAction.IsSuccessed) { JobManager.RemoveJob(jobParam.JobName); } |
| | | return apiAction; |
| | | } |
| | | |
| | | /// <summary> |
| | | /// 删除作业 |
| | | /// </summary> |
| | | /// <param name="jobname"></param> |
| | | public void RemovefJob(string jobname) |
| | | { |
| | | if (JobManager.AllSchedules.Any(q => q.Name == jobname)) |
| | | { |
| | | string guid = Guid.NewGuid().ToString("N"); |
| | | TSK_JOB job = Biz.Db.Queryable<TSK_JOB>().Where(x => x.JobName == jobname).First(); |
| | | var trigger = Biz.Db.Queryable<TSK_TRIG>().Where(x => x.JobId == job.ID).First(); |
| | | trigger.Status = TSK_TRIG.Statuss.Stop.GetValue(); |
| | | var db = Biz.Db; |
| | | //保存到数据库 |
| | | var dbTran = db.UseTran(() => |
| | | { |
| | | db.Updateable(trigger, "system_job").UpdateColumns(q=>new { q.Status,q.UPDATE_TIME,q.UPDATE_USER}).ExecuteCommand(); |
| | | }); |
| | | if (!dbTran.IsSuccess) |
| | | { |
| | | Logger.Scheduler.Fatal(dbTran.ErrorException, $"添加工作任务时保存到数据库异常"); |
| | | } |
| | | } |
| | | JobManager.RemoveJob(jobname); |
| | | } |
| | | |
| | | /// <summary> |
| | |
| | | var jobs = Biz.Db.Queryable<TSK_JOB>().IncludesAllFirstLayer().ToList(); |
| | | foreach (var job in jobs) |
| | | { |
| | | Type type = Type.GetType($"{job.AssemblyName}.{job.JobType},{job.AssemblyName}", throwOnError: true); |
| | | dynamic _type = Activator.CreateInstance(type); |
| | | TrigArgs args = JsonConvert.DeserializeObject<TrigArgs>(job.Triggers[0].Args); |
| | | Schedule schedule = Schedule(_type, job.JobName, registry); |
| | | //生成计划 |
| | | GenerateSchedule(args, schedule); |
| | | if (job.JobName == "fU9C_MES_GetWo" || job.JobName == "fTest") |
| | | { |
| | | Type type = Type.GetType($"{job.AssemblyName}.{job.JobType},{job.AssemblyName}", throwOnError: true); |
| | | dynamic _type = Activator.CreateInstance(type); |
| | | TrigArgs args = JsonConvert.DeserializeObject<TrigArgs>(job.Triggers[0].Args); |
| | | Schedule schedule = Schedule(_type, job.JobName, registry); |
| | | //生成计划 |
| | | GenerateSchedule(args, schedule); |
| | | } |
| | | } |
| | | } |
| | | catch (System.Exception ex) |
| | |
| | | /// <param name="JobName"></param> |
| | | /// <param name="registry"></param> |
| | | /// <returns></returns> |
| | | private Schedule Schedule<TJob>(TJob entity, string JobName, Registry registry) where TJob : class, IJob |
| | | private Schedule Schedule<TJob>(TJob entity, string JobName, Registry registry) where TJob : class, ITJob |
| | | { |
| | | return registry.Schedule<TJob>().WithName(JobName); |
| | | return registry.Schedule<TJob>().WithName(typeof(TJob).Name); |
| | | } |
| | | |
| | | /// <summary> |
| | |
| | | { |
| | | SpecificTimeUnit specificTimeUnit = null; |
| | | TimeUnit timeUnit = null; |
| | | if (args.NonReentrant == "Y") { schedule = schedule.NonReentrant(); } |
| | | if (args.ToRunNow == "Y") { specificTimeUnit = schedule.ToRunNow(); } |
| | | if (args.ToRunOnceAt == "Y") { specificTimeUnit = schedule.ToRunOnceAt(args.ToRunOnceAtDt.AddMinutes(args.NowAddMinutes)); } |
| | | if (args.ToRunOnceIn > 0) { timeUnit = schedule.ToRunOnceIn(args.ToRunOnceIn); } |
| | | else if (args.ToRunEvery >= 0) { timeUnit = schedule.ToRunEvery(args.ToRunEvery); } |
| | | if (specificTimeUnit != null && args.AndEvery>0) { timeUnit = specificTimeUnit.AndEvery(args.AndEvery); } |
| | | if (args.NonReentrant == "Y") { schedule = schedule.NonReentrant(); } //就是是否跟之前的还没运行完的计划同时运行。 |
| | | switch (args.runType) |
| | | { |
| | | case RunType.ToRunNow: |
| | | specificTimeUnit = schedule.ToRunNow(); //马上运行 |
| | | break; |
| | | case RunType.ToRunOnceAt: |
| | | specificTimeUnit = schedule.ToRunOnceAt(args.ToRunOnceAtDt.AddMinutes(args.NowAddMinutes)); //运行一次在哪个时间, |
| | | break; |
| | | case RunType.ToRunOnceIn: |
| | | timeUnit = schedule.ToRunOnceIn(args.ToRunOnceIn); //在多少后运行一次 |
| | | break; |
| | | case RunType.ToRunEvery: |
| | | timeUnit = schedule.ToRunEvery(args.ToRunEvery); //每隔多少运行一回 |
| | | break; |
| | | } |
| | | if (specificTimeUnit != null) { timeUnit = specificTimeUnit.AndEvery(args.ToRunEvery); } //每隔多少运行一回,加在ToRunNow、ToRunOnceAt后面 |
| | | //按时间类型来运行 |
| | | switch (args.Type) |
| | | { |
| | |
| | | timeUnit.Milliseconds(); |
| | | break; |
| | | case TrigArgs.EveryType.Seconds: |
| | | timeUnit.Seconds(); |
| | | timeUnit.Seconds(); //表示按秒,多少秒要看前面Every的数,比如:ToRunEvery(5).Seconds() 这个就表示 隔5秒运行一次 |
| | | break; |
| | | case TrigArgs.EveryType.Minutes: |
| | | timeUnit.Minutes(); |
| | |
| | | case TrigArgs.EveryType.Hours: |
| | | if (args.Minutes > 0) |
| | | { |
| | | timeUnit.Hours().At(args.Minutes); |
| | | timeUnit.Hours().At(args.Minutes); //表示按每小时的分钟,多少分要看前面Every的数,比如:ToRunEvery(1).Hours().At(5) 这个就表示 隔1小时的第5分钟运行一次 |
| | | } |
| | | else |
| | | { |
| | |
| | | } |
| | | break; |
| | | case TrigArgs.EveryType.Days: |
| | | if (args.Hours > 0 && args.Minutes > 0) |
| | | if (args.Hours > 0) |
| | | { |
| | | timeUnit.Days().At(args.Hours, args.Minutes); |
| | | timeUnit.Days().At(args.Hours, args.Minutes); //表示按天,多少天要看前面Every的数,比如:ToRunEvery(2).Days().At(14, 15) 这个就表示 隔2天的14:15运行一次 |
| | | } |
| | | else |
| | | { |
| | |
| | | } |
| | | break; |
| | | case TrigArgs.EveryType.Weeks: |
| | | if (args.Hours > 0 && args.Minutes > 0) |
| | | if (args.Hours > 0) |
| | | { |
| | | timeUnit.Weeks().At(args.Hours, args.Minutes); |
| | | timeUnit.Weeks().At(args.Hours, args.Minutes); //表示按周,多少周要看前面Every的数,比如:ToRunEvery(1).Weeks().At(14, 15) 这个就表示 隔1周的14:15运行一次 |
| | | } |
| | | else |
| | | { |
| | |
| | | } |
| | | break; |
| | | case TrigArgs.EveryType.Weekdays: |
| | | if (args.Hours > 0 && args.Minutes > 0) |
| | | if (args.Hours > 0) |
| | | { |
| | | timeUnit.Weekdays().At(args.Hours, args.Minutes); |
| | | } |
| | |
| | | timeUnit.Weekdays(); |
| | | } |
| | | break; |
| | | case TrigArgs.EveryType.Months: |
| | | if (args.Days > 0 && args.Hours > 0) |
| | | { |
| | | timeUnit.Months().On(args.Days).At(args.Hours, args.Minutes); |
| | | } |
| | | else |
| | | { |
| | | timeUnit.Months(); |
| | | } |
| | | break; |
| | | } |
| | | } |
| | | |