using System.Net; using System.Net.Http.Json; using System.Text.Json; using Microsoft.AspNetCore.Builder; using Microsoft.AspNetCore.Hosting; using Microsoft.AspNetCore.TestHost; using Microsoft.Extensions.Diagnostics.HealthChecks; using Microsoft.Extensions.DependencyInjection; using ZB.MOM.WW.Health; namespace ZB.MOM.WW.Health.Tests; /// /// Verifies the canonical JSON response writer (): /// the JSON body shape, the application/json content type, and that the framework's /// status-to-HTTP mapping (Healthy/Degraded โ†’ 200, Unhealthy โ†’ 503) is preserved when the /// writer is wired onto the ready/active tiers by . /// public sealed class ResponseWriterTests { private sealed class StubHealthCheck : IHealthCheck { private readonly HealthCheckResult _result; public StubHealthCheck(HealthStatus status, string? description = null) => _result = new HealthCheckResult(status, description); public Task CheckHealthAsync( HealthCheckContext context, CancellationToken cancellationToken = default) => Task.FromResult(_result); } private static async Task GetReadyAsync( HealthStatus status, string? description = "db reachable") { var builder = WebApplication.CreateBuilder(); builder.WebHost.UseTestServer(); builder.Services.AddHealthChecks() .AddCheck("db", new StubHealthCheck(status, description), tags: new[] { ZbHealthTags.Ready }); await using var app = builder.Build(); app.MapZbHealth(); await app.StartAsync(); var client = app.GetTestClient(); return await client.GetAsync("/health/ready"); } [Fact] public async Task ReadyEndpoint_Healthy_WritesJsonBody_With200() { var response = await GetReadyAsync(HealthStatus.Healthy); Assert.Equal(HttpStatusCode.OK, response.StatusCode); Assert.Equal("application/json", response.Content.Headers.ContentType?.MediaType); using var doc = JsonDocument.Parse(await response.Content.ReadAsStringAsync()); var root = doc.RootElement; Assert.Equal("Healthy", root.GetProperty("status").GetString()); Assert.Equal(JsonValueKind.Number, root.GetProperty("totalDurationMs").ValueKind); var entries = root.GetProperty("entries"); var db = entries.GetProperty("db"); Assert.Equal("Healthy", db.GetProperty("status").GetString()); Assert.Equal("db reachable", db.GetProperty("description").GetString()); } [Fact] public async Task ReadyEndpoint_NullDescription_EmitsDescriptionKeyAsNull() { // A check that produces no description must still emit the "description" key with a JSON null // value (matching the spec ยง3 example and the HealthChecks.UI.Client shape) rather than // dropping the key โ€” so consumers can read entries..description without handling a // missing property. var response = await GetReadyAsync(HealthStatus.Healthy, description: null); Assert.Equal(HttpStatusCode.OK, response.StatusCode); using var doc = JsonDocument.Parse(await response.Content.ReadAsStringAsync()); var db = doc.RootElement.GetProperty("entries").GetProperty("db"); Assert.True(db.TryGetProperty("description", out var description), "description key must be present"); Assert.Equal(JsonValueKind.Null, description.ValueKind); } [Fact] public async Task ReadyEndpoint_Degraded_Returns200_WithDegradedStatus() { var response = await GetReadyAsync(HealthStatus.Degraded); Assert.Equal(HttpStatusCode.OK, response.StatusCode); Assert.Equal("application/json", response.Content.Headers.ContentType?.MediaType); using var doc = JsonDocument.Parse(await response.Content.ReadAsStringAsync()); Assert.Equal("Degraded", doc.RootElement.GetProperty("status").GetString()); } [Fact] public async Task ReadyEndpoint_Unhealthy_Returns503_WithUnhealthyStatus() { var response = await GetReadyAsync(HealthStatus.Unhealthy); Assert.Equal(HttpStatusCode.ServiceUnavailable, response.StatusCode); Assert.Equal("application/json", response.Content.Headers.ContentType?.MediaType); using var doc = JsonDocument.Parse(await response.Content.ReadAsStringAsync()); Assert.Equal("Unhealthy", doc.RootElement.GetProperty("status").GetString()); } }