-
Notifications
You must be signed in to change notification settings - Fork 300
/
Copy pathPrometheusHandler.cs
79 lines (71 loc) · 3.42 KB
/
PrometheusHandler.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
using System.Diagnostics;
using System.Diagnostics.Metrics;
using System.Net.Http;
namespace k8s
{
/// <summary>
/// Implements legacy Prometheus metrics
/// </summary>
/// <remarks>Provided for compatibility for existing usages of PrometheusHandler. It is recommended
/// to transition to using OpenTelemetry and the default HttpClient metrics.
///
/// Note that the tags/labels are not appropriately named for some metrics. This
/// incorrect naming is retained to maintain compatibility and won't be fixed on this implementation.
/// Use OpenTelemetry and the standard HttpClient metrics instead.</remarks>
public class PrometheusHandler : DelegatingHandler
{
private const string Prefix = "k8s_dotnet";
private static readonly Meter Meter = new Meter("k8s.dotnet");
private static readonly Counter<int> RequestsSent = Meter.CreateCounter<int>(
$"{Prefix}_request_total",
description: "Number of requests sent by this client");
private static readonly Histogram<double> RequestLatency = Meter.CreateHistogram<double>(
$"{Prefix}_request_latency_seconds", unit: "milliseconds",
description: "Latency of requests sent by this client");
private static readonly Counter<int> ResponseCodes = Meter.CreateCounter<int>(
$"{Prefix}_response_code_total",
description: "Number of response codes received by the client");
private static readonly UpDownCounter<int> ActiveRequests =
Meter.CreateUpDownCounter<int>(
$"{Prefix}_active_requests",
description: "Number of requests currently in progress");
/// <inheritdoc />
protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
{
if (request == null)
{
throw new ArgumentNullException(nameof(request));
}
var digest = KubernetesRequestDigest.Parse(request);
var timer = Stopwatch.StartNew();
// Note that this is a tag called "method" but the value is the Verb.
// This is incorrect, but existing behavior.
var methodWithVerbValue = new KeyValuePair<string, object>("method", digest.Verb);
try
{
ActiveRequests.Add(1, methodWithVerbValue);
RequestsSent.Add(1, methodWithVerbValue);
var resp = await base.SendAsync(request, cancellationToken).ConfigureAwait(false);
ResponseCodes.Add(
1,
new KeyValuePair<string, object>("method", request.Method.ToString()),
new KeyValuePair<string, object>("code", (int)resp.StatusCode));
return resp;
}
finally
{
timer.Stop();
ActiveRequests.Add(-1, methodWithVerbValue);
var tags = new TagList
{
{ "verb", digest.Verb },
{ "group", digest.ApiGroup },
{ "version", digest.ApiVersion },
{ "kind", digest.Kind },
}
;
RequestLatency.Record(timer.Elapsed.TotalMilliseconds, tags);
}
}
}
}