wabbajack/Wabbajack.BuildServer/JobManager.cs

141 lines
5.1 KiB
C#
Raw Normal View History

2020-01-09 04:42:25 +00:00
using System;
2020-01-09 12:14:48 +00:00
using System.Linq;
2020-01-09 04:42:25 +00:00
using System.Threading.Tasks;
using Microsoft.Extensions.Logging;
using MongoDB.Driver;
using MongoDB.Driver.Linq;
using Nettle;
2020-01-09 04:42:25 +00:00
using Wabbajack.BuildServer.Models;
using Wabbajack.BuildServer.Models.JobQueue;
using Wabbajack.BuildServer.Models.Jobs;
using Wabbajack.Common;
namespace Wabbajack.BuildServer
{
public class JobManager
{
protected readonly ILogger<JobManager> Logger;
protected readonly DBContext Db;
protected readonly AppSettings Settings;
public JobManager(ILogger<JobManager> logger, DBContext db, AppSettings settings)
{
Db = db;
Logger = logger;
Settings = settings;
}
public void StartJobRunners()
{
2020-01-16 05:06:25 +00:00
if (Settings.MinimalMode) return;
2020-01-09 04:42:25 +00:00
for (var idx = 0; idx < 2; idx++)
{
Task.Run(async () =>
{
while (true)
{
try
{
var job = await Job.GetNext(Db);
if (job == null)
{
await Task.Delay(5000);
continue;
}
2020-01-13 21:11:07 +00:00
Logger.Log(LogLevel.Information, $"Starting job: {job.Payload.Description}");
2020-01-09 04:42:25 +00:00
JobResult result;
try
{
result = await job.Payload.Execute(Db, Settings);
}
catch (Exception ex)
{
Logger.Log(LogLevel.Error, ex, $"Error while running job: {job.Payload.Description}");
result = JobResult.Error(ex);
}
await Job.Finish(Db, job, result);
}
catch (Exception ex)
{
2020-01-13 21:11:07 +00:00
Logger.Log(LogLevel.Error, ex, $"Error getting or updating job");
2020-01-09 04:42:25 +00:00
}
}
});
}
}
public async Task JobScheduler()
{
2020-01-16 05:06:25 +00:00
if (Settings.MinimalMode) return;
2020-01-09 04:42:25 +00:00
Utils.LogMessages.Subscribe(msg => Logger.Log(LogLevel.Information, msg.ToString()));
while (true)
{
await KillOrphanedJobs();
2020-01-11 04:34:01 +00:00
await ScheduledJob<GetNexusUpdatesJob>(TimeSpan.FromHours(2), Job.JobPriority.High);
await ScheduledJob<UpdateModLists>(TimeSpan.FromMinutes(30), Job.JobPriority.High);
await ScheduledJob<EnqueueAllArchives>(TimeSpan.FromHours(2), Job.JobPriority.Low);
await ScheduledJob<EnqueueAllGameFiles>(TimeSpan.FromHours(24), Job.JobPriority.High);
2020-01-13 23:06:09 +00:00
await ScheduledJob<EnqueueRecentFiles>(TimeSpan.FromHours(6), Job.JobPriority.Low);
await ScheduledJob<IndexDynDOLOD>(TimeSpan.FromHours(1), Job.JobPriority.Normal);
2020-01-09 04:42:25 +00:00
await Task.Delay(10000);
}
}
private async Task KillOrphanedJobs()
{
try
{
var started = await Db.Jobs.AsQueryable()
.Where(j => j.Started != null && j.Ended == null)
.ToListAsync();
foreach (var job in started)
{
var runtime = DateTime.Now - job.Started;
if (runtime > TimeSpan.FromMinutes(30))
{
await Job.Finish(Db, job, JobResult.Error(new Exception($"Timeout after {runtime.Value.TotalMinutes}")));
}
}
}
catch (Exception ex)
{
2020-01-09 12:14:48 +00:00
Logger.Log(LogLevel.Error, ex, "Error in JobScheduler when scheduling KillOrphanedJobs");
2020-01-09 04:42:25 +00:00
}
}
2020-01-09 12:14:48 +00:00
2020-01-11 04:34:01 +00:00
private async Task ScheduledJob<T>(TimeSpan span, Job.JobPriority priority) where T : AJobPayload, new()
2020-01-09 04:42:25 +00:00
{
if (!Settings.RunBackEndJobs && typeof(T).ImplementsInterface(typeof(IBackEndJob))) return;
if (!Settings.RunFrontEndJobs && typeof(T).ImplementsInterface(typeof(IFrontEndJob))) return;
2020-01-09 04:42:25 +00:00
try
{
2020-01-09 12:14:48 +00:00
var jobs = await Db.Jobs.AsQueryable()
.Where(j => j.Payload is T)
.OrderByDescending(j => j.Created)
.Take(10)
2020-01-09 04:42:25 +00:00
.ToListAsync();
2020-01-09 12:14:48 +00:00
foreach (var job in jobs)
2020-01-09 04:42:25 +00:00
{
2020-01-09 12:14:48 +00:00
if (job.Started == null || job.Ended == null) return;
if (DateTime.Now - job.Ended < span) return;
2020-01-09 04:42:25 +00:00
}
2020-01-09 12:14:48 +00:00
await Db.Jobs.InsertOneAsync(new Job
{
2020-01-11 04:34:01 +00:00
Priority = priority,
2020-01-09 12:14:48 +00:00
Payload = new T()
});
2020-01-09 04:42:25 +00:00
}
catch (Exception ex)
{
2020-01-09 12:14:48 +00:00
Logger.Log(LogLevel.Error, ex, $"Error in JobScheduler when scheduling {typeof(T).Name}");
2020-01-09 04:42:25 +00:00
}
}
2020-01-09 12:14:48 +00:00
2020-01-09 04:42:25 +00:00
}
}