mirror of
				https://github.com/jellyfin/jellyfin.git
				synced 2025-10-26 08:12:42 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			183 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			183 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| #nullable disable
 | |
| 
 | |
| #pragma warning disable CS1591
 | |
| 
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Globalization;
 | |
| using System.IO;
 | |
| using System.Linq;
 | |
| using System.Text.Json;
 | |
| using System.Threading;
 | |
| using System.Threading.Tasks;
 | |
| using Jellyfin.Extensions.Json;
 | |
| using MediaBrowser.Common.Configuration;
 | |
| using MediaBrowser.Common.Extensions;
 | |
| using MediaBrowser.Controller.MediaEncoding;
 | |
| using MediaBrowser.Model.Dlna;
 | |
| using MediaBrowser.Model.Dto;
 | |
| using MediaBrowser.Model.Entities;
 | |
| using MediaBrowser.Model.IO;
 | |
| using MediaBrowser.Model.MediaInfo;
 | |
| using Microsoft.Extensions.Logging;
 | |
| 
 | |
| namespace Emby.Server.Implementations.Library
 | |
| {
 | |
|     public class LiveStreamHelper
 | |
|     {
 | |
|         private readonly IMediaEncoder _mediaEncoder;
 | |
|         private readonly ILogger _logger;
 | |
|         private readonly IApplicationPaths _appPaths;
 | |
|         private readonly JsonSerializerOptions _jsonOptions = JsonDefaults.Options;
 | |
| 
 | |
|         public LiveStreamHelper(IMediaEncoder mediaEncoder, ILogger logger, IApplicationPaths appPaths)
 | |
|         {
 | |
|             _mediaEncoder = mediaEncoder;
 | |
|             _logger = logger;
 | |
|             _appPaths = appPaths;
 | |
|         }
 | |
| 
 | |
|         public async Task AddMediaInfoWithProbe(MediaSourceInfo mediaSource, bool isAudio, string cacheKey, bool addProbeDelay, CancellationToken cancellationToken)
 | |
|         {
 | |
|             var originalRuntime = mediaSource.RunTimeTicks;
 | |
| 
 | |
|             var now = DateTime.UtcNow;
 | |
| 
 | |
|             MediaInfo mediaInfo = null;
 | |
|             var cacheFilePath = string.IsNullOrEmpty(cacheKey) ? null : Path.Combine(_appPaths.CachePath, "mediainfo", cacheKey.GetMD5().ToString("N", CultureInfo.InvariantCulture) + ".json");
 | |
| 
 | |
|             if (!string.IsNullOrEmpty(cacheKey))
 | |
|             {
 | |
|                 try
 | |
|                 {
 | |
|                     await using FileStream jsonStream = AsyncFile.OpenRead(cacheFilePath);
 | |
|                     mediaInfo = await JsonSerializer.DeserializeAsync<MediaInfo>(jsonStream, _jsonOptions, cancellationToken).ConfigureAwait(false);
 | |
| 
 | |
|                     // _logger.LogDebug("Found cached media info");
 | |
|                 }
 | |
|                 catch
 | |
|                 {
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (mediaInfo == null)
 | |
|             {
 | |
|                 if (addProbeDelay)
 | |
|                 {
 | |
|                     var delayMs = mediaSource.AnalyzeDurationMs ?? 0;
 | |
|                     delayMs = Math.Max(3000, delayMs);
 | |
|                     _logger.LogInformation("Waiting {0}ms before probing the live stream", delayMs);
 | |
|                     await Task.Delay(delayMs, cancellationToken).ConfigureAwait(false);
 | |
|                 }
 | |
| 
 | |
|                 mediaSource.AnalyzeDurationMs = 3000;
 | |
| 
 | |
|                 mediaInfo = await _mediaEncoder.GetMediaInfo(
 | |
|                     new MediaInfoRequest
 | |
|                     {
 | |
|                         MediaSource = mediaSource,
 | |
|                         MediaType = isAudio ? DlnaProfileType.Audio : DlnaProfileType.Video,
 | |
|                         ExtractChapters = false
 | |
|                     },
 | |
|                     cancellationToken).ConfigureAwait(false);
 | |
| 
 | |
|                 if (cacheFilePath != null)
 | |
|                 {
 | |
|                     Directory.CreateDirectory(Path.GetDirectoryName(cacheFilePath));
 | |
|                     await using FileStream createStream = AsyncFile.OpenWrite(cacheFilePath);
 | |
|                     await JsonSerializer.SerializeAsync(createStream, mediaInfo, _jsonOptions, cancellationToken).ConfigureAwait(false);
 | |
| 
 | |
|                     // _logger.LogDebug("Saved media info to {0}", cacheFilePath);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             var mediaStreams = mediaInfo.MediaStreams;
 | |
| 
 | |
|             if (!string.IsNullOrEmpty(cacheKey))
 | |
|             {
 | |
|                 var newList = new List<MediaStream>();
 | |
|                 newList.AddRange(mediaStreams.Where(i => i.Type == MediaStreamType.Video).Take(1));
 | |
|                 newList.AddRange(mediaStreams.Where(i => i.Type == MediaStreamType.Audio).Take(1));
 | |
| 
 | |
|                 foreach (var stream in newList)
 | |
|                 {
 | |
|                     stream.Index = -1;
 | |
|                     stream.Language = null;
 | |
|                 }
 | |
| 
 | |
|                 mediaStreams = newList;
 | |
|             }
 | |
| 
 | |
|             _logger.LogInformation("Live tv media info probe took {0} seconds", (DateTime.UtcNow - now).TotalSeconds.ToString(CultureInfo.InvariantCulture));
 | |
| 
 | |
|             mediaSource.Bitrate = mediaInfo.Bitrate;
 | |
|             mediaSource.Container = mediaInfo.Container;
 | |
|             mediaSource.Formats = mediaInfo.Formats;
 | |
|             mediaSource.MediaStreams = mediaStreams;
 | |
|             mediaSource.RunTimeTicks = mediaInfo.RunTimeTicks;
 | |
|             mediaSource.Size = mediaInfo.Size;
 | |
|             mediaSource.Timestamp = mediaInfo.Timestamp;
 | |
|             mediaSource.Video3DFormat = mediaInfo.Video3DFormat;
 | |
|             mediaSource.VideoType = mediaInfo.VideoType;
 | |
| 
 | |
|             mediaSource.DefaultSubtitleStreamIndex = null;
 | |
| 
 | |
|             // Null this out so that it will be treated like a live stream
 | |
|             if (!originalRuntime.HasValue)
 | |
|             {
 | |
|                 mediaSource.RunTimeTicks = null;
 | |
|             }
 | |
| 
 | |
|             var audioStream = mediaStreams.FirstOrDefault(i => i.Type == MediaStreamType.Audio);
 | |
| 
 | |
|             if (audioStream == null || audioStream.Index == -1)
 | |
|             {
 | |
|                 mediaSource.DefaultAudioStreamIndex = null;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 mediaSource.DefaultAudioStreamIndex = audioStream.Index;
 | |
|             }
 | |
| 
 | |
|             var videoStream = mediaStreams.FirstOrDefault(i => i.Type == MediaStreamType.Video);
 | |
|             if (videoStream != null)
 | |
|             {
 | |
|                 if (!videoStream.BitRate.HasValue)
 | |
|                 {
 | |
|                     var width = videoStream.Width ?? 1920;
 | |
| 
 | |
|                     if (width >= 3000)
 | |
|                     {
 | |
|                         videoStream.BitRate = 30000000;
 | |
|                     }
 | |
|                     else if (width >= 1900)
 | |
|                     {
 | |
|                         videoStream.BitRate = 20000000;
 | |
|                     }
 | |
|                     else if (width >= 1200)
 | |
|                     {
 | |
|                         videoStream.BitRate = 8000000;
 | |
|                     }
 | |
|                     else if (width >= 700)
 | |
|                     {
 | |
|                         videoStream.BitRate = 2000000;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // This is coming up false and preventing stream copy
 | |
|                 videoStream.IsAVC = null;
 | |
|             }
 | |
| 
 | |
|             mediaSource.AnalyzeDurationMs = 3000;
 | |
| 
 | |
|             // Try to estimate this
 | |
|             mediaSource.InferTotalBitrate(true);
 | |
|         }
 | |
| 
 | |
|         public Task AddMediaInfoWithProbe(MediaSourceInfo mediaSource, bool isAudio, bool addProbeDelay, CancellationToken cancellationToken)
 | |
|         {
 | |
|             return AddMediaInfoWithProbe(mediaSource, isAudio, null, addProbeDelay, cancellationToken);
 | |
|         }
 | |
|     }
 | |
| }
 |