using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.Linq; using System.Threading.Tasks; using Jellyfin.Data.Entities; using Jellyfin.Data.Entities.Security; using Jellyfin.Data.Enums; using Jellyfin.Data.Events; using Jellyfin.Data.Queries; using Jellyfin.Extensions; using MediaBrowser.Common.Extensions; using MediaBrowser.Controller.Devices; using MediaBrowser.Controller.Library; using MediaBrowser.Model.Devices; using MediaBrowser.Model.Querying; using MediaBrowser.Model.Session; using Microsoft.EntityFrameworkCore; namespace Jellyfin.Server.Implementations.Devices { /// /// Manages the creation, updating, and retrieval of devices. /// public class DeviceManager : IDeviceManager { private readonly IDbContextFactory _dbProvider; private readonly IUserManager _userManager; private readonly ConcurrentDictionary _capabilitiesMap = new(); /// /// Initializes a new instance of the class. /// /// The database provider. /// The user manager. public DeviceManager(IDbContextFactory dbProvider, IUserManager userManager) { _dbProvider = dbProvider; _userManager = userManager; } /// public event EventHandler>>? DeviceOptionsUpdated; /// public void SaveCapabilities(string deviceId, ClientCapabilities capabilities) { _capabilitiesMap[deviceId] = capabilities; } /// public async Task UpdateDeviceOptions(string deviceId, string deviceName) { DeviceOptions? deviceOptions; var dbContext = await _dbProvider.CreateDbContextAsync().ConfigureAwait(false); await using (dbContext.ConfigureAwait(false)) { deviceOptions = await dbContext.DeviceOptions.FirstOrDefaultAsync(dev => dev.DeviceId == deviceId).ConfigureAwait(false); if (deviceOptions is null) { deviceOptions = new DeviceOptions(deviceId); dbContext.DeviceOptions.Add(deviceOptions); } deviceOptions.CustomName = deviceName; await dbContext.SaveChangesAsync().ConfigureAwait(false); } DeviceOptionsUpdated?.Invoke(this, new GenericEventArgs>(new Tuple(deviceId, deviceOptions))); } /// public async Task CreateDevice(Device device) { var dbContext = await _dbProvider.CreateDbContextAsync().ConfigureAwait(false); await using (dbContext.ConfigureAwait(false)) { dbContext.Devices.Add(device); await dbContext.SaveChangesAsync().ConfigureAwait(false); } return device; } /// public async Task GetDeviceOptions(string deviceId) { var dbContext = await _dbProvider.CreateDbContextAsync().ConfigureAwait(false); DeviceOptions? deviceOptions; await using (dbContext.ConfigureAwait(false)) { deviceOptions = await dbContext.DeviceOptions .AsNoTracking() .FirstOrDefaultAsync(d => d.DeviceId == deviceId) .ConfigureAwait(false); } return deviceOptions ?? new DeviceOptions(deviceId); } /// public ClientCapabilities GetCapabilities(string deviceId) { return _capabilitiesMap.TryGetValue(deviceId, out ClientCapabilities? result) ? result : new ClientCapabilities(); } /// public async Task GetDevice(string id) { Device? device; var dbContext = await _dbProvider.CreateDbContextAsync().ConfigureAwait(false); await using (dbContext.ConfigureAwait(false)) { device = await dbContext.Devices .Where(d => d.DeviceId == id) .OrderByDescending(d => d.DateLastActivity) .Include(d => d.User) .FirstOrDefaultAsync() .ConfigureAwait(false); } var deviceInfo = device is null ? null : ToDeviceInfo(device); return deviceInfo; } /// public async Task> GetDevices(DeviceQuery query) { var dbContext = await _dbProvider.CreateDbContextAsync().ConfigureAwait(false); await using (dbContext.ConfigureAwait(false)) { var devices = dbContext.Devices .OrderBy(d => d.Id) .Where(device => !query.UserId.HasValue || device.UserId.Equals(query.UserId.Value)) .Where(device => query.DeviceId == null || device.DeviceId == query.DeviceId) .Where(device => query.AccessToken == null || device.AccessToken == query.AccessToken); var count = await devices.CountAsync().ConfigureAwait(false); if (query.Skip.HasValue) { devices = devices.Skip(query.Skip.Value); } if (query.Limit.HasValue) { devices = devices.Take(query.Limit.Value); } return new QueryResult(query.Skip, count, await devices.ToListAsync().ConfigureAwait(false)); } } /// public async Task> GetDeviceInfos(DeviceQuery query) { var devices = await GetDevices(query).ConfigureAwait(false); return new QueryResult( devices.StartIndex, devices.TotalRecordCount, devices.Items.Select(device => ToDeviceInfo(device)).ToList()); } /// public async Task> GetDevicesForUser(Guid? userId, bool? supportsSync) { var dbContext = await _dbProvider.CreateDbContextAsync().ConfigureAwait(false); await using (dbContext.ConfigureAwait(false)) { IAsyncEnumerable sessions = dbContext.Devices .Include(d => d.User) .OrderByDescending(d => d.DateLastActivity) .ThenBy(d => d.DeviceId) .AsAsyncEnumerable(); IAsyncEnumerable deviceOptions = dbContext.DeviceOptions.AsAsyncEnumerable(); if (supportsSync.HasValue) { sessions = sessions.Where(i => GetCapabilities(i.DeviceId).SupportsSync == supportsSync.Value); } if (userId.HasValue) { var user = _userManager.GetUserById(userId.Value); if (user is null) { throw new ResourceNotFoundException(); } sessions = sessions.Where(i => CanAccessDevice(user, i.DeviceId)); } var array = await sessions.Select(device => ToDeviceInfo(device)).ToArrayAsync().ConfigureAwait(false); await foreach (var deviceOption in deviceOptions) { var deviceInfo = array.FirstOrDefault(d => d.Id.Equals(deviceOption.DeviceId, StringComparison.OrdinalIgnoreCase)); if (deviceInfo != null) { deviceInfo.CustomName = deviceOption.CustomName; } } return new QueryResult(array); } } /// public async Task DeleteDevice(Device device) { var dbContext = await _dbProvider.CreateDbContextAsync().ConfigureAwait(false); await using (dbContext.ConfigureAwait(false)) { dbContext.Devices.Remove(device); await dbContext.SaveChangesAsync().ConfigureAwait(false); } } /// public bool CanAccessDevice(User user, string deviceId) { ArgumentNullException.ThrowIfNull(user); ArgumentException.ThrowIfNullOrEmpty(deviceId); if (user.HasPermission(PermissionKind.EnableAllDevices) || user.HasPermission(PermissionKind.IsAdministrator)) { return true; } return user.GetPreference(PreferenceKind.EnabledDevices).Contains(deviceId, StringComparison.OrdinalIgnoreCase) || !GetCapabilities(deviceId).SupportsPersistentIdentifier; } private DeviceInfo ToDeviceInfo(Device authInfo) { var caps = GetCapabilities(authInfo.DeviceId); return new DeviceInfo { AppName = authInfo.AppName, AppVersion = authInfo.AppVersion, Id = authInfo.DeviceId, LastUserId = authInfo.UserId, LastUserName = authInfo.User.Username, Name = authInfo.DeviceName, DateLastActivity = authInfo.DateLastActivity, IconUrl = caps.IconUrl }; } } }