Backend:
Diseño de un AuditoriaController con un patrón para añadir endpoints de historial para diferentes entidades. Implementación de la lógica de servicio y repositorio para obtener datos de las tablas _H para: Usuarios (gral_Usuarios_H) Pagos de Distribuidores (cue_PagosDistribuidor_H) Notas de Crédito/Débito (cue_CreditosDebitos_H) Entradas/Salidas de Distribuidores (dist_EntradasSalidas_H) Entradas/Salidas de Canillitas (dist_EntradasSalidasCanillas_H) Novedades de Canillitas (dist_dtNovedadesCanillas_H) Ajustes Manuales de Saldo (cue_SaldoAjustesHistorial) Tipos de Pago (cue_dtTipopago_H) Canillitas (Maestro) (dist_dtCanillas_H) Distribuidores (Maestro) (dist_dtDistribuidores_H) Empresas (Maestro) (dist_dtEmpresas_H) DTOs específicos para cada tipo de historial, incluyendo NombreUsuarioModifico. Frontend: Servicio auditoriaService.ts con métodos para llamar a cada endpoint de historial. Página AuditoriaGeneralPage.tsx con: Selector de "Tipo de Entidad a Auditar". Filtros comunes (Fechas, Usuario Modificador, Tipo de Modificación, ID Entidad). Un DataGrid que muestra las columnas dinámicamente según el tipo de entidad seleccionada. Lógica para cargar los datos correspondientes. DTOs de historial en TypeScript. Actualizaciones en AppRoutes.tsx y MainLayout.tsx para la nueva sección de Auditoría (restringida a SuperAdmin).
This commit is contained in:
		| @@ -1,6 +1,7 @@ | ||||
| using GestionIntegral.Api.Data; | ||||
| using GestionIntegral.Api.Data.Repositories.Contables; | ||||
| using GestionIntegral.Api.Data.Repositories.Distribucion; // Para IDistribuidorRepository, ICanillaRepository | ||||
| using GestionIntegral.Api.Dtos.Auditoria; | ||||
| using GestionIntegral.Api.Dtos.Contables; | ||||
| using GestionIntegral.Api.Models.Contables; | ||||
| using Microsoft.Extensions.Logging; | ||||
| @@ -52,7 +53,7 @@ namespace GestionIntegral.Api.Services.Contables | ||||
|                 var canData = await _canillaRepo.GetByIdAsync(saldo.IdDestino); | ||||
|                 nombreDestinatario = canData.Canilla?.NomApe ?? $"Can. ID {saldo.IdDestino}"; | ||||
|             } | ||||
|              | ||||
|  | ||||
|             var empresa = await _empresaRepo.GetByIdAsync(saldo.IdEmpresa); | ||||
|  | ||||
|             return new SaldoGestionDto | ||||
| @@ -94,11 +95,13 @@ namespace GestionIntegral.Api.Services.Contables | ||||
|             { | ||||
|                 if (await _canillaRepo.GetByIdSimpleAsync(ajusteDto.IdDestino) == null) | ||||
|                     return (false, "El canillita especificado no existe.", null); | ||||
|             } else { | ||||
|                  return (false, "Tipo de destino inválido.", null); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 return (false, "Tipo de destino inválido.", null); | ||||
|             } | ||||
|             if (await _empresaRepo.GetByIdAsync(ajusteDto.IdEmpresa) == null) | ||||
|                  return (false, "La empresa especificada no existe.", null); | ||||
|                 return (false, "La empresa especificada no existe.", null); | ||||
|  | ||||
|  | ||||
|             using var connection = _connectionFactory.CreateConnection(); | ||||
| @@ -116,7 +119,7 @@ namespace GestionIntegral.Api.Services.Contables | ||||
|                 } | ||||
|  | ||||
|                 decimal saldoAnterior = saldoActual.Monto; | ||||
|                  | ||||
|  | ||||
|                 bool modificado = await _saldoRepo.ModificarSaldoAsync(ajusteDto.Destino, ajusteDto.IdDestino, ajusteDto.IdEmpresa, ajusteDto.MontoAjuste, transaction); | ||||
|                 if (!modificado) | ||||
|                 { | ||||
| @@ -125,7 +128,7 @@ namespace GestionIntegral.Api.Services.Contables | ||||
|  | ||||
|                 // Obtener el saldo después de la modificación para el historial | ||||
|                 var saldoDespuesDeModificacion = await _saldoRepo.GetSaldoAsync(ajusteDto.Destino, ajusteDto.IdDestino, ajusteDto.IdEmpresa, transaction); | ||||
|                 if(saldoDespuesDeModificacion == null) throw new DataException("No se pudo obtener el saldo después de la modificación."); | ||||
|                 if (saldoDespuesDeModificacion == null) throw new DataException("No se pudo obtener el saldo después de la modificación."); | ||||
|  | ||||
|  | ||||
|                 var historial = new SaldoAjusteHistorial | ||||
| @@ -145,20 +148,44 @@ namespace GestionIntegral.Api.Services.Contables | ||||
|                 transaction.Commit(); | ||||
|                 _logger.LogInformation("Ajuste manual de saldo realizado para {Destino} ID {IdDestino}, Empresa ID {IdEmpresa} por Usuario ID {IdUsuarioAjuste}. Monto: {MontoAjuste}", | ||||
|                     ajusteDto.Destino, ajusteDto.IdDestino, ajusteDto.IdEmpresa, idUsuarioAjuste, ajusteDto.MontoAjuste); | ||||
|                  | ||||
|  | ||||
|                 var saldoDtoActualizado = await MapToGestionDto(saldoDespuesDeModificacion); | ||||
|                 return (true, null, saldoDtoActualizado); | ||||
|             } | ||||
|             catch (Exception ex) | ||||
|             { | ||||
|                 try { transaction.Rollback(); } catch (Exception rbEx){ _logger.LogError(rbEx, "Error en Rollback de RealizarAjusteManualSaldoAsync."); } | ||||
|                 try { transaction.Rollback(); } catch (Exception rbEx) { _logger.LogError(rbEx, "Error en Rollback de RealizarAjusteManualSaldoAsync."); } | ||||
|                 _logger.LogError(ex, "Error en RealizarAjusteManualSaldoAsync."); | ||||
|                 return (false, $"Error interno al realizar el ajuste: {ex.Message}", null); | ||||
|             } | ||||
|             finally | ||||
|             { | ||||
|                  if (connection.State == ConnectionState.Open) { if (connection is System.Data.Common.DbConnection dbConn) await dbConn.CloseAsync(); else connection.Close(); } | ||||
|                 if (connection.State == ConnectionState.Open) { if (connection is System.Data.Common.DbConnection dbConn) await dbConn.CloseAsync(); else connection.Close(); } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         public async Task<IEnumerable<SaldoAjusteHistorialDto>> ObtenerHistorialAjustesAsync( | ||||
|         DateTime? fechaDesde, DateTime? fechaHasta, | ||||
|         int? idUsuarioModifico, | ||||
|         string? destino, int? idDestino, int? idEmpresa) | ||||
|         { | ||||
|             var historialData = await _saldoRepo.GetHistorialAjustesAsync(fechaDesde, fechaHasta, idUsuarioModifico, destino, idDestino, idEmpresa); | ||||
|  | ||||
|             return historialData.Select(h => new SaldoAjusteHistorialDto | ||||
|             { | ||||
|                 IdSaldoAjusteHist = h.Historial.IdSaldoAjusteHist, | ||||
|                 Destino = h.Historial.Destino, | ||||
|                 Id_Destino = h.Historial.IdDestino, | ||||
|                 Id_Empresa = h.Historial.IdEmpresa, | ||||
|                 MontoAjuste = h.Historial.MontoAjuste, | ||||
|                 SaldoAnterior = h.Historial.SaldoAnterior, | ||||
|                 SaldoNuevo = h.Historial.SaldoNuevo, | ||||
|                 Justificacion = h.Historial.Justificacion, | ||||
|                 FechaAjuste = h.Historial.FechaAjuste, | ||||
|                 Id_UsuarioAjuste = h.Historial.IdUsuarioAjuste, | ||||
|                 NombreUsuarioModifico = h.NombreUsuarioModifico | ||||
|                 // TipoMod es implícito "AjusteManualSaldo" | ||||
|             }).ToList(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
		Reference in New Issue
	
	Block a user