mirror of https://github.com/ErsatzTV/ErsatzTV.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
165 lines
7.0 KiB
165 lines
7.0 KiB
using System.Threading.Channels; |
|
using ErsatzTV.Core; |
|
using ErsatzTV.Core.Domain; |
|
using ErsatzTV.Core.Scheduling; |
|
using ErsatzTV.Infrastructure.Data; |
|
using ErsatzTV.Infrastructure.Extensions; |
|
using Microsoft.EntityFrameworkCore; |
|
using Microsoft.Extensions.Logging; |
|
|
|
namespace ErsatzTV.Application.Playouts; |
|
|
|
public class ReplacePlayoutAlternateScheduleItemsHandler : |
|
IRequestHandler<ReplacePlayoutAlternateScheduleItems, Either<BaseError, Unit>> |
|
{ |
|
private readonly ChannelWriter<IBackgroundServiceRequest> _channel; |
|
private readonly IDbContextFactory<TvContext> _dbContextFactory; |
|
private readonly ILogger<ReplacePlayoutAlternateScheduleItemsHandler> _logger; |
|
|
|
public ReplacePlayoutAlternateScheduleItemsHandler( |
|
IDbContextFactory<TvContext> dbContextFactory, |
|
ChannelWriter<IBackgroundServiceRequest> channel, |
|
ILogger<ReplacePlayoutAlternateScheduleItemsHandler> logger) |
|
{ |
|
_dbContextFactory = dbContextFactory; |
|
_channel = channel; |
|
_logger = logger; |
|
} |
|
|
|
public async Task<Either<BaseError, Unit>> Handle( |
|
ReplacePlayoutAlternateScheduleItems request, |
|
CancellationToken cancellationToken) |
|
{ |
|
// TODO: validate that items is not empty |
|
|
|
try |
|
{ |
|
await using TvContext dbContext = await _dbContextFactory.CreateDbContextAsync(cancellationToken); |
|
|
|
Option<Playout> maybePlayout = await dbContext.Playouts |
|
.Include(p => p.ProgramSchedule) |
|
.Include(p => p.ProgramScheduleAlternates) |
|
.ThenInclude(p => p.ProgramSchedule) |
|
.SelectOneAsync(p => p.Id, p => p.Id == request.PlayoutId); |
|
|
|
foreach (Playout playout in maybePlayout) |
|
{ |
|
var existingScheduleMap = new Dictionary<DateTimeOffset, ProgramSchedule>(); |
|
var daysToCheck = new List<DateTimeOffset>(); |
|
|
|
Option<PlayoutItem> maybeLastPlayoutItem = await dbContext.PlayoutItems |
|
.Filter(pi => pi.PlayoutId == request.PlayoutId) |
|
.OrderByDescending(pi => pi.Start) |
|
.FirstOrDefaultAsync(cancellationToken) |
|
.Map(Optional); |
|
|
|
foreach (PlayoutItem lastPlayoutItem in maybeLastPlayoutItem) |
|
{ |
|
DateTimeOffset start = DateTimeOffset.Now; |
|
daysToCheck = Enumerable.Range(0, (lastPlayoutItem.StartOffset - start).Days + 1) |
|
.Select(d => start.AddDays(d)) |
|
.ToList(); |
|
|
|
foreach (DateTimeOffset dayToCheck in daysToCheck) |
|
{ |
|
ProgramSchedule schedule = PlayoutScheduleSelector.GetProgramScheduleFor( |
|
playout.ProgramSchedule, |
|
playout.ProgramScheduleAlternates, |
|
dayToCheck); |
|
|
|
existingScheduleMap.Add(dayToCheck, schedule); |
|
} |
|
} |
|
|
|
// exclude highest index |
|
int maxIndex = request.Items.Map(x => x.Index).Max(); |
|
ReplacePlayoutAlternateSchedule highest = request.Items.First(x => x.Index == maxIndex); |
|
|
|
ProgramScheduleAlternate[] existing = playout.ProgramScheduleAlternates.ToArray(); |
|
|
|
var incoming = request.Items.Except(new[] { highest }).ToList(); |
|
|
|
var toAdd = incoming.Filter(x => existing.All(e => e.Id != x.Id)).ToList(); |
|
var toRemove = existing.Filter(e => incoming.All(m => m.Id != e.Id)).ToList(); |
|
var toUpdate = incoming.Except(toAdd).ToList(); |
|
|
|
playout.ProgramScheduleAlternates.RemoveAll(toRemove.Contains); |
|
|
|
foreach (ReplacePlayoutAlternateSchedule add in toAdd) |
|
{ |
|
playout.ProgramScheduleAlternates.Add( |
|
new ProgramScheduleAlternate |
|
{ |
|
PlayoutId = playout.Id, |
|
Index = add.Index, |
|
ProgramScheduleId = add.ProgramScheduleId, |
|
DaysOfWeek = add.DaysOfWeek, |
|
DaysOfMonth = add.DaysOfMonth, |
|
MonthsOfYear = add.MonthsOfYear |
|
}); |
|
} |
|
|
|
foreach (ReplacePlayoutAlternateSchedule update in toUpdate) |
|
{ |
|
foreach (ProgramScheduleAlternate ex in existing.Filter(x => x.Id == update.Id)) |
|
{ |
|
ex.Index = update.Index; |
|
ex.ProgramScheduleId = update.ProgramScheduleId; |
|
ex.DaysOfWeek = update.DaysOfWeek; |
|
ex.DaysOfMonth = update.DaysOfMonth; |
|
ex.MonthsOfYear = update.MonthsOfYear; |
|
} |
|
} |
|
|
|
// save highest index directly to playout |
|
if (playout.ProgramScheduleId != highest.ProgramScheduleId) |
|
{ |
|
playout.ProgramScheduleId = highest.ProgramScheduleId; |
|
} |
|
|
|
await dbContext.SaveChangesAsync(cancellationToken); |
|
|
|
// load newly-added schedules |
|
foreach (ProgramScheduleAlternate alternate in playout.ProgramScheduleAlternates |
|
.Where(alternate => alternate.ProgramSchedule is null)) |
|
{ |
|
await dbContext.Entry(alternate).Reference(a => a.ProgramSchedule).LoadAsync(cancellationToken); |
|
} |
|
|
|
foreach (PlayoutItem _ in maybeLastPlayoutItem) |
|
{ |
|
foreach (DateTimeOffset dayToCheck in daysToCheck) |
|
{ |
|
ProgramSchedule schedule = PlayoutScheduleSelector.GetProgramScheduleFor( |
|
playout.ProgramSchedule, |
|
playout.ProgramScheduleAlternates, |
|
dayToCheck); |
|
|
|
if (existingScheduleMap.TryGetValue(dayToCheck, out ProgramSchedule existingValue) && |
|
existingValue.Id != schedule.Id) |
|
{ |
|
_logger.LogInformation( |
|
"Alternate schedule change detected for day {Day}, schedule {One} => {Two}; will refresh playout", |
|
dayToCheck, |
|
existingValue.Name, |
|
schedule.Name); |
|
|
|
await _channel.WriteAsync( |
|
new BuildPlayout(request.PlayoutId, PlayoutBuildMode.Refresh), |
|
cancellationToken); |
|
|
|
break; |
|
} |
|
} |
|
} |
|
} |
|
|
|
return Unit.Default; |
|
} |
|
catch (Exception ex) |
|
{ |
|
_logger.LogError(ex, "Error saving alternate schedule items"); |
|
return BaseError.New(ex.Message); |
|
} |
|
} |
|
}
|
|
|