forked from dotnet/runtime
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathNameResolutionTelemetry.cs
More file actions
135 lines (107 loc) · 4.72 KB
/
NameResolutionTelemetry.cs
File metadata and controls
135 lines (107 loc) · 4.72 KB
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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
using System.Collections.Generic;
using System.Diagnostics;
using System.Diagnostics.Tracing;
using System.Net.Sockets;
using System.Threading;
namespace System.Net
{
[EventSource(Name = "System.Net.NameResolution")]
internal sealed class NameResolutionTelemetry : EventSource
{
public static readonly NameResolutionTelemetry Log = new NameResolutionTelemetry();
private const int ResolutionStartEventId = 1;
private const int ResolutionStopEventId = 2;
private const int ResolutionFailedEventId = 3;
private PollingCounter? _lookupsRequestedCounter;
private PollingCounter? _currentLookupsCounter;
private EventCounter? _lookupsDuration;
private long _lookupsRequested;
private long _currentLookups;
protected override void OnEventCommand(EventCommandEventArgs command)
{
if (command.Command == EventCommand.Enable)
{
// The cumulative number of name resolution requests started since events were enabled
_lookupsRequestedCounter ??= new PollingCounter("dns-lookups-requested", this, () => Interlocked.Read(ref _lookupsRequested))
{
DisplayName = "DNS Lookups Requested"
};
// Current number of DNS requests pending
_currentLookupsCounter ??= new PollingCounter("current-dns-lookups", this, () => Interlocked.Read(ref _currentLookups))
{
DisplayName = "Current DNS Lookups"
};
_lookupsDuration ??= new EventCounter("dns-lookups-duration", this)
{
DisplayName = "Average DNS Lookup Duration",
DisplayUnits = "ms"
};
}
}
[Event(ResolutionStartEventId, Level = EventLevel.Informational)]
private void ResolutionStart(string hostNameOrAddress) => WriteEvent(ResolutionStartEventId, hostNameOrAddress);
[Event(ResolutionStopEventId, Level = EventLevel.Informational)]
private void ResolutionStop() => WriteEvent(ResolutionStopEventId);
[Event(ResolutionFailedEventId, Level = EventLevel.Informational)]
private void ResolutionFailed() => WriteEvent(ResolutionFailedEventId);
[NonEvent]
public long BeforeResolution(object hostNameOrAddress)
{
if (IsEnabled())
{
Interlocked.Increment(ref _lookupsRequested);
Interlocked.Increment(ref _currentLookups);
if (IsEnabled(EventLevel.Informational, EventKeywords.None))
{
string host = GetHostnameFromStateObject(hostNameOrAddress);
ResolutionStart(host);
}
return Stopwatch.GetTimestamp();
}
return NameResolutionMetrics.IsEnabled() ? Stopwatch.GetTimestamp() : 0;
}
[NonEvent]
public void AfterResolution(object hostNameOrAddress, long? startingTimestamp, Exception? exception = null)
{
Debug.Assert(startingTimestamp.HasValue);
if (startingTimestamp == 0)
{
return;
}
TimeSpan duration = Stopwatch.GetElapsedTime(startingTimestamp.Value);
if (IsEnabled())
{
Interlocked.Decrement(ref _currentLookups);
_lookupsDuration?.WriteMetric(duration.TotalMilliseconds);
if (IsEnabled(EventLevel.Informational, EventKeywords.None))
{
if (exception is not null)
{
ResolutionFailed();
}
ResolutionStop();
}
}
if (NameResolutionMetrics.IsEnabled())
{
NameResolutionMetrics.AfterResolution(duration, GetHostnameFromStateObject(hostNameOrAddress), exception);
}
}
private static string GetHostnameFromStateObject(object hostNameOrAddress)
{
Debug.Assert(hostNameOrAddress is not null);
string host = hostNameOrAddress switch
{
string h => h,
KeyValuePair<string, AddressFamily> t => t.Key,
IPAddress a => a.ToString(),
KeyValuePair<IPAddress, AddressFamily> t => t.Key.ToString(),
_ => null!
};
Debug.Assert(host is not null, $"Unknown hostNameOrAddress type: {hostNameOrAddress.GetType().Name}");
return host;
}
}
}