using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.IO; using System.Threading; using System.Threading.Tasks; using MediaBrowser.Controller.Library; using MediaBrowser.Model.Tasks; using Microsoft.Extensions.Logging; namespace ConfusedPolarBear.Plugin.IntroSkipper; /// /// Analyze all television episodes for introduction sequences. /// TODO: FIXME: rename task and file to DetectIntroductionsTask. /// public class AnalyzeEpisodesTask : IScheduledTask { private readonly ILogger _logger; private readonly ILoggerFactory _loggerFactory; private readonly ILibraryManager? _libraryManager; /// /// Initializes a new instance of the class. /// /// Logger factory. /// Library manager. public AnalyzeEpisodesTask( ILoggerFactory loggerFactory, ILibraryManager libraryManager) : this(loggerFactory) { _libraryManager = libraryManager; } /// /// Initializes a new instance of the class. /// /// Logger factory. public AnalyzeEpisodesTask(ILoggerFactory loggerFactory) { _logger = loggerFactory.CreateLogger(); _loggerFactory = loggerFactory; EdlManager.Initialize(_logger); } /// /// Gets the task name. /// public string Name => "Detect Introductions"; /// /// Gets the task category. /// public string Category => "Intro Skipper"; /// /// Gets the task description. /// public string Description => "Analyzes the audio of all television episodes to find introduction sequences."; /// /// Gets the task key. /// public string Key => "CPBIntroSkipperDetectIntroductions"; /// /// Analyze all episodes in the queue. Only one instance of this task should be run at a time. /// /// Task progress. /// Cancellation token. /// Task. public Task ExecuteAsync(IProgress progress, CancellationToken cancellationToken) { if (_libraryManager is null) { throw new InvalidOperationException("Library manager must not be null"); } // Make sure the analysis queue matches what's currently in Jellyfin. var queueManager = new QueueManager( _loggerFactory.CreateLogger(), _libraryManager); queueManager.EnqueueAllEpisodes(); var queue = Plugin.Instance!.AnalysisQueue; if (queue.Count == 0) { throw new FingerprintException( "No episodes to analyze. If you are limiting the list of libraries to analyze, check that all library names have been spelled correctly."); } // Log EDL settings EdlManager.LogConfiguration(); var totalProcessed = 0; var options = new ParallelOptions() { MaxDegreeOfParallelism = Plugin.Instance!.Configuration.MaxParallelism }; // TODO: if the queue is modified while the task is running, the task will fail. // clone the queue before running the task to prevent this. // Analyze all episodes in the queue using the degrees of parallelism the user specified. Parallel.ForEach(queue, options, (season) => { var (episodes, unanalyzed) = VerifyEpisodes(season.Value.AsReadOnly()); if (episodes.Count == 0) { return; } var first = episodes[0]; var writeEdl = false; if (!unanalyzed) { _logger.LogDebug( "All episodes in {Name} season {Season} have already been analyzed", first.SeriesName, first.SeasonNumber); return; } try { if (cancellationToken.IsCancellationRequested) { return; } // Increment totalProcessed by the number of episodes in this season that were actually analyzed // (instead of just using the number of episodes in the current season). var analyzed = AnalyzeSeason(episodes, cancellationToken); Interlocked.Add(ref totalProcessed, analyzed); writeEdl = analyzed > 0 || Plugin.Instance!.Configuration.RegenerateEdlFiles; } catch (FingerprintException ex) { _logger.LogWarning( "Unable to analyze {Series} season {Season}: unable to fingerprint: {Ex}", first.SeriesName, first.SeasonNumber, ex); } catch (KeyNotFoundException ex) { _logger.LogWarning( "Unable to analyze {Series} season {Season}: cache miss: {Ex}", first.SeriesName, first.SeasonNumber, ex); } if (writeEdl && Plugin.Instance!.Configuration.EdlAction != EdlAction.None) { EdlManager.UpdateEDLFiles(episodes); } progress.Report((totalProcessed * 100) / Plugin.Instance!.TotalQueued); }); // Turn the regenerate EDL flag off after the scan completes. if (Plugin.Instance!.Configuration.RegenerateEdlFiles) { _logger.LogInformation("Turning EDL file regeneration flag off"); Plugin.Instance!.Configuration.RegenerateEdlFiles = false; Plugin.Instance!.SaveConfiguration(); } return Task.CompletedTask; } /// /// Verify that all episodes in a season exist in Jellyfin and as a file in storage. /// TODO: FIXME: move to queue manager. /// /// QueuedEpisodes. /// Verified QueuedEpisodes and a flag indicating if any episode in this season has not been analyzed yet. private ( ReadOnlyCollection VerifiedEpisodes, bool AnyUnanalyzed) VerifyEpisodes(ReadOnlyCollection candidates) { var unanalyzed = false; var verified = new List(); foreach (var candidate in candidates) { try { // Verify that the episode exists in Jellyfin and in storage var path = Plugin.Instance!.GetItemPath(candidate.EpisodeId); if (File.Exists(path)) { verified.Add(candidate); } // Flag this season for analysis if the current episode hasn't been analyzed yet if (!Plugin.Instance.Intros.ContainsKey(candidate.EpisodeId)) { unanalyzed = true; } } catch (Exception ex) { _logger.LogDebug( "Skipping analysis of {Name} ({Id}): {Exception}", candidate.Name, candidate.EpisodeId, ex); } } return (verified.AsReadOnly(), unanalyzed); } /// /// Fingerprints all episodes in the provided season and stores the timestamps of all introductions. /// /// Episodes in this season. /// Cancellation token provided by the scheduled task. /// Number of episodes from the provided season that were analyzed. private int AnalyzeSeason( ReadOnlyCollection episodes, CancellationToken cancellationToken) { // Skip seasons with an insufficient number of episodes. if (episodes.Count <= 1) { return episodes.Count; } // Only analyze specials (season 0) if the user has opted in. var first = episodes[0]; if (first.SeasonNumber == 0 && !Plugin.Instance!.Configuration.AnalyzeSeasonZero) { return 0; } _logger.LogInformation( "Analyzing {Count} episodes from {Name} season {Season}", episodes.Count, first.SeriesName, first.SeasonNumber); // Analyze the season with Chromaprint var chromaprint = new ChromaprintAnalyzer(_loggerFactory.CreateLogger()); chromaprint.AnalyzeMediaFiles(episodes, AnalysisMode.Introduction, cancellationToken); return episodes.Count; } /// /// Get task triggers. /// /// Task triggers. public IEnumerable GetDefaultTriggers() { return new[] { new TaskTriggerInfo { Type = TaskTriggerInfo.TriggerDaily, TimeOfDayTicks = TimeSpan.FromHours(0).Ticks } }; } }