123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186 |
- #pragma warning disable CS1591
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Net.WebSockets;
- using System.Threading;
- using System.Threading.Tasks;
- using MediaBrowser.Controller.Net;
- using MediaBrowser.Controller.Net.WebSocketMessages;
- using MediaBrowser.Controller.Session;
- using MediaBrowser.Model.Session;
- using Microsoft.Extensions.Logging;
- namespace Emby.Server.Implementations.Session
- {
- public sealed class WebSocketController : ISessionController, IAsyncDisposable, IDisposable
- {
- private readonly ILogger<WebSocketController> _logger;
- private readonly ISessionManager _sessionManager;
- private readonly SessionInfo _session;
- private readonly List<IWebSocketConnection> _sockets;
- private readonly ReaderWriterLockSlim _socketsLock;
- private bool _disposed = false;
- public WebSocketController(
- ILogger<WebSocketController> logger,
- SessionInfo session,
- ISessionManager sessionManager)
- {
- _logger = logger;
- _session = session;
- _sessionManager = sessionManager;
- _sockets = new();
- _socketsLock = new();
- }
- private bool HasOpenSockets
- {
- get
- {
- ObjectDisposedException.ThrowIf(_disposed, this);
- try
- {
- _socketsLock.EnterReadLock();
- return _sockets.Any(i => i.State == WebSocketState.Open);
- }
- finally
- {
- _socketsLock.ExitReadLock();
- }
- }
- }
- /// <inheritdoc />
- public bool SupportsMediaControl => HasOpenSockets;
- /// <inheritdoc />
- public bool IsSessionActive => HasOpenSockets;
- public void AddWebSocket(IWebSocketConnection connection)
- {
- _logger.LogDebug("Adding websocket to session {Session}", _session.Id);
- ObjectDisposedException.ThrowIf(_disposed, this);
- try
- {
- _socketsLock.EnterWriteLock();
- _sockets.Add(connection);
- connection.Closed += OnConnectionClosed;
- }
- finally
- {
- _socketsLock.ExitWriteLock();
- }
- }
- private async void OnConnectionClosed(object? sender, EventArgs e)
- {
- var connection = sender as IWebSocketConnection ?? throw new ArgumentException($"{nameof(sender)} is not of type {nameof(IWebSocketConnection)}", nameof(sender));
- _logger.LogDebug("Removing websocket from session {Session}", _session.Id);
- ObjectDisposedException.ThrowIf(_disposed, this);
- try
- {
- _socketsLock.EnterWriteLock();
- _sockets.Remove(connection);
- connection.Closed -= OnConnectionClosed;
- }
- finally
- {
- _socketsLock.ExitWriteLock();
- }
- await _sessionManager.CloseIfNeededAsync(_session).ConfigureAwait(false);
- }
- /// <inheritdoc />
- public Task SendMessage<T>(
- SessionMessageType name,
- Guid messageId,
- T data,
- CancellationToken cancellationToken)
- {
- ObjectDisposedException.ThrowIf(_disposed, this);
- IWebSocketConnection? socket;
- try
- {
- _socketsLock.EnterReadLock();
- socket = _sockets.Where(i => i.State == WebSocketState.Open).MaxBy(i => i.LastActivityDate);
- }
- finally
- {
- _socketsLock.ExitReadLock();
- }
- if (socket is null)
- {
- return Task.CompletedTask;
- }
- return socket.SendAsync(
- new OutboundWebSocketMessage<T>
- {
- Data = data,
- MessageType = name,
- MessageId = messageId
- },
- cancellationToken);
- }
- /// <inheritdoc />
- public void Dispose()
- {
- if (_disposed)
- {
- return;
- }
- try
- {
- _socketsLock.EnterWriteLock();
- foreach (var socket in _sockets)
- {
- socket.Closed -= OnConnectionClosed;
- socket.Dispose();
- }
- _sockets.Clear();
- }
- finally
- {
- _socketsLock.ExitWriteLock();
- }
- _socketsLock.Dispose();
- _disposed = true;
- }
- public async ValueTask DisposeAsync()
- {
- if (_disposed)
- {
- return;
- }
- try
- {
- _socketsLock.EnterWriteLock();
- foreach (var socket in _sockets)
- {
- socket.Closed -= OnConnectionClosed;
- await socket.DisposeAsync().ConfigureAwait(false);
- }
- _sockets.Clear();
- }
- finally
- {
- _socketsLock.ExitWriteLock();
- }
- _socketsLock.Dispose();
- _disposed = true;
- }
- }
- }
|