| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526 | #pragma warning disable CS1591using System;using System.Collections.Frozen;using System.Collections.Generic;using System.Globalization;using System.IO;using System.Linq;using Jellyfin.Data.Enums;using Jellyfin.Database.Implementations.Entities;using Jellyfin.Extensions;using MediaBrowser.Common;using MediaBrowser.Controller.Channels;using MediaBrowser.Controller.Chapters;using MediaBrowser.Controller.Drawing;using MediaBrowser.Controller.Dto;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Entities.Audio;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.LiveTv;using MediaBrowser.Controller.Playlists;using MediaBrowser.Controller.Providers;using MediaBrowser.Controller.Trickplay;using MediaBrowser.Model.Dto;using MediaBrowser.Model.Entities;using MediaBrowser.Model.Querying;using Microsoft.Extensions.Logging;using Book = MediaBrowser.Controller.Entities.Book;using Episode = MediaBrowser.Controller.Entities.TV.Episode;using Movie = MediaBrowser.Controller.Entities.Movies.Movie;using MusicAlbum = MediaBrowser.Controller.Entities.Audio.MusicAlbum;using Person = MediaBrowser.Controller.Entities.Person;using Photo = MediaBrowser.Controller.Entities.Photo;using Season = MediaBrowser.Controller.Entities.TV.Season;using Series = MediaBrowser.Controller.Entities.TV.Series;namespace Emby.Server.Implementations.Dto{    public class DtoService : IDtoService    {        private static readonly FrozenDictionary<BaseItemKind, BaseItemKind[]> _relatedItemKinds = new Dictionary<BaseItemKind, BaseItemKind[]>        {            {                BaseItemKind.Genre, [                    BaseItemKind.Audio,                    BaseItemKind.Episode,                    BaseItemKind.Movie,                    BaseItemKind.LiveTvProgram,                    BaseItemKind.MusicAlbum,                    BaseItemKind.MusicArtist,                    BaseItemKind.MusicVideo,                    BaseItemKind.Series,                    BaseItemKind.Trailer                ]            },            {                BaseItemKind.MusicArtist, [                    BaseItemKind.Audio,                    BaseItemKind.MusicAlbum,                    BaseItemKind.MusicVideo                ]            },            {                BaseItemKind.MusicGenre, [                    BaseItemKind.Audio,                    BaseItemKind.MusicAlbum,                    BaseItemKind.MusicArtist,                    BaseItemKind.MusicVideo                ]            },            {                BaseItemKind.Person, [                    BaseItemKind.Audio,                    BaseItemKind.Episode,                    BaseItemKind.Movie,                    BaseItemKind.LiveTvProgram,                    BaseItemKind.MusicAlbum,                    BaseItemKind.MusicArtist,                    BaseItemKind.MusicVideo,                    BaseItemKind.Series,                    BaseItemKind.Trailer                ]            },            {                BaseItemKind.Studio, [                    BaseItemKind.Audio,                    BaseItemKind.Episode,                    BaseItemKind.Movie,                    BaseItemKind.LiveTvProgram,                    BaseItemKind.MusicAlbum,                    BaseItemKind.MusicArtist,                    BaseItemKind.MusicVideo,                    BaseItemKind.Series,                    BaseItemKind.Trailer                ]            },            {                BaseItemKind.Year, [                    BaseItemKind.Audio,                    BaseItemKind.Episode,                    BaseItemKind.Movie,                    BaseItemKind.LiveTvProgram,                    BaseItemKind.MusicAlbum,                    BaseItemKind.MusicArtist,                    BaseItemKind.MusicVideo,                    BaseItemKind.Series,                    BaseItemKind.Trailer                ]            }        }.ToFrozenDictionary();        private readonly ILogger<DtoService> _logger;        private readonly ILibraryManager _libraryManager;        private readonly IUserDataManager _userDataRepository;        private readonly IImageProcessor _imageProcessor;        private readonly IProviderManager _providerManager;        private readonly IRecordingsManager _recordingsManager;        private readonly IApplicationHost _appHost;        private readonly IMediaSourceManager _mediaSourceManager;        private readonly Lazy<ILiveTvManager> _livetvManagerFactory;        private readonly ITrickplayManager _trickplayManager;        private readonly IChapterManager _chapterManager;        public DtoService(            ILogger<DtoService> logger,            ILibraryManager libraryManager,            IUserDataManager userDataRepository,            IImageProcessor imageProcessor,            IProviderManager providerManager,            IRecordingsManager recordingsManager,            IApplicationHost appHost,            IMediaSourceManager mediaSourceManager,            Lazy<ILiveTvManager> livetvManagerFactory,            ITrickplayManager trickplayManager,            IChapterManager chapterManager)        {            _logger = logger;            _libraryManager = libraryManager;            _userDataRepository = userDataRepository;            _imageProcessor = imageProcessor;            _providerManager = providerManager;            _recordingsManager = recordingsManager;            _appHost = appHost;            _mediaSourceManager = mediaSourceManager;            _livetvManagerFactory = livetvManagerFactory;            _trickplayManager = trickplayManager;            _chapterManager = chapterManager;        }        private ILiveTvManager LivetvManager => _livetvManagerFactory.Value;        /// <inheritdoc />        public IReadOnlyList<BaseItemDto> GetBaseItemDtos(IReadOnlyList<BaseItem> items, DtoOptions options, User? user = null, BaseItem? owner = null)        {            var accessibleItems = user is null ? items : items.Where(x => x.IsVisible(user)).ToList();            var returnItems = new BaseItemDto[accessibleItems.Count];            List<(BaseItem, BaseItemDto)>? programTuples = null;            List<(BaseItemDto, LiveTvChannel)>? channelTuples = null;            for (int index = 0; index < accessibleItems.Count; index++)            {                var item = accessibleItems[index];                var dto = GetBaseItemDtoInternal(item, options, user, owner);                if (item is LiveTvChannel tvChannel)                {                    (channelTuples ??= []).Add((dto, tvChannel));                }                else if (item is LiveTvProgram)                {                    (programTuples ??= []).Add((item, dto));                }                if (options.ContainsField(ItemFields.ItemCounts))                {                    SetItemByNameInfo(dto, user);                }                returnItems[index] = dto;            }            if (programTuples is not null)            {                LivetvManager.AddInfoToProgramDto(programTuples, options.Fields, user).GetAwaiter().GetResult();            }            if (channelTuples is not null)            {                LivetvManager.AddChannelInfo(channelTuples, options, user);            }            return returnItems;        }        public BaseItemDto GetBaseItemDto(BaseItem item, DtoOptions options, User? user = null, BaseItem? owner = null)        {            var dto = GetBaseItemDtoInternal(item, options, user, owner);            if (item is LiveTvChannel tvChannel)            {                LivetvManager.AddChannelInfo(new[] { (dto, tvChannel) }, options, user);            }            else if (item is LiveTvProgram)            {                LivetvManager.AddInfoToProgramDto(new[] { (item, dto) }, options.Fields, user).GetAwaiter().GetResult();            }            if (options.ContainsField(ItemFields.ItemCounts))            {                SetItemByNameInfo(dto, user);            }            return dto;        }        private BaseItemDto GetBaseItemDtoInternal(BaseItem item, DtoOptions options, User? user = null, BaseItem? owner = null)        {            var dto = new BaseItemDto            {                ServerId = _appHost.SystemId            };            if (item.SourceType == SourceType.Channel)            {                dto.SourceType = item.SourceType.ToString();            }            if (options.ContainsField(ItemFields.People))            {                AttachPeople(dto, item, user);            }            if (options.ContainsField(ItemFields.PrimaryImageAspectRatio))            {                try                {                    AttachPrimaryImageAspectRatio(dto, item);                }                catch (Exception ex)                {                    // Have to use a catch-all unfortunately because some .net image methods throw plain Exceptions                    _logger.LogError(ex, "Error generating PrimaryImageAspectRatio for {ItemName}", item.Name);                }            }            if (options.ContainsField(ItemFields.DisplayPreferencesId))            {                dto.DisplayPreferencesId = item.DisplayPreferencesId.ToString("N", CultureInfo.InvariantCulture);            }            if (user is not null)            {                AttachUserSpecificInfo(dto, item, user, options);            }            if (item is IHasMediaSources                && options.ContainsField(ItemFields.MediaSources))            {                dto.MediaSources = _mediaSourceManager.GetStaticMediaSources(item, true, user).ToArray();                NormalizeMediaSourceContainers(dto);            }            if (options.ContainsField(ItemFields.Studios))            {                AttachStudios(dto, item);            }            AttachBasicFields(dto, item, owner, options);            if (options.ContainsField(ItemFields.CanDelete))            {                dto.CanDelete = user is null                    ? item.CanDelete()                    : item.CanDelete(user);            }            if (options.ContainsField(ItemFields.CanDownload))            {                dto.CanDownload = user is null                    ? item.CanDownload()                    : item.CanDownload(user);            }            if (options.ContainsField(ItemFields.Etag))            {                dto.Etag = item.GetEtag(user);            }            var activeRecording = _recordingsManager.GetActiveRecordingInfo(item.Path);            if (activeRecording is not null)            {                dto.Type = BaseItemKind.Recording;                dto.CanDownload = false;                dto.RunTimeTicks = null;                if (!string.IsNullOrEmpty(dto.SeriesName))                {                    dto.EpisodeTitle = dto.Name;                    dto.Name = dto.SeriesName;                }                LivetvManager.AddInfoToRecordingDto(item, dto, activeRecording, user);            }            if (item is Audio audio)            {                dto.HasLyrics = audio.GetMediaStreams().Any(s => s.Type == MediaStreamType.Lyric);            }            return dto;        }        private static void NormalizeMediaSourceContainers(BaseItemDto dto)        {            foreach (var mediaSource in dto.MediaSources)            {                var container = mediaSource.Container;                if (string.IsNullOrEmpty(container))                {                    continue;                }                var containers = container.Split(',');                if (containers.Length < 2)                {                    continue;                }                var path = mediaSource.Path;                string? fileExtensionContainer = null;                if (!string.IsNullOrEmpty(path))                {                    path = Path.GetExtension(path);                    if (!string.IsNullOrEmpty(path))                    {                        path = Path.GetExtension(path);                        if (!string.IsNullOrEmpty(path))                        {                            path = path.TrimStart('.');                        }                        if (!string.IsNullOrEmpty(path) && containers.Contains(path, StringComparison.OrdinalIgnoreCase))                        {                            fileExtensionContainer = path;                        }                    }                }                mediaSource.Container = fileExtensionContainer ?? containers[0];            }        }        /// <inheritdoc />        /// TODO refactor this to use the new SetItemByNameInfo.        /// Some callers already have the counts extracted so no reason to retrieve them again.        public BaseItemDto GetItemByNameDto(BaseItem item, DtoOptions options, List<BaseItem>? taggedItems, User? user = null)        {            var dto = GetBaseItemDtoInternal(item, options, user);            if (options.ContainsField(ItemFields.ItemCounts)                && taggedItems is not null                && taggedItems.Count != 0)            {                SetItemByNameInfo(item, dto, taggedItems);            }            return dto;        }        private void SetItemByNameInfo(BaseItemDto dto, User? user)        {            if (!_relatedItemKinds.TryGetValue(dto.Type, out var relatedItemKinds))            {                return;            }            var query = new InternalItemsQuery(user)            {                Recursive = true,                DtoOptions = new DtoOptions(false) { EnableImages = false },                IncludeItemTypes = relatedItemKinds            };            switch (dto.Type)            {                case BaseItemKind.Genre:                case BaseItemKind.MusicGenre:                    query.GenreIds = [dto.Id];                    break;                case BaseItemKind.MusicArtist:                    query.ArtistIds = [dto.Id];                    break;                case BaseItemKind.Person:                    query.PersonIds = [dto.Id];                    break;                case BaseItemKind.Studio:                    query.StudioIds = [dto.Id];                    break;                case BaseItemKind.Year                    when int.TryParse(dto.Name, NumberStyles.Integer, CultureInfo.InvariantCulture, out var year):                    query.Years = [year];                    break;                default:                    return;            }            var counts = _libraryManager.GetItemCounts(query);            dto.AlbumCount = counts.AlbumCount;            dto.ArtistCount = counts.ArtistCount;            dto.EpisodeCount = counts.EpisodeCount;            dto.MovieCount = counts.MovieCount;            dto.MusicVideoCount = counts.MusicVideoCount;            dto.ProgramCount = counts.ProgramCount;            dto.SeriesCount = counts.SeriesCount;            dto.SongCount = counts.SongCount;            dto.TrailerCount = counts.TrailerCount;            dto.ChildCount = counts.TotalItemCount();        }        private static void SetItemByNameInfo(BaseItem item, BaseItemDto dto, IReadOnlyList<BaseItem> taggedItems)        {            if (item is MusicArtist)            {                dto.AlbumCount = taggedItems.Count(i => i is MusicAlbum);                dto.MusicVideoCount = taggedItems.Count(i => i is MusicVideo);                dto.SongCount = taggedItems.Count(i => i is Audio);            }            else if (item is MusicGenre)            {                dto.ArtistCount = taggedItems.Count(i => i is MusicArtist);                dto.AlbumCount = taggedItems.Count(i => i is MusicAlbum);                dto.MusicVideoCount = taggedItems.Count(i => i is MusicVideo);                dto.SongCount = taggedItems.Count(i => i is Audio);            }            else            {                // This populates them all and covers Genre, Person, Studio, Year                dto.ArtistCount = taggedItems.Count(i => i is MusicArtist);                dto.AlbumCount = taggedItems.Count(i => i is MusicAlbum);                dto.EpisodeCount = taggedItems.Count(i => i is Episode);                dto.MovieCount = taggedItems.Count(i => i is Movie);                dto.TrailerCount = taggedItems.Count(i => i is Trailer);                dto.MusicVideoCount = taggedItems.Count(i => i is MusicVideo);                dto.SeriesCount = taggedItems.Count(i => i is Series);                dto.ProgramCount = taggedItems.Count(i => i is LiveTvProgram);                dto.SongCount = taggedItems.Count(i => i is Audio);            }            dto.ChildCount = taggedItems.Count;        }        /// <summary>        /// Attaches the user specific info.        /// </summary>        private void AttachUserSpecificInfo(BaseItemDto dto, BaseItem item, User user, DtoOptions options)        {            if (item.IsFolder)            {                var folder = (Folder)item;                if (options.EnableUserData)                {                    dto.UserData = _userDataRepository.GetUserDataDto(item, dto, user, options);                }                if (!dto.ChildCount.HasValue && item.SourceType == SourceType.Library)                {                    // For these types we can try to optimize and assume these values will be equal                    if (item is MusicAlbum || item is Season || item is Playlist)                    {                        dto.ChildCount = dto.RecursiveItemCount;                        var folderChildCount = folder.LinkedChildren.Length;                        // The default is an empty array, so we can't reliably use the count when it's empty                        if (folderChildCount > 0)                        {                            dto.ChildCount ??= folderChildCount;                        }                    }                    if (options.ContainsField(ItemFields.ChildCount))                    {                        dto.ChildCount ??= GetChildCount(folder, user);                    }                }                if (options.ContainsField(ItemFields.CumulativeRunTimeTicks))                {                    dto.CumulativeRunTimeTicks = item.RunTimeTicks;                }                if (options.ContainsField(ItemFields.DateLastMediaAdded))                {                    dto.DateLastMediaAdded = folder.DateLastMediaAdded;                }            }            else            {                if (options.EnableUserData)                {                    dto.UserData = _userDataRepository.GetUserDataDto(item, user);                }            }            if (options.ContainsField(ItemFields.PlayAccess))            {                dto.PlayAccess = item.GetPlayAccess(user);            }        }        private static int GetChildCount(Folder folder, User user)        {            // Right now this is too slow to calculate for top level folders on a per-user basis            // Just return something so that apps that are expecting a value won't think the folders are empty            if (folder is ICollectionFolder || folder is UserView)            {                return Random.Shared.Next(1, 10);            }            return folder.GetChildCount(user);        }        private static void SetBookProperties(BaseItemDto dto, Book item)        {            dto.SeriesName = item.SeriesName;        }        private static void SetPhotoProperties(BaseItemDto dto, Photo item)        {            dto.CameraMake = item.CameraMake;            dto.CameraModel = item.CameraModel;            dto.Software = item.Software;            dto.ExposureTime = item.ExposureTime;            dto.FocalLength = item.FocalLength;            dto.ImageOrientation = item.Orientation;            dto.Aperture = item.Aperture;            dto.ShutterSpeed = item.ShutterSpeed;            dto.Latitude = item.Latitude;            dto.Longitude = item.Longitude;            dto.Altitude = item.Altitude;            dto.IsoSpeedRating = item.IsoSpeedRating;            var album = item.AlbumEntity;            if (album is not null)            {                dto.Album = album.Name;                dto.AlbumId = album.Id;            }        }        private void SetMusicVideoProperties(BaseItemDto dto, MusicVideo item)        {            if (!string.IsNullOrEmpty(item.Album))            {                var parentAlbumIds = _libraryManager.GetItemIds(new InternalItemsQuery                {                    IncludeItemTypes = new[] { BaseItemKind.MusicAlbum },                    Name = item.Album,                    Limit = 1                });                if (parentAlbumIds.Count > 0)                {                    dto.AlbumId = parentAlbumIds[0];                }            }            dto.Album = item.Album;        }        private string[] GetImageTags(BaseItem item, List<ItemImageInfo> images)        {            return images                .Select(p => GetImageCacheTag(item, p))                .Where(i => i is not null)                .ToArray()!; // null values got filtered out        }        private string? GetImageCacheTag(BaseItem item, ItemImageInfo image)        {            try            {                return _imageProcessor.GetImageCacheTag(item, image);            }            catch (Exception ex)            {                _logger.LogError(ex, "Error getting {ImageType} image info for {Path}", image.Type, image.Path);                return null;            }        }        /// <summary>        /// Attaches People DTO's to a DTOBaseItem.        /// </summary>        /// <param name="dto">The dto.</param>        /// <param name="item">The item.</param>        /// <param name="user">The requesting user.</param>        private void AttachPeople(BaseItemDto dto, BaseItem item, User? user = null)        {            // Ordering by person type to ensure actors and artists are at the front.            // This is taking advantage of the fact that they both begin with A            // This should be improved in the future            var people = _libraryManager.GetPeople(item).OrderBy(i => i.SortOrder ?? int.MaxValue)                .ThenBy(i =>                {                    if (i.IsType(PersonKind.Actor))                    {                        return 0;                    }                    if (i.IsType(PersonKind.GuestStar))                    {                        return 1;                    }                    if (i.IsType(PersonKind.Director))                    {                        return 2;                    }                    if (i.IsType(PersonKind.Writer))                    {                        return 3;                    }                    if (i.IsType(PersonKind.Producer))                    {                        return 4;                    }                    if (i.IsType(PersonKind.Composer))                    {                        return 4;                    }                    return 10;                })                .ToList();            var list = new List<BaseItemPerson>();            Dictionary<string, Person> dictionary = people.Select(p => p.Name)                .Distinct(StringComparer.OrdinalIgnoreCase).Select(c =>                {                    try                    {                        return _libraryManager.GetPerson(c);                    }                    catch (Exception ex)                    {                        _logger.LogError(ex, "Error getting person {Name}", c);                        return null;                    }                }).Where(i => i is not null)                .Where(i => user is null || i!.IsVisible(user))                .DistinctBy(x => x!.Name, StringComparer.OrdinalIgnoreCase)                .ToDictionary(i => i!.Name, StringComparer.OrdinalIgnoreCase)!; // null values got filtered out            for (var i = 0; i < people.Count; i++)            {                var person = people[i];                var baseItemPerson = new BaseItemPerson                {                    Name = person.Name,                    Role = person.Role,                    Type = person.Type                };                if (dictionary.TryGetValue(person.Name, out Person? entity))                {                    baseItemPerson.PrimaryImageTag = GetTagAndFillBlurhash(dto, entity, ImageType.Primary);                    baseItemPerson.Id = entity.Id;                    if (dto.ImageBlurHashes is not null)                    {                        // Only add BlurHash for the person's image.                        baseItemPerson.ImageBlurHashes = [];                        foreach (var (imageType, blurHash) in dto.ImageBlurHashes)                        {                            if (blurHash is not null)                            {                                baseItemPerson.ImageBlurHashes[imageType] = [];                                foreach (var (imageId, blurHashValue) in blurHash)                                {                                    if (string.Equals(baseItemPerson.PrimaryImageTag, imageId, StringComparison.OrdinalIgnoreCase))                                    {                                        baseItemPerson.ImageBlurHashes[imageType][imageId] = blurHashValue;                                    }                                }                            }                        }                    }                    list.Add(baseItemPerson);                }            }            dto.People = list.ToArray();        }        /// <summary>        /// Attaches the studios.        /// </summary>        /// <param name="dto">The dto.</param>        /// <param name="item">The item.</param>        private void AttachStudios(BaseItemDto dto, BaseItem item)        {            dto.Studios = item.Studios                .Where(i => !string.IsNullOrEmpty(i))                .Select(i => new NameGuidPair                {                    Name = i,                    Id = _libraryManager.GetStudioId(i)                })                .ToArray();        }        private void AttachGenreItems(BaseItemDto dto, BaseItem item)        {            dto.GenreItems = item.Genres                .Where(i => !string.IsNullOrEmpty(i))                .Select(i => new NameGuidPair                {                    Name = i,                    Id = GetGenreId(i, item)                })                .ToArray();        }        private Guid GetGenreId(string name, BaseItem owner)        {            if (owner is IHasMusicGenres)            {                return _libraryManager.GetMusicGenreId(name);            }            return _libraryManager.GetGenreId(name);        }        private string? GetTagAndFillBlurhash(BaseItemDto dto, BaseItem item, ImageType imageType, int imageIndex = 0)        {            var image = item.GetImageInfo(imageType, imageIndex);            if (image is not null)            {                return GetTagAndFillBlurhash(dto, item, image);            }            return null;        }        private string? GetTagAndFillBlurhash(BaseItemDto dto, BaseItem item, ItemImageInfo image)        {            var tag = GetImageCacheTag(item, image);            if (tag is null)            {                return null;            }            if (!string.IsNullOrEmpty(image.BlurHash))            {                dto.ImageBlurHashes ??= [];                if (!dto.ImageBlurHashes.TryGetValue(image.Type, out var value))                {                    value = [];                    dto.ImageBlurHashes[image.Type] = value;                }                value[tag] = image.BlurHash;            }            return tag;        }        private string[] GetTagsAndFillBlurhashes(BaseItemDto dto, BaseItem item, ImageType imageType, int limit)        {            return GetTagsAndFillBlurhashes(dto, item, imageType, item.GetImages(imageType).Take(limit).ToList());        }        private string[] GetTagsAndFillBlurhashes(BaseItemDto dto, BaseItem item, ImageType imageType, List<ItemImageInfo> images)        {            var tags = GetImageTags(item, images);            var hashes = new Dictionary<string, string>();            for (int i = 0; i < images.Count; i++)            {                var img = images[i];                if (!string.IsNullOrEmpty(img.BlurHash))                {                    var tag = tags[i];                    hashes[tag] = img.BlurHash;                }            }            if (hashes.Count > 0)            {                dto.ImageBlurHashes ??= [];                dto.ImageBlurHashes[imageType] = hashes;            }            return tags;        }        /// <summary>        /// Sets simple property values on a DTOBaseItem.        /// </summary>        /// <param name="dto">The dto.</param>        /// <param name="item">The item.</param>        /// <param name="owner">The owner.</param>        /// <param name="options">The options.</param>        private void AttachBasicFields(BaseItemDto dto, BaseItem item, BaseItem? owner, DtoOptions options)        {            if (options.ContainsField(ItemFields.DateCreated))            {                dto.DateCreated = item.DateCreated;            }            if (options.ContainsField(ItemFields.Settings))            {                dto.LockedFields = item.LockedFields;                dto.LockData = item.IsLocked;                dto.ForcedSortName = item.ForcedSortName;            }            dto.Container = item.Container;            dto.EndDate = item.EndDate;            if (options.ContainsField(ItemFields.ExternalUrls))            {                dto.ExternalUrls = _providerManager.GetExternalUrls(item).ToArray();            }            if (options.ContainsField(ItemFields.Tags))            {                dto.Tags = item.Tags;            }            if (item is IHasAspectRatio hasAspectRatio)            {                dto.AspectRatio = hasAspectRatio.AspectRatio;            }            dto.ImageBlurHashes = [];            var backdropLimit = options.GetImageLimit(ImageType.Backdrop);            if (backdropLimit > 0)            {                dto.BackdropImageTags = GetTagsAndFillBlurhashes(dto, item, ImageType.Backdrop, backdropLimit);            }            if (options.ContainsField(ItemFields.Genres))            {                dto.Genres = item.Genres;                AttachGenreItems(dto, item);            }            if (options.EnableImages)            {                dto.ImageTags = [];                // Prevent implicitly captured closure                var currentItem = item;                foreach (var image in currentItem.ImageInfos.Where(i => !currentItem.AllowsMultipleImages(i.Type)))                {                    if (options.GetImageLimit(image.Type) > 0)                    {                        var tag = GetTagAndFillBlurhash(dto, item, image);                        if (tag is not null)                        {                            dto.ImageTags[image.Type] = tag;                        }                    }                }            }            dto.Id = item.Id;            dto.IndexNumber = item.IndexNumber;            dto.ParentIndexNumber = item.ParentIndexNumber;            if (item.IsFolder)            {                dto.IsFolder = true;            }            else if (item is IHasMediaSources)            {                dto.IsFolder = false;            }            dto.MediaType = item.MediaType;            if (item is not LiveTvProgram)            {                dto.LocationType = item.LocationType;            }            dto.Audio = item.Audio;            if (options.ContainsField(ItemFields.Settings))            {                dto.PreferredMetadataCountryCode = item.PreferredMetadataCountryCode;                dto.PreferredMetadataLanguage = item.PreferredMetadataLanguage;            }            dto.CriticRating = item.CriticRating;            if (item is IHasDisplayOrder hasDisplayOrder)            {                dto.DisplayOrder = hasDisplayOrder.DisplayOrder;            }            if (item is IHasCollectionType hasCollectionType)            {                dto.CollectionType = hasCollectionType.CollectionType;            }            if (options.ContainsField(ItemFields.RemoteTrailers))            {                dto.RemoteTrailers = item.RemoteTrailers;            }            dto.Name = item.Name;            dto.OfficialRating = item.OfficialRating;            if (options.ContainsField(ItemFields.Overview))            {                dto.Overview = item.Overview;            }            if (options.ContainsField(ItemFields.OriginalTitle))            {                dto.OriginalTitle = item.OriginalTitle;            }            if (options.ContainsField(ItemFields.ParentId))            {                dto.ParentId = item.DisplayParentId;            }            AddInheritedImages(dto, item, options, owner);            if (options.ContainsField(ItemFields.Path))            {                dto.Path = GetMappedPath(item, owner);            }            if (options.ContainsField(ItemFields.EnableMediaSourceDisplay))            {                dto.EnableMediaSourceDisplay = item.EnableMediaSourceDisplay;            }            dto.PremiereDate = item.PremiereDate;            dto.ProductionYear = item.ProductionYear;            if (options.ContainsField(ItemFields.ProviderIds))            {                dto.ProviderIds = item.ProviderIds;            }            dto.RunTimeTicks = item.RunTimeTicks;            if (options.ContainsField(ItemFields.SortName))            {                dto.SortName = item.SortName;            }            if (options.ContainsField(ItemFields.CustomRating))            {                dto.CustomRating = item.CustomRating;            }            if (options.ContainsField(ItemFields.Taglines))            {                if (!string.IsNullOrEmpty(item.Tagline))                {                    dto.Taglines = new string[] { item.Tagline };                }                dto.Taglines ??= Array.Empty<string>();            }            dto.Type = item.GetBaseItemKind();            if ((item.CommunityRating ?? 0) > 0)            {                dto.CommunityRating = item.CommunityRating;            }            if (item is ISupportsPlaceHolders supportsPlaceHolders && supportsPlaceHolders.IsPlaceHolder)            {                dto.IsPlaceHolder = supportsPlaceHolders.IsPlaceHolder;            }            if (item.LUFS.HasValue)            {                // -18 LUFS reference, same as ReplayGain 2.0, compatible with ReplayGain 1.0                dto.NormalizationGain = -18f - item.LUFS;            }            else if (item.NormalizationGain.HasValue)            {                dto.NormalizationGain = item.NormalizationGain;            }            // Add audio info            if (item is Audio audio)            {                dto.Album = audio.Album;                dto.ExtraType = audio.ExtraType;                var albumParent = audio.AlbumEntity;                if (albumParent is not null)                {                    dto.AlbumId = albumParent.Id;                    dto.AlbumPrimaryImageTag = GetTagAndFillBlurhash(dto, albumParent, ImageType.Primary);                }                // if (options.ContainsField(ItemFields.MediaSourceCount))                // {                // Songs always have one                // }            }            if (item is IHasArtist hasArtist)            {                dto.Artists = hasArtist.Artists;                // var artistItems = _libraryManager.GetArtists(new InternalItemsQuery                // {                //    EnableTotalRecordCount = false,                //    ItemIds = new[] { item.Id.ToString("N", CultureInfo.InvariantCulture) }                // });                // dto.ArtistItems = artistItems.Items                //    .Select(i =>                //    {                //        var artist = i.Item1;                //        return new NameIdPair                //        {                //            Name = artist.Name,                //            Id = artist.Id.ToString("N", CultureInfo.InvariantCulture)                //        };                //    })                //    .ToList();                // Include artists that are not in the database yet, e.g., just added via metadata editor                // var foundArtists = artistItems.Items.Select(i => i.Item1.Name).ToList();                dto.ArtistItems = _libraryManager.GetArtists([.. hasArtist.Artists.Where(e => !string.IsNullOrWhiteSpace(e))])                    .Where(e => e.Value.Length > 0)                    .Select(i =>                    {                        return new NameGuidPair                        {                            Name = i.Key,                            Id = i.Value.First().Id                        };                    }).Where(i => i is not null).ToArray();            }            if (item is IHasAlbumArtist hasAlbumArtist)            {                dto.AlbumArtist = hasAlbumArtist.AlbumArtists.FirstOrDefault();                // var artistItems = _libraryManager.GetAlbumArtists(new InternalItemsQuery                // {                //    EnableTotalRecordCount = false,                //    ItemIds = new[] { item.Id.ToString("N", CultureInfo.InvariantCulture) }                // });                // dto.AlbumArtists = artistItems.Items                //    .Select(i =>                //    {                //        var artist = i.Item1;                //        return new NameIdPair                //        {                //            Name = artist.Name,                //            Id = artist.Id.ToString("N", CultureInfo.InvariantCulture)                //        };                //    })                //    .ToList();                dto.AlbumArtists = hasAlbumArtist.AlbumArtists                    // .Except(foundArtists, new DistinctNameComparer())                    .Select(i =>                    {                        // This should not be necessary but we're seeing some cases of it                        if (string.IsNullOrEmpty(i))                        {                            return null;                        }                        var artist = _libraryManager.GetArtist(i, new DtoOptions(false)                        {                            EnableImages = false                        });                        if (artist is not null)                        {                            return new NameGuidPair                            {                                Name = artist.Name,                                Id = artist.Id                            };                        }                        return null;                    }).Where(i => i is not null).ToArray();            }            // Add video info            if (item is Video video)            {                dto.VideoType = video.VideoType;                dto.Video3DFormat = video.Video3DFormat;                dto.IsoType = video.IsoType;                if (video.HasSubtitles)                {                    dto.HasSubtitles = video.HasSubtitles;                }                if (video.AdditionalParts.Length != 0)                {                    dto.PartCount = video.AdditionalParts.Length + 1;                }                if (options.ContainsField(ItemFields.MediaSourceCount))                {                    var mediaSourceCount = video.MediaSourceCount;                    if (mediaSourceCount != 1)                    {                        dto.MediaSourceCount = mediaSourceCount;                    }                }                if (options.ContainsField(ItemFields.Chapters))                {                    dto.Chapters = _chapterManager.GetChapters(item.Id).ToList();                }                if (options.ContainsField(ItemFields.Trickplay))                {                    var trickplay = _trickplayManager.GetTrickplayManifest(item).GetAwaiter().GetResult();                    dto.Trickplay = trickplay.ToDictionary(                        mediaStream => mediaStream.Key,                        mediaStream => mediaStream.Value.ToDictionary(                            width => width.Key,                            width => new TrickplayInfoDto(width.Value)));                }                dto.ExtraType = video.ExtraType;            }            if (options.ContainsField(ItemFields.MediaStreams))            {                // Add VideoInfo                if (item is IHasMediaSources)                {                    MediaStream[] mediaStreams;                    if (dto.MediaSources is not null && dto.MediaSources.Length > 0)                    {                        if (item.SourceType == SourceType.Channel)                        {                            mediaStreams = dto.MediaSources[0].MediaStreams.ToArray();                        }                        else                        {                            string id = item.Id.ToString("N", CultureInfo.InvariantCulture);                            mediaStreams = dto.MediaSources.Where(i => string.Equals(i.Id, id, StringComparison.OrdinalIgnoreCase))                                .SelectMany(i => i.MediaStreams)                                .ToArray();                        }                    }                    else                    {                        mediaStreams = _mediaSourceManager.GetStaticMediaSources(item, true)[0].MediaStreams.ToArray();                    }                    dto.MediaStreams = mediaStreams;                }            }            BaseItem[]? allExtras = null;            if (options.ContainsField(ItemFields.SpecialFeatureCount))            {                allExtras = item.GetExtras().ToArray();                dto.SpecialFeatureCount = allExtras.Count(i => i.ExtraType.HasValue && BaseItem.DisplayExtraTypes.Contains(i.ExtraType.Value));            }            if (options.ContainsField(ItemFields.LocalTrailerCount))            {                if (item is IHasTrailers hasTrailers)                {                    dto.LocalTrailerCount = hasTrailers.LocalTrailers.Count;                }                else                {                    dto.LocalTrailerCount = (allExtras ?? item.GetExtras()).Count(i => i.ExtraType == ExtraType.Trailer);                }            }            // Add EpisodeInfo            if (item is Episode episode)            {                dto.IndexNumberEnd = episode.IndexNumberEnd;                dto.SeriesName = episode.SeriesName;                if (options.ContainsField(ItemFields.SpecialEpisodeNumbers))                {                    dto.AirsAfterSeasonNumber = episode.AirsAfterSeasonNumber;                    dto.AirsBeforeEpisodeNumber = episode.AirsBeforeEpisodeNumber;                    dto.AirsBeforeSeasonNumber = episode.AirsBeforeSeasonNumber;                }                dto.SeasonName = episode.SeasonName;                dto.SeasonId = episode.SeasonId;                dto.SeriesId = episode.SeriesId;                Series? episodeSeries = null;                // this block will add the series poster for episodes without a poster                // TODO maybe remove the if statement entirely                // if (options.ContainsField(ItemFields.SeriesPrimaryImage))                {                    episodeSeries ??= episode.Series;                    if (episodeSeries is not null)                    {                        dto.SeriesPrimaryImageTag = GetTagAndFillBlurhash(dto, episodeSeries, ImageType.Primary);                        if (dto.ImageTags is null || !dto.ImageTags.ContainsKey(ImageType.Primary))                        {                            AttachPrimaryImageAspectRatio(dto, episodeSeries);                        }                    }                }                if (options.ContainsField(ItemFields.SeriesStudio))                {                    episodeSeries ??= episode.Series;                    if (episodeSeries is not null)                    {                        dto.SeriesStudio = episodeSeries.Studios.FirstOrDefault();                    }                }            }            // Add SeriesInfo            Series? series;            if (item is Series tmp)            {                series = tmp;                dto.AirDays = series.AirDays;                dto.AirTime = series.AirTime;                dto.Status = series.Status?.ToString();            }            // Add SeasonInfo            if (item is Season season)            {                dto.SeriesName = season.SeriesName;                dto.SeriesId = season.SeriesId;                series = null;                if (options.ContainsField(ItemFields.SeriesStudio))                {                    series ??= season.Series;                    if (series is not null)                    {                        dto.SeriesStudio = series.Studios.FirstOrDefault();                    }                }                // this block will add the series poster for seasons without a poster                // TODO maybe remove the if statement entirely                // if (options.ContainsField(ItemFields.SeriesPrimaryImage))                {                    series ??= season.Series;                    if (series is not null)                    {                        dto.SeriesPrimaryImageTag = GetTagAndFillBlurhash(dto, series, ImageType.Primary);                        if (dto.ImageTags is null || !dto.ImageTags.ContainsKey(ImageType.Primary))                        {                            AttachPrimaryImageAspectRatio(dto, series);                        }                    }                }            }            if (item is MusicVideo musicVideo)            {                SetMusicVideoProperties(dto, musicVideo);            }            if (item is Book book)            {                SetBookProperties(dto, book);            }            if (options.ContainsField(ItemFields.ProductionLocations))            {                if (item.ProductionLocations.Length > 0 || item is Movie)                {                    dto.ProductionLocations = item.ProductionLocations;                }            }            if (options.ContainsField(ItemFields.Width))            {                var width = item.Width;                if (width > 0)                {                    dto.Width = width;                }            }            if (options.ContainsField(ItemFields.Height))            {                var height = item.Height;                if (height > 0)                {                    dto.Height = height;                }            }            if (options.ContainsField(ItemFields.IsHD))            {                // Compatibility                if (item.IsHD)                {                    dto.IsHD = true;                }            }            if (item is Photo photo)            {                SetPhotoProperties(dto, photo);            }            dto.ChannelId = item.ChannelId;            if (item.SourceType == SourceType.Channel)            {                var channel = _libraryManager.GetItemById(item.ChannelId);                if (channel is not null)                {                    dto.ChannelName = channel.Name;                }            }        }        private BaseItem? GetImageDisplayParent(BaseItem currentItem, BaseItem originalItem)        {            if (currentItem is MusicAlbum musicAlbum)            {                var artist = musicAlbum.GetMusicArtist(new DtoOptions(false));                if (artist is not null)                {                    return artist;                }            }            var parent = currentItem.DisplayParent ?? currentItem.GetOwner() ?? currentItem.GetParent();            if (parent is null && originalItem is not UserRootFolder && originalItem is not UserView && originalItem is not AggregateFolder && originalItem is not ICollectionFolder && originalItem is not Channel)            {                parent = _libraryManager.GetCollectionFolders(originalItem).FirstOrDefault();            }            return parent;        }        private void AddInheritedImages(BaseItemDto dto, BaseItem item, DtoOptions options, BaseItem? owner)        {            if (!item.SupportsInheritedParentImages)            {                return;            }            var logoLimit = options.GetImageLimit(ImageType.Logo);            var artLimit = options.GetImageLimit(ImageType.Art);            var thumbLimit = options.GetImageLimit(ImageType.Thumb);            var backdropLimit = options.GetImageLimit(ImageType.Backdrop);            // For now. Emby apps are not using this            artLimit = 0;            if (logoLimit == 0 && artLimit == 0 && thumbLimit == 0 && backdropLimit == 0)            {                return;            }            BaseItem? parent = null;            var isFirst = true;            var imageTags = dto.ImageTags;            while ((!(imageTags is not null && imageTags.ContainsKey(ImageType.Logo)) && logoLimit > 0)                || (!(imageTags is not null && imageTags.ContainsKey(ImageType.Art)) && artLimit > 0)                || (!(imageTags is not null && imageTags.ContainsKey(ImageType.Thumb)) && thumbLimit > 0)                || parent is Series)            {                parent ??= isFirst ? GetImageDisplayParent(item, item) ?? owner : parent;                if (parent is null)                {                    break;                }                var allImages = parent.ImageInfos;                if (logoLimit > 0 && !(imageTags is not null && imageTags.ContainsKey(ImageType.Logo)) && dto.ParentLogoItemId is null)                {                    var image = allImages.FirstOrDefault(i => i.Type == ImageType.Logo);                    if (image is not null)                    {                        dto.ParentLogoItemId = parent.Id;                        dto.ParentLogoImageTag = GetTagAndFillBlurhash(dto, parent, image);                    }                }                if (artLimit > 0 && !(imageTags is not null && imageTags.ContainsKey(ImageType.Art)) && dto.ParentArtItemId is null)                {                    var image = allImages.FirstOrDefault(i => i.Type == ImageType.Art);                    if (image is not null)                    {                        dto.ParentArtItemId = parent.Id;                        dto.ParentArtImageTag = GetTagAndFillBlurhash(dto, parent, image);                    }                }                if (thumbLimit > 0 && !(imageTags is not null && imageTags.ContainsKey(ImageType.Thumb)) && (dto.ParentThumbItemId is null || parent is Series) && parent is not ICollectionFolder && parent is not UserView)                {                    var image = allImages.FirstOrDefault(i => i.Type == ImageType.Thumb);                    if (image is not null)                    {                        dto.ParentThumbItemId = parent.Id;                        dto.ParentThumbImageTag = GetTagAndFillBlurhash(dto, parent, image);                    }                }                if (backdropLimit > 0 && !((dto.BackdropImageTags is not null && dto.BackdropImageTags.Length > 0) || (dto.ParentBackdropImageTags is not null && dto.ParentBackdropImageTags.Length > 0)))                {                    var images = allImages.Where(i => i.Type == ImageType.Backdrop).Take(backdropLimit).ToList();                    if (images.Count > 0)                    {                        dto.ParentBackdropItemId = parent.Id;                        dto.ParentBackdropImageTags = GetTagsAndFillBlurhashes(dto, parent, ImageType.Backdrop, images);                    }                }                isFirst = false;                if (!parent.SupportsInheritedParentImages)                {                    break;                }                parent = GetImageDisplayParent(parent, item);            }        }        private string GetMappedPath(BaseItem item, BaseItem? ownerItem)        {            var path = item.Path;            if (item.IsFileProtocol)            {                path = _libraryManager.GetPathAfterNetworkSubstitution(path, ownerItem ?? item);            }            return path;        }        /// <summary>        /// Attaches the primary image aspect ratio.        /// </summary>        /// <param name="dto">The dto.</param>        /// <param name="item">The item.</param>        public void AttachPrimaryImageAspectRatio(IItemDto dto, BaseItem item)        {            dto.PrimaryImageAspectRatio = GetPrimaryImageAspectRatio(item);        }        public double? GetPrimaryImageAspectRatio(BaseItem item)        {            var imageInfo = item.GetImageInfo(ImageType.Primary, 0);            if (imageInfo is null)            {                return null;            }            if (!imageInfo.IsLocalFile)            {                return item.GetDefaultPrimaryImageAspectRatio();            }            try            {                var size = _imageProcessor.GetImageDimensions(item, imageInfo);                var width = size.Width;                var height = size.Height;                if (width > 0 && height > 0)                {                    return (double)width / height;                }            }            catch (Exception ex)            {                _logger.LogError(ex, "Failed to determine primary image aspect ratio for {ImagePath}", imageInfo.Path);            }            return item.GetDefaultPrimaryImageAspectRatio();        }    }}
 |