Files
satellite-provider/SatelliteProvider.IntegrationTests/RegionTests.cs
T
Oleksandr Bezdieniezhnykh fcd494f67e [AZ-812] Region API: rename Latitude/Longitude → Lat/Lon (OSM convention)
Mirror of AZ-794 (inventory z/x/y rename). RequestRegionRequest.cs renames C#
props Latitude→Lat / Longitude→Lon and adds [JsonPropertyName("lat"/"lon")] so
the wire format is unambiguous under the AZ-795 strict-parsing stack
(UnmappedMemberHandling.Disallow → legacy {"latitude":..,"longitude":..} now
returns HTTP 400 instead of silently coercing).

Updates all in-repo consumers: API handler (Program.cs), integration tests
(Models.cs, RegionTests.cs, IdempotentPostTests.cs, SecurityTests.cs), the
performance harness (run-performance-tests.sh PT-03/04/05/07), and module
docs (common_dtos.md, api_program.md; system-flows.md F2 already used
lat/lon). New RegionFieldRenameTests.cs covers AC-4 both directions (new
format → 200, legacy format → 400). Smoke green; no regressions.

region-request.md contract doc not bumped here — AZ-808 publishes v1.0.0
directly with the post-rename names per AZ-812 coordination clause.

Batch 01 of cycle 8. PASS_WITH_WARNINGS (one Low DRY finding for follow-up
test-helper consolidation; details in
_docs/03_implementation/reviews/batch_01_cycle8_review.md).

Co-authored-by: Cursor <cursoragent@cursor.com>
2026-05-22 15:54:53 +03:00

181 lines
6.5 KiB
C#

using System.Net.Http.Json;
using System.Text.Json;
namespace SatelliteProvider.IntegrationTests;
public static class RegionTests
{
private static readonly JsonSerializerOptions JsonOptions = new()
{
PropertyNameCaseInsensitive = true
};
public static async Task RunRegionProcessingTest_200m_Zoom18(HttpClient httpClient)
{
Console.WriteLine();
Console.WriteLine("Test: Region Processing 200m at Zoom 18");
Console.WriteLine("------------------------------------------------------------------");
const double latitude = 47.461747;
const double longitude = 37.647063;
const double sizeMeters = 200;
const int zoomLevel = 18;
const bool stitchTiles = false;
await RunRegionProcessingTest(httpClient, latitude, longitude, sizeMeters, zoomLevel, stitchTiles);
Console.WriteLine();
Console.WriteLine("Region Processing Test (200m, Zoom 18): PASSED");
}
public static async Task RunRegionProcessingTest_400m_Zoom17(HttpClient httpClient)
{
Console.WriteLine();
Console.WriteLine("Test: Region Processing 400m at Zoom 17");
Console.WriteLine("------------------------------------------------------------------");
const double latitude = 47.461747;
const double longitude = 37.647063;
const double sizeMeters = 400;
const int zoomLevel = 17;
const bool stitchTiles = false;
await RunRegionProcessingTest(httpClient, latitude, longitude, sizeMeters, zoomLevel, stitchTiles);
Console.WriteLine();
Console.WriteLine("Region Processing Test (400m, Zoom 17): PASSED");
}
public static async Task RunRegionProcessingTest_500m_Zoom18(HttpClient httpClient)
{
Console.WriteLine();
Console.WriteLine("Test: Region Processing 500m at Zoom 18 with Stitching");
Console.WriteLine("------------------------------------------------------------------");
const double latitude = 47.461747;
const double longitude = 37.647063;
const double sizeMeters = 500;
const int zoomLevel = 18;
const bool stitchTiles = true;
await RunRegionProcessingTest(httpClient, latitude, longitude, sizeMeters, zoomLevel, stitchTiles);
Console.WriteLine();
Console.WriteLine("Region Processing Test (500m, Zoom 18 with Stitching): PASSED");
}
private static async Task RunRegionProcessingTest(
HttpClient httpClient,
double latitude,
double longitude,
double sizeMeters,
int zoomLevel,
bool stitchTiles)
{
var regionId = Guid.NewGuid();
Console.WriteLine($"Requesting region: ID={regionId}");
Console.WriteLine($" Coordinates: ({latitude}, {longitude})");
Console.WriteLine($" Size: {sizeMeters}m");
Console.WriteLine($" Zoom Level: {zoomLevel}");
Console.WriteLine($" Stitch Tiles: {stitchTiles}");
Console.WriteLine();
var requestRegion = new RequestRegionRequest
{
Id = regionId,
Lat = latitude,
Lon = longitude,
SizeMeters = sizeMeters,
ZoomLevel = zoomLevel,
StitchTiles = stitchTiles
};
var requestResponse = await httpClient.PostAsJsonAsync("/api/satellite/request", requestRegion);
if (!requestResponse.IsSuccessStatusCode)
{
var errorContent = await requestResponse.Content.ReadAsStringAsync();
throw new Exception($"Region request failed with status {requestResponse.StatusCode}: {errorContent}");
}
var initialStatus = await requestResponse.Content.ReadFromJsonAsync<RegionStatusResponse>(JsonOptions);
if (initialStatus == null)
{
throw new Exception("No status returned from region request");
}
Console.WriteLine($"✓ Region queued successfully");
Console.WriteLine($" Initial Status: {initialStatus.Status}");
Console.WriteLine();
Console.WriteLine("Polling for region status updates...");
RegionStatusResponse? finalStatus = null;
int maxAttempts = TestRunMode.RegionPollAttempts;
for (int i = 0; i < maxAttempts; i++)
{
await Task.Delay(2000);
var statusResponse = await httpClient.GetAsync($"/api/satellite/region/{regionId}");
if (!statusResponse.IsSuccessStatusCode)
{
var errorContent = await statusResponse.Content.ReadAsStringAsync();
throw new Exception($"Status check failed with status {statusResponse.StatusCode}: {errorContent}");
}
var status = await statusResponse.Content.ReadFromJsonAsync<RegionStatusResponse>(JsonOptions);
if (status == null)
{
throw new Exception("No status returned");
}
Console.WriteLine($" Attempt {i + 1}/{maxAttempts}: Status = {status.Status}");
if (status.Status == "completed" || status.Status == "failed")
{
finalStatus = status;
break;
}
}
if (finalStatus == null)
{
throw new Exception($"Region processing did not complete in time (waited {maxAttempts * 2} seconds)");
}
Console.WriteLine();
Console.WriteLine("Region Processing Results:");
Console.WriteLine($" Status: {finalStatus.Status}");
Console.WriteLine($" Tiles Downloaded: {finalStatus.TilesDownloaded}");
Console.WriteLine($" Tiles Reused: {finalStatus.TilesReused}");
Console.WriteLine($" CSV File: {finalStatus.CsvFilePath}");
Console.WriteLine($" Summary File: {finalStatus.SummaryFilePath}");
Console.WriteLine($" Created At: {finalStatus.CreatedAt:yyyy-MM-dd HH:mm:ss}");
Console.WriteLine($" Updated At: {finalStatus.UpdatedAt:yyyy-MM-dd HH:mm:ss}");
if (finalStatus.Status != "completed")
{
throw new Exception($"Expected status 'completed', got '{finalStatus.Status}'");
}
if (string.IsNullOrEmpty(finalStatus.CsvFilePath))
{
throw new Exception("CSV file path is empty");
}
if (string.IsNullOrEmpty(finalStatus.SummaryFilePath))
{
throw new Exception("Summary file path is empty");
}
Console.WriteLine();
Console.WriteLine("✓ Region processed successfully");
Console.WriteLine("✓ CSV and summary files created");
}
}