< Summary - Jellyfin

Information
Class: MediaBrowser.Providers.Manager.MetadataService<T1, T2>
Assembly: MediaBrowser.Providers
File(s): /srv/git/jellyfin/MediaBrowser.Providers/Manager/MetadataService.cs
Line coverage
67%
Covered lines: 233
Uncovered lines: 110
Coverable lines: 343
Total lines: 1289
Line coverage: 67.9%
Branch coverage
67%
Covered branches: 235
Total branches: 350
Branch coverage: 67.1%
Method coverage

Feature is only available for sponsors

Upgrade to PRO version

Coverage history

Coverage history 0 25 50 75 100

Metrics

File(s)

/srv/git/jellyfin/MediaBrowser.Providers/Manager/MetadataService.cs

#LineLine coverage
 1#nullable disable
 2
 3#pragma warning disable CS1591
 4
 5using System;
 6using System.Collections.Generic;
 7using System.Linq;
 8using System.Net.Http;
 9using System.Threading;
 10using System.Threading.Tasks;
 11using Jellyfin.Extensions;
 12using MediaBrowser.Controller.Configuration;
 13using MediaBrowser.Controller.Entities;
 14using MediaBrowser.Controller.Entities.Audio;
 15using MediaBrowser.Controller.IO;
 16using MediaBrowser.Controller.Library;
 17using MediaBrowser.Controller.Persistence;
 18using MediaBrowser.Controller.Providers;
 19using MediaBrowser.Model.Configuration;
 20using MediaBrowser.Model.Entities;
 21using MediaBrowser.Model.IO;
 22using MediaBrowser.Model.Providers;
 23using Microsoft.Extensions.Logging;
 24
 25namespace MediaBrowser.Providers.Manager
 26{
 27    public abstract class MetadataService<TItemType, TIdType> : IMetadataService
 28        where TItemType : BaseItem, IHasLookupInfo<TIdType>, new()
 29        where TIdType : ItemLookupInfo, new()
 30    {
 31        protected MetadataService(
 32            IServerConfigurationManager serverConfigurationManager,
 33            ILogger<MetadataService<TItemType, TIdType>> logger,
 34            IProviderManager providerManager,
 35            IFileSystem fileSystem,
 36            ILibraryManager libraryManager,
 37            IExternalDataManager externalDataManager,
 38            IItemRepository itemRepository)
 39        {
 54640            ServerConfigurationManager = serverConfigurationManager;
 54641            Logger = logger;
 54642            ProviderManager = providerManager;
 54643            FileSystem = fileSystem;
 54644            LibraryManager = libraryManager;
 54645            ExternalDataManager = externalDataManager;
 54646            ItemRepository = itemRepository;
 54647            ImageProvider = new ItemImageProvider(Logger, ProviderManager, FileSystem);
 54648        }
 49
 50        protected ItemImageProvider ImageProvider { get; }
 51
 52        protected IServerConfigurationManager ServerConfigurationManager { get; }
 53
 54        protected ILogger<MetadataService<TItemType, TIdType>> Logger { get; }
 55
 56        protected IProviderManager ProviderManager { get; }
 57
 58        protected IFileSystem FileSystem { get; }
 59
 60        protected ILibraryManager LibraryManager { get; }
 61
 62        protected IExternalDataManager ExternalDataManager { get; }
 63
 64        protected IItemRepository ItemRepository { get; }
 65
 3266        protected virtual bool EnableUpdatingPremiereDateFromChildren => false;
 67
 3268        protected virtual bool EnableUpdatingGenresFromChildren => false;
 69
 3270        protected virtual bool EnableUpdatingStudiosFromChildren => false;
 71
 3272        protected virtual bool EnableUpdatingOfficialRatingFromChildren => false;
 73
 50474        public virtual int Order => 0;
 75
 76        private FileSystemMetadata TryGetFileSystemMetadata(string path, IDirectoryService directoryService)
 77        {
 78            try
 79            {
 6480                return directoryService.GetFileSystemEntry(path);
 81            }
 082            catch (Exception ex)
 83            {
 084                Logger.LogError(ex, "Error getting file {Path}", path);
 085                return null;
 86            }
 6487        }
 88
 89        public virtual async Task<ItemUpdateType> RefreshMetadata(BaseItem item, MetadataRefreshOptions refreshOptions, 
 90        {
 91            var itemOfType = (TItemType)item;
 92            var updateType = ItemUpdateType.None;
 93
 94            var libraryOptions = LibraryManager.GetLibraryOptions(item);
 95            var isFirstRefresh = item.DateLastRefreshed == DateTime.MinValue;
 96            var hasRefreshedMetadata = true;
 97            var hasRefreshedImages = true;
 98
 99            var requiresRefresh = libraryOptions.AutomaticRefreshIntervalDays > 0 && (DateTime.UtcNow - item.DateLastRef
 100
 101            if (!requiresRefresh && refreshOptions.MetadataRefreshMode != MetadataRefreshMode.None)
 102            {
 103                // TODO: If this returns true, should we instead just change metadata refresh mode to Full?
 104                requiresRefresh = item.RequiresRefresh();
 105
 106                if (requiresRefresh)
 107                {
 108                    Logger.LogDebug("Refreshing {Type} {Item} because item.RequiresRefresh() returned true", typeof(TIte
 109                }
 110            }
 111
 112            if (refreshOptions.RemoveOldMetadata && refreshOptions.ReplaceAllImages)
 113            {
 114                if (ImageProvider.RemoveImages(item))
 115                {
 116                    updateType |= ItemUpdateType.ImageUpdate;
 117                }
 118            }
 119
 120            var localImagesFailed = false;
 121            var allImageProviders = ProviderManager.GetImageProviders(item, refreshOptions).ToList();
 122
 123            // Only validate already registered images if we are replacing and saving locally
 124            if (item.IsSaveLocalMetadataEnabled() && refreshOptions.ReplaceAllImages)
 125            {
 126                item.ValidateImages();
 127            }
 128            else
 129            {
 130                // Run full image validation and register new local images
 131                try
 132                {
 133                    if (ImageProvider.ValidateImages(item, allImageProviders.OfType<ILocalImageProvider>(), refreshOptio
 134                    {
 135                        updateType |= ItemUpdateType.ImageUpdate;
 136                    }
 137                }
 138                catch (Exception ex)
 139                {
 140                    localImagesFailed = true;
 141                    Logger.LogError(ex, "Error validating images for {Item}", item.Path ?? item.Name ?? "Unknown name");
 142                }
 143            }
 144
 145            var metadataResult = new MetadataResult<TItemType>
 146            {
 147                Item = itemOfType
 148            };
 149
 150            var beforeSaveResult = await BeforeSave(itemOfType, isFirstRefresh || refreshOptions.ReplaceAllMetadata || r
 151                .ConfigureAwait(false);
 152            updateType |= beforeSaveResult;
 153
 154            updateType = await SaveInternal(item, refreshOptions, updateType, isFirstRefresh, requiresRefresh, metadataR
 155
 156            // Next run metadata providers
 157            if (refreshOptions.MetadataRefreshMode != MetadataRefreshMode.None)
 158            {
 159                var providers = GetProviders(item, libraryOptions, refreshOptions, isFirstRefresh, requiresRefresh)
 160                    .ToList();
 161
 162                if (providers.Count > 0 || isFirstRefresh || requiresRefresh)
 163                {
 164                    if (item.BeforeMetadataRefresh(refreshOptions.ReplaceAllMetadata))
 165                    {
 166                        updateType |= ItemUpdateType.MetadataImport;
 167                    }
 168                }
 169
 170                if (providers.Count > 0)
 171                {
 172                    var id = itemOfType.GetLookupInfo();
 173
 174                    if (refreshOptions.SearchResult is not null)
 175                    {
 176                        ApplySearchResult(id, refreshOptions.SearchResult);
 177                    }
 178
 179                    id.IsAutomated = refreshOptions.IsAutomated;
 180
 181                    var hasMetadataSavers = ProviderManager.GetMetadataSavers(item, libraryOptions).Any();
 182                    var result = await RefreshWithProviders(metadataResult, id, refreshOptions, providers, ImageProvider
 183
 184                    updateType |= result.UpdateType;
 185                    if (result.Failures > 0)
 186                    {
 187                        hasRefreshedMetadata = false;
 188                    }
 189                }
 190            }
 191
 192            // Next run remote image providers, but only if local image providers didn't throw an exception
 193            if (!localImagesFailed && refreshOptions.ImageRefreshMode > MetadataRefreshMode.ValidationOnly)
 194            {
 195                var providers = GetNonLocalImageProviders(item, allImageProviders, refreshOptions).ToList();
 196
 197                if (providers.Count > 0)
 198                {
 199                    var result = await ImageProvider.RefreshImages(itemOfType, libraryOptions, providers, refreshOptions
 200
 201                    updateType |= result.UpdateType;
 202                    if (result.Failures > 0)
 203                    {
 204                        hasRefreshedImages = false;
 205                    }
 206                }
 207            }
 208
 209            if (hasRefreshedMetadata && hasRefreshedImages)
 210            {
 211                item.DateLastRefreshed = DateTime.UtcNow;
 212                updateType |= item.OnMetadataChanged();
 213            }
 214
 215            updateType = await SaveInternal(item, refreshOptions, updateType, isFirstRefresh, requiresRefresh, metadataR
 216
 217            await AfterMetadataRefresh(itemOfType, refreshOptions, cancellationToken).ConfigureAwait(false);
 218
 219            return updateType;
 220
 221            async Task<ItemUpdateType> SaveInternal(BaseItem item, MetadataRefreshOptions refreshOptions, ItemUpdateType
 222            {
 223                // Save if changes were made, or it's never been saved before
 224                if (refreshOptions.ForceSave || updateType > ItemUpdateType.None || isFirstRefresh || refreshOptions.Rep
 225                {
 226                    if (item.IsFileProtocol)
 227                    {
 228                        var file = TryGetFileSystemMetadata(item.Path, refreshOptions.DirectoryService);
 229                        if (file is not null)
 230                        {
 231                            item.DateModified = file.LastWriteTimeUtc;
 232                        }
 233                    }
 234
 235                    // If any of these properties are set then make sure the updateType is not None, just to force every
 236                    if (refreshOptions.ForceSave || refreshOptions.ReplaceAllMetadata)
 237                    {
 238                        updateType |= ItemUpdateType.MetadataDownload;
 239                    }
 240
 241                    // Save to database
 242                    await SaveItemAsync(metadataResult, updateType, cancellationToken).ConfigureAwait(false);
 243                }
 244
 245                return updateType;
 246            }
 247        }
 248
 249        private void ApplySearchResult(ItemLookupInfo lookupInfo, RemoteSearchResult result)
 250        {
 251            // Episode and Season do not support Identify, so the search results are the Series'
 252            switch (lookupInfo)
 253            {
 254                case EpisodeInfo episodeInfo:
 0255                    episodeInfo.SeriesProviderIds = result.ProviderIds;
 0256                    episodeInfo.ProviderIds.Clear();
 0257                    break;
 258                case SeasonInfo seasonInfo:
 0259                    seasonInfo.SeriesProviderIds = result.ProviderIds;
 0260                    seasonInfo.ProviderIds.Clear();
 0261                    break;
 262                default:
 0263                    lookupInfo.ProviderIds = result.ProviderIds;
 0264                    lookupInfo.Name = result.Name;
 0265                    lookupInfo.Year = result.ProductionYear;
 266                    break;
 267            }
 0268        }
 269
 270        protected async Task SaveItemAsync(MetadataResult<TItemType> result, ItemUpdateType reason, CancellationToken ca
 271        {
 272            await result.Item.UpdateToRepositoryAsync(reason, cancellationToken).ConfigureAwait(false);
 273            if (result.Item.SupportsPeople && result.People is not null)
 274            {
 275                var baseItem = result.Item;
 276
 277                await LibraryManager.UpdatePeopleAsync(baseItem, result.People, cancellationToken).ConfigureAwait(false)
 278            }
 279        }
 280
 281        protected virtual Task AfterMetadataRefresh(TItemType item, MetadataRefreshOptions refreshOptions, CancellationT
 282        {
 51283            item.AfterMetadataRefresh();
 51284            return Task.CompletedTask;
 285        }
 286
 287        /// <summary>
 288        /// Before the save.
 289        /// </summary>
 290        /// <param name="item">The item.</param>
 291        /// <param name="isFullRefresh">if set to <c>true</c> [is full refresh].</param>
 292        /// <param name="currentUpdateType">Type of the current update.</param>
 293        /// <returns>ItemUpdateType.</returns>
 294        private async Task<ItemUpdateType> BeforeSave(TItemType item, bool isFullRefresh, ItemUpdateType currentUpdateTy
 295        {
 296            var updateType = BeforeSaveInternal(item, isFullRefresh, currentUpdateType);
 297
 298            updateType |= item.OnMetadataChanged();
 299
 300            if (updateType == ItemUpdateType.None)
 301            {
 302                if (!await ItemRepository.ItemExistsAsync(item.Id).ConfigureAwait(false))
 303                {
 304                    return ItemUpdateType.MetadataImport;
 305                }
 306            }
 307
 308            return updateType;
 309        }
 310
 311        protected virtual ItemUpdateType BeforeSaveInternal(TItemType item, bool isFullRefresh, ItemUpdateType updateTyp
 312        {
 53313            if (EnableUpdateMetadataFromChildren(item, isFullRefresh, updateType))
 314            {
 0315                if (isFullRefresh || updateType > ItemUpdateType.None)
 316                {
 0317                    var children = GetChildrenForMetadataUpdates(item);
 318
 0319                    updateType = UpdateMetadataFromChildren(item, children, isFullRefresh, updateType);
 320                }
 321            }
 322
 53323            var presentationUniqueKey = item.CreatePresentationUniqueKey();
 53324            if (!string.Equals(item.PresentationUniqueKey, presentationUniqueKey, StringComparison.Ordinal))
 325            {
 28326                item.PresentationUniqueKey = presentationUniqueKey;
 28327                updateType |= ItemUpdateType.MetadataImport;
 328            }
 329
 330            // Cleanup extracted files if source file was modified
 53331            var itemPath = item.Path;
 53332            if (!string.IsNullOrEmpty(itemPath))
 333            {
 53334                var info = FileSystem.GetFileSystemInfo(itemPath);
 53335                if (info.Exists && item.HasChanged(info.LastWriteTimeUtc))
 336                {
 1337                    Logger.LogDebug("File modification time changed from {Then} to {Now}: {Path}", item.DateModified, in
 338
 1339                    item.DateModified = info.LastWriteTimeUtc;
 1340                    if (ServerConfigurationManager.GetMetadataConfiguration().UseFileCreationTimeForDateAdded)
 341                    {
 1342                        item.DateCreated = info.CreationTimeUtc;
 343                    }
 344
 1345                    if (item is Video video)
 346                    {
 0347                        Logger.LogInformation("File changed, pruning extracted data: {Path}", item.Path);
 0348                        ExternalDataManager.DeleteExternalItemDataAsync(video, CancellationToken.None).GetAwaiter().GetR
 349                    }
 350
 1351                    updateType |= ItemUpdateType.MetadataImport;
 352                }
 353            }
 354
 53355            return updateType;
 356        }
 357
 358        protected virtual bool EnableUpdateMetadataFromChildren(TItemType item, bool isFullRefresh, ItemUpdateType curre
 359        {
 53360            if (isFullRefresh || currentUpdateType > ItemUpdateType.None)
 361            {
 32362                if (EnableUpdatingPremiereDateFromChildren || EnableUpdatingGenresFromChildren || EnableUpdatingStudiosF
 363                {
 0364                    return true;
 365                }
 366
 32367                if (item is Folder folder)
 368                {
 32369                    return folder.SupportsDateLastMediaAdded || folder.SupportsCumulativeRunTimeTicks;
 370                }
 371            }
 372
 21373            return false;
 374        }
 375
 376        protected virtual IReadOnlyList<BaseItem> GetChildrenForMetadataUpdates(TItemType item)
 377        {
 0378            if (item is Folder folder)
 379            {
 0380                return folder.GetRecursiveChildren();
 381            }
 382
 0383            return [];
 384        }
 385
 386        protected virtual ItemUpdateType UpdateMetadataFromChildren(TItemType item, IReadOnlyList<BaseItem> children, bo
 387        {
 0388            var updateType = ItemUpdateType.None;
 389
 0390            if (isFullRefresh || currentUpdateType > ItemUpdateType.None)
 391            {
 0392                updateType |= UpdateCumulativeRunTimeTicks(item, children);
 0393                updateType |= UpdateDateLastMediaAdded(item, children);
 394
 395                // don't update user-changeable metadata for locked items
 0396                if (item.IsLocked)
 397                {
 0398                    return updateType;
 399                }
 400
 0401                if (EnableUpdatingPremiereDateFromChildren)
 402                {
 0403                    updateType |= UpdatePremiereDate(item, children);
 404                }
 405
 0406                if (EnableUpdatingGenresFromChildren)
 407                {
 0408                    updateType |= UpdateGenres(item, children);
 409                }
 410
 0411                if (EnableUpdatingStudiosFromChildren)
 412                {
 0413                    updateType |= UpdateStudios(item, children);
 414                }
 415
 0416                if (EnableUpdatingOfficialRatingFromChildren)
 417                {
 0418                    updateType |= UpdateOfficialRating(item, children);
 419                }
 420            }
 421
 0422            return updateType;
 423        }
 424
 425        private ItemUpdateType UpdateCumulativeRunTimeTicks(TItemType item, IReadOnlyList<BaseItem> children)
 426        {
 0427            if (item is Folder folder && folder.SupportsCumulativeRunTimeTicks)
 428            {
 0429                long ticks = 0;
 430
 0431                foreach (var child in children)
 432                {
 0433                    if (!child.IsFolder)
 434                    {
 0435                        ticks += child.RunTimeTicks ?? 0;
 436                    }
 437                }
 438
 0439                if (!folder.RunTimeTicks.HasValue || folder.RunTimeTicks.Value != ticks)
 440                {
 0441                    folder.RunTimeTicks = ticks;
 0442                    return ItemUpdateType.MetadataImport;
 443                }
 444            }
 445
 0446            return ItemUpdateType.None;
 447        }
 448
 449        private ItemUpdateType UpdateDateLastMediaAdded(TItemType item, IReadOnlyList<BaseItem> children)
 450        {
 0451            var updateType = ItemUpdateType.None;
 452
 0453            if (item is Folder folder && folder.SupportsDateLastMediaAdded)
 454            {
 0455                var dateLastMediaAdded = DateTime.MinValue;
 0456                var any = false;
 457
 0458                foreach (var child in children)
 459                {
 460                    // Exclude any folders and virtual items since they are only placeholders
 0461                    if (!child.IsFolder && !child.IsVirtualItem)
 462                    {
 0463                        var childDateCreated = child.DateCreated;
 0464                        if (childDateCreated > dateLastMediaAdded)
 465                        {
 0466                            dateLastMediaAdded = childDateCreated;
 467                        }
 468
 0469                        any = true;
 470                    }
 471                }
 472
 0473                if ((!folder.DateLastMediaAdded.HasValue && any) || folder.DateLastMediaAdded != dateLastMediaAdded)
 474                {
 0475                    folder.DateLastMediaAdded = dateLastMediaAdded;
 0476                    updateType = ItemUpdateType.MetadataImport;
 477                }
 478            }
 479
 0480            return updateType;
 481        }
 482
 483        private ItemUpdateType UpdatePremiereDate(TItemType item, IReadOnlyList<BaseItem> children)
 484        {
 0485            var updateType = ItemUpdateType.None;
 486
 0487            if (children.Count == 0)
 488            {
 0489                return updateType;
 490            }
 491
 0492            var date = children.Select(i => i.PremiereDate ?? DateTime.MaxValue).Min();
 493
 0494            var originalPremiereDate = item.PremiereDate;
 0495            var originalProductionYear = item.ProductionYear;
 496
 0497            if (date > DateTime.MinValue && date < DateTime.MaxValue)
 498            {
 0499                item.PremiereDate = date;
 0500                item.ProductionYear = date.Year;
 501            }
 502            else
 503            {
 0504                var year = children.Select(i => i.ProductionYear ?? 0).Min();
 505
 0506                if (year > 0)
 507                {
 0508                    item.ProductionYear = year;
 509                }
 510            }
 511
 0512            if ((originalPremiereDate ?? DateTime.MinValue) != (item.PremiereDate ?? DateTime.MinValue)
 0513                || (originalProductionYear ?? -1) != (item.ProductionYear ?? -1))
 514            {
 0515                updateType |= ItemUpdateType.MetadataEdit;
 516            }
 517
 0518            return updateType;
 519        }
 520
 521        private ItemUpdateType UpdateGenres(TItemType item, IReadOnlyList<BaseItem> children)
 522        {
 0523            var updateType = ItemUpdateType.None;
 524
 0525            if (!item.LockedFields.Contains(MetadataField.Genres))
 526            {
 0527                var currentList = item.Genres;
 528
 0529                item.Genres = children.SelectMany(i => i.Genres)
 0530                    .Distinct(StringComparer.OrdinalIgnoreCase)
 0531                    .ToArray();
 532
 0533                if (currentList.Length != item.Genres.Length || !currentList.Order().SequenceEqual(item.Genres.Order(), 
 534                {
 0535                    updateType |= ItemUpdateType.MetadataEdit;
 536                }
 537            }
 538
 0539            return updateType;
 540        }
 541
 542        private ItemUpdateType UpdateStudios(TItemType item, IReadOnlyList<BaseItem> children)
 543        {
 0544            var updateType = ItemUpdateType.None;
 545
 0546            if (!item.LockedFields.Contains(MetadataField.Studios))
 547            {
 0548                var currentList = item.Studios;
 549
 0550                item.Studios = children.SelectMany(i => i.Studios)
 0551                    .Distinct(StringComparer.OrdinalIgnoreCase)
 0552                    .ToArray();
 553
 0554                if (currentList.Length != item.Studios.Length || !currentList.Order().SequenceEqual(item.Studios.Order()
 555                {
 0556                    updateType |= ItemUpdateType.MetadataEdit;
 557                }
 558            }
 559
 0560            return updateType;
 561        }
 562
 563        private ItemUpdateType UpdateOfficialRating(TItemType item, IReadOnlyList<BaseItem> children)
 564        {
 0565            var updateType = ItemUpdateType.None;
 566
 0567            if (!item.LockedFields.Contains(MetadataField.OfficialRating))
 568            {
 0569                if (item.UpdateRatingToItems(children))
 570                {
 0571                    updateType |= ItemUpdateType.MetadataEdit;
 572                }
 573            }
 574
 0575            return updateType;
 576        }
 577
 578        /// <summary>
 579        /// Gets the providers.
 580        /// </summary>
 581        /// <param name="item">A media item.</param>
 582        /// <param name="libraryOptions">The LibraryOptions to use.</param>
 583        /// <param name="options">The MetadataRefreshOptions to use.</param>
 584        /// <param name="isFirstRefresh">Specifies first refresh mode.</param>
 585        /// <param name="requiresRefresh">Specifies refresh mode.</param>
 586        /// <returns>IEnumerable{`0}.</returns>
 587        protected IEnumerable<IMetadataProvider> GetProviders(BaseItem item, LibraryOptions libraryOptions, MetadataRefr
 588        {
 589            // Get providers to refresh
 53590            var providers = ProviderManager.GetMetadataProviders<TItemType>(item, libraryOptions).ToList();
 591
 53592            var metadataRefreshMode = options.MetadataRefreshMode;
 593
 594            // Run all if either of these flags are true
 53595            var runAllProviders = options.ReplaceAllMetadata ||
 53596                metadataRefreshMode == MetadataRefreshMode.FullRefresh ||
 53597                (isFirstRefresh && metadataRefreshMode >= MetadataRefreshMode.Default) ||
 53598                (requiresRefresh && metadataRefreshMode >= MetadataRefreshMode.Default);
 599
 53600            if (!runAllProviders)
 601            {
 21602                var providersWithChanges = providers
 21603                    .Where(i =>
 21604                    {
 21605                        if (i is IHasItemChangeMonitor hasFileChangeMonitor)
 21606                        {
 21607                            return HasChanged(item, hasFileChangeMonitor, options.DirectoryService);
 21608                        }
 21609
 21610                        return false;
 21611                    })
 21612                    .ToList();
 613
 21614                if (providersWithChanges.Count == 0)
 615                {
 0616                    providers = new List<IMetadataProvider<TItemType>>();
 617                }
 618                else
 619                {
 21620                    var anyRemoteProvidersChanged = providersWithChanges.OfType<IRemoteMetadataProvider>()
 21621                        .Any();
 622
 21623                    var anyLocalProvidersChanged = providersWithChanges.OfType<ILocalMetadataProvider>()
 21624                        .Any();
 625
 21626                    var anyLocalPreRefreshProvidersChanged = providersWithChanges.OfType<IPreRefreshProvider>()
 21627                        .Any();
 628
 21629                    providers = providers.Where(i =>
 21630                    {
 21631                        // If any provider reports a change, always run local ones as well
 21632                        if (i is ILocalMetadataProvider)
 21633                        {
 21634                            return anyRemoteProvidersChanged || anyLocalProvidersChanged || anyLocalPreRefreshProvidersC
 21635                        }
 21636
 21637                        // If any remote providers changed, run them all so that priorities can be honored
 21638                        if (i is IRemoteMetadataProvider)
 21639                        {
 21640                            if (options.MetadataRefreshMode == MetadataRefreshMode.ValidationOnly)
 21641                            {
 21642                                return false;
 21643                            }
 21644
 21645                            return anyRemoteProvidersChanged;
 21646                        }
 21647
 21648                        // Run custom refresh providers if they report a change or any remote providers change
 21649                        return anyRemoteProvidersChanged || providersWithChanges.Contains(i);
 21650                    }).ToList();
 651                }
 652            }
 653
 53654            return providers;
 655        }
 656
 657        protected virtual IEnumerable<IImageProvider> GetNonLocalImageProviders(BaseItem item, IEnumerable<IImageProvide
 658        {
 659            // Get providers to refresh
 53660            var providers = allImageProviders.Where(i => i is not ILocalImageProvider);
 661
 53662            var dateLastImageRefresh = item.DateLastRefreshed;
 663
 664            // Run all if either of these flags are true
 53665            var runAllProviders = options.ImageRefreshMode == MetadataRefreshMode.FullRefresh || dateLastImageRefresh.Da
 666
 53667            if (!runAllProviders)
 668            {
 25669                providers = providers
 25670                    .Where(i =>
 25671                    {
 25672                        if (i is IHasItemChangeMonitor hasFileChangeMonitor)
 25673                        {
 25674                            return HasChanged(item, hasFileChangeMonitor, options.DirectoryService);
 25675                        }
 25676
 25677                        return false;
 25678                    });
 679            }
 680
 53681            return providers;
 682        }
 683
 684        public bool CanRefresh(BaseItem item)
 685        {
 962686            return item is TItemType;
 687        }
 688
 689        public bool CanRefreshPrimary(Type type)
 690        {
 1285691            return type == typeof(TItemType);
 692        }
 693
 694        protected virtual async Task<RefreshResult> RefreshWithProviders(
 695            MetadataResult<TItemType> metadata,
 696            TIdType id,
 697            MetadataRefreshOptions options,
 698            ICollection<IMetadataProvider> providers,
 699            ItemImageProvider imageService,
 700            bool isSavingMetadata,
 701            CancellationToken cancellationToken)
 702        {
 703            var refreshResult = new RefreshResult
 704            {
 705                UpdateType = ItemUpdateType.None
 706            };
 707
 708            var item = metadata.Item;
 709
 710            var customProviders = providers.OfType<ICustomMetadataProvider<TItemType>>().ToList();
 711            var logName = !item.IsFileProtocol ? item.Name ?? item.Path : item.Path ?? item.Name;
 712
 713            foreach (var provider in customProviders.Where(i => i is IPreRefreshProvider))
 714            {
 715                await RunCustomProvider(provider, item, logName, options, refreshResult, cancellationToken).ConfigureAwa
 716            }
 717
 718            if (item.IsLocked)
 719            {
 720                return refreshResult;
 721            }
 722
 723            var temp = new MetadataResult<TItemType>
 724            {
 725                Item = CreateNew()
 726            };
 727            temp.Item.Path = item.Path;
 728            temp.Item.Id = item.Id;
 729            temp.Item.ParentIndexNumber = item.ParentIndexNumber;
 730            temp.Item.PreferredMetadataCountryCode = item.PreferredMetadataCountryCode;
 731            temp.Item.PreferredMetadataLanguage = item.PreferredMetadataLanguage;
 732
 733            var foundImageTypes = new List<ImageType>();
 734
 735            // Do not execute local providers if we are identifying or replacing with local metadata saving enabled
 736            if (options.SearchResult is null && !(isSavingMetadata && options.ReplaceAllMetadata))
 737            {
 738                foreach (var provider in providers.OfType<ILocalMetadataProvider<TItemType>>())
 739                {
 740                    var providerName = provider.GetType().Name;
 741                    Logger.LogDebug("Running {Provider} for {Item}", providerName, logName);
 742
 743                    var itemInfo = new ItemInfo(item);
 744
 745                    try
 746                    {
 747                        var localItem = await provider.GetMetadata(itemInfo, options.DirectoryService, cancellationToken
 748
 749                        if (localItem.HasMetadata)
 750                        {
 751                            foreach (var remoteImage in localItem.RemoteImages)
 752                            {
 753                                try
 754                                {
 755                                    if (item.ImageInfos.Any(x => x.Type == remoteImage.Type)
 756                                        && !options.IsReplacingImage(remoteImage.Type))
 757                                    {
 758                                        continue;
 759                                    }
 760
 761                                    await ProviderManager.SaveImage(item, remoteImage.Url, remoteImage.Type, null, cance
 762                                    refreshResult.UpdateType |= ItemUpdateType.ImageUpdate;
 763
 764                                    // remember imagetype that has just been downloaded
 765                                    foundImageTypes.Add(remoteImage.Type);
 766                                }
 767                                catch (HttpRequestException ex)
 768                                {
 769                                    Logger.LogError(ex, "Could not save {ImageType} image: {Url}", Enum.GetName(remoteIm
 770                                }
 771                            }
 772
 773                            if (foundImageTypes.Count > 0)
 774                            {
 775                                imageService.UpdateReplaceImages(options, foundImageTypes);
 776                            }
 777
 778                            if (imageService.MergeImages(item, localItem.Images, options))
 779                            {
 780                                refreshResult.UpdateType |= ItemUpdateType.ImageUpdate;
 781                            }
 782
 783                            MergeData(localItem, temp, [], false, true);
 784                            refreshResult.UpdateType |= ItemUpdateType.MetadataImport;
 785
 786                            break;
 787                        }
 788
 789                        Logger.LogDebug("{Provider} returned no metadata for {Item}", providerName, logName);
 790                    }
 791                    catch (OperationCanceledException)
 792                    {
 793                        throw;
 794                    }
 795                    catch (Exception ex)
 796                    {
 797                        Logger.LogError(ex, "Error in {Provider}", provider.Name);
 798
 799                        // If a local provider fails, consider that a failure
 800                        refreshResult.ErrorMessage = ex.Message;
 801                    }
 802                }
 803            }
 804
 805            var isLocalLocked = temp.Item.IsLocked;
 806            if (!isLocalLocked && (options.ReplaceAllMetadata || options.MetadataRefreshMode > MetadataRefreshMode.Valid
 807            {
 808                var remoteResult = await ExecuteRemoteProviders(temp, logName, false, id, providers.OfType<IRemoteMetada
 809                    .ConfigureAwait(false);
 810
 811                refreshResult.UpdateType |= remoteResult.UpdateType;
 812                refreshResult.ErrorMessage = remoteResult.ErrorMessage;
 813                refreshResult.Failures += remoteResult.Failures;
 814            }
 815
 816            if (providers.Any(i => i is not ICustomMetadataProvider))
 817            {
 818                if (refreshResult.UpdateType > ItemUpdateType.None)
 819                {
 820                    if (!options.RemoveOldMetadata)
 821                    {
 822                        // Add existing metadata to provider result if it does not exist there
 823                        MergeData(metadata, temp, [], false, false);
 824                    }
 825
 826                    if (isLocalLocked)
 827                    {
 828                        MergeData(temp, metadata, item.LockedFields, true, true);
 829                    }
 830                    else
 831                    {
 832                        var shouldReplace = (options.MetadataRefreshMode > MetadataRefreshMode.ValidationOnly && options
 833                            // Case for Scan for new and updated files
 834                            || (options.MetadataRefreshMode == MetadataRefreshMode.Default && !options.ReplaceAllMetadat
 835                        MergeData(temp, metadata, item.LockedFields, shouldReplace, true);
 836                    }
 837                }
 838            }
 839
 840            foreach (var provider in customProviders.Where(i => i is not IPreRefreshProvider))
 841            {
 842                await RunCustomProvider(provider, item, logName, options, refreshResult, cancellationToken).ConfigureAwa
 843            }
 844
 845            return refreshResult;
 846        }
 847
 848        private async Task RunCustomProvider(ICustomMetadataProvider<TItemType> provider, TItemType item, string logName
 849        {
 850            Logger.LogDebug("Running {Provider} for {Item}", provider.GetType().Name, logName);
 851
 852            try
 853            {
 854                refreshResult.UpdateType |= await provider.FetchAsync(item, options, cancellationToken).ConfigureAwait(f
 855            }
 856            catch (OperationCanceledException)
 857            {
 858                throw;
 859            }
 860            catch (Exception ex)
 861            {
 862                refreshResult.ErrorMessage = ex.Message;
 863                Logger.LogError(ex, "Error in {Provider}", provider.Name);
 864            }
 865        }
 866
 867        protected virtual TItemType CreateNew()
 868        {
 53869            return new TItemType();
 870        }
 871
 872        private async Task<RefreshResult> ExecuteRemoteProviders(MetadataResult<TItemType> temp, string logName, bool re
 873        {
 874            var refreshResult = new RefreshResult();
 875
 876            if (id is not null)
 877            {
 878                MergeNewData(temp.Item, id);
 879            }
 880
 881            foreach (var provider in providers)
 882            {
 883                var providerName = provider.GetType().Name;
 884                Logger.LogDebug("Running {Provider} for {Item}", providerName, logName);
 885
 886                try
 887                {
 888                    var result = await provider.GetMetadata(id, cancellationToken).ConfigureAwait(false);
 889
 890                    if (result.HasMetadata)
 891                    {
 892                        result.Provider = provider.Name;
 893
 894                        MergeData(result, temp, [], replaceData, false);
 895                        MergeNewData(temp.Item, id);
 896
 897                        refreshResult.UpdateType |= ItemUpdateType.MetadataDownload;
 898                    }
 899                    else
 900                    {
 901                        Logger.LogDebug("{Provider} returned no metadata for {Item}", providerName, logName);
 902                    }
 903                }
 904                catch (OperationCanceledException)
 905                {
 906                    throw;
 907                }
 908                catch (Exception ex)
 909                {
 910                    refreshResult.Failures++;
 911                    refreshResult.ErrorMessage = ex.Message;
 912                    Logger.LogError(ex, "Error in {Provider}", provider.Name);
 913                }
 914            }
 915
 916            return refreshResult;
 917        }
 918
 919        private void MergeNewData(TItemType source, TIdType lookupInfo)
 920        {
 921            // Copy new provider id's that may have been obtained
 106922            foreach (var providerId in source.ProviderIds)
 923            {
 0924                var key = providerId.Key;
 925
 926                // Don't replace existing Id's.
 0927                lookupInfo.ProviderIds.TryAdd(key, providerId.Value);
 928            }
 53929        }
 930
 931        private bool HasChanged(BaseItem item, IHasItemChangeMonitor changeMonitor, IDirectoryService directoryService)
 932        {
 933            try
 934            {
 21935                var hasChanged = changeMonitor.HasChanged(item, directoryService);
 936
 21937                if (hasChanged)
 938                {
 21939                    Logger.LogDebug("{Monitor} reports change to {Item}", changeMonitor.GetType().Name, item.Path ?? ite
 940                }
 941
 21942                return hasChanged;
 943            }
 0944            catch (Exception ex)
 945            {
 0946                Logger.LogError(ex, "Error in {Monitor}.HasChanged", changeMonitor.GetType().Name);
 0947                return false;
 948            }
 21949        }
 950
 951        /// <summary>
 952        /// Merges metadata from source into target.
 953        /// </summary>
 954        /// <param name="source">The source for new metadata.</param>
 955        /// <param name="target">The target to insert new metadata into.</param>
 956        /// <param name="lockedFields">The fields that are locked and should not be updated.</param>
 957        /// <param name="replaceData"><c>true</c> if existing data should be replaced.</param>
 958        /// <param name="mergeMetadataSettings"><c>true</c> if the metadata settings in target should be updated to matc
 959        /// <exception cref="ArgumentException">Thrown if source or target are null.</exception>
 960        protected virtual void MergeData(
 961            MetadataResult<TItemType> source,
 962            MetadataResult<TItemType> target,
 963            MetadataField[] lockedFields,
 964            bool replaceData,
 965            bool mergeMetadataSettings)
 966        {
 0967            MergeBaseItemData(source, target, lockedFields, replaceData, mergeMetadataSettings);
 0968        }
 969
 970        internal static void MergeBaseItemData(
 971            MetadataResult<TItemType> sourceResult,
 972            MetadataResult<TItemType> targetResult,
 973            MetadataField[] lockedFields,
 974            bool replaceData,
 975            bool mergeMetadataSettings)
 976        {
 129977            var source = sourceResult.Item;
 129978            var target = targetResult.Item;
 979
 129980            ArgumentNullException.ThrowIfNull(sourceResult);
 129981            ArgumentNullException.ThrowIfNull(targetResult);
 982
 129983            if (!lockedFields.Contains(MetadataField.Name))
 984            {
 126985                if (replaceData || string.IsNullOrEmpty(target.Name))
 986                {
 987                    // Safeguard against incoming data having an empty name
 125988                    if (!string.IsNullOrWhiteSpace(source.Name))
 989                    {
 3990                        target.Name = source.Name;
 991                    }
 992                }
 993            }
 994
 129995            if (replaceData || string.IsNullOrEmpty(target.OriginalTitle))
 996            {
 128997                target.OriginalTitle = source.OriginalTitle;
 998            }
 999
 1291000            if (replaceData || !target.CommunityRating.HasValue)
 1001            {
 1281002                target.CommunityRating = source.CommunityRating;
 1003            }
 1004
 1291005            if (replaceData || !target.EndDate.HasValue)
 1006            {
 1281007                target.EndDate = source.EndDate;
 1008            }
 1009
 1291010            if (!lockedFields.Contains(MetadataField.Genres))
 1011            {
 1271012                if (replaceData || target.Genres.Length == 0)
 1013                {
 1261014                    target.Genres = source.Genres;
 1015                }
 1016            }
 1017
 1291018            if (replaceData || !target.IndexNumber.HasValue)
 1019            {
 1281020                target.IndexNumber = source.IndexNumber;
 1021            }
 1022
 1291023            if (!lockedFields.Contains(MetadataField.OfficialRating))
 1024            {
 1261025                if (replaceData || string.IsNullOrEmpty(target.OfficialRating))
 1026                {
 1251027                    target.OfficialRating = source.OfficialRating;
 1028                }
 1029            }
 1030
 1291031            if (replaceData || string.IsNullOrEmpty(target.CustomRating))
 1032            {
 1281033                target.CustomRating = source.CustomRating;
 1034            }
 1035
 1291036            if (replaceData || string.IsNullOrEmpty(target.Tagline))
 1037            {
 1281038                target.Tagline = source.Tagline;
 1039            }
 1040
 1291041            if (!lockedFields.Contains(MetadataField.Overview))
 1042            {
 1261043                if (replaceData || string.IsNullOrEmpty(target.Overview))
 1044                {
 1251045                    target.Overview = source.Overview;
 1046                }
 1047            }
 1048
 1291049            if (replaceData || !target.ParentIndexNumber.HasValue)
 1050            {
 1281051                target.ParentIndexNumber = source.ParentIndexNumber;
 1052            }
 1053
 1291054            if (!lockedFields.Contains(MetadataField.Cast))
 1055            {
 1281056                if (replaceData || targetResult.People is null || targetResult.People.Count == 0)
 1057                {
 1241058                    targetResult.People = sourceResult.People;
 1059                }
 41060                else if (sourceResult.People is not null && sourceResult.People.Count > 0)
 1061                {
 41062                    MergePeople(sourceResult.People, targetResult.People);
 1063                }
 1064            }
 1065
 1291066            if (replaceData || !target.PremiereDate.HasValue)
 1067            {
 1281068                target.PremiereDate = source.PremiereDate;
 1069            }
 1070
 1291071            if (replaceData || !target.ProductionYear.HasValue)
 1072            {
 1281073                target.ProductionYear = source.ProductionYear;
 1074            }
 1075
 1291076            if (!lockedFields.Contains(MetadataField.Runtime))
 1077            {
 1291078                if (replaceData || !target.RunTimeTicks.HasValue)
 1079                {
 1291080                    if (target is not Audio && target is not Video)
 1081                    {
 491082                        target.RunTimeTicks = source.RunTimeTicks;
 1083                    }
 1084                }
 1085            }
 1086
 1291087            if (!lockedFields.Contains(MetadataField.Studios))
 1088            {
 1271089                if (replaceData || target.Studios.Length == 0)
 1090                {
 1261091                    target.Studios = source.Studios;
 1092                }
 1093                else
 1094                {
 11095                    target.Studios = target.Studios.Concat(source.Studios).Distinct(StringComparer.OrdinalIgnoreCase).To
 1096                }
 1097            }
 1098
 1291099            if (!lockedFields.Contains(MetadataField.Tags))
 1100            {
 1271101                if (replaceData || target.Tags.Length == 0)
 1102                {
 1261103                    target.Tags = source.Tags;
 1104                }
 1105                else
 1106                {
 11107                    target.Tags = target.Tags.Concat(source.Tags).Distinct(StringComparer.OrdinalIgnoreCase).ToArray();
 1108                }
 1109            }
 1110
 1291111            if (!lockedFields.Contains(MetadataField.ProductionLocations))
 1112            {
 1271113                if (replaceData || target.ProductionLocations.Length == 0)
 1114                {
 1261115                    target.ProductionLocations = source.ProductionLocations;
 1116                }
 1117                else
 1118                {
 11119                    target.ProductionLocations = target.ProductionLocations.Concat(source.ProductionLocations).Distinct(
 1120                }
 1121            }
 1122
 2661123            foreach (var id in source.ProviderIds)
 1124            {
 41125                var key = id.Key;
 1126
 1127                // Don't replace existing Id's.
 41128                if (replaceData)
 1129                {
 11130                    target.ProviderIds[key] = id.Value;
 1131                }
 1132                else
 1133                {
 31134                    target.ProviderIds.TryAdd(key, id.Value);
 1135                }
 1136            }
 1137
 1291138            if (replaceData || !target.CriticRating.HasValue)
 1139            {
 1281140                target.CriticRating = source.CriticRating;
 1141            }
 1142
 1291143            if (replaceData || target.RemoteTrailers.Count == 0)
 1144            {
 1281145                target.RemoteTrailers = source.RemoteTrailers;
 1146            }
 1147            else
 1148            {
 11149                target.RemoteTrailers = target.RemoteTrailers.Concat(source.RemoteTrailers).DistinctBy(t => t.Url).ToArr
 1150            }
 1151
 1291152            MergeAlbumArtist(source, target, replaceData);
 1291153            MergeVideoInfo(source, target, replaceData);
 1291154            MergeDisplayOrder(source, target, replaceData);
 1155
 1291156            if (replaceData || string.IsNullOrEmpty(target.ForcedSortName))
 1157            {
 1281158                var forcedSortName = source.ForcedSortName;
 1281159                if (!string.IsNullOrEmpty(forcedSortName))
 1160                {
 31161                    target.ForcedSortName = forcedSortName;
 1162                }
 1163            }
 1164
 1291165            if (mergeMetadataSettings)
 1166            {
 21167                if (replaceData || !target.IsLocked)
 1168                {
 21169                    target.IsLocked = target.IsLocked || source.IsLocked;
 1170                }
 1171
 21172                if (target.LockedFields.Length == 0)
 1173                {
 01174                    target.LockedFields = source.LockedFields;
 1175                }
 1176                else
 1177                {
 21178                    target.LockedFields = target.LockedFields.Concat(source.LockedFields).Distinct().ToArray();
 1179                }
 1180
 21181                if (source.DateCreated != DateTime.MinValue)
 1182                {
 11183                    target.DateCreated = source.DateCreated;
 1184                }
 1185
 21186                if (replaceData || source.DateModified != DateTime.MinValue)
 1187                {
 21188                    target.DateModified = source.DateModified;
 1189                }
 1190
 21191                if (replaceData || string.IsNullOrEmpty(target.PreferredMetadataCountryCode))
 1192                {
 21193                    target.PreferredMetadataCountryCode = source.PreferredMetadataCountryCode;
 1194                }
 1195
 21196                if (replaceData || string.IsNullOrEmpty(target.PreferredMetadataLanguage))
 1197                {
 21198                    target.PreferredMetadataLanguage = source.PreferredMetadataLanguage;
 1199                }
 1200            }
 1291201        }
 1202
 1203        private static void MergePeople(IReadOnlyList<PersonInfo> source, IReadOnlyList<PersonInfo> target)
 1204        {
 41205            var sourceByName = source.ToLookup(p => p.Name.RemoveDiacritics(), StringComparer.OrdinalIgnoreCase);
 41206            var targetByName = target.ToLookup(p => p.Name.RemoveDiacritics(), StringComparer.OrdinalIgnoreCase);
 1207
 161208            foreach (var name in targetByName.Select(g => g.Key))
 1209            {
 41210                var targetPeople = targetByName[name].ToArray();
 41211                var sourcePeople = sourceByName[name].ToArray();
 1212
 41213                if (sourcePeople.Length == 0)
 1214                {
 1215                    continue;
 1216                }
 1217
 121218                for (int i = 0; i < targetPeople.Length; i++)
 1219                {
 31220                    var person = targetPeople[i];
 31221                    var personInSource = i < sourcePeople.Length ? sourcePeople[i] : sourcePeople[0];
 1222
 101223                    foreach (var providerId in personInSource.ProviderIds)
 1224                    {
 21225                        person.ProviderIds.TryAdd(providerId.Key, providerId.Value);
 1226                    }
 1227
 31228                    if (string.IsNullOrWhiteSpace(person.ImageUrl))
 1229                    {
 21230                        person.ImageUrl = personInSource.ImageUrl;
 1231                    }
 1232
 31233                    if (!string.IsNullOrWhiteSpace(personInSource.Role) && string.IsNullOrWhiteSpace(person.Role))
 1234                    {
 01235                        person.Role = personInSource.Role;
 1236                    }
 1237
 31238                    if (personInSource.SortOrder.HasValue && !person.SortOrder.HasValue)
 1239                    {
 01240                        person.SortOrder = personInSource.SortOrder;
 1241                    }
 1242                }
 1243            }
 41244        }
 1245
 1246        private static void MergeDisplayOrder(BaseItem source, BaseItem target, bool replaceData)
 1247        {
 1291248            if (source is IHasDisplayOrder sourceHasDisplayOrder
 1291249                && target is IHasDisplayOrder targetHasDisplayOrder)
 1250            {
 491251                if (replaceData || string.IsNullOrEmpty(targetHasDisplayOrder.DisplayOrder))
 1252                {
 481253                    var displayOrder = sourceHasDisplayOrder.DisplayOrder;
 481254                    if (!string.IsNullOrWhiteSpace(displayOrder))
 1255                    {
 31256                        targetHasDisplayOrder.DisplayOrder = displayOrder;
 1257                    }
 1258                }
 1259            }
 1291260        }
 1261
 1262        private static void MergeAlbumArtist(BaseItem source, BaseItem target, bool replaceData)
 1263        {
 1291264            if (source is IHasAlbumArtist sourceHasAlbumArtist
 1291265                && target is IHasAlbumArtist targetHasAlbumArtist)
 1266            {
 281267                if (replaceData || targetHasAlbumArtist.AlbumArtists.Count == 0)
 1268                {
 271269                    targetHasAlbumArtist.AlbumArtists = sourceHasAlbumArtist.AlbumArtists;
 1270                }
 11271                else if (sourceHasAlbumArtist.AlbumArtists.Count > 0)
 1272                {
 11273                    targetHasAlbumArtist.AlbumArtists = targetHasAlbumArtist.AlbumArtists.Concat(sourceHasAlbumArtist.Al
 1274                }
 1275            }
 1021276        }
 1277
 1278        private static void MergeVideoInfo(BaseItem source, BaseItem target, bool replaceData)
 1279        {
 1291280            if (source is Video sourceCast && target is Video targetCast)
 1281            {
 521282                if (replaceData || !targetCast.Video3DFormat.HasValue)
 1283                {
 511284                    targetCast.Video3DFormat = sourceCast.Video3DFormat;
 1285                }
 1286            }
 1291287        }
 1288    }
 1289}

Methods/Properties

.ctor(MediaBrowser.Controller.Configuration.IServerConfigurationManager,Microsoft.Extensions.Logging.ILogger`1<MediaBrowser.Providers.Manager.MetadataService`2<TItemType,TIdType>>,MediaBrowser.Controller.Providers.IProviderManager,MediaBrowser.Model.IO.IFileSystem,MediaBrowser.Controller.Library.ILibraryManager,MediaBrowser.Controller.IO.IExternalDataManager,MediaBrowser.Controller.Persistence.IItemRepository)
get_EnableUpdatingPremiereDateFromChildren()
get_EnableUpdatingGenresFromChildren()
get_EnableUpdatingStudiosFromChildren()
get_EnableUpdatingOfficialRatingFromChildren()
get_Order()
TryGetFileSystemMetadata(System.String,MediaBrowser.Controller.Providers.IDirectoryService)
ApplySearchResult(MediaBrowser.Controller.Providers.ItemLookupInfo,MediaBrowser.Model.Providers.RemoteSearchResult)
AfterMetadataRefresh(TItemType,MediaBrowser.Controller.Providers.MetadataRefreshOptions,System.Threading.CancellationToken)
BeforeSaveInternal(TItemType,System.Boolean,MediaBrowser.Controller.Library.ItemUpdateType)
EnableUpdateMetadataFromChildren(TItemType,System.Boolean,MediaBrowser.Controller.Library.ItemUpdateType)
GetChildrenForMetadataUpdates(TItemType)
UpdateMetadataFromChildren(TItemType,System.Collections.Generic.IReadOnlyList`1<MediaBrowser.Controller.Entities.BaseItem>,System.Boolean,MediaBrowser.Controller.Library.ItemUpdateType)
UpdateCumulativeRunTimeTicks(TItemType,System.Collections.Generic.IReadOnlyList`1<MediaBrowser.Controller.Entities.BaseItem>)
UpdateDateLastMediaAdded(TItemType,System.Collections.Generic.IReadOnlyList`1<MediaBrowser.Controller.Entities.BaseItem>)
UpdatePremiereDate(TItemType,System.Collections.Generic.IReadOnlyList`1<MediaBrowser.Controller.Entities.BaseItem>)
UpdateGenres(TItemType,System.Collections.Generic.IReadOnlyList`1<MediaBrowser.Controller.Entities.BaseItem>)
UpdateStudios(TItemType,System.Collections.Generic.IReadOnlyList`1<MediaBrowser.Controller.Entities.BaseItem>)
UpdateOfficialRating(TItemType,System.Collections.Generic.IReadOnlyList`1<MediaBrowser.Controller.Entities.BaseItem>)
GetProviders(MediaBrowser.Controller.Entities.BaseItem,MediaBrowser.Model.Configuration.LibraryOptions,MediaBrowser.Controller.Providers.MetadataRefreshOptions,System.Boolean,System.Boolean)
GetNonLocalImageProviders(MediaBrowser.Controller.Entities.BaseItem,System.Collections.Generic.IEnumerable`1<MediaBrowser.Controller.Providers.IImageProvider>,MediaBrowser.Controller.Providers.ImageRefreshOptions)
CanRefresh(MediaBrowser.Controller.Entities.BaseItem)
CanRefreshPrimary(System.Type)
CreateNew()
MergeNewData(TItemType,TIdType)
HasChanged(MediaBrowser.Controller.Entities.BaseItem,MediaBrowser.Controller.Providers.IHasItemChangeMonitor,MediaBrowser.Controller.Providers.IDirectoryService)
MergeData(MediaBrowser.Controller.Providers.MetadataResult`1<TItemType>,MediaBrowser.Controller.Providers.MetadataResult`1<TItemType>,MediaBrowser.Model.Entities.MetadataField[],System.Boolean,System.Boolean)
MergeBaseItemData(MediaBrowser.Controller.Providers.MetadataResult`1<TItemType>,MediaBrowser.Controller.Providers.MetadataResult`1<TItemType>,MediaBrowser.Model.Entities.MetadataField[],System.Boolean,System.Boolean)
MergePeople(System.Collections.Generic.IReadOnlyList`1<MediaBrowser.Controller.Entities.PersonInfo>,System.Collections.Generic.IReadOnlyList`1<MediaBrowser.Controller.Entities.PersonInfo>)
MergeDisplayOrder(MediaBrowser.Controller.Entities.BaseItem,MediaBrowser.Controller.Entities.BaseItem,System.Boolean)
MergeAlbumArtist(MediaBrowser.Controller.Entities.BaseItem,MediaBrowser.Controller.Entities.BaseItem,System.Boolean)
MergeVideoInfo(MediaBrowser.Controller.Entities.BaseItem,MediaBrowser.Controller.Entities.BaseItem,System.Boolean)