Production code:
- POST /api/satellite/tiles/inventory (XOR body, 5000-cap,
most-recent-per-location_hash select, present/absent shaping).
- Kestrel HttpProtocols.Http1AndHttp2 on every listener (AC-5).
- Migration 015 creates tiles_leaflet_path covering index over
(location_hash, captured_at DESC, updated_at DESC, id DESC)
INCLUDE (file_path, source); drops superseded idx_tiles_location_hash.
- TileRepository.GetByTileCoordinatesAsync rewired to filter by
location_hash (Index Only Scan via tiles_leaflet_path).
- TileRepository.GetTilesByLocationHashesAsync added with Npgsql-
direct ANY($1::uuid[]) binding (Dapper IEnumerable expansion is
incompatible with the array form).
- Uuidv5.LocationHashForTile centralises the UUIDv5(TileNamespace,
"{z}/{x}/{y}") formula — single source of truth for the cross-repo
invariant (gps-denied-onboard parity).
Contracts:
- New: contracts/api/tile-inventory.md v1.0.0.
- Bumped: contracts/data-access/tile-storage.md to v2.0.0 (joint
ownership by AZ-503-foundation + AZ-505: schema + covering index +
GetByTileCoordinatesAsync rewrite).
Tests:
- TileInventoryTests covers AC-1, AC-2 (DB-level), AC-4, AC-6.
- Http2MultiplexingTests covers AC-5 (20 concurrent multiplexed GETs
over h2c via SocketsHttpHandler + AppContext Http2Unencrypted switch).
- LeafletPathIndexOnlyTests covers AC-3 (EXPLAIN (ANALYZE, BUFFERS)
asserts Index Only Scan over tiles_leaflet_path with heap_blocks=0).
Docs:
- architecture.md, system-flows.md, data_model.md, module-layout.md,
glossary.md, modules/api_program.md, modules/dataaccess_tile_repository.md,
components/02_data_access/description.md all updated to reference the
v2.0.0 tile-storage contract + new tile-inventory contract + AC-7.
Reports:
- batch_01_cycle6_report.md, batch_01_cycle6_review.md,
implementation_completeness_cycle6_report.md (PASS),
implementation_report_tile_inventory_cycle6.md.
Task spec moved todo/ -> done/.
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)). Enables HTTP/2 over plaintext (h2c) on the dev endpoint so programmatic clients (HttpClientwithVersion20+RequestVersionExact, httpxhttp2=True) can multiplex tile reads on a single TCP connection. Browsers still negotiate HTTP/1.1 over plaintext — browser Leaflet performance is unaffected by the H2 flip and depends instead on thetiles_leaflet_pathcovering index.
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.