using Microsoft.AspNetCore.Mvc; using Microsoft.OpenApi.Models; using Swashbuckle.AspNetCore.SwaggerGen; using SatelliteProvider.Api; using SatelliteProvider.Api.DTOs; using SatelliteProvider.Api.Swagger; using SatelliteProvider.DataAccess; using SatelliteProvider.DataAccess.Repositories; using SatelliteProvider.Common.Configs; using SatelliteProvider.Common.DTO; using SatelliteProvider.Common.Interfaces; using SatelliteProvider.Services.RegionProcessing; using SatelliteProvider.Services.RouteManagement; using SatelliteProvider.Services.TileDownloader; using Serilog; var builder = WebApplication.CreateBuilder(args); builder.Host.UseSerilog((context, configuration) => configuration.ReadFrom.Configuration(context.Configuration)); var connectionString = builder.Configuration.GetConnectionString("DefaultConnection") ?? throw new InvalidOperationException("Connection string 'DefaultConnection' not found."); builder.Services.Configure(builder.Configuration.GetSection("MapConfig")); builder.Services.Configure(builder.Configuration.GetSection("StorageConfig")); builder.Services.Configure(builder.Configuration.GetSection("ProcessingConfig")); builder.Services.AddSingleton(sp => new TileRepository(connectionString, sp.GetRequiredService>())); builder.Services.AddSingleton(sp => new RegionRepository(connectionString, sp.GetRequiredService>())); builder.Services.AddSingleton(sp => new RouteRepository(connectionString, sp.GetRequiredService>())); builder.Services.AddHttpClient(); builder.Services.AddTileDownloader(); builder.Services.AddRegionProcessing(); builder.Services.AddRouteManagement(); var allowedOrigins = builder.Configuration.GetSection("CorsConfig:AllowedOrigins").Get() ?? Array.Empty(); var allowAnyOrigin = builder.Configuration.GetValue("CorsConfig:AllowAnyOrigin"); CorsConfigurationValidator.EnsureSafeForEnvironment(allowedOrigins, allowAnyOrigin, builder.Environment.EnvironmentName); builder.Services.AddCors(options => { options.AddPolicy("TilesCors", policy => { if (CorsConfigurationValidator.ShouldUsePermissivePolicy(allowedOrigins, allowAnyOrigin)) policy.AllowAnyOrigin().AllowAnyHeader().AllowAnyMethod(); else policy.WithOrigins(allowedOrigins).AllowAnyHeader().AllowAnyMethod(); }); }); builder.Services.AddProblemDetails(); builder.Services.AddExceptionHandler(); builder.Services.ConfigureHttpJsonOptions(options => { options.SerializerOptions.PropertyNamingPolicy = System.Text.Json.JsonNamingPolicy.CamelCase; options.SerializerOptions.PropertyNameCaseInsensitive = true; }); builder.Services.AddEndpointsApiExplorer(); builder.Services.AddSwaggerGen(c => { c.SwaggerDoc("v1", new OpenApiInfo { Title = "Satellite Provider API", Version = "v1" }); c.MapType(() => new OpenApiSchema { Type = "object", Properties = new Dictionary { ["timestamp"] = new() { Type = "string", Format = "date-time", Description = "Image capture timestamp" }, ["image"] = new() { Type = "string", Format = "binary", Description = "Image file to upload" }, ["lat"] = new() { Type = "number", Format = "double", Description = "Latitude coordinate where image was captured" }, ["lon"] = new() { Type = "number", Format = "double", Description = "Longitude coordinate where image was captured" }, ["height"] = new() { Type = "number", Format = "double", Description = "Height/altitude in meters where image was captured" }, ["focalLength"] = new() { Type = "number", Format = "double", Description = "Camera focal length in millimeters" }, ["sensorWidth"] = new() { Type = "number", Format = "double", Description = "Camera sensor width in millimeters" }, ["sensorHeight"] = new() { Type = "number", Format = "double", Description = "Camera sensor height in millimeters" } }, Required = new HashSet { "timestamp", "image", "lat", "lon", "height", "focalLength", "sensorWidth", "sensorHeight" } }); c.OperationFilter(); }); var app = builder.Build(); if (CorsConfigurationValidator.ShouldWarnAboutPermissiveDefault(allowedOrigins, allowAnyOrigin)) { app.Services .GetRequiredService>() .LogWarning(CorsConfigurationValidator.PermissiveDefaultWarning, app.Environment.EnvironmentName); } var migratorLogger = app.Services.GetRequiredService>(); var migrator = new DatabaseMigrator(connectionString, migratorLogger); if (!migrator.RunMigrations()) { throw new Exception("Database migration failed. Application cannot start."); } var storageConfig = app.Configuration.GetSection("StorageConfig").Get() ?? new StorageConfig(); Directory.CreateDirectory(storageConfig.TilesDirectory); Directory.CreateDirectory(storageConfig.ReadyDirectory); if (app.Environment.IsDevelopment()) { app.UseSwagger(); app.UseSwaggerUI(); } app.UseExceptionHandler(); app.UseHttpsRedirection(); app.UseCors("TilesCors"); app.MapGet("/tiles/{z:int}/{x:int}/{y:int}", ServeTile) .WithOpenApi(op => new(op) { Summary = "Get satellite tile image by z/x/y coordinates (Slippy Map tile server)" }); app.MapGet("/api/satellite/tiles/latlon", GetTileByLatLon) .WithOpenApi(op => new(op) { Summary = "Get satellite tile by latitude and longitude coordinates" }); app.MapGet("/api/satellite/tiles/mgrs", GetSatelliteTilesByMgrs) .ProducesProblem(StatusCodes.Status501NotImplemented) .WithOpenApi(op => new(op) { Summary = "Get satellite tiles by MGRS coordinates (NOT IMPLEMENTED)" }); app.MapPost("/api/satellite/upload", UploadImage) .Accepts("multipart/form-data") .ProducesProblem(StatusCodes.Status501NotImplemented) .WithOpenApi(op => new(op) { Summary = "Upload image with metadata (NOT IMPLEMENTED)" }) .DisableAntiforgery(); app.MapPost("/api/satellite/request", RequestRegion) .WithOpenApi(op => new(op) { Summary = "Request tiles for a region", Description = "Idempotent (AZ-362): POSTing the same `id` twice returns the existing region resource with HTTP 200 and does not enqueue duplicate background processing.", }); app.MapGet("/api/satellite/region/{id:guid}", GetRegionStatus) .WithOpenApi(op => new(op) { Summary = "Get region status and file paths" }); app.MapPost("/api/satellite/route", CreateRoute) .WithOpenApi(op => new(op) { Summary = "Create a route with intermediate points", Description = "Idempotent (AZ-362): POSTing the same `id` twice returns the existing route resource with HTTP 200 and does not regenerate intermediate points or re-queue geofence regions.", }); app.MapGet("/api/satellite/route/{id:guid}", GetRoute) .WithOpenApi(op => new(op) { Summary = "Get route information with calculated points" }); app.Run(); async Task ServeTile(int z, int x, int y, HttpContext httpContext, ITileService tileService) { var tile = await tileService.GetOrDownloadTileAsync(z, x, y, httpContext.RequestAborted); httpContext.Response.Headers.CacheControl = $"public, max-age={(long)tile.MaxAge.TotalSeconds}"; httpContext.Response.Headers.ETag = tile.ETag; return Results.Bytes(tile.Bytes, tile.ContentType); } async Task GetTileByLatLon([FromQuery] double Latitude, [FromQuery] double Longitude, [FromQuery] int ZoomLevel, ITileService tileService) { var tile = await tileService.DownloadAndStoreSingleTileAsync(Latitude, Longitude, ZoomLevel); var response = new DownloadTileResponse { Id = tile.Id, ZoomLevel = tile.TileZoom, Latitude = tile.Latitude, Longitude = tile.Longitude, TileSizeMeters = tile.TileSizeMeters, TileSizePixels = tile.TileSizePixels, ImageType = tile.ImageType, MapsVersion = tile.MapsVersion, Version = tile.Version, FilePath = tile.FilePath, CreatedAt = tile.CreatedAt, UpdatedAt = tile.UpdatedAt }; return Results.Ok(response); } IResult GetSatelliteTilesByMgrs(string mgrs, double squareSideMeters) { return Results.Problem( statusCode: StatusCodes.Status501NotImplemented, title: "Not implemented", detail: "MGRS-based tile retrieval is not implemented."); } IResult UploadImage([FromForm] UploadImageRequest request) { return Results.Problem( statusCode: StatusCodes.Status501NotImplemented, title: "Not implemented", detail: "Image upload is not implemented."); } async Task RequestRegion([FromBody] RequestRegionRequest request, IRegionService regionService) { if (request.SizeMeters < 100 || request.SizeMeters > 10000) { return Results.BadRequest(new { error = "Size must be between 100 and 10000 meters" }); } var status = await regionService.RequestRegionAsync( request.Id, request.Latitude, request.Longitude, request.SizeMeters, request.ZoomLevel, request.StitchTiles); return Results.Ok(status); } async Task GetRegionStatus(Guid id, IRegionService regionService) { var status = await regionService.GetRegionStatusAsync(id); if (status == null) { return Results.NotFound(new { error = $"Region {id} not found" }); } return Results.Ok(status); } async Task CreateRoute([FromBody] CreateRouteRequest request, IRouteService routeService, ILogger logger) { try { var route = await routeService.CreateRouteAsync(request); return Results.Ok(route); } catch (ArgumentException ex) { logger.LogWarning(ex, "Invalid route request"); return Results.BadRequest(new { error = ex.Message }); } } async Task GetRoute(Guid id, IRouteService routeService) { var route = await routeService.GetRouteAsync(id); if (route == null) { return Results.NotFound(new { error = $"Route {id} not found" }); } return Results.Ok(route); }