| | 1 | | #nullable disable |
| | 2 | |
|
| | 3 | | #pragma warning disable CS1591 |
| | 4 | |
|
| | 5 | | using System; |
| | 6 | | using System.Collections.Generic; |
| | 7 | | using System.Diagnostics; |
| | 8 | | using System.Globalization; |
| | 9 | | using System.IO; |
| | 10 | | using System.Text; |
| | 11 | | using System.Text.Json; |
| | 12 | | using System.Threading; |
| | 13 | | using System.Threading.Tasks; |
| | 14 | | using Jellyfin.Extensions; |
| | 15 | | using Jellyfin.Extensions.Json; |
| | 16 | | using MediaBrowser.Common; |
| | 17 | | using MediaBrowser.Common.Configuration; |
| | 18 | | using MediaBrowser.Controller; |
| | 19 | | using MediaBrowser.Controller.Configuration; |
| | 20 | | using MediaBrowser.Controller.Library; |
| | 21 | | using MediaBrowser.Controller.MediaEncoding; |
| | 22 | | using MediaBrowser.Model.Dto; |
| | 23 | | using MediaBrowser.Model.IO; |
| | 24 | | using Microsoft.Extensions.Logging; |
| | 25 | |
|
| | 26 | | namespace Jellyfin.LiveTv.IO |
| | 27 | | { |
| | 28 | | public class EncodedRecorder : IRecorder |
| | 29 | | { |
| | 30 | | private readonly ILogger _logger; |
| | 31 | | private readonly IMediaEncoder _mediaEncoder; |
| | 32 | | private readonly IServerApplicationPaths _appPaths; |
| 0 | 33 | | private readonly TaskCompletionSource<bool> _taskCompletionSource = new TaskCompletionSource<bool>(TaskCreationO |
| | 34 | | private readonly IServerConfigurationManager _serverConfigurationManager; |
| 0 | 35 | | private readonly JsonSerializerOptions _jsonOptions = JsonDefaults.Options; |
| | 36 | | private bool _hasExited; |
| | 37 | | private FileStream _logFileStream; |
| | 38 | | private string _targetPath; |
| | 39 | | private Process _process; |
| | 40 | | private bool _disposed; |
| | 41 | |
|
| | 42 | | public EncodedRecorder( |
| | 43 | | ILogger logger, |
| | 44 | | IMediaEncoder mediaEncoder, |
| | 45 | | IServerApplicationPaths appPaths, |
| | 46 | | IServerConfigurationManager serverConfigurationManager) |
| | 47 | | { |
| 0 | 48 | | _logger = logger; |
| 0 | 49 | | _mediaEncoder = mediaEncoder; |
| 0 | 50 | | _appPaths = appPaths; |
| 0 | 51 | | _serverConfigurationManager = serverConfigurationManager; |
| 0 | 52 | | } |
| | 53 | |
|
| 0 | 54 | | private static bool CopySubtitles => false; |
| | 55 | |
|
| | 56 | | public string GetOutputPath(MediaSourceInfo mediaSource, string targetFile) |
| | 57 | | { |
| 0 | 58 | | return Path.ChangeExtension(targetFile, ".ts"); |
| | 59 | | } |
| | 60 | |
|
| | 61 | | public async Task Record(IDirectStreamProvider directStreamProvider, MediaSourceInfo mediaSource, string targetF |
| | 62 | | { |
| | 63 | | // The media source is infinite so we need to handle stopping ourselves |
| | 64 | | using var durationToken = new CancellationTokenSource(duration); |
| | 65 | | using var cancellationTokenSource = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken, durat |
| | 66 | |
|
| | 67 | | await RecordFromFile(mediaSource, mediaSource.Path, targetFile, onStarted, cancellationTokenSource.Token).Co |
| | 68 | |
|
| | 69 | | _logger.LogInformation("Recording completed to file {Path}", targetFile); |
| | 70 | | } |
| | 71 | |
|
| | 72 | | private async Task RecordFromFile(MediaSourceInfo mediaSource, string inputFile, string targetFile, Action onSta |
| | 73 | | { |
| | 74 | | _targetPath = targetFile; |
| | 75 | | Directory.CreateDirectory(Path.GetDirectoryName(targetFile)); |
| | 76 | |
|
| | 77 | | var processStartInfo = new ProcessStartInfo |
| | 78 | | { |
| | 79 | | CreateNoWindow = true, |
| | 80 | | UseShellExecute = false, |
| | 81 | |
|
| | 82 | | RedirectStandardError = true, |
| | 83 | | RedirectStandardInput = true, |
| | 84 | |
|
| | 85 | | FileName = _mediaEncoder.EncoderPath, |
| | 86 | | Arguments = GetCommandLineArgs(mediaSource, inputFile, targetFile), |
| | 87 | |
|
| | 88 | | WindowStyle = ProcessWindowStyle.Hidden, |
| | 89 | | ErrorDialog = false |
| | 90 | | }; |
| | 91 | |
|
| | 92 | | _logger.LogInformation("{Filename} {Arguments}", processStartInfo.FileName, processStartInfo.Arguments); |
| | 93 | |
|
| | 94 | | var logFilePath = Path.Combine(_appPaths.LogDirectoryPath, "record-transcode-" + Guid.NewGuid() + ".txt"); |
| | 95 | | Directory.CreateDirectory(Path.GetDirectoryName(logFilePath)); |
| | 96 | |
|
| | 97 | | // FFMpeg writes debug/error info to stderr. This is useful when debugging so let's put it in the log direct |
| | 98 | | _logFileStream = new FileStream(logFilePath, FileMode.CreateNew, FileAccess.Write, FileShare.Read, IODefault |
| | 99 | |
|
| | 100 | | await JsonSerializer.SerializeAsync(_logFileStream, mediaSource, _jsonOptions, cancellationToken).ConfigureA |
| | 101 | | await _logFileStream.WriteAsync(Encoding.UTF8.GetBytes(Environment.NewLine + Environment.NewLine + processSt |
| | 102 | |
|
| | 103 | | _process = new Process |
| | 104 | | { |
| | 105 | | StartInfo = processStartInfo, |
| | 106 | | EnableRaisingEvents = true |
| | 107 | | }; |
| | 108 | | _process.Exited += (_, _) => OnFfMpegProcessExited(_process); |
| | 109 | |
|
| | 110 | | _process.Start(); |
| | 111 | |
|
| | 112 | | cancellationToken.Register(Stop); |
| | 113 | |
|
| | 114 | | onStarted(); |
| | 115 | |
|
| | 116 | | // Important - don't await the log task or we won't be able to kill ffmpeg when the user stops playback |
| | 117 | | _ = StartStreamingLog(_process.StandardError.BaseStream, _logFileStream); |
| | 118 | |
|
| | 119 | | _logger.LogInformation("ffmpeg recording process started for {Path}", _targetPath); |
| | 120 | |
|
| | 121 | | // Block until ffmpeg exits |
| | 122 | | await _taskCompletionSource.Task.ConfigureAwait(false); |
| | 123 | | } |
| | 124 | |
|
| | 125 | | private string GetCommandLineArgs(MediaSourceInfo mediaSource, string inputTempFile, string targetFile) |
| | 126 | | { |
| 0 | 127 | | string videoArgs = "-codec:v:0 copy -fflags +genpts"; |
| | 128 | |
|
| 0 | 129 | | var flags = new List<string>(); |
| 0 | 130 | | if (mediaSource.IgnoreDts) |
| | 131 | | { |
| 0 | 132 | | flags.Add("+igndts"); |
| | 133 | | } |
| | 134 | |
|
| 0 | 135 | | if (mediaSource.IgnoreIndex) |
| | 136 | | { |
| 0 | 137 | | flags.Add("+ignidx"); |
| | 138 | | } |
| | 139 | |
|
| 0 | 140 | | if (mediaSource.GenPtsInput) |
| | 141 | | { |
| 0 | 142 | | flags.Add("+genpts"); |
| | 143 | | } |
| | 144 | |
|
| 0 | 145 | | var inputModifier = "-async 1"; |
| | 146 | |
|
| 0 | 147 | | if (flags.Count > 0) |
| | 148 | | { |
| 0 | 149 | | inputModifier += " -fflags " + string.Join(string.Empty, flags); |
| | 150 | | } |
| | 151 | |
|
| 0 | 152 | | if (mediaSource.ReadAtNativeFramerate) |
| | 153 | | { |
| 0 | 154 | | inputModifier += " -re"; |
| | 155 | | } |
| | 156 | |
|
| 0 | 157 | | if (mediaSource.RequiresLooping) |
| | 158 | | { |
| 0 | 159 | | inputModifier += " -stream_loop -1 -reconnect_at_eof 1 -reconnect_streamed 1 -reconnect_delay_max 2"; |
| | 160 | | } |
| | 161 | |
|
| 0 | 162 | | var analyzeDurationSeconds = 5; |
| 0 | 163 | | var analyzeDuration = " -analyzeduration " + |
| 0 | 164 | | (analyzeDurationSeconds * 1000000).ToString(CultureInfo.InvariantCulture); |
| 0 | 165 | | inputModifier += analyzeDuration; |
| | 166 | |
|
| 0 | 167 | | var subtitleArgs = CopySubtitles ? " -codec:s copy" : " -sn"; |
| | 168 | |
|
| | 169 | | // var outputParam = string.Equals(Path.GetExtension(targetFile), ".mp4", StringComparison.OrdinalIgnoreCase |
| | 170 | | // " -f mp4 -movflags frag_keyframe+empty_moov" : |
| | 171 | | // string.Empty; |
| | 172 | |
|
| 0 | 173 | | var outputParam = string.Empty; |
| | 174 | |
|
| 0 | 175 | | var threads = EncodingHelper.GetNumberOfThreads(null, _serverConfigurationManager.GetEncodingOptions(), null |
| 0 | 176 | | var commandLineArgs = string.Format( |
| 0 | 177 | | CultureInfo.InvariantCulture, |
| 0 | 178 | | "-i \"{0}\" {2} -map_metadata -1 -threads {6} {3}{4}{5} -y \"{1}\"", |
| 0 | 179 | | inputTempFile, |
| 0 | 180 | | targetFile.Replace("\"", "\\\"", StringComparison.Ordinal), // Escape quotes in filename |
| 0 | 181 | | videoArgs, |
| 0 | 182 | | GetAudioArgs(mediaSource), |
| 0 | 183 | | subtitleArgs, |
| 0 | 184 | | outputParam, |
| 0 | 185 | | threads); |
| | 186 | |
|
| 0 | 187 | | return inputModifier + " " + commandLineArgs; |
| | 188 | | } |
| | 189 | |
|
| | 190 | | private static string GetAudioArgs(MediaSourceInfo mediaSource) |
| | 191 | | { |
| 0 | 192 | | return "-codec:a:0 copy"; |
| | 193 | | } |
| | 194 | |
|
| | 195 | | protected string GetOutputSizeParam() |
| 0 | 196 | | => "-vf \"yadif=0:-1:0\""; |
| | 197 | |
|
| | 198 | | private void Stop() |
| | 199 | | { |
| 0 | 200 | | if (!_hasExited) |
| | 201 | | { |
| | 202 | | try |
| | 203 | | { |
| 0 | 204 | | _logger.LogInformation("Stopping ffmpeg recording process for {Path}", _targetPath); |
| | 205 | |
|
| 0 | 206 | | _process.StandardInput.WriteLine("q"); |
| 0 | 207 | | } |
| 0 | 208 | | catch (Exception ex) |
| | 209 | | { |
| 0 | 210 | | _logger.LogError(ex, "Error stopping recording transcoding job for {Path}", _targetPath); |
| 0 | 211 | | } |
| | 212 | |
|
| 0 | 213 | | if (_hasExited) |
| | 214 | | { |
| 0 | 215 | | return; |
| | 216 | | } |
| | 217 | |
|
| | 218 | | try |
| | 219 | | { |
| 0 | 220 | | _logger.LogInformation("Calling recording process.WaitForExit for {Path}", _targetPath); |
| | 221 | |
|
| 0 | 222 | | if (_process.WaitForExit(10000)) |
| | 223 | | { |
| 0 | 224 | | return; |
| | 225 | | } |
| 0 | 226 | | } |
| 0 | 227 | | catch (Exception ex) |
| | 228 | | { |
| 0 | 229 | | _logger.LogError(ex, "Error waiting for recording process to exit for {Path}", _targetPath); |
| 0 | 230 | | } |
| | 231 | |
|
| 0 | 232 | | if (_hasExited) |
| | 233 | | { |
| 0 | 234 | | return; |
| | 235 | | } |
| | 236 | |
|
| | 237 | | try |
| | 238 | | { |
| 0 | 239 | | _logger.LogInformation("Killing ffmpeg recording process for {Path}", _targetPath); |
| | 240 | |
|
| 0 | 241 | | _process.Kill(); |
| 0 | 242 | | } |
| 0 | 243 | | catch (Exception ex) |
| | 244 | | { |
| 0 | 245 | | _logger.LogError(ex, "Error killing recording transcoding job for {Path}", _targetPath); |
| 0 | 246 | | } |
| | 247 | | } |
| 0 | 248 | | } |
| | 249 | |
|
| | 250 | | /// <summary> |
| | 251 | | /// Processes the exited. |
| | 252 | | /// </summary> |
| | 253 | | private void OnFfMpegProcessExited(Process process) |
| | 254 | | { |
| 0 | 255 | | using (process) |
| | 256 | | { |
| 0 | 257 | | _hasExited = true; |
| | 258 | |
|
| 0 | 259 | | _logFileStream?.Dispose(); |
| 0 | 260 | | _logFileStream = null; |
| | 261 | |
|
| 0 | 262 | | var exitCode = process.ExitCode; |
| | 263 | |
|
| 0 | 264 | | _logger.LogInformation("FFMpeg recording exited with code {ExitCode} for {Path}", exitCode, _targetPath) |
| | 265 | |
|
| 0 | 266 | | if (exitCode == 0) |
| | 267 | | { |
| 0 | 268 | | _taskCompletionSource.TrySetResult(true); |
| | 269 | | } |
| | 270 | | else |
| | 271 | | { |
| 0 | 272 | | _taskCompletionSource.TrySetException( |
| 0 | 273 | | new FfmpegException( |
| 0 | 274 | | string.Format( |
| 0 | 275 | | CultureInfo.InvariantCulture, |
| 0 | 276 | | "Recording for {0} failed. Exit code {1}", |
| 0 | 277 | | _targetPath, |
| 0 | 278 | | exitCode))); |
| | 279 | | } |
| 0 | 280 | | } |
| 0 | 281 | | } |
| | 282 | |
|
| | 283 | | private async Task StartStreamingLog(Stream source, FileStream target) |
| | 284 | | { |
| | 285 | | try |
| | 286 | | { |
| | 287 | | using (var reader = new StreamReader(source)) |
| | 288 | | { |
| | 289 | | await foreach (var line in reader.ReadAllLinesAsync().ConfigureAwait(false)) |
| | 290 | | { |
| | 291 | | var bytes = Encoding.UTF8.GetBytes(Environment.NewLine + line); |
| | 292 | |
|
| | 293 | | await target.WriteAsync(bytes.AsMemory()).ConfigureAwait(false); |
| | 294 | | await target.FlushAsync().ConfigureAwait(false); |
| | 295 | | } |
| | 296 | | } |
| | 297 | | } |
| | 298 | | catch (Exception ex) |
| | 299 | | { |
| | 300 | | _logger.LogError(ex, "Error reading ffmpeg recording log"); |
| | 301 | | } |
| | 302 | | } |
| | 303 | |
|
| | 304 | | /// <inheritdoc /> |
| | 305 | | public void Dispose() |
| | 306 | | { |
| 0 | 307 | | Dispose(true); |
| 0 | 308 | | GC.SuppressFinalize(this); |
| 0 | 309 | | } |
| | 310 | |
|
| | 311 | | /// <summary> |
| | 312 | | /// Releases unmanaged and optionally managed resources. |
| | 313 | | /// </summary> |
| | 314 | | /// <param name="disposing"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release |
| | 315 | | protected virtual void Dispose(bool disposing) |
| | 316 | | { |
| 0 | 317 | | if (_disposed) |
| | 318 | | { |
| 0 | 319 | | return; |
| | 320 | | } |
| | 321 | |
|
| 0 | 322 | | if (disposing) |
| | 323 | | { |
| 0 | 324 | | _logFileStream?.Dispose(); |
| 0 | 325 | | _process?.Dispose(); |
| | 326 | | } |
| | 327 | |
|
| 0 | 328 | | _logFileStream = null; |
| 0 | 329 | | _process = null; |
| | 330 | |
|
| 0 | 331 | | _disposed = true; |
| 0 | 332 | | } |
| | 333 | | } |
| | 334 | | } |