| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131 |
- using System;
- using System.Collections.Generic;
- using System.Diagnostics;
- using System.Globalization;
- using System.IO;
- namespace Jellyfin.MediaEncoding.Keyframes.FfProbe;
- /// <summary>
- /// FfProbe based keyframe extractor.
- /// </summary>
- public static class FfProbeKeyframeExtractor
- {
- /// <summary>
- /// Extracts the keyframes using the ffprobe executable at the specified path.
- /// </summary>
- /// <param name="ffProbePath">The path to the ffprobe executable.</param>
- /// <param name="filePath">The file path.</param>
- /// <returns>An instance of <see cref="KeyframeData"/>.</returns>
- public static KeyframeData GetKeyframeData(string ffProbePath, string filePath)
- {
- using var process = new Process
- {
- StartInfo = new ProcessStartInfo
- {
- FileName = ffProbePath,
- Arguments = string.Format(
- CultureInfo.InvariantCulture,
- "-fflags +genpts -v error -skip_frame nokey -show_entries format=duration -show_entries stream=duration -show_entries packet=pts_time,flags -select_streams v -of csv \"{0}\"",
- filePath),
- CreateNoWindow = true,
- UseShellExecute = false,
- RedirectStandardOutput = true,
- WindowStyle = ProcessWindowStyle.Hidden,
- ErrorDialog = false,
- },
- EnableRaisingEvents = true
- };
- try
- {
- process.Start();
- try
- {
- process.PriorityClass = ProcessPriorityClass.BelowNormal;
- }
- catch
- {
- // We do not care if process priority setting fails
- // Ideally log a warning but this does not have a logger available
- }
- return ParseStream(process.StandardOutput);
- }
- catch (Exception)
- {
- try
- {
- if (!process.HasExited)
- {
- process.Kill();
- }
- }
- catch
- {
- // We do not care if this fails
- }
- throw;
- }
- }
- internal static KeyframeData ParseStream(StreamReader reader)
- {
- var keyframes = new List<long>();
- double streamDuration = 0;
- double formatDuration = 0;
- using (reader)
- {
- while (!reader.EndOfStream)
- {
- var line = reader.ReadLine().AsSpan();
- if (line.IsEmpty)
- {
- continue;
- }
- var firstComma = line.IndexOf(',');
- var lineType = line[..firstComma];
- var rest = line[(firstComma + 1)..];
- if (lineType.Equals("packet", StringComparison.OrdinalIgnoreCase))
- {
- // Split time and flags from the packet line. Example line: packet,7169.079000,K_
- var secondComma = rest.IndexOf(',');
- var ptsTime = rest[..secondComma];
- var flags = rest[(secondComma + 1)..];
- if (flags.StartsWith("K_"))
- {
- if (double.TryParse(ptsTime, NumberStyles.AllowDecimalPoint, CultureInfo.InvariantCulture, out var keyframe))
- {
- // Have to manually convert to ticks to avoid rounding errors as TimeSpan is only precise down to 1 ms when converting double.
- keyframes.Add(Convert.ToInt64(keyframe * TimeSpan.TicksPerSecond));
- }
- }
- }
- else if (lineType.Equals("stream", StringComparison.OrdinalIgnoreCase))
- {
- if (double.TryParse(rest, NumberStyles.AllowDecimalPoint, CultureInfo.InvariantCulture, out var streamDurationResult))
- {
- streamDuration = streamDurationResult;
- }
- }
- else if (lineType.Equals("format", StringComparison.OrdinalIgnoreCase))
- {
- if (double.TryParse(rest, NumberStyles.AllowDecimalPoint, CultureInfo.InvariantCulture, out var formatDurationResult))
- {
- formatDuration = formatDurationResult;
- }
- }
- }
- // Prefer the stream duration as it should be more accurate
- var duration = streamDuration > 0 ? streamDuration : formatDuration;
- return new KeyframeData(TimeSpan.FromSeconds(duration).Ticks, keyframes);
- }
- }
- }
|