* use browser's accept-language header * add fill with group mode to schedule items * update dependencies * fixes * fix tests
133 lines
5.3 KiB
C#
133 lines
5.3 KiB
C#
using System.Threading.Channels;
|
|
using ErsatzTV.Application.Playouts;
|
|
using ErsatzTV.Core;
|
|
using ErsatzTV.Core.Domain;
|
|
using ErsatzTV.Core.Scheduling;
|
|
using ErsatzTV.Infrastructure.Data;
|
|
using Microsoft.EntityFrameworkCore;
|
|
using static ErsatzTV.Application.ProgramSchedules.Mapper;
|
|
|
|
namespace ErsatzTV.Application.ProgramSchedules;
|
|
|
|
public class ReplaceProgramScheduleItemsHandler : ProgramScheduleItemCommandBase,
|
|
IRequestHandler<ReplaceProgramScheduleItems, Either<BaseError, IEnumerable<ProgramScheduleItemViewModel>>>
|
|
{
|
|
private readonly ChannelWriter<IBackgroundServiceRequest> _channel;
|
|
private readonly IDbContextFactory<TvContext> _dbContextFactory;
|
|
|
|
public ReplaceProgramScheduleItemsHandler(
|
|
IDbContextFactory<TvContext> dbContextFactory,
|
|
ChannelWriter<IBackgroundServiceRequest> channel)
|
|
{
|
|
_dbContextFactory = dbContextFactory;
|
|
_channel = channel;
|
|
}
|
|
|
|
public async Task<Either<BaseError, IEnumerable<ProgramScheduleItemViewModel>>> Handle(
|
|
ReplaceProgramScheduleItems request,
|
|
CancellationToken cancellationToken)
|
|
{
|
|
await using TvContext dbContext = await _dbContextFactory.CreateDbContextAsync(cancellationToken);
|
|
Validation<BaseError, ProgramSchedule> validation = await Validate(dbContext, request);
|
|
return await validation.Apply(ps => PersistItems(dbContext, request, ps));
|
|
}
|
|
|
|
private async Task<IEnumerable<ProgramScheduleItemViewModel>> PersistItems(
|
|
TvContext dbContext,
|
|
ReplaceProgramScheduleItems request,
|
|
ProgramSchedule programSchedule)
|
|
{
|
|
dbContext.RemoveRange(programSchedule.Items);
|
|
programSchedule.Items = request.Items.Map(i => BuildItem(programSchedule, i.Index, i)).ToList();
|
|
|
|
await dbContext.SaveChangesAsync();
|
|
|
|
// refresh any playouts that use this schedule
|
|
foreach (Playout playout in programSchedule.Playouts)
|
|
{
|
|
await _channel.WriteAsync(new BuildPlayout(playout.Id, PlayoutBuildMode.Refresh));
|
|
}
|
|
|
|
return programSchedule.Items.Map(ProjectToViewModel);
|
|
}
|
|
|
|
private static Task<Validation<BaseError, ProgramSchedule>> Validate(
|
|
TvContext dbContext,
|
|
ReplaceProgramScheduleItems request) =>
|
|
ProgramScheduleMustExist(dbContext, request.ProgramScheduleId)
|
|
.BindT(programSchedule => PlayoutModesMustBeValid(request, programSchedule))
|
|
.BindT(programSchedule => CollectionTypesMustBeValid(request, programSchedule))
|
|
.BindT(programSchedule => PlaybackOrdersMustBeValid(request, programSchedule))
|
|
.BindT(programSchedule => FillerConfigurationsMustBeValid(dbContext, request, programSchedule));
|
|
|
|
private static Validation<BaseError, ProgramSchedule> PlayoutModesMustBeValid(
|
|
ReplaceProgramScheduleItems request,
|
|
ProgramSchedule programSchedule) =>
|
|
request.Items.Map(item => PlayoutModeMustBeValid(item, programSchedule)).Sequence()
|
|
.Map(_ => programSchedule);
|
|
|
|
private static Validation<BaseError, ProgramSchedule> CollectionTypesMustBeValid(
|
|
ReplaceProgramScheduleItems request,
|
|
ProgramSchedule programSchedule) =>
|
|
request.Items.Map(item => CollectionTypeMustBeValid(item, programSchedule)).Sequence()
|
|
.Map(_ => programSchedule);
|
|
|
|
private static async Task<Validation<BaseError, ProgramSchedule>> FillerConfigurationsMustBeValid(
|
|
TvContext dbContext,
|
|
ReplaceProgramScheduleItems request,
|
|
ProgramSchedule programSchedule)
|
|
{
|
|
foreach (ReplaceProgramScheduleItem item in request.Items)
|
|
{
|
|
Either<BaseError, ProgramSchedule> result = await FillerConfigurationMustBeValid(
|
|
dbContext,
|
|
item,
|
|
programSchedule);
|
|
if (result.IsLeft)
|
|
{
|
|
return result.ToValidation();
|
|
}
|
|
}
|
|
|
|
return programSchedule;
|
|
}
|
|
|
|
private static Validation<BaseError, ProgramSchedule> PlaybackOrdersMustBeValid(
|
|
ReplaceProgramScheduleItems request,
|
|
ProgramSchedule programSchedule)
|
|
{
|
|
var keyOrders = new Dictionary<CollectionKey, System.Collections.Generic.HashSet<PlaybackOrder>>();
|
|
foreach (ReplaceProgramScheduleItem item in request.Items)
|
|
{
|
|
var key = new CollectionKey(
|
|
item.CollectionType,
|
|
item.CollectionId,
|
|
item.MediaItemId,
|
|
item.MultiCollectionId,
|
|
item.SmartCollectionId);
|
|
|
|
if (keyOrders.TryGetValue(key, out System.Collections.Generic.HashSet<PlaybackOrder> playbackOrders))
|
|
{
|
|
playbackOrders.Add(item.PlaybackOrder);
|
|
keyOrders[key] = playbackOrders;
|
|
}
|
|
else
|
|
{
|
|
keyOrders.Add(key, new System.Collections.Generic.HashSet<PlaybackOrder> { item.PlaybackOrder });
|
|
}
|
|
}
|
|
|
|
return Optional(keyOrders.Values.Count(set => set.Count != 1))
|
|
.Filter(count => count == 0)
|
|
.Map(_ => programSchedule)
|
|
.ToValidation<BaseError>("A collection must not use multiple playback orders");
|
|
}
|
|
|
|
private sealed record CollectionKey(
|
|
ProgramScheduleItemCollectionType CollectionType,
|
|
int? CollectionId,
|
|
int? MediaItemId,
|
|
int? MultiCollectionId,
|
|
int? SmartCollectionId);
|
|
}
|