| | 1 | | #nullable disable |
| | 2 | |
|
| | 3 | | #pragma warning disable CA1819, CS1591 |
| | 4 | |
|
| | 5 | | using System; |
| | 6 | | using System.Collections.Concurrent; |
| | 7 | | using System.Collections.Generic; |
| | 8 | | using System.Linq; |
| | 9 | | using System.Text.Json.Serialization; |
| | 10 | | using System.Threading; |
| | 11 | | using System.Threading.Tasks; |
| | 12 | | using Jellyfin.Extensions; |
| | 13 | | using MediaBrowser.Controller.IO; |
| | 14 | | using MediaBrowser.Controller.Library; |
| | 15 | | using MediaBrowser.Controller.Providers; |
| | 16 | | using MediaBrowser.Model.IO; |
| | 17 | |
|
| | 18 | | namespace MediaBrowser.Controller.Entities |
| | 19 | | { |
| | 20 | | /// <summary> |
| | 21 | | /// Specialized folder that can have items added to it's children by external entities. |
| | 22 | | /// Used for our RootFolder so plugins can add items. |
| | 23 | | /// </summary> |
| | 24 | | public class AggregateFolder : Folder |
| | 25 | | { |
| 43 | 26 | | private readonly Lock _childIdsLock = new(); |
| | 27 | |
|
| | 28 | | /// <summary> |
| | 29 | | /// The _virtual children. |
| | 30 | | /// </summary> |
| 43 | 31 | | private readonly ConcurrentBag<BaseItem> _virtualChildren = new ConcurrentBag<BaseItem>(); |
| | 32 | | private bool _requiresRefresh; |
| | 33 | | private Guid[] _childrenIds = null; |
| | 34 | |
|
| 43 | 35 | | public AggregateFolder() |
| | 36 | | { |
| 43 | 37 | | PhysicalLocationsList = Array.Empty<string>(); |
| 43 | 38 | | } |
| | 39 | |
|
| | 40 | | /// <summary> |
| | 41 | | /// Gets the virtual children. |
| | 42 | | /// </summary> |
| | 43 | | /// <value>The virtual children.</value> |
| 44 | 44 | | public ConcurrentBag<BaseItem> VirtualChildren => _virtualChildren; |
| | 45 | |
|
| | 46 | | [JsonIgnore] |
| 0 | 47 | | public override bool IsPhysicalRoot => true; |
| | 48 | |
|
| | 49 | | [JsonIgnore] |
| 0 | 50 | | public override bool SupportsPlayedStatus => false; |
| | 51 | |
|
| | 52 | | [JsonIgnore] |
| 61 | 53 | | public override string[] PhysicalLocations => PhysicalLocationsList; |
| | 54 | |
|
| | 55 | | public string[] PhysicalLocationsList { get; set; } |
| | 56 | |
|
| | 57 | | public override bool CanDelete() |
| | 58 | | { |
| 0 | 59 | | return false; |
| | 60 | | } |
| | 61 | |
|
| | 62 | | protected override FileSystemMetadata[] GetFileSystemChildren(IDirectoryService directoryService) |
| | 63 | | { |
| 26 | 64 | | return CreateResolveArgs(directoryService, true).FileSystemChildren; |
| | 65 | | } |
| | 66 | |
|
| | 67 | | protected override IReadOnlyList<BaseItem> LoadChildren() |
| 112 | 68 | | { |
| | 69 | | lock (_childIdsLock) |
| | 70 | | { |
| 112 | 71 | | if (_childrenIds is null || _childrenIds.Length == 0) |
| | 72 | | { |
| 76 | 73 | | var list = base.LoadChildren(); |
| 74 | 74 | | _childrenIds = list.Select(i => i.Id).ToArray(); |
| 74 | 75 | | return list; |
| | 76 | | } |
| | 77 | |
|
| 36 | 78 | | return _childrenIds.Select(LibraryManager.GetItemById).Where(i => i is not null).ToList(); |
| | 79 | | } |
| 110 | 80 | | } |
| | 81 | |
|
| | 82 | | private void ClearCache() |
| 127 | 83 | | { |
| | 84 | | lock (_childIdsLock) |
| | 85 | | { |
| 127 | 86 | | _childrenIds = null; |
| 127 | 87 | | } |
| 127 | 88 | | } |
| | 89 | |
|
| | 90 | | public override bool RequiresRefresh() |
| | 91 | | { |
| 22 | 92 | | var changed = base.RequiresRefresh() || _requiresRefresh; |
| | 93 | |
|
| 22 | 94 | | if (!changed) |
| | 95 | | { |
| 21 | 96 | | var locations = PhysicalLocations; |
| | 97 | |
|
| 21 | 98 | | var newLocations = CreateResolveArgs(new DirectoryService(FileSystem), false).PhysicalLocations; |
| | 99 | |
|
| 21 | 100 | | if (!locations.SequenceEqual(newLocations)) |
| | 101 | | { |
| 16 | 102 | | changed = true; |
| | 103 | | } |
| | 104 | | } |
| | 105 | |
|
| 22 | 106 | | return changed; |
| | 107 | | } |
| | 108 | |
|
| | 109 | | public override bool BeforeMetadataRefresh(bool replaceAllMetadata) |
| | 110 | | { |
| 22 | 111 | | ClearCache(); |
| | 112 | |
|
| 22 | 113 | | var changed = base.BeforeMetadataRefresh(replaceAllMetadata) || _requiresRefresh; |
| 22 | 114 | | _requiresRefresh = false; |
| 22 | 115 | | return changed; |
| | 116 | | } |
| | 117 | |
|
| | 118 | | private ItemResolveArgs CreateResolveArgs(IDirectoryService directoryService, bool setPhysicalLocations) |
| | 119 | | { |
| 47 | 120 | | ClearCache(); |
| | 121 | |
|
| 47 | 122 | | var path = ContainingFolderPath; |
| | 123 | |
|
| 47 | 124 | | var args = new ItemResolveArgs(ConfigurationManager.ApplicationPaths, LibraryManager) |
| 47 | 125 | | { |
| 47 | 126 | | FileInfo = FileSystem.GetDirectoryInfo(path) |
| 47 | 127 | | }; |
| | 128 | |
|
| | 129 | | // Gather child folder and files |
| 47 | 130 | | if (args.IsDirectory) |
| | 131 | | { |
| | 132 | | // When resolving the root, we need it's grandchildren (children of user views) |
| 47 | 133 | | var flattenFolderDepth = 2; |
| | 134 | |
|
| 47 | 135 | | var files = FileData.GetFilteredFileSystemEntries(directoryService, args.Path, FileSystem, CollectionFol |
| | 136 | |
|
| | 137 | | // Need to remove subpaths that may have been resolved from shortcuts |
| | 138 | | // Example: if \\server\movies exists, then strip out \\server\movies\action |
| 47 | 139 | | files = LibraryManager.NormalizeRootPathList(files).ToArray(); |
| | 140 | |
|
| 47 | 141 | | args.FileSystemChildren = files; |
| | 142 | | } |
| | 143 | |
|
| 47 | 144 | | _requiresRefresh = _requiresRefresh || !args.PhysicalLocations.SequenceEqual(PhysicalLocations); |
| 47 | 145 | | if (setPhysicalLocations) |
| | 146 | | { |
| 26 | 147 | | PhysicalLocationsList = args.PhysicalLocations; |
| | 148 | | } |
| | 149 | |
|
| 47 | 150 | | return args; |
| | 151 | | } |
| | 152 | |
|
| | 153 | | protected override IEnumerable<BaseItem> GetNonCachedChildren(IDirectoryService directoryService) |
| | 154 | | { |
| 26 | 155 | | return base.GetNonCachedChildren(directoryService).Concat(_virtualChildren); |
| | 156 | | } |
| | 157 | |
|
| | 158 | | protected override async Task ValidateChildrenInternal(IProgress<double> progress, bool recursive, bool refreshC |
| | 159 | | { |
| | 160 | | ClearCache(); |
| | 161 | |
|
| | 162 | | await base.ValidateChildrenInternal(progress, recursive, refreshChildMetadata, allowRemoveRoot, refreshOptio |
| | 163 | | .ConfigureAwait(false); |
| | 164 | |
|
| | 165 | | ClearCache(); |
| | 166 | | } |
| | 167 | |
|
| | 168 | | /// <summary> |
| | 169 | | /// Adds the virtual child. |
| | 170 | | /// </summary> |
| | 171 | | /// <param name="child">The child.</param> |
| | 172 | | /// <exception cref="ArgumentNullException">Throws if child is null.</exception> |
| | 173 | | public void AddVirtualChild(BaseItem child) |
| | 174 | | { |
| 21 | 175 | | ArgumentNullException.ThrowIfNull(child); |
| | 176 | |
|
| 21 | 177 | | _virtualChildren.Add(child); |
| 21 | 178 | | } |
| | 179 | |
|
| | 180 | | /// <summary> |
| | 181 | | /// Finds the virtual child. |
| | 182 | | /// </summary> |
| | 183 | | /// <param name="id">The id.</param> |
| | 184 | | /// <returns>BaseItem.</returns> |
| | 185 | | /// <exception cref="ArgumentNullException">The id is empty.</exception> |
| | 186 | | public BaseItem FindVirtualChild(Guid id) |
| | 187 | | { |
| 0 | 188 | | if (id.IsEmpty()) |
| | 189 | | { |
| 0 | 190 | | throw new ArgumentNullException(nameof(id)); |
| | 191 | | } |
| | 192 | |
|
| 0 | 193 | | foreach (var child in _virtualChildren) |
| | 194 | | { |
| 0 | 195 | | if (child.Id.Equals(id)) |
| | 196 | | { |
| 0 | 197 | | return child; |
| | 198 | | } |
| | 199 | | } |
| | 200 | |
|
| 0 | 201 | | return null; |
| 0 | 202 | | } |
| | 203 | | } |
| | 204 | | } |