using NSubstitute; using SIGCM2.Application.Abstractions.Persistence; using SIGCM2.Application.Audit; using SIGCM2.Application.Secciones.Deactivate; using SIGCM2.Domain.Entities; using SIGCM2.Domain.Exceptions; namespace SIGCM2.Application.Tests.Secciones.Deactivate; public class DeactivateSeccionCommandHandlerTests { private readonly ISeccionRepository _repo = Substitute.For(); private readonly IMedioRepository _medioRepo = Substitute.For(); private readonly IAuditLogger _audit = Substitute.For(); private readonly DeactivateSeccionCommandHandler _handler; private static Seccion MakeSeccion(int id = 1, bool activo = true) => new(id, 1, "COD" + id, "Nombre", "clasificados", activo, DateTime.UtcNow, null); private static Medio MakeMedio(int id = 1, bool activo = true) => new(id, "COD" + id, "Medio " + id, TipoMedio.Diario, null, activo, DateTime.UtcNow, null); public DeactivateSeccionCommandHandlerTests() { _handler = new DeactivateSeccionCommandHandler(_repo, _medioRepo, _audit); // Default: medio is active _medioRepo.GetByIdAsync(Arg.Any(), Arg.Any()).Returns(MakeMedio(1, true)); } [Fact] public async Task Handle_NotFound_ThrowsSeccionNotFoundException() { _repo.GetByIdAsync(999, Arg.Any()).Returns((Seccion?)null); await Assert.ThrowsAsync( () => _handler.Handle(new DeactivateSeccionCommand(999))); } [Fact] public async Task Handle_AlreadyInactive_IsIdempotentAndDoesNotCallUpdateAsync() { _repo.GetByIdAsync(1, Arg.Any()).Returns(MakeSeccion(1, false)); await _handler.Handle(new DeactivateSeccionCommand(1)); await _repo.DidNotReceive().UpdateAsync(Arg.Any(), Arg.Any()); } [Fact] public async Task Handle_AlreadyInactive_DoesNotWriteAuditEvent() { _repo.GetByIdAsync(1, Arg.Any()).Returns(MakeSeccion(1, false)); await _handler.Handle(new DeactivateSeccionCommand(1)); await _audit.DidNotReceive().LogAsync( Arg.Any(), Arg.Any(), Arg.Any(), Arg.Any(), Arg.Any()); } [Fact] public async Task Handle_ActiveSeccion_CallsUpdateAsyncWithInactiveEntity() { _repo.GetByIdAsync(1, Arg.Any()).Returns(MakeSeccion(1, true)); await _handler.Handle(new DeactivateSeccionCommand(1)); await _repo.Received(1).UpdateAsync( Arg.Is(s => !s.Activo), Arg.Any()); } [Fact] public async Task Handle_ActiveSeccion_WritesAuditWithDeactivateAction() { _repo.GetByIdAsync(1, Arg.Any()).Returns(MakeSeccion(1, true)); await _handler.Handle(new DeactivateSeccionCommand(1)); await _audit.Received(1).LogAsync( action: "seccion.deactivate", targetType: "Seccion", targetId: "1", metadata: Arg.Any(), ct: Arg.Any()); } [Fact] public async Task Handle_MedioInactivo_ThrowsMedioInactivoExceptionAndNoAuditLogged() { _repo.GetByIdAsync(1, Arg.Any()).Returns(MakeSeccion(1, true)); _medioRepo.GetByIdAsync(1, Arg.Any()).Returns(MakeMedio(1, activo: false)); await Assert.ThrowsAsync( () => _handler.Handle(new DeactivateSeccionCommand(1))); await _audit.DidNotReceive().LogAsync( Arg.Any(), Arg.Any(), Arg.Any(), Arg.Any(), Arg.Any()); } }