Kestrel with HttpProtocols.Http1AndHttp2 on a plaintext listener silently downgrades to HTTP/1.1-only (logs "HTTP/2 is not enabled ... TLS is not enabled"), so AC-5's multiplexed-GET test failed with HTTP_1_1_REQUIRED. ALPN cannot run over plaintext, so the fix switches the dev listener to TLS on https://+:8080: - scripts/run-tests.sh generates a self-signed dev cert idempotently (./certs/api.pfx + api.crt) via openssl in an alpine container; certs/ is gitignored. - docker-compose.yml binds Kestrel to ASPNETCORE_URLS=https://+:8080 with Kestrel__Certificates__Default__Path bound to the .pfx. - docker-compose.tests.yml mounts api.crt into the integration-tests container's CA store and runs update-ca-certificates so HttpClient trusts the cert transparently; default API_URL is now https://api:8080. - Drop the obsolete Http2UnencryptedSupport AppContext switch from Http2MultiplexingTests; ALPN over TLS handles negotiation. Test-data fixes caught on the post-TLS rerun (independent of the TLS switch but surfaced together): - Http2MultiplexingTests: switch slippy coords from (154321, 95812) -- which Google Maps returns 404 for -- to (158485, 91707), the slippy projection of (47.461747, 37.647063) already exercised by JwtIntegrationTests. - TileInventoryTests + LeafletPathIndexOnlyTests: SpecifyKind to Unspecified at the binding site for raw Npgsql seed paths writing into tiles.captured_at / created_at / updated_at (TIMESTAMP without tz). Npgsql v6+ refuses Kind=Utc into plain timestamp columns; production goes through Dapper and never hits this code path. - MigrationTests Az503NewUniqueIndexCoversIntegerKeyAndFlightId: accept either idx_tiles_location_hash (migration 014) or its AZ-505 successor tiles_leaflet_path (migration 015) -- both have location_hash as the leading column, which is the AC-9 intent. Docs updated to reflect the TLS+ALPN path: tile-inventory.md Non-Goals, modules/api_program.md, module-layout.md, the AZ-505 task spec's Risk 3, and the cycle 6 implementation + completeness reports. The full integration test suite passes (mode=full, exit 0). Co-authored-by: Cursor <cursoragent@cursor.com>
12 KiB
Module: Api/Program.cs
Purpose
Application entry point. Configures DI container, sets up middleware, defines minimal API endpoints, runs database migrations on startup, and starts background services.
Public Interface
API Endpoints
| Method | Route | Handler | Description |
|---|---|---|---|
| GET | /tiles/{z}/{x}/{y} |
ServeTile |
Slippy map tile server with in-memory caching. AZ-505 rewired the DB lookup to filter on location_hash (deterministic UUIDv5) so the read becomes an Index Only Scan against tiles_leaflet_path; the wire response is byte-identical to pre-AZ-505. |
| GET | /api/satellite/tiles/latlon |
GetTileByLatLon |
Download single tile by lat/lon/zoom |
| POST | /api/satellite/tiles/inventory |
GetTilesInventory |
Bulk tile-existence/metadata lookup (AZ-505) — body is XOR of tiles[{tileZoom,tileX,tileY}] (Form A) and locationHashes[uuid] (Form B), each capped at 5000 entries. Response is one entry per request entry, in input order. Contract: _docs/02_document/contracts/api/tile-inventory.md v1.0.0. |
| GET | /api/satellite/tiles/mgrs |
GetSatelliteTilesByMgrs |
MGRS stub (returns empty) |
| POST | /api/satellite/upload |
UploadUavTileBatch |
UAV tile batch upload (AZ-488) — multipart envelope, 5-rule quality gate, per-source UPSERT with source='uav'. Requires the RequiresGpsPermission policy. |
| POST | /api/satellite/request |
RequestRegion |
Queue region for async tile processing |
| GET | /api/satellite/region/{id} |
GetRegionStatus |
Get region processing status |
| POST | /api/satellite/route |
CreateRoute |
Create route with intermediate points |
| GET | /api/satellite/route/{id} |
GetRoute |
Get route with all points |
Local Records (defined in Program.cs)
GetSatelliteTilesResponse,SatelliteTile— MGRS response stubsDownloadTileResponse— tile download responseRequestRegionRequest— region request bodyParameterDescriptionFilter— Swagger operation filter
Api/DTOs (AZ-488)
UavTileBatchUploadRequest— multipart envelope withmetadata(JSON string) andfiles(IFormFileCollection)
Common/DTO (AZ-488)
UavTileMetadata,UavTileBatchMetadataPayload— per-item metadata + envelope shapeUavTileBatchUploadResponse,UavTileUploadResultItem— per-item response shapeUavTileUploadStatus,UavTileRejectReasons— string-constant enumerations exposed in the v1.0.0 contract
Common/DTO (AZ-505)
TileInventoryRequest— XOR body envelope withTiles(Form A) ORLocationHashes(Form B)TileCoord—{TileZoom, TileX, TileY}per-entry coord under Form ATileInventoryResponse—{Results: TileInventoryEntry[]}response shape; ordering matches requestTileInventoryEntry— per-entry response shape (Present,LocationHash, optionalId/CapturedAt/Source/FlightId/ResolutionMPerPx)TileInventoryLimits.MaxEntriesPerRequest— hard cap (5000) consumed by request validation
Internal Logic
DI Registration
- Serilog configured from
appsettings.json - Connection string extracted from
ConnectionStrings:DefaultConnection - Config bindings:
MapConfig,StorageConfig,ProcessingConfig,UavQualityConfig(AZ-488) - Request size limits (AZ-488):
KestrelServerOptions.Limits.MaxRequestBodySizeandFormOptions.MultipartBodyLengthLimitare set toUavQualityConfig.MaxBatchSize × UavQualityConfig.MaxBytes(default 100 × 5 MiB = 500 MiB) so an oversized UAV batch is rejected at the framework layer before reaching the handler. - Singletons: repositories (
TileRepository,RegionRepository,RouteRepository),GoogleMapsDownloaderV2,ITileService,IRegionService,IRouteService,IUavTileQualityGate,IUavTileUploadHandler(AZ-488) IRegionRequestQueuewith configurable capacity- Hosted services:
RegionProcessingService,RouteProcessingService - CORS policy:
TilesCors— configured origins fromCorsConfig:AllowedOrigins, falls back to allow-any - JSON options: camelCase, case-insensitive
- JWT authentication (AZ-487 + AZ-494):
AddSatelliteJwt(builder.Configuration)(extension inSatelliteProvider.Api.Authentication) registersJwtBearerwithTokenValidationParametersset per the suite auth contract: signature + lifetime + issuer + audience validation, 30 s clock skew, ≥ 32-byte HMAC key. Theissvalue comes fromJWT_ISSUERenv (fallbackJwt:Issuerconfig); theaudvalue comes fromJWT_AUDIENCEenv (fallbackJwt:Audienceconfig). All three values (secret, iss, aud) are fail-fast — the API throwsInvalidOperationExceptionat startup if any is unset or whitespace-only. Production deploys MUST set the env vars with admin-team-confirmed values;appsettings.jsonships empty so the fail-fast triggers.appsettings.Development.jsonships clearly-tagged DEV-ONLY values (DEV-ONLY-iss-admin-azaion-local/DEV-ONLY-aud-satellite-provider) so local dev works out-of-the-box. Followed byAddAuthorizationwith theRequiresGpsPermissionpolicy (AZ-488). - Kestrel HTTP/2 (AZ-505):
builder.WebHost.ConfigureKestrel(opts => opts.ConfigureEndpointDefaults(lo => lo.Protocols = HttpProtocols.Http1AndHttp2)). The dev listener is nowhttps://+:8080with a self-signed cert (./certs/api.pfx, generated idempotently byscripts/run-tests.shand bound viaASPNETCORE_Kestrel__Certificates__Default__Path/__Passwordindocker-compose.yml). Kestrel needs TLS for HTTP/2 protocol negotiation; ALPN advertises bothh2andhttp/1.1so HTTP/2-capable clients (browser Leaflet,HttpClientwithVersion20+RequestVersionExact, httpxhttp2=True) multiplex tile reads on a single TLS connection, and legacy clients fall back to HTTP/1.1. The integration-test container trusts the dev cert via/usr/local/share/ca-certificates/+update-ca-certificates. AZ-505 AC-5 verifies the multiplex semantics here; production termination is expected at the ingress (Envoy / nginx / ALB) — Kestrel can then drop to HTTP/2 cleartext behind it without changing this code.
Startup
- Database migration via
DatabaseMigrator.RunMigrations()— throws on failure - Creates tiles and ready directories
- Swagger enabled in Development mode
- Middleware chain (order matters):
UseExceptionHandler→UseHttpsRedirection→UseCors("TilesCors")→UseAuthentication→UseAuthorization→ endpoint mapping. - Every
MapGet/MapPostendpoint is decorated with.RequireAuthorization(); the framework returns 401 before the handler runs for any anonymous, expired, or invalid-signature request.
ServeTile Handler
- Checks
IMemoryCachefor tile bytes (1h absolute, 30min sliding expiration) - If cache miss: queries
ITileRepository.GetByTileCoordinatesAsync— AZ-505 rewired this method to computelocation_hash = Uuidv5(TileNamespace, "{z}/{x}/{y}")and filter byWHERE location_hash = $1, hittingtiles_leaflet_pathas anIndex Only ScanwithHeap Fetches ≤ 1. Selection rule is unchanged (most-recent across sources/flights); wire response is byte-identical. - If no DB record: downloads tile via
GoogleMapsDownloaderV2.DownloadSingleTileAsync, createsTileEntity, inserts - Returns image bytes with cache headers (
Cache-Control: public, max-age=86400)
GetTilesInventory Handler (AZ-505)
- Validates XOR body shape: 400 if both
tilesandlocationHashesare populated, 400 if neither is populated, 400 if either exceedsTileInventoryLimits.MaxEntriesPerRequest(5000) - Delegates to
ITileService.GetInventoryAsync(request, ct) - Service computes
location_hashfor Form A entries viaUuidv5.Create(TileNamespace, "{z}/{x}/{y}"), callsITileRepository.GetTilesByLocationHashesAsync(IReadOnlyList<Guid>), re-aligns results back to input order - Returns
TileInventoryResponsewith one entry per input —present=trueentries carryid/capturedAt/source/flightId/resolutionMPerPx;present=falseentries carry onlylocationHash - Authenticated by
.RequireAuthorization()(401 before handler for anonymous)
GetTileByLatLon Handler
Downloads a tile, persists it, returns metadata as DownloadTileResponse.
RequestRegion Handler
Validates size (100–10000m), delegates to IRegionService.RequestRegionAsync.
UploadUavTileBatch Handler (AZ-488)
Buffers each IFormFile into memory, packages them as UavUploadFile records (filename, content-type, bytes), and delegates to IUavTileUploadHandler.HandleAsync. Envelope-level errors (mismatched batch, oversized batch, malformed metadata) are surfaced as HTTP 400 ProblemDetails; per-item rejects are returned in the HTTP 200 response payload. The endpoint is protected by .RequireAuthorization(SatellitePermissions.UavUploadPolicy) so 401 (no token) and 403 (no GPS permission) are returned before the handler runs.
Dependencies
All project references: Common, DataAccess, Services.
NuGet: Serilog.AspNetCore (8.0.3 — fallback retained on .NET 10 per AZ-500 Risk #4: no 10.x line published as of cycle 4; documented in AGENTS.md), Swashbuckle.AspNetCore (10.1.7 — bumped from 6.6.2 by AZ-500 to land Microsoft.OpenApi 2.x compat required by ASP.NET Core 10), Microsoft.AspNetCore.OpenApi (10.0.7 — bumped from 8.0.25 by AZ-500), Microsoft.AspNetCore.Authentication.JwtBearer (10.0.7 — added at 8.0.21 by AZ-487, bumped to 8.0.25 by AZ-496, bumped to 10.0.7 by AZ-500), SixLabors.ImageSharp, Newtonsoft.Json.
Microsoft.OpenApi 2.x refactor note (AZ-500): the major bump (1.x → 2.x) drove three internal Swashbuckle-setup edits in this file — using Microsoft.OpenApi.Models; → using Microsoft.OpenApi;; AddSecurityRequirement(...) rewritten to take a Func<OpenApiDocument, OpenApiSecurityRequirement> and use OpenApiSecuritySchemeReference("Bearer") instead of the removed OpenApiSecurityScheme.Reference shape; MapType<UavTileBatchUploadRequest> rewritten to use the new JsonSchemaType enum and IDictionary<string, IOpenApiSchema> properties bag. The Swagger document shape (paths, operations, the Bearer Authorize button, the multipart-batch upload schema) is preserved exactly — SwaggerDocument_AdvertisesBearerSecurityScheme and the AZ-353 swagger-ready integration assertions still pass. Eight ASPDEPR002 deprecation warnings (WithOpenApi(...)) remain — they're recorded in _docs/03_implementation/reviews/batch_01_cycle4_review.md as a follow-up PBI; the API is still fully functional in .NET 10 (deprecated, not removed).
Consumers
- HTTP clients (external)
- Integration tests (via HTTP)
Data Models
Defines several local request/response records that are not shared with other projects.
Configuration
All configuration sections are consumed here:
ConnectionStrings:DefaultConnectionMapConfig,StorageConfig,ProcessingConfigUavQuality(AZ-488) —MinBytes,MaxBytes,MaxAgeDays,CapturedAtFutureSkewSeconds,MinLuminanceVariance,MaxBatchSize,LuminanceSampleSize. Drives the 5-rule quality gate AND the per-request body-size limits.CorsConfig:AllowedOriginsJwt:Secret— HMAC-SHA256 signing key for JWT validation (AZ-487). Resolution:JWT_SECRETenv var (preferred, opaque production secret) →Jwt:Secretconfiguration key (appsettings.Development.jsonplaceholder only). Startup fails fast if the resolved value is unset, empty, or shorter than 32 bytes.Jwt:Issuer— Expectedissclaim value (AZ-494). Resolution:JWT_ISSUERenv →Jwt:Issuerconfig. Startup fails fast if unset/empty.Jwt:Audience— Expectedaudclaim value (AZ-494). Resolution:JWT_AUDIENCEenv →Jwt:Audienceconfig. Startup fails fast if unset/empty.Serilogsection
External Integrations
- Google Maps (indirectly via
GoogleMapsDownloaderV2) - PostgreSQL (via repositories and DatabaseMigrator)
- File system (
./tiles/,./ready/)
Security
- CORS configured (permissive by default when no origins specified)
- Swagger only in Development; Bearer token "Authorize" button registered via
AddSecurityDefinition/AddSecurityRequirement(AZ-487) - HTTPS redirection enabled
- JWT bearer authentication (AZ-487) — every endpoint requires a valid HS256-signed token. Anonymous, expired, or signature-tampered requests return 401 before the handler runs.
- Permission-claim policies (AZ-488) —
POST /api/satellite/uploadis wrapped in.RequireAuthorization(SatellitePermissions.UavUploadPolicy). ThePermissionsAuthorizationHandlerreads thepermissionsclaim (repeated-string OR JSON-array shape) and returns 403 ifGPSis not present.
Tests
Integration tests exercise all endpoints. Unit test project has only a dummy test.