VideoHlsService.cs 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. using System.Threading;
  2. using MediaBrowser.Common.IO;
  3. using MediaBrowser.Common.Net;
  4. using MediaBrowser.Controller.Channels;
  5. using MediaBrowser.Controller.Configuration;
  6. using MediaBrowser.Controller.Dlna;
  7. using MediaBrowser.Controller.Dto;
  8. using MediaBrowser.Controller.Library;
  9. using MediaBrowser.Controller.LiveTv;
  10. using MediaBrowser.Controller.MediaEncoding;
  11. using MediaBrowser.Controller.Persistence;
  12. using MediaBrowser.Model.IO;
  13. using ServiceStack;
  14. using System;
  15. using System.IO;
  16. using System.Linq;
  17. using System.Threading.Tasks;
  18. namespace MediaBrowser.Api.Playback.Hls
  19. {
  20. /// <summary>
  21. /// Class GetHlsVideoStream
  22. /// </summary>
  23. [Route("/Videos/{Id}/stream.m3u8", "GET")]
  24. [Api(Description = "Gets a video stream using HTTP live streaming.")]
  25. public class GetHlsVideoStream : VideoStreamRequest
  26. {
  27. [ApiMember(Name = "BaselineStreamAudioBitRate", Description = "Optional. Specify the audio bitrate for the baseline stream.", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  28. public int? BaselineStreamAudioBitRate { get; set; }
  29. [ApiMember(Name = "AppendBaselineStream", Description = "Optional. Whether or not to include a baseline audio-only stream in the master playlist.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  30. public bool AppendBaselineStream { get; set; }
  31. [ApiMember(Name = "TimeStampOffsetMs", Description = "Optional. Alter the timestamps in the playlist by a given amount, in ms. Default is 1000.", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  32. public int TimeStampOffsetMs { get; set; }
  33. }
  34. /// <summary>
  35. /// Class GetHlsVideoSegment
  36. /// </summary>
  37. [Route("/Videos/{Id}/hls/{PlaylistId}/{SegmentId}.ts", "GET")]
  38. [Api(Description = "Gets an Http live streaming segment file. Internal use only.")]
  39. public class GetHlsVideoSegment : VideoStreamRequest
  40. {
  41. public string PlaylistId { get; set; }
  42. /// <summary>
  43. /// Gets or sets the segment id.
  44. /// </summary>
  45. /// <value>The segment id.</value>
  46. public string SegmentId { get; set; }
  47. }
  48. /// <summary>
  49. /// Class VideoHlsService
  50. /// </summary>
  51. public class VideoHlsService : BaseHlsService
  52. {
  53. public VideoHlsService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IDtoService dtoService, IFileSystem fileSystem, IItemRepository itemRepository, ILiveTvManager liveTvManager, IEncodingManager encodingManager, IDlnaManager dlnaManager, IChannelManager channelManager, IHttpClient httpClient) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, dtoService, fileSystem, itemRepository, liveTvManager, encodingManager, dlnaManager, channelManager, httpClient)
  54. {
  55. }
  56. /// <summary>
  57. /// Gets the specified request.
  58. /// </summary>
  59. /// <param name="request">The request.</param>
  60. /// <returns>System.Object.</returns>
  61. public object Get(GetHlsVideoSegment request)
  62. {
  63. var file = request.SegmentId + Path.GetExtension(Request.PathInfo);
  64. file = Path.Combine(ServerConfigurationManager.ApplicationPaths.TranscodingTempPath, file);
  65. OnBeginRequest(request.PlaylistId);
  66. return ResultFactory.GetStaticFileResult(Request, file);
  67. }
  68. /// <summary>
  69. /// Called when [begin request].
  70. /// </summary>
  71. /// <param name="playlistId">The playlist id.</param>
  72. protected void OnBeginRequest(string playlistId)
  73. {
  74. var normalizedPlaylistId = playlistId.Replace("-low", string.Empty);
  75. foreach (var playlist in Directory.EnumerateFiles(ServerConfigurationManager.ApplicationPaths.TranscodingTempPath, "*.m3u8")
  76. .Where(i => i.IndexOf(normalizedPlaylistId, StringComparison.OrdinalIgnoreCase) != -1)
  77. .ToList())
  78. {
  79. ExtendPlaylistTimer(playlist);
  80. }
  81. }
  82. private async void ExtendPlaylistTimer(string playlist)
  83. {
  84. ApiEntryPoint.Instance.OnTranscodeBeginRequest(playlist, TranscodingJobType.Hls);
  85. await Task.Delay(20000).ConfigureAwait(false);
  86. ApiEntryPoint.Instance.OnTranscodeEndRequest(playlist, TranscodingJobType.Hls);
  87. }
  88. /// <summary>
  89. /// Gets the specified request.
  90. /// </summary>
  91. /// <param name="request">The request.</param>
  92. /// <returns>System.Object.</returns>
  93. public object Get(GetHlsVideoStream request)
  94. {
  95. return ProcessRequest(request);
  96. }
  97. /// <summary>
  98. /// Gets the audio arguments.
  99. /// </summary>
  100. /// <param name="state">The state.</param>
  101. /// <returns>System.String.</returns>
  102. protected override string GetAudioArguments(StreamState state)
  103. {
  104. var codec = state.OutputAudioCodec;
  105. if (codec.Equals("copy", StringComparison.OrdinalIgnoreCase))
  106. {
  107. return "-codec:a:0 copy";
  108. }
  109. var args = "-codec:a:0 " + codec;
  110. if (state.AudioStream != null)
  111. {
  112. var channels = state.OutputAudioChannels;
  113. if (channels.HasValue)
  114. {
  115. args += " -ac " + channels.Value;
  116. }
  117. var bitrate = state.OutputAudioBitrate;
  118. if (bitrate.HasValue)
  119. {
  120. args += " -ab " + bitrate.Value.ToString(UsCulture);
  121. }
  122. args += " " + GetAudioFilterParam(state, true);
  123. return args;
  124. }
  125. return args;
  126. }
  127. /// <summary>
  128. /// Gets the video arguments.
  129. /// </summary>
  130. /// <param name="state">The state.</param>
  131. /// <param name="performSubtitleConversion">if set to <c>true</c> [perform subtitle conversion].</param>
  132. /// <returns>System.String.</returns>
  133. protected override string GetVideoArguments(StreamState state,
  134. bool performSubtitleConversion)
  135. {
  136. var codec = state.OutputVideoCodec;
  137. // See if we can save come cpu cycles by avoiding encoding
  138. if (codec.Equals("copy", StringComparison.OrdinalIgnoreCase))
  139. {
  140. return IsH264(state.VideoStream) ? "-codec:v:0 copy -bsf h264_mp4toannexb" : "-codec:v:0 copy";
  141. }
  142. var keyFrameArg = state.ReadInputAtNativeFramerate ?
  143. " -force_key_frames expr:if(isnan(prev_forced_t),gte(t,.1),gte(t,prev_forced_t+1))" :
  144. " -force_key_frames expr:if(isnan(prev_forced_t),gte(t,.1),gte(t,prev_forced_t+5))";
  145. var hasGraphicalSubs = state.SubtitleStream != null && state.SubtitleStream.IsGraphicalSubtitleStream;
  146. var args = "-codec:v:0 " + codec + " " + GetVideoQualityParam(state, "libx264", true) + keyFrameArg;
  147. // Add resolution params, if specified
  148. if (!hasGraphicalSubs)
  149. {
  150. if (state.VideoRequest.Width.HasValue || state.VideoRequest.Height.HasValue || state.VideoRequest.MaxHeight.HasValue || state.VideoRequest.MaxWidth.HasValue)
  151. {
  152. args += GetOutputSizeParam(state, codec, performSubtitleConversion, CancellationToken.None);
  153. }
  154. }
  155. // This is for internal graphical subs
  156. if (hasGraphicalSubs)
  157. {
  158. args += GetInternalGraphicalSubtitleParam(state, codec);
  159. }
  160. return args;
  161. }
  162. /// <summary>
  163. /// Gets the segment file extension.
  164. /// </summary>
  165. /// <param name="state">The state.</param>
  166. /// <returns>System.String.</returns>
  167. protected override string GetSegmentFileExtension(StreamState state)
  168. {
  169. return ".ts";
  170. }
  171. }
  172. }