Este commit introduce una refactorización significativa en el módulo de
suscripciones para alinear el sistema con reglas de negocio clave:
facturación consolidada por empresa, cobro a mes adelantado con
imputación de ajustes diferida, y una interfaz de usuario más clara.
Backend:
- **Facturación por Empresa:** Se modifica `FacturacionService` para
agrupar las suscripciones por cliente y empresa, generando una
factura consolidada para cada combinación. Esto asegura la correcta
separación fiscal.
- **Imputación de Ajustes:** Se ajusta la lógica para que la facturación
de un período (ej. Septiembre) aplique únicamente los ajustes
pendientes cuya fecha corresponde al período anterior (Agosto).
- **Cierre Secuencial:** Se implementa una validación en
`GenerarFacturacionMensual` que impide generar la facturación de un
período si el anterior no ha sido cerrado, garantizando el orden
cronológico.
- **Emails Consolidados:** El proceso de notificación automática al
generar el cierre ahora envía un único email consolidado por
suscriptor, detallando los cargos de todas sus facturas/empresas.
- **Envío de PDF Individual:** Se refactoriza el endpoint de envío manual
para que opere sobre una `idFactura` individual y adjunte el PDF
correspondiente si existe.
- **Repositorios Mejorados:** Se optimizan y añaden métodos en
`FacturaRepository` y `AjusteRepository` para soportar los nuevos
requisitos de filtrado y consulta de datos consolidados.
Frontend:
- **Separación de Vistas:** La página de "Facturación" se divide en dos:
- `ProcesosPage`: Para acciones masivas (generar cierre, archivo de
débito, procesar respuesta).
- `ConsultaFacturasPage`: Una nueva página dedicada a buscar,
filtrar y gestionar facturas individuales con una interfaz de doble
acordeón (Suscriptor -> Empresa).
- **Filtros Avanzados:** La página `ConsultaFacturasPage` ahora incluye
filtros por nombre de suscriptor, estado de pago y estado de
facturación.
- **Filtros de Fecha por Defecto:** La página de "Cuenta Corriente"
ahora filtra por el mes actual por defecto para mejorar el rendimiento
y la usabilidad.
- **Validación de Fechas:** Se añade lógica en los filtros de fecha para
impedir la selección de rangos inválidos.
- **Validación de Monto de Pago:** El modal de pago manual ahora impide
registrar un monto superior al saldo pendiente de la factura.
218 lines
10 KiB
C#
218 lines
10 KiB
C#
using GestionIntegral.Api.Data;
|
|
using GestionIntegral.Api.Data.Repositories.Suscripciones;
|
|
using GestionIntegral.Api.Dtos.Suscripciones;
|
|
using GestionIntegral.Api.Models.Suscripciones;
|
|
using System.Data;
|
|
|
|
namespace GestionIntegral.Api.Services.Suscripciones
|
|
{
|
|
public class SuscriptorService : ISuscriptorService
|
|
{
|
|
private readonly ISuscriptorRepository _suscriptorRepository;
|
|
private readonly IFormaPagoRepository _formaPagoRepository;
|
|
private readonly DbConnectionFactory _connectionFactory;
|
|
private readonly ILogger<SuscriptorService> _logger;
|
|
|
|
public SuscriptorService(
|
|
ISuscriptorRepository suscriptorRepository,
|
|
IFormaPagoRepository formaPagoRepository,
|
|
DbConnectionFactory connectionFactory,
|
|
ILogger<SuscriptorService> logger)
|
|
{
|
|
_suscriptorRepository = suscriptorRepository;
|
|
_formaPagoRepository = formaPagoRepository;
|
|
_connectionFactory = connectionFactory;
|
|
_logger = logger;
|
|
}
|
|
|
|
// Helper para mapear Modelo -> DTO, enriqueciendo con el nombre de la forma de pago
|
|
private async Task<SuscriptorDto?> MapToDto(Suscriptor suscriptor)
|
|
{
|
|
if (suscriptor == null) return null;
|
|
|
|
var formaPago = await _formaPagoRepository.GetByIdAsync(suscriptor.IdFormaPagoPreferida);
|
|
|
|
return new SuscriptorDto
|
|
{
|
|
IdSuscriptor = suscriptor.IdSuscriptor,
|
|
NombreCompleto = suscriptor.NombreCompleto,
|
|
Email = suscriptor.Email,
|
|
Telefono = suscriptor.Telefono,
|
|
Direccion = suscriptor.Direccion,
|
|
TipoDocumento = suscriptor.TipoDocumento,
|
|
NroDocumento = suscriptor.NroDocumento,
|
|
CBU = suscriptor.CBU,
|
|
IdFormaPagoPreferida = suscriptor.IdFormaPagoPreferida,
|
|
NombreFormaPagoPreferida = formaPago?.Nombre ?? "Desconocida",
|
|
Observaciones = suscriptor.Observaciones,
|
|
Activo = suscriptor.Activo
|
|
};
|
|
}
|
|
|
|
public async Task<IEnumerable<SuscriptorDto>> ObtenerTodos(string? nombreFilter, string? nroDocFilter, bool soloActivos)
|
|
{
|
|
var suscriptores = await _suscriptorRepository.GetAllAsync(nombreFilter, nroDocFilter, soloActivos);
|
|
var dtosTasks = suscriptores.Select(s => MapToDto(s));
|
|
var dtos = await Task.WhenAll(dtosTasks);
|
|
return dtos.Where(dto => dto != null).Select(dto => dto!);
|
|
}
|
|
|
|
public async Task<SuscriptorDto?> ObtenerPorId(int id)
|
|
{
|
|
var suscriptor = await _suscriptorRepository.GetByIdAsync(id);
|
|
if (suscriptor == null)
|
|
return null;
|
|
return await MapToDto(suscriptor);
|
|
}
|
|
|
|
public async Task<(SuscriptorDto? Suscriptor, string? Error)> Crear(CreateSuscriptorDto createDto, int idUsuario)
|
|
{
|
|
// Validación de Lógica de Negocio
|
|
if (await _suscriptorRepository.ExistsByDocumentoAsync(createDto.TipoDocumento, createDto.NroDocumento))
|
|
{
|
|
return (null, "Ya existe un suscriptor con el mismo tipo y número de documento.");
|
|
}
|
|
|
|
var formaPago = await _formaPagoRepository.GetByIdAsync(createDto.IdFormaPagoPreferida);
|
|
if (formaPago == null || !formaPago.Activo)
|
|
{
|
|
return (null, "La forma de pago seleccionada no es válida o está inactiva.");
|
|
}
|
|
if (formaPago.RequiereCBU && string.IsNullOrWhiteSpace(createDto.CBU))
|
|
{
|
|
return (null, "El CBU es obligatorio para la forma de pago seleccionada.");
|
|
}
|
|
|
|
var nuevoSuscriptor = new Suscriptor
|
|
{
|
|
NombreCompleto = createDto.NombreCompleto,
|
|
Email = createDto.Email,
|
|
Telefono = createDto.Telefono,
|
|
Direccion = createDto.Direccion,
|
|
TipoDocumento = createDto.TipoDocumento,
|
|
NroDocumento = createDto.NroDocumento,
|
|
CBU = createDto.CBU,
|
|
IdFormaPagoPreferida = createDto.IdFormaPagoPreferida,
|
|
Observaciones = createDto.Observaciones,
|
|
IdUsuarioAlta = idUsuario
|
|
};
|
|
|
|
using var connection = _connectionFactory.CreateConnection();
|
|
await (connection as System.Data.Common.DbConnection)!.OpenAsync();
|
|
using var transaction = connection.BeginTransaction();
|
|
|
|
try
|
|
{
|
|
var suscriptorCreado = await _suscriptorRepository.CreateAsync(nuevoSuscriptor, transaction);
|
|
if (suscriptorCreado == null) throw new DataException("La creación en el repositorio devolvió null.");
|
|
|
|
transaction.Commit();
|
|
_logger.LogInformation("Suscriptor ID {IdSuscriptor} creado por Usuario ID {IdUsuario}.", suscriptorCreado.IdSuscriptor, idUsuario);
|
|
|
|
var dtoCreado = await MapToDto(suscriptorCreado);
|
|
return (dtoCreado, null);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
try { transaction.Rollback(); } catch { }
|
|
_logger.LogError(ex, "Error al crear suscriptor: {Nombre}", createDto.NombreCompleto);
|
|
return (null, $"Error interno al crear el suscriptor: {ex.Message}");
|
|
}
|
|
}
|
|
|
|
public async Task<(bool Exito, string? Error)> Actualizar(int id, UpdateSuscriptorDto updateDto, int idUsuario)
|
|
{
|
|
var suscriptorExistente = await _suscriptorRepository.GetByIdAsync(id);
|
|
if (suscriptorExistente == null) return (false, "Suscriptor no encontrado.");
|
|
|
|
if (await _suscriptorRepository.ExistsByDocumentoAsync(updateDto.TipoDocumento, updateDto.NroDocumento, id))
|
|
{
|
|
return (false, "El tipo y número de documento ya pertenecen a otro suscriptor.");
|
|
}
|
|
|
|
var formaPago = await _formaPagoRepository.GetByIdAsync(updateDto.IdFormaPagoPreferida);
|
|
if (formaPago == null || !formaPago.Activo)
|
|
{
|
|
return (false, "La forma de pago seleccionada no es válida o está inactiva.");
|
|
}
|
|
if (formaPago.RequiereCBU && string.IsNullOrWhiteSpace(updateDto.CBU))
|
|
{
|
|
return (false, "El CBU es obligatorio para la forma de pago seleccionada.");
|
|
}
|
|
|
|
// Mapeo DTO -> Modelo
|
|
suscriptorExistente.NombreCompleto = updateDto.NombreCompleto;
|
|
suscriptorExistente.Email = updateDto.Email;
|
|
suscriptorExistente.Telefono = updateDto.Telefono;
|
|
suscriptorExistente.Direccion = updateDto.Direccion;
|
|
suscriptorExistente.TipoDocumento = updateDto.TipoDocumento;
|
|
suscriptorExistente.NroDocumento = updateDto.NroDocumento;
|
|
suscriptorExistente.CBU = updateDto.CBU;
|
|
suscriptorExistente.IdFormaPagoPreferida = updateDto.IdFormaPagoPreferida;
|
|
suscriptorExistente.Observaciones = updateDto.Observaciones;
|
|
suscriptorExistente.IdUsuarioMod = idUsuario;
|
|
suscriptorExistente.FechaMod = DateTime.Now;
|
|
|
|
using var connection = _connectionFactory.CreateConnection();
|
|
await (connection as System.Data.Common.DbConnection)!.OpenAsync();
|
|
using var transaction = connection.BeginTransaction();
|
|
|
|
try
|
|
{
|
|
var actualizado = await _suscriptorRepository.UpdateAsync(suscriptorExistente, transaction);
|
|
if (!actualizado) throw new DataException("La actualización en el repositorio devolvió false.");
|
|
|
|
transaction.Commit();
|
|
_logger.LogInformation("Suscriptor ID {IdSuscriptor} actualizado por Usuario ID {IdUsuario}.", id, idUsuario);
|
|
return (true, null);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
try { transaction.Rollback(); } catch { }
|
|
_logger.LogError(ex, "Error al actualizar suscriptor ID: {IdSuscriptor}", id);
|
|
return (false, $"Error interno al actualizar: {ex.Message}");
|
|
}
|
|
}
|
|
|
|
private async Task<(bool Exito, string? Error)> CambiarEstadoActivo(int id, bool activar, int idUsuario)
|
|
{
|
|
var suscriptor = await _suscriptorRepository.GetByIdAsync(id);
|
|
if (suscriptor == null) return (false, "Suscriptor no encontrado.");
|
|
|
|
if (!activar && await _suscriptorRepository.IsInUseAsync(id))
|
|
{
|
|
return (false, "No se puede desactivar un suscriptor con suscripciones activas.");
|
|
}
|
|
|
|
using var connection = _connectionFactory.CreateConnection();
|
|
await (connection as System.Data.Common.DbConnection)!.OpenAsync();
|
|
using var transaction = connection.BeginTransaction();
|
|
|
|
try
|
|
{
|
|
var actualizado = await _suscriptorRepository.ToggleActivoAsync(id, activar, idUsuario, transaction);
|
|
if (!actualizado) throw new DataException("No se pudo cambiar el estado del suscriptor.");
|
|
|
|
transaction.Commit();
|
|
_logger.LogInformation("El estado del Suscriptor ID {IdSuscriptor} se cambió a {Estado} por el Usuario ID {IdUsuario}.", id, activar ? "Activo" : "Inactivo", idUsuario);
|
|
return (true, null);
|
|
}
|
|
catch(Exception ex)
|
|
{
|
|
try { transaction.Rollback(); } catch { }
|
|
_logger.LogError(ex, "Error al cambiar estado del suscriptor ID: {IdSuscriptor}", id);
|
|
return (false, $"Error interno: {ex.Message}");
|
|
}
|
|
}
|
|
|
|
public Task<(bool Exito, string? Error)> Desactivar(int id, int idUsuario)
|
|
{
|
|
return CambiarEstadoActivo(id, false, idUsuario);
|
|
}
|
|
|
|
public Task<(bool Exito, string? Error)> Activar(int id, int idUsuario)
|
|
{
|
|
return CambiarEstadoActivo(id, true, idUsuario);
|
|
}
|
|
}
|
|
} |