Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: Refactor probes and add more health checks #1159

Merged
merged 55 commits into from
Oct 8, 2024
Merged
Show file tree
Hide file tree
Changes from 41 commits
Commits
Show all changes
55 commits
Select commit Hold shift + click to select a range
1317e97
sup
oskogstad Sep 18, 2024
51c3515
low cse
oskogstad Sep 18, 2024
b25034e
chore: mo health checks pls
arealmaas Sep 20, 2024
42326d1
chore: even mo health checks pls
arealmaas Sep 20, 2024
a275c12
Merge branch 'main' into chore/add-redis-postgres-health-checks
arealmaas Sep 20, 2024
ddf0ce7
cleanup
arealmaas Sep 20, 2024
d58b8e4
Update src/Digdir.Domain.Dialogporten.Infrastructure/HealthChecks/Wel…
arealmaas Sep 20, 2024
1cafcdf
Update src/Digdir.Domain.Dialogporten.Infrastructure/HealthChecks/Wel…
arealmaas Sep 20, 2024
3996060
cleanup
arealmaas Sep 20, 2024
8889b77
cleanup
arealmaas Sep 20, 2024
02c13b0
cleanup
arealmaas Sep 23, 2024
edb9f93
Merge branch 'main' into chore/add-redis-postgres-health-checks
oskogstad Oct 1, 2024
e9322f1
Update src/Digdir.Domain.Dialogporten.Infrastructure/HealthChecks/Wel…
oskogstad Oct 1, 2024
383f56a
chore: adjust health checks
arealmaas Oct 1, 2024
62bd903
chore: adjust health checks
arealmaas Oct 1, 2024
1129ce2
cleanup
arealmaas Oct 1, 2024
7ffc934
Merge branch 'main' into chore/add-redis-postgres-health-checks
arealmaas Oct 1, 2024
33e7c4e
chore(deps): update dependency htmlagilitypack to 1.11.66 (#1212)
renovate[bot] Oct 2, 2024
08e94cb
ci: ensure unique revisions for deployments (#1211)
arealmaas Oct 2, 2024
f8fe468
chore(main): release 1.20.1 (#1207)
dialogporten-bot Oct 2, 2024
ea2400b
ci: send message on release created and successfully validated (#1185)
arealmaas Oct 2, 2024
b755f0c
beginning
arealmaas Oct 2, 2024
84f0941
refactor
arealmaas Oct 3, 2024
4b8fdb6
refactor
arealmaas Oct 3, 2024
59ef954
ok
arealmaas Oct 3, 2024
2ca9fae
ok
arealmaas Oct 3, 2024
d7e3b3f
ok
arealmaas Oct 3, 2024
e41bd02
swaggah
arealmaas Oct 3, 2024
fcf6727
cleanup
arealmaas Oct 3, 2024
34ce8a8
cleanup
arealmaas Oct 3, 2024
62ced3e
cleanup
arealmaas Oct 3, 2024
3abf115
Update src/Digdir.Domain.Dialogporten.Infrastructure/HealthChecks/Red…
arealmaas Oct 3, 2024
3173da9
cleanup
arealmaas Oct 3, 2024
2f267c8
cleanup
arealmaas Oct 3, 2024
48bab84
cleanup
arealmaas Oct 3, 2024
1a4dddb
cleanup
arealmaas Oct 3, 2024
296c01b
Update src/Digdir.Domain.Dialogporten.Infrastructure/HealthChecks/Red…
arealmaas Oct 3, 2024
4038e3b
cleanup
arealmaas Oct 3, 2024
f5eea3a
cleanup
arealmaas Oct 3, 2024
1210864
cleanup
arealmaas Oct 3, 2024
4ba55db
Update src/Digdir.Library.Utils.AspNet/HealthChecks/EndpointsHealthCh…
arealmaas Oct 3, 2024
f1b0186
Update src/Digdir.Domain.Dialogporten.Infrastructure/HealthChecks/Red…
arealmaas Oct 4, 2024
dd80084
add to sln
arealmaas Oct 4, 2024
f6d385b
doc
arealmaas Oct 7, 2024
9792cda
Merge branch 'main' into chore/add-redis-postgres-health-checks
arealmaas Oct 7, 2024
e282c76
redo options
arealmaas Oct 7, 2024
51af7ed
cleanup
arealmaas Oct 7, 2024
776cb9d
cleanup
arealmaas Oct 7, 2024
49a3b89
cleanup
arealmaas Oct 7, 2024
89accc8
Some suggestions (#1243)
MagnusSandgren Oct 7, 2024
abcf931
parallelize
arealmaas Oct 7, 2024
284cd17
cleanup
arealmaas Oct 7, 2024
8b8806d
Merge branch 'main' into chore/add-redis-postgres-health-checks
arealmaas Oct 7, 2024
353138e
cleanup
arealmaas Oct 8, 2024
de44087
cleanup
arealmaas Oct 8, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
33 changes: 33 additions & 0 deletions .azure/applications/web-api-eu/main.bicep
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,38 @@ resource environmentKeyVaultResource 'Microsoft.KeyVault/vaults@2023-07-01' exis

var containerAppName = '${namePrefix}-webapi-eu-ca'

var port = 8080

var probes = [
{
periodSeconds: 5
initialDelaySeconds: 2
type: 'Liveness'
httpGet: {
path: '/liveness'
port: port
}
}
{
periodSeconds: 5
initialDelaySeconds: 2
type: 'Readiness'
httpGet: {
path: '/readiness'
port: port
}
}
{
periodSeconds: 5
initialDelaySeconds: 2
type: 'Startup'
httpGet: {
path: '/startup'
port: port
}
}
]

module containerApp '../../modules/containerApp/main.bicep' = {
name: containerAppName
params: {
Expand All @@ -94,6 +126,7 @@ module containerApp '../../modules/containerApp/main.bicep' = {
apimIp: apimIp
tags: tags
resources: resources
probes: probes
revisionSuffix: revisionSuffix
}
}
Expand Down
34 changes: 34 additions & 0 deletions .azure/applications/web-api-so/main.bicep
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,38 @@ resource environmentKeyVaultResource 'Microsoft.KeyVault/vaults@2023-07-01' exis

var containerAppName = '${namePrefix}-webapi-so-ca'

var port = 8080

var probes = [
{
periodSeconds: 5
initialDelaySeconds: 2
type: 'Liveness'
httpGet: {
path: '/liveness'
port: port
}
}
{
periodSeconds: 5
initialDelaySeconds: 2
type: 'Readiness'
httpGet: {
path: '/readiness'
port: port
}
}
{
periodSeconds: 5
initialDelaySeconds: 2
type: 'Startup'
httpGet: {
path: '/startup'
port: port
}
}
]
arealmaas marked this conversation as resolved.
Show resolved Hide resolved

module containerApp '../../modules/containerApp/main.bicep' = {
name: containerAppName
params: {
Expand All @@ -98,6 +130,8 @@ module containerApp '../../modules/containerApp/main.bicep' = {
apimIp: apimIp
tags: tags
resources: resources
probes: probes
port: port
revisionSuffix: revisionSuffix
}
}
Expand Down
25 changes: 3 additions & 22 deletions .azure/modules/containerApp/main.bicep
Original file line number Diff line number Diff line change
Expand Up @@ -28,30 +28,12 @@ param resources object?
@description('The suffix for the revision of the container app')
param revisionSuffix string

@description('The probes for the container app')
param probes array = []
arealmaas marked this conversation as resolved.
Show resolved Hide resolved

// Container app revision name does not allow '.' character
var cleanedRevisionSuffix = replace(revisionSuffix, '.', '-')

var probes = [
{
periodSeconds: 5
initialDelaySeconds: 2
type: 'Liveness'
httpGet: {
path: '/healthz'
port: port
}
}
{
periodSeconds: 5
initialDelaySeconds: 2
type: 'Readiness'
httpGet: {
path: '/healthz'
port: port
}
}
]

var ipSecurityRestrictions = empty(apimIp)
? []
: [
Expand All @@ -74,7 +56,6 @@ resource containerApp 'Microsoft.App/containerApps@2024-03-01' = {
identity: {
type: 'SystemAssigned'
}

properties: {
configuration: {
ingress: ingress
Expand Down
2 changes: 1 addition & 1 deletion docs/schema/V1/swagger.verified.json
Original file line number Diff line number Diff line change
Expand Up @@ -6460,4 +6460,4 @@
"url": "https://altinn-dev-api.azure-api.net/dialogporten"
}
]
}
}
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<TreatAsUsed>true</TreatAsUsed>
</PackageReference>
<PackageReference Include="Microsoft.Extensions.Diagnostics.HealthChecks" Version="8.0.0"/>
<PackageReference Include="Microsoft.Extensions.Diagnostics.HealthChecks.EntityFrameworkCore" Version="8.0.0"/>
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
<PackageReference Include="Microsoft.Extensions.Hosting.Abstractions" Version="8.0.0"/>
<PackageReference Include="Polly.Contrib.WaitAndRetry" Version="1.1.1"/>
<PackageReference Include="ZiggyCreatures.FusionCache" Version="1.4.0" />
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
using System;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.Extensions.Diagnostics.HealthChecks;
using StackExchange.Redis;
using Microsoft.Extensions.Options;

namespace Digdir.Domain.Dialogporten.Infrastructure.HealthChecks;

internal sealed class RedisHealthCheck : IHealthCheck
{
private readonly InfrastructureSettings _settings;

public RedisHealthCheck(IOptions<InfrastructureSettings> options)
{
ArgumentNullException.ThrowIfNull(options);
if (options.Value == null)
{
throw new ArgumentException("InfrastructureSettings cannot be null.", nameof(options));
}
_settings = options.Value;
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
}

public async Task<HealthCheckResult> CheckHealthAsync(HealthCheckContext context, CancellationToken cancellationToken = default)
{
try
{
using var redis = await ConnectionMultiplexer.ConnectAsync(_settings.Redis.ConnectionString);
var db = redis.GetDatabase();
await db.PingAsync();
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
return HealthCheckResult.Healthy("Redis connection is healthy.");
}
catch (RedisConnectionException ex)
{
return HealthCheckResult.Unhealthy("Unable to connect to Redis.", exception: ex);
}
catch (Exception ex)
{
return HealthCheckResult.Unhealthy("An unexpected error occurred while checking Redis health.", exception: ex);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,8 @@
using StackExchange.Redis;
using ZiggyCreatures.Caching.Fusion;
using ZiggyCreatures.Caching.Fusion.NullObjects;
using Microsoft.Extensions.Diagnostics.HealthChecks;
using Digdir.Domain.Dialogporten.Infrastructure.HealthChecks;

namespace Digdir.Domain.Dialogporten.Infrastructure;

Expand Down Expand Up @@ -199,6 +201,8 @@ public static IServiceCollection AddInfrastructure(this IServiceCollection servi
})
.AddPolicyHandlerFromRegistry(PollyPolicy.DefaultHttpRetryPolicy);

services.AddCustomHealthChecks();

if (environment.IsDevelopment())
{
var localDeveloperSettings = configuration.GetLocalDevelopmentSettings();
Expand All @@ -212,6 +216,24 @@ public static IServiceCollection AddInfrastructure(this IServiceCollection servi
return services;
}

private static IServiceCollection AddCustomHealthChecks(this IServiceCollection services)
{
services.AddHttpClient("HealthCheckClient")
.ConfigureHttpClient(client =>
{
client.Timeout = TimeSpan.FromSeconds(5);
})
.AddPolicyHandlerFromRegistry(PollyPolicy.DefaultHttpRetryPolicy); ;
arealmaas marked this conversation as resolved.
Show resolved Hide resolved

services.AddHealthChecks()
MagnusSandgren marked this conversation as resolved.
Show resolved Hide resolved
.AddCheck<RedisHealthCheck>("redis", tags: ["dependencies", "redis"])
.AddDbContextCheck<DialogDbContext>("postgres", tags: ["dependencies", "critical"]);

services.AddSingleton<RedisHealthCheck>();
arealmaas marked this conversation as resolved.
Show resolved Hide resolved

return services;
}
arealmaas marked this conversation as resolved.
Show resolved Hide resolved

private static IServiceCollection AddGraphQlRedisSubscriptions(this IServiceCollection services,
string redisConnectionString)
{
Expand Down Expand Up @@ -295,4 +317,4 @@ private static IServiceCollection ConfigureFusionCache(this IServiceCollection s

return services;
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,8 @@
<ItemGroup>
<ProjectReference Include="..\Digdir.Domain.Dialogporten.Application\Digdir.Domain.Dialogporten.Application.csproj"/>
<ProjectReference Include="..\Digdir.Domain.Dialogporten.Infrastructure\Digdir.Domain.Dialogporten.Infrastructure.csproj"/>
<ProjectReference Include="..\Digdir.Library.Utils.AspNet\Digdir.Library.Utils.AspNet.csproj"/>
<ProjectReference Include="..\Digdir.Tool.Dialogporten.GenerateFakeData\Digdir.Tool.Dialogporten.GenerateFakeData.csproj"/>
</ItemGroup>

</Project>
</Project>
16 changes: 12 additions & 4 deletions src/Digdir.Domain.Dialogporten.WebApi/Program.cs
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
using Microsoft.AspNetCore.Authorization;
using NSwag;
using Serilog;
using Digdir.Library.Utils.AspNet;

// Using two-stage initialization to catch startup errors.
Log.Logger = new LoggerConfiguration()
Expand Down Expand Up @@ -126,11 +127,15 @@ static void BuildAndRun(string[] args)
.AddNewtonsoftJson()
.Services


// Auth
.AddDialogportenAuthentication(builder.Configuration)
.AddAuthorization()
.AddHealthChecks();

// Health checks with configuration
.AddAspNetHealthChecks(builder.Configuration, options =>
{
options.WellKnownEndpointsConfigurationSectionPath = "WebApi:Authentication:JwtBearerTokenSchemas";
});

if (builder.Environment.IsDevelopment())
{
Expand Down Expand Up @@ -200,7 +205,10 @@ static void BuildAndRun(string[] args)
uiConfig.DocumentPath = dialogPrefix + "/swagger/{documentName}/swagger.json";
});
app.MapControllers();
app.MapHealthChecks("/healthz");

// Map Health Checks
app.MapAspNetHealthChecks();
arealmaas marked this conversation as resolved.
Show resolved Hide resolved

app.Run();
}

Expand All @@ -216,4 +224,4 @@ static void IgnoreEmptyCollections(JsonTypeInfo typeInfo)
}

// ReSharper disable once ClassNeverInstantiated.Global
public sealed partial class Program;
public sealed partial class Program;
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
MagnusSandgren marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
<Project Sdk="Microsoft.NET.Sdk">

<PropertyGroup>
<GenerateDocumentationFile>true</GenerateDocumentationFile>
<NoWarn>1591</NoWarn> <!-- Disable warnings for missing XML comments -->
</PropertyGroup>
arealmaas marked this conversation as resolved.
Show resolved Hide resolved

<ItemGroup>
<PackageReference Include="Microsoft.Extensions.Http" Version="8.0.0" />
<PackageReference Include="AspNetCore.HealthChecks.UI.Client" Version="8.0.1" />
</ItemGroup>
arealmaas marked this conversation as resolved.
Show resolved Hide resolved

</Project>
78 changes: 78 additions & 0 deletions src/Digdir.Library.Utils.AspNet/HealthCheckExtensions.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
using Digdir.Library.Utils.AspNet.HealthChecks;
using HealthChecks.UI.Client;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Diagnostics.HealthChecks;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Diagnostics.HealthChecks;

namespace Digdir.Library.Utils.AspNet;

public static class HealthCheckExtensions
{
public class AspNetHealthChecksOptions
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
{
public string WellKnownEndpointsConfigurationSectionPath { get; set; } = string.Empty;
}

public class JwtBearerTokenSchema
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
{
public string Name { get; set; } = string.Empty;
public string WellKnown { get; set; } = string.Empty;
}
arealmaas marked this conversation as resolved.
Show resolved Hide resolved

public static IServiceCollection AddAspNetHealthChecks(this IServiceCollection services, IConfiguration configuration, Action<AspNetHealthChecksOptions> configure)
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
{
var options = new AspNetHealthChecksOptions();
configure(options);
arealmaas marked this conversation as resolved.
Show resolved Hide resolved

var wellKnownSchemas = configuration
.GetSection(options.WellKnownEndpointsConfigurationSectionPath)
.Get<List<JwtBearerTokenSchema>>();

var healthChecks = services.AddHealthChecks();

healthChecks.AddCheck("self", () => HealthCheckResult.Healthy(), tags: ["self"]);

var wellKnownEndpoints = wellKnownSchemas?.Select(schema => schema.WellKnown).ToList() ?? new List<string>();

if (wellKnownEndpoints.Count > 0)
{
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
services.Configure<EndpointsHealthCheckOptions>(opts =>
{
opts.Endpoints = wellKnownEndpoints;
});

healthChecks.AddCheck<EndpointsHealthCheck>(
"Endpoints",
failureStatus: HealthStatus.Unhealthy,
tags: ["dependencies"]);
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
}

return services;
}

public static void MapAspNetHealthChecks(this WebApplication app)
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
{
app.MapHealthChecks("/startup", new HealthCheckOptions
{
Predicate = check => check.Tags.Contains("dependencies"),
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
ResponseWriter = UIResponseWriter.WriteHealthCheckUIResponse
});
app.MapHealthChecks("/liveness", new HealthCheckOptions
{
Predicate = check => check.Tags.Contains("self"),
ResponseWriter = UIResponseWriter.WriteHealthCheckUIResponse
});
app.MapHealthChecks("/readiness", new HealthCheckOptions
{
Predicate = check => check.Tags.Contains("critical"),
ResponseWriter = UIResponseWriter.WriteHealthCheckUIResponse
});
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
app.MapHealthChecks("/health", new HealthCheckOptions
{
Predicate = check => check.Tags.Contains("dependencies"),
ResponseWriter = UIResponseWriter.WriteHealthCheckUIResponse
});
}
arealmaas marked this conversation as resolved.
Show resolved Hide resolved
}
Loading
Loading