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;
|
2020-01-19 05:51:12 +00:00
|
|
|
|
using Nettle;
|
2020-01-26 04:50:17 +00:00
|
|
|
|
using Wabbajack.BuildServer.Model.Models;
|
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;
|
2020-01-26 04:50:17 +00:00
|
|
|
|
protected SqlService Sql;
|
2020-01-09 04:42:25 +00:00
|
|
|
|
|
2020-01-26 04:50:17 +00:00
|
|
|
|
public JobManager(ILogger<JobManager> logger, DBContext db, SqlService sql, AppSettings settings)
|
2020-01-09 04:42:25 +00:00
|
|
|
|
{
|
|
|
|
|
Db = db;
|
|
|
|
|
Logger = logger;
|
|
|
|
|
Settings = settings;
|
2020-01-26 04:50:17 +00:00
|
|
|
|
Sql = sql;
|
2020-01-09 04:42:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-01-26 04:50:17 +00:00
|
|
|
|
|
2020-01-09 04:42:25 +00:00
|
|
|
|
public void StartJobRunners()
|
|
|
|
|
{
|
2020-01-26 04:50:17 +00:00
|
|
|
|
if (!Settings.JobRunner) 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
|
|
|
|
|
{
|
2020-01-26 04:50:17 +00:00
|
|
|
|
result = await job.Payload.Execute(Db, Sql, Settings);
|
2020-01-09 04:42:25 +00:00
|
|
|
|
}
|
|
|
|
|
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()
|
|
|
|
|
{
|
|
|
|
|
Utils.LogMessages.Subscribe(msg => Logger.Log(LogLevel.Information, msg.ToString()));
|
2020-01-26 04:50:17 +00:00
|
|
|
|
if (!Settings.JobScheduler) return;
|
2020-01-09 04:42:25 +00:00
|
|
|
|
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);
|
2020-01-12 05:00:41 +00:00
|
|
|
|
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);
|
2020-01-22 23:08:22 +00:00
|
|
|
|
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
|
|
|
|
{
|
2020-01-19 05:51:12 +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
|
|
|
|
}
|
|
|
|
|
}
|