915 lines
33 KiB
C#
Raw Normal View History

2022-05-01 00:33:22 -05:00
using System;
using System.Collections.Generic;
2022-05-03 01:09:50 -05:00
using System.Collections.ObjectModel;
2022-05-16 19:20:35 -05:00
using System.Numerics;
2022-05-01 00:33:22 -05:00
using System.Threading;
using System.Threading.Tasks;
2022-06-22 22:03:34 -05:00
using MediaBrowser.Controller.Library;
2022-05-01 00:33:22 -05:00
using MediaBrowser.Model.Tasks;
using Microsoft.Extensions.Logging;
namespace ConfusedPolarBear.Plugin.IntroSkipper;
/// <summary>
/// Fingerprint and analyze all queued episodes for common audio sequences.
2022-05-01 00:33:22 -05:00
/// </summary>
public class FingerprinterTask : IScheduledTask
{
2022-05-01 00:33:22 -05:00
/// <summary>
/// Maximum number of bits (out of 32 total) that can be different between segments before they are considered dissimilar.
2022-06-01 13:45:34 -05:00
/// 6 bits means the audio must be at least 81% similar (1 - 6 / 32).
2022-05-01 00:33:22 -05:00
/// </summary>
2022-06-01 02:51:57 -05:00
private const double MaximumDifferences = 6;
2022-05-01 00:33:22 -05:00
/// <summary>
/// Maximum time (in seconds) permitted between timestamps before they are considered non-contiguous.
2022-05-01 00:33:22 -05:00
/// </summary>
2022-06-01 02:51:57 -05:00
private const double MaximumDistance = 3.5;
2022-05-01 00:33:22 -05:00
/// <summary>
/// Seconds of audio in one fingerprint point. This value is defined by the Chromaprint library and should not be changed.
2022-05-01 00:33:22 -05:00
/// </summary>
2022-05-09 22:56:03 -05:00
private const double SamplesToSeconds = 0.128;
2022-05-01 00:33:22 -05:00
2022-05-13 01:13:13 -05:00
/// <summary>
/// Bucket size used in the reanalysis histogram.
/// </summary>
private const int ReanalysisBucketWidth = 5;
/// <summary>
/// Maximum time (in seconds) that an intro's duration can be different from a typical intro's duration before marking it for reanalysis.
/// </summary>
private const double ReanalysisTolerance = ReanalysisBucketWidth * 1.5;
2022-05-09 22:56:03 -05:00
private readonly ILogger<FingerprinterTask> _logger;
2022-06-22 22:03:34 -05:00
private readonly ILogger<QueueManager> _queueLogger;
private readonly ILibraryManager? _libraryManager;
2022-05-17 02:06:34 -05:00
/// <summary>
/// Lock which guards the fingerprint cache dictionary.
/// </summary>
private readonly object _fingerprintCacheLock = new object();
2022-05-17 02:06:34 -05:00
/// <summary>
/// Lock which guards the shared dictionary of intros.
/// </summary>
private readonly object _introsLock = new object();
/// <summary>
/// Temporary fingerprint cache to speed up reanalysis.
/// Fingerprints are removed from this after a season is analyzed.
/// </summary>
private Dictionary<Guid, ReadOnlyCollection<uint>> _fingerprintCache;
2022-07-04 02:03:10 -05:00
/// <summary>
/// Statistics for the currently running analysis task.
/// </summary>
private AnalysisStatistics analysisStatistics = new AnalysisStatistics();
/// <summary>
/// Minimum duration of similar audio that will be considered an introduction.
/// </summary>
private static int minimumIntroDuration = 15;
2022-05-01 00:33:22 -05:00
/// <summary>
/// Initializes a new instance of the <see cref="FingerprinterTask"/> class.
2022-05-01 00:33:22 -05:00
/// </summary>
2022-06-22 22:03:34 -05:00
/// <param name="loggerFactory">Logger factory.</param>
/// <param name="libraryManager">Library manager.</param>
public FingerprinterTask(ILoggerFactory loggerFactory, ILibraryManager libraryManager) : this(loggerFactory)
2022-05-01 00:33:22 -05:00
{
2022-06-22 22:03:34 -05:00
_libraryManager = libraryManager;
}
/// <summary>
/// Initializes a new instance of the <see cref="FingerprinterTask"/> class.
/// </summary>
/// <param name="loggerFactory">Logger factory.</param>
public FingerprinterTask(ILoggerFactory loggerFactory)
{
_logger = loggerFactory.CreateLogger<FingerprinterTask>();
_queueLogger = loggerFactory.CreateLogger<QueueManager>();
_fingerprintCache = new Dictionary<Guid, ReadOnlyCollection<uint>>();
EdlManager.Initialize(_logger);
2022-05-01 00:33:22 -05:00
}
/// <summary>
/// Gets the task name.
2022-05-01 00:33:22 -05:00
/// </summary>
public string Name => "Analyze episodes";
/// <summary>
/// Gets the task category.
2022-05-01 00:33:22 -05:00
/// </summary>
public string Category => "Intro Skipper";
/// <summary>
/// Gets the task description.
2022-05-01 00:33:22 -05:00
/// </summary>
public string Description => "Analyzes the audio of all television episodes to find introduction sequences.";
/// <summary>
/// Gets the task key.
2022-05-01 00:33:22 -05:00
/// </summary>
public string Key => "CPBIntroSkipperRunFingerprinter";
/// <summary>
2022-06-22 22:03:34 -05:00
/// Analyze all episodes in the queue. Only one instance of this task should be run at a time.
2022-05-01 00:33:22 -05:00
/// </summary>
/// <param name="progress">Task progress.</param>
2022-05-01 00:33:22 -05:00
/// <param name="cancellationToken">Cancellation token.</param>
/// <returns>Task.</returns>
2022-05-01 00:33:22 -05:00
public Task ExecuteAsync(IProgress<double> progress, CancellationToken cancellationToken)
{
2022-06-22 22:03:34 -05:00
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(_queueLogger, _libraryManager);
queueManager.EnqueueAllEpisodes();
2022-05-01 00:33:22 -05:00
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.");
}
2022-06-24 00:02:08 -05:00
// Log EDL settings
EdlManager.LogConfiguration();
// Include the previously processed episodes in the percentage reported to the UI.
var totalProcessed = CountProcessedEpisodes();
var options = new ParallelOptions()
{
MaxDegreeOfParallelism = Plugin.Instance!.Configuration.MaxParallelism
};
2022-05-17 02:06:34 -05:00
2022-07-04 02:03:10 -05:00
var taskStart = DateTime.Now;
analysisStatistics = new AnalysisStatistics();
analysisStatistics.TotalQueuedEpisodes = Plugin.Instance!.TotalQueued;
minimumIntroDuration = Plugin.Instance!.Configuration.MinimumIntroDuration;
2022-06-24 00:02:08 -05:00
// Analyze all episodes in the queue using the degrees of parallelism the user specified.
2022-05-17 02:06:34 -05:00
Parallel.ForEach(queue, options, (season) =>
{
2022-07-04 02:03:10 -05:00
var workerStart = DateTime.Now;
var first = season.Value[0];
var writeEdl = false;
try
{
// 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(season, cancellationToken);
Interlocked.Add(ref totalProcessed, analyzed);
2022-06-24 00:02:08 -05:00
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);
}
// Clear this season's episodes from the temporary fingerprint cache.
lock (_fingerprintCacheLock)
{
foreach (var ep in season.Value)
{
_fingerprintCache.Remove(ep.EpisodeId);
}
}
2022-05-01 00:33:22 -05:00
if (writeEdl && Plugin.Instance!.Configuration.EdlAction != EdlAction.None)
{
EdlManager.UpdateEDLFiles(season.Value.AsReadOnly());
}
2022-05-13 01:28:06 -05:00
progress.Report((totalProcessed * 100) / Plugin.Instance!.TotalQueued);
2022-07-04 02:03:10 -05:00
analysisStatistics.TotalCPUTime.AddDuration(workerStart);
Plugin.Instance!.AnalysisStatistics = analysisStatistics;
2022-05-16 17:06:46 -05:00
});
2022-05-13 01:28:06 -05:00
2022-07-08 01:02:50 -05:00
// Update analysis statistics
2022-07-04 02:03:10 -05:00
analysisStatistics.TotalTaskTime.AddDuration(taskStart);
Plugin.Instance!.AnalysisStatistics = analysisStatistics;
2022-06-24 00:02:08 -05:00
// 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();
}
2022-05-13 01:28:06 -05:00
return Task.CompletedTask;
}
/// <summary>
/// Count the number of previously processed episodes to ensure the reported progress is correct.
/// </summary>
/// <returns>Number of previously processed episodes.</returns>
private int CountProcessedEpisodes()
{
var previous = 0;
foreach (var season in Plugin.Instance!.AnalysisQueue)
{
foreach (var episode in season.Value)
{
if (!Plugin.Instance!.Intros.TryGetValue(episode.EpisodeId, out var intro) || !intro.Valid)
{
continue;
}
previous++;
}
}
return previous;
}
/// <summary>
/// Fingerprints all episodes in the provided season and stores the timestamps of all introductions.
/// </summary>
/// <param name="season">Pairing of season GUID to a list of QueuedEpisode objects.</param>
/// <param name="cancellationToken">Cancellation token provided by the scheduled task.</param>
/// <returns>Number of episodes from the provided season that were analyzed.</returns>
private int AnalyzeSeason(
2022-05-13 01:28:06 -05:00
KeyValuePair<Guid, List<QueuedEpisode>> season,
CancellationToken cancellationToken)
{
2022-05-17 02:06:34 -05:00
var seasonIntros = new Dictionary<Guid, Intro>();
var episodes = season.Value;
var first = episodes[0];
2022-05-13 01:28:06 -05:00
/* Don't analyze specials or seasons with an insufficient number of episodes.
* A season with only 1 episode can't be analyzed as it would compare the episode to itself,
* which would result in the entire episode being marked as an introduction, as the audio is identical.
*/
if (season.Value.Count < 2 || first.SeasonNumber == 0)
{
return episodes.Count;
2022-05-13 01:28:06 -05:00
}
2022-06-19 00:03:10 -05:00
var unanalyzed = false;
// Only log an analysis message if there are unanalyzed episodes in this season.
foreach (var episode in episodes)
{
if (!Plugin.Instance!.Intros.ContainsKey(episode.EpisodeId))
{
unanalyzed = true;
break;
}
}
if (unanalyzed)
{
_logger.LogInformation(
"Analyzing {Count} episodes from {Name} season {Season}",
season.Value.Count,
first.SeriesName,
first.SeasonNumber);
2022-06-19 00:03:10 -05:00
}
else
{
_logger.LogDebug(
"All episodes from {Name} season {Season} have already been analyzed",
first.SeriesName,
first.SeasonNumber);
return 0;
2022-06-19 00:03:10 -05:00
}
2022-05-13 01:28:06 -05:00
// Ensure there are an even number of episodes
if (episodes.Count % 2 != 0)
{
episodes.Add(episodes[episodes.Count - 2]);
}
// Analyze each pair of episodes in the current season
var everFoundIntro = false;
var failures = 0;
for (var i = 0; i < episodes.Count; i += 2)
{
if (cancellationToken.IsCancellationRequested)
2022-05-02 01:18:31 -05:00
{
2022-05-13 01:28:06 -05:00
break;
2022-05-02 01:18:31 -05:00
}
2022-05-13 01:28:06 -05:00
var lhs = episodes[i];
var rhs = episodes[i + 1];
2022-05-01 00:33:22 -05:00
if (!everFoundIntro && failures >= 20)
{
2022-05-13 01:28:06 -05:00
_logger.LogWarning(
"Failed to find an introduction in {Series} season {Season}",
lhs.SeriesName,
lhs.SeasonNumber);
break;
2022-05-01 00:33:22 -05:00
}
if (Plugin.Instance!.Intros.ContainsKey(lhs.EpisodeId) && Plugin.Instance!.Intros.ContainsKey(rhs.EpisodeId))
2022-05-01 00:33:22 -05:00
{
2022-06-12 17:31:42 -05:00
_logger.LogTrace(
2022-05-13 01:28:06 -05:00
"Episodes {LHS} and {RHS} have both already been fingerprinted",
lhs.EpisodeId,
rhs.EpisodeId);
2022-05-02 01:18:31 -05:00
2022-05-13 01:28:06 -05:00
continue;
}
2022-05-03 01:09:50 -05:00
2022-05-13 01:28:06 -05:00
try
{
2022-06-12 17:31:42 -05:00
_logger.LogTrace("Analyzing {LHS} and {RHS}", lhs.Path, rhs.Path);
2022-05-03 01:09:50 -05:00
2022-05-13 01:28:06 -05:00
var (lhsIntro, rhsIntro) = FingerprintEpisodes(lhs, rhs);
2022-05-17 02:06:34 -05:00
seasonIntros[lhsIntro.EpisodeId] = lhsIntro;
seasonIntros[rhsIntro.EpisodeId] = rhsIntro;
2022-07-04 02:03:10 -05:00
analysisStatistics.TotalAnalyzedEpisodes.Add(2);
2022-05-13 01:28:06 -05:00
if (!lhsIntro.Valid)
{
failures += 2;
2022-05-01 00:33:22 -05:00
continue;
}
2022-05-13 01:28:06 -05:00
everFoundIntro = true;
2022-05-01 00:33:22 -05:00
}
2022-05-13 01:28:06 -05:00
catch (FingerprintException ex)
2022-05-02 01:18:31 -05:00
{
2022-05-13 01:28:06 -05:00
_logger.LogError("Caught fingerprint error: {Ex}", ex);
2022-05-02 01:18:31 -05:00
}
2022-05-13 01:28:06 -05:00
}
2022-05-13 01:13:13 -05:00
2022-05-17 02:06:34 -05:00
// Ensure only one thread at a time can update the shared intro dictionary.
lock (_introsLock)
{
foreach (var intro in seasonIntros)
{
Plugin.Instance!.Intros[intro.Key] = intro.Value;
}
}
2022-05-13 01:28:06 -05:00
2022-05-19 00:59:58 -05:00
// Only run the second pass if the user hasn't requested cancellation and we found an intro
if (!cancellationToken.IsCancellationRequested && everFoundIntro)
2022-05-13 01:28:06 -05:00
{
2022-07-04 02:03:10 -05:00
var start = DateTime.Now;
2022-05-19 00:59:58 -05:00
// Run a second pass over this season to remove outliers and fix episodes that failed in the first pass.
RunSecondPass(season.Value);
2022-07-04 02:03:10 -05:00
analysisStatistics.SecondPassCPUTime.AddDuration(start);
2022-05-01 00:33:22 -05:00
}
2022-05-19 00:59:58 -05:00
lock (_introsLock)
{
Plugin.Instance!.SaveTimestamps();
}
return episodes.Count;
2022-05-01 00:33:22 -05:00
}
2022-05-03 01:09:50 -05:00
/// <summary>
/// Analyze two episodes to find an introduction sequence shared between them.
/// </summary>
/// <param name="lhsEpisode">First episode to analyze.</param>
/// <param name="rhsEpisode">Second episode to analyze.</param>
2022-05-13 01:13:13 -05:00
/// <returns>Intros for the first and second episodes.</returns>
public (Intro Lhs, Intro Rhs) FingerprintEpisodes(QueuedEpisode lhsEpisode, QueuedEpisode rhsEpisode)
2022-05-01 00:33:22 -05:00
{
2022-07-04 02:03:10 -05:00
var start = DateTime.Now;
2022-06-09 17:33:39 -05:00
var lhsFingerprint = Chromaprint.Fingerprint(lhsEpisode);
var rhsFingerprint = Chromaprint.Fingerprint(rhsEpisode);
2022-07-04 02:03:10 -05:00
analysisStatistics.FingerprintCPUTime.AddDuration(start);
2022-05-13 01:13:13 -05:00
// Cache the fingerprints for quicker recall in the second pass (if one is needed).
lock (_fingerprintCacheLock)
{
_fingerprintCache[lhsEpisode.EpisodeId] = lhsFingerprint;
_fingerprintCache[rhsEpisode.EpisodeId] = rhsFingerprint;
}
2022-05-13 01:13:13 -05:00
return FingerprintEpisodes(
lhsEpisode.EpisodeId,
lhsFingerprint,
rhsEpisode.EpisodeId,
2022-07-05 16:16:48 -05:00
rhsFingerprint,
true);
2022-05-13 01:13:13 -05:00
}
2022-05-01 00:33:22 -05:00
2022-05-13 01:13:13 -05:00
/// <summary>
/// Analyze two episodes to find an introduction sequence shared between them.
/// </summary>
/// <param name="lhsId">First episode id.</param>
2022-05-16 23:08:20 -05:00
/// <param name="lhsPoints">First episode fingerprint points.</param>
2022-05-13 01:13:13 -05:00
/// <param name="rhsId">Second episode id.</param>
2022-05-16 23:08:20 -05:00
/// <param name="rhsPoints">Second episode fingerprint points.</param>
2022-07-05 16:16:48 -05:00
/// <param name="isFirstPass">If this was called as part of the first analysis pass, add the elapsed time to the statistics.</param>
2022-05-13 01:13:13 -05:00
/// <returns>Intros for the first and second episodes.</returns>
public (Intro Lhs, Intro Rhs) FingerprintEpisodes(
Guid lhsId,
2022-05-16 23:08:20 -05:00
ReadOnlyCollection<uint> lhsPoints,
2022-05-13 01:13:13 -05:00
Guid rhsId,
2022-07-05 16:16:48 -05:00
ReadOnlyCollection<uint> rhsPoints,
bool isFirstPass)
2022-05-13 01:13:13 -05:00
{
2022-07-05 16:16:48 -05:00
// If this isn't running as part of the first analysis pass, don't count this CPU time as first pass time.
var start = isFirstPass ? DateTime.Now : DateTime.MinValue;
2022-05-01 00:33:22 -05:00
2022-07-05 16:16:48 -05:00
// ===== Method 1: Inverted indexes =====
// Creates an inverted fingerprint point index for both episodes.
// For every point which is a 100% match, search for an introduction at that point.
var (lhsRanges, rhsRanges) = SearchInvertedIndex(lhsPoints, rhsPoints);
2022-05-01 00:33:22 -05:00
2022-07-05 16:16:48 -05:00
if (lhsRanges.Count > 0)
2022-05-03 01:09:50 -05:00
{
2022-07-05 16:16:48 -05:00
_logger.LogTrace("Index search successful");
analysisStatistics.IndexSearches.Increment();
analysisStatistics.FirstPassCPUTime.AddDuration(start);
2022-05-01 00:33:22 -05:00
2022-07-05 16:16:48 -05:00
return GetLongestTimeRange(lhsId, lhsRanges, rhsId, rhsRanges);
2022-05-03 01:09:50 -05:00
}
2022-07-05 16:16:48 -05:00
// ===== Method 2: Quick scan =====
// Tests if an intro can be found within the first 5 seconds of the episodes. ±5/0.128 = ±40 samples.
(lhsRanges, rhsRanges) = ShiftEpisodes(lhsPoints, rhsPoints, -40, 40);
if (lhsRanges.Count > 0)
2022-05-03 01:09:50 -05:00
{
2022-07-05 16:16:48 -05:00
_logger.LogTrace("Quick scan successful");
2022-07-04 02:03:10 -05:00
analysisStatistics.QuickScans.Increment();
2022-07-05 16:16:48 -05:00
analysisStatistics.FirstPassCPUTime.AddDuration(start);
return GetLongestTimeRange(lhsId, lhsRanges, rhsId, rhsRanges);
2022-05-01 00:33:22 -05:00
}
2022-07-05 16:16:48 -05:00
// ===== Method 3: Full scan =====
// Compares all elements of the shortest fingerprint to the other fingerprint.
var limit = Math.Min(lhsPoints.Count, rhsPoints.Count);
(lhsRanges, rhsRanges) = ShiftEpisodes(lhsPoints, rhsPoints, -1 * limit, limit);
2022-05-01 00:33:22 -05:00
2022-07-05 16:16:48 -05:00
if (lhsRanges.Count > 0)
{
_logger.LogTrace("Full scan successful");
analysisStatistics.FullScans.Increment();
2022-07-04 02:03:10 -05:00
analysisStatistics.FirstPassCPUTime.AddDuration(start);
2022-07-05 16:16:48 -05:00
return GetLongestTimeRange(lhsId, lhsRanges, rhsId, rhsRanges);
2022-05-01 00:33:22 -05:00
}
2022-07-05 16:16:48 -05:00
// No method was able to find an introduction, return nothing.
_logger.LogTrace(
"Unable to find a shared introduction sequence between {LHS} and {RHS}",
lhsId,
rhsId);
analysisStatistics.FirstPassCPUTime.AddDuration(start);
return (new Intro(lhsId), new Intro(rhsId));
}
/// <summary>
/// Locates the longest range of similar audio and returns an Intro class for each range.
/// </summary>
/// <param name="lhsId">First episode id.</param>
/// <param name="lhsRanges">First episode shared timecodes.</param>
/// <param name="rhsId">Second episode id.</param>
/// <param name="rhsRanges">Second episode shared timecodes.</param>
/// <returns>Intros for the first and second episodes.</returns>
private (Intro Lhs, Intro Rhs) GetLongestTimeRange(
Guid lhsId,
List<TimeRange> lhsRanges,
Guid rhsId,
List<TimeRange> rhsRanges)
{
// Store the longest time range as the introduction.
2022-05-01 00:33:22 -05:00
lhsRanges.Sort();
rhsRanges.Sort();
var lhsIntro = lhsRanges[0];
var rhsIntro = rhsRanges[0];
2022-05-03 01:09:50 -05:00
// If the intro starts early in the episode, move it to the beginning.
2022-05-01 00:33:22 -05:00
if (lhsIntro.Start <= 5)
{
lhsIntro.Start = 0;
}
if (rhsIntro.Start <= 5)
{
rhsIntro.Start = 0;
}
2022-07-05 16:16:48 -05:00
// Create Intro classes for each time range.
2022-05-16 23:08:20 -05:00
return (new Intro(lhsId, lhsIntro), new Intro(rhsId, rhsIntro));
2022-05-03 01:09:50 -05:00
}
2022-07-05 16:16:48 -05:00
/// <summary>
/// Search for a shared introduction sequence using inverted indexes.
/// </summary>
/// <param name="lhsPoints">Left episode fingerprint points.</param>
/// <param name="rhsPoints">Right episode fingerprint points.</param>
/// <returns>List of shared TimeRanges between the left and right episodes.</returns>
private (List<TimeRange> Lhs, List<TimeRange> Rhs) SearchInvertedIndex(
ReadOnlyCollection<uint> lhsPoints,
ReadOnlyCollection<uint> rhsPoints)
{
var lhsRanges = new List<TimeRange>();
var rhsRanges = new List<TimeRange>();
// Generate inverted indexes for the left and right episodes.
var lhsIndex = Chromaprint.CreateInvertedIndex(lhsPoints);
var rhsIndex = Chromaprint.CreateInvertedIndex(rhsPoints);
var indexShifts = new HashSet<int>();
// For all audio points in the left episode, check if the right episode has a point which matches exactly.
// If an exact match is found, calculate the shift that must be used to align the points.
foreach (var kvp in lhsIndex)
{
var point = kvp.Key;
if (rhsIndex.ContainsKey(point))
{
2022-07-05 17:08:52 -05:00
var lhsFirst = (int)lhsIndex[point];
var rhsFirst = (int)rhsIndex[point];
2022-07-05 16:16:48 -05:00
indexShifts.Add(rhsFirst - lhsFirst);
}
}
// Use all discovered shifts to compare the episodes.
foreach (var shift in indexShifts)
{
var (lhsIndexContiguous, rhsIndexContiguous) = ShiftEpisodes(lhsPoints, rhsPoints, shift, shift);
lhsRanges.AddRange(lhsIndexContiguous);
rhsRanges.AddRange(rhsIndexContiguous);
}
return (lhsRanges, rhsRanges);
}
2022-05-03 01:09:50 -05:00
/// <summary>
2022-05-16 23:08:20 -05:00
/// Shifts a pair of episodes through the range of provided shift amounts and returns discovered contiguous time ranges.
2022-05-03 01:09:50 -05:00
/// </summary>
/// <param name="lhs">First episode fingerprint.</param>
/// <param name="rhs">Second episode fingerprint.</param>
/// <param name="lower">Lower end of the shift range.</param>
/// <param name="upper">Upper end of the shift range.</param>
private static (List<TimeRange> Lhs, List<TimeRange> Rhs) ShiftEpisodes(
2022-05-03 01:09:50 -05:00
ReadOnlyCollection<uint> lhs,
ReadOnlyCollection<uint> rhs,
int lower,
int upper)
{
2022-05-03 01:09:50 -05:00
var lhsRanges = new List<TimeRange>();
var rhsRanges = new List<TimeRange>();
for (int amount = lower; amount <= upper; amount++)
{
var (lRange, rRange) = FindContiguous(lhs, rhs, amount);
2022-05-03 01:09:50 -05:00
if (lRange.End == 0 && rRange.End == 0)
{
continue;
}
lhsRanges.Add(lRange);
rhsRanges.Add(rRange);
}
return (lhsRanges, rhsRanges);
}
/// <summary>
/// Finds the longest contiguous region of similar audio between two fingerprints using the provided shift amount.
/// </summary>
/// <param name="lhs">First fingerprint to compare.</param>
/// <param name="rhs">Second fingerprint to compare.</param>
/// <param name="shiftAmount">Amount to shift one fingerprint by.</param>
private static (TimeRange Lhs, TimeRange Rhs) FindContiguous(
2022-05-03 01:09:50 -05:00
ReadOnlyCollection<uint> lhs,
ReadOnlyCollection<uint> rhs,
int shiftAmount)
{
2022-05-03 01:09:50 -05:00
var leftOffset = 0;
var rightOffset = 0;
// Calculate the offsets for the left and right hand sides.
if (shiftAmount < 0)
{
2022-05-03 01:09:50 -05:00
leftOffset -= shiftAmount;
}
else
{
2022-05-03 01:09:50 -05:00
rightOffset += shiftAmount;
}
// Store similar times for both LHS and RHS.
var lhsTimes = new List<double>();
var rhsTimes = new List<double>();
var upperLimit = Math.Min(lhs.Count, rhs.Count) - Math.Abs(shiftAmount);
// XOR all elements in LHS and RHS, using the shift amount from above.
for (var i = 0; i < upperLimit; i++)
{
2022-05-03 01:09:50 -05:00
// XOR both samples at the current position.
var lhsPosition = i + leftOffset;
var rhsPosition = i + rightOffset;
var diff = lhs[lhsPosition] ^ rhs[rhsPosition];
2022-05-07 21:11:59 -05:00
// If the difference between the samples is small, flag both times as similar.
2022-05-09 22:56:03 -05:00
if (CountBits(diff) > MaximumDifferences)
2022-05-03 01:09:50 -05:00
{
continue;
}
2022-05-09 22:56:03 -05:00
var lhsTime = lhsPosition * SamplesToSeconds;
var rhsTime = rhsPosition * SamplesToSeconds;
2022-05-03 01:09:50 -05:00
lhsTimes.Add(lhsTime);
rhsTimes.Add(rhsTime);
}
// Ensure the last timestamp is checked
lhsTimes.Add(double.MaxValue);
rhsTimes.Add(double.MaxValue);
2022-05-03 01:09:50 -05:00
// Now that both fingerprints have been compared at this shift, see if there's a contiguous time range.
2022-05-09 22:56:03 -05:00
var lContiguous = TimeRangeHelpers.FindContiguous(lhsTimes.ToArray(), MaximumDistance);
if (lContiguous is null || lContiguous.Duration < minimumIntroDuration)
2022-05-03 01:09:50 -05:00
{
return (new TimeRange(), new TimeRange());
}
// Since LHS had a contiguous time range, RHS must have one also.
2022-05-09 22:56:03 -05:00
var rContiguous = TimeRangeHelpers.FindContiguous(rhsTimes.ToArray(), MaximumDistance)!;
2022-05-03 01:09:50 -05:00
// Tweak the end timestamps just a bit to ensure as little content as possible is skipped over.
if (lContiguous.Duration >= 90)
{
lContiguous.End -= 2 * MaximumDistance;
rContiguous.End -= 2 * MaximumDistance;
2022-05-03 01:09:50 -05:00
}
2022-06-01 02:51:57 -05:00
else if (lContiguous.Duration >= 30)
2022-05-03 01:09:50 -05:00
{
lContiguous.End -= MaximumDistance;
rContiguous.End -= MaximumDistance;
2022-05-03 01:09:50 -05:00
}
return (lContiguous, rContiguous);
2022-05-01 00:33:22 -05:00
}
/// <summary>
/// Count the number of bits that are set in the provided number.
/// </summary>
/// <param name="number">Number to count bits in.</param>
/// <returns>Number of bits that are equal to 1.</returns>
public static int CountBits(uint number)
{
2022-05-16 19:20:35 -05:00
return BitOperations.PopCount(number);
2022-05-01 00:33:22 -05:00
}
2022-05-13 01:13:13 -05:00
/// <summary>
/// Reanalyze the most recently analyzed season.
/// Looks for and fixes intro durations that were either not found or are statistical outliers.
/// </summary>
/// <param name="episodes">List of episodes that was just analyzed.</param>
2022-05-19 00:59:58 -05:00
private void RunSecondPass(List<QueuedEpisode> episodes)
2022-05-13 01:13:13 -05:00
{
// First, assert that at least half of the episodes in this season have an intro.
var validCount = 0;
var totalCount = episodes.Count;
foreach (var episode in episodes)
{
2022-06-24 16:00:33 -05:00
if (Plugin.Instance!.Intros.TryGetValue(episode.EpisodeId, out var intro) && intro.Valid)
2022-05-13 01:13:13 -05:00
{
validCount++;
}
}
var percentValid = (validCount * 100) / totalCount;
2022-06-12 17:31:42 -05:00
_logger.LogTrace("Found intros in {Valid}/{Total} ({Percent}%) of episodes", validCount, totalCount, percentValid);
2022-05-13 01:13:13 -05:00
if (percentValid < 50)
{
return;
}
// Create a histogram of all episode durations
var histogram = new Dictionary<int, SeasonHistogram>();
foreach (var episode in episodes)
{
var id = episode.EpisodeId;
var duration = GetIntroDuration(id);
if (duration < minimumIntroDuration)
2022-05-13 01:13:13 -05:00
{
continue;
}
// Bucket the duration into equally sized groups
var bucket = Convert.ToInt32(Math.Floor(duration / ReanalysisBucketWidth)) * ReanalysisBucketWidth;
// TryAdd returns true when the key was successfully added (i.e. for newly created buckets).
// Newly created buckets are initialized with the provided episode ID, so nothing else needs to be done for them.
if (histogram.TryAdd(bucket, new SeasonHistogram(id)))
{
continue;
}
histogram[bucket].Episodes.Add(id);
}
// Find the bucket that was seen most often, as this is likely to be the true intro length.
var maxDuration = 0;
var maxBucket = new SeasonHistogram(Guid.Empty);
foreach (var entry in histogram)
{
if (entry.Value.Count > maxBucket.Count)
{
maxDuration = entry.Key;
maxBucket = entry.Value;
}
}
// Ensure that the most frequently seen bucket has a majority
percentValid = (maxBucket.Count * 100) / validCount;
2022-06-12 17:31:42 -05:00
_logger.LogTrace(
2022-05-13 01:13:13 -05:00
"Intro duration {Duration} appeared {Frequency} times ({Percent}%)",
maxDuration,
maxBucket.Count,
percentValid);
if (percentValid < 50 || maxBucket.Episodes[0].Equals(Guid.Empty))
{
return;
}
2022-06-12 17:31:42 -05:00
_logger.LogTrace("Second pass is processing {Count} episodes", totalCount - maxBucket.Count);
2022-05-13 01:13:13 -05:00
2022-05-19 00:59:58 -05:00
// Calculate a range of intro durations that are most likely to be correct.
var maxEpisode = episodes.Find(x => x.EpisodeId == maxBucket.Episodes[0]);
if (maxEpisode is null)
2022-05-13 01:13:13 -05:00
{
2022-05-19 00:59:58 -05:00
_logger.LogError("Second pass failed to get episode from bucket");
2022-05-13 01:13:13 -05:00
return;
}
2022-05-19 00:59:58 -05:00
var lhsDuration = GetIntroDuration(maxEpisode.EpisodeId);
2022-05-13 01:13:13 -05:00
var (lowTargetDuration, highTargetDuration) = (
lhsDuration - ReanalysisTolerance,
lhsDuration + ReanalysisTolerance);
2022-05-19 00:59:58 -05:00
// TODO: add limit and make it customizable
var count = maxBucket.Episodes.Count - 1;
var goodFingerprints = new List<ReadOnlyCollection<uint>>();
foreach (var id in maxBucket.Episodes)
{
// Sometimes an episode isn't in the fingerprint cache. When this occurs, it has to be fingerprinted again.
_logger.LogTrace("Second pass: searching cache for {Id}", id);
if (_fingerprintCache.ContainsKey(id))
{
_logger.LogTrace("Second pass: cache hit for {Id}", id);
goodFingerprints.Add(_fingerprintCache[id]);
}
else
{
_logger.LogTrace("Second pass: cache miss for {Id}", id);
var fullEp = episodes.Find((e) => { return e.EpisodeId == id; });
if (fullEp is not null)
{
goodFingerprints.Add(Chromaprint.Fingerprint(fullEp));
}
else
{
_logger.LogTrace("Second pass: unable to find episode {Id}", id);
}
}
2022-05-19 00:59:58 -05:00
}
2022-05-13 01:13:13 -05:00
foreach (var episode in episodes)
{
// Don't reanalyze episodes from the max bucket
if (maxBucket.Episodes.Contains(episode.EpisodeId))
{
continue;
}
var oldDuration = GetIntroDuration(episode.EpisodeId);
// If the episode's intro duration is close enough to the targeted bucket, leave it alone.
if (Math.Abs(lhsDuration - oldDuration) <= ReanalysisTolerance)
{
2022-06-12 17:31:42 -05:00
_logger.LogTrace(
2022-05-13 01:13:13 -05:00
"Not reanalyzing episode {Path} (intro is {Initial}, target is {Max})",
2022-07-05 16:17:25 -05:00
episode.Path,
2022-05-13 01:13:13 -05:00
Math.Round(oldDuration, 2),
maxDuration);
continue;
}
2022-06-12 17:31:42 -05:00
_logger.LogTrace(
2022-05-13 01:13:13 -05:00
"Reanalyzing episode {Path} (intro is {Initial}, target is {Max})",
2022-07-05 16:17:25 -05:00
episode.Path,
2022-05-13 01:13:13 -05:00
Math.Round(oldDuration, 2),
maxDuration);
// Analyze the episode again, ignoring whatever is returned for the known good episode.
2022-05-19 00:59:58 -05:00
foreach (var lhsFingerprint in goodFingerprints)
2022-05-13 01:13:13 -05:00
{
2022-07-03 02:47:48 -05:00
if (!_fingerprintCache.TryGetValue(episode.EpisodeId, out var fp))
{
_logger.LogTrace("Unable to get cached fingerprint for {Id}, skipping", episode.EpisodeId);
continue;
}
2022-05-19 00:59:58 -05:00
var (_, newRhs) = FingerprintEpisodes(
maxEpisode.EpisodeId,
lhsFingerprint,
episode.EpisodeId,
2022-07-05 16:17:25 -05:00
fp,
false);
2022-05-19 00:59:58 -05:00
// Ensure that the new intro duration is within the targeted bucket and longer than what was found previously.
var newDuration = Math.Round(newRhs.IntroEnd - newRhs.IntroStart, 2);
if (newDuration < oldDuration || newDuration < lowTargetDuration || newDuration > highTargetDuration)
{
2022-06-12 17:31:42 -05:00
_logger.LogTrace(
2022-05-19 00:59:58 -05:00
"Ignoring reanalysis for {Path} (was {Initial}, now is {New})",
2022-07-05 16:17:25 -05:00
episode.Path,
2022-05-19 00:59:58 -05:00
oldDuration,
newDuration);
continue;
}
2022-06-12 17:31:42 -05:00
_logger.LogTrace(
2022-05-19 00:59:58 -05:00
"Reanalysis succeeded for {Path} (was {Initial}, now is {New})",
2022-07-05 16:17:25 -05:00
episode.Path,
2022-05-13 01:13:13 -05:00
oldDuration,
newDuration);
2022-05-19 00:59:58 -05:00
lock (_introsLock)
{
Plugin.Instance!.Intros[episode.EpisodeId] = newRhs;
}
2022-05-13 01:13:13 -05:00
2022-05-19 00:59:58 -05:00
break;
2022-05-17 02:06:34 -05:00
}
2022-05-13 01:13:13 -05:00
}
}
private double GetIntroDuration(Guid id)
{
2022-06-24 16:00:33 -05:00
if (!Plugin.Instance!.Intros.TryGetValue(id, out var episode))
{
return 0;
}
2022-05-13 01:13:13 -05:00
return episode.Valid ? Math.Round(episode.IntroEnd - episode.IntroStart, 2) : 0;
}
2022-05-01 00:33:22 -05:00
/// <summary>
/// Get task triggers.
/// </summary>
/// <returns>Task triggers.</returns>
2022-05-01 00:33:22 -05:00
public IEnumerable<TaskTriggerInfo> GetDefaultTriggers()
{
return new[]
{
new TaskTriggerInfo
{
Type = TaskTriggerInfo.TriggerDaily,
TimeOfDayTicks = TimeSpan.FromDays(24).Ticks
}
};
}
}