-
Notifications
You must be signed in to change notification settings - Fork 869
/
Copy pathWebSocketsTelemetryMiddleware.cs
87 lines (75 loc) · 3.19 KB
/
WebSocketsTelemetryMiddleware.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
80
81
82
83
84
85
86
87
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
using System;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Http.Features;
namespace Yarp.ReverseProxy.WebSocketsTelemetry;
internal sealed class WebSocketsTelemetryMiddleware
{
private readonly RequestDelegate _next;
private readonly TimeProvider _timeProvider;
public WebSocketsTelemetryMiddleware(RequestDelegate next, TimeProvider timeProvider)
{
_next = next ?? throw new ArgumentNullException(nameof(next));
_timeProvider = timeProvider ?? throw new ArgumentNullException(nameof(timeProvider));
}
public Task InvokeAsync(HttpContext context)
{
if (WebSocketsTelemetry.Log.IsEnabled())
{
if (context.Features.Get<IHttpUpgradeFeature>() is { IsUpgradableRequest: true } upgradeFeature)
{
var upgradeWrapper = new HttpUpgradeFeatureWrapper(_timeProvider, context, upgradeFeature);
return InvokeAsyncCore(upgradeWrapper, _next);
}
else if (context.Features.Get<IHttpExtendedConnectFeature>() is { IsExtendedConnect: true } connectFeature
&& string.Equals("websocket", connectFeature.Protocol, StringComparison.OrdinalIgnoreCase))
{
var connectWrapper = new HttpConnectFeatureWrapper(_timeProvider, context, connectFeature);
return InvokeAsyncCore(connectWrapper, _next);
}
}
return _next(context);
}
private static async Task InvokeAsyncCore(HttpUpgradeFeatureWrapper upgradeWrapper, RequestDelegate next)
{
upgradeWrapper.HttpContext.Features.Set<IHttpUpgradeFeature>(upgradeWrapper);
try
{
await next(upgradeWrapper.HttpContext);
}
finally
{
if (upgradeWrapper.TelemetryStream is { } telemetryStream)
{
WebSocketsTelemetry.Log.WebSocketClosed(
telemetryStream.EstablishedTime.Ticks,
telemetryStream.GetCloseReason(upgradeWrapper.HttpContext),
telemetryStream.MessagesRead,
telemetryStream.MessagesWritten);
}
upgradeWrapper.HttpContext.Features.Set(upgradeWrapper.InnerUpgradeFeature);
}
}
private static async Task InvokeAsyncCore(HttpConnectFeatureWrapper connectWrapper, RequestDelegate next)
{
connectWrapper.HttpContext.Features.Set<IHttpExtendedConnectFeature>(connectWrapper);
try
{
await next(connectWrapper.HttpContext);
}
finally
{
if (connectWrapper.TelemetryStream is { } telemetryStream)
{
WebSocketsTelemetry.Log.WebSocketClosed(
telemetryStream.EstablishedTime.Ticks,
telemetryStream.GetCloseReason(connectWrapper.HttpContext),
telemetryStream.MessagesRead,
telemetryStream.MessagesWritten);
}
connectWrapper.HttpContext.Features.Set(connectWrapper.InnerConnectFeature);
}
}
}