Added inital job entity and services. Added released works API integration.
All checks were successful
ci / build-test (push) Successful in 2m21s
ci / publish-image (push) Successful in 2m19s

This commit is contained in:
2026-03-27 01:32:39 -04:00
parent 1c016ac62e
commit d9e421178f
36 changed files with 5596 additions and 43 deletions

View File

@@ -0,0 +1,80 @@
using JSMR.Application.Jobs;
using JSMR.Domain.Entities;
using JSMR.Domain.Enums;
using Microsoft.EntityFrameworkCore;
namespace JSMR.Infrastructure.Data.Repositories.Jobs;
public sealed class JobProgressWriter(AppDbContext dbContext) : IJobProgressWriter
{
public async Task SetStepAsync(int jobId, string step, CancellationToken canellationToken)
{
Job? job = await dbContext.Jobs.FirstOrDefaultAsync(x => x.Id == jobId, canellationToken);
if (job is null)
return;
job.CurrentStep = step;
job.HeartbeatUtc = DateTime.UtcNow;
await dbContext.SaveChangesAsync(canellationToken);
}
public async Task SetProgressAsync(int jobId, int? current, int? total, CancellationToken canellationToken)
{
Job? job = await dbContext.Jobs.FirstOrDefaultAsync(x => x.Id == jobId, canellationToken);
if (job is null)
return;
job.ProgressCurrent = current;
job.ProgressTotal = total;
job.HeartbeatUtc = DateTime.UtcNow;
await dbContext.SaveChangesAsync(canellationToken);
}
public async Task SetHeartbeatAsync(int jobId, CancellationToken canellationToken)
{
Job? job = await dbContext.Jobs.FirstOrDefaultAsync(x => x.Id == jobId, canellationToken);
if (job is null)
return;
job.HeartbeatUtc = DateTime.UtcNow;
await dbContext.SaveChangesAsync(canellationToken);
}
public async Task CompleteAsync(int jobId, string? summary, CancellationToken canellationToken)
{
Job? job = await dbContext.Jobs.FirstOrDefaultAsync(x => x.Id == jobId, canellationToken);
if (job is null)
return;
job.Status = JobStatus.Succeeded;
job.CompletedUtc = DateTime.UtcNow;
job.HeartbeatUtc = DateTime.UtcNow;
job.ResultSummary = summary;
job.CurrentStep = "Completed";
await dbContext.SaveChangesAsync(canellationToken);
}
public async Task FailAsync(int jobId, string error, CancellationToken canellationToken)
{
Job? job = await dbContext.Jobs.FirstOrDefaultAsync(x => x.Id == jobId, canellationToken);
if (job is null)
return;
job.Status = JobStatus.Failed;
job.CompletedUtc = DateTime.UtcNow;
job.HeartbeatUtc = DateTime.UtcNow;
job.Error = error;
job.CurrentStep = "Failed";
await dbContext.SaveChangesAsync(canellationToken);
}
}

View File

@@ -0,0 +1,52 @@
using JSMR.Application.Jobs;
using JSMR.Domain.Entities;
using JSMR.Domain.Enums;
using Microsoft.EntityFrameworkCore;
namespace JSMR.Infrastructure.Data.Repositories.Jobs;
public sealed class JobRepository(AppDbContext dbContext) : IJobRepository
{
public async Task<Job> AddAsync(Job job, CancellationToken cancellationToken)
{
await dbContext.Jobs.AddAsync(job, cancellationToken);
return job;
}
public Task<Job?> GetByIdAsync(int id, CancellationToken cancellationToken)
=> dbContext.Jobs.FirstOrDefaultAsync(x => x.Id == id, cancellationToken);
public async Task<IReadOnlyList<Job>> GetRecentAsync(int take, CancellationToken cancellationToken)
=> await dbContext.Jobs
.OrderByDescending(x => x.CreatedUtc)
.Take(take)
.ToListAsync(cancellationToken);
public Task<bool> AnyRunningAsync(CancellationToken cancellationToken)
=> dbContext.Jobs.AnyAsync(x => x.Status == JobStatus.Running, cancellationToken);
public async Task<Job?> TryClaimNextQueuedAsync(string workerName, CancellationToken cancellationToken)
{
Job? next = await dbContext.Jobs
.Where(x => x.Status == JobStatus.Queued)
.OrderBy(x => x.CreatedUtc)
.FirstOrDefaultAsync(cancellationToken);
if (next is null)
return null;
next.Status = JobStatus.Running;
next.StartedUtc = DateTime.UtcNow;
next.HeartbeatUtc = DateTime.UtcNow;
next.WorkerName = workerName;
next.AttemptCount += 1;
await dbContext.SaveChangesAsync(cancellationToken);
return next;
}
public Task SaveChangesAsync(CancellationToken ct)
=> dbContext.SaveChangesAsync(ct);
}