< Summary - Jellyfin

Information
Class: MediaBrowser.Controller.Entities.Folder
Assembly: MediaBrowser.Controller
File(s): /srv/git/jellyfin/MediaBrowser.Controller/Entities/Folder.cs
Line coverage
33%
Covered lines: 180
Uncovered lines: 364
Coverable lines: 544
Total lines: 1814
Line coverage: 33%
Branch coverage
29%
Covered branches: 107
Total branches: 364
Branch coverage: 29.3%
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_SupportsThemeMedia()100%210%
get_IsPreSorted()100%210%
get_IsPhysicalRoot()100%11100%
get_SupportsInheritedParentImages()100%210%
get_SupportsPlayedStatus()100%210%
get_IsFolder()100%11100%
get_IsDisplayedAsFolder()100%210%
get_SupportsCumulativeRunTimeTicks()100%11100%
get_SupportsDateLastMediaAdded()100%11100%
get_FileNameWithoutExtension()50%2266.66%
get_Children()100%11100%
get_RecursiveChildren()100%210%
get_SupportsShortcutChildren()100%11100%
get_FilterLinkedChildrenPerUser()100%11100%
get_SupportsOwnedItems()100%22100%
get_SupportsUserDataFromChildren()25%841635.71%
CanDelete()50%2266.66%
RequiresRefresh()50%4475%
AddChild(...)0%4260%
IsVisible(...)50%801222.22%
LoadChildren()100%11100%
GetRefreshProgress()100%210%
ValidateChildren(...)100%210%
ValidateChildren(...)100%11100%
GetActualChildrenDictionary()37.5%11863.63%
IsLibraryFolderAccessible(...)80%101083.33%
GetNonCachedChildren(...)100%11100%
GetCachedChildren()100%11100%
GetChildCount(...)0%2040%
GetRecursiveChildCount(...)100%210%
QueryRecursive(...)71.42%291457.14%
QueryWithPostFiltering2(...)62.5%9877.27%
RequiresPostFiltering2(...)50%6450%
RequiresPostFiltering(...)50%4203633.33%
SortItemsByRequest(...)100%210%
GetItems(...)0%4260%
GetItemList(...)16.66%13642.85%
GetItemsInternal(...)33.33%28615.78%
PostFilterAndSort(...)0%110100%
CollapseBoxSetItemsIfNeeded(...)0%620%
CollapseBoxSetItems(...)40.9%1332238.88%
AllowBoxSetCollapsing(...)0%5256720%
GetChildren(...)100%11100%
GetChildren(...)50%2283.33%
GetEligibleChildrenForRecursiveChildren(...)100%11100%
AddChildren(...)71.42%161476.92%
AddChildrenFromCollection(...)83.33%141275%
GetRecursiveChildren(...)100%210%
GetRecursiveChildren()100%210%
GetRecursiveChildren(...)100%11100%
GetRecursiveChildren(...)100%210%
GetRecursiveChildren(...)100%11100%
AddChildrenToList(...)22.22%1431827.27%
GetLinkedChildren()25%5457.14%
ContainsLinkedChildByItemId(...)0%110100%
GetLinkedChildren(...)9.09%406227.4%
GetLinkedChildrenInfos()100%210%
RefreshLinkedChildren(...)66.66%6687.8%
MarkPlayed(...)0%110100%
MarkUnplayed(...)0%620%
IsPlayed(...)100%210%
IsUnplayed(...)100%210%
FillUserDataDtoValues(...)8.33%124128%
GetProgress(...)100%11100%

File(s)

/srv/git/jellyfin/MediaBrowser.Controller/Entities/Folder.cs

#LineLine coverage
 1#nullable disable
 2
 3#pragma warning disable CA1002, CA1721, CA1819, CS1591
 4
 5using System;
 6using System.Collections.Generic;
 7using System.Collections.Immutable;
 8using System.IO;
 9using System.Linq;
 10using System.Security;
 11using System.Text.Json.Serialization;
 12using System.Threading;
 13using System.Threading.Tasks;
 14using System.Threading.Tasks.Dataflow;
 15using J2N.Collections.Generic.Extensions;
 16using Jellyfin.Data;
 17using Jellyfin.Data.Enums;
 18using Jellyfin.Database.Implementations.Entities;
 19using Jellyfin.Database.Implementations.Enums;
 20using Jellyfin.Extensions;
 21using MediaBrowser.Controller.Channels;
 22using MediaBrowser.Controller.Collections;
 23using MediaBrowser.Controller.Configuration;
 24using MediaBrowser.Controller.Dto;
 25using MediaBrowser.Controller.Entities.Audio;
 26using MediaBrowser.Controller.Entities.Movies;
 27using MediaBrowser.Controller.Library;
 28using MediaBrowser.Controller.Providers;
 29using MediaBrowser.Model.Dto;
 30using MediaBrowser.Model.IO;
 31using MediaBrowser.Model.Querying;
 32using Microsoft.Extensions.Logging;
 33using Episode = MediaBrowser.Controller.Entities.TV.Episode;
 34using MusicAlbum = MediaBrowser.Controller.Entities.Audio.MusicAlbum;
 35using Season = MediaBrowser.Controller.Entities.TV.Season;
 36using Series = MediaBrowser.Controller.Entities.TV.Series;
 37
 38namespace MediaBrowser.Controller.Entities
 39{
 40    /// <summary>
 41    /// Class Folder.
 42    /// </summary>
 43    public class Folder : BaseItem
 44    {
 38845        public Folder()
 46        {
 38847            LinkedChildren = Array.Empty<LinkedChild>();
 38848        }
 49
 50        public static IUserViewManager UserViewManager { get; set; }
 51
 52        /// <summary>
 53        /// Gets or sets a value indicating whether this instance is root.
 54        /// </summary>
 55        /// <value><c>true</c> if this instance is root; otherwise, <c>false</c>.</value>
 56        public bool IsRoot { get; set; }
 57
 58        public LinkedChild[] LinkedChildren { get; set; }
 59
 60        [JsonIgnore]
 61        public DateTime? DateLastMediaAdded { get; set; }
 62
 63        [JsonIgnore]
 064        public override bool SupportsThemeMedia => true;
 65
 66        [JsonIgnore]
 067        public virtual bool IsPreSorted => false;
 68
 69        [JsonIgnore]
 1070        public virtual bool IsPhysicalRoot => false;
 71
 72        [JsonIgnore]
 073        public override bool SupportsInheritedParentImages => true;
 74
 75        [JsonIgnore]
 076        public override bool SupportsPlayedStatus => true;
 77
 78        /// <summary>
 79        /// Gets a value indicating whether this instance is folder.
 80        /// </summary>
 81        /// <value><c>true</c> if this instance is folder; otherwise, <c>false</c>.</value>
 82        [JsonIgnore]
 69283        public override bool IsFolder => true;
 84
 85        [JsonIgnore]
 086        public override bool IsDisplayedAsFolder => true;
 87
 88        [JsonIgnore]
 8289        public virtual bool SupportsCumulativeRunTimeTicks => false;
 90
 91        [JsonIgnore]
 2992        public virtual bool SupportsDateLastMediaAdded => false;
 93
 94        [JsonIgnore]
 95        public override string FileNameWithoutExtension
 96        {
 97            get
 98            {
 28399                if (IsFileProtocol)
 100                {
 283101                    return System.IO.Path.GetFileName(Path);
 102                }
 103
 0104                return null;
 105            }
 106        }
 107
 108        /// <summary>
 109        /// Gets the actual children.
 110        /// </summary>
 111        /// <value>The actual children.</value>
 112        [JsonIgnore]
 861113        public virtual IEnumerable<BaseItem> Children => LoadChildren();
 114
 115        /// <summary>
 116        /// Gets thread-safe access to all recursive children of this folder - without regard to user.
 117        /// </summary>
 118        /// <value>The recursive children.</value>
 119        [JsonIgnore]
 0120        public IEnumerable<BaseItem> RecursiveChildren => GetRecursiveChildren();
 121
 122        [JsonIgnore]
 22123        protected virtual bool SupportsShortcutChildren => false;
 124
 10125        protected virtual bool FilterLinkedChildrenPerUser => false;
 126
 127        [JsonIgnore]
 84128        protected override bool SupportsOwnedItems => base.SupportsOwnedItems || SupportsShortcutChildren;
 129
 130        [JsonIgnore]
 131        public virtual bool SupportsUserDataFromChildren
 132        {
 133            get
 134            {
 135                // These are just far too slow.
 9136                if (this is ICollectionFolder)
 137                {
 3138                    return false;
 139                }
 140
 6141                if (this is UserView)
 142                {
 0143                    return false;
 144                }
 145
 6146                if (this is UserRootFolder)
 147                {
 6148                    return false;
 149                }
 150
 0151                if (this is Channel)
 152                {
 0153                    return false;
 154                }
 155
 0156                if (SourceType != SourceType.Library)
 157                {
 0158                    return false;
 159                }
 160
 0161                if (this is IItemByName)
 162                {
 0163                    if (this is not IHasDualAccess hasDualAccess || hasDualAccess.IsAccessedByName)
 164                    {
 0165                        return false;
 166                    }
 167                }
 168
 0169                return true;
 170            }
 171        }
 172
 173        public static ICollectionManager CollectionManager { get; set; }
 174
 175        public override bool CanDelete()
 176        {
 6177            if (IsRoot)
 178            {
 6179                return false;
 180            }
 181
 0182            return base.CanDelete();
 183        }
 184
 185        public override bool RequiresRefresh()
 186        {
 53187            var baseResult = base.RequiresRefresh();
 188
 53189            if (SupportsCumulativeRunTimeTicks && !RunTimeTicks.HasValue)
 190            {
 0191                baseResult = true;
 192            }
 193
 53194            return baseResult;
 195        }
 196
 197        /// <summary>
 198        /// Adds the child.
 199        /// </summary>
 200        /// <param name="item">The item.</param>
 201        /// <exception cref="InvalidOperationException">Unable to add  + item.Name.</exception>
 202        public void AddChild(BaseItem item)
 203        {
 0204            item.SetParent(this);
 205
 0206            if (item.Id.IsEmpty())
 207            {
 0208                item.Id = LibraryManager.GetNewItemId(item.Path, item.GetType());
 209            }
 210
 0211            if (item.DateCreated == DateTime.MinValue)
 212            {
 0213                item.DateCreated = DateTime.UtcNow;
 214            }
 215
 0216            if (item.DateModified == DateTime.MinValue)
 217            {
 0218                item.DateModified = DateTime.UtcNow;
 219            }
 220
 0221            LibraryManager.CreateItem(item, this);
 0222        }
 223
 224        public override bool IsVisible(User user, bool skipAllowedTagsCheck = false)
 225        {
 13226            if (this is ICollectionFolder && this is not BasePluginFolder)
 227            {
 0228                var blockedMediaFolders = user.GetPreferenceValues<Guid>(PreferenceKind.BlockedMediaFolders);
 0229                if (blockedMediaFolders.Length > 0)
 230                {
 0231                    if (blockedMediaFolders.Contains(Id))
 232                    {
 0233                        return false;
 234                    }
 235                }
 236                else
 237                {
 0238                    if (!user.HasPermission(PermissionKind.EnableAllFolders)
 0239                        && !user.GetPreferenceValues<Guid>(PreferenceKind.EnabledFolders).Contains(Id))
 240                    {
 0241                        return false;
 242                    }
 243                }
 244            }
 245
 13246            return base.IsVisible(user, skipAllowedTagsCheck);
 247        }
 248
 249        /// <summary>
 250        /// Loads our children.  Validation will occur externally.
 251        /// We want this synchronous.
 252        /// </summary>
 253        /// <returns>Returns children.</returns>
 254        protected virtual IReadOnlyList<BaseItem> LoadChildren()
 255        {
 256            // logger.LogDebug("Loading children from {0} {1} {2}", GetType().Name, Id, Path);
 257            // just load our children from the repo - the library will be validated and maintained in other processes
 142258            return GetCachedChildren();
 259        }
 260
 261        public override double? GetRefreshProgress()
 262        {
 0263            return ProviderManager.GetRefreshProgress(Id);
 264        }
 265
 266        public Task ValidateChildren(IProgress<double> progress, CancellationToken cancellationToken)
 267        {
 0268            return ValidateChildren(progress, new MetadataRefreshOptions(new DirectoryService(FileSystem)), cancellation
 269        }
 270
 271        /// <summary>
 272        /// Validates that the children of the folder still exist.
 273        /// </summary>
 274        /// <param name="progress">The progress.</param>
 275        /// <param name="metadataRefreshOptions">The metadata refresh options.</param>
 276        /// <param name="recursive">if set to <c>true</c> [recursive].</param>
 277        /// <param name="allowRemoveRoot">remove item even this folder is root.</param>
 278        /// <param name="cancellationToken">The cancellation token.</param>
 279        /// <returns>Task.</returns>
 280        public Task ValidateChildren(IProgress<double> progress, MetadataRefreshOptions metadataRefreshOptions, bool rec
 281        {
 47282            return ValidateChildrenInternal(progress, recursive, true, allowRemoveRoot, metadataRefreshOptions, metadata
 283        }
 284
 285        private Dictionary<Guid, BaseItem> GetActualChildrenDictionary()
 286        {
 40287            var dictionary = new Dictionary<Guid, BaseItem>();
 288
 40289            var childrenList = Children.ToList();
 290
 142291            foreach (var child in childrenList)
 292            {
 32293                var id = child.Id;
 32294                if (dictionary.ContainsKey(id))
 295                {
 0296                    Logger.LogError(
 0297                        "Found folder containing items with duplicate id. Path: {Path}, Child Name: {ChildName}",
 0298                        Path ?? Name,
 0299                        child.Path ?? child.Name);
 300                }
 301                else
 302                {
 32303                    dictionary[id] = child;
 304                }
 305            }
 306
 39307            return dictionary;
 308        }
 309
 310        /// <summary>
 311        /// Validates the children internal.
 312        /// </summary>
 313        /// <param name="progress">The progress.</param>
 314        /// <param name="recursive">if set to <c>true</c> [recursive].</param>
 315        /// <param name="refreshChildMetadata">if set to <c>true</c> [refresh child metadata].</param>
 316        /// <param name="allowRemoveRoot">remove item even this folder is root.</param>
 317        /// <param name="refreshOptions">The refresh options.</param>
 318        /// <param name="directoryService">The directory service.</param>
 319        /// <param name="cancellationToken">The cancellation token.</param>
 320        /// <returns>Task.</returns>
 321        protected virtual async Task ValidateChildrenInternal(IProgress<double> progress, bool recursive, bool refreshCh
 322        {
 323            if (recursive)
 324            {
 325                ProviderManager.OnRefreshStart(this);
 326            }
 327
 328            try
 329            {
 330                await ValidateChildrenInternal2(progress, recursive, refreshChildMetadata, allowRemoveRoot, refreshOptio
 331            }
 332            finally
 333            {
 334                if (recursive)
 335                {
 336                    ProviderManager.OnRefreshComplete(this);
 337                }
 338            }
 339        }
 340
 341        private static bool IsLibraryFolderAccessible(IDirectoryService directoryService, BaseItem item, bool checkColle
 342        {
 80343            if (!checkCollection && (item is BoxSet || string.Equals(item.FileNameWithoutExtension, "collections", Strin
 344            {
 0345                return true;
 346            }
 347
 348            // For top parents i.e. Library folders, skip the validation if it's empty or inaccessible
 80349            if (item.IsTopParent && !directoryService.IsAccessible(item.ContainingFolderPath))
 350            {
 25351                Logger.LogWarning("Library folder {LibraryFolderPath} is inaccessible or empty, skipping", item.Containi
 25352                return false;
 353            }
 354
 55355            return true;
 356        }
 357
 358        private async Task ValidateChildrenInternal2(IProgress<double> progress, bool recursive, bool refreshChildMetada
 359        {
 360            if (!IsLibraryFolderAccessible(directoryService, this, allowRemoveRoot))
 361            {
 362                return;
 363            }
 364
 365            cancellationToken.ThrowIfCancellationRequested();
 366
 367            var validChildren = new List<BaseItem>();
 368            var validChildrenNeedGeneration = false;
 369
 370            if (IsFileProtocol)
 371            {
 372                IEnumerable<BaseItem> nonCachedChildren = [];
 373
 374                try
 375                {
 376                    nonCachedChildren = GetNonCachedChildren(directoryService);
 377                }
 378                catch (IOException ex)
 379                {
 380                    Logger.LogError(ex, "Error retrieving children from file system");
 381                }
 382                catch (SecurityException ex)
 383                {
 384                    Logger.LogError(ex, "Error retrieving children from file system");
 385                }
 386                catch (Exception ex)
 387                {
 388                    Logger.LogError(ex, "Error retrieving children");
 389                    return;
 390                }
 391
 392                progress.Report(ProgressHelpers.RetrievedChildren);
 393
 394                if (recursive)
 395                {
 396                    ProviderManager.OnRefreshProgress(this, ProgressHelpers.RetrievedChildren);
 397                }
 398
 399                // Build a dictionary of the current children we have now by Id so we can compare quickly and easily
 400                var currentChildren = GetActualChildrenDictionary();
 401
 402                // Create a list for our validated children
 403                var newItems = new List<BaseItem>();
 404
 405                cancellationToken.ThrowIfCancellationRequested();
 406
 407                foreach (var child in nonCachedChildren)
 408                {
 409                    if (!IsLibraryFolderAccessible(directoryService, child, allowRemoveRoot))
 410                    {
 411                        continue;
 412                    }
 413
 414                    if (currentChildren.TryGetValue(child.Id, out BaseItem currentChild))
 415                    {
 416                        validChildren.Add(currentChild);
 417
 418                        if (currentChild.UpdateFromResolvedItem(child) > ItemUpdateType.None)
 419                        {
 420                            await currentChild.UpdateToRepositoryAsync(ItemUpdateType.MetadataImport, cancellationToken)
 421                        }
 422                        else
 423                        {
 424                            // metadata is up-to-date; make sure DB has correct images dimensions and hash
 425                            await LibraryManager.UpdateImagesAsync(currentChild).ConfigureAwait(false);
 426                        }
 427
 428                        continue;
 429                    }
 430
 431                    // Brand new item - needs to be added
 432                    child.SetParent(this);
 433                    newItems.Add(child);
 434                    validChildren.Add(child);
 435                }
 436
 437                // That's all the new and changed ones - now see if any have been removed and need cleanup
 438                var itemsRemoved = currentChildren.Values.Except(validChildren).ToList();
 439                var shouldRemove = !IsRoot || allowRemoveRoot;
 440                // If it's an AggregateFolder, don't remove
 441                if (shouldRemove && itemsRemoved.Count > 0)
 442                {
 443                    foreach (var item in itemsRemoved)
 444                    {
 445                        if (item.IsFileProtocol)
 446                        {
 447                            Logger.LogDebug("Removed item: {Path}", item.Path);
 448
 449                            item.SetParent(null);
 450                            LibraryManager.DeleteItem(item, new DeleteOptions { DeleteFileLocation = false }, this, fals
 451                        }
 452                    }
 453                }
 454
 455                if (newItems.Count > 0)
 456                {
 457                    LibraryManager.CreateItems(newItems, this, cancellationToken);
 458                }
 459            }
 460            else
 461            {
 462                validChildrenNeedGeneration = true;
 463            }
 464
 465            progress.Report(ProgressHelpers.UpdatedChildItems);
 466
 467            if (recursive)
 468            {
 469                ProviderManager.OnRefreshProgress(this, ProgressHelpers.UpdatedChildItems);
 470            }
 471
 472            cancellationToken.ThrowIfCancellationRequested();
 473
 474            if (recursive)
 475            {
 476                var folder = this;
 477                var innerProgress = new Progress<double>(innerPercent =>
 478                {
 479                    var percent = ProgressHelpers.GetProgress(ProgressHelpers.UpdatedChildItems, ProgressHelpers.Scanned
 480
 481                    progress.Report(percent);
 482
 483                    ProviderManager.OnRefreshProgress(folder, percent);
 484                });
 485
 486                if (validChildrenNeedGeneration)
 487                {
 488                    validChildren = Children.ToList();
 489                    validChildrenNeedGeneration = false;
 490                }
 491
 492                await ValidateSubFolders(validChildren.OfType<Folder>().ToList(), directoryService, innerProgress, cance
 493            }
 494
 495            if (refreshChildMetadata)
 496            {
 497                progress.Report(ProgressHelpers.ScannedSubfolders);
 498
 499                if (recursive)
 500                {
 501                    ProviderManager.OnRefreshProgress(this, ProgressHelpers.ScannedSubfolders);
 502                }
 503
 504                var container = this as IMetadataContainer;
 505
 506                var folder = this;
 507                var innerProgress = new Progress<double>(innerPercent =>
 508                {
 509                    var percent = ProgressHelpers.GetProgress(ProgressHelpers.ScannedSubfolders, ProgressHelpers.Refresh
 510
 511                    progress.Report(percent);
 512
 513                    if (recursive)
 514                    {
 515                        ProviderManager.OnRefreshProgress(folder, percent);
 516                    }
 517                });
 518
 519                if (container is not null)
 520                {
 521                    await RefreshAllMetadataForContainer(container, refreshOptions, innerProgress, cancellationToken).Co
 522                }
 523                else
 524                {
 525                    if (validChildrenNeedGeneration)
 526                    {
 527                        validChildren = Children.ToList();
 528                    }
 529
 530                    await RefreshMetadataRecursive(validChildren, refreshOptions, recursive, innerProgress, cancellation
 531                }
 532            }
 533        }
 534
 535        private async Task RefreshMetadataRecursive(IList<BaseItem> children, MetadataRefreshOptions refreshOptions, boo
 536        {
 537            await RunTasks(
 538                (baseItem, innerProgress) => RefreshChildMetadata(baseItem, refreshOptions, recursive && baseItem.IsFold
 539                children,
 540                progress,
 541                cancellationToken).ConfigureAwait(false);
 542        }
 543
 544        private async Task RefreshAllMetadataForContainer(IMetadataContainer container, MetadataRefreshOptions refreshOp
 545        {
 546            if (container is Series series)
 547            {
 548                await series.RefreshMetadata(refreshOptions, cancellationToken).ConfigureAwait(false);
 549            }
 550
 551            await container.RefreshAllMetadata(refreshOptions, progress, cancellationToken).ConfigureAwait(false);
 552        }
 553
 554        private async Task RefreshChildMetadata(BaseItem child, MetadataRefreshOptions refreshOptions, bool recursive, I
 555        {
 556            if (child is IMetadataContainer container)
 557            {
 558                await RefreshAllMetadataForContainer(container, refreshOptions, progress, cancellationToken).ConfigureAw
 559            }
 560            else
 561            {
 562                if (refreshOptions.RefreshItem(child))
 563                {
 564                    await child.RefreshMetadata(refreshOptions, cancellationToken).ConfigureAwait(false);
 565                }
 566
 567                if (recursive && child is Folder folder)
 568                {
 569                    await folder.RefreshMetadataRecursive(folder.Children.ToList(), refreshOptions, true, progress, canc
 570                }
 571            }
 572        }
 573
 574        /// <summary>
 575        /// Refreshes the children.
 576        /// </summary>
 577        /// <param name="children">The children.</param>
 578        /// <param name="directoryService">The directory service.</param>
 579        /// <param name="progress">The progress.</param>
 580        /// <param name="cancellationToken">The cancellation token.</param>
 581        /// <returns>Task.</returns>
 582        private async Task ValidateSubFolders(IList<Folder> children, IDirectoryService directoryService, IProgress<doub
 583        {
 584            await RunTasks(
 585                (folder, innerProgress) => folder.ValidateChildrenInternal(innerProgress, true, false, false, null, dire
 586                children,
 587                progress,
 588                cancellationToken).ConfigureAwait(false);
 589        }
 590
 591        /// <summary>
 592        /// Runs an action block on a list of children.
 593        /// </summary>
 594        /// <param name="task">The task to run for each child.</param>
 595        /// <param name="children">The list of children.</param>
 596        /// <param name="progress">The progress.</param>
 597        /// <param name="cancellationToken">The cancellation token.</param>
 598        /// <returns>Task.</returns>
 599        private async Task RunTasks<T>(Func<T, IProgress<double>, Task> task, IList<T> children, IProgress<double> progr
 600        {
 601            var childrenCount = children.Count;
 602            var childrenProgress = new double[childrenCount];
 603
 604            void UpdateProgress()
 605            {
 606                progress.Report(childrenProgress.Average());
 607            }
 608
 609            var fanoutConcurrency = ConfigurationManager.Configuration.LibraryScanFanoutConcurrency;
 610            var parallelism = fanoutConcurrency > 0 ? fanoutConcurrency : Environment.ProcessorCount;
 611
 612            var actionBlock = new ActionBlock<int>(
 613                async i =>
 614                {
 615                    var innerProgress = new Progress<double>(innerPercent =>
 616                    {
 617                        // round the percent and only update progress if it changed to prevent excessive UpdateProgress 
 618                        var innerPercentRounded = Math.Round(innerPercent);
 619                        if (childrenProgress[i] != innerPercentRounded)
 620                        {
 621                            childrenProgress[i] = innerPercentRounded;
 622                            UpdateProgress();
 623                        }
 624                    });
 625
 626                    await task(children[i], innerProgress).ConfigureAwait(false);
 627
 628                    childrenProgress[i] = 100;
 629
 630                    UpdateProgress();
 631                },
 632                new ExecutionDataflowBlockOptions
 633                {
 634                    MaxDegreeOfParallelism = parallelism,
 635                    CancellationToken = cancellationToken,
 636                });
 637
 638            for (var i = 0; i < childrenCount; i++)
 639            {
 640                await actionBlock.SendAsync(i, cancellationToken).ConfigureAwait(false);
 641            }
 642
 643            actionBlock.Complete();
 644
 645            await actionBlock.Completion.ConfigureAwait(false);
 646        }
 647
 648        /// <summary>
 649        /// Get the children of this folder from the actual file system.
 650        /// </summary>
 651        /// <returns>IEnumerable{BaseItem}.</returns>
 652        /// <param name="directoryService">The directory service to use for operation.</param>
 653        /// <returns>Returns set of base items.</returns>
 654        protected virtual IEnumerable<BaseItem> GetNonCachedChildren(IDirectoryService directoryService)
 655        {
 40656            var collectionType = LibraryManager.GetContentType(this);
 40657            var libraryOptions = LibraryManager.GetLibraryOptions(this);
 658
 40659            return LibraryManager.ResolvePaths(GetFileSystemChildren(directoryService), directoryService, this, libraryO
 660        }
 661
 662        /// <summary>
 663        /// Get our children from the repo - stubbed for now.
 664        /// </summary>
 665        /// <returns>IEnumerable{BaseItem}.</returns>
 666        protected IReadOnlyList<BaseItem> GetCachedChildren()
 667        {
 142668            return ItemRepository.GetItemList(new InternalItemsQuery
 142669            {
 142670                Parent = this,
 142671                GroupByPresentationUniqueKey = false,
 142672                DtoOptions = new DtoOptions(true)
 142673            });
 674        }
 675
 676        public virtual int GetChildCount(User user)
 677        {
 0678            if (LinkedChildren.Length > 0)
 679            {
 0680                if (this is not ICollectionFolder)
 681                {
 0682                    return GetChildren(user, true).Count;
 683                }
 684            }
 685
 0686            var result = GetItems(new InternalItemsQuery(user)
 0687            {
 0688                Recursive = false,
 0689                Limit = 0,
 0690                Parent = this,
 0691                DtoOptions = new DtoOptions(false)
 0692                {
 0693                    EnableImages = false
 0694                }
 0695            });
 696
 0697            return result.TotalRecordCount;
 698        }
 699
 700        public virtual int GetRecursiveChildCount(User user)
 701        {
 0702            return GetItems(new InternalItemsQuery(user)
 0703            {
 0704                Recursive = true,
 0705                IsFolder = false,
 0706                IsVirtualItem = false,
 0707                EnableTotalRecordCount = true,
 0708                Limit = 0,
 0709                DtoOptions = new DtoOptions(false)
 0710                {
 0711                    EnableImages = false
 0712                }
 0713            }).TotalRecordCount;
 714        }
 715
 716        public QueryResult<BaseItem> QueryRecursive(InternalItemsQuery query)
 717        {
 16718            var user = query.User;
 719
 16720            if (!query.ForceDirect && RequiresPostFiltering(query))
 721            {
 722                IEnumerable<BaseItem> items;
 0723                Func<BaseItem, bool> filter = i => UserViewBuilder.Filter(i, user, query, UserDataManager, LibraryManage
 724
 0725                if (query.User is null)
 726                {
 0727                    items = GetRecursiveChildren(filter);
 728                }
 729                else
 730                {
 0731                    items = GetRecursiveChildren(user, query);
 732                }
 733
 0734                return PostFilterAndSort(items, query, true);
 735            }
 736
 16737            if (this is not UserRootFolder
 16738                && this is not AggregateFolder
 16739                && query.ParentId.IsEmpty())
 740            {
 16741                query.Parent = this;
 742            }
 743
 16744            if (RequiresPostFiltering2(query))
 745            {
 0746                return QueryWithPostFiltering2(query);
 747            }
 748
 16749            return LibraryManager.GetItemsResult(query);
 750        }
 751
 752        protected QueryResult<BaseItem> QueryWithPostFiltering2(InternalItemsQuery query)
 753        {
 1754            var startIndex = query.StartIndex;
 1755            var limit = query.Limit;
 756
 1757            query.StartIndex = null;
 1758            query.Limit = null;
 759
 1760            IEnumerable<BaseItem> itemsList = LibraryManager.GetItemList(query);
 1761            var user = query.User;
 762
 1763            if (user is not null)
 764            {
 765                // needed for boxsets
 1766                itemsList = itemsList.Where(i => i.IsVisibleStandalone(query.User));
 767            }
 768
 769            IEnumerable<BaseItem> returnItems;
 1770            int totalCount = 0;
 771
 1772            if (query.EnableTotalRecordCount)
 773            {
 0774                var itemArray = itemsList.ToArray();
 0775                totalCount = itemArray.Length;
 0776                returnItems = itemArray;
 777            }
 778            else
 779            {
 1780                returnItems = itemsList;
 781            }
 782
 1783            if (limit.HasValue)
 784            {
 0785                returnItems = returnItems.Skip(startIndex ?? 0).Take(limit.Value);
 786            }
 1787            else if (startIndex.HasValue)
 788            {
 0789                returnItems = returnItems.Skip(startIndex.Value);
 790            }
 791
 1792            return new QueryResult<BaseItem>(
 1793                query.StartIndex,
 1794                totalCount,
 1795                returnItems.ToArray());
 796        }
 797
 798        private bool RequiresPostFiltering2(InternalItemsQuery query)
 799        {
 16800            if (query.IncludeItemTypes.Length == 1 && query.IncludeItemTypes[0] == BaseItemKind.BoxSet)
 801            {
 0802                Logger.LogDebug("Query requires post-filtering due to BoxSet query");
 0803                return true;
 804            }
 805
 16806            return false;
 807        }
 808
 809        private bool RequiresPostFiltering(InternalItemsQuery query)
 810        {
 16811            if (LinkedChildren.Length > 0)
 812            {
 0813                if (this is not ICollectionFolder)
 814                {
 0815                    Logger.LogDebug("{Type}: Query requires post-filtering due to LinkedChildren.", GetType().Name);
 0816                    return true;
 817                }
 818            }
 819
 820            // Filter by Video3DFormat
 16821            if (query.Is3D.HasValue)
 822            {
 0823                Logger.LogDebug("Query requires post-filtering due to Is3D");
 0824                return true;
 825            }
 826
 16827            if (query.HasOfficialRating.HasValue)
 828            {
 0829                Logger.LogDebug("Query requires post-filtering due to HasOfficialRating");
 0830                return true;
 831            }
 832
 16833            if (query.IsPlaceHolder.HasValue)
 834            {
 0835                Logger.LogDebug("Query requires post-filtering due to IsPlaceHolder");
 0836                return true;
 837            }
 838
 16839            if (query.HasSpecialFeature.HasValue)
 840            {
 0841                Logger.LogDebug("Query requires post-filtering due to HasSpecialFeature");
 0842                return true;
 843            }
 844
 16845            if (query.HasSubtitles.HasValue)
 846            {
 0847                Logger.LogDebug("Query requires post-filtering due to HasSubtitles");
 0848                return true;
 849            }
 850
 16851            if (query.HasTrailer.HasValue)
 852            {
 0853                Logger.LogDebug("Query requires post-filtering due to HasTrailer");
 0854                return true;
 855            }
 856
 16857            if (query.HasThemeSong.HasValue)
 858            {
 0859                Logger.LogDebug("Query requires post-filtering due to HasThemeSong");
 0860                return true;
 861            }
 862
 16863            if (query.HasThemeVideo.HasValue)
 864            {
 0865                Logger.LogDebug("Query requires post-filtering due to HasThemeVideo");
 0866                return true;
 867            }
 868
 869            // Filter by VideoType
 16870            if (query.VideoTypes.Length > 0)
 871            {
 0872                Logger.LogDebug("Query requires post-filtering due to VideoTypes");
 0873                return true;
 874            }
 875
 16876            if (CollapseBoxSetItems(query, this, query.User, ConfigurationManager))
 877            {
 0878                Logger.LogDebug("Query requires post-filtering due to CollapseBoxSetItems");
 0879                return true;
 880            }
 881
 16882            if (!query.AdjacentTo.IsNullOrEmpty())
 883            {
 0884                Logger.LogDebug("Query requires post-filtering due to AdjacentTo");
 0885                return true;
 886            }
 887
 16888            if (query.SeriesStatuses.Length > 0)
 889            {
 0890                Logger.LogDebug("Query requires post-filtering due to SeriesStatuses");
 0891                return true;
 892            }
 893
 16894            if (query.AiredDuringSeason.HasValue)
 895            {
 0896                Logger.LogDebug("Query requires post-filtering due to AiredDuringSeason");
 0897                return true;
 898            }
 899
 16900            if (query.IsPlayed.HasValue)
 901            {
 0902                if (query.IncludeItemTypes.Length == 1 && query.IncludeItemTypes.Contains(BaseItemKind.Series))
 903                {
 0904                    Logger.LogDebug("Query requires post-filtering due to IsPlayed");
 0905                    return true;
 906                }
 907            }
 908
 16909            return false;
 910        }
 911
 912        private static BaseItem[] SortItemsByRequest(InternalItemsQuery query, IReadOnlyList<BaseItem> items)
 913        {
 0914            return items.OrderBy(i => Array.IndexOf(query.ItemIds, i.Id)).ToArray();
 915        }
 916
 917        public QueryResult<BaseItem> GetItems(InternalItemsQuery query)
 918        {
 0919            if (query.ItemIds.Length > 0)
 920            {
 0921                var result = LibraryManager.GetItemsResult(query);
 922
 0923                if (query.OrderBy.Count == 0 && query.ItemIds.Length > 1)
 924                {
 0925                    result.Items = SortItemsByRequest(query, result.Items);
 926                }
 927
 0928                return result;
 929            }
 930
 0931            return GetItemsInternal(query);
 932        }
 933
 934        public IReadOnlyList<BaseItem> GetItemList(InternalItemsQuery query)
 935        {
 17936            query.EnableTotalRecordCount = false;
 937
 17938            if (query.ItemIds.Length > 0)
 939            {
 0940                var result = LibraryManager.GetItemList(query);
 941
 0942                if (query.OrderBy.Count == 0 && query.ItemIds.Length > 1)
 943                {
 0944                    return SortItemsByRequest(query, result);
 945                }
 946
 0947                return result;
 948            }
 949
 17950            return GetItemsInternal(query).Items;
 951        }
 952
 953        protected virtual QueryResult<BaseItem> GetItemsInternal(InternalItemsQuery query)
 954        {
 16955            if (SourceType == SourceType.Channel)
 956            {
 957                try
 958                {
 0959                    query.Parent = this;
 0960                    query.ChannelIds = new[] { ChannelId };
 961
 962                    // Don't blow up here because it could cause parent screens with other content to fail
 0963                    return ChannelManager.GetChannelItemsInternal(query, new Progress<double>(), CancellationToken.None)
 964                }
 0965                catch
 966                {
 967                    // Already logged at lower levels
 0968                    return new QueryResult<BaseItem>();
 969                }
 970            }
 971
 16972            if (query.Recursive)
 973            {
 16974                return QueryRecursive(query);
 975            }
 976
 0977            var user = query.User;
 978
 0979            Func<BaseItem, bool> filter = i => UserViewBuilder.Filter(i, user, query, UserDataManager, LibraryManager);
 980
 981            IEnumerable<BaseItem> items;
 982
 0983            if (query.User is null)
 984            {
 0985                items = Children.Where(filter);
 986            }
 987            else
 988            {
 989                // need to pass this param to the children.
 0990                var childQuery = new InternalItemsQuery
 0991                {
 0992                    DisplayAlbumFolders = query.DisplayAlbumFolders
 0993                };
 994
 0995                items = GetChildren(user, true, childQuery).Where(filter);
 996            }
 997
 0998            return PostFilterAndSort(items, query, true);
 0999        }
 1000
 1001        protected QueryResult<BaseItem> PostFilterAndSort(IEnumerable<BaseItem> items, InternalItemsQuery query, bool en
 1002        {
 01003            var user = query.User;
 1004
 1005            // Check recursive - don't substitute in plain folder views
 01006            if (user is not null)
 1007            {
 01008                items = CollapseBoxSetItemsIfNeeded(items, query, this, user, ConfigurationManager, CollectionManager);
 1009            }
 1010
 1011            #pragma warning disable CA1309
 01012            if (!string.IsNullOrEmpty(query.NameStartsWithOrGreater))
 1013            {
 01014                items = items.Where(i => string.Compare(query.NameStartsWithOrGreater, i.SortName, StringComparison.Inva
 1015            }
 1016
 01017            if (!string.IsNullOrEmpty(query.NameStartsWith))
 1018            {
 01019                items = items.Where(i => i.SortName.StartsWith(query.NameStartsWith, StringComparison.InvariantCultureIg
 1020            }
 1021
 01022            if (!string.IsNullOrEmpty(query.NameLessThan))
 1023            {
 01024                items = items.Where(i => string.Compare(query.NameLessThan, i.SortName, StringComparison.InvariantCultur
 1025            }
 1026            #pragma warning restore CA1309
 1027
 1028            // This must be the last filter
 01029            if (!query.AdjacentTo.IsNullOrEmpty())
 1030            {
 01031                items = UserViewBuilder.FilterForAdjacency(items.ToList(), query.AdjacentTo.Value);
 1032            }
 1033
 01034            return UserViewBuilder.SortAndPage(items, null, query, LibraryManager, enableSorting);
 1035        }
 1036
 1037        private static IEnumerable<BaseItem> CollapseBoxSetItemsIfNeeded(
 1038            IEnumerable<BaseItem> items,
 1039            InternalItemsQuery query,
 1040            BaseItem queryParent,
 1041            User user,
 1042            IServerConfigurationManager configurationManager,
 1043            ICollectionManager collectionManager)
 1044        {
 01045            ArgumentNullException.ThrowIfNull(items);
 1046
 01047            if (CollapseBoxSetItems(query, queryParent, user, configurationManager))
 1048            {
 01049                items = collectionManager.CollapseItemsWithinBoxSets(items, user);
 1050            }
 1051
 01052            return items;
 1053        }
 1054
 1055        private static bool CollapseBoxSetItems(
 1056            InternalItemsQuery query,
 1057            BaseItem queryParent,
 1058            User user,
 1059            IServerConfigurationManager configurationManager)
 1060        {
 1061            // Could end up stuck in a loop like this
 161062            if (queryParent is BoxSet)
 1063            {
 01064                return false;
 1065            }
 1066
 161067            if (queryParent is Season)
 1068            {
 01069                return false;
 1070            }
 1071
 161072            if (queryParent is MusicAlbum)
 1073            {
 01074                return false;
 1075            }
 1076
 161077            if (queryParent is MusicArtist)
 1078            {
 01079                return false;
 1080            }
 1081
 161082            var param = query.CollapseBoxSetItems;
 1083
 161084            if (!param.HasValue)
 1085            {
 01086                if (user is not null && query.IncludeItemTypes.Any(type =>
 01087                    (type == BaseItemKind.Movie && !configurationManager.Configuration.EnableGroupingMoviesIntoCollectio
 01088                    (type == BaseItemKind.Series && !configurationManager.Configuration.EnableGroupingShowsIntoCollectio
 1089                {
 01090                    return false;
 1091                }
 1092
 01093                if (query.IncludeItemTypes.Length == 0
 01094                    || query.IncludeItemTypes.Any(type => type == BaseItemKind.Movie || type == BaseItemKind.Series))
 1095                {
 01096                    param = true;
 1097                }
 1098            }
 1099
 161100            return param.HasValue && param.Value && AllowBoxSetCollapsing(query);
 1101        }
 1102
 1103        private static bool AllowBoxSetCollapsing(InternalItemsQuery request)
 1104        {
 01105            if (request.IsFavorite.HasValue)
 1106            {
 01107                return false;
 1108            }
 1109
 01110            if (request.IsFavoriteOrLiked.HasValue)
 1111            {
 01112                return false;
 1113            }
 1114
 01115            if (request.IsLiked.HasValue)
 1116            {
 01117                return false;
 1118            }
 1119
 01120            if (request.IsPlayed.HasValue)
 1121            {
 01122                return false;
 1123            }
 1124
 01125            if (request.IsResumable.HasValue)
 1126            {
 01127                return false;
 1128            }
 1129
 01130            if (request.IsFolder.HasValue)
 1131            {
 01132                return false;
 1133            }
 1134
 01135            if (request.Genres.Count > 0)
 1136            {
 01137                return false;
 1138            }
 1139
 01140            if (request.GenreIds.Count > 0)
 1141            {
 01142                return false;
 1143            }
 1144
 01145            if (request.HasImdbId.HasValue)
 1146            {
 01147                return false;
 1148            }
 1149
 01150            if (request.HasOfficialRating.HasValue)
 1151            {
 01152                return false;
 1153            }
 1154
 01155            if (request.HasOverview.HasValue)
 1156            {
 01157                return false;
 1158            }
 1159
 01160            if (request.HasParentalRating.HasValue)
 1161            {
 01162                return false;
 1163            }
 1164
 01165            if (request.HasSpecialFeature.HasValue)
 1166            {
 01167                return false;
 1168            }
 1169
 01170            if (request.HasSubtitles.HasValue)
 1171            {
 01172                return false;
 1173            }
 1174
 01175            if (request.HasThemeSong.HasValue)
 1176            {
 01177                return false;
 1178            }
 1179
 01180            if (request.HasThemeVideo.HasValue)
 1181            {
 01182                return false;
 1183            }
 1184
 01185            if (request.HasTmdbId.HasValue)
 1186            {
 01187                return false;
 1188            }
 1189
 01190            if (request.HasTrailer.HasValue)
 1191            {
 01192                return false;
 1193            }
 1194
 01195            if (request.ImageTypes.Length > 0)
 1196            {
 01197                return false;
 1198            }
 1199
 01200            if (request.Is3D.HasValue)
 1201            {
 01202                return false;
 1203            }
 1204
 01205            if (request.Is4K.HasValue)
 1206            {
 01207                return false;
 1208            }
 1209
 01210            if (request.IsHD.HasValue)
 1211            {
 01212                return false;
 1213            }
 1214
 01215            if (request.IsLocked.HasValue)
 1216            {
 01217                return false;
 1218            }
 1219
 01220            if (request.IsPlaceHolder.HasValue)
 1221            {
 01222                return false;
 1223            }
 1224
 01225            if (request.IsPlayed.HasValue)
 1226            {
 01227                return false;
 1228            }
 1229
 01230            if (!string.IsNullOrWhiteSpace(request.Person))
 1231            {
 01232                return false;
 1233            }
 1234
 01235            if (request.PersonIds.Length > 0)
 1236            {
 01237                return false;
 1238            }
 1239
 01240            if (request.ItemIds.Length > 0)
 1241            {
 01242                return false;
 1243            }
 1244
 01245            if (request.StudioIds.Length > 0)
 1246            {
 01247                return false;
 1248            }
 1249
 01250            if (request.VideoTypes.Length > 0)
 1251            {
 01252                return false;
 1253            }
 1254
 01255            if (request.Years.Length > 0)
 1256            {
 01257                return false;
 1258            }
 1259
 01260            if (request.Tags.Length > 0)
 1261            {
 01262                return false;
 1263            }
 1264
 01265            if (request.OfficialRatings.Length > 0)
 1266            {
 01267                return false;
 1268            }
 1269
 01270            if (request.MinCommunityRating.HasValue)
 1271            {
 01272                return false;
 1273            }
 1274
 01275            if (request.MinCriticRating.HasValue)
 1276            {
 01277                return false;
 1278            }
 1279
 01280            if (request.MinIndexNumber.HasValue)
 1281            {
 01282                return false;
 1283            }
 1284
 01285            return true;
 1286        }
 1287
 1288        public IReadOnlyList<BaseItem> GetChildren(User user, bool includeLinkedChildren)
 1289        {
 101290            ArgumentNullException.ThrowIfNull(user);
 1291
 101292            return GetChildren(user, includeLinkedChildren, new InternalItemsQuery(user));
 1293        }
 1294
 1295        public virtual IReadOnlyList<BaseItem> GetChildren(User user, bool includeLinkedChildren, InternalItemsQuery que
 1296        {
 101297            ArgumentNullException.ThrowIfNull(user);
 1298
 1299            // the true root should return our users root folder children
 101300            if (IsPhysicalRoot)
 1301            {
 01302                return LibraryManager.GetUserRootFolder().GetChildren(user, includeLinkedChildren);
 1303            }
 1304
 101305            var result = new Dictionary<Guid, BaseItem>();
 1306
 101307            AddChildren(user, includeLinkedChildren, result, false, query);
 1308
 101309            return result.Values.ToArray();
 1310        }
 1311
 1312        protected virtual IEnumerable<BaseItem> GetEligibleChildrenForRecursiveChildren(User user)
 1313        {
 101314            return Children;
 1315        }
 1316
 1317        /// <summary>
 1318        /// Adds the children to list.
 1319        /// </summary>
 1320        private void AddChildren(User user, bool includeLinkedChildren, Dictionary<Guid, BaseItem> result, bool recursiv
 1321        {
 1322            // Prevent infinite recursion of nested folders
 101323            visitedFolders ??= new HashSet<Folder>();
 101324            if (!visitedFolders.Add(this))
 1325            {
 01326                return;
 1327            }
 1328
 1329            // If Query.AlbumFolders is set, then enforce the format as per the db in that it permits sub-folders in mus
 101330            IEnumerable<BaseItem> children = null;
 101331            if ((query?.DisplayAlbumFolders ?? false) && (this is MusicAlbum))
 1332            {
 01333                children = Children;
 01334                query = null;
 1335            }
 1336
 1337            // If there are not sub-folders, proceed as normal.
 101338            if (children is null)
 1339            {
 101340                children = GetEligibleChildrenForRecursiveChildren(user);
 1341            }
 1342
 101343            AddChildrenFromCollection(children, user, includeLinkedChildren, result, recursive, query, visitedFolders);
 1344
 101345            if (includeLinkedChildren)
 1346            {
 101347                AddChildrenFromCollection(GetLinkedChildren(user), user, includeLinkedChildren, result, recursive, query
 1348            }
 101349        }
 1350
 1351        private void AddChildrenFromCollection(IEnumerable<BaseItem> children, User user, bool includeLinkedChildren, Di
 1352        {
 601353            foreach (var child in children)
 1354            {
 101355                if (!child.IsVisible(user))
 1356                {
 1357                    continue;
 1358                }
 1359
 101360                if (query is null || UserViewBuilder.FilterItem(child, query))
 1361                {
 101362                    result[child.Id] = child;
 1363                }
 1364
 101365                if (recursive && child.IsFolder)
 1366                {
 01367                    var folder = (Folder)child;
 1368
 01369                    folder.AddChildren(user, includeLinkedChildren, result, true, query, visitedFolders);
 1370                }
 1371            }
 201372        }
 1373
 1374        public virtual IReadOnlyList<BaseItem> GetRecursiveChildren(User user, InternalItemsQuery query)
 1375        {
 01376            ArgumentNullException.ThrowIfNull(user);
 1377
 01378            var result = new Dictionary<Guid, BaseItem>();
 1379
 01380            AddChildren(user, true, result, true, query);
 1381
 01382            return result.Values.ToArray();
 1383        }
 1384
 1385        /// <summary>
 1386        /// Gets the recursive children.
 1387        /// </summary>
 1388        /// <returns>IList{BaseItem}.</returns>
 1389        public IReadOnlyList<BaseItem> GetRecursiveChildren()
 1390        {
 01391            return GetRecursiveChildren(true);
 1392        }
 1393
 1394        public IReadOnlyList<BaseItem> GetRecursiveChildren(bool includeLinkedChildren)
 1395        {
 21396            return GetRecursiveChildren(i => true, includeLinkedChildren);
 1397        }
 1398
 1399        public IReadOnlyList<BaseItem> GetRecursiveChildren(Func<BaseItem, bool> filter)
 1400        {
 01401            return GetRecursiveChildren(filter, true);
 1402        }
 1403
 1404        public IReadOnlyList<BaseItem> GetRecursiveChildren(Func<BaseItem, bool> filter, bool includeLinkedChildren)
 1405        {
 21406            var result = new Dictionary<Guid, BaseItem>();
 1407
 21408            AddChildrenToList(result, includeLinkedChildren, true, filter);
 1409
 21410            return result.Values.ToArray();
 1411        }
 1412
 1413        /// <summary>
 1414        /// Adds the children to list.
 1415        /// </summary>
 1416        private void AddChildrenToList(Dictionary<Guid, BaseItem> result, bool includeLinkedChildren, bool recursive, Fu
 1417        {
 41418            foreach (var child in Children)
 1419            {
 01420                if (filter is null || filter(child))
 1421                {
 01422                    result[child.Id] = child;
 1423                }
 1424
 01425                if (recursive && child.IsFolder)
 1426                {
 01427                    var folder = (Folder)child;
 1428
 1429                    // We can only support includeLinkedChildren for the first folder, or we might end up stuck in a loo
 01430                    folder.AddChildrenToList(result, false, true, filter);
 1431                }
 1432            }
 1433
 21434            if (includeLinkedChildren)
 1435            {
 01436                foreach (var child in GetLinkedChildren())
 1437                {
 01438                    if (filter is null || filter(child))
 1439                    {
 01440                        result[child.Id] = child;
 1441                    }
 1442                }
 1443            }
 21444        }
 1445
 1446        /// <summary>
 1447        /// Gets the linked children.
 1448        /// </summary>
 1449        /// <returns>IEnumerable{BaseItem}.</returns>
 1450        public List<BaseItem> GetLinkedChildren()
 1451        {
 101452            var linkedChildren = LinkedChildren;
 101453            var list = new List<BaseItem>(linkedChildren.Length);
 1454
 201455            foreach (var i in linkedChildren)
 1456            {
 01457                var child = GetLinkedChild(i);
 1458
 01459                if (child is not null)
 1460                {
 01461                    list.Add(child);
 1462                }
 1463            }
 1464
 101465            return list;
 1466        }
 1467
 1468        public bool ContainsLinkedChildByItemId(Guid itemId)
 1469        {
 01470            var linkedChildren = LinkedChildren;
 01471            foreach (var i in linkedChildren)
 1472            {
 01473                if (i.ItemId.HasValue)
 1474                {
 01475                    if (i.ItemId.Value.Equals(itemId))
 1476                    {
 01477                        return true;
 1478                    }
 1479
 1480                    continue;
 1481                }
 1482
 01483                var child = GetLinkedChild(i);
 1484
 01485                if (child is not null && child.Id.Equals(itemId))
 1486                {
 01487                    return true;
 1488                }
 1489            }
 1490
 01491            return false;
 1492        }
 1493
 1494        public List<BaseItem> GetLinkedChildren(User user)
 1495        {
 101496            if (!FilterLinkedChildrenPerUser || user is null)
 1497            {
 101498                return GetLinkedChildren();
 1499            }
 1500
 01501            var linkedChildren = LinkedChildren;
 01502            var list = new List<BaseItem>(linkedChildren.Length);
 1503
 01504            if (linkedChildren.Length == 0)
 1505            {
 01506                return list;
 1507            }
 1508
 01509            var allUserRootChildren = LibraryManager.GetUserRootFolder()
 01510                .GetChildren(user, true)
 01511                .OfType<Folder>()
 01512                .ToList();
 1513
 01514            var collectionFolderIds = allUserRootChildren
 01515                .Select(i => i.Id)
 01516                .ToList();
 1517
 01518            foreach (var i in linkedChildren)
 1519            {
 01520                var child = GetLinkedChild(i);
 1521
 01522                if (child is null)
 1523                {
 1524                    continue;
 1525                }
 1526
 01527                var childOwner = child.GetOwner() ?? child;
 1528
 01529                if (child is not IItemByName)
 1530                {
 01531                    var childProtocol = childOwner.PathProtocol;
 01532                    if (!childProtocol.HasValue || childProtocol.Value != Model.MediaInfo.MediaProtocol.File)
 1533                    {
 01534                        if (!childOwner.IsVisibleStandalone(user))
 1535                        {
 01536                            continue;
 1537                        }
 1538                    }
 1539                    else
 1540                    {
 01541                        var itemCollectionFolderIds =
 01542                            LibraryManager.GetCollectionFolders(childOwner, allUserRootChildren).Select(f => f.Id);
 1543
 01544                        if (!itemCollectionFolderIds.Any(collectionFolderIds.Contains))
 1545                        {
 1546                            continue;
 1547                        }
 1548                    }
 1549                }
 1550
 01551                list.Add(child);
 1552            }
 1553
 01554            return list;
 1555        }
 1556
 1557        /// <summary>
 1558        /// Gets the linked children.
 1559        /// </summary>
 1560        /// <returns>IEnumerable{BaseItem}.</returns>
 1561        public IReadOnlyList<Tuple<LinkedChild, BaseItem>> GetLinkedChildrenInfos()
 1562        {
 01563            return LinkedChildren
 01564                .Select(i => new Tuple<LinkedChild, BaseItem>(i, GetLinkedChild(i)))
 01565                .Where(i => i.Item2 is not null)
 01566                .ToArray();
 1567        }
 1568
 1569        protected override async Task<bool> RefreshedOwnedItems(MetadataRefreshOptions options, IReadOnlyList<FileSystem
 1570        {
 1571            var changesFound = false;
 1572
 1573            if (IsFileProtocol)
 1574            {
 1575                if (RefreshLinkedChildren(fileSystemChildren))
 1576                {
 1577                    changesFound = true;
 1578                }
 1579            }
 1580
 1581            var baseHasChanges = await base.RefreshedOwnedItems(options, fileSystemChildren, cancellationToken).Configur
 1582
 1583            return baseHasChanges || changesFound;
 1584        }
 1585
 1586        /// <summary>
 1587        /// Refreshes the linked children.
 1588        /// </summary>
 1589        /// <param name="fileSystemChildren">The enumerable of file system metadata.</param>
 1590        /// <returns><c>true</c> if the linked children were updated, <c>false</c> otherwise.</returns>
 1591        protected virtual bool RefreshLinkedChildren(IEnumerable<FileSystemMetadata> fileSystemChildren)
 1592        {
 151593            if (SupportsShortcutChildren)
 1594            {
 151595                var newShortcutLinks = fileSystemChildren
 151596                    .Where(i => !i.IsDirectory && FileSystem.IsShortcut(i.FullName))
 151597                    .Select(i =>
 151598                    {
 151599                        try
 151600                        {
 151601                            Logger.LogDebug("Found shortcut at {0}", i.FullName);
 151602
 151603                            var resolvedPath = CollectionFolder.ApplicationHost.ExpandVirtualPath(FileSystem.ResolveShor
 151604
 151605                            if (!string.IsNullOrEmpty(resolvedPath))
 151606                            {
 151607                                return new LinkedChild
 151608                                {
 151609                                    Path = resolvedPath,
 151610                                    Type = LinkedChildType.Shortcut
 151611                                };
 151612                            }
 151613
 151614                            Logger.LogError("Error resolving shortcut {0}", i.FullName);
 151615
 151616                            return null;
 151617                        }
 151618                        catch (IOException ex)
 151619                        {
 151620                            Logger.LogError(ex, "Error resolving shortcut {0}", i.FullName);
 151621                            return null;
 151622                        }
 151623                    })
 151624                    .Where(i => i is not null)
 151625                    .ToList();
 1626
 151627                var currentShortcutLinks = LinkedChildren.Where(i => i.Type == LinkedChildType.Shortcut).ToList();
 1628
 151629                if (!newShortcutLinks.SequenceEqual(currentShortcutLinks, new LinkedChildComparer(FileSystem)))
 1630                {
 01631                    Logger.LogInformation("Shortcut links have changed for {0}", Path);
 1632
 01633                    newShortcutLinks.AddRange(LinkedChildren.Where(i => i.Type == LinkedChildType.Manual));
 01634                    LinkedChildren = newShortcutLinks.ToArray();
 01635                    return true;
 1636                }
 1637            }
 1638
 301639            foreach (var child in LinkedChildren)
 1640            {
 1641                // Reset the cached value
 01642                child.ItemId = null;
 1643            }
 1644
 151645            return false;
 1646        }
 1647
 1648        /// <summary>
 1649        /// Marks the played.
 1650        /// </summary>
 1651        /// <param name="user">The user.</param>
 1652        /// <param name="datePlayed">The date played.</param>
 1653        /// <param name="resetPosition">if set to <c>true</c> [reset position].</param>
 1654        public override void MarkPlayed(
 1655            User user,
 1656            DateTime? datePlayed,
 1657            bool resetPosition)
 1658        {
 01659            var query = new InternalItemsQuery
 01660            {
 01661                User = user,
 01662                Recursive = true,
 01663                IsFolder = false,
 01664                EnableTotalRecordCount = false
 01665            };
 1666
 01667            if (!user.DisplayMissingEpisodes)
 1668            {
 01669                query.IsVirtualItem = false;
 1670            }
 1671
 01672            var itemsResult = GetItemList(query);
 1673
 1674            // Sweep through recursively and update status
 01675            foreach (var item in itemsResult)
 1676            {
 01677                if (item.IsVirtualItem)
 1678                {
 1679                    // The querying doesn't support virtual unaired
 01680                    var episode = item as Episode;
 01681                    if (episode is not null && episode.IsUnaired)
 1682                    {
 1683                        continue;
 1684                    }
 1685                }
 1686
 01687                item.MarkPlayed(user, datePlayed, resetPosition);
 1688            }
 01689        }
 1690
 1691        /// <summary>
 1692        /// Marks the unplayed.
 1693        /// </summary>
 1694        /// <param name="user">The user.</param>
 1695        public override void MarkUnplayed(User user)
 1696        {
 01697            var itemsResult = GetItemList(new InternalItemsQuery
 01698            {
 01699                User = user,
 01700                Recursive = true,
 01701                IsFolder = false,
 01702                EnableTotalRecordCount = false
 01703            });
 1704
 1705            // Sweep through recursively and update status
 01706            foreach (var item in itemsResult)
 1707            {
 01708                item.MarkUnplayed(user);
 1709            }
 01710        }
 1711
 1712        public override bool IsPlayed(User user)
 1713        {
 01714            var itemsResult = GetItemList(new InternalItemsQuery(user)
 01715            {
 01716                Recursive = true,
 01717                IsFolder = false,
 01718                IsVirtualItem = false,
 01719                EnableTotalRecordCount = false
 01720            });
 1721
 01722            return itemsResult
 01723                .All(i => i.IsPlayed(user));
 1724        }
 1725
 1726        public override bool IsUnplayed(User user)
 1727        {
 01728            return !IsPlayed(user);
 1729        }
 1730
 1731        public override void FillUserDataDtoValues(UserItemDataDto dto, UserItemData userData, BaseItemDto itemDto, User
 1732        {
 91733            if (!SupportsUserDataFromChildren)
 1734            {
 91735                return;
 1736            }
 1737
 01738            if (itemDto is not null && fields.ContainsField(ItemFields.RecursiveItemCount))
 1739            {
 01740                itemDto.RecursiveItemCount = GetRecursiveChildCount(user);
 1741            }
 1742
 01743            if (SupportsPlayedStatus)
 1744            {
 01745                var unplayedQueryResult = GetItems(new InternalItemsQuery(user)
 01746                {
 01747                    Recursive = true,
 01748                    IsFolder = false,
 01749                    IsVirtualItem = false,
 01750                    EnableTotalRecordCount = true,
 01751                    Limit = 0,
 01752                    IsPlayed = false,
 01753                    DtoOptions = new DtoOptions(false)
 01754                    {
 01755                        EnableImages = false
 01756                    }
 01757                }).TotalRecordCount;
 1758
 01759                dto.UnplayedItemCount = unplayedQueryResult;
 1760
 01761                if (itemDto?.RecursiveItemCount > 0)
 1762                {
 01763                    var unplayedPercentage = ((double)unplayedQueryResult / itemDto.RecursiveItemCount.Value) * 100;
 01764                    dto.PlayedPercentage = 100 - unplayedPercentage;
 01765                    dto.Played = dto.PlayedPercentage.Value >= 100;
 1766                }
 1767                else
 1768                {
 01769                    dto.Played = (dto.UnplayedItemCount ?? 0) == 0;
 1770                }
 1771            }
 01772        }
 1773
 1774        /// <summary>
 1775        /// Contains constants used when reporting scan progress.
 1776        /// </summary>
 1777        private static class ProgressHelpers
 1778        {
 1779            /// <summary>
 1780            /// Reported after the folders immediate children are retrieved.
 1781            /// </summary>
 1782            public const int RetrievedChildren = 5;
 1783
 1784            /// <summary>
 1785            /// Reported after add, updating, or deleting child items from the LibraryManager.
 1786            /// </summary>
 1787            public const int UpdatedChildItems = 10;
 1788
 1789            /// <summary>
 1790            /// Reported once subfolders are scanned.
 1791            /// When scanning subfolders, the progress will be between [UpdatedItems, ScannedSubfolders].
 1792            /// </summary>
 1793            public const int ScannedSubfolders = 50;
 1794
 1795            /// <summary>
 1796            /// Reported once metadata is refreshed.
 1797            /// When refreshing metadata, the progress will be between [ScannedSubfolders, MetadataRefreshed].
 1798            /// </summary>
 1799            public const int RefreshedMetadata = 100;
 1800
 1801            /// <summary>
 1802            /// Gets the current progress given the previous step, next step, and progress in between.
 1803            /// </summary>
 1804            /// <param name="previousProgressStep">The previous progress step.</param>
 1805            /// <param name="nextProgressStep">The next progress step.</param>
 1806            /// <param name="currentProgress">The current progress step.</param>
 1807            /// <returns>The progress.</returns>
 1808            public static double GetProgress(int previousProgressStep, int nextProgressStep, double currentProgress)
 1809            {
 81810                return previousProgressStep + ((nextProgressStep - previousProgressStep) * (currentProgress / 100));
 1811            }
 1812        }
 1813    }
 1814}

Methods/Properties

.ctor()
get_SupportsThemeMedia()
get_IsPreSorted()
get_IsPhysicalRoot()
get_SupportsInheritedParentImages()
get_SupportsPlayedStatus()
get_IsFolder()
get_IsDisplayedAsFolder()
get_SupportsCumulativeRunTimeTicks()
get_SupportsDateLastMediaAdded()
get_FileNameWithoutExtension()
get_Children()
get_RecursiveChildren()
get_SupportsShortcutChildren()
get_FilterLinkedChildrenPerUser()
get_SupportsOwnedItems()
get_SupportsUserDataFromChildren()
CanDelete()
RequiresRefresh()
AddChild(MediaBrowser.Controller.Entities.BaseItem)
IsVisible(Jellyfin.Database.Implementations.Entities.User,System.Boolean)
LoadChildren()
GetRefreshProgress()
ValidateChildren(System.IProgress`1<System.Double>,System.Threading.CancellationToken)
ValidateChildren(System.IProgress`1<System.Double>,MediaBrowser.Controller.Providers.MetadataRefreshOptions,System.Boolean,System.Boolean,System.Threading.CancellationToken)
GetActualChildrenDictionary()
IsLibraryFolderAccessible(MediaBrowser.Controller.Providers.IDirectoryService,MediaBrowser.Controller.Entities.BaseItem,System.Boolean)
GetNonCachedChildren(MediaBrowser.Controller.Providers.IDirectoryService)
GetCachedChildren()
GetChildCount(Jellyfin.Database.Implementations.Entities.User)
GetRecursiveChildCount(Jellyfin.Database.Implementations.Entities.User)
QueryRecursive(MediaBrowser.Controller.Entities.InternalItemsQuery)
QueryWithPostFiltering2(MediaBrowser.Controller.Entities.InternalItemsQuery)
RequiresPostFiltering2(MediaBrowser.Controller.Entities.InternalItemsQuery)
RequiresPostFiltering(MediaBrowser.Controller.Entities.InternalItemsQuery)
SortItemsByRequest(MediaBrowser.Controller.Entities.InternalItemsQuery,System.Collections.Generic.IReadOnlyList`1<MediaBrowser.Controller.Entities.BaseItem>)
GetItems(MediaBrowser.Controller.Entities.InternalItemsQuery)
GetItemList(MediaBrowser.Controller.Entities.InternalItemsQuery)
GetItemsInternal(MediaBrowser.Controller.Entities.InternalItemsQuery)
PostFilterAndSort(System.Collections.Generic.IEnumerable`1<MediaBrowser.Controller.Entities.BaseItem>,MediaBrowser.Controller.Entities.InternalItemsQuery,System.Boolean)
CollapseBoxSetItemsIfNeeded(System.Collections.Generic.IEnumerable`1<MediaBrowser.Controller.Entities.BaseItem>,MediaBrowser.Controller.Entities.InternalItemsQuery,MediaBrowser.Controller.Entities.BaseItem,Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Controller.Configuration.IServerConfigurationManager,MediaBrowser.Controller.Collections.ICollectionManager)
CollapseBoxSetItems(MediaBrowser.Controller.Entities.InternalItemsQuery,MediaBrowser.Controller.Entities.BaseItem,Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Controller.Configuration.IServerConfigurationManager)
AllowBoxSetCollapsing(MediaBrowser.Controller.Entities.InternalItemsQuery)
GetChildren(Jellyfin.Database.Implementations.Entities.User,System.Boolean)
GetChildren(Jellyfin.Database.Implementations.Entities.User,System.Boolean,MediaBrowser.Controller.Entities.InternalItemsQuery)
GetEligibleChildrenForRecursiveChildren(Jellyfin.Database.Implementations.Entities.User)
AddChildren(Jellyfin.Database.Implementations.Entities.User,System.Boolean,System.Collections.Generic.Dictionary`2<System.Guid,MediaBrowser.Controller.Entities.BaseItem>,System.Boolean,MediaBrowser.Controller.Entities.InternalItemsQuery,System.Collections.Generic.HashSet`1<MediaBrowser.Controller.Entities.Folder>)
AddChildrenFromCollection(System.Collections.Generic.IEnumerable`1<MediaBrowser.Controller.Entities.BaseItem>,Jellyfin.Database.Implementations.Entities.User,System.Boolean,System.Collections.Generic.Dictionary`2<System.Guid,MediaBrowser.Controller.Entities.BaseItem>,System.Boolean,MediaBrowser.Controller.Entities.InternalItemsQuery,System.Collections.Generic.HashSet`1<MediaBrowser.Controller.Entities.Folder>)
GetRecursiveChildren(Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Controller.Entities.InternalItemsQuery)
GetRecursiveChildren()
GetRecursiveChildren(System.Boolean)
GetRecursiveChildren(System.Func`2<MediaBrowser.Controller.Entities.BaseItem,System.Boolean>)
GetRecursiveChildren(System.Func`2<MediaBrowser.Controller.Entities.BaseItem,System.Boolean>,System.Boolean)
AddChildrenToList(System.Collections.Generic.Dictionary`2<System.Guid,MediaBrowser.Controller.Entities.BaseItem>,System.Boolean,System.Boolean,System.Func`2<MediaBrowser.Controller.Entities.BaseItem,System.Boolean>)
GetLinkedChildren()
ContainsLinkedChildByItemId(System.Guid)
GetLinkedChildren(Jellyfin.Database.Implementations.Entities.User)
GetLinkedChildrenInfos()
RefreshLinkedChildren(System.Collections.Generic.IEnumerable`1<MediaBrowser.Model.IO.FileSystemMetadata>)
MarkPlayed(Jellyfin.Database.Implementations.Entities.User,System.Nullable`1<System.DateTime>,System.Boolean)
MarkUnplayed(Jellyfin.Database.Implementations.Entities.User)
IsPlayed(Jellyfin.Database.Implementations.Entities.User)
IsUnplayed(Jellyfin.Database.Implementations.Entities.User)
FillUserDataDtoValues(MediaBrowser.Model.Dto.UserItemDataDto,MediaBrowser.Controller.Entities.UserItemData,MediaBrowser.Model.Dto.BaseItemDto,Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Controller.Dto.DtoOptions)
GetProgress(System.Int32,System.Int32,System.Double)