using System; using System.Collections.Generic; using ErsatzTV.FFmpeg.Encoder; using ErsatzTV.FFmpeg.Format; using ErsatzTV.FFmpeg.OutputFormat; using ErsatzTV.FFmpeg.State; using FluentAssertions; using LanguageExt; using Microsoft.Extensions.Logging; using Moq; using NUnit.Framework; using static LanguageExt.Prelude; namespace ErsatzTV.FFmpeg.Tests; [TestFixture] public class PipelineGeneratorTests { private readonly ILogger _logger = new Mock().Object; [Test] public void Incorrect_Video_Codec_Should_Use_Encoder() { var videoInputFile = new VideoInputFile( "/tmp/whatever.mkv", new List { new(0, VideoFormat.H264, new PixelFormatYuv420P(), new FrameSize(1920, 1080), "24", false) }); var audioInputFile = new AudioInputFile( "/tmp/whatever.mkv", new List { new(1, AudioFormat.Aac, 2) }, new AudioState( AudioFormat.Aac, 2, 320, 640, 48, Option.None, false)); var desiredState = new FrameState( true, false, VideoFormat.Hevc, new PixelFormatYuv420P(), new FrameSize(1920, 1080), new FrameSize(1920, 1080), Option.None, 2000, 4000, 90_000, false); var ffmpegState = new FFmpegState( false, HardwareAccelerationMode.None, Option.None, Option.None, Option.None, Option.None, false, Option.None, Option.None, Option.None, OutputFormatKind.MpegTs, Option.None, Option.None, 0); var builder = new PipelineBuilder(videoInputFile, audioInputFile, None, "", _logger); FFmpegPipeline result = builder.Build(ffmpegState, desiredState); result.PipelineSteps.Should().HaveCountGreaterThan(0); result.PipelineSteps.Should().Contain(ps => ps is EncoderLibx265); PrintCommand(videoInputFile, audioInputFile, None, None, result); } [Test] public void Concat_Test() { var resolution = new FrameSize(1920, 1080); var concatInputFile = new ConcatInputFile("http://localhost:8080/ffmpeg/concat/1", resolution); var builder = new PipelineBuilder(None, None, None, "", _logger); FFmpegPipeline result = builder.Concat(concatInputFile, FFmpegState.Concat(false, "Some Channel")); result.PipelineSteps.Should().HaveCountGreaterThan(0); string command = PrintCommand(None, None, None, concatInputFile, result); command.Should().Be( "-threads 1 -nostdin -hide_banner -nostats -loglevel error -fflags +genpts+discardcorrupt+igndts -f concat -safe 0 -protocol_whitelist file,http,tcp,https,tcp,tls -probesize 32 -re -stream_loop -1 -i http://localhost:8080/ffmpeg/concat/1 -muxdelay 0 -muxpreload 0 -movflags +faststart -flags cgop -sc_threshold 0 -c copy -map_metadata -1 -metadata service_provider=\"ErsatzTV\" -metadata service_name=\"Some Channel\" -f mpegts -mpegts_flags +initial_discontinuity pipe:1"); } [Test] public void HlsDirect_Test() { var videoInputFile = new VideoInputFile( "/tmp/whatever.mkv", new List { new(0, VideoFormat.H264, new PixelFormatYuv420P(), new FrameSize(1920, 1080), "24", false) }); var audioInputFile = new AudioInputFile( "/tmp/whatever.mkv", new List { new(1, AudioFormat.Aac, 2) }, new AudioState( AudioFormat.Copy, None, None, None, None, None, false)); var desiredState = new FrameState( true, false, VideoFormat.Copy, new PixelFormatYuv420P(), new FrameSize(1920, 1080), new FrameSize(1920, 1080), Option.None, 2000, 4000, 90_000, false); var ffmpegState = new FFmpegState( false, HardwareAccelerationMode.None, Option.None, Option.None, Option.None, Option.None, false, Option.None, Option.None, Option.None, OutputFormatKind.MpegTs, Option.None, Option.None, 0); var builder = new PipelineBuilder(videoInputFile, audioInputFile, None, "", _logger); FFmpegPipeline result = builder.Build(ffmpegState, desiredState); result.PipelineSteps.Should().HaveCountGreaterThan(0); result.PipelineSteps.Should().Contain(ps => ps is EncoderCopyVideo); result.PipelineSteps.Should().Contain(ps => ps is EncoderCopyAudio); string command = PrintCommand(videoInputFile, audioInputFile, None, None, result); command.Should().Be( "-threads 1 -nostdin -hide_banner -nostats -loglevel error -fflags +genpts+discardcorrupt+igndts -i /tmp/whatever.mkv -map 0:1 -map 0:0 -muxdelay 0 -muxpreload 0 -movflags +faststart -flags cgop -sc_threshold 0 -c:v copy -c:a copy -f mpegts -mpegts_flags +initial_discontinuity pipe:1"); } private static string PrintCommand( Option videoInputFile, Option audioInputFile, Option watermarkInputFile, Option concatInputFile, FFmpegPipeline pipeline) { IList arguments = CommandGenerator.GenerateArguments( videoInputFile, audioInputFile, watermarkInputFile, concatInputFile, pipeline.PipelineSteps); var command = string.Join(" ", arguments); Console.WriteLine($"Generated command: ffmpeg {string.Join(" ", arguments)}"); return command; } }