using System.Linq; using System.Threading; using System.Threading.Channels; using System.Threading.Tasks; using ErsatzTV.Core; using ErsatzTV.Core.Domain; using ErsatzTV.Infrastructure.Data; using ErsatzTV.Infrastructure.Extensions; using LanguageExt; using MediatR; using Microsoft.EntityFrameworkCore; using static LanguageExt.Prelude; using Channel = ErsatzTV.Core.Domain.Channel; namespace ErsatzTV.Application.Playouts.Commands { public class CreatePlayoutHandler : IRequestHandler> { private readonly ChannelWriter _channel; private readonly IDbContextFactory _dbContextFactory; public CreatePlayoutHandler( ChannelWriter channel, IDbContextFactory dbContextFactory) { _channel = channel; _dbContextFactory = dbContextFactory; } public async Task> Handle( CreatePlayout request, CancellationToken cancellationToken) { await using TvContext dbContext = _dbContextFactory.CreateDbContext(); Validation validation = await Validate(dbContext, request); return await validation.Apply(playout => PersistPlayout(dbContext, playout)); } private async Task PersistPlayout(TvContext dbContext, Playout playout) { await dbContext.Playouts.AddAsync(playout); await dbContext.SaveChangesAsync(); await _channel.WriteAsync(new BuildPlayout(playout.Id)); return new CreatePlayoutResponse(playout.Id); } private async Task> Validate(TvContext dbContext, CreatePlayout request) => (await ValidateChannel(dbContext, request), await ValidateProgramSchedule(dbContext, request), ValidatePlayoutType(request)) .Apply( (channel, programSchedule, playoutType) => new Playout { ChannelId = channel.Id, ProgramScheduleId = programSchedule.Id, ProgramSchedulePlayoutType = playoutType }); private static Task> ValidateChannel( TvContext dbContext, CreatePlayout createPlayout) => dbContext.Channels .Include(c => c.Playouts) .SelectOneAsync(c => c.Id, c => c.Id == createPlayout.ChannelId) .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 static Task> ValidateProgramSchedule( TvContext dbContext, CreatePlayout createPlayout) => dbContext.ProgramSchedules .Include(ps => ps.Items) .SelectOneAsync(ps => ps.Id, ps => ps.Id == createPlayout.ProgramScheduleId) .Map(o => o.ToValidation("Program schedule does not exist")) .BindT(ProgramScheduleMustHaveItems); private static Validation ProgramScheduleMustHaveItems( ProgramSchedule programSchedule) => Optional(programSchedule) .Filter(ps => ps.Items.Any()) .ToValidation("Program schedule must have items"); private static Validation ValidatePlayoutType( CreatePlayout createPlayout) => Optional(createPlayout.ProgramSchedulePlayoutType) .Filter(playoutType => playoutType != ProgramSchedulePlayoutType.None) .ToValidation("[ProgramSchedulePlayoutType] must not be None"); } }