| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487 | using MediaBrowser.Common.Extensions;using MediaBrowser.Common.IO;using MediaBrowser.Controller.Configuration;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.Providers;using MediaBrowser.Model.Configuration;using MediaBrowser.Model.Entities;using MediaBrowser.Model.Logging;using MediaBrowser.Model.Net;using MediaBrowser.Model.Providers;using System;using System.Collections.Generic;using System.IO;using System.Linq;using System.Net;using System.Threading;using System.Threading.Tasks;namespace MediaBrowser.Providers.Manager{    public class ItemImageProvider    {        private readonly ILogger _logger;        private readonly IProviderManager _providerManager;        private readonly IServerConfigurationManager _config;        private readonly IFileSystem _fileSystem;        public ItemImageProvider(ILogger logger, IProviderManager providerManager, IServerConfigurationManager config, IFileSystem fileSystem)        {            _logger = logger;            _providerManager = providerManager;            _config = config;            _fileSystem = fileSystem;        }        public bool ValidateImages(IHasImages item, IEnumerable<IImageProvider> providers, IDirectoryService directoryService)        {            var hasChanges = item.ValidateImages(directoryService);            foreach (var provider in providers.OfType<ILocalImageFileProvider>())            {                var images = provider.GetImages(item, directoryService);                if (MergeImages(item, images))                {                    hasChanges = true;                }            }            return hasChanges;        }        public async Task<RefreshResult> RefreshImages(IHasImages item, IEnumerable<IImageProvider> imageProviders, ImageRefreshOptions refreshOptions, MetadataOptions savedOptions, CancellationToken cancellationToken)        {            if (refreshOptions.IsReplacingImage(ImageType.Backdrop))            {                ClearImages(item, ImageType.Backdrop);            }            if (refreshOptions.IsReplacingImage(ImageType.Screenshot))            {                ClearImages(item, ImageType.Screenshot);            }            var result = new RefreshResult { UpdateType = ItemUpdateType.None };            var providers = imageProviders.ToList();            var providerIds = new List<Guid>();            // In order to avoid duplicates, only download these if there are none already            var backdropLimit = savedOptions.GetLimit(ImageType.Backdrop);            var screenshotLimit = savedOptions.GetLimit(ImageType.Screenshot);            var downloadedImages = new List<ImageType>();            foreach (var provider in providers)            {                var remoteProvider = provider as IRemoteImageProvider;                if (remoteProvider != null)                {                    await RefreshFromProvider(item, remoteProvider, refreshOptions, savedOptions, backdropLimit, screenshotLimit, downloadedImages, result, cancellationToken).ConfigureAwait(false);                    providerIds.Add(provider.GetType().FullName.GetMD5());                    continue;                }                var dynamicImageProvider = provider as IDynamicImageProvider;                if (dynamicImageProvider != null)                {                    await RefreshFromProvider(item, dynamicImageProvider, refreshOptions, savedOptions, downloadedImages, result, cancellationToken).ConfigureAwait(false);                    providerIds.Add(provider.GetType().FullName.GetMD5());                }            }            result.Providers = providerIds;            return result;        }        /// <summary>        /// Refreshes from 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="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(IHasImages item,            IDynamicImageProvider provider,            ImageRefreshOptions refreshOptions,            MetadataOptions savedOptions,            ICollection<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.Debug("Running {0} for {1}", 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 = "image/" + Path.GetExtension(response.Path).TrimStart('.').ToLower();                                var stream = _fileSystem.GetFileStream(response.Path, FileMode.Open, FileAccess.Read, FileShare.Read, true);                                await _providerManager.SaveImage(item, stream, mimeType, imageType, null, cancellationToken).ConfigureAwait(false);                            }                            else                            {                                var mimeType = "image/" + response.Format.ToString().ToLower();                                await _providerManager.SaveImage(item, response.Stream, mimeType, imageType, null, cancellationToken).ConfigureAwait(false);                            }                            downloadedImages.Add(imageType);                            result.UpdateType = result.UpdateType | ItemUpdateType.ImageUpdate;                        }                    }                }            }            catch (OperationCanceledException)            {                throw;            }            catch (Exception ex)            {                result.ErrorMessage = ex.Message;                result.Status = ProviderRefreshStatus.CompletedWithErrors;                _logger.ErrorException("Error in {0}", ex, provider.Name);            }        }        /// <summary>        /// Image types that are only one per item        /// </summary>        private readonly ImageType[] _singularImages =        {            ImageType.Primary,            ImageType.Art,            ImageType.Banner,            ImageType.Box,            ImageType.BoxRear,            ImageType.Disc,            ImageType.Logo,            ImageType.Menu,            ImageType.Thumb        };        /// <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>        /// <param name="screenshotLimit">The screenshot limit.</param>        /// <returns><c>true</c> if the specified item contains images; otherwise, <c>false</c>.</returns>        private bool ContainsImages(IHasImages item, List<ImageType> images, MetadataOptions savedOptions, int backdropLimit, int screenshotLimit)        {            if (_singularImages.Any(i => images.Contains(i) && !item.HasImage(i) && savedOptions.GetLimit(i) > 0))            {                return false;            }            if (images.Contains(ImageType.Backdrop) && item.GetImages(ImageType.Backdrop).Count() < backdropLimit)            {                return false;            }            if (images.Contains(ImageType.Screenshot) && item.GetImages(ImageType.Screenshot).Count() < screenshotLimit)            {                return false;            }            return true;        }        /// <summary>        /// Refreshes from 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="screenshotLimit">The screenshot 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(IHasImages item,             IRemoteImageProvider provider,             ImageRefreshOptions refreshOptions,             MetadataOptions savedOptions,             int backdropLimit,             int screenshotLimit,             ICollection<ImageType> downloadedImages,            RefreshResult result,             CancellationToken cancellationToken)        {            try            {                if (!refreshOptions.ReplaceAllImages &&                     refreshOptions.ReplaceImages.Count == 0 &&                     ContainsImages(item, provider.GetSupportedImages(item).ToList(), savedOptions, backdropLimit, screenshotLimit))                {                    return;                }                _logger.Debug("Running {0} for {1}", provider.GetType().Name, item.Path ?? item.Name);                var images = await _providerManager.GetAvailableRemoteImages(item, new RemoteImageQuery                {                    ProviderName = provider.Name,                    IncludeAllLanguages = false,                    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);                await DownloadBackdrops(item, ImageType.Backdrop, backdropLimit, provider, result, list, minWidth, cancellationToken).ConfigureAwait(false);                var hasScreenshots = item as IHasScreenshots;                if (hasScreenshots != null)                {                    minWidth = savedOptions.GetMinWidth(ImageType.Screenshot);                    await DownloadBackdrops(item, ImageType.Screenshot, screenshotLimit, provider, result, list, minWidth, cancellationToken).ConfigureAwait(false);                }            }            catch (OperationCanceledException)            {                throw;            }            catch (Exception ex)            {                result.ErrorMessage = ex.Message;                result.Status = ProviderRefreshStatus.CompletedWithErrors;                _logger.ErrorException("Error in {0}", ex, provider.Name);            }        }        private void ClearImages(IHasImages item, ImageType type)        {            var deleted = false;            foreach (var image in item.GetImages(type).ToList())            {                // Delete the source file                var currentFile = new FileInfo(image.Path);                // Deletion will fail if the file is hidden so remove the attribute first                if (currentFile.Exists)                {                    if ((currentFile.Attributes & FileAttributes.Hidden) == FileAttributes.Hidden)                    {                        currentFile.Attributes &= ~FileAttributes.Hidden;                    }                    currentFile.Delete();                    deleted = true;                }            }            if (deleted)            {                item.ValidateImages(new DirectoryService(_logger));            }        }        public bool MergeImages(IHasImages item, List<LocalImageInfo> images)        {            var changed = false;            foreach (var type in _singularImages)            {                var image = images.FirstOrDefault(i => i.Type == type);                if (image != null)                {                    var currentImage = item.GetImageInfo(type, 0);                    if (currentImage == null)                    {                        item.SetImagePath(type, image.FileInfo);                        changed = true;                    }                    else if (!string.Equals(currentImage.Path, image.FileInfo.FullName,                            StringComparison.OrdinalIgnoreCase))                    {                        item.SetImagePath(type, image.FileInfo);                        changed = true;                    }                    else                    {                        currentImage.DateModified = _fileSystem.GetLastWriteTimeUtc(image.FileInfo);                    }                }            }            if (UpdateMultiImages(item, images, ImageType.Backdrop))            {                changed = true;            }            var hasScreenshots = item as IHasScreenshots;            if (hasScreenshots != null)            {                if (UpdateMultiImages(item, images, ImageType.Screenshot))                {                    changed = true;                }            }            return changed;        }        private bool UpdateMultiImages(IHasImages item, List<LocalImageInfo> images, ImageType type)        {            var changed = false;            var backdrops = images.Where(i => i.Type == type).ToList();            if (backdrops.Count > 0)            {                var foundImages = images.Where(i => i.Type == type)                    .Select(i => i.FileInfo)                    .ToList();                if (foundImages.Count > 0)                {                    if (item.AddImages(type, foundImages))                    {                        changed = true;                    }                }            }            return changed;        }        private async Task<bool> DownloadImage(IHasImages item,             IRemoteImageProvider provider,             RefreshResult result,             IEnumerable<RemoteImageInfo> images,             int minWidth,             ImageType type,             CancellationToken cancellationToken)        {            foreach (var image in images.Where(i => i.Type == type))            {                if (image.Width.HasValue && image.Width.Value < minWidth)                {                    continue;                }                var url = image.Url;                try                {                    var response = await provider.GetImageResponse(url, cancellationToken).ConfigureAwait(false);                    await _providerManager.SaveImage(item, response.Content, response.ContentType, type, null, cancellationToken).ConfigureAwait(false);                    result.UpdateType = result.UpdateType | ItemUpdateType.ImageUpdate;                    return true;                }                catch (HttpException ex)                {                    // Sometimes providers send back bad url's. Just move to the next image                    if (ex.StatusCode.HasValue && ex.StatusCode.Value == HttpStatusCode.NotFound)                    {                        continue;                    }                    break;                }            }            return false;        }        private async Task DownloadBackdrops(IHasImages item, ImageType imageType, 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;                try                {                    var response = await provider.GetImageResponse(url, cancellationToken).ConfigureAwait(false);                    // If there's already an image of the same size, skip it                    if (response.ContentLength.HasValue)                    {                        try                        {                            if (item.GetImages(imageType).Any(i => new FileInfo(i.Path).Length == response.ContentLength.Value))                            {                                response.Content.Dispose();                                continue;                            }                        }                        catch (IOException ex)                        {                            _logger.ErrorException("Error examining images", ex);                        }                    }                    await _providerManager.SaveImage(item, response.Content, response.ContentType, imageType, null, cancellationToken).ConfigureAwait(false);                    result.UpdateType = result.UpdateType | ItemUpdateType.ImageUpdate;                }                catch (HttpException ex)                {                    // Sometimes providers send back bad url's. Just move onto the next image                    if (ex.StatusCode.HasValue && ex.StatusCode.Value == HttpStatusCode.NotFound)                    {                        continue;                    }                    break;                }            }        }    }}
 |