using System.IO.Abstractions; using System.Threading.Channels; using ErsatzTV.Application.Channels; using ErsatzTV.Core; using ErsatzTV.Core.Domain; using ErsatzTV.Core.Scheduling; using ErsatzTV.Infrastructure.Data; using ErsatzTV.Infrastructure.Extensions; using Microsoft.EntityFrameworkCore; using Channel = ErsatzTV.Core.Domain.Channel; namespace ErsatzTV.Application.Playouts; public class CreateSequentialPlayoutHandler( IFileSystem fileSystem, ChannelWriter channel, IDbContextFactory dbContextFactory) : IRequestHandler> { public async Task> Handle( CreateSequentialPlayout request, CancellationToken cancellationToken) { await using TvContext dbContext = await dbContextFactory.CreateDbContextAsync(cancellationToken); Validation validation = await Validate(dbContext, request, cancellationToken); return await validation.Apply(playout => PersistPlayout(dbContext, playout, cancellationToken)); } private async Task PersistPlayout( TvContext dbContext, Playout playout, CancellationToken cancellationToken) { await dbContext.Playouts.AddAsync(playout, cancellationToken); await dbContext.SaveChangesAsync(cancellationToken); await channel.WriteAsync(new BuildPlayout(playout.Id, PlayoutBuildMode.Reset), cancellationToken); if (playout.Channel.PlayoutMode is ChannelPlayoutMode.OnDemand) { await channel.WriteAsync( new TimeShiftOnDemandPlayout(playout.Id, DateTimeOffset.Now, false), cancellationToken); } await channel.WriteAsync(new RefreshChannelList(), cancellationToken); return new CreatePlayoutResponse(playout.Id); } private async Task> Validate( TvContext dbContext, CreateSequentialPlayout request, CancellationToken cancellationToken) => (await ValidateChannel(dbContext, request, cancellationToken), ValidateYamlFile(request), ValidateScheduleKind(request)) .Apply((channel, yamlFile, scheduleKind) => new Playout { ChannelId = channel.Id, ScheduleFile = yamlFile, ScheduleKind = scheduleKind, Seed = new Random().Next() }); private static Task> ValidateChannel( TvContext dbContext, CreateSequentialPlayout createSequentialPlayout, CancellationToken cancellationToken) => dbContext.Channels .Include(c => c.Playouts) .SelectOneAsync(c => c.Id, c => c.Id == createSequentialPlayout.ChannelId, cancellationToken) .Map(o => o.ToValidation("Channel does not exist")) .BindT(ChannelMustNotHavePlayouts); private static Validation ChannelMustNotHavePlayouts(Channel channel) => Optional(channel.Playouts.Count) .Filter(count => count == 0) .Map(_ => channel) .ToValidation("Channel already has one playout"); private Validation ValidateYamlFile(CreateSequentialPlayout request) { if (!fileSystem.File.Exists(request.ScheduleFile)) { return BaseError.New("Sequential schedule does not exist!"); } return request.ScheduleFile; } private static Validation ValidateScheduleKind( CreateSequentialPlayout createSequentialPlayout) => Optional(createSequentialPlayout.ScheduleKind) .Filter(scheduleKind => scheduleKind == PlayoutScheduleKind.Sequential) .ToValidation("[ScheduleKind] must be Sequential"); }