#nullable disable #pragma warning disable CS1591 using System; using System.Collections.Generic; using System.Linq; using System.Text.Json.Serialization; using System.Threading; using System.Threading.Tasks; using Jellyfin.Data.Enums; using MediaBrowser.Controller.Entities; using MediaBrowser.Model.Dto; using MediaBrowser.Model.Session; using Microsoft.Extensions.Logging; namespace MediaBrowser.Controller.Session { /// /// Class SessionInfo. /// public sealed class SessionInfo : IAsyncDisposable, IDisposable { // 1 second private const long ProgressIncrement = 10000000; private readonly ISessionManager _sessionManager; private readonly ILogger _logger; private readonly object _progressLock = new object(); private Timer _progressTimer; private PlaybackProgressInfo _lastProgressInfo; private bool _disposed = false; public SessionInfo(ISessionManager sessionManager, ILogger logger) { _sessionManager = sessionManager; _logger = logger; AdditionalUsers = Array.Empty(); PlayState = new PlayerStateInfo(); SessionControllers = Array.Empty(); NowPlayingQueue = Array.Empty(); NowPlayingQueueFullItems = Array.Empty(); } public PlayerStateInfo PlayState { get; set; } public SessionUserInfo[] AdditionalUsers { get; set; } public ClientCapabilities Capabilities { get; set; } /// /// Gets or sets the remote end point. /// /// The remote end point. public string RemoteEndPoint { get; set; } /// /// Gets the playable media types. /// /// The playable media types. public IReadOnlyList PlayableMediaTypes { get { if (Capabilities is null) { return Array.Empty(); } return Capabilities.PlayableMediaTypes; } } /// /// Gets or sets the id. /// /// The id. public string Id { get; set; } /// /// Gets or sets the user id. /// /// The user id. public Guid UserId { get; set; } /// /// Gets or sets the username. /// /// The username. public string UserName { get; set; } /// /// Gets or sets the type of the client. /// /// The type of the client. public string Client { get; set; } /// /// Gets or sets the last activity date. /// /// The last activity date. public DateTime LastActivityDate { get; set; } /// /// Gets or sets the last playback check in. /// /// The last playback check in. public DateTime LastPlaybackCheckIn { get; set; } /// /// Gets or sets the name of the device. /// /// The name of the device. public string DeviceName { get; set; } /// /// Gets or sets the type of the device. /// /// The type of the device. public string DeviceType { get; set; } /// /// Gets or sets the now playing item. /// /// The now playing item. public BaseItemDto NowPlayingItem { get; set; } public BaseItem FullNowPlayingItem { get; set; } public BaseItemDto NowViewingItem { get; set; } /// /// Gets or sets the device id. /// /// The device id. public string DeviceId { get; set; } /// /// Gets or sets the application version. /// /// The application version. public string ApplicationVersion { get; set; } /// /// Gets or sets the session controller. /// /// The session controller. [JsonIgnore] public ISessionController[] SessionControllers { get; set; } public TranscodingInfo TranscodingInfo { get; set; } /// /// Gets a value indicating whether this instance is active. /// /// true if this instance is active; otherwise, false. public bool IsActive { get { var controllers = SessionControllers; foreach (var controller in controllers) { if (controller.IsSessionActive) { return true; } } if (controllers.Length > 0) { return false; } return true; } } public bool SupportsMediaControl { get { if (Capabilities is null || !Capabilities.SupportsMediaControl) { return false; } var controllers = SessionControllers; foreach (var controller in controllers) { if (controller.SupportsMediaControl) { return true; } } return false; } } public bool SupportsRemoteControl { get { if (Capabilities is null || !Capabilities.SupportsMediaControl) { return false; } var controllers = SessionControllers; foreach (var controller in controllers) { if (controller.SupportsMediaControl) { return true; } } return false; } } public IReadOnlyList NowPlayingQueue { get; set; } public IReadOnlyList NowPlayingQueueFullItems { get; set; } public bool HasCustomDeviceName { get; set; } public string PlaylistItemId { get; set; } public string ServerId { get; set; } public string UserPrimaryImageTag { get; set; } /// /// Gets the supported commands. /// /// The supported commands. public IReadOnlyList SupportedCommands => Capabilities is null ? Array.Empty() : Capabilities.SupportedCommands; public Tuple EnsureController(Func factory) { var controllers = SessionControllers.ToList(); foreach (var controller in controllers) { if (controller is T) { return new Tuple(controller, false); } } var newController = factory(this); _logger.LogDebug("Creating new {0}", newController.GetType().Name); controllers.Add(newController); SessionControllers = controllers.ToArray(); return new Tuple(newController, true); } public void AddController(ISessionController controller) { var controllers = SessionControllers.ToList(); controllers.Add(controller); SessionControllers = controllers.ToArray(); } public bool ContainsUser(Guid userId) { if (UserId.Equals(userId)) { return true; } foreach (var additionalUser in AdditionalUsers) { if (additionalUser.UserId.Equals(userId)) { return true; } } return false; } public void StartAutomaticProgress(PlaybackProgressInfo progressInfo) { if (_disposed) { return; } lock (_progressLock) { _lastProgressInfo = progressInfo; if (_progressTimer is null) { _progressTimer = new Timer(OnProgressTimerCallback, null, 1000, 1000); } else { _progressTimer.Change(1000, 1000); } } } private async void OnProgressTimerCallback(object state) { if (_disposed) { return; } var progressInfo = _lastProgressInfo; if (progressInfo is null) { return; } if (progressInfo.IsPaused) { return; } var positionTicks = progressInfo.PositionTicks ?? 0; if (positionTicks < 0) { positionTicks = 0; } var newPositionTicks = positionTicks + ProgressIncrement; var item = progressInfo.Item; long? runtimeTicks = item?.RunTimeTicks; // Don't report beyond the runtime if (runtimeTicks.HasValue && newPositionTicks >= runtimeTicks.Value) { return; } progressInfo.PositionTicks = newPositionTicks; try { await _sessionManager.OnPlaybackProgress(progressInfo, true).ConfigureAwait(false); } catch (Exception ex) { _logger.LogError(ex, "Error reporting playback progress"); } } public void StopAutomaticProgress() { lock (_progressLock) { if (_progressTimer is not null) { _progressTimer.Dispose(); _progressTimer = null; } _lastProgressInfo = null; } } /// public void Dispose() { _disposed = true; StopAutomaticProgress(); var controllers = SessionControllers.ToList(); SessionControllers = Array.Empty(); foreach (var controller in controllers) { if (controller is IDisposable disposable) { _logger.LogDebug("Disposing session controller synchronously {TypeName}", disposable.GetType().Name); disposable.Dispose(); } } } public async ValueTask DisposeAsync() { _disposed = true; StopAutomaticProgress(); var controllers = SessionControllers.ToList(); foreach (var controller in controllers) { if (controller is IAsyncDisposable disposableAsync) { _logger.LogDebug("Disposing session controller asynchronously {TypeName}", disposableAsync.GetType().Name); await disposableAsync.DisposeAsync().ConfigureAwait(false); } } } } }