Feat Widgets Cards y Optimización de Consultas
This commit is contained in:
		| @@ -10,6 +10,8 @@ using Microsoft.IdentityModel.Tokens; | ||||
| using Elecciones.Database.Entities; | ||||
| using System.Text.Json.Serialization; | ||||
| using Microsoft.AspNetCore.HttpOverrides; | ||||
| using Elecciones.Core.Enums; | ||||
| using Microsoft.OpenApi.Models; | ||||
|  | ||||
| // Esta es la estructura estándar y recomendada. | ||||
| var builder = WebApplication.CreateBuilder(args); | ||||
| @@ -81,8 +83,40 @@ builder.Services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme) | ||||
|     }); | ||||
|  | ||||
| builder.Services.AddEndpointsApiExplorer(); | ||||
| builder.Services.AddSwaggerGen(); | ||||
|  | ||||
| //builder.Services.AddSwaggerGen(); | ||||
|  | ||||
| builder.Services.AddSwaggerGen(options => | ||||
| { | ||||
|     // 1. Definir el esquema de seguridad que usaremos (Bearer Token) | ||||
|     options.AddSecurityDefinition("Bearer", new OpenApiSecurityScheme | ||||
|     { | ||||
|         Description = "Autorización JWT usando el esquema Bearer. Ingresa 'Bearer' [espacio] y luego tu token. Ejemplo: 'Bearer 12345abcdef'", | ||||
|         Name = "Authorization", // El nombre del header HTTP | ||||
|         In = ParameterLocation.Header, // Dónde se ubicará el token (en el header) | ||||
|         Type = SecuritySchemeType.ApiKey, // El tipo de esquema | ||||
|         Scheme = "Bearer" // El nombre del esquema | ||||
|     }); | ||||
|  | ||||
|     // 2. Aplicar este requisito de seguridad a todos los endpoints que lo necesiten | ||||
|     options.AddSecurityRequirement(new OpenApiSecurityRequirement() | ||||
|     { | ||||
|         { | ||||
|             new OpenApiSecurityScheme | ||||
|             { | ||||
|                 Reference = new OpenApiReference | ||||
|                 { | ||||
|                     Type = ReferenceType.SecurityScheme, | ||||
|                     Id = "Bearer" // Debe coincidir con el nombre que le dimos en AddSecurityDefinition | ||||
|                 }, | ||||
|                 Scheme = "oauth2", | ||||
|                 Name = "Bearer", | ||||
|                 In = ParameterLocation.Header, | ||||
|             }, | ||||
|             new List<string>() | ||||
|         } | ||||
|     }); | ||||
| }); | ||||
|  | ||||
| builder.Services.Configure<ForwardedHeadersOptions>(options => | ||||
| { | ||||
| @@ -153,7 +187,23 @@ using (var scope = app.Services.CreateScope()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| // Seeder para las bancas vacías | ||||
| // --- SEEDER DE ELECCIONES (Añadir para asegurar que existan) --- | ||||
| using (var scope = app.Services.CreateScope()) | ||||
| { | ||||
|     var context = scope.ServiceProvider.GetRequiredService<EleccionesDbContext>(); | ||||
|     if (!context.Elecciones.Any()) | ||||
|     { | ||||
|         context.Elecciones.AddRange( | ||||
|             new Eleccion { Id = 1, Nombre = "Elecciones Provinciales 2025", Nivel = "Provincial", DistritoId = "02", Fecha = new DateOnly(2025, 10, 26) }, | ||||
|             new Eleccion { Id = 2, Nombre = "Elecciones Nacionales 2025", Nivel = "Nacional", DistritoId = "00", Fecha = new DateOnly(2025, 10, 26) } | ||||
|         ); | ||||
|         context.SaveChanges(); | ||||
|         Console.WriteLine("--> Seeded Eleccion entities."); | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| // --- SEEDER DE BANCAS (MODIFICADO Y COMPLETADO) --- | ||||
| using (var scope = app.Services.CreateScope()) | ||||
| { | ||||
|     var services = scope.ServiceProvider; | ||||
| @@ -161,27 +211,91 @@ using (var scope = app.Services.CreateScope()) | ||||
|     if (!context.Bancadas.Any()) | ||||
|     { | ||||
|         var bancas = new List<Bancada>(); | ||||
|         // 92 bancas de diputados | ||||
|         for (int i = 1; i <= 92; i++) // Bucle de 1 a 92 | ||||
|  | ||||
|         // --- BANCAS PROVINCIALES (EleccionId = 1) --- | ||||
|         // 92 bancas de diputados provinciales | ||||
|         for (int i = 1; i <= 92; i++) | ||||
|         { | ||||
|             bancas.Add(new Bancada | ||||
|             { | ||||
|                 EleccionId = 1, | ||||
|                 Camara = Elecciones.Core.Enums.TipoCamara.Diputados, | ||||
|                 NumeroBanca = i // Asignamos el número de banca | ||||
|                 NumeroBanca = i | ||||
|             }); | ||||
|         } | ||||
|         // 46 bancas de senadores | ||||
|         for (int i = 1; i <= 46; i++) // Bucle de 1 a 46 | ||||
|         // 46 bancas de senadores provinciales | ||||
|         for (int i = 1; i <= 46; i++) | ||||
|         { | ||||
|             bancas.Add(new Bancada | ||||
|             { | ||||
|                 EleccionId = 1, | ||||
|                 Camara = Elecciones.Core.Enums.TipoCamara.Senadores, | ||||
|                 NumeroBanca = i // Asignamos el número de banca | ||||
|                 NumeroBanca = i | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         // --- BANCAS NACIONALES (EleccionId = 2) --- | ||||
|         // 257 bancas de diputados nacionales | ||||
|         for (int i = 1; i <= 257; i++) | ||||
|         { | ||||
|             bancas.Add(new Bancada | ||||
|             { | ||||
|                 EleccionId = 2, | ||||
|                 Camara = TipoCamara.Diputados, | ||||
|                 NumeroBanca = i | ||||
|             }); | ||||
|         } | ||||
|         // 72 bancas de senadores nacionales | ||||
|         for (int i = 1; i <= 72; i++) | ||||
|         { | ||||
|             bancas.Add(new Bancada | ||||
|             { | ||||
|                 EleccionId = 2, | ||||
|                 Camara = TipoCamara.Senadores, | ||||
|                 NumeroBanca = i | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         context.Bancadas.AddRange(bancas); | ||||
|         context.SaveChanges(); | ||||
|         Console.WriteLine("--> Seeded 138 bancas físicas."); | ||||
|         Console.WriteLine($"--> Seeded {bancas.Count} bancas físicas para ambas elecciones."); | ||||
|     } | ||||
| } | ||||
|  | ||||
| // --- Seeder para Proyecciones de Bancas (Elección Nacional) --- | ||||
| using (var scope = app.Services.CreateScope()) | ||||
| { | ||||
|     var services = scope.ServiceProvider; | ||||
|     var context = services.GetRequiredService<EleccionesDbContext>(); | ||||
|  | ||||
|     const int eleccionNacionalId = 2; | ||||
|     // Categoría 2: Diputados Nacionales, Categoría 1: Senadores Nacionales | ||||
|     if (!context.ProyeccionesBancas.Any(p => p.EleccionId == eleccionNacionalId)) | ||||
|     { | ||||
|         var partidos = await context.AgrupacionesPoliticas.Take(5).ToListAsync(); | ||||
|         var provincia = await context.AmbitosGeograficos.FirstOrDefaultAsync(a => a.NivelId == 10); // Asumimos un ámbito provincial genérico para la proyección total | ||||
|  | ||||
|         if (partidos.Count >= 5 && provincia != null) | ||||
|         { | ||||
|             var proyecciones = new List<ProyeccionBanca> | ||||
|             { | ||||
|                 // -- DIPUTADOS (Se renuevan 127) -- | ||||
|                 new() { EleccionId = eleccionNacionalId, CategoriaId = 2, AmbitoGeograficoId = provincia.Id, AgrupacionPoliticaId = partidos[0].Id, NroBancas = 50, FechaTotalizacion = DateTime.UtcNow }, | ||||
|                 new() { EleccionId = eleccionNacionalId, CategoriaId = 2, AmbitoGeograficoId = provincia.Id, AgrupacionPoliticaId = partidos[1].Id, NroBancas = 40, FechaTotalizacion = DateTime.UtcNow }, | ||||
|                 new() { EleccionId = eleccionNacionalId, CategoriaId = 2, AmbitoGeograficoId = provincia.Id, AgrupacionPoliticaId = partidos[2].Id, NroBancas = 20, FechaTotalizacion = DateTime.UtcNow }, | ||||
|                 new() { EleccionId = eleccionNacionalId, CategoriaId = 2, AmbitoGeograficoId = provincia.Id, AgrupacionPoliticaId = partidos[3].Id, NroBancas = 10, FechaTotalizacion = DateTime.UtcNow }, | ||||
|                 new() { EleccionId = eleccionNacionalId, CategoriaId = 2, AmbitoGeograficoId = provincia.Id, AgrupacionPoliticaId = partidos[4].Id, NroBancas = 7, FechaTotalizacion = DateTime.UtcNow }, | ||||
|  | ||||
|                 // -- SENADORES (Se renuevan 24) -- | ||||
|                 new() { EleccionId = eleccionNacionalId, CategoriaId = 1, AmbitoGeograficoId = provincia.Id, AgrupacionPoliticaId = partidos[0].Id, NroBancas = 10, FechaTotalizacion = DateTime.UtcNow }, | ||||
|                 new() { EleccionId = eleccionNacionalId, CategoriaId = 1, AmbitoGeograficoId = provincia.Id, AgrupacionPoliticaId = partidos[1].Id, NroBancas = 8, FechaTotalizacion = DateTime.UtcNow }, | ||||
|                 new() { EleccionId = eleccionNacionalId, CategoriaId = 1, AmbitoGeograficoId = provincia.Id, AgrupacionPoliticaId = partidos[2].Id, NroBancas = 4, FechaTotalizacion = DateTime.UtcNow }, | ||||
|                 new() { EleccionId = eleccionNacionalId, CategoriaId = 1, AmbitoGeograficoId = provincia.Id, AgrupacionPoliticaId = partidos[3].Id, NroBancas = 2, FechaTotalizacion = DateTime.UtcNow }, | ||||
|             }; | ||||
|             await context.ProyeccionesBancas.AddRangeAsync(proyecciones); | ||||
|             await context.SaveChangesAsync(); | ||||
|             Console.WriteLine("--> Seeded Proyecciones de Bancas para la Elección Nacional."); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -200,7 +314,10 @@ using (var scope = app.Services.CreateScope()) | ||||
|         { "Worker_Resultados_Activado", "false" }, | ||||
|         { "Worker_Bajas_Activado", "false" }, | ||||
|         { "Worker_Prioridad", "Resultados" }, | ||||
|         { "Logging_Level", "Information" } | ||||
|         { "Logging_Level", "Information" }, | ||||
|         { "PresidenciaDiputadosNacional", "" }, | ||||
|         { "PresidenciaDiputadosNacional_TipoBanca", "ganada" }, | ||||
|         { "PresidenciaSenadoNacional_TipoBanca", "ganada" } | ||||
|     }; | ||||
|  | ||||
|     foreach (var config in defaultConfiguraciones) | ||||
| @@ -230,7 +347,7 @@ using (var scope = app.Services.CreateScope()) | ||||
|  | ||||
|         var eleccionNacional = await context.Elecciones.FindAsync(eleccionNacionalId) ?? new Eleccion { Id = eleccionNacionalId, Nombre = "Elecciones Nacionales 2025", Nivel = "Nacional", DistritoId = "00", Fecha = new DateOnly(2025, 10, 26) }; | ||||
|         if (!context.Elecciones.Local.Any(e => e.Id == eleccionNacionalId)) context.Elecciones.Add(eleccionNacional); | ||||
|          | ||||
|  | ||||
|         var categoriaDiputadosNac = await context.CategoriasElectorales.FindAsync(2) ?? new CategoriaElectoral { Id = 2, Nombre = "DIPUTADOS NACIONALES", Orden = 3 }; | ||||
|         if (!context.CategoriasElectorales.Local.Any(c => c.Id == 2)) context.CategoriasElectorales.Add(categoriaDiputadosNac); | ||||
|         await context.SaveChangesAsync(); | ||||
| @@ -270,7 +387,8 @@ using (var scope = app.Services.CreateScope()) | ||||
|         await context.SaveChangesAsync(); | ||||
|  | ||||
|         var todosLosPartidos = await context.AgrupacionesPoliticas.Take(5).ToListAsync(); | ||||
|         if (!todosLosPartidos.Any()) { | ||||
|         if (!todosLosPartidos.Any()) | ||||
|         { | ||||
|             logger.LogWarning("--> No hay partidos, no se pueden generar votos."); | ||||
|             return; | ||||
|         } | ||||
| @@ -278,7 +396,7 @@ using (var scope = app.Services.CreateScope()) | ||||
|         var nuevosResultados = new List<ResultadoVoto>(); | ||||
|         var nuevosEstados = new List<EstadoRecuentoGeneral>(); | ||||
|         var rand = new Random(); | ||||
|          | ||||
|  | ||||
|         long totalVotosNacional = 0; | ||||
|         int totalMesasNacional = 0; | ||||
|         int totalMesasEscrutadasNacional = 0; | ||||
| @@ -287,9 +405,9 @@ using (var scope = app.Services.CreateScope()) | ||||
|         { | ||||
|             var municipiosDeProvincia = await context.AmbitosGeograficos.AsNoTracking().Where(a => a.NivelId == 30 && a.DistritoId == provincia.DistritoId).ToListAsync(); | ||||
|             if (!municipiosDeProvincia.Any()) continue; | ||||
|              | ||||
|  | ||||
|             long totalVotosProvincia = 0; | ||||
|              | ||||
|  | ||||
|             int partidoIndex = rand.Next(todosLosPartidos.Count); | ||||
|             foreach (var municipio in municipiosDeProvincia) | ||||
|             { | ||||
| @@ -299,7 +417,8 @@ using (var scope = app.Services.CreateScope()) | ||||
|                 totalVotosProvincia += votosGanador; | ||||
|  | ||||
|                 var otrosPartidos = todosLosPartidos.Where(p => p.Id != partidoGanador.Id).OrderBy(p => rand.Next()).Take(rand.Next(3, todosLosPartidos.Count)); | ||||
|                 foreach (var competidor in otrosPartidos) { | ||||
|                 foreach (var competidor in otrosPartidos) | ||||
|                 { | ||||
|                     var votosCompetidor = rand.Next(1000, 24000); | ||||
|                     nuevosResultados.Add(new ResultadoVoto { EleccionId = eleccionNacionalId, AmbitoGeograficoId = municipio.Id, CategoriaId = categoriaDiputadosNac.Id, AgrupacionPoliticaId = competidor.Id, CantidadVotos = votosCompetidor }); | ||||
|                     totalVotosProvincia += votosCompetidor; | ||||
| @@ -312,8 +431,11 @@ using (var scope = app.Services.CreateScope()) | ||||
|             var cantidadElectoresProvincia = mesasEsperadasProvincia * 350; | ||||
|             var participacionProvincia = (decimal)(rand.Next(65, 85) / 100.0); | ||||
|  | ||||
|             nuevosEstados.Add(new EstadoRecuentoGeneral { | ||||
|                 EleccionId = eleccionNacionalId, AmbitoGeograficoId = provincia.Id, CategoriaId = categoriaDiputadosNac.Id, | ||||
|             nuevosEstados.Add(new EstadoRecuentoGeneral | ||||
|             { | ||||
|                 EleccionId = eleccionNacionalId, | ||||
|                 AmbitoGeograficoId = provincia.Id, | ||||
|                 CategoriaId = categoriaDiputadosNac.Id, | ||||
|                 FechaTotalizacion = DateTime.UtcNow, | ||||
|                 MesasEsperadas = mesasEsperadasProvincia, | ||||
|                 MesasTotalizadas = mesasTotalizadasProvincia, | ||||
| @@ -322,7 +444,7 @@ using (var scope = app.Services.CreateScope()) | ||||
|                 CantidadVotantes = (int)(cantidadElectoresProvincia * participacionProvincia), | ||||
|                 ParticipacionPorcentaje = participacionProvincia * 100 | ||||
|             }); | ||||
|              | ||||
|  | ||||
|             totalVotosNacional += totalVotosProvincia; | ||||
|             totalMesasNacional += mesasEsperadasProvincia; | ||||
|             totalMesasEscrutadasNacional += mesasTotalizadasProvincia; | ||||
| @@ -330,14 +452,18 @@ using (var scope = app.Services.CreateScope()) | ||||
|  | ||||
|         // --- LÓGICA DE DATOS DE RECUENTO A NIVEL NACIONAL --- | ||||
|         var ambitoNacional = await context.AmbitosGeograficos.AsNoTracking().FirstOrDefaultAsync(a => a.NivelId == 0); | ||||
|         if (ambitoNacional == null) { | ||||
|         if (ambitoNacional == null) | ||||
|         { | ||||
|             ambitoNacional = new AmbitoGeografico { Nombre = "Nacional", NivelId = 0, DistritoId = "00" }; | ||||
|             context.AmbitosGeograficos.Add(ambitoNacional); | ||||
|             await context.SaveChangesAsync(); | ||||
|         } | ||||
|         var participacionNacional = (decimal)(rand.Next(70, 88) / 100.0); | ||||
|         nuevosEstados.Add(new EstadoRecuentoGeneral { | ||||
|             EleccionId = eleccionNacionalId, AmbitoGeograficoId = ambitoNacional.Id, CategoriaId = categoriaDiputadosNac.Id, | ||||
|         nuevosEstados.Add(new EstadoRecuentoGeneral | ||||
|         { | ||||
|             EleccionId = eleccionNacionalId, | ||||
|             AmbitoGeograficoId = ambitoNacional.Id, | ||||
|             CategoriaId = categoriaDiputadosNac.Id, | ||||
|             FechaTotalizacion = DateTime.UtcNow, | ||||
|             MesasEsperadas = totalMesasNacional, | ||||
|             MesasTotalizadas = totalMesasEscrutadasNacional, | ||||
| @@ -347,17 +473,56 @@ using (var scope = app.Services.CreateScope()) | ||||
|             ParticipacionPorcentaje = participacionNacional * 100 | ||||
|         }); | ||||
|  | ||||
|         if (nuevosResultados.Any()) { | ||||
|         if (nuevosResultados.Any()) | ||||
|         { | ||||
|             await context.ResultadosVotos.AddRangeAsync(nuevosResultados); | ||||
|             await context.EstadosRecuentosGenerales.AddRangeAsync(nuevosEstados); | ||||
|             await context.SaveChangesAsync(); | ||||
|             logger.LogInformation("--> Se generaron {Votos} registros de votos y {Estados} de estados de recuento.", nuevosResultados.Count, nuevosEstados.Count); | ||||
|         } else { | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             logger.LogWarning("--> No se generaron datos de simulación."); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| // --- Seeder para Bancas Previas (Composición Nacional 2025) --- | ||||
| using (var scope = app.Services.CreateScope()) | ||||
| { | ||||
|     var services = scope.ServiceProvider; | ||||
|     var context = services.GetRequiredService<EleccionesDbContext>(); | ||||
|  | ||||
|     const int eleccionNacionalId = 2; | ||||
|  | ||||
|     if (!context.BancasPrevias.Any(b => b.EleccionId == eleccionNacionalId)) | ||||
|     { | ||||
|         var partidos = await context.AgrupacionesPoliticas.Take(5).ToListAsync(); | ||||
|         if (partidos.Count >= 5) | ||||
|         { | ||||
|             var bancasPrevias = new List<BancaPrevia> | ||||
|             { | ||||
|                 // -- DIPUTADOS (Total: 257, se renuevan 127, quedan 130) -- | ||||
|                 new() { EleccionId = eleccionNacionalId, Camara = TipoCamara.Diputados, AgrupacionPoliticaId = partidos[0].Id, Cantidad = 40 }, | ||||
|                 new() { EleccionId = eleccionNacionalId, Camara = TipoCamara.Diputados, AgrupacionPoliticaId = partidos[1].Id, Cantidad = 35 }, | ||||
|                 new() { EleccionId = eleccionNacionalId, Camara = TipoCamara.Diputados, AgrupacionPoliticaId = partidos[2].Id, Cantidad = 30 }, | ||||
|                 new() { EleccionId = eleccionNacionalId, Camara = TipoCamara.Diputados, AgrupacionPoliticaId = partidos[3].Id, Cantidad = 15 }, | ||||
|                 new() { EleccionId = eleccionNacionalId, Camara = TipoCamara.Diputados, AgrupacionPoliticaId = partidos[4].Id, Cantidad = 10 }, | ||||
|  | ||||
|                 // -- SENADORES (Total: 72, se renuevan 24, quedan 48) -- | ||||
|                 new() { EleccionId = eleccionNacionalId, Camara = TipoCamara.Senadores, AgrupacionPoliticaId = partidos[0].Id, Cantidad = 18 }, | ||||
|                 new() { EleccionId = eleccionNacionalId, Camara = TipoCamara.Senadores, AgrupacionPoliticaId = partidos[1].Id, Cantidad = 15 }, | ||||
|                 new() { EleccionId = eleccionNacionalId, Camara = TipoCamara.Senadores, AgrupacionPoliticaId = partidos[2].Id, Cantidad = 8 }, | ||||
|                 new() { EleccionId = eleccionNacionalId, Camara = TipoCamara.Senadores, AgrupacionPoliticaId = partidos[3].Id, Cantidad = 4 }, | ||||
|                 new() { EleccionId = eleccionNacionalId, Camara = TipoCamara.Senadores, AgrupacionPoliticaId = partidos[4].Id, Cantidad = 3 }, | ||||
|             }; | ||||
|             await context.BancasPrevias.AddRangeAsync(bancasPrevias); | ||||
|             await context.SaveChangesAsync(); | ||||
|             Console.WriteLine("--> Seeded Bancas Previas para la Elección Nacional."); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| // Configurar el pipeline de peticiones HTTP. | ||||
| // Añadimos el logging de peticiones de Serilog aquí. | ||||
| app.UseSerilogRequestLogging(); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user