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:
@@ -41,7 +41,7 @@ namespace GestionIntegral.Api.Data.Repositories.Contables
|
||||
if (idDestino.HasValue) { sqlBuilder.Append(" AND Id_Destino = @IdDestinoParam"); parameters.Add("IdDestinoParam", idDestino.Value); }
|
||||
if (idEmpresa.HasValue) { sqlBuilder.Append(" AND Id_Empresa = @IdEmpresaParam"); parameters.Add("IdEmpresaParam", idEmpresa.Value); }
|
||||
if (!string.IsNullOrWhiteSpace(tipoNota)) { sqlBuilder.Append(" AND Tipo = @TipoParam"); parameters.Add("TipoParam", tipoNota); }
|
||||
|
||||
|
||||
sqlBuilder.Append(" ORDER BY Fecha DESC, Id_Nota DESC;");
|
||||
|
||||
try
|
||||
@@ -86,11 +86,20 @@ namespace GestionIntegral.Api.Data.Repositories.Contables
|
||||
var inserted = await transaction.Connection!.QuerySingleAsync<NotaCreditoDebito>(sqlInsert, nuevaNota, transaction);
|
||||
if (inserted == null || inserted.IdNota == 0) throw new DataException("Error al crear la nota o ID no generado.");
|
||||
|
||||
await transaction.Connection!.ExecuteAsync(sqlHistorico, new {
|
||||
IdNotaHist = inserted.IdNota, DestinoHist = inserted.Destino, IdDestinoHist = inserted.IdDestino,
|
||||
ReferenciaHist = inserted.Referencia, TipoHist = inserted.Tipo, FechaHist = inserted.Fecha, MontoHist = inserted.Monto,
|
||||
ObservacionesHist = inserted.Observaciones, IdEmpresaHist = inserted.IdEmpresa,
|
||||
IdUsuarioHist = idUsuario, FechaModHist = DateTime.Now, TipoModHist = "Creada"
|
||||
await transaction.Connection!.ExecuteAsync(sqlHistorico, new
|
||||
{
|
||||
IdNotaHist = inserted.IdNota,
|
||||
DestinoHist = inserted.Destino,
|
||||
IdDestinoHist = inserted.IdDestino,
|
||||
ReferenciaHist = inserted.Referencia,
|
||||
TipoHist = inserted.Tipo,
|
||||
FechaHist = inserted.Fecha,
|
||||
MontoHist = inserted.Monto,
|
||||
ObservacionesHist = inserted.Observaciones,
|
||||
IdEmpresaHist = inserted.IdEmpresa,
|
||||
IdUsuarioHist = idUsuario,
|
||||
FechaModHist = DateTime.Now,
|
||||
TipoModHist = "Creada"
|
||||
}, transaction);
|
||||
return inserted;
|
||||
}
|
||||
@@ -107,23 +116,33 @@ namespace GestionIntegral.Api.Data.Repositories.Contables
|
||||
UPDATE dbo.cue_CreditosDebitos SET
|
||||
Monto = @Monto, Observaciones = @Observaciones
|
||||
WHERE Id_Nota = @IdNota;";
|
||||
const string sqlHistorico = @"
|
||||
const string sqlHistorico = @"
|
||||
INSERT INTO dbo.cue_CreditosDebitos_H
|
||||
(Id_Nota, Destino, Id_Destino, Referencia, Tipo, Fecha, Monto, Observaciones, Id_Empresa, Id_Usuario, FechaMod, TipoMod)
|
||||
VALUES (@IdNotaHist, @DestinoHist, @IdDestinoHist, @ReferenciaHist, @TipoHist, @FechaHist, @MontoHist, @ObservacionesHist, @IdEmpresaHist, @IdUsuarioHist, @FechaModHist, @TipoModHist);";
|
||||
|
||||
await transaction.Connection!.ExecuteAsync(sqlHistorico, new {
|
||||
IdNotaHist = actual.IdNota, DestinoHist = actual.Destino, IdDestinoHist = actual.IdDestino, ReferenciaHist = actual.Referencia,
|
||||
TipoHist = actual.Tipo, FechaHist = actual.Fecha, MontoHist = actual.Monto, // Valor ANTERIOR
|
||||
ObservacionesHist = actual.Observaciones, IdEmpresaHist = actual.IdEmpresa, // Valor ANTERIOR
|
||||
IdUsuarioHist = idUsuario, FechaModHist = DateTime.Now, TipoModHist = "Actualizada"
|
||||
|
||||
await transaction.Connection!.ExecuteAsync(sqlHistorico, new
|
||||
{
|
||||
IdNotaHist = actual.IdNota,
|
||||
DestinoHist = actual.Destino,
|
||||
IdDestinoHist = actual.IdDestino,
|
||||
ReferenciaHist = actual.Referencia,
|
||||
TipoHist = actual.Tipo,
|
||||
FechaHist = actual.Fecha,
|
||||
MontoHist = actual.Monto, // Valor ANTERIOR
|
||||
ObservacionesHist = actual.Observaciones,
|
||||
IdEmpresaHist = actual.IdEmpresa, // Valor ANTERIOR
|
||||
IdUsuarioHist = idUsuario,
|
||||
FechaModHist = DateTime.Now,
|
||||
TipoModHist = "Actualizada"
|
||||
}, transaction);
|
||||
|
||||
var rowsAffected = await transaction.Connection!.ExecuteAsync(sqlUpdate, new {
|
||||
var rowsAffected = await transaction.Connection!.ExecuteAsync(sqlUpdate, new
|
||||
{
|
||||
notaAActualizar.Monto,
|
||||
notaAActualizar.Observaciones,
|
||||
notaAActualizar.IdNota
|
||||
} , transaction);
|
||||
}, transaction);
|
||||
return rowsAffected == 1;
|
||||
}
|
||||
|
||||
@@ -140,14 +159,67 @@ namespace GestionIntegral.Api.Data.Repositories.Contables
|
||||
(Id_Nota, Destino, Id_Destino, Referencia, Tipo, Fecha, Monto, Observaciones, Id_Empresa, Id_Usuario, FechaMod, TipoMod)
|
||||
VALUES (@IdNotaHist, @DestinoHist, @IdDestinoHist, @ReferenciaHist, @TipoHist, @FechaHist, @MontoHist, @ObservacionesHist, @IdEmpresaHist, @IdUsuarioHist, @FechaModHist, @TipoModHist);";
|
||||
|
||||
await transaction.Connection!.ExecuteAsync(sqlHistorico, new {
|
||||
IdNotaHist = actual.IdNota, DestinoHist = actual.Destino, IdDestinoHist = actual.IdDestino, ReferenciaHist = actual.Referencia,
|
||||
TipoHist = actual.Tipo, FechaHist = actual.Fecha, MontoHist = actual.Monto, ObservacionesHist = actual.Observaciones, IdEmpresaHist = actual.IdEmpresa,
|
||||
IdUsuarioHist = idUsuario, FechaModHist = DateTime.Now, TipoModHist = "Eliminada"
|
||||
await transaction.Connection!.ExecuteAsync(sqlHistorico, new
|
||||
{
|
||||
IdNotaHist = actual.IdNota,
|
||||
DestinoHist = actual.Destino,
|
||||
IdDestinoHist = actual.IdDestino,
|
||||
ReferenciaHist = actual.Referencia,
|
||||
TipoHist = actual.Tipo,
|
||||
FechaHist = actual.Fecha,
|
||||
MontoHist = actual.Monto,
|
||||
ObservacionesHist = actual.Observaciones,
|
||||
IdEmpresaHist = actual.IdEmpresa,
|
||||
IdUsuarioHist = idUsuario,
|
||||
FechaModHist = DateTime.Now,
|
||||
TipoModHist = "Eliminada"
|
||||
}, transaction);
|
||||
|
||||
var rowsAffected = await transaction.Connection!.ExecuteAsync(sqlDelete, new { IdNotaParam = idNota }, transaction);
|
||||
return rowsAffected == 1;
|
||||
}
|
||||
|
||||
public async Task<IEnumerable<(NotaCreditoDebitoHistorico Historial, string NombreUsuarioModifico)>> GetHistorialAsync(
|
||||
DateTime? fechaDesde, DateTime? fechaHasta,
|
||||
int? idUsuarioModifico, string? tipoModificacion,
|
||||
int? idNotaOriginal)
|
||||
{
|
||||
using var connection = _cf.CreateConnection();
|
||||
var sqlBuilder = new StringBuilder(@"
|
||||
SELECT
|
||||
h.Id_Nota, h.Destino, h.Id_Destino, h.Referencia, h.Tipo, h.Fecha, h.Monto,
|
||||
h.Observaciones, h.Id_Empresa,
|
||||
h.Id_Usuario, h.FechaMod, h.TipoMod,
|
||||
u.Nombre + ' ' + u.Apellido AS NombreUsuarioModifico
|
||||
FROM dbo.cue_CreditosDebitos_H h
|
||||
JOIN dbo.gral_Usuarios u ON h.Id_Usuario = u.Id
|
||||
WHERE 1=1");
|
||||
|
||||
var parameters = new DynamicParameters();
|
||||
|
||||
if (fechaDesde.HasValue) { sqlBuilder.Append(" AND h.FechaMod >= @FechaDesdeParam"); parameters.Add("FechaDesdeParam", fechaDesde.Value.Date); }
|
||||
if (fechaHasta.HasValue) { sqlBuilder.Append(" AND h.FechaMod <= @FechaHastaParam"); parameters.Add("FechaHastaParam", fechaHasta.Value.Date.AddDays(1).AddTicks(-1)); }
|
||||
if (idUsuarioModifico.HasValue) { sqlBuilder.Append(" AND h.Id_Usuario = @IdUsuarioModificoParam"); parameters.Add("IdUsuarioModificoParam", idUsuarioModifico.Value); }
|
||||
if (!string.IsNullOrWhiteSpace(tipoModificacion)) { sqlBuilder.Append(" AND h.TipoMod = @TipoModParam"); parameters.Add("TipoModParam", tipoModificacion); }
|
||||
if (idNotaOriginal.HasValue) { sqlBuilder.Append(" AND h.Id_Nota = @IdNotaOriginalParam"); parameters.Add("IdNotaOriginalParam", idNotaOriginal.Value); }
|
||||
|
||||
sqlBuilder.Append(" ORDER BY h.FechaMod DESC;");
|
||||
|
||||
try
|
||||
{
|
||||
var result = await connection.QueryAsync<NotaCreditoDebitoHistorico, string, (NotaCreditoDebitoHistorico, string)>(
|
||||
sqlBuilder.ToString(),
|
||||
(hist, userName) => (hist, userName),
|
||||
parameters,
|
||||
splitOn: "NombreUsuarioModifico"
|
||||
);
|
||||
return result;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_log.LogError(ex, "Error al obtener historial de Notas C/D.");
|
||||
return Enumerable.Empty<(NotaCreditoDebitoHistorico, string)>();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user