< Summary - Jellyfin

Information
Class: MediaBrowser.MediaEncoding.Encoder.EncoderValidator
Assembly: MediaBrowser.MediaEncoding
File(s): /srv/git/jellyfin/MediaBrowser.MediaEncoding/Encoder/EncoderValidator.cs
Line coverage
53%
Covered lines: 192
Uncovered lines: 168
Coverable lines: 360
Total lines: 662
Line coverage: 53.3%
Branch coverage
28%
Covered branches: 23
Total branches: 80
Branch coverage: 28.7%
Method coverage

Feature is only available for sponsors

Upgrade to PRO version

Coverage history

Coverage history 0 25 50 75 100

Metrics

File(s)

/srv/git/jellyfin/MediaBrowser.MediaEncoding/Encoder/EncoderValidator.cs

#LineLine coverage
 1#pragma warning disable CS1591
 2
 3using System;
 4using System.Collections.Generic;
 5using System.Diagnostics;
 6using System.Globalization;
 7using System.Linq;
 8using System.Runtime.Versioning;
 9using System.Text.RegularExpressions;
 10using Microsoft.Extensions.Logging;
 11
 12namespace MediaBrowser.MediaEncoding.Encoder
 13{
 14    public partial class EncoderValidator
 15    {
 116        private static readonly string[] _requiredDecoders = new[]
 117        {
 118            "h264",
 119            "hevc",
 120            "vp8",
 121            "libvpx",
 122            "vp9",
 123            "libvpx-vp9",
 124            "av1",
 125            "libdav1d",
 126            "mpeg2video",
 127            "mpeg4",
 128            "msmpeg4",
 129            "dca",
 130            "ac3",
 131            "ac4",
 132            "aac",
 133            "mp3",
 134            "flac",
 135            "truehd",
 136            "h264_qsv",
 137            "hevc_qsv",
 138            "mpeg2_qsv",
 139            "vc1_qsv",
 140            "vp8_qsv",
 141            "vp9_qsv",
 142            "av1_qsv",
 143            "h264_cuvid",
 144            "hevc_cuvid",
 145            "mpeg2_cuvid",
 146            "vc1_cuvid",
 147            "mpeg4_cuvid",
 148            "vp8_cuvid",
 149            "vp9_cuvid",
 150            "av1_cuvid",
 151            "h264_rkmpp",
 152            "hevc_rkmpp",
 153            "mpeg1_rkmpp",
 154            "mpeg2_rkmpp",
 155            "mpeg4_rkmpp",
 156            "vp8_rkmpp",
 157            "vp9_rkmpp",
 158            "av1_rkmpp"
 159        };
 60
 161        private static readonly string[] _requiredEncoders = new[]
 162        {
 163            "libx264",
 164            "libx265",
 165            "libsvtav1",
 166            "aac",
 167            "aac_at",
 168            "libfdk_aac",
 169            "ac3",
 170            "alac",
 171            "dca",
 172            "libmp3lame",
 173            "libopus",
 174            "libvorbis",
 175            "flac",
 176            "truehd",
 177            "srt",
 178            "h264_amf",
 179            "hevc_amf",
 180            "av1_amf",
 181            "h264_qsv",
 182            "hevc_qsv",
 183            "mjpeg_qsv",
 184            "av1_qsv",
 185            "h264_nvenc",
 186            "hevc_nvenc",
 187            "av1_nvenc",
 188            "h264_vaapi",
 189            "hevc_vaapi",
 190            "av1_vaapi",
 191            "mjpeg_vaapi",
 192            "h264_v4l2m2m",
 193            "h264_videotoolbox",
 194            "hevc_videotoolbox",
 195            "mjpeg_videotoolbox",
 196            "h264_rkmpp",
 197            "hevc_rkmpp",
 198            "mjpeg_rkmpp"
 199        };
 100
 1101        private static readonly string[] _requiredFilters = new[]
 1102        {
 1103            // sw
 1104            "alphasrc",
 1105            "zscale",
 1106            "tonemapx",
 1107            // qsv
 1108            "scale_qsv",
 1109            "vpp_qsv",
 1110            "deinterlace_qsv",
 1111            "overlay_qsv",
 1112            // cuda
 1113            "scale_cuda",
 1114            "yadif_cuda",
 1115            "bwdif_cuda",
 1116            "tonemap_cuda",
 1117            "overlay_cuda",
 1118            "transpose_cuda",
 1119            "hwupload_cuda",
 1120            // opencl
 1121            "scale_opencl",
 1122            "tonemap_opencl",
 1123            "overlay_opencl",
 1124            "transpose_opencl",
 1125            // vaapi
 1126            "scale_vaapi",
 1127            "deinterlace_vaapi",
 1128            "tonemap_vaapi",
 1129            "procamp_vaapi",
 1130            "overlay_vaapi",
 1131            "transpose_vaapi",
 1132            "hwupload_vaapi",
 1133            // vulkan
 1134            "libplacebo",
 1135            "scale_vulkan",
 1136            "overlay_vulkan",
 1137            "transpose_vulkan",
 1138            "flip_vulkan",
 1139            // videotoolbox
 1140            "yadif_videotoolbox",
 1141            "bwdif_videotoolbox",
 1142            "scale_vt",
 1143            "transpose_vt",
 1144            "overlay_videotoolbox",
 1145            "tonemap_videotoolbox",
 1146            // rkrga
 1147            "scale_rkrga",
 1148            "vpp_rkrga",
 1149            "overlay_rkrga"
 1150        };
 151
 1152        private static readonly Dictionary<int, string[]> _filterOptionsDict = new Dictionary<int, string[]>
 1153        {
 1154            { 0, new string[] { "scale_cuda", "format" } },
 1155            { 1, new string[] { "tonemap_cuda", "GPU accelerated HDR to SDR tonemapping" } },
 1156            { 2, new string[] { "tonemap_opencl", "bt2390" } },
 1157            { 3, new string[] { "overlay_opencl", "Action to take when encountering EOF from secondary input" } },
 1158            { 4, new string[] { "overlay_vaapi", "Action to take when encountering EOF from secondary input" } },
 1159            { 5, new string[] { "overlay_vulkan", "Action to take when encountering EOF from secondary input" } },
 1160            { 6, new string[] { "transpose_opencl", "rotate by half-turn" } }
 1161        };
 162
 163        // These are the library versions that corresponds to our minimum ffmpeg version 4.4 according to the version ta
 164        // Refers to the versions in https://ffmpeg.org/download.html
 1165        private static readonly Dictionary<string, Version> _ffmpegMinimumLibraryVersions = new Dictionary<string, Versi
 1166        {
 1167            { "libavutil", new Version(56, 70) },
 1168            { "libavcodec", new Version(58, 134) },
 1169            { "libavformat", new Version(58, 76) },
 1170            { "libavdevice", new Version(58, 13) },
 1171            { "libavfilter", new Version(7, 110) },
 1172            { "libswscale", new Version(5, 9) },
 1173            { "libswresample", new Version(3, 9) },
 1174            { "libpostproc", new Version(55, 9) }
 1175        };
 176
 177        private readonly ILogger _logger;
 178
 179        private readonly string _encoderPath;
 180
 16181        private readonly Version _minFFmpegMultiThreadedCli = new Version(7, 0);
 182
 183        public EncoderValidator(ILogger logger, string encoderPath)
 184        {
 16185            _logger = logger;
 16186            _encoderPath = encoderPath;
 16187        }
 188
 189        private enum Codec
 190        {
 191            Encoder,
 192            Decoder
 193        }
 194
 195        // When changing this, also change the minimum library versions in _ffmpegMinimumLibraryVersions
 1196        public static Version MinVersion { get; } = new Version(4, 4);
 197
 1198        public static Version? MaxVersion { get; } = null;
 199
 200        [GeneratedRegex(@"^ffmpeg version n?((?:[0-9]+\.?)+)")]
 201        private static partial Regex FfmpegVersionRegex();
 202
 203        [GeneratedRegex(@"((?<name>lib\w+)\s+(?<major>[0-9]+)\.\s*(?<minor>[0-9]+))", RegexOptions.Multiline)]
 204        private static partial Regex LibraryRegex();
 205
 206        public bool ValidateVersion()
 207        {
 208            string output;
 209            try
 210            {
 0211                output = GetProcessOutput(_encoderPath, "-version", false, null);
 0212            }
 0213            catch (Exception ex)
 214            {
 0215                _logger.LogError(ex, "Error validating encoder");
 0216                return false;
 217            }
 218
 0219            if (string.IsNullOrWhiteSpace(output))
 220            {
 0221                _logger.LogError("FFmpeg validation: The process returned no result");
 0222                return false;
 223            }
 224
 0225            _logger.LogDebug("ffmpeg output: {Output}", output);
 226
 0227            return ValidateVersionInternal(output);
 0228        }
 229
 230        internal bool ValidateVersionInternal(string versionOutput)
 231        {
 8232            if (versionOutput.Contains("Libav developers", StringComparison.OrdinalIgnoreCase))
 233            {
 0234                _logger.LogError("FFmpeg validation: avconv instead of ffmpeg is not supported");
 0235                return false;
 236            }
 237
 238            // Work out what the version under test is
 8239            var version = GetFFmpegVersionInternal(versionOutput);
 240
 8241            _logger.LogInformation("Found ffmpeg version {Version}", version is not null ? version.ToString() : "unknown
 242
 8243            if (version is null)
 244            {
 1245                if (MaxVersion is not null) // Version is unknown
 246                {
 0247                    if (MinVersion == MaxVersion)
 248                    {
 0249                        _logger.LogWarning("FFmpeg validation: We recommend version {MinVersion}", MinVersion);
 250                    }
 251                    else
 252                    {
 0253                        _logger.LogWarning("FFmpeg validation: We recommend a minimum of {MinVersion} and maximum of {Ma
 254                    }
 255                }
 256                else
 257                {
 1258                    _logger.LogWarning("FFmpeg validation: We recommend minimum version {MinVersion}", MinVersion);
 259                }
 260
 1261                return false;
 262            }
 263
 7264            if (version < MinVersion) // Version is below what we recommend
 265            {
 1266                _logger.LogWarning("FFmpeg validation: The minimum recommended version is {MinVersion}", MinVersion);
 1267                return false;
 268            }
 269
 6270            if (MaxVersion is not null && version > MaxVersion) // Version is above what we recommend
 271            {
 0272                _logger.LogWarning("FFmpeg validation: The maximum recommended version is {MaxVersion}", MaxVersion);
 0273                return false;
 274            }
 275
 6276            return true;
 277        }
 278
 0279        public IEnumerable<string> GetDecoders() => GetCodecs(Codec.Decoder);
 280
 0281        public IEnumerable<string> GetEncoders() => GetCodecs(Codec.Encoder);
 282
 0283        public IEnumerable<string> GetHwaccels() => GetHwaccelTypes();
 284
 0285        public IEnumerable<string> GetFilters() => GetFFmpegFilters();
 286
 0287        public IDictionary<int, bool> GetFiltersWithOption() => GetFFmpegFiltersWithOption();
 288
 289        public Version? GetFFmpegVersion()
 290        {
 291            string output;
 292            try
 293            {
 0294                output = GetProcessOutput(_encoderPath, "-version", false, null);
 0295            }
 0296            catch (Exception ex)
 297            {
 0298                _logger.LogError(ex, "Error validating encoder");
 0299                return null;
 300            }
 301
 0302            if (string.IsNullOrWhiteSpace(output))
 303            {
 0304                _logger.LogError("FFmpeg validation: The process returned no result");
 0305                return null;
 306            }
 307
 0308            _logger.LogDebug("ffmpeg output: {Output}", output);
 309
 0310            return GetFFmpegVersionInternal(output);
 0311        }
 312
 313        /// <summary>
 314        /// Using the output from "ffmpeg -version" work out the FFmpeg version.
 315        /// For pre-built binaries the first line should contain a string like "ffmpeg version x.y", which is easy
 316        /// to parse. If this is not available, then we try to match known library versions to FFmpeg versions.
 317        /// If that fails then we test the libraries to determine if they're newer than our minimum versions.
 318        /// </summary>
 319        /// <param name="output">The output from "ffmpeg -version".</param>
 320        /// <returns>The FFmpeg version.</returns>
 321        internal Version? GetFFmpegVersionInternal(string output)
 322        {
 323            // For pre-built binaries the FFmpeg version should be mentioned at the very start of the output
 16324            var match = FfmpegVersionRegex().Match(output);
 325
 16326            if (match.Success)
 327            {
 12328                if (Version.TryParse(match.Groups[1].ValueSpan, out var result))
 329                {
 12330                    return result;
 331                }
 332            }
 333
 4334            var versionMap = GetFFmpegLibraryVersions(output);
 335
 4336            var allVersionsValidated = true;
 337
 72338            foreach (var minimumVersion in _ffmpegMinimumLibraryVersions)
 339            {
 32340                if (versionMap.TryGetValue(minimumVersion.Key, out var foundVersion))
 341                {
 32342                    if (foundVersion >= minimumVersion.Value)
 343                    {
 16344                        _logger.LogInformation("Found {Library} version {FoundVersion} ({MinimumVersion})", minimumVersi
 345                    }
 346                    else
 347                    {
 16348                        _logger.LogWarning("Found {Library} version {FoundVersion} lower than recommended version {Minim
 16349                        allVersionsValidated = false;
 350                    }
 351                }
 352                else
 353                {
 0354                    _logger.LogError("{Library} version not found", minimumVersion.Key);
 0355                    allVersionsValidated = false;
 356                }
 357            }
 358
 4359            return allVersionsValidated ? MinVersion : null;
 360        }
 361
 362        /// <summary>
 363        /// Grabs the library names and major.minor version numbers from the 'ffmpeg -version' output
 364        /// and condenses them on to one line.  Output format is "name1=major.minor,name2=major.minor,etc.".
 365        /// </summary>
 366        /// <param name="output">The 'ffmpeg -version' output.</param>
 367        /// <returns>The library names and major.minor version numbers.</returns>
 368        private static Dictionary<string, Version> GetFFmpegLibraryVersions(string output)
 369        {
 4370            var map = new Dictionary<string, Version>();
 371
 72372            foreach (Match match in LibraryRegex().Matches(output))
 373            {
 32374                var version = new Version(
 32375                    int.Parse(match.Groups["major"].ValueSpan, CultureInfo.InvariantCulture),
 32376                    int.Parse(match.Groups["minor"].ValueSpan, CultureInfo.InvariantCulture));
 377
 32378                map.Add(match.Groups["name"].Value, version);
 379            }
 380
 4381            return map;
 382        }
 383
 384        public bool CheckVaapiDeviceByDriverName(string driverName, string renderNodePath)
 385        {
 0386            if (!OperatingSystem.IsLinux())
 387            {
 0388                return false;
 389            }
 390
 0391            if (string.IsNullOrEmpty(driverName) || string.IsNullOrEmpty(renderNodePath))
 392            {
 0393                return false;
 394            }
 395
 396            try
 397            {
 0398                var output = GetProcessOutput(_encoderPath, "-v verbose -hide_banner -init_hw_device vaapi=va:" + render
 0399                return output.Contains(driverName, StringComparison.Ordinal);
 400            }
 0401            catch (Exception ex)
 402            {
 0403                _logger.LogError(ex, "Error detecting the given vaapi render node path");
 0404                return false;
 405            }
 0406        }
 407
 408        public bool CheckVulkanDrmDeviceByExtensionName(string renderNodePath, string[] vulkanExtensions)
 409        {
 0410            if (!OperatingSystem.IsLinux())
 411            {
 0412                return false;
 413            }
 414
 0415            if (string.IsNullOrEmpty(renderNodePath))
 416            {
 0417                return false;
 418            }
 419
 420            try
 421            {
 0422                var command = "-v verbose -hide_banner -init_hw_device drm=dr:" + renderNodePath + " -init_hw_device vul
 0423                var output = GetProcessOutput(_encoderPath, command, true, null);
 0424                foreach (string ext in vulkanExtensions)
 425                {
 0426                    if (!output.Contains(ext, StringComparison.Ordinal))
 427                    {
 0428                        return false;
 429                    }
 430                }
 431
 0432                return true;
 433            }
 0434            catch (Exception ex)
 435            {
 0436                _logger.LogError(ex, "Error detecting the given drm render node path");
 0437                return false;
 438            }
 0439        }
 440
 441        [SupportedOSPlatform("macos")]
 442        public bool CheckIsVideoToolboxAv1DecodeAvailable()
 443        {
 0444            return ApplePlatformHelper.HasAv1HardwareAccel(_logger);
 445        }
 446
 447        private IEnumerable<string> GetHwaccelTypes()
 448        {
 0449            string? output = null;
 450            try
 451            {
 0452                output = GetProcessOutput(_encoderPath, "-hwaccels", false, null);
 0453            }
 0454            catch (Exception ex)
 455            {
 0456                _logger.LogError(ex, "Error detecting available hwaccel types");
 0457            }
 458
 0459            if (string.IsNullOrWhiteSpace(output))
 460            {
 0461                return Enumerable.Empty<string>();
 462            }
 463
 0464            var found = output.Split(new char[] { '\r', '\n' }, StringSplitOptions.RemoveEmptyEntries).Skip(1).Distinct(
 0465            _logger.LogInformation("Available hwaccel types: {Types}", found);
 466
 0467            return found;
 468        }
 469
 470        public bool CheckFilterWithOption(string filter, string option)
 471        {
 0472            if (string.IsNullOrEmpty(filter) || string.IsNullOrEmpty(option))
 473            {
 0474                return false;
 475            }
 476
 477            string output;
 478            try
 479            {
 0480                output = GetProcessOutput(_encoderPath, "-h filter=" + filter, false, null);
 0481            }
 0482            catch (Exception ex)
 483            {
 0484                _logger.LogError(ex, "Error detecting the given filter");
 0485                return false;
 486            }
 487
 0488            if (output.Contains("Filter " + filter, StringComparison.Ordinal))
 489            {
 0490                return output.Contains(option, StringComparison.Ordinal);
 491            }
 492
 0493            _logger.LogWarning("Filter: {Name} with option {Option} is not available", filter, option);
 494
 0495            return false;
 0496        }
 497
 498        public bool CheckSupportedRuntimeKey(string keyDesc, Version? ffmpegVersion)
 499        {
 0500            if (string.IsNullOrEmpty(keyDesc))
 501            {
 0502                return false;
 503            }
 504
 505            string output;
 506            try
 507            {
 508                // With multi-threaded cli support, FFmpeg 7 is less sensitive to keyboard input
 0509                var duration = ffmpegVersion >= _minFFmpegMultiThreadedCli ? 10000 : 1000;
 0510                output = GetProcessOutput(_encoderPath, $"-hide_banner -f lavfi -i nullsrc=s=1x1:d={duration} -f null -"
 0511            }
 0512            catch (Exception ex)
 513            {
 0514                _logger.LogError(ex, "Error checking supported runtime key");
 0515                return false;
 516            }
 517
 0518            return output.Contains(keyDesc, StringComparison.Ordinal);
 0519        }
 520
 521        public bool CheckSupportedHwaccelFlag(string flag)
 522        {
 0523            return !string.IsNullOrEmpty(flag) && GetProcessExitCode(_encoderPath, $"-loglevel quiet -hwaccel_flags +{fl
 524        }
 525
 526        private IEnumerable<string> GetCodecs(Codec codec)
 527        {
 0528            string codecstr = codec == Codec.Encoder ? "encoders" : "decoders";
 529            string output;
 530            try
 531            {
 0532                output = GetProcessOutput(_encoderPath, "-" + codecstr, false, null);
 0533            }
 0534            catch (Exception ex)
 535            {
 0536                _logger.LogError(ex, "Error detecting available {Codec}", codecstr);
 0537                return Enumerable.Empty<string>();
 538            }
 539
 0540            if (string.IsNullOrWhiteSpace(output))
 541            {
 0542                return Enumerable.Empty<string>();
 543            }
 544
 0545            var required = codec == Codec.Encoder ? _requiredEncoders : _requiredDecoders;
 546
 0547            var found = CodecRegex()
 0548                .Matches(output)
 0549                .Select(x => x.Groups["codec"].Value)
 0550                .Where(x => required.Contains(x));
 551
 0552            _logger.LogInformation("Available {Codec}: {Codecs}", codecstr, found);
 553
 0554            return found;
 0555        }
 556
 557        private IEnumerable<string> GetFFmpegFilters()
 558        {
 559            string output;
 560            try
 561            {
 0562                output = GetProcessOutput(_encoderPath, "-filters", false, null);
 0563            }
 0564            catch (Exception ex)
 565            {
 0566                _logger.LogError(ex, "Error detecting available filters");
 0567                return Enumerable.Empty<string>();
 568            }
 569
 0570            if (string.IsNullOrWhiteSpace(output))
 571            {
 0572                return Enumerable.Empty<string>();
 573            }
 574
 0575            var found = FilterRegex()
 0576                .Matches(output)
 0577                .Select(x => x.Groups["filter"].Value)
 0578                .Where(x => _requiredFilters.Contains(x));
 579
 0580            _logger.LogInformation("Available filters: {Filters}", found);
 581
 0582            return found;
 0583        }
 584
 585        private Dictionary<int, bool> GetFFmpegFiltersWithOption()
 586        {
 0587            Dictionary<int, bool> dict = new Dictionary<int, bool>();
 0588            for (int i = 0; i < _filterOptionsDict.Count; i++)
 589            {
 0590                if (_filterOptionsDict.TryGetValue(i, out var val) && val.Length == 2)
 591                {
 0592                    dict.Add(i, CheckFilterWithOption(val[0], val[1]));
 593                }
 594            }
 595
 0596            return dict;
 597        }
 598
 599        private string GetProcessOutput(string path, string arguments, bool readStdErr, string? testKey)
 600        {
 0601            var redirectStandardIn = !string.IsNullOrEmpty(testKey);
 0602            using (var process = new Process
 0603            {
 0604                StartInfo = new ProcessStartInfo(path, arguments)
 0605                {
 0606                    CreateNoWindow = true,
 0607                    UseShellExecute = false,
 0608                    WindowStyle = ProcessWindowStyle.Hidden,
 0609                    ErrorDialog = false,
 0610                    RedirectStandardInput = redirectStandardIn,
 0611                    RedirectStandardOutput = true,
 0612                    RedirectStandardError = true
 0613                }
 0614            })
 615            {
 0616                _logger.LogDebug("Running {Path} {Arguments}", path, arguments);
 617
 0618                process.Start();
 619
 0620                if (redirectStandardIn)
 621                {
 0622                    using var writer = process.StandardInput;
 0623                    writer.Write(testKey);
 624                }
 625
 0626                using var reader = readStdErr ? process.StandardError : process.StandardOutput;
 0627                return reader.ReadToEnd();
 628            }
 0629        }
 630
 631        private bool GetProcessExitCode(string path, string arguments)
 632        {
 0633            using var process = new Process();
 0634            process.StartInfo = new ProcessStartInfo(path, arguments)
 0635            {
 0636                CreateNoWindow = true,
 0637                UseShellExecute = false,
 0638                WindowStyle = ProcessWindowStyle.Hidden,
 0639                ErrorDialog = false
 0640            };
 0641            _logger.LogDebug("Running {Path} {Arguments}", path, arguments);
 642
 643            try
 644            {
 0645                process.Start();
 0646                process.WaitForExit();
 0647                return process.ExitCode == 0;
 648            }
 0649            catch (Exception ex)
 650            {
 0651                _logger.LogError("Running {Path} {Arguments} failed with exception {Exception}", path, arguments, ex.Mes
 0652                return false;
 653            }
 0654        }
 655
 656        [GeneratedRegex("^\\s\\S{6}\\s(?<codec>[\\w|-]+)\\s+.+$", RegexOptions.Multiline)]
 657        private static partial Regex CodecRegex();
 658
 659        [GeneratedRegex("^\\s\\S{3}\\s(?<filter>[\\w|-]+)\\s+.+$", RegexOptions.Multiline)]
 660        private static partial Regex FilterRegex();
 661    }
 662}