2024-10-25 14:31:50 -04:00
|
|
|
// Copyright (C) 2024 Intro-Skipper contributors <intro-skipper.org>
|
|
|
|
// SPDX-License-Identifier: GPL-3.0-only.
|
2024-10-25 14:15:12 -04:00
|
|
|
|
2022-05-01 00:33:22 -05:00
|
|
|
using System;
|
2024-04-19 21:21:06 +02:00
|
|
|
using System.Collections.Concurrent;
|
2022-05-01 00:33:22 -05:00
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Diagnostics;
|
|
|
|
using System.Globalization;
|
2022-05-05 18:10:34 -05:00
|
|
|
using System.IO;
|
|
|
|
using System.Text;
|
2022-08-29 23:56:13 -05:00
|
|
|
using System.Text.RegularExpressions;
|
2024-10-19 23:50:41 +02:00
|
|
|
using IntroSkipper.Data;
|
2022-05-01 00:33:22 -05:00
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
|
2024-10-19 23:50:41 +02:00
|
|
|
namespace IntroSkipper;
|
2022-05-01 00:33:22 -05:00
|
|
|
|
|
|
|
/// <summary>
|
2022-08-28 22:35:43 -05:00
|
|
|
/// Wrapper for libchromaprint and the silencedetect filter.
|
2022-05-01 00:33:22 -05:00
|
|
|
/// </summary>
|
2024-09-10 18:08:42 +02:00
|
|
|
public static partial class FFmpegWrapper
|
2022-05-09 22:50:41 -05:00
|
|
|
{
|
2022-08-29 23:56:13 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Used with FFmpeg's silencedetect filter to extract the start and end times of silence.
|
|
|
|
/// </summary>
|
2024-09-10 18:08:42 +02:00
|
|
|
private static readonly Regex _silenceDetectionExpression = SilenceRegex();
|
2022-08-29 23:56:13 -05:00
|
|
|
|
2022-11-01 00:53:56 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Used with FFmpeg's blackframe filter to extract the time and percentage of black pixels.
|
|
|
|
/// </summary>
|
2024-09-10 18:08:42 +02:00
|
|
|
private static readonly Regex _blackFrameRegex = BlackFrameRegex();
|
2022-11-01 00:53:56 -05:00
|
|
|
|
2022-05-01 00:33:22 -05:00
|
|
|
/// <summary>
|
2022-05-09 22:50:41 -05:00
|
|
|
/// Gets or sets the logger.
|
2022-05-01 00:33:22 -05:00
|
|
|
/// </summary>
|
|
|
|
public static ILogger? Logger { get; set; }
|
|
|
|
|
2024-09-10 18:08:42 +02:00
|
|
|
private static Dictionary<string, string> ChromaprintLogs { get; set; } = [];
|
2022-08-25 00:39:20 -05:00
|
|
|
|
2024-10-16 16:05:59 +02:00
|
|
|
private static ConcurrentDictionary<(Guid Id, AnalysisMode Mode), Dictionary<uint, int>> InvertedIndexCache { get; set; } = new();
|
2022-08-26 00:50:45 -05:00
|
|
|
|
2022-05-01 00:33:22 -05:00
|
|
|
/// <summary>
|
2022-06-09 14:07:40 -05:00
|
|
|
/// Check that the installed version of ffmpeg supports chromaprint.
|
2022-05-01 00:33:22 -05:00
|
|
|
/// </summary>
|
2022-06-09 14:07:40 -05:00
|
|
|
/// <returns>true if a compatible version of ffmpeg is installed, false on any error.</returns>
|
|
|
|
public static bool CheckFFmpegVersion()
|
2022-05-01 00:33:22 -05:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2022-09-01 03:28:19 -05:00
|
|
|
// Always log ffmpeg's version information.
|
|
|
|
if (!CheckFFmpegRequirement(
|
|
|
|
"-version",
|
|
|
|
"ffmpeg",
|
|
|
|
"version",
|
|
|
|
"Unknown error with FFmpeg version"))
|
2022-06-13 17:26:05 -05:00
|
|
|
{
|
2022-09-01 03:28:19 -05:00
|
|
|
ChromaprintLogs["error"] = "unknown_error";
|
2022-11-06 21:25:23 -06:00
|
|
|
WarningManager.SetFlag(PluginWarning.IncompatibleFFmpegBuild);
|
2022-06-13 17:26:05 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-09-01 03:28:19 -05:00
|
|
|
// First, validate that the installed version of ffmpeg supports chromaprint at all.
|
|
|
|
if (!CheckFFmpegRequirement(
|
|
|
|
"-muxers",
|
|
|
|
"chromaprint",
|
|
|
|
"muxer list",
|
|
|
|
"The installed version of ffmpeg does not support chromaprint"))
|
2022-06-13 17:26:05 -05:00
|
|
|
{
|
2022-09-01 03:28:19 -05:00
|
|
|
ChromaprintLogs["error"] = "chromaprint_not_supported";
|
2022-11-06 21:25:23 -06:00
|
|
|
WarningManager.SetFlag(PluginWarning.IncompatibleFFmpegBuild);
|
2022-06-13 17:26:05 -05:00
|
|
|
return false;
|
|
|
|
}
|
2022-09-01 03:28:19 -05:00
|
|
|
|
|
|
|
// Second, validate that the Chromaprint muxer understands the "-fp_format raw" option.
|
|
|
|
if (!CheckFFmpegRequirement(
|
|
|
|
"-h muxer=chromaprint",
|
|
|
|
"binary raw fingerprint",
|
|
|
|
"chromaprint options",
|
|
|
|
"The installed version of ffmpeg does not support raw binary fingerprints"))
|
2022-06-13 17:26:05 -05:00
|
|
|
{
|
2022-09-01 03:28:19 -05:00
|
|
|
ChromaprintLogs["error"] = "fp_format_not_supported";
|
2022-11-06 21:25:23 -06:00
|
|
|
WarningManager.SetFlag(PluginWarning.IncompatibleFFmpegBuild);
|
2022-06-13 17:26:05 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-09-01 00:18:51 -05:00
|
|
|
// Third, validate that ffmpeg supports of the all required silencedetect options.
|
2022-09-01 03:28:19 -05:00
|
|
|
if (!CheckFFmpegRequirement(
|
|
|
|
"-h filter=silencedetect",
|
|
|
|
"noise tolerance",
|
|
|
|
"silencedetect options",
|
|
|
|
"The installed version of ffmpeg does not support the silencedetect filter"))
|
2022-09-01 00:18:51 -05:00
|
|
|
{
|
2022-09-01 03:28:19 -05:00
|
|
|
ChromaprintLogs["error"] = "silencedetect_not_supported";
|
2022-11-06 21:25:23 -06:00
|
|
|
WarningManager.SetFlag(PluginWarning.IncompatibleFFmpegBuild);
|
2022-09-01 00:18:51 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-06-13 17:26:05 -05:00
|
|
|
Logger?.LogDebug("Installed version of ffmpeg meets fingerprinting requirements");
|
2022-08-25 00:39:20 -05:00
|
|
|
ChromaprintLogs["error"] = "okay";
|
2022-06-13 17:26:05 -05:00
|
|
|
return true;
|
2022-05-01 00:33:22 -05:00
|
|
|
}
|
|
|
|
catch
|
|
|
|
{
|
2022-08-25 00:39:20 -05:00
|
|
|
ChromaprintLogs["error"] = "unknown_error";
|
2022-11-06 21:25:23 -06:00
|
|
|
WarningManager.SetFlag(PluginWarning.IncompatibleFFmpegBuild);
|
2022-05-01 00:33:22 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Fingerprint a queued episode.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="episode">Queued episode to fingerprint.</param>
|
2022-10-31 01:00:39 -05:00
|
|
|
/// <param name="mode">Portion of media file to fingerprint. Introduction = first 25% / 10 minutes and Credits = last 4 minutes.</param>
|
2022-05-09 22:50:41 -05:00
|
|
|
/// <returns>Numerical fingerprint points.</returns>
|
2024-10-16 16:05:59 +02:00
|
|
|
public static uint[] Fingerprint(QueuedEpisode episode, AnalysisMode mode)
|
2022-05-01 00:33:22 -05:00
|
|
|
{
|
2022-10-31 01:00:39 -05:00
|
|
|
int start, end;
|
|
|
|
|
2024-10-16 16:05:59 +02:00
|
|
|
if (mode == AnalysisMode.Introduction)
|
2022-05-05 18:10:34 -05:00
|
|
|
{
|
2022-10-31 01:00:39 -05:00
|
|
|
start = 0;
|
|
|
|
end = episode.IntroFingerprintEnd;
|
2022-05-05 18:10:34 -05:00
|
|
|
}
|
2024-10-16 16:05:59 +02:00
|
|
|
else if (mode == AnalysisMode.Credits)
|
2022-05-01 00:33:22 -05:00
|
|
|
{
|
2022-10-31 01:00:39 -05:00
|
|
|
start = episode.CreditsFingerprintStart;
|
|
|
|
end = episode.Duration;
|
2022-05-01 00:33:22 -05:00
|
|
|
}
|
2022-10-31 01:00:39 -05:00
|
|
|
else
|
2022-05-01 00:33:22 -05:00
|
|
|
{
|
2024-04-20 12:58:29 +02:00
|
|
|
throw new ArgumentException("Unknown analysis mode " + mode);
|
2022-05-01 00:33:22 -05:00
|
|
|
}
|
|
|
|
|
2022-10-31 01:00:39 -05:00
|
|
|
return Fingerprint(episode, mode, start, end);
|
2022-05-01 00:33:22 -05:00
|
|
|
}
|
|
|
|
|
2022-07-05 16:16:48 -05:00
|
|
|
/// <summary>
|
2022-07-05 17:08:52 -05:00
|
|
|
/// Transforms a Chromaprint into an inverted index of fingerprint points to the last index it appeared at.
|
2022-07-05 16:16:48 -05:00
|
|
|
/// </summary>
|
2022-08-26 00:50:45 -05:00
|
|
|
/// <param name="id">Episode ID.</param>
|
2022-07-05 16:16:48 -05:00
|
|
|
/// <param name="fingerprint">Chromaprint fingerprint.</param>
|
2024-04-20 12:29:40 +02:00
|
|
|
/// <param name="mode">Mode.</param>
|
2022-07-05 16:16:48 -05:00
|
|
|
/// <returns>Inverted index.</returns>
|
2024-10-16 16:05:59 +02:00
|
|
|
public static Dictionary<uint, int> CreateInvertedIndex(Guid id, uint[] fingerprint, AnalysisMode mode)
|
2022-07-05 16:16:48 -05:00
|
|
|
{
|
2024-05-08 16:27:16 +02:00
|
|
|
if (InvertedIndexCache.TryGetValue((id, mode), out var cached))
|
2022-08-26 00:50:45 -05:00
|
|
|
{
|
2024-04-19 21:21:06 +02:00
|
|
|
return cached;
|
2022-08-26 00:50:45 -05:00
|
|
|
}
|
|
|
|
|
2022-07-05 17:08:52 -05:00
|
|
|
var invIndex = new Dictionary<uint, int>();
|
2022-07-05 16:16:48 -05:00
|
|
|
|
2022-07-09 00:24:58 -05:00
|
|
|
for (int i = 0; i < fingerprint.Length; i++)
|
2022-07-05 16:16:48 -05:00
|
|
|
{
|
|
|
|
// Get the current point.
|
|
|
|
var point = fingerprint[i];
|
|
|
|
|
|
|
|
// Append the current sample's timecode to the collection for this point.
|
2022-07-05 17:08:52 -05:00
|
|
|
invIndex[point] = i;
|
2022-07-05 16:16:48 -05:00
|
|
|
}
|
|
|
|
|
2024-05-08 16:27:16 +02:00
|
|
|
InvertedIndexCache[(id, mode)] = invIndex;
|
2022-08-26 00:50:45 -05:00
|
|
|
|
2022-07-05 16:16:48 -05:00
|
|
|
return invIndex;
|
|
|
|
}
|
|
|
|
|
2022-05-05 18:10:34 -05:00
|
|
|
/// <summary>
|
2022-08-29 23:56:13 -05:00
|
|
|
/// Detect ranges of silence in the provided episode.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="episode">Queued episode.</param>
|
2024-08-31 19:32:37 +02:00
|
|
|
/// <param name="range">Time range to search.</param>
|
2022-08-29 23:56:13 -05:00
|
|
|
/// <returns>Array of TimeRange objects that are silent in the queued episode.</returns>
|
2024-08-31 19:32:37 +02:00
|
|
|
public static TimeRange[] DetectSilence(QueuedEpisode episode, TimeRange range)
|
2022-08-29 23:56:13 -05:00
|
|
|
{
|
|
|
|
Logger?.LogTrace(
|
2024-08-31 19:32:37 +02:00
|
|
|
"Detecting silence in \"{File}\" (range {Start}-{End}, id {Id})",
|
2022-08-29 23:56:13 -05:00
|
|
|
episode.Path,
|
2024-08-31 19:32:37 +02:00
|
|
|
range.Start,
|
|
|
|
range.End,
|
2022-08-29 23:56:13 -05:00
|
|
|
episode.EpisodeId);
|
|
|
|
|
|
|
|
// -vn, -sn, -dn: ignore video, subtitle, and data tracks
|
|
|
|
var args = string.Format(
|
|
|
|
CultureInfo.InvariantCulture,
|
|
|
|
"-vn -sn -dn " +
|
2024-08-31 19:32:37 +02:00
|
|
|
"-ss {0} -i \"{1}\" -to {2} -af \"silencedetect=noise={3}dB:duration=0.1\" -f null -",
|
|
|
|
range.Start,
|
2022-08-29 23:56:13 -05:00
|
|
|
episode.Path,
|
2024-08-31 19:32:37 +02:00
|
|
|
range.End - range.Start,
|
2022-08-29 23:56:13 -05:00
|
|
|
Plugin.Instance?.Configuration.SilenceDetectionMaximumNoise ?? -50);
|
|
|
|
|
2024-08-31 19:32:37 +02:00
|
|
|
// Cache the output of this command to "GUID-intro-silence-v2"
|
|
|
|
var cacheKey = string.Format(
|
|
|
|
CultureInfo.InvariantCulture,
|
|
|
|
"{0}-silence-{1}-{2}-v2",
|
|
|
|
episode.EpisodeId.ToString("N"),
|
|
|
|
range.Start,
|
|
|
|
range.End);
|
2022-08-29 23:56:13 -05:00
|
|
|
|
|
|
|
var currentRange = new TimeRange();
|
|
|
|
var silenceRanges = new List<TimeRange>();
|
|
|
|
|
2022-11-01 00:53:56 -05:00
|
|
|
/* Each match will have a type (either "start" or "end") and a timecode (a double).
|
|
|
|
*
|
|
|
|
* Sample output:
|
|
|
|
* [silencedetect @ 0x000000000000] silence_start: 12.34
|
|
|
|
* [silencedetect @ 0x000000000000] silence_end: 56.123 | silence_duration: 43.783
|
|
|
|
*/
|
2022-08-29 23:56:13 -05:00
|
|
|
var raw = Encoding.UTF8.GetString(GetOutput(args, cacheKey, true));
|
2024-09-10 18:08:42 +02:00
|
|
|
foreach (Match match in _silenceDetectionExpression.Matches(raw))
|
2022-08-29 23:56:13 -05:00
|
|
|
{
|
|
|
|
var isStart = match.Groups["type"].Value == "start";
|
|
|
|
var time = Convert.ToDouble(match.Groups["time"].Value, CultureInfo.InvariantCulture);
|
|
|
|
|
|
|
|
if (isStart)
|
|
|
|
{
|
2024-08-31 19:32:37 +02:00
|
|
|
currentRange.Start = time + range.Start;
|
2022-08-29 23:56:13 -05:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-08-31 19:32:37 +02:00
|
|
|
currentRange.End = time + range.Start;
|
2022-08-29 23:56:13 -05:00
|
|
|
silenceRanges.Add(new TimeRange(currentRange));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return silenceRanges.ToArray();
|
|
|
|
}
|
|
|
|
|
2022-11-01 00:53:56 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Finds the location of all black frames in a media file within a time range.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="episode">Media file to analyze.</param>
|
|
|
|
/// <param name="range">Time range to search.</param>
|
2022-11-02 17:06:21 -05:00
|
|
|
/// <param name="minimum">Percentage of the frame that must be black.</param>
|
|
|
|
/// <returns>Array of frames that are mostly black.</returns>
|
|
|
|
public static BlackFrame[] DetectBlackFrames(
|
|
|
|
QueuedEpisode episode,
|
|
|
|
TimeRange range,
|
|
|
|
int minimum)
|
2022-11-01 00:53:56 -05:00
|
|
|
{
|
|
|
|
// Seek to the start of the time range and find frames that are at least 50% black.
|
|
|
|
var args = string.Format(
|
|
|
|
CultureInfo.InvariantCulture,
|
|
|
|
"-ss {0} -i \"{1}\" -to {2} -an -dn -sn -vf \"blackframe=amount=50\" -f null -",
|
|
|
|
range.Start,
|
|
|
|
episode.Path,
|
|
|
|
range.End - range.Start);
|
|
|
|
|
2022-11-02 17:06:21 -05:00
|
|
|
// Cache the results to GUID-blackframes-START-END-v1.
|
2022-11-01 00:53:56 -05:00
|
|
|
var cacheKey = string.Format(
|
|
|
|
CultureInfo.InvariantCulture,
|
2022-11-02 17:06:21 -05:00
|
|
|
"{0}-blackframes-{1}-{2}-v1",
|
2022-11-01 00:53:56 -05:00
|
|
|
episode.EpisodeId.ToString("N"),
|
|
|
|
range.Start,
|
|
|
|
range.End);
|
|
|
|
|
|
|
|
var blackFrames = new List<BlackFrame>();
|
|
|
|
|
|
|
|
/* Run the blackframe filter.
|
|
|
|
*
|
|
|
|
* Sample output:
|
|
|
|
* [Parsed_blackframe_0 @ 0x0000000] frame:1 pblack:99 pts:43 t:0.043000 type:B last_keyframe:0
|
|
|
|
* [Parsed_blackframe_0 @ 0x0000000] frame:2 pblack:99 pts:85 t:0.085000 type:B last_keyframe:0
|
|
|
|
*/
|
|
|
|
var raw = Encoding.UTF8.GetString(GetOutput(args, cacheKey, true));
|
|
|
|
foreach (var line in raw.Split('\n'))
|
|
|
|
{
|
2024-04-15 22:14:00 +02:00
|
|
|
// There is no FFmpeg flag to hide metadata such as description
|
|
|
|
// In our case, the metadata contained something that matched the regex.
|
|
|
|
if (line.StartsWith("[Parsed_blackframe_", StringComparison.OrdinalIgnoreCase))
|
2022-11-01 00:53:56 -05:00
|
|
|
{
|
2024-09-10 18:08:42 +02:00
|
|
|
var matches = _blackFrameRegex.Matches(line);
|
2024-04-15 22:14:00 +02:00
|
|
|
if (matches.Count != 2)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
var (strPercent, strTime) = (
|
|
|
|
matches[0].Value.Split(':')[1],
|
|
|
|
matches[1].Value.Split(':')[1]
|
|
|
|
);
|
|
|
|
|
|
|
|
var bf = new BlackFrame(
|
|
|
|
Convert.ToInt32(strPercent, CultureInfo.InvariantCulture),
|
|
|
|
Convert.ToDouble(strTime, CultureInfo.InvariantCulture));
|
|
|
|
|
|
|
|
if (bf.Percentage > minimum)
|
|
|
|
{
|
|
|
|
blackFrames.Add(bf);
|
|
|
|
}
|
2022-11-02 17:06:21 -05:00
|
|
|
}
|
2022-11-01 00:53:56 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return blackFrames.ToArray();
|
|
|
|
}
|
|
|
|
|
2022-10-31 01:47:41 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Gets Chromaprint debugging logs.
|
|
|
|
/// </summary>
|
|
|
|
/// <returns>Markdown formatted logs.</returns>
|
|
|
|
public static string GetChromaprintLogs()
|
|
|
|
{
|
|
|
|
// Print the FFmpeg detection status at the top.
|
|
|
|
// Format: "* FFmpeg: `error`"
|
|
|
|
// Append two newlines to separate the bulleted list from the logs
|
|
|
|
var logs = string.Format(
|
|
|
|
CultureInfo.InvariantCulture,
|
|
|
|
"* FFmpeg: `{0}`\n\n",
|
|
|
|
ChromaprintLogs["error"]);
|
|
|
|
|
|
|
|
// Always include ffmpeg version information
|
|
|
|
logs += FormatFFmpegLog("version");
|
|
|
|
|
|
|
|
// Don't print feature detection logs if the plugin started up okay
|
|
|
|
if (ChromaprintLogs["error"] == "okay")
|
|
|
|
{
|
|
|
|
return logs;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Print all remaining logs
|
|
|
|
foreach (var kvp in ChromaprintLogs)
|
|
|
|
{
|
|
|
|
if (kvp.Key == "error" || kvp.Key == "version")
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
logs += FormatFFmpegLog(kvp.Key);
|
|
|
|
}
|
|
|
|
|
|
|
|
return logs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Run an FFmpeg command with the provided arguments and validate that the output contains
|
|
|
|
/// the provided string.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="arguments">Arguments to pass to FFmpeg.</param>
|
|
|
|
/// <param name="mustContain">String that the output must contain. Case insensitive.</param>
|
|
|
|
/// <param name="bundleName">Support bundle key to store FFmpeg's output under.</param>
|
|
|
|
/// <param name="errorMessage">Error message to log if this requirement is not met.</param>
|
|
|
|
/// <returns>true on success, false on error.</returns>
|
|
|
|
private static bool CheckFFmpegRequirement(
|
|
|
|
string arguments,
|
|
|
|
string mustContain,
|
|
|
|
string bundleName,
|
|
|
|
string errorMessage)
|
|
|
|
{
|
|
|
|
Logger?.LogDebug("Checking FFmpeg requirement {Arguments}", arguments);
|
|
|
|
|
|
|
|
var output = Encoding.UTF8.GetString(GetOutput(arguments, string.Empty, false, 2000));
|
|
|
|
Logger?.LogTrace("Output of ffmpeg {Arguments}: {Output}", arguments, output);
|
|
|
|
ChromaprintLogs[bundleName] = output;
|
|
|
|
|
|
|
|
if (!output.Contains(mustContain, StringComparison.OrdinalIgnoreCase))
|
|
|
|
{
|
|
|
|
Logger?.LogError("{ErrorMessage}", errorMessage);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Logger?.LogDebug("FFmpeg requirement {Arguments} met", arguments);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-08-29 23:56:13 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Runs ffmpeg and returns standard output (or error).
|
|
|
|
/// If caching is enabled, will use cacheFilename to cache the output of this command.
|
2022-05-05 18:10:34 -05:00
|
|
|
/// </summary>
|
2022-06-09 14:07:40 -05:00
|
|
|
/// <param name="args">Arguments to pass to ffmpeg.</param>
|
2022-08-29 23:56:13 -05:00
|
|
|
/// <param name="cacheFilename">Filename to cache the output of this command to, or string.Empty if this command should not be cached.</param>
|
|
|
|
/// <param name="stderr">If standard error should be returned.</param>
|
|
|
|
/// <param name="timeout">Timeout (in miliseconds) to wait for ffmpeg to exit.</param>
|
|
|
|
private static ReadOnlySpan<byte> GetOutput(
|
|
|
|
string args,
|
|
|
|
string cacheFilename,
|
|
|
|
bool stderr = false,
|
|
|
|
int timeout = 60 * 1000)
|
2022-05-01 00:33:22 -05:00
|
|
|
{
|
2022-06-09 14:07:40 -05:00
|
|
|
var ffmpegPath = Plugin.Instance?.FFmpegPath ?? "ffmpeg";
|
2022-05-01 00:33:22 -05:00
|
|
|
|
2022-11-01 00:53:56 -05:00
|
|
|
// The silencedetect and blackframe filters output data at the info log level.
|
|
|
|
var useInfoLevel = args.Contains("silencedetect", StringComparison.OrdinalIgnoreCase) ||
|
|
|
|
args.Contains("blackframe", StringComparison.OrdinalIgnoreCase);
|
|
|
|
|
|
|
|
var logLevel = useInfoLevel ? "info" : "warning";
|
2022-09-01 03:54:29 -05:00
|
|
|
|
2022-08-29 23:56:13 -05:00
|
|
|
var cacheOutput =
|
|
|
|
(Plugin.Instance?.Configuration.CacheFingerprints ?? false) &&
|
|
|
|
!string.IsNullOrEmpty(cacheFilename);
|
|
|
|
|
|
|
|
// If caching is enabled, try to load the output of this command from the cached file.
|
|
|
|
if (cacheOutput)
|
|
|
|
{
|
|
|
|
// Calculate the absolute path to the cached file.
|
|
|
|
cacheFilename = Path.Join(Plugin.Instance!.FingerprintCachePath, cacheFilename);
|
|
|
|
|
|
|
|
// If the cached file exists, return whatever it holds.
|
|
|
|
if (File.Exists(cacheFilename))
|
|
|
|
{
|
|
|
|
Logger?.LogTrace("Returning contents of cache {Cache}", cacheFilename);
|
|
|
|
return File.ReadAllBytes(cacheFilename);
|
|
|
|
}
|
|
|
|
|
|
|
|
Logger?.LogTrace("Not returning contents of cache {Cache} (not found)", cacheFilename);
|
|
|
|
}
|
|
|
|
|
2022-06-16 14:59:25 -05:00
|
|
|
// Prepend some flags to prevent FFmpeg from logging it's banner and progress information
|
|
|
|
// for each file that is fingerprinted.
|
2022-09-01 03:54:29 -05:00
|
|
|
var prependArgument = string.Format(
|
|
|
|
CultureInfo.InvariantCulture,
|
2024-03-02 22:19:51 -05:00
|
|
|
"-hide_banner -loglevel {0} -threads {1} ",
|
|
|
|
logLevel,
|
|
|
|
Plugin.Instance?.Configuration.ProcessThreads ?? 0);
|
2022-09-01 03:54:29 -05:00
|
|
|
|
|
|
|
var info = new ProcessStartInfo(ffmpegPath, args.Insert(0, prependArgument))
|
2022-06-09 14:07:40 -05:00
|
|
|
{
|
2022-06-16 03:45:32 +08:00
|
|
|
WindowStyle = ProcessWindowStyle.Hidden,
|
2022-06-09 14:07:40 -05:00
|
|
|
CreateNoWindow = true,
|
2022-06-16 03:45:32 +08:00
|
|
|
UseShellExecute = false,
|
|
|
|
ErrorDialog = false,
|
|
|
|
|
2022-08-29 23:56:13 -05:00
|
|
|
RedirectStandardOutput = !stderr,
|
|
|
|
RedirectStandardError = stderr
|
2022-06-09 14:07:40 -05:00
|
|
|
};
|
2022-05-01 00:33:22 -05:00
|
|
|
|
2024-09-10 18:08:42 +02:00
|
|
|
using var ffmpeg = new Process { StartInfo = info };
|
|
|
|
Logger?.LogDebug("Starting ffmpeg with the following arguments: {Arguments}", ffmpeg.StartInfo.Arguments);
|
|
|
|
|
|
|
|
ffmpeg.Start();
|
|
|
|
|
|
|
|
try
|
2022-06-09 14:07:40 -05:00
|
|
|
{
|
2024-09-10 18:08:42 +02:00
|
|
|
ffmpeg.PriorityClass = Plugin.Instance?.Configuration.ProcessPriority ?? ProcessPriorityClass.BelowNormal;
|
|
|
|
}
|
|
|
|
catch (Exception e)
|
|
|
|
{
|
|
|
|
Logger?.LogDebug("ffmpeg priority could not be modified. {Message}", e.Message);
|
|
|
|
}
|
2022-05-01 00:33:22 -05:00
|
|
|
|
2024-09-10 18:08:42 +02:00
|
|
|
using var ms = new MemoryStream();
|
|
|
|
var buf = new byte[4096];
|
|
|
|
int bytesRead;
|
2022-08-29 23:56:13 -05:00
|
|
|
|
2024-09-10 18:08:42 +02:00
|
|
|
using (var streamReader = stderr ? ffmpeg.StandardError : ffmpeg.StandardOutput)
|
|
|
|
{
|
|
|
|
while ((bytesRead = streamReader.BaseStream.Read(buf, 0, buf.Length)) > 0)
|
2024-06-14 17:11:04 +02:00
|
|
|
{
|
2024-09-10 18:08:42 +02:00
|
|
|
ms.Write(buf, 0, bytesRead);
|
2024-06-14 17:11:04 +02:00
|
|
|
}
|
2024-09-10 18:08:42 +02:00
|
|
|
}
|
2022-05-01 00:33:22 -05:00
|
|
|
|
2024-09-10 18:08:42 +02:00
|
|
|
ffmpeg.WaitForExit(timeout);
|
2022-06-16 03:45:32 +08:00
|
|
|
|
2024-09-10 18:08:42 +02:00
|
|
|
var output = ms.ToArray();
|
2022-08-29 23:56:13 -05:00
|
|
|
|
2024-09-10 18:08:42 +02:00
|
|
|
// If caching is enabled, cache the output of this command.
|
|
|
|
if (cacheOutput)
|
|
|
|
{
|
|
|
|
File.WriteAllBytes(cacheFilename, output);
|
2022-06-09 14:07:40 -05:00
|
|
|
}
|
2024-09-10 18:08:42 +02:00
|
|
|
|
|
|
|
return output;
|
2022-05-01 00:33:22 -05:00
|
|
|
}
|
2022-05-05 18:10:34 -05:00
|
|
|
|
2022-10-31 01:47:41 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Fingerprint a queued episode.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="episode">Queued episode to fingerprint.</param>
|
|
|
|
/// <param name="mode">Portion of media file to fingerprint.</param>
|
|
|
|
/// <param name="start">Time (in seconds) relative to the start of the file to start fingerprinting from.</param>
|
|
|
|
/// <param name="end">Time (in seconds) relative to the start of the file to stop fingerprinting at.</param>
|
|
|
|
/// <returns>Numerical fingerprint points.</returns>
|
2024-10-16 16:05:59 +02:00
|
|
|
private static uint[] Fingerprint(QueuedEpisode episode, AnalysisMode mode, int start, int end)
|
2022-10-31 01:47:41 -05:00
|
|
|
{
|
|
|
|
// Try to load this episode from cache before running ffmpeg.
|
|
|
|
if (LoadCachedFingerprint(episode, mode, out uint[] cachedFingerprint))
|
|
|
|
{
|
|
|
|
Logger?.LogTrace("Fingerprint cache hit on {File}", episode.Path);
|
|
|
|
return cachedFingerprint;
|
|
|
|
}
|
|
|
|
|
|
|
|
Logger?.LogDebug(
|
|
|
|
"Fingerprinting [{Start}, {End}] from \"{File}\" (id {Id})",
|
|
|
|
start,
|
|
|
|
end,
|
|
|
|
episode.Path,
|
|
|
|
episode.EpisodeId);
|
|
|
|
|
|
|
|
var args = string.Format(
|
|
|
|
CultureInfo.InvariantCulture,
|
|
|
|
"-ss {0} -i \"{1}\" -to {2} -ac 2 -f chromaprint -fp_format raw -",
|
|
|
|
start,
|
|
|
|
episode.Path,
|
|
|
|
end - start);
|
|
|
|
|
|
|
|
// Returns all fingerprint points as raw 32 bit unsigned integers (little endian).
|
|
|
|
var rawPoints = GetOutput(args, string.Empty);
|
|
|
|
if (rawPoints.Length == 0 || rawPoints.Length % 4 != 0)
|
|
|
|
{
|
|
|
|
Logger?.LogWarning("Chromaprint returned {Count} points for \"{Path}\"", rawPoints.Length, episode.Path);
|
|
|
|
throw new FingerprintException("chromaprint output for \"" + episode.Path + "\" was malformed");
|
|
|
|
}
|
|
|
|
|
|
|
|
var results = new List<uint>();
|
|
|
|
for (var i = 0; i < rawPoints.Length; i += 4)
|
|
|
|
{
|
|
|
|
var rawPoint = rawPoints.Slice(i, 4);
|
|
|
|
results.Add(BitConverter.ToUInt32(rawPoint));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to cache this fingerprint.
|
|
|
|
CacheFingerprint(episode, mode, results);
|
|
|
|
|
|
|
|
return results.ToArray();
|
|
|
|
}
|
|
|
|
|
2022-05-05 18:10:34 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Tries to load an episode's fingerprint from cache. If caching is not enabled, calling this function is a no-op.
|
2022-08-29 23:56:13 -05:00
|
|
|
/// This function was created before the unified caching mechanism was introduced (in v0.1.7).
|
2022-05-05 18:10:34 -05:00
|
|
|
/// </summary>
|
|
|
|
/// <param name="episode">Episode to try to load from cache.</param>
|
2022-10-31 01:00:39 -05:00
|
|
|
/// <param name="mode">Analysis mode.</param>
|
2022-07-09 00:24:58 -05:00
|
|
|
/// <param name="fingerprint">Array to store the fingerprint in.</param>
|
2022-05-05 18:10:34 -05:00
|
|
|
/// <returns>true if the episode was successfully loaded from cache, false on any other error.</returns>
|
2022-10-31 01:00:39 -05:00
|
|
|
private static bool LoadCachedFingerprint(
|
|
|
|
QueuedEpisode episode,
|
2024-10-16 16:05:59 +02:00
|
|
|
AnalysisMode mode,
|
2022-10-31 01:00:39 -05:00
|
|
|
out uint[] fingerprint)
|
2022-05-05 18:10:34 -05:00
|
|
|
{
|
2022-07-09 00:24:58 -05:00
|
|
|
fingerprint = Array.Empty<uint>();
|
2022-05-05 18:10:34 -05:00
|
|
|
|
|
|
|
// If fingerprint caching isn't enabled, don't try to load anything.
|
2022-05-09 03:31:10 -05:00
|
|
|
if (!(Plugin.Instance?.Configuration.CacheFingerprints ?? false))
|
2022-05-05 18:10:34 -05:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-10-31 01:00:39 -05:00
|
|
|
var path = GetFingerprintCachePath(episode, mode);
|
2022-05-05 18:10:34 -05:00
|
|
|
|
|
|
|
// If this episode isn't cached, bail out.
|
|
|
|
if (!File.Exists(path))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
var raw = File.ReadAllLines(path, Encoding.UTF8);
|
|
|
|
var result = new List<uint>();
|
|
|
|
|
|
|
|
// Read each stringified uint.
|
|
|
|
result.EnsureCapacity(raw.Length);
|
2022-06-20 01:07:03 -05:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
foreach (var rawNumber in raw)
|
|
|
|
{
|
|
|
|
result.Add(Convert.ToUInt32(rawNumber, CultureInfo.InvariantCulture));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (FormatException)
|
2022-05-05 18:10:34 -05:00
|
|
|
{
|
2022-06-20 01:07:03 -05:00
|
|
|
// Occurs when the cached fingerprint is corrupt.
|
|
|
|
Logger?.LogDebug(
|
|
|
|
"Cached fingerprint for {Path} ({Id}) is corrupt, ignoring cache",
|
|
|
|
episode.Path,
|
|
|
|
episode.EpisodeId);
|
|
|
|
|
|
|
|
return false;
|
2022-05-05 18:10:34 -05:00
|
|
|
}
|
|
|
|
|
2022-07-09 00:24:58 -05:00
|
|
|
fingerprint = result.ToArray();
|
2022-05-05 18:10:34 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Cache an episode's fingerprint to disk. If caching is not enabled, calling this function is a no-op.
|
2022-08-29 23:56:13 -05:00
|
|
|
/// This function was created before the unified caching mechanism was introduced (in v0.1.7).
|
2022-05-05 18:10:34 -05:00
|
|
|
/// </summary>
|
|
|
|
/// <param name="episode">Episode to store in cache.</param>
|
2022-10-31 01:00:39 -05:00
|
|
|
/// <param name="mode">Analysis mode.</param>
|
2022-05-05 18:10:34 -05:00
|
|
|
/// <param name="fingerprint">Fingerprint of the episode to store.</param>
|
2022-10-31 01:00:39 -05:00
|
|
|
private static void CacheFingerprint(
|
|
|
|
QueuedEpisode episode,
|
2024-10-16 16:05:59 +02:00
|
|
|
AnalysisMode mode,
|
2022-10-31 01:00:39 -05:00
|
|
|
List<uint> fingerprint)
|
2022-05-05 18:10:34 -05:00
|
|
|
{
|
|
|
|
// Bail out if caching isn't enabled.
|
2022-05-09 03:31:10 -05:00
|
|
|
if (!(Plugin.Instance?.Configuration.CacheFingerprints ?? false))
|
2022-05-05 18:10:34 -05:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stringify each data point.
|
|
|
|
var lines = new List<string>();
|
|
|
|
foreach (var number in fingerprint)
|
|
|
|
{
|
|
|
|
lines.Add(number.ToString(CultureInfo.InvariantCulture));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cache the episode.
|
2022-10-31 01:00:39 -05:00
|
|
|
File.WriteAllLinesAsync(
|
|
|
|
GetFingerprintCachePath(episode, mode),
|
|
|
|
lines,
|
|
|
|
Encoding.UTF8).ConfigureAwait(false);
|
2022-05-05 18:10:34 -05:00
|
|
|
}
|
|
|
|
|
2024-03-05 17:58:32 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Remove a cached episode fingerprint from disk.
|
|
|
|
/// </summary>
|
2024-06-07 17:09:48 +02:00
|
|
|
/// <param name="id">Episode to remove from cache.</param>
|
|
|
|
public static void DeleteEpisodeCache(Guid id)
|
2024-03-05 17:58:32 -05:00
|
|
|
{
|
|
|
|
var cachePath = Path.Join(
|
|
|
|
Plugin.Instance!.FingerprintCachePath,
|
2024-06-07 17:09:48 +02:00
|
|
|
id.ToString("N"));
|
|
|
|
|
|
|
|
// File.Delete(cachePath);
|
|
|
|
// File.Delete(cachePath + "-intro-silence-v1");
|
|
|
|
// File.Delete(cachePath + "-credits");
|
2024-03-05 17:58:32 -05:00
|
|
|
|
2024-06-07 17:09:48 +02:00
|
|
|
var filePattern = Path.GetFileName(cachePath) + "*";
|
|
|
|
foreach (var filePath in Directory.EnumerateFiles(Plugin.Instance!.FingerprintCachePath, filePattern))
|
2024-03-05 17:58:32 -05:00
|
|
|
{
|
2024-09-21 18:06:11 +02:00
|
|
|
Logger?.LogDebug("DeleteEpisodeCache {FilePath}", filePath);
|
2024-06-07 17:09:48 +02:00
|
|
|
File.Delete(filePath);
|
2024-03-05 17:58:32 -05:00
|
|
|
}
|
2024-06-07 17:09:48 +02:00
|
|
|
}
|
2024-03-05 17:58:32 -05:00
|
|
|
|
2024-06-07 17:09:48 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Remove cached fingerprints from disk by mode.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="mode">Analysis mode.</param>
|
2024-10-16 16:05:59 +02:00
|
|
|
public static void DeleteCacheFiles(AnalysisMode mode)
|
2024-06-07 17:09:48 +02:00
|
|
|
{
|
|
|
|
foreach (var filePath in Directory.EnumerateFiles(Plugin.Instance!.FingerprintCachePath))
|
|
|
|
{
|
2024-10-16 16:05:59 +02:00
|
|
|
var shouldDelete = (mode == AnalysisMode.Introduction)
|
2024-06-07 17:09:48 +02:00
|
|
|
? !filePath.Contains("credit", StringComparison.OrdinalIgnoreCase)
|
|
|
|
&& !filePath.Contains("blackframes", StringComparison.OrdinalIgnoreCase)
|
|
|
|
: filePath.Contains("credit", StringComparison.OrdinalIgnoreCase)
|
|
|
|
|| filePath.Contains("blackframes", StringComparison.OrdinalIgnoreCase);
|
|
|
|
|
|
|
|
if (shouldDelete)
|
|
|
|
{
|
|
|
|
File.Delete(filePath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-05 18:10:34 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Determines the path an episode should be cached at.
|
2022-08-29 23:56:13 -05:00
|
|
|
/// This function was created before the unified caching mechanism was introduced (in v0.1.7).
|
2022-05-05 18:10:34 -05:00
|
|
|
/// </summary>
|
|
|
|
/// <param name="episode">Episode.</param>
|
2022-10-31 01:00:39 -05:00
|
|
|
/// <param name="mode">Analysis mode.</param>
|
2024-06-15 13:16:47 +02:00
|
|
|
/// <returns>Path.</returns>
|
2024-10-16 16:05:59 +02:00
|
|
|
public static string GetFingerprintCachePath(QueuedEpisode episode, AnalysisMode mode)
|
2022-05-05 18:10:34 -05:00
|
|
|
{
|
2022-10-31 01:00:39 -05:00
|
|
|
var basePath = Path.Join(
|
|
|
|
Plugin.Instance!.FingerprintCachePath,
|
|
|
|
episode.EpisodeId.ToString("N"));
|
2022-08-25 00:39:20 -05:00
|
|
|
|
2024-10-16 16:05:59 +02:00
|
|
|
if (mode == AnalysisMode.Introduction)
|
2022-09-01 03:28:19 -05:00
|
|
|
{
|
2022-10-31 01:00:39 -05:00
|
|
|
return basePath;
|
2022-09-01 03:28:19 -05:00
|
|
|
}
|
2024-04-20 12:58:29 +02:00
|
|
|
|
2024-10-16 16:05:59 +02:00
|
|
|
if (mode == AnalysisMode.Credits)
|
2022-08-25 00:39:20 -05:00
|
|
|
{
|
2022-10-31 01:00:39 -05:00
|
|
|
return basePath + "-credits";
|
|
|
|
}
|
2024-04-20 12:58:29 +02:00
|
|
|
|
|
|
|
throw new ArgumentException("Unknown analysis mode " + mode);
|
2022-10-06 22:40:05 -05:00
|
|
|
}
|
2022-08-25 00:39:20 -05:00
|
|
|
|
2022-10-06 22:40:05 -05:00
|
|
|
private static string FormatFFmpegLog(string key)
|
|
|
|
{
|
|
|
|
/* Format:
|
|
|
|
* FFmpeg NAME:
|
|
|
|
* ```
|
|
|
|
* LOGS
|
|
|
|
* ```
|
|
|
|
*/
|
|
|
|
|
|
|
|
var formatted = string.Format(CultureInfo.InvariantCulture, "FFmpeg {0}:\n```\n", key);
|
|
|
|
formatted += ChromaprintLogs[key];
|
|
|
|
|
|
|
|
// Ensure the closing triple backtick is on a separate line
|
|
|
|
if (!formatted.EndsWith('\n'))
|
|
|
|
{
|
|
|
|
formatted += "\n";
|
2022-08-25 00:39:20 -05:00
|
|
|
}
|
|
|
|
|
2022-10-06 22:40:05 -05:00
|
|
|
formatted += "```\n\n";
|
|
|
|
|
|
|
|
return formatted;
|
2022-08-25 00:39:20 -05:00
|
|
|
}
|
2024-09-10 18:08:42 +02:00
|
|
|
|
|
|
|
[GeneratedRegex("silence_(?<type>start|end): (?<time>[0-9\\.]+)")]
|
|
|
|
private static partial Regex SilenceRegex();
|
|
|
|
|
|
|
|
[GeneratedRegex("(pblack|t):[0-9.]+")]
|
|
|
|
private static partial Regex BlackFrameRegex();
|
2022-05-01 00:33:22 -05:00
|
|
|
}
|