< Summary - Jellyfin

Information
Class: Emby.Server.Implementations.ScheduledTasks.Tasks.ChapterImagesTask
Assembly: Emby.Server.Implementations
File(s): /srv/git/jellyfin/Emby.Server.Implementations/ScheduledTasks/Tasks/ChapterImagesTask.cs
Line coverage
86%
Covered lines: 19
Uncovered lines: 3
Coverable lines: 22
Total lines: 182
Line coverage: 86.3%
Branch coverage
N/A
Covered branches: 0
Total branches: 0
Branch coverage: N/A
Method coverage

Feature is only available for sponsors

Upgrade to PRO version

Coverage history

Coverage history 0 25 50 75 100

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity Line coverage
.ctor(...)100%11100%
get_Name()100%11100%
get_Description()100%210%
get_Category()100%210%
get_Key()100%210%
GetDefaultTriggers()100%11100%

File(s)

/srv/git/jellyfin/Emby.Server.Implementations/ScheduledTasks/Tasks/ChapterImagesTask.cs

#LineLine coverage
 1using System;
 2using System.Collections.Generic;
 3using System.IO;
 4using System.Linq;
 5using System.Threading;
 6using System.Threading.Tasks;
 7using Jellyfin.Data.Enums;
 8using Jellyfin.Extensions;
 9using MediaBrowser.Common.Configuration;
 10using MediaBrowser.Controller.Chapters;
 11using MediaBrowser.Controller.Dto;
 12using MediaBrowser.Controller.Entities;
 13using MediaBrowser.Controller.Library;
 14using MediaBrowser.Controller.MediaEncoding;
 15using MediaBrowser.Controller.Persistence;
 16using MediaBrowser.Controller.Providers;
 17using MediaBrowser.Model.Globalization;
 18using MediaBrowser.Model.IO;
 19using MediaBrowser.Model.Tasks;
 20using Microsoft.Extensions.Logging;
 21
 22namespace Emby.Server.Implementations.ScheduledTasks.Tasks
 23{
 24    /// <summary>
 25    /// Class ChapterImagesTask.
 26    /// </summary>
 27    public class ChapterImagesTask : IScheduledTask
 28    {
 29        private readonly ILogger<ChapterImagesTask> _logger;
 30        private readonly ILibraryManager _libraryManager;
 31        private readonly IItemRepository _itemRepo;
 32        private readonly IApplicationPaths _appPaths;
 33        private readonly IEncodingManager _encodingManager;
 34        private readonly IFileSystem _fileSystem;
 35        private readonly ILocalizationManager _localization;
 36        private readonly IChapterRepository _chapterRepository;
 37
 38        /// <summary>
 39        /// Initializes a new instance of the <see cref="ChapterImagesTask" /> class.
 40        /// </summary>
 41        /// <param name="logger">Instance of the <see cref="ILogger"/> interface.</param>
 42        /// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>
 43        /// <param name="itemRepo">Instance of the <see cref="IItemRepository"/> interface.</param>
 44        /// <param name="appPaths">Instance of the <see cref="IApplicationPaths"/> interface.</param>
 45        /// <param name="encodingManager">Instance of the <see cref="IEncodingManager"/> interface.</param>
 46        /// <param name="fileSystem">Instance of the <see cref="IFileSystem"/> interface.</param>
 47        /// <param name="localization">Instance of the <see cref="ILocalizationManager"/> interface.</param>
 48        /// <param name="chapterRepository">Instance of the <see cref="IChapterRepository"/> interface.</param>
 49        public ChapterImagesTask(
 50            ILogger<ChapterImagesTask> logger,
 51            ILibraryManager libraryManager,
 52            IItemRepository itemRepo,
 53            IApplicationPaths appPaths,
 54            IEncodingManager encodingManager,
 55            IFileSystem fileSystem,
 56            ILocalizationManager localization,
 57            IChapterRepository chapterRepository)
 58        {
 2159            _logger = logger;
 2160            _libraryManager = libraryManager;
 2161            _itemRepo = itemRepo;
 2162            _appPaths = appPaths;
 2163            _encodingManager = encodingManager;
 2164            _fileSystem = fileSystem;
 2165            _localization = localization;
 2166            _chapterRepository = chapterRepository;
 2167        }
 68
 69        /// <inheritdoc />
 2170        public string Name => _localization.GetLocalizedString("TaskRefreshChapterImages");
 71
 72        /// <inheritdoc />
 073        public string Description => _localization.GetLocalizedString("TaskRefreshChapterImagesDescription");
 74
 75        /// <inheritdoc />
 076        public string Category => _localization.GetLocalizedString("TasksLibraryCategory");
 77
 78        /// <inheritdoc />
 079        public string Key => "RefreshChapterImages";
 80
 81        /// <inheritdoc />
 82        public IEnumerable<TaskTriggerInfo> GetDefaultTriggers()
 83        {
 2184            return
 2185            [
 2186                new TaskTriggerInfo
 2187                {
 2188                    Type = TaskTriggerInfoType.DailyTrigger,
 2189                    TimeOfDayTicks = TimeSpan.FromHours(2).Ticks,
 2190                    MaxRuntimeTicks = TimeSpan.FromHours(4).Ticks
 2191                }
 2192            ];
 93        }
 94
 95        /// <inheritdoc />
 96        public async Task ExecuteAsync(IProgress<double> progress, CancellationToken cancellationToken)
 97        {
 98            var videos = _libraryManager.GetItemList(new InternalItemsQuery
 99            {
 100                MediaTypes = [MediaType.Video],
 101                IsFolder = false,
 102                Recursive = true,
 103                DtoOptions = new DtoOptions(false)
 104                {
 105                    EnableImages = false
 106                },
 107                SourceTypes = [SourceType.Library],
 108                IsVirtualItem = false
 109            })
 110            .OfType<Video>()
 111            .ToList();
 112
 113            var numComplete = 0;
 114
 115            var failHistoryPath = Path.Combine(_appPaths.CachePath, "chapter-failures.txt");
 116
 117            List<string> previouslyFailedImages;
 118
 119            if (File.Exists(failHistoryPath))
 120            {
 121                try
 122                {
 123                    previouslyFailedImages = (await File.ReadAllTextAsync(failHistoryPath, cancellationToken).ConfigureA
 124                        .Split('|', StringSplitOptions.RemoveEmptyEntries)
 125                        .ToList();
 126                }
 127                catch (IOException)
 128                {
 129                    previouslyFailedImages = new List<string>();
 130                }
 131            }
 132            else
 133            {
 134                previouslyFailedImages = new List<string>();
 135            }
 136
 137            var directoryService = new DirectoryService(_fileSystem);
 138
 139            foreach (var video in videos)
 140            {
 141                cancellationToken.ThrowIfCancellationRequested();
 142
 143                var key = video.Path + video.DateModified.Ticks;
 144
 145                var extract = !previouslyFailedImages.Contains(key, StringComparison.OrdinalIgnoreCase);
 146
 147                try
 148                {
 149                    var chapters = _chapterRepository.GetChapters(video.Id);
 150
 151                    var success = await _encodingManager.RefreshChapterImages(video, directoryService, chapters, extract
 152
 153                    if (!success)
 154                    {
 155                        previouslyFailedImages.Add(key);
 156
 157                        var parentPath = Path.GetDirectoryName(failHistoryPath);
 158                        if (parentPath is not null)
 159                        {
 160                            Directory.CreateDirectory(parentPath);
 161                        }
 162
 163                        string text = string.Join('|', previouslyFailedImages);
 164                        await File.WriteAllTextAsync(failHistoryPath, text, cancellationToken).ConfigureAwait(false);
 165                    }
 166
 167                    numComplete++;
 168                    double percent = numComplete;
 169                    percent /= videos.Count;
 170
 171                    progress.Report(100 * percent);
 172                }
 173                catch (ObjectDisposedException ex)
 174                {
 175                    // TODO Investigate and properly fix.
 176                    _logger.LogError(ex, "Object Disposed");
 177                    break;
 178                }
 179            }
 180        }
 181    }
 182}