< Summary - Jellyfin

Information
Class: Emby.Server.Implementations.Session.SessionManager
Assembly: Emby.Server.Implementations
File(s): /srv/git/jellyfin/Emby.Server.Implementations/Session/SessionManager.cs
Line coverage
22%
Covered lines: 113
Uncovered lines: 391
Coverable lines: 504
Total lines: 2144
Line coverage: 22.4%
Branch coverage
12%
Covered branches: 23
Total branches: 178
Branch coverage: 12.9%
Method coverage

Feature is only available for sponsors

Upgrade to PRO version

Coverage history

Coverage history 0 25 50 75 100

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity Line coverage
.ctor(...)100%11100%
get_Sessions()100%11100%
OnDeviceManagerDeviceOptionsUpdated(...)0%4260%
CheckDisposed()100%11100%
OnSessionStarted(...)100%44100%
UpdateDeviceName(...)0%620%
OnSessionControllerConnected(...)100%210%
GetMediaSource(...)100%210%
RemoveNowPlayingItem(...)0%620%
GetSessionKey(...)100%11100%
GetSessionInfo(...)62.5%161694.44%
CreateSession(...)57.14%141488%
GetUsers(...)0%2040%
StartCheckTimers()0%4260%
StopIdleCheckTimer()0%620%
StopInactiveCheckTimer()0%620%
GetNowPlayingItem(...)0%4260%
OnPlaybackStart(...)0%2040%
OnPlaybackProgress(...)100%210%
UpdateLiveStreamActiveSessionMappings(...)0%110100%
OnPlaybackProgress(...)0%4260%
UpdatePlaybackSettings(...)0%156120%
OnPlaybackStopped(...)0%2040%
GetSession(...)0%620%
GetSessionToRemoteControl(...)0%620%
ToSessionInfoDto(...)100%11100%
SendMessageCommand(...)0%620%
SendGeneralCommand(...)0%620%
SendMessageToSessions(...)100%11100%
TranslateItemForPlayback(...)0%4260%
TranslateItemForInstantMix(...)0%620%
SendBrowseCommand(...)100%210%
SendPlaystateCommand(...)0%2040%
AssertCanControl(...)100%210%
SendRestartRequiredNotification(...)100%210%
AddAdditionalUser(...)0%4260%
RemoveAdditionalUser(...)0%2040%
AuthenticateNewSession(...)100%11100%
AuthenticateDirect(...)100%210%
ReportCapabilities(...)100%210%
ReportCapabilities(...)25%9430%
GetItemInfo(...)0%2040%
GetImageCacheTag(...)100%210%
ReportNowViewingItem(...)100%210%
ReportTranscodingInfo(...)0%620%
ClearTranscodingInfo(...)100%210%
GetSession(...)100%210%
GetSessionByAuthenticationToken(...)0%7280%
GetSessions(...)0%600240%
SendMessageToAdminSessions(...)0%2040%
SendMessageToUserSessions(...)0%620%
SendMessageToUserSessions(...)100%210%
SendMessageToUserDeviceSessions(...)100%210%

File(s)

/srv/git/jellyfin/Emby.Server.Implementations/Session/SessionManager.cs

#LineLine coverage
 1#nullable disable
 2
 3using System;
 4using System.Collections.Concurrent;
 5using System.Collections.Generic;
 6using System.Globalization;
 7using System.Linq;
 8using System.Threading;
 9using System.Threading.Tasks;
 10using Jellyfin.Data;
 11using Jellyfin.Data.Enums;
 12using Jellyfin.Data.Events;
 13using Jellyfin.Data.Queries;
 14using Jellyfin.Database.Implementations.Entities;
 15using Jellyfin.Database.Implementations.Entities.Security;
 16using Jellyfin.Database.Implementations.Enums;
 17using Jellyfin.Extensions;
 18using MediaBrowser.Common.Events;
 19using MediaBrowser.Common.Extensions;
 20using MediaBrowser.Controller;
 21using MediaBrowser.Controller.Authentication;
 22using MediaBrowser.Controller.Configuration;
 23using MediaBrowser.Controller.Devices;
 24using MediaBrowser.Controller.Drawing;
 25using MediaBrowser.Controller.Dto;
 26using MediaBrowser.Controller.Entities;
 27using MediaBrowser.Controller.Events;
 28using MediaBrowser.Controller.Events.Authentication;
 29using MediaBrowser.Controller.Events.Session;
 30using MediaBrowser.Controller.Library;
 31using MediaBrowser.Controller.Net;
 32using MediaBrowser.Controller.Session;
 33using MediaBrowser.Model.Dto;
 34using MediaBrowser.Model.Entities;
 35using MediaBrowser.Model.Library;
 36using MediaBrowser.Model.Querying;
 37using MediaBrowser.Model.Session;
 38using MediaBrowser.Model.SyncPlay;
 39using Microsoft.EntityFrameworkCore;
 40using Microsoft.Extensions.Hosting;
 41using Microsoft.Extensions.Logging;
 42using Episode = MediaBrowser.Controller.Entities.TV.Episode;
 43
 44namespace Emby.Server.Implementations.Session
 45{
 46    /// <summary>
 47    /// Class SessionManager.
 48    /// </summary>
 49    public sealed class SessionManager : ISessionManager, IAsyncDisposable
 50    {
 51        private readonly IUserDataManager _userDataManager;
 52        private readonly IServerConfigurationManager _config;
 53        private readonly ILogger<SessionManager> _logger;
 54        private readonly IEventManager _eventManager;
 55        private readonly ILibraryManager _libraryManager;
 56        private readonly IUserManager _userManager;
 57        private readonly IMusicManager _musicManager;
 58        private readonly IDtoService _dtoService;
 59        private readonly IImageProcessor _imageProcessor;
 60        private readonly IMediaSourceManager _mediaSourceManager;
 61        private readonly IServerApplicationHost _appHost;
 62        private readonly IDeviceManager _deviceManager;
 63        private readonly CancellationTokenRegistration _shutdownCallback;
 3164        private readonly ConcurrentDictionary<string, SessionInfo> _activeConnections
 3165            = new(StringComparer.OrdinalIgnoreCase);
 66
 3167        private readonly ConcurrentDictionary<string, ConcurrentDictionary<string, string>> _activeLiveStreamSessions
 3168            = new(StringComparer.OrdinalIgnoreCase);
 69
 70        private Timer _idleTimer;
 71        private Timer _inactiveTimer;
 72
 73        private DtoOptions _itemInfoDtoOptions;
 74        private bool _disposed;
 75
 76        /// <summary>
 77        /// Initializes a new instance of the <see cref="SessionManager"/> class.
 78        /// </summary>
 79        /// <param name="logger">Instance of <see cref="ILogger{SessionManager}"/> interface.</param>
 80        /// <param name="eventManager">Instance of <see cref="IEventManager"/> interface.</param>
 81        /// <param name="userDataManager">Instance of <see cref="IUserDataManager"/> interface.</param>
 82        /// <param name="serverConfigurationManager">Instance of <see cref="IServerConfigurationManager"/> interface.</p
 83        /// <param name="libraryManager">Instance of <see cref="ILibraryManager"/> interface.</param>
 84        /// <param name="userManager">Instance of <see cref="IUserManager"/> interface.</param>
 85        /// <param name="musicManager">Instance of <see cref="IMusicManager"/> interface.</param>
 86        /// <param name="dtoService">Instance of <see cref="IDtoService"/> interface.</param>
 87        /// <param name="imageProcessor">Instance of <see cref="IImageProcessor"/> interface.</param>
 88        /// <param name="appHost">Instance of <see cref="IServerApplicationHost"/> interface.</param>
 89        /// <param name="deviceManager">Instance of <see cref="IDeviceManager"/> interface.</param>
 90        /// <param name="mediaSourceManager">Instance of <see cref="IMediaSourceManager"/> interface.</param>
 91        /// <param name="hostApplicationLifetime">Instance of <see cref="IHostApplicationLifetime"/> interface.</param>
 92        public SessionManager(
 93            ILogger<SessionManager> logger,
 94            IEventManager eventManager,
 95            IUserDataManager userDataManager,
 96            IServerConfigurationManager serverConfigurationManager,
 97            ILibraryManager libraryManager,
 98            IUserManager userManager,
 99            IMusicManager musicManager,
 100            IDtoService dtoService,
 101            IImageProcessor imageProcessor,
 102            IServerApplicationHost appHost,
 103            IDeviceManager deviceManager,
 104            IMediaSourceManager mediaSourceManager,
 105            IHostApplicationLifetime hostApplicationLifetime)
 106        {
 31107            _logger = logger;
 31108            _eventManager = eventManager;
 31109            _userDataManager = userDataManager;
 31110            _config = serverConfigurationManager;
 31111            _libraryManager = libraryManager;
 31112            _userManager = userManager;
 31113            _musicManager = musicManager;
 31114            _dtoService = dtoService;
 31115            _imageProcessor = imageProcessor;
 31116            _appHost = appHost;
 31117            _deviceManager = deviceManager;
 31118            _mediaSourceManager = mediaSourceManager;
 31119            _shutdownCallback = hostApplicationLifetime.ApplicationStopping.Register(OnApplicationStopping);
 120
 31121            _deviceManager.DeviceOptionsUpdated += OnDeviceManagerDeviceOptionsUpdated;
 31122        }
 123
 124        /// <summary>
 125        /// Occurs when playback has started.
 126        /// </summary>
 127        public event EventHandler<PlaybackProgressEventArgs> PlaybackStart;
 128
 129        /// <summary>
 130        /// Occurs when playback has progressed.
 131        /// </summary>
 132        public event EventHandler<PlaybackProgressEventArgs> PlaybackProgress;
 133
 134        /// <summary>
 135        /// Occurs when playback has stopped.
 136        /// </summary>
 137        public event EventHandler<PlaybackStopEventArgs> PlaybackStopped;
 138
 139        /// <inheritdoc />
 140        public event EventHandler<SessionEventArgs> SessionStarted;
 141
 142        /// <inheritdoc />
 143        public event EventHandler<SessionEventArgs> CapabilitiesChanged;
 144
 145        /// <inheritdoc />
 146        public event EventHandler<SessionEventArgs> SessionEnded;
 147
 148        /// <inheritdoc />
 149        public event EventHandler<SessionEventArgs> SessionActivity;
 150
 151        /// <inheritdoc />
 152        public event EventHandler<SessionEventArgs> SessionControllerConnected;
 153
 154        /// <summary>
 155        /// Gets all connections.
 156        /// </summary>
 157        /// <value>All connections.</value>
 36158        public IEnumerable<SessionInfo> Sessions => _activeConnections.Values.OrderByDescending(c => c.LastActivityDate)
 159
 160        private void OnDeviceManagerDeviceOptionsUpdated(object sender, GenericEventArgs<Tuple<string, DeviceOptions>> e
 161        {
 0162            foreach (var session in Sessions)
 163            {
 0164                if (string.Equals(session.DeviceId, e.Argument.Item1, StringComparison.Ordinal))
 165                {
 0166                    if (!string.IsNullOrWhiteSpace(e.Argument.Item2.CustomName))
 167                    {
 0168                        session.HasCustomDeviceName = true;
 0169                        session.DeviceName = e.Argument.Item2.CustomName;
 170                    }
 171                    else
 172                    {
 0173                        session.HasCustomDeviceName = false;
 174                    }
 175                }
 176            }
 0177        }
 178
 179        private void CheckDisposed()
 180        {
 70181            ObjectDisposedException.ThrowIf(_disposed, this);
 70182        }
 183
 184        private void OnSessionStarted(SessionInfo info)
 185        {
 15186            if (!string.IsNullOrEmpty(info.DeviceId))
 187            {
 15188                var capabilities = _deviceManager.GetCapabilities(info.DeviceId);
 189
 15190                if (capabilities is not null)
 191                {
 15192                    ReportCapabilities(info, capabilities, false);
 193                }
 194            }
 195
 15196            _eventManager.Publish(new SessionStartedEventArgs(info));
 197
 15198            EventHelper.QueueEventIfNotNull(
 15199                SessionStarted,
 15200                this,
 15201                new SessionEventArgs
 15202                {
 15203                    SessionInfo = info
 15204                },
 15205                _logger);
 15206        }
 207
 208        private async ValueTask OnSessionEnded(SessionInfo info)
 209        {
 210            EventHelper.QueueEventIfNotNull(
 211                SessionEnded,
 212                this,
 213                new SessionEventArgs
 214                {
 215                    SessionInfo = info
 216                },
 217                _logger);
 218
 219            _eventManager.Publish(new SessionEndedEventArgs(info));
 220
 221            await info.DisposeAsync().ConfigureAwait(false);
 222        }
 223
 224        /// <inheritdoc />
 225        public void UpdateDeviceName(string sessionId, string reportedDeviceName)
 226        {
 0227            var session = GetSession(sessionId);
 0228            if (session is not null)
 229            {
 0230                session.DeviceName = reportedDeviceName;
 231            }
 0232        }
 233
 234        /// <summary>
 235        /// Logs the user activity.
 236        /// </summary>
 237        /// <param name="appName">Type of the client.</param>
 238        /// <param name="appVersion">The app version.</param>
 239        /// <param name="deviceId">The device id.</param>
 240        /// <param name="deviceName">Name of the device.</param>
 241        /// <param name="remoteEndPoint">The remote end point.</param>
 242        /// <param name="user">The user.</param>
 243        /// <returns>SessionInfo.</returns>
 244        public async Task<SessionInfo> LogSessionActivity(
 245            string appName,
 246            string appVersion,
 247            string deviceId,
 248            string deviceName,
 249            string remoteEndPoint,
 250            User user)
 251        {
 252            CheckDisposed();
 253
 254            ArgumentException.ThrowIfNullOrEmpty(appName);
 255            ArgumentException.ThrowIfNullOrEmpty(appVersion);
 256            ArgumentException.ThrowIfNullOrEmpty(deviceId);
 257
 258            var activityDate = DateTime.UtcNow;
 259            var session = GetSessionInfo(appName, appVersion, deviceId, deviceName, remoteEndPoint, user);
 260            var lastActivityDate = session.LastActivityDate;
 261            session.LastActivityDate = activityDate;
 262
 263            if (user is not null)
 264            {
 265                var userLastActivityDate = user.LastActivityDate ?? DateTime.MinValue;
 266
 267                if ((activityDate - userLastActivityDate).TotalSeconds > 60)
 268                {
 269                    try
 270                    {
 271                        user.LastActivityDate = activityDate;
 272                        await _userManager.UpdateUserAsync(user).ConfigureAwait(false);
 273                    }
 274                    catch (DbUpdateConcurrencyException e)
 275                    {
 276                        _logger.LogDebug(e, "Error updating user's last activity date.");
 277                    }
 278                }
 279            }
 280
 281            if ((activityDate - lastActivityDate).TotalSeconds > 10)
 282            {
 283                SessionActivity?.Invoke(
 284                    this,
 285                    new SessionEventArgs
 286                    {
 287                        SessionInfo = session
 288                    });
 289            }
 290
 291            return session;
 292        }
 293
 294        /// <inheritdoc />
 295        public void OnSessionControllerConnected(SessionInfo session)
 296        {
 0297            EventHelper.QueueEventIfNotNull(
 0298                SessionControllerConnected,
 0299                this,
 0300                new SessionEventArgs
 0301                {
 0302                    SessionInfo = session
 0303                },
 0304                _logger);
 0305        }
 306
 307        /// <inheritdoc />
 308        public async Task CloseIfNeededAsync(SessionInfo session)
 309        {
 310            if (!session.SessionControllers.Any(i => i.IsSessionActive))
 311            {
 312                var key = GetSessionKey(session.Client, session.DeviceId);
 313
 314                _activeConnections.TryRemove(key, out _);
 315                if (!string.IsNullOrEmpty(session.PlayState?.LiveStreamId))
 316                {
 317                    await CloseLiveStreamIfNeededAsync(session.PlayState.LiveStreamId, session.Id).ConfigureAwait(false)
 318                }
 319
 320                await OnSessionEnded(session).ConfigureAwait(false);
 321            }
 322        }
 323
 324        /// <inheritdoc />
 325        public async Task CloseLiveStreamIfNeededAsync(string liveStreamId, string sessionIdOrPlaySessionId)
 326        {
 327            bool liveStreamNeedsToBeClosed = false;
 328
 329            if (_activeLiveStreamSessions.TryGetValue(liveStreamId, out var activeSessionMappings))
 330            {
 331                if (activeSessionMappings.TryRemove(sessionIdOrPlaySessionId, out var correspondingId))
 332                {
 333                    if (!string.IsNullOrEmpty(correspondingId))
 334                    {
 335                        activeSessionMappings.TryRemove(correspondingId, out _);
 336                    }
 337
 338                    liveStreamNeedsToBeClosed = true;
 339                }
 340
 341                if (activeSessionMappings.IsEmpty)
 342                {
 343                    _activeLiveStreamSessions.TryRemove(liveStreamId, out _);
 344                }
 345            }
 346
 347            if (liveStreamNeedsToBeClosed)
 348            {
 349                try
 350                {
 351                    await _mediaSourceManager.CloseLiveStream(liveStreamId).ConfigureAwait(false);
 352                }
 353                catch (Exception ex)
 354                {
 355                    _logger.LogError(ex, "Error closing live stream");
 356                }
 357            }
 358        }
 359
 360        /// <inheritdoc />
 361        public async ValueTask ReportSessionEnded(string sessionId)
 362        {
 363            CheckDisposed();
 364            var session = GetSession(sessionId, false);
 365
 366            if (session is not null)
 367            {
 368                var key = GetSessionKey(session.Client, session.DeviceId);
 369
 370                _activeConnections.TryRemove(key, out _);
 371
 372                await OnSessionEnded(session).ConfigureAwait(false);
 373            }
 374        }
 375
 376        private Task<MediaSourceInfo> GetMediaSource(BaseItem item, string mediaSourceId, string liveStreamId)
 377        {
 0378            return _mediaSourceManager.GetMediaSource(item, mediaSourceId, liveStreamId, false, CancellationToken.None);
 379        }
 380
 381        /// <summary>
 382        /// Updates the now playing item id.
 383        /// </summary>
 384        /// <returns>Task.</returns>
 385        private async Task UpdateNowPlayingItem(SessionInfo session, PlaybackProgressInfo info, BaseItem libraryItem, bo
 386        {
 387            if (session is null)
 388            {
 389               return;
 390            }
 391
 392            if (string.IsNullOrEmpty(info.MediaSourceId))
 393            {
 394                info.MediaSourceId = info.ItemId.ToString("N", CultureInfo.InvariantCulture);
 395            }
 396
 397            if (!info.ItemId.IsEmpty() && info.Item is null && libraryItem is not null)
 398            {
 399                var current = session.NowPlayingItem;
 400
 401                if (current is null || !info.ItemId.Equals(current.Id))
 402                {
 403                    var runtimeTicks = libraryItem.RunTimeTicks;
 404
 405                    MediaSourceInfo mediaSource = null;
 406                    if (libraryItem is IHasMediaSources)
 407                    {
 408                        mediaSource = await GetMediaSource(libraryItem, info.MediaSourceId, info.LiveStreamId).Configure
 409
 410                        if (mediaSource is not null)
 411                        {
 412                            runtimeTicks = mediaSource.RunTimeTicks;
 413                        }
 414                    }
 415
 416                    info.Item = GetItemInfo(libraryItem, mediaSource);
 417
 418                    info.Item.RunTimeTicks = runtimeTicks;
 419                }
 420                else
 421                {
 422                    info.Item = current;
 423                }
 424            }
 425
 426            session.NowPlayingItem = info.Item;
 427            session.LastActivityDate = DateTime.UtcNow;
 428
 429            if (updateLastCheckInTime)
 430            {
 431                session.LastPlaybackCheckIn = DateTime.UtcNow;
 432            }
 433
 434            if (info.IsPaused && session.LastPausedDate is null)
 435            {
 436                session.LastPausedDate = DateTime.UtcNow;
 437            }
 438            else if (!info.IsPaused)
 439            {
 440                session.LastPausedDate = null;
 441            }
 442
 443            session.PlayState.IsPaused = info.IsPaused;
 444            session.PlayState.PositionTicks = info.PositionTicks;
 445            session.PlayState.MediaSourceId = info.MediaSourceId;
 446            session.PlayState.LiveStreamId = info.LiveStreamId;
 447            session.PlayState.CanSeek = info.CanSeek;
 448            session.PlayState.IsMuted = info.IsMuted;
 449            session.PlayState.VolumeLevel = info.VolumeLevel;
 450            session.PlayState.AudioStreamIndex = info.AudioStreamIndex;
 451            session.PlayState.SubtitleStreamIndex = info.SubtitleStreamIndex;
 452            session.PlayState.PlayMethod = info.PlayMethod;
 453            session.PlayState.RepeatMode = info.RepeatMode;
 454            session.PlayState.PlaybackOrder = info.PlaybackOrder;
 455            session.PlaylistItemId = info.PlaylistItemId;
 456
 457            var nowPlayingQueue = info.NowPlayingQueue;
 458
 459            if (nowPlayingQueue?.Length > 0)
 460            {
 461                session.NowPlayingQueue = nowPlayingQueue;
 462
 463                var itemIds = Array.ConvertAll(nowPlayingQueue, queue => queue.Id);
 464                session.NowPlayingQueueFullItems = _dtoService.GetBaseItemDtos(
 465                    _libraryManager.GetItemList(new InternalItemsQuery { ItemIds = itemIds }),
 466                    new DtoOptions(true));
 467            }
 468        }
 469
 470        /// <summary>
 471        /// Removes the now playing item id.
 472        /// </summary>
 473        /// <param name="session">The session.</param>
 474        private void RemoveNowPlayingItem(SessionInfo session)
 475        {
 0476            session.NowPlayingItem = null;
 0477            session.PlayState = new PlayerStateInfo();
 478
 0479            if (!string.IsNullOrEmpty(session.DeviceId))
 480            {
 0481                ClearTranscodingInfo(session.DeviceId);
 482            }
 0483        }
 484
 485        private static string GetSessionKey(string appName, string deviceId)
 15486            => appName + deviceId;
 487
 488        /// <summary>
 489        /// Gets the connection.
 490        /// </summary>
 491        /// <param name="appName">Type of the client.</param>
 492        /// <param name="appVersion">The app version.</param>
 493        /// <param name="deviceId">The device id.</param>
 494        /// <param name="deviceName">Name of the device.</param>
 495        /// <param name="remoteEndPoint">The remote end point.</param>
 496        /// <param name="user">The user.</param>
 497        /// <returns>SessionInfo.</returns>
 498        private SessionInfo GetSessionInfo(
 499            string appName,
 500            string appVersion,
 501            string deviceId,
 502            string deviceName,
 503            string remoteEndPoint,
 504            User user)
 505        {
 15506            CheckDisposed();
 507
 15508            ArgumentException.ThrowIfNullOrEmpty(deviceId);
 509
 15510            var key = GetSessionKey(appName, deviceId);
 511
 15512            CheckDisposed();
 513
 15514            if (!_activeConnections.TryGetValue(key, out var sessionInfo))
 515            {
 15516                sessionInfo = CreateSession(key, appName, appVersion, deviceId, deviceName, remoteEndPoint, user);
 15517                _activeConnections[key] = sessionInfo;
 518            }
 519
 15520            sessionInfo.UserId = user?.Id ?? Guid.Empty;
 15521            sessionInfo.UserName = user?.Username;
 15522            sessionInfo.UserPrimaryImageTag = user?.ProfileImage is null ? null : GetImageCacheTag(user);
 15523            sessionInfo.RemoteEndPoint = remoteEndPoint;
 15524            sessionInfo.Client = appName;
 525
 15526            if (!sessionInfo.HasCustomDeviceName || string.IsNullOrEmpty(sessionInfo.DeviceName))
 527            {
 15528                sessionInfo.DeviceName = deviceName;
 529            }
 530
 15531            sessionInfo.ApplicationVersion = appVersion;
 532
 15533            if (user is null)
 534            {
 0535                sessionInfo.AdditionalUsers = Array.Empty<SessionUserInfo>();
 536            }
 537
 15538            return sessionInfo;
 539        }
 540
 541        private SessionInfo CreateSession(
 542            string key,
 543            string appName,
 544            string appVersion,
 545            string deviceId,
 546            string deviceName,
 547            string remoteEndPoint,
 548            User user)
 549        {
 15550            var sessionInfo = new SessionInfo(this, _logger)
 15551            {
 15552                Client = appName,
 15553                DeviceId = deviceId,
 15554                ApplicationVersion = appVersion,
 15555                Id = key.GetMD5().ToString("N", CultureInfo.InvariantCulture),
 15556                ServerId = _appHost.SystemId
 15557            };
 558
 15559            var username = user?.Username;
 560
 15561            sessionInfo.UserId = user?.Id ?? Guid.Empty;
 15562            sessionInfo.UserName = username;
 15563            sessionInfo.UserPrimaryImageTag = user?.ProfileImage is null ? null : GetImageCacheTag(user);
 15564            sessionInfo.RemoteEndPoint = remoteEndPoint;
 565
 15566            if (string.IsNullOrEmpty(deviceName))
 567            {
 0568                deviceName = "Network Device";
 569            }
 570
 15571            var deviceOptions = _deviceManager.GetDeviceOptions(deviceId) ?? new()
 15572            {
 15573                DeviceId = deviceId
 15574            };
 15575            if (string.IsNullOrEmpty(deviceOptions.CustomName))
 576            {
 15577                sessionInfo.DeviceName = deviceName;
 578            }
 579            else
 580            {
 0581                sessionInfo.DeviceName = deviceOptions.CustomName;
 0582                sessionInfo.HasCustomDeviceName = true;
 583            }
 584
 15585            OnSessionStarted(sessionInfo);
 15586            return sessionInfo;
 587        }
 588
 589        private List<User> GetUsers(SessionInfo session)
 590        {
 0591            var users = new List<User>();
 592
 0593            if (session.UserId.IsEmpty())
 594            {
 0595                return users;
 596            }
 597
 0598            var user = _userManager.GetUserById(session.UserId);
 599
 0600            if (user is null)
 601            {
 0602                throw new InvalidOperationException("User not found");
 603            }
 604
 0605            users.Add(user);
 606
 0607            users.AddRange(session.AdditionalUsers
 0608                .Select(i => _userManager.GetUserById(i.UserId))
 0609                .Where(i => i is not null));
 610
 0611            return users;
 612        }
 613
 614        private void StartCheckTimers()
 615        {
 0616            _idleTimer ??= new Timer(CheckForIdlePlayback, null, TimeSpan.FromMinutes(5), TimeSpan.FromMinutes(5));
 617
 0618            if (_config.Configuration.InactiveSessionThreshold > 0)
 619            {
 0620                _inactiveTimer ??= new Timer(CheckForInactiveSteams, null, TimeSpan.FromMinutes(1), TimeSpan.FromMinutes
 621            }
 622            else
 623            {
 0624                StopInactiveCheckTimer();
 625            }
 0626        }
 627
 628        private void StopIdleCheckTimer()
 629        {
 0630            if (_idleTimer is not null)
 631            {
 0632                _idleTimer.Dispose();
 0633                _idleTimer = null;
 634            }
 0635        }
 636
 637        private void StopInactiveCheckTimer()
 638        {
 0639            if (_inactiveTimer is not null)
 640            {
 0641                _inactiveTimer.Dispose();
 0642                _inactiveTimer = null;
 643            }
 0644        }
 645
 646        private async void CheckForIdlePlayback(object state)
 647        {
 648            var playingSessions = Sessions.Where(i => i.NowPlayingItem is not null)
 649                .ToList();
 650
 651            if (playingSessions.Count > 0)
 652            {
 653                var idle = playingSessions
 654                    .Where(i => (DateTime.UtcNow - i.LastPlaybackCheckIn).TotalMinutes > 5)
 655                    .ToList();
 656
 657                foreach (var session in idle)
 658                {
 659                    _logger.LogDebug("Session {0} has gone idle while playing", session.Id);
 660
 661                    try
 662                    {
 663                        await OnPlaybackStopped(new PlaybackStopInfo
 664                        {
 665                            Item = session.NowPlayingItem,
 666                            ItemId = session.NowPlayingItem is null ? Guid.Empty : session.NowPlayingItem.Id,
 667                            SessionId = session.Id,
 668                            MediaSourceId = session.PlayState?.MediaSourceId,
 669                            PositionTicks = session.PlayState?.PositionTicks
 670                        }).ConfigureAwait(false);
 671                    }
 672                    catch (Exception ex)
 673                    {
 674                        _logger.LogDebug(ex, "Error calling OnPlaybackStopped");
 675                    }
 676                }
 677            }
 678            else
 679            {
 680                StopIdleCheckTimer();
 681            }
 682        }
 683
 684        private async void CheckForInactiveSteams(object state)
 685        {
 686            var inactiveSessions = Sessions.Where(i =>
 687                    i.NowPlayingItem is not null
 688                    && i.PlayState.IsPaused
 689                    && (DateTime.UtcNow - i.LastPausedDate).Value.TotalMinutes > _config.Configuration.InactiveSessionTh
 690
 691            foreach (var session in inactiveSessions)
 692            {
 693                _logger.LogDebug("Session {Session} has been inactive for {InactiveTime} minutes. Stopping it.", session
 694
 695                try
 696                {
 697                    await SendPlaystateCommand(
 698                        session.Id,
 699                        session.Id,
 700                        new PlaystateRequest()
 701                        {
 702                            Command = PlaystateCommand.Stop,
 703                            ControllingUserId = session.UserId.ToString(),
 704                            SeekPositionTicks = session.PlayState?.PositionTicks
 705                        },
 706                        CancellationToken.None).ConfigureAwait(true);
 707                }
 708                catch (Exception ex)
 709                {
 710                    _logger.LogDebug(ex, "Error calling SendPlaystateCommand for stopping inactive session {Session}.", 
 711                }
 712            }
 713
 714            bool playingSessions = Sessions.Any(i => i.NowPlayingItem is not null);
 715
 716            if (!playingSessions)
 717            {
 718                StopInactiveCheckTimer();
 719            }
 720        }
 721
 722        private BaseItem GetNowPlayingItem(SessionInfo session, Guid itemId)
 723        {
 0724            if (session is null)
 725            {
 0726                return null;
 727            }
 728
 0729            var item = session.FullNowPlayingItem;
 0730            if (item is not null && item.Id.Equals(itemId))
 731            {
 0732                return item;
 733            }
 734
 0735            item = _libraryManager.GetItemById(itemId);
 736
 0737            session.FullNowPlayingItem = item;
 738
 0739            return item;
 740        }
 741
 742        /// <summary>
 743        /// Used to report that playback has started for an item.
 744        /// </summary>
 745        /// <param name="info">The info.</param>
 746        /// <returns>Task.</returns>
 747        /// <exception cref="ArgumentNullException"><c>info</c> is <c>null</c>.</exception>
 748        public async Task OnPlaybackStart(PlaybackStartInfo info)
 749        {
 750            CheckDisposed();
 751
 752            ArgumentNullException.ThrowIfNull(info);
 753
 754            var session = GetSession(info.SessionId);
 755
 756            var libraryItem = info.ItemId.IsEmpty()
 757                ? null
 758                : GetNowPlayingItem(session, info.ItemId);
 759
 760            await UpdateNowPlayingItem(session, info, libraryItem, true).ConfigureAwait(false);
 761
 762            if (!string.IsNullOrEmpty(session.DeviceId) && info.PlayMethod != PlayMethod.Transcode)
 763            {
 764                ClearTranscodingInfo(session.DeviceId);
 765            }
 766
 767            session.StartAutomaticProgress(info);
 768
 769            var users = GetUsers(session);
 770
 771            if (libraryItem is not null)
 772            {
 773                foreach (var user in users)
 774                {
 775                    OnPlaybackStart(user, libraryItem);
 776                }
 777            }
 778
 779            if (!string.IsNullOrEmpty(info.LiveStreamId))
 780            {
 781                UpdateLiveStreamActiveSessionMappings(info.LiveStreamId, info.SessionId, info.PlaySessionId);
 782            }
 783
 784            var eventArgs = new PlaybackStartEventArgs
 785            {
 786                Item = libraryItem,
 787                Users = users,
 788                MediaSourceId = info.MediaSourceId,
 789                MediaInfo = info.Item,
 790                DeviceName = session.DeviceName,
 791                ClientName = session.Client,
 792                DeviceId = session.DeviceId,
 793                Session = session,
 794                PlaybackPositionTicks = info.PositionTicks,
 795                PlaySessionId = info.PlaySessionId
 796            };
 797
 798            await _eventManager.PublishAsync(eventArgs).ConfigureAwait(false);
 799
 800            // Nothing to save here
 801            // Fire events to inform plugins
 802            EventHelper.QueueEventIfNotNull(
 803                PlaybackStart,
 804                this,
 805                eventArgs,
 806                _logger);
 807
 808            StartCheckTimers();
 809        }
 810
 811        /// <summary>
 812        /// Called when [playback start].
 813        /// </summary>
 814        /// <param name="user">The user object.</param>
 815        /// <param name="item">The item.</param>
 816        private void OnPlaybackStart(User user, BaseItem item)
 817        {
 0818            var data = _userDataManager.GetUserData(user, item);
 819
 0820            data.PlayCount++;
 0821            data.LastPlayedDate = DateTime.UtcNow;
 822
 0823            if (item.SupportsPlayedStatus && !item.SupportsPositionTicksResume)
 824            {
 0825                data.Played = true;
 826            }
 827            else
 828            {
 0829                data.Played = false;
 830            }
 831
 0832            _userDataManager.SaveUserData(user, item, data, UserDataSaveReason.PlaybackStart, CancellationToken.None);
 0833        }
 834
 835        /// <inheritdoc />
 836        public Task OnPlaybackProgress(PlaybackProgressInfo info)
 837        {
 0838            return OnPlaybackProgress(info, false);
 839        }
 840
 841        private void UpdateLiveStreamActiveSessionMappings(string liveStreamId, string sessionId, string playSessionId)
 842        {
 0843            var activeSessionMappings = _activeLiveStreamSessions.GetOrAdd(liveStreamId, _ => new ConcurrentDictionary<s
 844
 0845            if (!string.IsNullOrEmpty(playSessionId))
 846            {
 0847                if (!activeSessionMappings.TryGetValue(sessionId, out var currentPlaySessionId) || currentPlaySessionId 
 848                {
 0849                    if (!string.IsNullOrEmpty(currentPlaySessionId))
 850                    {
 0851                        activeSessionMappings.TryRemove(currentPlaySessionId, out _);
 852                    }
 853
 0854                    activeSessionMappings[sessionId] = playSessionId;
 0855                    activeSessionMappings[playSessionId] = sessionId;
 856                }
 857            }
 858            else
 859            {
 0860                if (!activeSessionMappings.TryGetValue(sessionId, out _))
 861                {
 0862                    activeSessionMappings[sessionId] = string.Empty;
 863                }
 864            }
 0865        }
 866
 867        /// <summary>
 868        /// Used to report playback progress for an item.
 869        /// </summary>
 870        /// <param name="info">The playback progress info.</param>
 871        /// <param name="isAutomated">Whether this is an automated update.</param>
 872        /// <returns>Task.</returns>
 873        public async Task OnPlaybackProgress(PlaybackProgressInfo info, bool isAutomated)
 874        {
 875            CheckDisposed();
 876
 877            ArgumentNullException.ThrowIfNull(info);
 878
 879            var session = GetSession(info.SessionId, false);
 880            if (session is null)
 881            {
 882                return;
 883            }
 884
 885            var libraryItem = info.ItemId.IsEmpty()
 886                ? null
 887                : GetNowPlayingItem(session, info.ItemId);
 888
 889            await UpdateNowPlayingItem(session, info, libraryItem, !isAutomated).ConfigureAwait(false);
 890
 891            if (!string.IsNullOrEmpty(session.DeviceId) && info.PlayMethod != PlayMethod.Transcode)
 892            {
 893                ClearTranscodingInfo(session.DeviceId);
 894            }
 895
 896            var users = GetUsers(session);
 897
 898            // only update saved user data on actual check-ins, not automated ones
 899            if (libraryItem is not null && !isAutomated)
 900            {
 901                foreach (var user in users)
 902                {
 903                    OnPlaybackProgress(user, libraryItem, info);
 904                }
 905            }
 906
 907            if (!string.IsNullOrEmpty(info.LiveStreamId))
 908            {
 909                UpdateLiveStreamActiveSessionMappings(info.LiveStreamId, info.SessionId, info.PlaySessionId);
 910            }
 911
 912            var eventArgs = new PlaybackProgressEventArgs
 913            {
 914                Item = libraryItem,
 915                Users = users,
 916                PlaybackPositionTicks = session.PlayState.PositionTicks,
 917                MediaSourceId = session.PlayState.MediaSourceId,
 918                MediaInfo = info.Item,
 919                DeviceName = session.DeviceName,
 920                ClientName = session.Client,
 921                DeviceId = session.DeviceId,
 922                IsPaused = info.IsPaused,
 923                PlaySessionId = info.PlaySessionId,
 924                IsAutomated = isAutomated,
 925                Session = session
 926            };
 927
 928            await _eventManager.PublishAsync(eventArgs).ConfigureAwait(false);
 929
 930            PlaybackProgress?.Invoke(this, eventArgs);
 931
 932            if (!isAutomated)
 933            {
 934                session.StartAutomaticProgress(info);
 935            }
 936
 937            StartCheckTimers();
 938        }
 939
 940        private void OnPlaybackProgress(User user, BaseItem item, PlaybackProgressInfo info)
 941        {
 0942            var data = _userDataManager.GetUserData(user, item);
 943
 0944            var positionTicks = info.PositionTicks;
 945
 0946            var changed = false;
 947
 0948            if (positionTicks.HasValue)
 949            {
 0950                _userDataManager.UpdatePlayState(item, data, positionTicks.Value);
 0951                changed = true;
 952            }
 953
 0954            var tracksChanged = UpdatePlaybackSettings(user, info, data);
 0955            if (!tracksChanged)
 956            {
 0957                changed = true;
 958            }
 959
 0960            if (changed)
 961            {
 0962                _userDataManager.SaveUserData(user, item, data, UserDataSaveReason.PlaybackProgress, CancellationToken.N
 963            }
 0964        }
 965
 966        private static bool UpdatePlaybackSettings(User user, PlaybackProgressInfo info, UserItemData data)
 967        {
 0968            var changed = false;
 969
 0970            if (user.RememberAudioSelections)
 971            {
 0972                if (data.AudioStreamIndex != info.AudioStreamIndex)
 973                {
 0974                    data.AudioStreamIndex = info.AudioStreamIndex;
 0975                    changed = true;
 976                }
 977            }
 978            else
 979            {
 0980                if (data.AudioStreamIndex.HasValue)
 981                {
 0982                    data.AudioStreamIndex = null;
 0983                    changed = true;
 984                }
 985            }
 986
 0987            if (user.RememberSubtitleSelections)
 988            {
 0989                if (data.SubtitleStreamIndex != info.SubtitleStreamIndex)
 990                {
 0991                    data.SubtitleStreamIndex = info.SubtitleStreamIndex;
 0992                    changed = true;
 993                }
 994            }
 995            else
 996            {
 0997                if (data.SubtitleStreamIndex.HasValue)
 998                {
 0999                    data.SubtitleStreamIndex = null;
 01000                    changed = true;
 1001                }
 1002            }
 1003
 01004            return changed;
 1005        }
 1006
 1007        /// <summary>
 1008        /// Used to report that playback has ended for an item.
 1009        /// </summary>
 1010        /// <param name="info">The info.</param>
 1011        /// <returns>Task.</returns>
 1012        /// <exception cref="ArgumentNullException"><c>info</c> is <c>null</c>.</exception>
 1013        /// <exception cref="ArgumentOutOfRangeException"><c>info.PositionTicks</c> is <c>null</c> or negative.</excepti
 1014        public async Task OnPlaybackStopped(PlaybackStopInfo info)
 1015        {
 1016            CheckDisposed();
 1017
 1018            ArgumentNullException.ThrowIfNull(info);
 1019
 1020            if (info.PositionTicks.HasValue && info.PositionTicks.Value < 0)
 1021            {
 1022                throw new ArgumentOutOfRangeException(nameof(info), "The PlaybackStopInfo's PositionTicks was negative."
 1023            }
 1024
 1025            var session = GetSession(info.SessionId);
 1026
 1027            session.StopAutomaticProgress();
 1028
 1029            var libraryItem = info.ItemId.IsEmpty()
 1030                ? null
 1031                : GetNowPlayingItem(session, info.ItemId);
 1032
 1033            // Normalize
 1034            if (string.IsNullOrEmpty(info.MediaSourceId))
 1035            {
 1036                info.MediaSourceId = info.ItemId.ToString("N", CultureInfo.InvariantCulture);
 1037            }
 1038
 1039            if (!info.ItemId.IsEmpty() && info.Item is null && libraryItem is not null)
 1040            {
 1041                var current = session.NowPlayingItem;
 1042
 1043                if (current is null || !info.ItemId.Equals(current.Id))
 1044                {
 1045                    MediaSourceInfo mediaSource = null;
 1046
 1047                    if (libraryItem is IHasMediaSources)
 1048                    {
 1049                        mediaSource = await GetMediaSource(libraryItem, info.MediaSourceId, info.LiveStreamId).Configure
 1050                    }
 1051
 1052                    info.Item = GetItemInfo(libraryItem, mediaSource);
 1053                }
 1054                else
 1055                {
 1056                    info.Item = current;
 1057                }
 1058            }
 1059
 1060            if (info.Item is not null)
 1061            {
 1062                var msString = info.PositionTicks.HasValue ? (info.PositionTicks.Value / 10000).ToString(CultureInfo.Inv
 1063
 1064                _logger.LogInformation(
 1065                    "Playback stopped reported by app {0} {1} playing {2}. Stopped at {3} ms",
 1066                    session.Client,
 1067                    session.ApplicationVersion,
 1068                    info.Item.Name,
 1069                    msString);
 1070            }
 1071
 1072            if (info.NowPlayingQueue is not null)
 1073            {
 1074                session.NowPlayingQueue = info.NowPlayingQueue;
 1075            }
 1076
 1077            session.PlaylistItemId = info.PlaylistItemId;
 1078
 1079            RemoveNowPlayingItem(session);
 1080
 1081            var users = GetUsers(session);
 1082            var playedToCompletion = false;
 1083
 1084            if (libraryItem is not null)
 1085            {
 1086                foreach (var user in users)
 1087                {
 1088                    playedToCompletion = OnPlaybackStopped(user, libraryItem, info.PositionTicks, info.Failed);
 1089                }
 1090            }
 1091
 1092            if (!string.IsNullOrEmpty(info.LiveStreamId))
 1093            {
 1094                await CloseLiveStreamIfNeededAsync(info.LiveStreamId, session.Id).ConfigureAwait(false);
 1095            }
 1096
 1097            var eventArgs = new PlaybackStopEventArgs
 1098            {
 1099                Item = libraryItem,
 1100                Users = users,
 1101                PlaybackPositionTicks = info.PositionTicks,
 1102                PlayedToCompletion = playedToCompletion,
 1103                MediaSourceId = info.MediaSourceId,
 1104                MediaInfo = info.Item,
 1105                DeviceName = session.DeviceName,
 1106                ClientName = session.Client,
 1107                DeviceId = session.DeviceId,
 1108                Session = session,
 1109                PlaySessionId = info.PlaySessionId
 1110            };
 1111
 1112            await _eventManager.PublishAsync(eventArgs).ConfigureAwait(false);
 1113
 1114            EventHelper.QueueEventIfNotNull(PlaybackStopped, this, eventArgs, _logger);
 1115        }
 1116
 1117        private bool OnPlaybackStopped(User user, BaseItem item, long? positionTicks, bool playbackFailed)
 1118        {
 01119            if (playbackFailed)
 1120            {
 01121                return false;
 1122            }
 1123
 01124            var data = _userDataManager.GetUserData(user, item);
 1125            bool playedToCompletion;
 01126            if (positionTicks.HasValue)
 1127            {
 01128                playedToCompletion = _userDataManager.UpdatePlayState(item, data, positionTicks.Value);
 1129            }
 1130            else
 1131            {
 1132                // If the client isn't able to report this, then we'll just have to make an assumption
 01133                data.PlayCount++;
 01134                data.Played = item.SupportsPlayedStatus;
 01135                data.PlaybackPositionTicks = 0;
 01136                playedToCompletion = true;
 1137            }
 1138
 01139            _userDataManager.SaveUserData(user, item, data, UserDataSaveReason.PlaybackFinished, CancellationToken.None)
 1140
 01141            return playedToCompletion;
 1142        }
 1143
 1144        /// <summary>
 1145        /// Gets the session.
 1146        /// </summary>
 1147        /// <param name="sessionId">The session identifier.</param>
 1148        /// <param name="throwOnMissing">if set to <c>true</c> [throw on missing].</param>
 1149        /// <returns>SessionInfo.</returns>
 1150        /// <exception cref="ResourceNotFoundException">
 1151        /// No session with an Id equal to <c>sessionId</c> was found
 1152        /// and <c>throwOnMissing</c> is <c>true</c>.
 1153        /// </exception>
 1154        private SessionInfo GetSession(string sessionId, bool throwOnMissing = true)
 1155        {
 01156            var session = Sessions.FirstOrDefault(i => string.Equals(i.Id, sessionId, StringComparison.Ordinal));
 01157            if (session is null && throwOnMissing)
 1158            {
 01159                throw new ResourceNotFoundException(
 01160                    string.Format(CultureInfo.InvariantCulture, "Session {0} not found.", sessionId));
 1161            }
 1162
 01163            return session;
 1164        }
 1165
 1166        private SessionInfo GetSessionToRemoteControl(string sessionId)
 1167        {
 1168            // Accept either device id or session id
 01169            var session = Sessions.FirstOrDefault(i => string.Equals(i.Id, sessionId, StringComparison.Ordinal));
 1170
 01171            if (session is null)
 1172            {
 01173                throw new ResourceNotFoundException(
 01174                    string.Format(CultureInfo.InvariantCulture, "Session {0} not found.", sessionId));
 1175            }
 1176
 01177            return session;
 1178        }
 1179
 1180        private SessionInfoDto ToSessionInfoDto(SessionInfo sessionInfo)
 1181        {
 151182            return new SessionInfoDto
 151183            {
 151184                PlayState = sessionInfo.PlayState,
 151185                AdditionalUsers = sessionInfo.AdditionalUsers,
 151186                Capabilities = _deviceManager.ToClientCapabilitiesDto(sessionInfo.Capabilities),
 151187                RemoteEndPoint = sessionInfo.RemoteEndPoint,
 151188                PlayableMediaTypes = sessionInfo.PlayableMediaTypes,
 151189                Id = sessionInfo.Id,
 151190                UserId = sessionInfo.UserId,
 151191                UserName = sessionInfo.UserName,
 151192                Client = sessionInfo.Client,
 151193                LastActivityDate = sessionInfo.LastActivityDate,
 151194                LastPlaybackCheckIn = sessionInfo.LastPlaybackCheckIn,
 151195                LastPausedDate = sessionInfo.LastPausedDate,
 151196                DeviceName = sessionInfo.DeviceName,
 151197                DeviceType = sessionInfo.DeviceType,
 151198                NowPlayingItem = sessionInfo.NowPlayingItem,
 151199                NowViewingItem = sessionInfo.NowViewingItem,
 151200                DeviceId = sessionInfo.DeviceId,
 151201                ApplicationVersion = sessionInfo.ApplicationVersion,
 151202                TranscodingInfo = sessionInfo.TranscodingInfo,
 151203                IsActive = sessionInfo.IsActive,
 151204                SupportsMediaControl = sessionInfo.SupportsMediaControl,
 151205                SupportsRemoteControl = sessionInfo.SupportsRemoteControl,
 151206                NowPlayingQueue = sessionInfo.NowPlayingQueue,
 151207                NowPlayingQueueFullItems = sessionInfo.NowPlayingQueueFullItems,
 151208                HasCustomDeviceName = sessionInfo.HasCustomDeviceName,
 151209                PlaylistItemId = sessionInfo.PlaylistItemId,
 151210                ServerId = sessionInfo.ServerId,
 151211                UserPrimaryImageTag = sessionInfo.UserPrimaryImageTag,
 151212                SupportedCommands = sessionInfo.SupportedCommands
 151213            };
 1214        }
 1215
 1216        /// <inheritdoc />
 1217        public Task SendMessageCommand(string controllingSessionId, string sessionId, MessageCommand command, Cancellati
 1218        {
 01219            CheckDisposed();
 1220
 01221            var generalCommand = new GeneralCommand
 01222            {
 01223                Name = GeneralCommandType.DisplayMessage
 01224            };
 1225
 01226            generalCommand.Arguments["Header"] = command.Header;
 01227            generalCommand.Arguments["Text"] = command.Text;
 1228
 01229            if (command.TimeoutMs.HasValue)
 1230            {
 01231                generalCommand.Arguments["TimeoutMs"] = command.TimeoutMs.Value.ToString(CultureInfo.InvariantCulture);
 1232            }
 1233
 01234            return SendGeneralCommand(controllingSessionId, sessionId, generalCommand, cancellationToken);
 1235        }
 1236
 1237        /// <inheritdoc />
 1238        public Task SendGeneralCommand(string controllingSessionId, string sessionId, GeneralCommand command, Cancellati
 1239        {
 01240            CheckDisposed();
 1241
 01242            var session = GetSessionToRemoteControl(sessionId);
 1243
 01244            if (!string.IsNullOrEmpty(controllingSessionId))
 1245            {
 01246                var controllingSession = GetSession(controllingSessionId);
 01247                AssertCanControl(session, controllingSession);
 1248            }
 1249
 01250            return SendMessageToSession(session, SessionMessageType.GeneralCommand, command, cancellationToken);
 1251        }
 1252
 1253        private static async Task SendMessageToSession<T>(SessionInfo session, SessionMessageType name, T data, Cancella
 1254        {
 1255            var controllers = session.SessionControllers;
 1256            var messageId = Guid.NewGuid();
 1257
 1258            foreach (var controller in controllers)
 1259            {
 1260                await controller.SendMessage(name, messageId, data, cancellationToken).ConfigureAwait(false);
 1261            }
 1262        }
 1263
 1264        private static Task SendMessageToSessions<T>(IEnumerable<SessionInfo> sessions, SessionMessageType name, T data,
 1265        {
 1266            IEnumerable<Task> GetTasks()
 1267            {
 1268                var messageId = Guid.NewGuid();
 1269                foreach (var session in sessions)
 1270                {
 1271                    var controllers = session.SessionControllers;
 1272                    foreach (var controller in controllers)
 1273                    {
 1274                        yield return controller.SendMessage(name, messageId, data, cancellationToken);
 1275                    }
 1276                }
 1277            }
 1278
 211279            return Task.WhenAll(GetTasks());
 1280        }
 1281
 1282        /// <inheritdoc />
 1283        public async Task SendPlayCommand(string controllingSessionId, string sessionId, PlayRequest command, Cancellati
 1284        {
 1285            CheckDisposed();
 1286
 1287            var session = GetSessionToRemoteControl(sessionId);
 1288
 1289            var user = session.UserId.IsEmpty() ? null : _userManager.GetUserById(session.UserId);
 1290
 1291            List<BaseItem> items;
 1292
 1293            if (command.PlayCommand == PlayCommand.PlayInstantMix)
 1294            {
 1295                items = command.ItemIds.SelectMany(i => TranslateItemForInstantMix(i, user))
 1296                    .ToList();
 1297
 1298                command.PlayCommand = PlayCommand.PlayNow;
 1299            }
 1300            else
 1301            {
 1302                var list = new List<BaseItem>();
 1303                foreach (var itemId in command.ItemIds)
 1304                {
 1305                    var subItems = TranslateItemForPlayback(itemId, user);
 1306                    list.AddRange(subItems);
 1307                }
 1308
 1309                items = list;
 1310            }
 1311
 1312            if (command.PlayCommand == PlayCommand.PlayShuffle)
 1313            {
 1314                items.Shuffle();
 1315                command.PlayCommand = PlayCommand.PlayNow;
 1316            }
 1317
 1318            command.ItemIds = items.Select(i => i.Id).ToArray();
 1319
 1320            if (user is not null)
 1321            {
 1322                if (items.Any(i => i.GetPlayAccess(user) != PlayAccess.Full))
 1323                {
 1324                    throw new ArgumentException(
 1325                        string.Format(CultureInfo.InvariantCulture, "{0} is not allowed to play media.", user.Username))
 1326                }
 1327            }
 1328
 1329            if (user is not null
 1330                && command.ItemIds.Length == 1
 1331                && user.EnableNextEpisodeAutoPlay
 1332                && _libraryManager.GetItemById(command.ItemIds[0]) is Episode episode)
 1333            {
 1334                var series = episode.Series;
 1335                if (series is not null)
 1336                {
 1337                    var episodes = series.GetEpisodes(
 1338                            user,
 1339                            new DtoOptions(false)
 1340                            {
 1341                                EnableImages = false
 1342                            },
 1343                            user.DisplayMissingEpisodes)
 1344                        .Where(i => !i.IsVirtualItem)
 1345                        .SkipWhile(i => !i.Id.Equals(episode.Id))
 1346                        .ToList();
 1347
 1348                    if (episodes.Count > 0)
 1349                    {
 1350                        command.ItemIds = episodes.Select(i => i.Id).ToArray();
 1351                    }
 1352                }
 1353            }
 1354
 1355            if (!string.IsNullOrEmpty(controllingSessionId))
 1356            {
 1357                var controllingSession = GetSession(controllingSessionId);
 1358                AssertCanControl(session, controllingSession);
 1359                if (!controllingSession.UserId.IsEmpty())
 1360                {
 1361                    command.ControllingUserId = controllingSession.UserId;
 1362                }
 1363            }
 1364
 1365            await SendMessageToSession(session, SessionMessageType.Play, command, cancellationToken).ConfigureAwait(fals
 1366        }
 1367
 1368        /// <inheritdoc />
 1369        public async Task SendSyncPlayCommand(string sessionId, SendCommand command, CancellationToken cancellationToken
 1370        {
 1371            CheckDisposed();
 1372            var session = GetSession(sessionId);
 1373            await SendMessageToSession(session, SessionMessageType.SyncPlayCommand, command, cancellationToken).Configur
 1374        }
 1375
 1376        /// <inheritdoc />
 1377        public async Task SendSyncPlayGroupUpdate<T>(string sessionId, GroupUpdate<T> command, CancellationToken cancell
 1378        {
 1379            CheckDisposed();
 1380            var session = GetSession(sessionId);
 1381            await SendMessageToSession(session, SessionMessageType.SyncPlayGroupUpdate, command, cancellationToken).Conf
 1382        }
 1383
 1384        private IEnumerable<BaseItem> TranslateItemForPlayback(Guid id, User user)
 1385        {
 01386            var item = _libraryManager.GetItemById(id);
 1387
 01388            if (item is null)
 1389            {
 01390                _logger.LogError("A nonexistent item Id {0} was passed into TranslateItemForPlayback", id);
 01391                return Array.Empty<BaseItem>();
 1392            }
 1393
 01394            if (item is IItemByName byName)
 1395            {
 01396                return byName.GetTaggedItems(new InternalItemsQuery(user)
 01397                {
 01398                    IsFolder = false,
 01399                    Recursive = true,
 01400                    DtoOptions = new DtoOptions(false)
 01401                    {
 01402                        EnableImages = false,
 01403                        Fields = new[]
 01404                        {
 01405                            ItemFields.SortName
 01406                        }
 01407                    },
 01408                    IsVirtualItem = false,
 01409                    OrderBy = new[] { (ItemSortBy.SortName, SortOrder.Ascending) }
 01410                });
 1411            }
 1412
 01413            if (item.IsFolder)
 1414            {
 01415                var folder = (Folder)item;
 1416
 01417                return folder.GetItemList(new InternalItemsQuery(user)
 01418                {
 01419                    Recursive = true,
 01420                    IsFolder = false,
 01421                    DtoOptions = new DtoOptions(false)
 01422                    {
 01423                        EnableImages = false,
 01424                        Fields = new ItemFields[]
 01425                        {
 01426                            ItemFields.SortName
 01427                        }
 01428                    },
 01429                    IsVirtualItem = false,
 01430                    OrderBy = new[] { (ItemSortBy.SortName, SortOrder.Ascending) }
 01431                });
 1432            }
 1433
 01434            return new[] { item };
 1435        }
 1436
 1437        private List<BaseItem> TranslateItemForInstantMix(Guid id, User user)
 1438        {
 01439            var item = _libraryManager.GetItemById(id);
 1440
 01441            if (item is null)
 1442            {
 01443                _logger.LogError("A nonexistent item Id {0} was passed into TranslateItemForInstantMix", id);
 01444                return new List<BaseItem>();
 1445            }
 1446
 01447            return _musicManager.GetInstantMixFromItem(item, user, new DtoOptions(false) { EnableImages = false }).ToLis
 1448        }
 1449
 1450        /// <inheritdoc />
 1451        public Task SendBrowseCommand(string controllingSessionId, string sessionId, BrowseRequest command, Cancellation
 1452        {
 01453            var generalCommand = new GeneralCommand
 01454            {
 01455                Name = GeneralCommandType.DisplayContent,
 01456                Arguments =
 01457                {
 01458                    ["ItemId"] = command.ItemId,
 01459                    ["ItemName"] = command.ItemName,
 01460                    ["ItemType"] = command.ItemType.ToString()
 01461                }
 01462            };
 1463
 01464            return SendGeneralCommand(controllingSessionId, sessionId, generalCommand, cancellationToken);
 1465        }
 1466
 1467        /// <inheritdoc />
 1468        public Task SendPlaystateCommand(string controllingSessionId, string sessionId, PlaystateRequest command, Cancel
 1469        {
 01470            CheckDisposed();
 1471
 01472            var session = GetSessionToRemoteControl(sessionId);
 1473
 01474            if (!string.IsNullOrEmpty(controllingSessionId))
 1475            {
 01476                var controllingSession = GetSession(controllingSessionId);
 01477                AssertCanControl(session, controllingSession);
 01478                if (!controllingSession.UserId.IsEmpty())
 1479                {
 01480                    command.ControllingUserId = controllingSession.UserId.ToString("N", CultureInfo.InvariantCulture);
 1481                }
 1482            }
 1483
 01484            return SendMessageToSession(session, SessionMessageType.Playstate, command, cancellationToken);
 1485        }
 1486
 1487        private static void AssertCanControl(SessionInfo session, SessionInfo controllingSession)
 1488        {
 01489            ArgumentNullException.ThrowIfNull(session);
 1490
 01491            ArgumentNullException.ThrowIfNull(controllingSession);
 01492        }
 1493
 1494        /// <summary>
 1495        /// Sends the restart required message.
 1496        /// </summary>
 1497        /// <param name="cancellationToken">The cancellation token.</param>
 1498        /// <returns>Task.</returns>
 1499        public Task SendRestartRequiredNotification(CancellationToken cancellationToken)
 1500        {
 01501            CheckDisposed();
 1502
 01503            return SendMessageToSessions(Sessions, SessionMessageType.RestartRequired, string.Empty, cancellationToken);
 1504        }
 1505
 1506        /// <summary>
 1507        /// Adds the additional user.
 1508        /// </summary>
 1509        /// <param name="sessionId">The session identifier.</param>
 1510        /// <param name="userId">The user identifier.</param>
 1511        /// <exception cref="UnauthorizedAccessException">Cannot modify additional users without authenticating first.</
 1512        /// <exception cref="ArgumentException">The requested user is already the primary user of the session.</exceptio
 1513        public void AddAdditionalUser(string sessionId, Guid userId)
 1514        {
 01515            CheckDisposed();
 1516
 01517            var session = GetSession(sessionId);
 1518
 01519            if (session.UserId.Equals(userId))
 1520            {
 01521                throw new ArgumentException("The requested user is already the primary user of the session.");
 1522            }
 1523
 01524            if (session.AdditionalUsers.All(i => !i.UserId.Equals(userId)))
 1525            {
 01526                var user = _userManager.GetUserById(userId);
 01527                var newUser = new SessionUserInfo
 01528                {
 01529                    UserId = userId,
 01530                    UserName = user.Username
 01531                };
 1532
 01533                session.AdditionalUsers = [.. session.AdditionalUsers, newUser];
 1534            }
 01535        }
 1536
 1537        /// <summary>
 1538        /// Removes the additional user.
 1539        /// </summary>
 1540        /// <param name="sessionId">The session identifier.</param>
 1541        /// <param name="userId">The user identifier.</param>
 1542        /// <exception cref="UnauthorizedAccessException">Cannot modify additional users without authenticating first.</
 1543        /// <exception cref="ArgumentException">The requested user is already the primary user of the session.</exceptio
 1544        public void RemoveAdditionalUser(string sessionId, Guid userId)
 1545        {
 01546            CheckDisposed();
 1547
 01548            var session = GetSession(sessionId);
 1549
 01550            if (session.UserId.Equals(userId))
 1551            {
 01552                throw new ArgumentException("The requested user is already the primary user of the session.");
 1553            }
 1554
 01555            var user = session.AdditionalUsers.FirstOrDefault(i => i.UserId.Equals(userId));
 1556
 01557            if (user is not null)
 1558            {
 01559                var list = session.AdditionalUsers.ToList();
 01560                list.Remove(user);
 1561
 01562                session.AdditionalUsers = list.ToArray();
 1563            }
 01564        }
 1565
 1566        /// <summary>
 1567        /// Authenticates the new session.
 1568        /// </summary>
 1569        /// <param name="request">The authenticationrequest.</param>
 1570        /// <returns>The authentication result.</returns>
 1571        public Task<AuthenticationResult> AuthenticateNewSession(AuthenticationRequest request)
 1572        {
 151573            return AuthenticateNewSessionInternal(request, true);
 1574        }
 1575
 1576        /// <summary>
 1577        /// Directly authenticates the session without enforcing password.
 1578        /// </summary>
 1579        /// <param name="request">The authentication request.</param>
 1580        /// <returns>The authentication result.</returns>
 1581        public Task<AuthenticationResult> AuthenticateDirect(AuthenticationRequest request)
 1582        {
 01583            return AuthenticateNewSessionInternal(request, false);
 1584        }
 1585
 1586        internal async Task<AuthenticationResult> AuthenticateNewSessionInternal(AuthenticationRequest request, bool enf
 1587        {
 1588            CheckDisposed();
 1589
 1590            ArgumentException.ThrowIfNullOrEmpty(request.App);
 1591            ArgumentException.ThrowIfNullOrEmpty(request.DeviceId);
 1592            ArgumentException.ThrowIfNullOrEmpty(request.DeviceName);
 1593            ArgumentException.ThrowIfNullOrEmpty(request.AppVersion);
 1594
 1595            User user = null;
 1596            if (!request.UserId.IsEmpty())
 1597            {
 1598                user = _userManager.GetUserById(request.UserId);
 1599            }
 1600
 1601            user ??= _userManager.GetUserByName(request.Username);
 1602
 1603            if (enforcePassword)
 1604            {
 1605                user = await _userManager.AuthenticateUser(
 1606                    request.Username,
 1607                    request.Password,
 1608                    request.RemoteEndPoint,
 1609                    true).ConfigureAwait(false);
 1610            }
 1611
 1612            if (user is null)
 1613            {
 1614                await _eventManager.PublishAsync(new AuthenticationRequestEventArgs(request)).ConfigureAwait(false);
 1615                throw new AuthenticationException("Invalid username or password entered.");
 1616            }
 1617
 1618            if (!string.IsNullOrEmpty(request.DeviceId)
 1619                && !_deviceManager.CanAccessDevice(user, request.DeviceId))
 1620            {
 1621                throw new SecurityException("User is not allowed access from this device.");
 1622            }
 1623
 1624            int sessionsCount = Sessions.Count(i => i.UserId.Equals(user.Id));
 1625            int maxActiveSessions = user.MaxActiveSessions;
 1626            _logger.LogInformation("Current/Max sessions for user {User}: {Sessions}/{Max}", user.Username, sessionsCoun
 1627            if (maxActiveSessions >= 1 && sessionsCount >= maxActiveSessions)
 1628            {
 1629                throw new SecurityException("User is at their maximum number of sessions.");
 1630            }
 1631
 1632            var token = await GetAuthorizationToken(user, request.DeviceId, request.App, request.AppVersion, request.Dev
 1633
 1634            var session = await LogSessionActivity(
 1635                request.App,
 1636                request.AppVersion,
 1637                request.DeviceId,
 1638                request.DeviceName,
 1639                request.RemoteEndPoint,
 1640                user).ConfigureAwait(false);
 1641
 1642            var returnResult = new AuthenticationResult
 1643            {
 1644                User = _userManager.GetUserDto(user, request.RemoteEndPoint),
 1645                SessionInfo = ToSessionInfoDto(session),
 1646                AccessToken = token,
 1647                ServerId = _appHost.SystemId
 1648            };
 1649
 1650            await _eventManager.PublishAsync(new AuthenticationResultEventArgs(returnResult)).ConfigureAwait(false);
 1651            return returnResult;
 1652        }
 1653
 1654        internal async Task<string> GetAuthorizationToken(User user, string deviceId, string app, string appVersion, str
 1655        {
 1656            // This should be validated above, but if it isn't don't delete all tokens.
 1657            ArgumentException.ThrowIfNullOrEmpty(deviceId);
 1658
 1659            var existing = _deviceManager.GetDevices(
 1660                new DeviceQuery
 1661                {
 1662                    DeviceId = deviceId,
 1663                    UserId = user.Id
 1664                }).Items;
 1665
 1666            foreach (var auth in existing)
 1667            {
 1668                try
 1669                {
 1670                    // Logout any existing sessions for the user on this device
 1671                    await Logout(auth).ConfigureAwait(false);
 1672                }
 1673                catch (Exception ex)
 1674                {
 1675                    _logger.LogError(ex, "Error while logging out existing session.");
 1676                }
 1677            }
 1678
 1679            _logger.LogInformation("Creating new access token for user {0}", user.Id);
 1680            var device = await _deviceManager.CreateDevice(new Device(user.Id, app, appVersion, deviceName, deviceId)).C
 1681
 1682            return device.AccessToken;
 1683        }
 1684
 1685        /// <inheritdoc />
 1686        public async Task Logout(string accessToken)
 1687        {
 1688            CheckDisposed();
 1689
 1690            ArgumentException.ThrowIfNullOrEmpty(accessToken);
 1691
 1692            var existing = _deviceManager.GetDevices(
 1693                new DeviceQuery
 1694                {
 1695                    Limit = 1,
 1696                    AccessToken = accessToken
 1697                }).Items;
 1698
 1699            if (existing.Count > 0)
 1700            {
 1701                await Logout(existing[0]).ConfigureAwait(false);
 1702            }
 1703        }
 1704
 1705        /// <inheritdoc />
 1706        public async Task Logout(Device device)
 1707        {
 1708            CheckDisposed();
 1709
 1710            _logger.LogInformation("Logging out access token {0}", device.AccessToken);
 1711
 1712            await _deviceManager.DeleteDevice(device).ConfigureAwait(false);
 1713
 1714            var sessions = Sessions
 1715                .Where(i => string.Equals(i.DeviceId, device.DeviceId, StringComparison.OrdinalIgnoreCase))
 1716                .ToList();
 1717
 1718            foreach (var session in sessions)
 1719            {
 1720                try
 1721                {
 1722                    await ReportSessionEnded(session.Id).ConfigureAwait(false);
 1723                }
 1724                catch (Exception ex)
 1725                {
 1726                    _logger.LogError(ex, "Error reporting session ended");
 1727                }
 1728            }
 1729        }
 1730
 1731        /// <inheritdoc />
 1732        public async Task RevokeUserTokens(Guid userId, string currentAccessToken)
 1733        {
 1734            CheckDisposed();
 1735
 1736            var existing = _deviceManager.GetDevices(new DeviceQuery
 1737            {
 1738                UserId = userId
 1739            });
 1740
 1741            foreach (var info in existing.Items)
 1742            {
 1743                if (!string.Equals(currentAccessToken, info.AccessToken, StringComparison.OrdinalIgnoreCase))
 1744                {
 1745                    await Logout(info).ConfigureAwait(false);
 1746                }
 1747            }
 1748        }
 1749
 1750        /// <summary>
 1751        /// Reports the capabilities.
 1752        /// </summary>
 1753        /// <param name="sessionId">The session identifier.</param>
 1754        /// <param name="capabilities">The capabilities.</param>
 1755        public void ReportCapabilities(string sessionId, ClientCapabilities capabilities)
 1756        {
 01757            CheckDisposed();
 1758
 01759            var session = GetSession(sessionId);
 1760
 01761            ReportCapabilities(session, capabilities, true);
 01762        }
 1763
 1764        private void ReportCapabilities(
 1765            SessionInfo session,
 1766            ClientCapabilities capabilities,
 1767            bool saveCapabilities)
 1768        {
 151769            session.Capabilities = capabilities;
 1770
 151771            if (saveCapabilities)
 1772            {
 01773                CapabilitiesChanged?.Invoke(
 01774                    this,
 01775                    new SessionEventArgs
 01776                    {
 01777                        SessionInfo = session
 01778                    });
 1779
 01780                _deviceManager.SaveCapabilities(session.DeviceId, capabilities);
 1781            }
 151782        }
 1783
 1784        /// <summary>
 1785        /// Converts a BaseItem to a BaseItemInfo.
 1786        /// </summary>
 1787        private BaseItemDto GetItemInfo(BaseItem item, MediaSourceInfo mediaSource)
 1788        {
 01789            ArgumentNullException.ThrowIfNull(item);
 1790
 01791            var dtoOptions = _itemInfoDtoOptions;
 1792
 01793            if (_itemInfoDtoOptions is null)
 1794            {
 01795                dtoOptions = new DtoOptions
 01796                {
 01797                    AddProgramRecordingInfo = false
 01798                };
 1799
 01800                var fields = dtoOptions.Fields.ToList();
 1801
 01802                fields.Remove(ItemFields.CanDelete);
 01803                fields.Remove(ItemFields.CanDownload);
 01804                fields.Remove(ItemFields.ChildCount);
 01805                fields.Remove(ItemFields.CustomRating);
 01806                fields.Remove(ItemFields.DateLastMediaAdded);
 01807                fields.Remove(ItemFields.DateLastRefreshed);
 01808                fields.Remove(ItemFields.DateLastSaved);
 01809                fields.Remove(ItemFields.DisplayPreferencesId);
 01810                fields.Remove(ItemFields.Etag);
 01811                fields.Remove(ItemFields.ItemCounts);
 01812                fields.Remove(ItemFields.MediaSourceCount);
 01813                fields.Remove(ItemFields.MediaStreams);
 01814                fields.Remove(ItemFields.MediaSources);
 01815                fields.Remove(ItemFields.People);
 01816                fields.Remove(ItemFields.PlayAccess);
 01817                fields.Remove(ItemFields.People);
 01818                fields.Remove(ItemFields.ProductionLocations);
 01819                fields.Remove(ItemFields.RecursiveItemCount);
 01820                fields.Remove(ItemFields.RemoteTrailers);
 01821                fields.Remove(ItemFields.SeasonUserData);
 01822                fields.Remove(ItemFields.Settings);
 01823                fields.Remove(ItemFields.SortName);
 01824                fields.Remove(ItemFields.Tags);
 01825                fields.Remove(ItemFields.ExtraIds);
 1826
 01827                dtoOptions.Fields = fields.ToArray();
 1828
 01829                _itemInfoDtoOptions = dtoOptions;
 1830            }
 1831
 01832            var info = _dtoService.GetBaseItemDto(item, dtoOptions);
 1833
 01834            if (mediaSource is not null)
 1835            {
 01836                info.MediaStreams = mediaSource.MediaStreams.ToArray();
 1837            }
 1838
 01839            return info;
 1840        }
 1841
 1842        private string GetImageCacheTag(User user)
 1843        {
 1844            try
 1845            {
 01846                return _imageProcessor.GetImageCacheTag(user);
 1847            }
 01848            catch (Exception e)
 1849            {
 01850                _logger.LogError(e, "Error getting image information for profile image");
 01851                return null;
 1852            }
 01853        }
 1854
 1855        /// <inheritdoc />
 1856        public void ReportNowViewingItem(string sessionId, string itemId)
 1857        {
 01858            ArgumentException.ThrowIfNullOrEmpty(itemId);
 1859
 01860            var item = _libraryManager.GetItemById(new Guid(itemId));
 01861            var session = GetSession(sessionId);
 1862
 01863            session.NowViewingItem = GetItemInfo(item, null);
 01864        }
 1865
 1866        /// <inheritdoc />
 1867        public void ReportTranscodingInfo(string deviceId, TranscodingInfo info)
 1868        {
 01869            var session = Sessions.FirstOrDefault(i =>
 01870                string.Equals(i.DeviceId, deviceId, StringComparison.OrdinalIgnoreCase));
 1871
 01872            if (session is not null)
 1873            {
 01874                session.TranscodingInfo = info;
 1875            }
 01876        }
 1877
 1878        /// <inheritdoc />
 1879        public void ClearTranscodingInfo(string deviceId)
 1880        {
 01881            ReportTranscodingInfo(deviceId, null);
 01882        }
 1883
 1884        /// <inheritdoc />
 1885        public SessionInfo GetSession(string deviceId, string client, string version)
 1886        {
 01887            return Sessions.FirstOrDefault(i =>
 01888                string.Equals(i.DeviceId, deviceId, StringComparison.OrdinalIgnoreCase)
 01889                    && string.Equals(i.Client, client, StringComparison.OrdinalIgnoreCase));
 1890        }
 1891
 1892        /// <inheritdoc />
 1893        public Task<SessionInfo> GetSessionByAuthenticationToken(Device info, string deviceId, string remoteEndpoint, st
 1894        {
 01895            ArgumentNullException.ThrowIfNull(info);
 1896
 01897            var user = info.UserId.IsEmpty()
 01898                ? null
 01899                : _userManager.GetUserById(info.UserId);
 1900
 01901            appVersion = string.IsNullOrEmpty(appVersion)
 01902                ? info.AppVersion
 01903                : appVersion;
 1904
 01905            var deviceName = info.DeviceName;
 01906            var appName = info.AppName;
 1907
 01908            if (string.IsNullOrEmpty(deviceId))
 1909            {
 01910                deviceId = info.DeviceId;
 1911            }
 1912
 1913            // Prevent argument exception
 01914            if (string.IsNullOrEmpty(appVersion))
 1915            {
 01916                appVersion = "1";
 1917            }
 1918
 01919            return LogSessionActivity(appName, appVersion, deviceId, deviceName, remoteEndpoint, user);
 1920        }
 1921
 1922        /// <inheritdoc />
 1923        public async Task<SessionInfo> GetSessionByAuthenticationToken(string token, string deviceId, string remoteEndpo
 1924        {
 1925            var items = _deviceManager.GetDevices(new DeviceQuery
 1926            {
 1927                AccessToken = token,
 1928                Limit = 1
 1929            }).Items;
 1930
 1931            if (items.Count == 0)
 1932            {
 1933                return null;
 1934            }
 1935
 1936            return await GetSessionByAuthenticationToken(items[0], deviceId, remoteEndpoint, null).ConfigureAwait(false)
 1937        }
 1938
 1939        /// <inheritdoc/>
 1940        public IReadOnlyList<SessionInfoDto> GetSessions(
 1941            Guid userId,
 1942            string deviceId,
 1943            int? activeWithinSeconds,
 1944            Guid? controllableUserToCheck,
 1945            bool isApiKey)
 1946        {
 01947            var result = Sessions;
 01948            if (!string.IsNullOrEmpty(deviceId))
 1949            {
 01950                result = result.Where(i => string.Equals(i.DeviceId, deviceId, StringComparison.OrdinalIgnoreCase));
 1951            }
 1952
 01953            var userCanControlOthers = false;
 01954            var userIsAdmin = false;
 01955            User user = null;
 1956
 01957            if (isApiKey)
 1958            {
 01959                userCanControlOthers = true;
 01960                userIsAdmin = true;
 1961            }
 01962            else if (!userId.IsEmpty())
 1963            {
 01964                user = _userManager.GetUserById(userId);
 01965                if (user is not null)
 1966                {
 01967                    userCanControlOthers = user.HasPermission(PermissionKind.EnableRemoteControlOfOtherUsers);
 01968                    userIsAdmin = user.HasPermission(PermissionKind.IsAdministrator);
 1969                }
 1970                else
 1971                {
 01972                    return [];
 1973                }
 1974            }
 1975
 01976            if (!controllableUserToCheck.IsNullOrEmpty())
 1977            {
 01978                result = result.Where(i => i.SupportsRemoteControl);
 1979
 01980                var controlledUser = _userManager.GetUserById(controllableUserToCheck.Value);
 01981                if (controlledUser is null)
 1982                {
 01983                    return [];
 1984                }
 1985
 01986                if (!controlledUser.HasPermission(PermissionKind.EnableSharedDeviceControl))
 1987                {
 1988                    // Controlled user has device sharing disabled
 01989                    result = result.Where(i => !i.UserId.IsEmpty());
 1990                }
 1991
 01992                if (!userCanControlOthers)
 1993                {
 1994                    // User cannot control other user's sessions, validate user id.
 01995                    result = result.Where(i => i.UserId.IsEmpty() || i.ContainsUser(userId));
 1996                }
 1997
 01998                result = result.Where(i =>
 01999                {
 02000                    if (isApiKey)
 02001                    {
 02002                        return true;
 02003                    }
 02004
 02005                    if (user is null)
 02006                    {
 02007                        return false;
 02008                    }
 02009
 02010                    return string.IsNullOrWhiteSpace(i.DeviceId) || _deviceManager.CanAccessDevice(user, i.DeviceId);
 02011                });
 2012            }
 02013            else if (!userIsAdmin)
 2014            {
 2015                // Request isn't from administrator, limit to "own" sessions.
 02016                result = result.Where(i => i.UserId.IsEmpty() || i.ContainsUser(userId));
 2017            }
 2018
 02019            if (!userIsAdmin)
 2020            {
 2021                // Don't report acceleration type for non-admin users.
 02022                result = result.Select(r =>
 02023                {
 02024                    if (r.TranscodingInfo is not null)
 02025                    {
 02026                        r.TranscodingInfo.HardwareAccelerationType = HardwareAccelerationType.none;
 02027                    }
 02028
 02029                    return r;
 02030                });
 2031            }
 2032
 02033            if (activeWithinSeconds.HasValue && activeWithinSeconds.Value > 0)
 2034            {
 02035                var minActiveDate = DateTime.UtcNow.AddSeconds(0 - activeWithinSeconds.Value);
 02036                result = result.Where(i => i.LastActivityDate >= minActiveDate);
 2037            }
 2038
 02039            return result.Select(ToSessionInfoDto).ToList();
 2040        }
 2041
 2042        /// <inheritdoc />
 2043        public Task SendMessageToAdminSessions<T>(SessionMessageType name, T data, CancellationToken cancellationToken)
 2044        {
 02045            CheckDisposed();
 2046
 02047            var adminUserIds = _userManager.Users
 02048                .Where(i => i.HasPermission(PermissionKind.IsAdministrator))
 02049                .Select(i => i.Id)
 02050                .ToList();
 2051
 02052            return SendMessageToUserSessions(adminUserIds, name, data, cancellationToken);
 2053        }
 2054
 2055        /// <inheritdoc />
 2056        public Task SendMessageToUserSessions<T>(List<Guid> userIds, SessionMessageType name, Func<T> dataFn, Cancellati
 2057        {
 02058            CheckDisposed();
 2059
 02060            var sessions = Sessions.Where(i => userIds.Any(i.ContainsUser)).ToList();
 2061
 02062            if (sessions.Count == 0)
 2063            {
 02064                return Task.CompletedTask;
 2065            }
 2066
 02067            return SendMessageToSessions(sessions, name, dataFn(), cancellationToken);
 2068        }
 2069
 2070        /// <inheritdoc />
 2071        public Task SendMessageToUserSessions<T>(List<Guid> userIds, SessionMessageType name, T data, CancellationToken 
 2072        {
 02073            CheckDisposed();
 2074
 02075            var sessions = Sessions.Where(i => userIds.Any(i.ContainsUser));
 02076            return SendMessageToSessions(sessions, name, data, cancellationToken);
 2077        }
 2078
 2079        /// <inheritdoc />
 2080        public Task SendMessageToUserDeviceSessions<T>(string deviceId, SessionMessageType name, T data, CancellationTok
 2081        {
 02082            CheckDisposed();
 2083
 02084            var sessions = Sessions.Where(i => string.Equals(i.DeviceId, deviceId, StringComparison.OrdinalIgnoreCase));
 2085
 02086            return SendMessageToSessions(sessions, name, data, cancellationToken);
 2087        }
 2088
 2089        /// <inheritdoc />
 2090        public async ValueTask DisposeAsync()
 2091        {
 2092            if (_disposed)
 2093            {
 2094                return;
 2095            }
 2096
 2097            foreach (var session in _activeConnections.Values)
 2098            {
 2099                await session.DisposeAsync().ConfigureAwait(false);
 2100            }
 2101
 2102            if (_idleTimer is not null)
 2103            {
 2104                await _idleTimer.DisposeAsync().ConfigureAwait(false);
 2105                _idleTimer = null;
 2106            }
 2107
 2108            if (_inactiveTimer is not null)
 2109            {
 2110                await _inactiveTimer.DisposeAsync().ConfigureAwait(false);
 2111                _inactiveTimer = null;
 2112            }
 2113
 2114            await _shutdownCallback.DisposeAsync().ConfigureAwait(false);
 2115
 2116            _deviceManager.DeviceOptionsUpdated -= OnDeviceManagerDeviceOptionsUpdated;
 2117            _disposed = true;
 2118        }
 2119
 2120        private async void OnApplicationStopping()
 2121        {
 2122            _logger.LogInformation("Sending shutdown notifications");
 2123            try
 2124            {
 2125                var messageType = _appHost.ShouldRestart ? SessionMessageType.ServerRestarting : SessionMessageType.Serv
 2126
 2127                await SendMessageToSessions(Sessions, messageType, string.Empty, CancellationToken.None).ConfigureAwait(
 2128            }
 2129            catch (Exception ex)
 2130            {
 2131                _logger.LogError(ex, "Error sending server shutdown notifications");
 2132            }
 2133
 2134            // Close open websockets to allow Kestrel to shut down cleanly
 2135            foreach (var session in _activeConnections.Values)
 2136            {
 2137                await session.DisposeAsync().ConfigureAwait(false);
 2138            }
 2139
 2140            _activeConnections.Clear();
 2141            _activeLiveStreamSessions.Clear();
 2142        }
 2143    }
 2144}

Methods/Properties

.ctor(Microsoft.Extensions.Logging.ILogger`1<Emby.Server.Implementations.Session.SessionManager>,MediaBrowser.Controller.Events.IEventManager,MediaBrowser.Controller.Library.IUserDataManager,MediaBrowser.Controller.Configuration.IServerConfigurationManager,MediaBrowser.Controller.Library.ILibraryManager,MediaBrowser.Controller.Library.IUserManager,MediaBrowser.Controller.Library.IMusicManager,MediaBrowser.Controller.Dto.IDtoService,MediaBrowser.Controller.Drawing.IImageProcessor,MediaBrowser.Controller.IServerApplicationHost,MediaBrowser.Controller.Devices.IDeviceManager,MediaBrowser.Controller.Library.IMediaSourceManager,Microsoft.Extensions.Hosting.IHostApplicationLifetime)
get_Sessions()
OnDeviceManagerDeviceOptionsUpdated(System.Object,Jellyfin.Data.Events.GenericEventArgs`1<System.Tuple`2<System.String,Jellyfin.Database.Implementations.Entities.Security.DeviceOptions>>)
CheckDisposed()
OnSessionStarted(MediaBrowser.Controller.Session.SessionInfo)
UpdateDeviceName(System.String,System.String)
OnSessionControllerConnected(MediaBrowser.Controller.Session.SessionInfo)
GetMediaSource(MediaBrowser.Controller.Entities.BaseItem,System.String,System.String)
RemoveNowPlayingItem(MediaBrowser.Controller.Session.SessionInfo)
GetSessionKey(System.String,System.String)
GetSessionInfo(System.String,System.String,System.String,System.String,System.String,Jellyfin.Database.Implementations.Entities.User)
CreateSession(System.String,System.String,System.String,System.String,System.String,System.String,Jellyfin.Database.Implementations.Entities.User)
GetUsers(MediaBrowser.Controller.Session.SessionInfo)
StartCheckTimers()
StopIdleCheckTimer()
StopInactiveCheckTimer()
GetNowPlayingItem(MediaBrowser.Controller.Session.SessionInfo,System.Guid)
OnPlaybackStart(Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Controller.Entities.BaseItem)
OnPlaybackProgress(MediaBrowser.Model.Session.PlaybackProgressInfo)
UpdateLiveStreamActiveSessionMappings(System.String,System.String,System.String)
OnPlaybackProgress(Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Controller.Entities.BaseItem,MediaBrowser.Model.Session.PlaybackProgressInfo)
UpdatePlaybackSettings(Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Model.Session.PlaybackProgressInfo,MediaBrowser.Controller.Entities.UserItemData)
OnPlaybackStopped(Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Controller.Entities.BaseItem,System.Nullable`1<System.Int64>,System.Boolean)
GetSession(System.String,System.Boolean)
GetSessionToRemoteControl(System.String)
ToSessionInfoDto(MediaBrowser.Controller.Session.SessionInfo)
SendMessageCommand(System.String,System.String,MediaBrowser.Model.Session.MessageCommand,System.Threading.CancellationToken)
SendGeneralCommand(System.String,System.String,MediaBrowser.Model.Session.GeneralCommand,System.Threading.CancellationToken)
SendMessageToSessions(System.Collections.Generic.IEnumerable`1<MediaBrowser.Controller.Session.SessionInfo>,MediaBrowser.Model.Session.SessionMessageType,T,System.Threading.CancellationToken)
TranslateItemForPlayback(System.Guid,Jellyfin.Database.Implementations.Entities.User)
TranslateItemForInstantMix(System.Guid,Jellyfin.Database.Implementations.Entities.User)
SendBrowseCommand(System.String,System.String,MediaBrowser.Model.Session.BrowseRequest,System.Threading.CancellationToken)
SendPlaystateCommand(System.String,System.String,MediaBrowser.Model.Session.PlaystateRequest,System.Threading.CancellationToken)
AssertCanControl(MediaBrowser.Controller.Session.SessionInfo,MediaBrowser.Controller.Session.SessionInfo)
SendRestartRequiredNotification(System.Threading.CancellationToken)
AddAdditionalUser(System.String,System.Guid)
RemoveAdditionalUser(System.String,System.Guid)
AuthenticateNewSession(MediaBrowser.Controller.Session.AuthenticationRequest)
AuthenticateDirect(MediaBrowser.Controller.Session.AuthenticationRequest)
ReportCapabilities(System.String,MediaBrowser.Model.Session.ClientCapabilities)
ReportCapabilities(MediaBrowser.Controller.Session.SessionInfo,MediaBrowser.Model.Session.ClientCapabilities,System.Boolean)
GetItemInfo(MediaBrowser.Controller.Entities.BaseItem,MediaBrowser.Model.Dto.MediaSourceInfo)
GetImageCacheTag(Jellyfin.Database.Implementations.Entities.User)
ReportNowViewingItem(System.String,System.String)
ReportTranscodingInfo(System.String,MediaBrowser.Model.Session.TranscodingInfo)
ClearTranscodingInfo(System.String)
GetSession(System.String,System.String,System.String)
GetSessionByAuthenticationToken(Jellyfin.Database.Implementations.Entities.Security.Device,System.String,System.String,System.String)
GetSessions(System.Guid,System.String,System.Nullable`1<System.Int32>,System.Nullable`1<System.Guid>,System.Boolean)
SendMessageToAdminSessions(MediaBrowser.Model.Session.SessionMessageType,T,System.Threading.CancellationToken)
SendMessageToUserSessions(System.Collections.Generic.List`1<System.Guid>,MediaBrowser.Model.Session.SessionMessageType,System.Func`1<T>,System.Threading.CancellationToken)
SendMessageToUserSessions(System.Collections.Generic.List`1<System.Guid>,MediaBrowser.Model.Session.SessionMessageType,T,System.Threading.CancellationToken)
SendMessageToUserDeviceSessions(System.String,MediaBrowser.Model.Session.SessionMessageType,T,System.Threading.CancellationToken)