Metrics /2 (#29646)
* Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Metrics /2 * Apply suggestions from code review Co-authored-by: James Newton-King <james@newtonking.com> * react to feedback --------- Co-authored-by: James Newton-King <james@newtonking.com>pull/29752/head
parent
b690c39156
commit
4c9a0ecb16
|
@ -0,0 +1,180 @@
|
|||
---
|
||||
title: ASP.NET Core metrics
|
||||
description: Metrics for ASP.NET Core apps
|
||||
author: rick-anderson
|
||||
ms.author: riande
|
||||
monikerRange: '>= aspnetcore-8.0'
|
||||
ms.date: 6/30/2023
|
||||
ms.topic: article
|
||||
ms.prod: aspnet-core
|
||||
uid: log-mon/metrics/metrics
|
||||
---
|
||||
|
||||
# ASP.NET Core metrics
|
||||
|
||||
Metrics are numerical measurements reported over time. They're typically used to monitor the health of an app and generate alerts. For example, a web service might track how many:
|
||||
|
||||
* Requests it received per second.
|
||||
* Milliseconds it took to respond.
|
||||
* Responses sent an error.
|
||||
|
||||
These metrics can be reported to a monitoring system at regular intervals. Dashboards can be setup to view metrics and alerts created to notify people of problems. If the web service is intended to respond to requests within 400 ms and starts responding in 600 ms, the monitoring system can notify the operations staff that the app response is slower than normal.
|
||||
|
||||
See [ASP.NET Core metrics](https://github.com/dotnet/aspnetcore/issues/47536) for ASP.NET Core specific metrics. See [.NET metrics](/dotnet/core/diagnostics/metrics) for .NET metrics.
|
||||
|
||||
## Using metrics
|
||||
|
||||
There are two parts to using metrics in a .NET app:
|
||||
|
||||
* **Instrumentation:** Code in .NET libraries takes measurements and associates these measurements with a metric name.
|
||||
* **Collection:** A .NET app configures named metrics to be transmitted from the app for external storage and analysis. Some tools may perform configuration outside the app using configuration files or a UI tool.
|
||||
|
||||
Instrumented code can record numeric measurements, but the measurements need to be aggregated, transmitted, and stored to create useful metrics for monitoring. The process of aggregating, transmitting, and storing data is called collection. This tutorial shows several examples of collecting metrics:
|
||||
|
||||
* Populating metrics in [Grafana](https://grafana.com/) with [OpenTelemetry](https://opentelemetry.io/) and [Prometheus](https://prometheus.io/).
|
||||
* Viewing metrics in real time with [`dotnet-counters`](/dotnet/core/diagnostics/dotnet-counters)
|
||||
|
||||
## Create the starter app
|
||||
|
||||
Create a new ASP.NET Core app with the following command:
|
||||
|
||||
```dotnetcli
|
||||
dotnet new web -o WebMetric
|
||||
cd WebMetric
|
||||
dotnet add package OpenTelemetry.Exporter.Prometheus.AspNetCore --prerelease
|
||||
dotnet add package OpenTelemetry.Extensions.Hosting
|
||||
```
|
||||
|
||||
Replace the contents of `Program.cs` with the following code:
|
||||
|
||||
:::code language="csharp" source="~/log-mon/metrics/metrics/samples/Program.cs":::
|
||||
|
||||
<!-- Review TODO: Add link to available meters -->
|
||||
|
||||
## View metrics with dotnet-counters
|
||||
|
||||
[dotnet-counters](/dotnet/core/diagnostics/dotnet-counters) is a command-line tool that can view live metrics for .NET Core apps on demand. It doesn't require setup, making it useful for ad-hoc investigations or verifying that metric instrumentation is working. It works with both <xref:System.Diagnostics.Metrics?displayProperty=nameWithType> based APIs and [EventCounters](/dotnet/core/diagnostics/event-counters).
|
||||
|
||||
If the [dotnet-counters](/dotnet/core/diagnostics/dotnet-counters) tool isn't installed, run the following command:
|
||||
|
||||
```dotnetcli
|
||||
dotnet tool update -g dotnet-counters
|
||||
```
|
||||
|
||||
While the test app is running, launch [dotnet-counters](/dotnet/core/diagnostics/dotnet-counters).
|
||||
The following command shows an example of `dotnet-counters` monitoring all metrics from the `Microsoft.AspNetCore.Hosting` meter.
|
||||
|
||||
```dotnetcli
|
||||
dotnet-counters monitor -n WebMetric --counters Microsoft.AspNetCore.Hosting
|
||||
```
|
||||
|
||||
Output similar to the following is displayed:
|
||||
|
||||
```dotnetcli
|
||||
Press p to pause, r to resume, q to quit.
|
||||
Status: Running
|
||||
|
||||
[Microsoft.AspNetCore.Hosting]
|
||||
http-server-current-requests
|
||||
host=localhost,method=GET,port=5045,scheme=http 0
|
||||
http-server-request-duration (s)
|
||||
host=localhost,method=GET,port=5045,protocol=HTTP/1.1,ro 0.001
|
||||
host=localhost,method=GET,port=5045,protocol=HTTP/1.1,ro 0.001
|
||||
host=localhost,method=GET,port=5045,protocol=HTTP/1.1,ro 0.001
|
||||
host=localhost,method=GET,port=5045,protocol=HTTP/1.1,ro 0
|
||||
host=localhost,method=GET,port=5045,protocol=HTTP/1.1,ro 0
|
||||
host=localhost,method=GET,port=5045,protocol=HTTP/1.1,ro 0 12
|
||||
```
|
||||
|
||||
Run `WebMetric>dotnet-counters list` to show all available metrics.
|
||||
|
||||
The following command shows the `Microsoft-AspNetCore-Server-Kestrel` meter with the `requests-per-second` and `total-requests` counters.
|
||||
|
||||
```dotnetcli
|
||||
dotnet-counters monitor -n WebMetric --counters Microsoft.AspNetCore.Hosting[requests-per-second,total-requests]
|
||||
```
|
||||
|
||||
For more information, see [dotnet-counters](/dotnet/core/diagnostics/dotnet-counters). To learn more about metrics in .NET, see [built-in metrics](/dotnet/core/diagnostics/available-counters).
|
||||
|
||||
## Create custom metrics
|
||||
|
||||
See [Create custom metrics](/dotnet/core/diagnostics/metrics#create-custom-metrics) for information on creating custom metrics.
|
||||
|
||||
## View metrics in Grafana with OpenTelemetry and Prometheus
|
||||
|
||||
### Overview
|
||||
|
||||
[OpenTelemetry](https://opentelemetry.io/):
|
||||
|
||||
* Is a vendor-neutral open-source project supported by the [Cloud Native Computing Foundation](https://www.cncf.io/).
|
||||
* Standardizes generating and collecting telemetry for cloud-native software.
|
||||
* Works with .NET using the .NET metric APIs.
|
||||
* Is endorsed by [Azure Monitor](/azure/azure-monitor/app/opentelemetry-overview) and many APM vendors.
|
||||
|
||||
This tutorial shows one of the integrations available for OpenTelemetry metrics using the OSS [Prometheus](https://prometheus.io/) and [Grafana](https://grafana.com/) projects. The metrics data flow:
|
||||
|
||||
1. The ASP.NET Core metric APIs record measurements from the example app.
|
||||
1. The OpenTelemetry .NET library running in the app aggregates the measurements.
|
||||
1. The Prometheus exporter library makes the aggregated data available via an HTTP metrics endpoint. 'Exporter' is what OpenTelemetry calls the libraries that transmit telemetry to vendor-specific backends.
|
||||
1. A Prometheus server:
|
||||
|
||||
* Polls the metrics endpoint
|
||||
* Reads the data
|
||||
* Stores the data in a database for long-term persistence. Prometheus refers to reading and storing data as *scraping* an endpoint.
|
||||
* Can run on a different machine
|
||||
|
||||
1. The Grafana server:
|
||||
|
||||
* Queries the data stored in Prometheus and displays it on a web-based monitoring dashboard.
|
||||
* Can run on a different machine.
|
||||
|
||||
### View metrics from sample app
|
||||
|
||||
Navigate to the sample app. The browser displays `Hello OpenTelemetry! ticks:<3digits>` where `3digits` are the last 3 digits of the current [DateTime.Ticks](/dotnet/api/system.datetime.ticks).
|
||||
|
||||
Append `/metrics` to the URL to view the metrics endpoint. The browser displays the metrics being collected:
|
||||
|
||||
![metrics 2](~/log-mon/metrics/metrics/static/metrics.png)
|
||||
|
||||
### Set up and configure Prometheus
|
||||
|
||||
Follow the [Prometheus first steps](https://prometheus.io/docs/introduction/first_steps/) to set up a Prometheus server and confirm it's working.
|
||||
|
||||
Modify the *prometheus.yml* configuration file so that Prometheus scrapes the metrics endpoint that the example app is exposing. Add the following highlighted text in the `scrape_configs` section:
|
||||
|
||||
:::code language="yaml" source="~/log-mon/metrics/metrics/samples/prometheus.yml" highlight="31-99":::
|
||||
|
||||
In the preceding highlighted YAML, replace `5045` with the port number that the example app is running on.
|
||||
|
||||
#### Start Prometheus
|
||||
|
||||
1. Reload the configuration or restart the Prometheus server.
|
||||
1. Confirm that OpenTelemetryTest is in the UP state in the **Status** > **Targets** page of the Prometheus web portal.
|
||||
|
||||
![Prometheus status](~/log-mon/metrics/metrics/static/prometheus_status.png)
|
||||
|
||||
Select the **Open metric explorer** icon to see available metrics:
|
||||
|
||||
![Prometheus open_metric_exp](~/log-mon/metrics/metrics/static/open_metric_exp.png)
|
||||
|
||||
Enter counter category such as `http_` in the **Expression** input box to see the available metrics:
|
||||
|
||||
![available metrics](~/log-mon/metrics/metrics/static/metrics2.png)
|
||||
|
||||
Alternatively, enter counter category such as `kestrel` in the **Expression** input box to see the available metrics:
|
||||
|
||||
![Prometheus kestrel](~/log-mon/metrics/metrics/static/kestrel.png)
|
||||
|
||||
### Show metrics on a Grafana dashboard
|
||||
|
||||
1. Follow the [installation instructions](https://prometheus.io/docs/visualization/grafana/#creating-a-prometheus-graph) to install Grafana and connect it to a Prometheus data source.
|
||||
|
||||
1. Follow [Creating a Prometheus graph](https://prometheus.io/docs/visualization/grafana/#creating-a-prometheus-graph). Alternatively, download a JSON file from [aspnetcore-grafana dashboards](https://github.com/JamesNK/aspnetcore-grafana/tree/main/dashboards) to configure Grafana.
|
||||
|
||||
![dashboard-screenshot2](~/log-mon/metrics/metrics/static/dashboard-screenshot.png)
|
||||
|
||||
## ASP.NET Core meters and counters
|
||||
|
||||
* [HostingMetrics](https://source.dot.net/#Microsoft.AspNetCore.Hosting/Internal/HostingMetrics.cs,0e3eae9d5e50ff0d)
|
||||
* [KestrelMetrics](https://source.dot.net/#Microsoft.AspNetCore.Server.Kestrel.Core/Internal/Infrastructure/KestrelMetrics.cs,8d505625d99068a6)
|
||||
* SignalR [HttpConnectionsMetrics](https://source.dot.net/#Microsoft.AspNetCore.Http.Connections/Internal/HttpConnectionsMetrics.cs,2d63d88d7202e42d,references)
|
|
@ -0,0 +1,25 @@
|
|||
using OpenTelemetry.Metrics;
|
||||
|
||||
var builder = WebApplication.CreateBuilder(args);
|
||||
builder.Services.AddOpenTelemetry()
|
||||
.WithMetrics(builder =>
|
||||
{
|
||||
builder.AddPrometheusExporter();
|
||||
|
||||
builder.AddMeter("Microsoft.AspNetCore.Hosting",
|
||||
"Microsoft.AspNetCore.Server.Kestrel");
|
||||
builder.AddView("http-server-request-duration",
|
||||
new ExplicitBucketHistogramConfiguration
|
||||
{
|
||||
Boundaries = new double[] { 0, 0.005, 0.01, 0.025, 0.05,
|
||||
0.075, 0.1, 0.25, 0.5, 0.75, 1, 2.5, 5, 7.5, 10 }
|
||||
});
|
||||
});
|
||||
var app = builder.Build();
|
||||
|
||||
app.MapPrometheusScrapingEndpoint();
|
||||
|
||||
app.MapGet("/", () => "Hello OpenTelemetry! ticks:"
|
||||
+ DateTime.Now.Ticks.ToString()[^3..]);
|
||||
|
||||
app.Run();
|
|
@ -0,0 +1,14 @@
|
|||
<Project Sdk="Microsoft.NET.Sdk.Web">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net8.0</TargetFramework>
|
||||
<Nullable>enable</Nullable>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="OpenTelemetry.Exporter.Prometheus.AspNetCore" Version="1.5.0-rc.1" />
|
||||
<PackageReference Include="OpenTelemetry.Extensions.Hosting" Version="1.5.1" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
|
@ -0,0 +1,9 @@
|
|||
{
|
||||
"Logging": {
|
||||
"LogLevel": {
|
||||
"Default": "Information",
|
||||
"Microsoft.AspNetCore": "Warning"
|
||||
}
|
||||
},
|
||||
"AllowedHosts": "*"
|
||||
}
|
|
@ -0,0 +1,34 @@
|
|||
# my global config
|
||||
global:
|
||||
scrape_interval: 15s # Set the scrape interval to every 15 seconds. Default is every 1 minute.
|
||||
evaluation_interval: 15s # Evaluate rules every 15 seconds. The default is every 1 minute.
|
||||
# scrape_timeout is set to the global default (10s).
|
||||
|
||||
# Alertmanager configuration
|
||||
alerting:
|
||||
alertmanagers:
|
||||
- static_configs:
|
||||
- targets:
|
||||
# - alertmanager:9093
|
||||
|
||||
# Load rules once and periodically evaluate them according to the global 'evaluation_interval'.
|
||||
rule_files:
|
||||
# - "first_rules.yml"
|
||||
# - "second_rules.yml"
|
||||
|
||||
# A scrape configuration containing exactly one endpoint to scrape:
|
||||
# Here it's Prometheus itself.
|
||||
scrape_configs:
|
||||
# The job name is added as a label `job=<job_name>` to any timeseries scraped from this config.
|
||||
- job_name: "prometheus"
|
||||
|
||||
# metrics_path defaults to '/metrics'
|
||||
# scheme defaults to 'http'.
|
||||
|
||||
static_configs:
|
||||
- targets: ["localhost:9090"]
|
||||
|
||||
- job_name: 'MyASPNETApp'
|
||||
scrape_interval: 5s # Poll every 5 seconds for a more responsive demo.
|
||||
static_configs:
|
||||
- targets: ["localhost:5045"] ## Enter the HTTP port number of the demo app.
|
Binary file not shown.
After Width: | Height: | Size: 438 KiB |
Binary file not shown.
After Width: | Height: | Size: 119 KiB |
Binary file not shown.
After Width: | Height: | Size: 108 KiB |
Binary file not shown.
After Width: | Height: | Size: 64 KiB |
Binary file not shown.
After Width: | Height: | Size: 59 KiB |
Binary file not shown.
After Width: | Height: | Size: 115 KiB |
Loading…
Reference in New Issue