Files
Elecciones-2025/Elecciones-Web/src/Elecciones.Api/Program.cs

559 lines
25 KiB
C#
Raw Normal View History

//Elecciones.Api/Program.cs
using Elecciones.Database;
using Microsoft.EntityFrameworkCore;
2025-08-15 17:31:51 -03:00
using Serilog;
using Elecciones.Core.Services;
using Elecciones.Infrastructure.Services;
using System.Text;
using Microsoft.AspNetCore.Authentication.JwtBearer;
using Microsoft.IdentityModel.Tokens;
using Elecciones.Database.Entities;
using System.Text.Json.Serialization;
2025-09-03 17:54:49 -03:00
using Microsoft.AspNetCore.HttpOverrides;
using Elecciones.Core.Enums;
using Microsoft.OpenApi.Models;
2025-08-15 17:31:51 -03:00
// Esta es la estructura estándar y recomendada.
var builder = WebApplication.CreateBuilder(args);
// 1. Registra el servicio del interruptor como un Singleton.
// Esto asegura que toda la aplicación comparta la MISMA instancia del interruptor.
builder.Services.AddSingleton<LoggingSwitchService>();
builder.Host.UseSerilog((context, services, configuration) =>
{
// 2. Obtenemos la instancia del interruptor que acabamos de registrar.
var loggingSwitch = services.GetRequiredService<LoggingSwitchService>();
configuration
.ReadFrom.Configuration(context.Configuration)
.ReadFrom.Services(services)
.Enrich.FromLogContext()
// 3. Establecemos el nivel mínimo de logging controlado por el interruptor.
.MinimumLevel.ControlledBy(loggingSwitch.LevelSwitch)
.WriteTo.Console()
.WriteTo.File("logs/api-.log", rollingInterval: RollingInterval.Day); // o "logs/worker-.log"
});
2025-08-15 17:31:51 -03:00
// 2. Añadir servicios al contenedor.
var connectionString = builder.Configuration.GetConnectionString("DefaultConnection");
builder.Services.AddDbContext<EleccionesDbContext>(options =>
options.UseSqlServer(connectionString));
builder.Services.AddScoped<IPasswordHasher, PasswordHasher>();
builder.Services.AddControllers().AddJsonOptions(options =>
{
// Esto le dice al serializador que maneje las referencias circulares
options.JsonSerializerOptions.ReferenceHandler = ReferenceHandler.IgnoreCycles;
});
var MyAllowSpecificOrigins = "_myAllowSpecificOrigins";
builder.Services.AddCors(options =>
{
options.AddPolicy(name: MyAllowSpecificOrigins,
policy =>
{
policy.WithOrigins(
2025-09-03 18:56:01 -03:00
"http://localhost:5173",
2025-09-03 17:06:39 -03:00
"http://localhost:5174",
2025-09-03 18:56:01 -03:00
"http://192.168.5.128:8700",
2025-09-03 17:06:39 -03:00
"https://www.eldia.com",
2025-09-05 13:40:33 -03:00
"https://extras.eldia.com",
"https://eldia.mustang.cloud"
2025-08-30 11:31:45 -03:00
)
.AllowAnyHeader()
.AllowAnyMethod();
});
});
builder.Services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme)
.AddJwtBearer(options =>
{
options.TokenValidationParameters = new TokenValidationParameters
{
ValidateIssuer = true,
ValidateAudience = true,
ValidateLifetime = true,
ValidateIssuerSigningKey = true,
ValidIssuer = builder.Configuration["Jwt:Issuer"],
ValidAudience = builder.Configuration["Jwt:Audience"],
IssuerSigningKey = new SymmetricSecurityKey(Encoding.UTF8.GetBytes(builder.Configuration["Jwt:Key"]!))
};
});
builder.Services.AddEndpointsApiExplorer();
//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>()
}
});
});
2025-09-03 17:54:49 -03:00
builder.Services.Configure<ForwardedHeadersOptions>(options =>
{
options.ForwardedHeaders =
ForwardedHeaders.XForwardedFor | ForwardedHeaders.XForwardedProto;
options.KnownNetworks.Clear();
options.KnownProxies.Clear();
});
2025-08-15 17:31:51 -03:00
// 3. Construir la aplicación.
var app = builder.Build();
// --- LÓGICA PARA LEER EL NIVEL DE LOGGING AL INICIO ---
// Creamos un scope temporal para leer la configuración de la BD
using (var scope = app.Services.CreateScope()) // O 'host.Services.CreateScope()'
{
var services = scope.ServiceProvider;
try
{
// El resto de la lógica no cambia
var dbContext = services.GetRequiredService<EleccionesDbContext>();
var loggingSwitchService = services.GetRequiredService<LoggingSwitchService>();
var logLevelConfig = await dbContext.Configuraciones
.AsNoTracking()
.FirstOrDefaultAsync(c => c.Clave == "Logging_Level");
if (logLevelConfig != null)
{
loggingSwitchService.SetLoggingLevel(logLevelConfig.Valor);
Console.WriteLine($"--> Nivel de logging inicial establecido desde la BD a: {logLevelConfig.Valor}");
}
}
catch (Exception ex)
{
// Si hay un error (ej. la BD no está disponible al arrancar), se usará el nivel por defecto 'Information'.
Console.WriteLine($"--> No se pudo establecer el nivel de logging desde la BD: {ex.Message}");
}
}
2025-09-03 17:54:49 -03:00
app.UseForwardedHeaders();
// Seeder para el usuario admin
using (var scope = app.Services.CreateScope())
{
var services = scope.ServiceProvider;
try
{
var context = services.GetRequiredService<EleccionesDbContext>();
var hasher = services.GetRequiredService<IPasswordHasher>();
if (!context.AdminUsers.Any())
{
var (hash, salt) = hasher.HashPassword("PTP847elec");
context.AdminUsers.Add(new Elecciones.Database.Entities.AdminUser
{
Username = "admin",
PasswordHash = hash,
PasswordSalt = salt
});
context.SaveChanges();
Console.WriteLine("--> Admin user seeded.");
}
}
catch (Exception ex)
{
var logger = services.GetRequiredService<ILogger<Program>>();
logger.LogError(ex, "An error occurred while seeding the database.");
}
}
// --- 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;
var context = services.GetRequiredService<EleccionesDbContext>();
if (!context.Bancadas.Any())
{
var bancas = new List<Bancada>();
// --- BANCAS PROVINCIALES (EleccionId = 1) ---
// 92 bancas de diputados provinciales
for (int i = 1; i <= 92; i++)
{
2025-08-30 11:31:45 -03:00
bancas.Add(new Bancada
{
EleccionId = 1,
2025-08-30 11:31:45 -03:00
Camara = Elecciones.Core.Enums.TipoCamara.Diputados,
NumeroBanca = i
2025-08-30 11:31:45 -03:00
});
}
// 46 bancas de senadores provinciales
for (int i = 1; i <= 46; i++)
{
2025-08-30 11:31:45 -03:00
bancas.Add(new Bancada
{
EleccionId = 1,
2025-08-30 11:31:45 -03:00
Camara = Elecciones.Core.Enums.TipoCamara.Senadores,
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
2025-08-30 11:31:45 -03:00
});
}
// 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 {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.");
}
}
}
// Seeder para las configuraciones por defecto
using (var scope = app.Services.CreateScope())
{
var services = scope.ServiceProvider;
var context = services.GetRequiredService<EleccionesDbContext>();
// Lista de configuraciones por defecto a asegurar
var defaultConfiguraciones = new Dictionary<string, string>
{
{ "MostrarOcupantes", "true" },
{ "TickerResultadosCantidad", "3" },
{ "ConcejalesResultadosCantidad", "5" },
{ "Worker_Resultados_Activado", "false" },
{ "Worker_Bajas_Activado", "false" },
{ "Worker_Prioridad", "Resultados" },
{ "Logging_Level", "Information" },
{ "PresidenciaDiputadosNacional", "" },
{ "PresidenciaDiputadosNacional_TipoBanca", "ganada" },
{ "PresidenciaSenadoNacional_TipoBanca", "ganada" }
};
foreach (var config in defaultConfiguraciones)
2025-09-01 14:04:40 -03:00
{
if (!context.Configuraciones.Any(c => c.Clave == config.Key))
{
context.Configuraciones.Add(new Configuracion { Clave = config.Key, Valor = config.Value });
}
2025-09-01 14:04:40 -03:00
}
context.SaveChanges();
Console.WriteLine("--> Seeded default configurations.");
}
2025-09-22 17:56:04 -03:00
// --- SEEDER FINAL Y AUTOSUFICIENTE (CON DATOS DE RECUENTO) ---
using (var scope = app.Services.CreateScope())
{
var services = scope.ServiceProvider;
var context = services.GetRequiredService<EleccionesDbContext>();
var logger = services.GetRequiredService<ILogger<Program>>();
const int eleccionNacionalId = 2;
if (!context.ResultadosVotos.Any(r => r.EleccionId == eleccionNacionalId))
{
2025-09-22 17:56:04 -03:00
logger.LogInformation("--> No se encontraron datos para la elección nacional ID {EleccionId}. Generando datos de simulación COMPLETOS...", eleccionNacionalId);
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();
var provinciasMaestras = new Dictionary<string, string>
{
2025-09-19 17:19:10 -03:00
{ "01", "CIUDAD AUTONOMA DE BUENOS AIRES" }, { "02", "BUENOS AIRES" }, { "03", "CATAMARCA" }, { "04", "CORDOBA" },
{ "05", "CORRIENTES" },{ "06", "CHACO" }, { "07", "CHUBUT" }, { "08", "ENTRE RIOS" },
{ "09", "FORMOSA" }, { "10", "JUJUY" }, { "11", "LA PAMPA" }, { "12", "LA RIOJA" },
{ "13", "MENDOZA" }, { "14", "MISIONES" }, { "15", "NEUQUEN" }, { "16", "RIO NEGRO" },
{ "17", "SALTA" }, { "18", "SAN JUAN" }, { "19", "SAN LUIS" }, { "20", "SANTA CRUZ" },
{ "21", "SANTA FE" }, { "22", "SANTIAGO DEL ESTERO" }, { "23", "TIERRA DEL FUEGO" }, { "24", "TUCUMAN" }
};
foreach (var p in provinciasMaestras)
{
if (!await context.AmbitosGeograficos.AnyAsync(a => a.NivelId == 10 && a.DistritoId == p.Key))
{
context.AmbitosGeograficos.Add(new AmbitoGeografico { Nombre = p.Value, NivelId = 10, DistritoId = p.Key });
}
}
await context.SaveChangesAsync();
logger.LogInformation("--> Verificados/creados los 24 ámbitos geográficos de Nivel 10.");
var provinciasEnDb = await context.AmbitosGeograficos.AsNoTracking().Where(a => a.NivelId == 10).ToListAsync();
foreach (var provincia in provinciasEnDb)
{
2025-09-22 17:56:04 -03:00
if (!await context.AmbitosGeograficos.AnyAsync(a => a.NivelId == 30 && a.DistritoId == provincia.DistritoId))
{
2025-09-22 17:56:04 -03:00
logger.LogWarning("--> No se encontraron municipios para {Provincia}. Creando 5 de ejemplo.", provincia.Nombre);
for (int i = 1; i <= 5; i++)
{
2025-09-22 17:56:04 -03:00
context.AmbitosGeograficos.Add(new AmbitoGeografico { Nombre = $"{provincia.Nombre} - Depto. {i}", NivelId = 30, DistritoId = provincia.DistritoId });
}
}
}
await context.SaveChangesAsync();
var todosLosPartidos = await context.AgrupacionesPoliticas.Take(5).ToListAsync();
if (!todosLosPartidos.Any())
{
2025-09-22 17:56:04 -03:00
logger.LogWarning("--> No hay partidos, no se pueden generar votos.");
return;
}
var nuevosResultados = new List<ResultadoVoto>();
2025-09-22 17:56:04 -03:00
var nuevosEstados = new List<EstadoRecuentoGeneral>();
var rand = new Random();
2025-09-22 17:56:04 -03:00
long totalVotosNacional = 0;
int totalMesasNacional = 0;
int totalMesasEscrutadasNacional = 0;
foreach (var provincia in provinciasEnDb)
{
2025-09-22 17:56:04 -03:00
var municipiosDeProvincia = await context.AmbitosGeograficos.AsNoTracking().Where(a => a.NivelId == 30 && a.DistritoId == provincia.DistritoId).ToListAsync();
if (!municipiosDeProvincia.Any()) continue;
2025-09-22 17:56:04 -03:00
long totalVotosProvincia = 0;
int partidoIndex = rand.Next(todosLosPartidos.Count);
foreach (var municipio in municipiosDeProvincia)
{
2025-09-22 17:56:04 -03:00
var partidoGanador = todosLosPartidos[partidoIndex++ % todosLosPartidos.Count];
var votosGanador = rand.Next(25000, 70000);
nuevosResultados.Add(new ResultadoVoto { EleccionId = eleccionNacionalId, AmbitoGeograficoId = municipio.Id, CategoriaId = categoriaDiputadosNac.Id, AgrupacionPoliticaId = partidoGanador.Id, CantidadVotos = votosGanador });
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)
{
2025-09-22 17:56:04 -03:00
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;
}
}
2025-09-22 17:56:04 -03:00
// --- LÓGICA DE DATOS DE RECUENTO POR PROVINCIA ---
var mesasEsperadasProvincia = municipiosDeProvincia.Count * rand.Next(15, 30);
var mesasTotalizadasProvincia = (int)(mesasEsperadasProvincia * (rand.Next(75, 99) / 100.0));
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,
2025-09-22 17:56:04 -03:00
FechaTotalizacion = DateTime.UtcNow,
MesasEsperadas = mesasEsperadasProvincia,
MesasTotalizadas = mesasTotalizadasProvincia,
MesasTotalizadasPorcentaje = (decimal)mesasTotalizadasProvincia * 100 / mesasEsperadasProvincia,
CantidadElectores = cantidadElectoresProvincia,
CantidadVotantes = (int)(cantidadElectoresProvincia * participacionProvincia),
ParticipacionPorcentaje = participacionProvincia * 100
});
2025-09-22 17:56:04 -03:00
totalVotosNacional += totalVotosProvincia;
totalMesasNacional += mesasEsperadasProvincia;
totalMesasEscrutadasNacional += mesasTotalizadasProvincia;
}
2025-09-22 17:56:04 -03:00
// --- LÓGICA DE DATOS DE RECUENTO A NIVEL NACIONAL ---
var ambitoNacional = await context.AmbitosGeograficos.AsNoTracking().FirstOrDefaultAsync(a => a.NivelId == 0);
if (ambitoNacional == null)
{
2025-09-22 17:56:04 -03:00
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,
2025-09-22 17:56:04 -03:00
FechaTotalizacion = DateTime.UtcNow,
MesasEsperadas = totalMesasNacional,
MesasTotalizadas = totalMesasEscrutadasNacional,
MesasTotalizadasPorcentaje = (decimal)totalMesasEscrutadasNacional * 100 / totalMesasNacional,
CantidadElectores = totalMesasNacional * 350,
CantidadVotantes = (int)((totalMesasNacional * 350) * participacionNacional),
ParticipacionPorcentaje = participacionNacional * 100
});
if (nuevosResultados.Any())
{
await context.ResultadosVotos.AddRangeAsync(nuevosResultados);
2025-09-22 17:56:04 -03:00
await context.EstadosRecuentosGenerales.AddRangeAsync(nuevosEstados);
await context.SaveChangesAsync();
2025-09-22 17:56:04 -03:00
logger.LogInformation("--> Se generaron {Votos} registros de votos y {Estados} de estados de recuento.", nuevosResultados.Count, nuevosEstados.Count);
}
else
{
2025-09-22 17:56:04 -03:00
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.
2025-08-15 17:31:51 -03:00
// Añadimos el logging de peticiones de Serilog aquí.
app.UseSerilogRequestLogging();
if (app.Environment.IsDevelopment())
{
app.UseSwagger();
app.UseSwaggerUI();
}
// 1. Redirección a HTTPS (si se usa)
//app.UseHttpsRedirection();
// 2. Middleware de Enrutamiento. ¡CLAVE!
// Determina a qué endpoint irá la petición.
app.UseRouting();
// 3. Middleware de CORS.
// Ahora se ejecuta sabiendo a qué endpoint se dirige la petición.
app.UseCors(MyAllowSpecificOrigins);
// 4. Middleware de Autenticación.
// Identifica quién es el usuario a partir del token.
app.UseAuthentication();
// 5. Middleware de Autorización.
// Verifica si el usuario identificado tiene permiso para acceder al endpoint.
app.UseAuthorization();
// 6. Mapea los controladores a los endpoints que el enrutador descubrió.
app.MapControllers();
2025-08-15 17:31:51 -03:00
// 5. Ejecutar la aplicación.
app.Run();