using MediaBrowser.Common.ScheduledTasks; using MediaBrowser.Controller; using MediaBrowser.Controller.Entities; using MediaBrowser.Controller.Entities.Movies; using MediaBrowser.Controller.Library; using MediaBrowser.Model.Logging; using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Threading; using System.Threading.Tasks; namespace MediaBrowser.Server.Implementations.ScheduledTasks.Tasks { /// /// Class ImageCleanupTask /// public class ImageCleanupTask : IScheduledTask { /// /// The _kernel /// private readonly Kernel _kernel; /// /// The _logger /// private readonly ILogger _logger; private readonly ILibraryManager _libraryManager; /// /// Initializes a new instance of the class. /// /// The kernel. /// The logger. public ImageCleanupTask(Kernel kernel, ILogger logger, ILibraryManager libraryManager) { _kernel = kernel; _logger = logger; _libraryManager = libraryManager; } /// /// Creates the triggers that define when the task will run /// /// IEnumerable{BaseTaskTrigger}. public IEnumerable GetDefaultTriggers() { return new ITaskTrigger[] { new DailyTrigger { TimeOfDay = TimeSpan.FromHours(2) } }; } /// /// Returns the task to be executed /// /// The cancellation token. /// The progress. /// Task. public async Task Execute(CancellationToken cancellationToken, IProgress progress) { await EnsureChapterImages(cancellationToken).ConfigureAwait(false); // First gather all image files var files = GetFiles(_kernel.FFMpegManager.AudioImagesDataPath) .Concat(GetFiles(_kernel.FFMpegManager.VideoImagesDataPath)) .Concat(GetFiles(_kernel.ProviderManager.ImagesDataPath)) .ToList(); // Now gather all items var items = _libraryManager.RootFolder.RecursiveChildren.ToList(); items.Add(_libraryManager.RootFolder); // Determine all possible image paths var pathsInUse = items.SelectMany(GetPathsInUse) .Distinct(StringComparer.OrdinalIgnoreCase) .ToDictionary(p => p, StringComparer.OrdinalIgnoreCase); var numComplete = 0; var tasks = files.Select(file => Task.Run(() => { cancellationToken.ThrowIfCancellationRequested(); if (!pathsInUse.ContainsKey(file)) { cancellationToken.ThrowIfCancellationRequested(); try { File.Delete(file); } catch (IOException ex) { _logger.ErrorException("Error deleting {0}", ex, file); } } // Update progress lock (progress) { numComplete++; double percent = numComplete; percent /= files.Count; progress.Report(100 * percent); } })); await Task.WhenAll(tasks).ConfigureAwait(false); } /// /// Ensures the chapter images. /// /// The cancellation token. /// Task. private Task EnsureChapterImages(CancellationToken cancellationToken) { var videos = _libraryManager.RootFolder.RecursiveChildren.OfType