| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715 | #nullable disableusing System;using System.Collections.Generic;using System.IO;using System.Linq;using System.Net;using System.Net.Http;using System.Net.Mime;using System.Threading;using System.Threading.Tasks;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Entities.Audio;using MediaBrowser.Controller.Entities.TV;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.LiveTv;using MediaBrowser.Controller.Providers;using MediaBrowser.Model.Configuration;using MediaBrowser.Model.Drawing;using MediaBrowser.Model.Entities;using MediaBrowser.Model.IO;using MediaBrowser.Model.MediaInfo;using MediaBrowser.Model.Net;using MediaBrowser.Model.Providers;using Microsoft.Extensions.Logging;namespace MediaBrowser.Providers.Manager{    /// <summary>    /// Utilities for managing images attached to items.    /// </summary>    public class ItemImageProvider    {        private readonly ILogger _logger;        private readonly IProviderManager _providerManager;        private readonly IFileSystem _fileSystem;        private static readonly ImageType[] AllImageTypes = Enum.GetValues<ImageType>();        /// <summary>        /// Image types that are only one per item.        /// </summary>        private static readonly ImageType[] _singularImages =        {            ImageType.Primary,            ImageType.Art,            ImageType.Banner,            ImageType.Box,            ImageType.BoxRear,            ImageType.Disc,            ImageType.Logo,            ImageType.Menu,            ImageType.Thumb        };        /// <summary>        /// Initializes a new instance of the <see cref="ItemImageProvider"/> class.        /// </summary>        /// <param name="logger">The logger.</param>        /// <param name="providerManager">The provider manager for interacting with provider image references.</param>        /// <param name="fileSystem">The filesystem.</param>        public ItemImageProvider(ILogger logger, IProviderManager providerManager, IFileSystem fileSystem)        {            _logger = logger;            _providerManager = providerManager;            _fileSystem = fileSystem;        }        /// <summary>        /// Removes all existing images from the provided item.        /// </summary>        /// <param name="item">The <see cref="BaseItem"/> to remove images from.</param>        /// <param name="canDeleteLocal">Whether removing images outside metadata folder is allowed.</param>        /// <returns><c>true</c> if changes were made to the item; otherwise <c>false</c>.</returns>        public bool RemoveImages(BaseItem item, bool canDeleteLocal = false)        {            var singular = new List<ItemImageInfo>();            var itemMetadataPath = item.GetInternalMetadataPath();            for (var i = 0; i < _singularImages.Length; i++)            {                var currentImage = item.GetImageInfo(_singularImages[i], 0);                if (currentImage is not null)                {                    var imageInMetadataFolder = currentImage.Path.StartsWith(itemMetadataPath, StringComparison.OrdinalIgnoreCase);                    if (imageInMetadataFolder || canDeleteLocal || item.IsSaveLocalMetadataEnabled())                    {                        singular.Add(currentImage);                    }                }            }            singular.AddRange(item.GetImages(ImageType.Backdrop));            PruneImages(item, singular);            return singular.Count > 0;        }        /// <summary>        /// Verifies existing images have valid paths and adds any new local images provided.        /// </summary>        /// <param name="item">The <see cref="BaseItem"/> to validate images for.</param>        /// <param name="providers">The providers to use, must include <see cref="ILocalImageProvider"/>(s) for local scanning.</param>        /// <param name="refreshOptions">The refresh options.</param>        /// <returns><c>true</c> if changes were made to the item; otherwise <c>false</c>.</returns>        public bool ValidateImages(BaseItem item, IEnumerable<IImageProvider> providers, ImageRefreshOptions refreshOptions)        {            var hasChanges = false;            var directoryService = refreshOptions?.DirectoryService;            if (item is not Photo)            {                var images = providers.OfType<ILocalImageProvider>()                    .SelectMany(i => i.GetImages(item, directoryService))                    .ToList();                if (MergeImages(item, images, refreshOptions))                {                    hasChanges = true;                }            }            return hasChanges;        }        /// <summary>        /// Refreshes from the providers according to the given options.        /// </summary>        /// <param name="item">The <see cref="BaseItem"/> to gather images for.</param>        /// <param name="libraryOptions">The library options.</param>        /// <param name="providers">The providers to query for images.</param>        /// <param name="refreshOptions">The refresh options.</param>        /// <param name="cancellationToken">The cancellation token.</param>        /// <returns>The refresh result.</returns>        public async Task<RefreshResult> RefreshImages(            BaseItem item,            LibraryOptions libraryOptions,            IEnumerable<IImageProvider> providers,            ImageRefreshOptions refreshOptions,            CancellationToken cancellationToken)        {            var oldBackdropImages = Array.Empty<ItemImageInfo>();            if (refreshOptions.IsReplacingImage(ImageType.Backdrop))            {                oldBackdropImages = item.GetImages(ImageType.Backdrop).ToArray();            }            var result = new RefreshResult { UpdateType = ItemUpdateType.None };            var typeName = item.GetType().Name;            var typeOptions = libraryOptions.GetTypeOptions(typeName) ?? new TypeOptions { Type = typeName };            // track library limits, adding buffer to allow lazy replacing of current images            var backdropLimit = typeOptions.GetLimit(ImageType.Backdrop) + oldBackdropImages.Length;            var downloadedImages = new List<ImageType>();            foreach (var provider in providers)            {                if (provider is IRemoteImageProvider remoteProvider)                {                    await RefreshFromProvider(item, remoteProvider, refreshOptions, typeOptions, backdropLimit, downloadedImages, result, cancellationToken).ConfigureAwait(false);                    continue;                }                if (provider is IDynamicImageProvider dynamicImageProvider)                {                    await RefreshFromProvider(item, dynamicImageProvider, refreshOptions, typeOptions, downloadedImages, result, cancellationToken).ConfigureAwait(false);                }            }            // Only delete existing multi-images if new ones were added            if (oldBackdropImages.Length > 0 && oldBackdropImages.Length < item.GetImages(ImageType.Backdrop).Count())            {                PruneImages(item, oldBackdropImages);            }            return result;        }        /// <summary>        /// Refreshes from a dynamic provider.        /// </summary>        private async Task RefreshFromProvider(            BaseItem item,            IDynamicImageProvider provider,            ImageRefreshOptions refreshOptions,            TypeOptions savedOptions,            List<ImageType> downloadedImages,            RefreshResult result,            CancellationToken cancellationToken)        {            try            {                var images = provider.GetSupportedImages(item);                foreach (var imageType in images)                {                    if (!savedOptions.IsEnabled(imageType))                    {                        continue;                    }                    if (!item.HasImage(imageType) || (refreshOptions.IsReplacingImage(imageType) && !downloadedImages.Contains(imageType)))                    {                        _logger.LogDebug("Running {Provider} for {Item}", provider.GetType().Name, item.Path ?? item.Name);                        var response = await provider.GetImage(item, imageType, cancellationToken).ConfigureAwait(false);                        if (response.HasImage)                        {                            if (string.IsNullOrEmpty(response.Path))                            {                                var mimeType = response.Format.GetMimeType();                                await _providerManager.SaveImage(item, response.Stream, mimeType, imageType, null, cancellationToken).ConfigureAwait(false);                            }                            else                            {                                if (response.Protocol == MediaProtocol.Http)                                {                                    _logger.LogDebug("Setting image url into item {Item}", item.Id);                                    var index = item.AllowsMultipleImages(imageType) ? item.GetImages(imageType).Count() : 0;                                    item.SetImage(                                        new ItemImageInfo                                        {                                            Path = response.Path,                                            Type = imageType                                        },                                        index);                                }                                else                                {                                    var mimeType = MimeTypes.GetMimeType(response.Path);                                    await _providerManager.SaveImage(item, response.Path, mimeType, imageType, null, null, cancellationToken).ConfigureAwait(false);                                }                            }                            downloadedImages.Add(imageType);                            result.UpdateType |= ItemUpdateType.ImageUpdate;                        }                    }                }            }            catch (OperationCanceledException)            {                throw;            }            catch (Exception ex)            {                result.ErrorMessage = ex.Message;                _logger.LogError(ex, "Error in {Provider}", provider.Name);            }        }        /// <summary>        /// Refreshes from a remote provider.        /// </summary>        /// <param name="item">The item.</param>        /// <param name="provider">The provider.</param>        /// <param name="refreshOptions">The refresh options.</param>        /// <param name="savedOptions">The saved options.</param>        /// <param name="backdropLimit">The backdrop limit.</param>        /// <param name="downloadedImages">The downloaded images.</param>        /// <param name="result">The result.</param>        /// <param name="cancellationToken">The cancellation token.</param>        /// <returns>Task.</returns>        private async Task RefreshFromProvider(            BaseItem item,            IRemoteImageProvider provider,            ImageRefreshOptions refreshOptions,            TypeOptions savedOptions,            int backdropLimit,            List<ImageType> downloadedImages,            RefreshResult result,            CancellationToken cancellationToken)        {            try            {                if (!item.SupportsRemoteImageDownloading)                {                    return;                }                if (!refreshOptions.ReplaceAllImages &&                    refreshOptions.ReplaceImages.Count == 0 &&                    ContainsImages(item, provider.GetSupportedImages(item).ToList(), savedOptions, backdropLimit))                {                    return;                }                _logger.LogDebug("Running {Provider} for {Item}", provider.GetType().Name, item.Path ?? item.Name);                var images = await _providerManager.GetAvailableRemoteImages(                    item,                    new RemoteImageQuery(provider.Name)                    {                        IncludeAllLanguages = true,                        IncludeDisabledProviders = false,                    },                    cancellationToken).ConfigureAwait(false);                var list = images.ToList();                int minWidth;                foreach (var imageType in _singularImages)                {                    if (!savedOptions.IsEnabled(imageType))                    {                        continue;                    }                    if (!item.HasImage(imageType) || (refreshOptions.IsReplacingImage(imageType) && !downloadedImages.Contains(imageType)))                    {                        minWidth = savedOptions.GetMinWidth(imageType);                        var downloaded = await DownloadImage(item, provider, result, list, minWidth, imageType, cancellationToken).ConfigureAwait(false);                        if (downloaded)                        {                            downloadedImages.Add(imageType);                        }                    }                }                minWidth = savedOptions.GetMinWidth(ImageType.Backdrop);                var listWithNoLangFirst = list.OrderByDescending(i => string.IsNullOrEmpty(i.Language));                await DownloadMultiImages(item, ImageType.Backdrop, refreshOptions, backdropLimit, provider, result, listWithNoLangFirst, minWidth, cancellationToken).ConfigureAwait(false);            }            catch (OperationCanceledException)            {                throw;            }            catch (Exception ex)            {                result.ErrorMessage = ex.Message;                _logger.LogError(ex, "Error in {Provider}", provider.Name);            }        }        /// <summary>        /// Determines if an item already contains the given images.        /// </summary>        /// <param name="item">The item.</param>        /// <param name="images">The images.</param>        /// <param name="savedOptions">The saved options.</param>        /// <param name="backdropLimit">The backdrop limit.</param>        /// <returns><c>true</c> if the specified item contains images; otherwise, <c>false</c>.</returns>        private bool ContainsImages(BaseItem item, List<ImageType> images, TypeOptions savedOptions, int backdropLimit)        {            // Using .Any causes the creation of a DisplayClass aka. variable capture            for (var i = 0; i < _singularImages.Length; i++)            {                var type = _singularImages[i];                if (images.Contains(type) && !item.HasImage(type) && savedOptions.GetLimit(type) > 0)                {                    return false;                }            }            if (images.Contains(ImageType.Backdrop) && item.GetImages(ImageType.Backdrop).Count() < backdropLimit)            {                return false;            }            return true;        }        private void PruneImages(BaseItem item, IReadOnlyList<ItemImageInfo> images)        {            foreach (var image in images)            {                if (image.IsLocalFile)                {                    try                    {                        _fileSystem.DeleteFile(image.Path);                    }                    catch (FileNotFoundException)                    {                        // Nothing to do, already gone                    }                    catch (DirectoryNotFoundException)                    {                        // Nothing to do, already gone                    }                    catch (UnauthorizedAccessException ex)                    {                        _logger.LogWarning(ex, "Unable to delete {Image}", image.Path);                    }                }            }            item.RemoveImages(images);            // Cleanup old metadata directory for episodes if empty, as long as it's not a virtual item            if (item is Episode && !item.IsVirtualItem)            {                var oldLocalMetadataDirectory = Path.Combine(item.ContainingFolderPath, "metadata");                if (_fileSystem.DirectoryExists(oldLocalMetadataDirectory) && !_fileSystem.GetFiles(oldLocalMetadataDirectory).Any())                {                    Directory.Delete(oldLocalMetadataDirectory);                }            }        }        /// <summary>        /// Merges a list of images into the provided item, validating existing images and replacing them or adding new images as necessary.        /// </summary>        /// <param name="refreshOptions">The refresh options.</param>        /// <param name="dontReplaceImages">List of imageTypes to remove from ReplaceImages.</param>        public void UpdateReplaceImages(ImageRefreshOptions refreshOptions, ICollection<ImageType> dontReplaceImages)        {            if (refreshOptions is not null)            {                if (refreshOptions.ReplaceAllImages)                {                    refreshOptions.ReplaceAllImages = false;                    refreshOptions.ReplaceImages = AllImageTypes.ToList();                }                refreshOptions.ReplaceImages = refreshOptions.ReplaceImages.Except(dontReplaceImages).ToList();            }        }        /// <summary>        /// Merges a list of images into the provided item, validating existing images and replacing them or adding new images as necessary.        /// </summary>        /// <param name="item">The <see cref="BaseItem"/> to modify.</param>        /// <param name="images">The new images to place in <c>item</c>.</param>        /// <param name="refreshOptions">The refresh options.</param>        /// <returns><c>true</c> if changes were made to the item; otherwise <c>false</c>.</returns>        public bool MergeImages(BaseItem item, IReadOnlyList<LocalImageInfo> images, ImageRefreshOptions refreshOptions)        {            var changed = item.ValidateImages();            var foundImageTypes = new List<ImageType>();            for (var i = 0; i < _singularImages.Length; i++)            {                var type = _singularImages[i];                var image = GetFirstLocalImageInfoByType(images, type);                if (image is not null)                {                    var currentImage = item.GetImageInfo(type, 0);                    // if image file is stored with media, don't replace that later                    if (item.ContainingFolderPath is not null && item.ContainingFolderPath.Contains(Path.GetDirectoryName(image.FileInfo.FullName), StringComparison.OrdinalIgnoreCase))                    {                        foundImageTypes.Add(type);                    }                    if (currentImage is null || !string.Equals(currentImage.Path, image.FileInfo.FullName, StringComparison.OrdinalIgnoreCase))                    {                        item.SetImagePath(type, image.FileInfo);                        changed = true;                    }                    else                    {                        var newDateModified = _fileSystem.GetLastWriteTimeUtc(image.FileInfo);                        // If date changed then we need to reset saved image dimensions                        if (currentImage.DateModified != newDateModified && (currentImage.Width > 0 || currentImage.Height > 0))                        {                            currentImage.Width = 0;                            currentImage.Height = 0;                            changed = true;                        }                        currentImage.DateModified = newDateModified;                    }                }            }            if (UpdateMultiImages(item, images, ImageType.Backdrop))            {                changed = true;                foundImageTypes.Add(ImageType.Backdrop);            }            if (foundImageTypes.Count > 0)            {                UpdateReplaceImages(refreshOptions, foundImageTypes);            }            return changed;        }        private static LocalImageInfo GetFirstLocalImageInfoByType(IReadOnlyList<LocalImageInfo> images, ImageType type)        {            var len = images.Count;            for (var i = 0; i < len; i++)            {                var image = images[i];                if (image.Type == type)                {                    return image;                }            }            return null;        }        private bool UpdateMultiImages(BaseItem item, IReadOnlyList<LocalImageInfo> images, ImageType type)        {            var changed = false;            var newImageFileInfos = images                .Where(i => i.Type == type)                .Select(i => i.FileInfo)                .ToList();            if (item.AddImages(type, newImageFileInfos))            {                changed = true;            }            return changed;        }        private async Task<bool> DownloadImage(            BaseItem item,            IRemoteImageProvider provider,            RefreshResult result,            IEnumerable<RemoteImageInfo> images,            int minWidth,            ImageType type,            CancellationToken cancellationToken)        {            var eligibleImages = images                .Where(i => i.Type == type && (i.Width is null || i.Width >= minWidth))                .ToList();            if (EnableImageStub(item) && eligibleImages.Count > 0)            {                SaveImageStub(item, type, eligibleImages.Select(i => i.Url));                result.UpdateType |= ItemUpdateType.ImageUpdate;                return true;            }            foreach (var image in eligibleImages)            {                var url = image.Url;                try                {                    using var response = await provider.GetImageResponse(url, cancellationToken).ConfigureAwait(false);                    // Sometimes providers send back bad urls. Just move to the next image                    if (response.StatusCode == HttpStatusCode.NotFound || response.StatusCode == HttpStatusCode.Forbidden)                    {                        _logger.LogDebug("{Url} returned {StatusCode}, ignoring", url, response.StatusCode);                        continue;                    }                    if (!response.IsSuccessStatusCode)                    {                        _logger.LogWarning("{Url} returned {StatusCode}, skipping all remaining requests", url, response.StatusCode);                        break;                    }                    var stream = await response.Content.ReadAsStreamAsync(cancellationToken).ConfigureAwait(false);                    await using (stream.ConfigureAwait(false))                    {                        var mimetype = response.Content.Headers.ContentType?.MediaType;                        if (mimetype is null || mimetype.Equals(MediaTypeNames.Application.Octet, StringComparison.OrdinalIgnoreCase))                        {                            mimetype = MimeTypes.GetMimeType(response.RequestMessage.RequestUri.GetLeftPart(UriPartial.Path));                        }                        await _providerManager.SaveImage(                            item,                            stream,                            mimetype,                            type,                            null,                            cancellationToken).ConfigureAwait(false);                    }                    result.UpdateType |= ItemUpdateType.ImageUpdate;                    return true;                }                catch (HttpRequestException)                {                    break;                }            }            return false;        }        private bool EnableImageStub(BaseItem item)        {            if (item is LiveTvProgram)            {                return true;            }            if (!item.IsFileProtocol)            {                return true;            }            if (item is IItemByName and not MusicArtist)            {                var hasDualAccess = item as IHasDualAccess;                if (hasDualAccess is null || hasDualAccess.IsAccessedByName)                {                    return true;                }            }            // We always want to use prefetched images            return false;        }        private void SaveImageStub(BaseItem item, ImageType imageType, IEnumerable<string> urls)        {            var newIndex = item.AllowsMultipleImages(imageType) ? item.GetImages(imageType).Count() : 0;            SaveImageStub(item, imageType, urls, newIndex);        }        private void SaveImageStub(BaseItem item, ImageType imageType, IEnumerable<string> urls, int newIndex)        {            var path = string.Join('|', urls.Take(1));            item.SetImage(                new ItemImageInfo                {                    Path = path,                    Type = imageType                },                newIndex);        }        private async Task DownloadMultiImages(BaseItem item, ImageType imageType, ImageRefreshOptions refreshOptions, int limit, IRemoteImageProvider provider, RefreshResult result, IEnumerable<RemoteImageInfo> images, int minWidth, CancellationToken cancellationToken)        {            foreach (var image in images.Where(i => i.Type == imageType))            {                if (item.GetImages(imageType).Count() >= limit)                {                    break;                }                if (image.Width.HasValue && image.Width.Value < minWidth)                {                    continue;                }                var url = image.Url;                if (EnableImageStub(item))                {                    SaveImageStub(item, imageType, new[] { url });                    result.UpdateType |= ItemUpdateType.ImageUpdate;                    continue;                }                try                {                    using var response = await provider.GetImageResponse(url, cancellationToken).ConfigureAwait(false);                    // Sometimes providers send back bad urls. Just move to the next image                    if (response.StatusCode == HttpStatusCode.NotFound || response.StatusCode == HttpStatusCode.Forbidden)                    {                        _logger.LogDebug("{Url} returned {StatusCode}, ignoring", url, response.StatusCode);                        continue;                    }                    if (!response.IsSuccessStatusCode)                    {                        _logger.LogWarning("{Url} returned {StatusCode}, skipping all remaining requests", url, response.StatusCode);                        break;                    }                    // If there's already an image of the same file size, skip it unless doing a full refresh                    if (response.Content.Headers.ContentLength.HasValue && !refreshOptions.IsReplacingImage(imageType))                    {                        try                        {                            if (item.GetImages(imageType).Any(i => _fileSystem.GetFileInfo(i.Path).Length == response.Content.Headers.ContentLength.Value))                            {                                response.Content.Dispose();                                continue;                            }                        }                        catch (IOException ex)                        {                            _logger.LogError(ex, "Error examining images");                        }                    }                    var stream = await response.Content.ReadAsStreamAsync(cancellationToken).ConfigureAwait(false);                    await using (stream.ConfigureAwait(false))                    {                        var mimetype = response.Content.Headers.ContentType?.MediaType;                        if (mimetype is null || mimetype.Equals(MediaTypeNames.Application.Octet, StringComparison.OrdinalIgnoreCase))                        {                            mimetype = MimeTypes.GetMimeType(response.RequestMessage.RequestUri.GetLeftPart(UriPartial.Path));                        }                        await _providerManager.SaveImage(                            item,                            stream,                            mimetype,                            imageType,                            null,                            cancellationToken).ConfigureAwait(false);                    }                    result.UpdateType |= ItemUpdateType.ImageUpdate;                }                catch (HttpRequestException)                {                    break;                }            }        }    }}
 |