[backend/queue] Clean completed jobs in a cron task instead of doing it at job completion
This prevents database deadlocks that can occur when many jobs are being processed simultaneously.
This commit is contained in:
parent
61d6f73e90
commit
97532c2b22
2 changed files with 40 additions and 15 deletions
|
@ -21,9 +21,11 @@ public class QueueService(
|
||||||
) : BackgroundService
|
) : BackgroundService
|
||||||
{
|
{
|
||||||
private readonly List<IPostgresJobQueue> _queues = [];
|
private readonly List<IPostgresJobQueue> _queues = [];
|
||||||
|
|
||||||
|
public IEnumerable<string> QueueNames => _queues.Select(p => p.Name);
|
||||||
|
|
||||||
public readonly BackgroundTaskQueue BackgroundTaskQueue = new();
|
public readonly BackgroundTaskQueue BackgroundTaskQueue = new();
|
||||||
public readonly DeliverQueue DeliverQueue = new();
|
public readonly DeliverQueue DeliverQueue = new();
|
||||||
|
|
||||||
public readonly InboxQueue InboxQueue = new();
|
public readonly InboxQueue InboxQueue = new();
|
||||||
public readonly PreDeliverQueue PreDeliverQueue = new();
|
public readonly PreDeliverQueue PreDeliverQueue = new();
|
||||||
|
|
||||||
|
@ -493,16 +495,6 @@ public class PostgresJobQueue<T>(
|
||||||
db.ChangeTracker.Clear();
|
db.ChangeTracker.Clear();
|
||||||
db.Update(job);
|
db.Update(job);
|
||||||
await db.SaveChangesAsync(token);
|
await db.SaveChangesAsync(token);
|
||||||
|
|
||||||
await db.Jobs.Where(p => p.Queue == name && p.Status == Job.JobStatus.Completed)
|
|
||||||
.OrderByDescending(p => p.FinishedAt)
|
|
||||||
.Skip(10)
|
|
||||||
.ExecuteDeleteAsync(token);
|
|
||||||
|
|
||||||
await db.Jobs.Where(p => p.Queue == name && p.Status == Job.JobStatus.Failed)
|
|
||||||
.OrderByDescending(p => p.FinishedAt)
|
|
||||||
.Skip(100)
|
|
||||||
.ExecuteDeleteAsync(token);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task EnqueueAsync(T jobData)
|
public async Task EnqueueAsync(T jobData)
|
||||||
|
|
33
Iceshrimp.Backend/Core/Tasks/JobCleanupTask.cs
Normal file
33
Iceshrimp.Backend/Core/Tasks/JobCleanupTask.cs
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
|
using Iceshrimp.Backend.Core.Database;
|
||||||
|
using Iceshrimp.Backend.Core.Database.Tables;
|
||||||
|
using Iceshrimp.Backend.Core.Services;
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
|
||||||
|
namespace Iceshrimp.Backend.Core.Tasks;
|
||||||
|
|
||||||
|
[SuppressMessage("ReSharper", "UnusedType.Global", Justification = "Instantiated at runtime by CronService")]
|
||||||
|
public class JobCleanupTask : ICronTask
|
||||||
|
{
|
||||||
|
public async Task Invoke(IServiceProvider provider)
|
||||||
|
{
|
||||||
|
var db = provider.GetRequiredService<DatabaseContext>();
|
||||||
|
var queue = provider.GetRequiredService<QueueService>();
|
||||||
|
|
||||||
|
foreach (var name in queue.QueueNames)
|
||||||
|
{
|
||||||
|
await db.Jobs.Where(p => p.Queue == name && p.Status == Job.JobStatus.Completed)
|
||||||
|
.OrderByDescending(p => p.FinishedAt)
|
||||||
|
.Skip(10)
|
||||||
|
.ExecuteDeleteAsync();
|
||||||
|
|
||||||
|
await db.Jobs.Where(p => p.Queue == name && p.Status == Job.JobStatus.Failed)
|
||||||
|
.OrderByDescending(p => p.FinishedAt)
|
||||||
|
.Skip(100)
|
||||||
|
.ExecuteDeleteAsync();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public CronTaskType Type => CronTaskType.Interval;
|
||||||
|
public TimeSpan Trigger => TimeSpan.FromMinutes(15);
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue