using ErsatzTV.Core; using ErsatzTV.Core.Domain.Scheduling; using ErsatzTV.Infrastructure.Data; using Microsoft.EntityFrameworkCore; namespace ErsatzTV.Application.Scheduling; public class CreateBlockGroupHandler(IDbContextFactory dbContextFactory) : IRequestHandler> { public async Task> Handle( CreateBlockGroup request, CancellationToken cancellationToken) { await using TvContext dbContext = await dbContextFactory.CreateDbContextAsync(cancellationToken); Validation validation = await Validate(dbContext, request); return await validation.Apply(profile => PersistBlockGroup(dbContext, profile)); } private static async Task PersistBlockGroup(TvContext dbContext, BlockGroup blockGroup) { await dbContext.BlockGroups.AddAsync(blockGroup); await dbContext.SaveChangesAsync(); return Mapper.ProjectToViewModel(blockGroup); } private static Task> Validate(TvContext dbContext, CreateBlockGroup request) => ValidateName(dbContext, request).MapT(name => new BlockGroup { Name = name, Blocks = [] }); private static async Task> ValidateName( TvContext dbContext, CreateBlockGroup createBlockGroup) { Validation result1 = createBlockGroup.NotEmpty(c => c.Name) .Bind(_ => createBlockGroup.NotLongerThan(50)(c => c.Name)); int duplicateNameCount = await dbContext.BlockGroups .CountAsync(ps => ps.Name == createBlockGroup.Name); var result2 = Optional(duplicateNameCount) .Where(count => count == 0) .ToValidation("Block group name must be unique"); return (result1, result2).Apply((_, _) => createBlockGroup.Name); } }