| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408 | #nullable disable#pragma warning disable CS1591using System;using System.Collections.Generic;using System.Linq;using System.Text.Json.Serialization;using System.Threading;using System.Threading.Tasks;using MediaBrowser.Controller.Entities;using MediaBrowser.Model.Dto;using MediaBrowser.Model.Session;using Microsoft.Extensions.Logging;namespace MediaBrowser.Controller.Session{    /// <summary>    /// Class SessionInfo.    /// </summary>    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<SessionUserInfo>();            PlayState = new PlayerStateInfo();            SessionControllers = Array.Empty<ISessionController>();            NowPlayingQueue = Array.Empty<QueueItem>();            NowPlayingQueueFullItems = Array.Empty<BaseItemDto>();        }        public PlayerStateInfo PlayState { get; set; }        public SessionUserInfo[] AdditionalUsers { get; set; }        public ClientCapabilities Capabilities { get; set; }        /// <summary>        /// Gets or sets the remote end point.        /// </summary>        /// <value>The remote end point.</value>        public string RemoteEndPoint { get; set; }        /// <summary>        /// Gets the playable media types.        /// </summary>        /// <value>The playable media types.</value>        public IReadOnlyList<string> PlayableMediaTypes        {            get            {                if (Capabilities == null)                {                    return Array.Empty<string>();                }                return Capabilities.PlayableMediaTypes;            }        }        /// <summary>        /// Gets or sets the id.        /// </summary>        /// <value>The id.</value>        public string Id { get; set; }        /// <summary>        /// Gets or sets the user id.        /// </summary>        /// <value>The user id.</value>        public Guid UserId { get; set; }        /// <summary>        /// Gets or sets the username.        /// </summary>        /// <value>The username.</value>        public string UserName { get; set; }        /// <summary>        /// Gets or sets the type of the client.        /// </summary>        /// <value>The type of the client.</value>        public string Client { get; set; }        /// <summary>        /// Gets or sets the last activity date.        /// </summary>        /// <value>The last activity date.</value>        public DateTime LastActivityDate { get; set; }        /// <summary>        /// Gets or sets the last playback check in.        /// </summary>        /// <value>The last playback check in.</value>        public DateTime LastPlaybackCheckIn { get; set; }        /// <summary>        /// Gets or sets the name of the device.        /// </summary>        /// <value>The name of the device.</value>        public string DeviceName { get; set; }        /// <summary>        /// Gets or sets the type of the device.        /// </summary>        /// <value>The type of the device.</value>        public string DeviceType { get; set; }        /// <summary>        /// Gets or sets the now playing item.        /// </summary>        /// <value>The now playing item.</value>        public BaseItemDto NowPlayingItem { get; set; }        public BaseItem FullNowPlayingItem { get; set; }        public BaseItemDto NowViewingItem { get; set; }        /// <summary>        /// Gets or sets the device id.        /// </summary>        /// <value>The device id.</value>        public string DeviceId { get; set; }        /// <summary>        /// Gets or sets the application version.        /// </summary>        /// <value>The application version.</value>        public string ApplicationVersion { get; set; }        /// <summary>        /// Gets or sets the session controller.        /// </summary>        /// <value>The session controller.</value>        [JsonIgnore]        public ISessionController[] SessionControllers { get; set; }        public TranscodingInfo TranscodingInfo { get; set; }        /// <summary>        /// Gets a value indicating whether this instance is active.        /// </summary>        /// <value><c>true</c> if this instance is active; otherwise, <c>false</c>.</value>        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 == 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 == null || !Capabilities.SupportsMediaControl)                {                    return false;                }                var controllers = SessionControllers;                foreach (var controller in controllers)                {                    if (controller.SupportsMediaControl)                    {                        return true;                    }                }                return false;            }        }        public IReadOnlyList<QueueItem> NowPlayingQueue { get; set; }        public IReadOnlyList<BaseItemDto> NowPlayingQueueFullItems { get; set; }        public bool HasCustomDeviceName { get; set; }        public string PlaylistItemId { get; set; }        public string ServerId { get; set; }        public string UserPrimaryImageTag { get; set; }        /// <summary>        /// Gets the supported commands.        /// </summary>        /// <value>The supported commands.</value>        public IReadOnlyList<GeneralCommandType> SupportedCommands            => Capabilities == null ? Array.Empty<GeneralCommandType>() : Capabilities.SupportedCommands;        public Tuple<ISessionController, bool> EnsureController<T>(Func<SessionInfo, ISessionController> factory)        {            var controllers = SessionControllers.ToList();            foreach (var controller in controllers)            {                if (controller is T)                {                    return new Tuple<ISessionController, bool>(controller, false);                }            }            var newController = factory(this);            _logger.LogDebug("Creating new {0}", newController.GetType().Name);            controllers.Add(newController);            SessionControllers = controllers.ToArray();            return new Tuple<ISessionController, bool>(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 == 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 == 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 != null)                {                    _progressTimer.Dispose();                    _progressTimer = null;                }                _lastProgressInfo = null;            }        }        /// <inheritdoc />        public void Dispose()        {            _disposed = true;            StopAutomaticProgress();            var controllers = SessionControllers.ToList();            SessionControllers = Array.Empty<ISessionController>();            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);                }            }        }    }}
 |