asp.netcore 实现健康检测
2021-01-24 20:15
标签:就是 services map edm sqlserver rabbit ali 行操作 art 建议大家使用VS2019开发时把自带的反编译功能打开: 进入正题,首先上一个本节最终能实现的健康检测UI界面: 本节实现的健康检测是通过引用以下系列nuget进行的实现: github上有该包的实现方式,使用过程中可以参考源码来编写相关参数,github地址:https://github.com/Xabaril/AspNetCore.Diagnostics.HealthChecks 在此也列一下清单吧,总共实现了对以下功能点的健康检测功能: 以上主要是些通用的检测点,如api是否正常,数据库、redis等存储库是否正常,rabbitmq是否正常等,除此之后我们还可以自己扩展,比如与我们系统业务相关的功能数据检测等,简单的实现估计也就几行代码,主要看需求。 如何实现web或api、数据库、redis等的状态检测呢?下面开始各种配置。 其中AddCustomHealthCheck的实现如下,实现的检测实例: 可以看到这里实现了对sqlserver数据库、rebbitmq、redis、webapi的健康检测,参数中最重要的是名称及配置如连接字符串的信息,如redis的连接字符串格式如何写,我们可以通过查看源码来了解,此处实现redis的健康检测源码如下: 实现就几行的代码,同时我们看到这里是通过StackExchange.Redis实现的redis连接,我们传递的连接字符串就很明显了,如下: 其他的相关参数也可以通过查看源码的方式来进行操作。。。 ---------------------------------------------------------------------------------------------------------------------------------- 设置数据持久化,可以通过查看源码得知此处是使用的EFCore,支持以下种类的持久化: services中添加配置(记得引用需要的持久化nuget包): 程序启动后会自动根据配置的数据库连接信息创建数据表,大概就是配置信息、当前状态信息、历史记录信息。 ---------------------------------------------------------------------------------------------------------------------------------- 如果健康检测异常需要发送消息至api,可以添加以下配置: 具体参数格式也可通过源码查看,此处是post方式数据传输,关键源码如下: 我们可以通过api接收异常消息后添加日志或进行发送邮件等提醒信息。 最后如果要配置UI界面,只需在endpoint中添加配置: 写的比较散,希望对大家学习有一丝丝帮助~~ asp.netcore 实现健康检测 标签:就是 services map edm sqlserver rabbit ali 行操作 art 原文地址:https://www.cnblogs.com/quluqi/p/13251547.html
1 public class Startup
2 {
3 public Startup(IConfiguration configuration)
4 {
5 Configuration = configuration;
6 }
7 public IConfiguration Configuration { get; }
8
9 public void ConfigureServices(IServiceCollection services)
10 {
11 services.AddCustomHealthCheck(this.Configuration);
12
13 services.AddHealthChecksUI(setupSettings =>
14 {
15 //检测站点,可以添加多条,UI中会把站点内的检测点分组显示(也可通过配置文件实现)
16 setupSettings.AddHealthCheckEndpoint(name: "localhost-5000", uri: "http://localhost:5000/health");
17 //当检测出异常结果时发送消息给api
18 setupSettings.AddWebhookNotification("webhook1",
19 uri: "http://localhost:5008/WeatherForecast/message",
20 payload: "{ \"message\": \"Webhook report for [[LIVENESS]]: [[FAILURE]] - Description: [[DESCRIPTIONS]]\"}",
21 restorePayload: "{ \"message\": \"[[LIVENESS]] is back to life\"}");
22 setupSettings.SetMinimumSecondsBetweenFailureNotifications(60);
23 setupSettings.SetEvaluationTimeInSeconds(10);
24 }).AddSqlServerStorage(Configuration["HealthStorageConnectionString"]);//数据库持久化
25 //.AddInMemoryStorage();//内存持久化
26 }
27
28 public void ConfigureContainer(ContainerBuilder builder)
29 {
30 //autofac
31 }
32
33 // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
34 public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
35 {
36 if (env.IsDevelopment())
37 {
38 app.UseDeveloperExceptionPage();
39 }
40
41 app.UseRouting();
42 app.UseEndpoints(endpoints =>
43 {
44 endpoints.MapHealthChecks("/health", new HealthCheckOptions()
45 {
46 Predicate = _ => true,
47 ResponseWriter = UIResponseWriter.WriteHealthCheckUIResponse
48 });
49 endpoints.MapHealthChecksUI();
50 });
51 }
52 }
1 public static IServiceCollection AddCustomHealthCheck(this IServiceCollection services, IConfiguration configuration)
2 {
3 var hcBuilder = services.AddHealthChecks();
4
5 hcBuilder.AddCheck("Jobs.MicrosoftBackgroundService-apiself", () => HealthCheckResult.Healthy());
6 hcBuilder.AddSqlServer(
7 configuration["ConnectionString"],
8 name: "Jobs.MicrosoftBackgroundService-sqlserver-check",
9 tags: new string[] { "sqlserverdb" });
10 hcBuilder.AddRabbitMQ(
11 $"amqp://{configuration["EventBusConnection"]}",
12 name: "Jobs.MicrosoftBackgroundService-rabbitmqbus-check",
13 tags: new string[] { "rabbitmqbus" });
14 hcBuilder.AddRedis(
15 configuration["RedisConnectionString"],
16 name: "Jobs.MicrosoftBackgroundService-redis-check",
17 tags: new string[] { "redisdb" });
18 hcBuilder.AddUrlGroup(new Uri(configuration["testApiUrl"]), name: "testApiUrl-check", tags: new string[] { "testApiUrl" });
19
20 return services;
21 }
1 using Microsoft.Extensions.Diagnostics.HealthChecks;
2 using StackExchange.Redis;
3 using System;
4 using System.Collections.Concurrent;
5 using System.Threading;
6 using System.Threading.Tasks;
7
8 namespace HealthChecks.Redis
9 {
10 public class RedisHealthCheck : IHealthCheck
11 {
12 private static readonly ConcurrentDictionarystring, ConnectionMultiplexer> _connections = new ConcurrentDictionarystring, ConnectionMultiplexer>();
13
14 private readonly string _redisConnectionString;
15
16 public RedisHealthCheck(string redisConnectionString)
17 {
18 _redisConnectionString = (redisConnectionString ?? throw new ArgumentNullException("redisConnectionString"));
19 }
20
21 public async Task
"RedisConnectionString": "192.168.217.128:6379,defaultDatabase=1,password=123!@#..."
services.AddSqlServerStorage(Configuration["HealthStorageConnectionString"]);
1 foreach (var webHook in _settings.Webhooks)
2 {
3 bool shouldNotify = webHook.ShouldNotifyFunc?.Invoke(report) ?? true;
4
5 if (!shouldNotify)
6 {
7 _logger.LogInformation("Webhook notification will not be sent because of user configuration");
8 continue;
9 };
10
11 if (!isHealthy)
12 {
13 failure = webHook.CustomMessageFunc?.Invoke(report) ?? GetFailedMessageFromContent(report);
14 description = webHook.CustomDescriptionFunc?.Invoke(report) ?? GetFailedDescriptionsFromContent(report);
15 }
16
17 var payload = isHealthy ? webHook.RestoredPayload : webHook.Payload;
18 payload = payload.Replace(Keys.LIVENESS_BOOKMARK, name)
19 .Replace(Keys.FAILURE_BOOKMARK, failure)
20 .Replace(Keys.DESCRIPTIONS_BOOKMARK, description);
21
22
23 Uri.TryCreate(webHook.Uri, UriKind.Absolute, out var absoluteUri);
24
25 if (absoluteUri == null || !absoluteUri.IsValidHealthCheckEndpoint())
26 {
27 Uri.TryCreate(_serverAddressesService.AbsoluteUriFromRelative(webHook.Uri), UriKind.Absolute, out absoluteUri);
28 }
29
30 await SendRequest(absoluteUri, webHook.Name, payload);
31 }
可以看到此处的Webhooks可以配置多个 1 private async Task SendRequest(Uri uri, string name, string payloadContent)
2 {
3 try
4 {
5 var payload = new StringContent(payloadContent, Encoding.UTF8, Keys.DEFAULT_RESPONSE_CONTENT_TYPE);
6 var response = await _httpClient.PostAsync(uri, payload);
7 if (!response.IsSuccessStatusCode)
8 {
9 _logger.LogError("The webhook notification has not executed successfully for {name} webhook. The error code is {statuscode}.", name, response.StatusCode);
10 }
11
12 }
13 catch (Exception exception)
14 {
15 _logger.LogError($"The failure notification for {name} has not executed successfully.", exception);
16 }
17 }
1 app.UseEndpoints(endpoints =>
2 {
3 endpoints.MapHealthChecks("/health", new HealthCheckOptions()
4 {
5 Predicate = _ => true,
6 ResponseWriter = UIResponseWriter.WriteHealthCheckUIResponse
7 });//要进行该站点检测应添加此代码
8 endpoints.MapHealthChecksUI();//添加UI界面支持
9 });