jellyfin/Emby.Server.Implementations/HttpServer/WebSocketManager.cs

99 lines
3.5 KiB
C#
Raw Permalink Normal View History

#nullable disable
2020-09-03 11:32:22 +02:00
#pragma warning disable CS1591
using System;
using System.Collections.Generic;
using System.Linq;
2020-09-03 11:32:22 +02:00
using System.Net.WebSockets;
using System.Threading.Tasks;
using MediaBrowser.Common.Extensions;
2020-09-03 11:32:22 +02:00
using MediaBrowser.Controller.Net;
using Microsoft.AspNetCore.Http;
using Microsoft.Extensions.Logging;
namespace Emby.Server.Implementations.HttpServer
{
public class WebSocketManager : IWebSocketManager
{
private readonly IWebSocketListener[] _webSocketListeners;
private readonly IAuthService _authService;
2020-09-03 11:32:22 +02:00
private readonly ILogger<WebSocketManager> _logger;
private readonly ILoggerFactory _loggerFactory;
2020-11-28 09:50:16 +01:00
2020-09-03 11:32:22 +02:00
public WebSocketManager(
IAuthService authService,
IEnumerable<IWebSocketListener> webSocketListeners,
2020-09-03 11:32:22 +02:00
ILogger<WebSocketManager> logger,
ILoggerFactory loggerFactory)
{
_webSocketListeners = webSocketListeners.ToArray();
_authService = authService;
2020-09-03 11:32:22 +02:00
_logger = logger;
_loggerFactory = loggerFactory;
}
/// <inheritdoc />
public async Task WebSocketRequestHandler(HttpContext context)
{
var authorizationInfo = await _authService.Authenticate(context.Request).ConfigureAwait(false);
if (!authorizationInfo.IsAuthenticated)
{
throw new SecurityException("Token is required");
}
2020-09-03 11:32:22 +02:00
try
{
_logger.LogInformation("WS {IP} request", context.Connection.RemoteIpAddress);
WebSocket webSocket = await context.WebSockets.AcceptWebSocketAsync().ConfigureAwait(false);
var connection = new WebSocketConnection(
2020-09-03 11:32:22 +02:00
_loggerFactory.CreateLogger<WebSocketConnection>(),
webSocket,
authorizationInfo,
2023-02-17 19:27:36 +01:00
context.GetNormalizedRemoteIP())
2020-09-03 11:32:22 +02:00
{
OnReceive = ProcessWebSocketMessageReceived
};
await using (connection.ConfigureAwait(false))
{
var tasks = new Task[_webSocketListeners.Length];
for (var i = 0; i < _webSocketListeners.Length; ++i)
{
tasks[i] = _webSocketListeners[i].ProcessWebSocketConnectedAsync(connection, context);
}
await Task.WhenAll(tasks).ConfigureAwait(false);
2020-09-03 11:32:22 +02:00
await connection.ReceiveAsync().ConfigureAwait(false);
_logger.LogInformation("WS {IP} closed", context.Connection.RemoteIpAddress);
}
2020-09-03 11:32:22 +02:00
}
catch (Exception ex) // Otherwise ASP.Net will ignore the exception
{
_logger.LogError(ex, "WS {IP} WebSocketRequestHandler error", context.Connection.RemoteIpAddress);
if (!context.Response.HasStarted)
{
context.Response.StatusCode = 500;
}
}
}
/// <summary>
/// Processes the web socket message received.
/// </summary>
/// <param name="result">The result.</param>
private Task ProcessWebSocketMessageReceived(WebSocketMessageInfo result)
{
var tasks = new Task[_webSocketListeners.Length];
for (var i = 0; i < _webSocketListeners.Length; ++i)
2020-11-28 09:50:16 +01:00
{
tasks[i] = _webSocketListeners[i].ProcessMessageAsync(result);
2020-11-28 09:50:16 +01:00
}
2020-09-03 11:32:22 +02:00
return Task.WhenAll(tasks);
2020-09-03 11:32:22 +02:00
}
}
}