VideoHlsService.cs 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. using MediaBrowser.Common.IO;
  2. using MediaBrowser.Controller.Configuration;
  3. using MediaBrowser.Controller.Devices;
  4. using MediaBrowser.Controller.Dlna;
  5. using MediaBrowser.Controller.Library;
  6. using MediaBrowser.Controller.LiveTv;
  7. using MediaBrowser.Controller.MediaEncoding;
  8. using MediaBrowser.Model.IO;
  9. using ServiceStack;
  10. using System;
  11. using System.IO;
  12. namespace MediaBrowser.Api.Playback.Hls
  13. {
  14. /// <summary>
  15. /// Class GetHlsVideoStream
  16. /// </summary>
  17. [Route("/Videos/{Id}/stream.m3u8", "GET")]
  18. [Api(Description = "Gets a video stream using HTTP live streaming.")]
  19. public class GetHlsVideoStream : VideoStreamRequest
  20. {
  21. [ApiMember(Name = "BaselineStreamAudioBitRate", Description = "Optional. Specify the audio bitrate for the baseline stream.", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  22. public int? BaselineStreamAudioBitRate { get; set; }
  23. [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")]
  24. public bool AppendBaselineStream { get; set; }
  25. [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")]
  26. public int TimeStampOffsetMs { get; set; }
  27. }
  28. [Route("/Videos/{Id}/live.m3u8", "GET")]
  29. [Api(Description = "Gets a video stream using HTTP live streaming.")]
  30. public class GetLiveHlsStream : VideoStreamRequest
  31. {
  32. }
  33. /// <summary>
  34. /// Class GetHlsVideoSegment
  35. /// </summary>
  36. [Route("/Videos/{Id}/hls/{PlaylistId}/{SegmentId}.ts", "GET")]
  37. [Api(Description = "Gets an Http live streaming segment file. Internal use only.")]
  38. public class GetHlsVideoSegment : VideoStreamRequest
  39. {
  40. public string PlaylistId { get; set; }
  41. /// <summary>
  42. /// Gets or sets the segment id.
  43. /// </summary>
  44. /// <value>The segment id.</value>
  45. public string SegmentId { get; set; }
  46. }
  47. /// <summary>
  48. /// Class VideoHlsService
  49. /// </summary>
  50. public class VideoHlsService : BaseHlsService
  51. {
  52. public VideoHlsService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, ILiveTvManager liveTvManager, IDlnaManager dlnaManager, ISubtitleEncoder subtitleEncoder, IDeviceManager deviceManager, IMediaSourceManager mediaSourceManager, IZipClient zipClient) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, fileSystem, liveTvManager, dlnaManager, subtitleEncoder, deviceManager, mediaSourceManager, zipClient)
  53. {
  54. }
  55. /// <summary>
  56. /// Gets the specified request.
  57. /// </summary>
  58. /// <param name="request">The request.</param>
  59. /// <returns>System.Object.</returns>
  60. public object Get(GetHlsVideoSegment request)
  61. {
  62. var file = request.SegmentId + Path.GetExtension(Request.PathInfo);
  63. file = Path.Combine(ServerConfigurationManager.ApplicationPaths.TranscodingTempPath, file);
  64. return ResultFactory.GetStaticFileResult(Request, file);
  65. }
  66. /// <summary>
  67. /// Gets the specified request.
  68. /// </summary>
  69. /// <param name="request">The request.</param>
  70. /// <returns>System.Object.</returns>
  71. public object Get(GetHlsVideoStream request)
  72. {
  73. return ProcessRequest(request, false);
  74. }
  75. public object Get(GetLiveHlsStream request)
  76. {
  77. return ProcessRequest(request, true);
  78. }
  79. /// <summary>
  80. /// Gets the audio arguments.
  81. /// </summary>
  82. /// <param name="state">The state.</param>
  83. /// <returns>System.String.</returns>
  84. protected override string GetAudioArguments(StreamState state)
  85. {
  86. var codec = state.OutputAudioCodec;
  87. if (codec.Equals("copy", StringComparison.OrdinalIgnoreCase))
  88. {
  89. return "-codec:a:0 copy";
  90. }
  91. var args = "-codec:a:0 " + codec;
  92. var channels = state.OutputAudioChannels;
  93. if (channels.HasValue)
  94. {
  95. args += " -ac " + channels.Value;
  96. }
  97. var bitrate = state.OutputAudioBitrate;
  98. if (bitrate.HasValue)
  99. {
  100. args += " -ab " + bitrate.Value.ToString(UsCulture);
  101. }
  102. args += " " + GetAudioFilterParam(state, true);
  103. return args;
  104. }
  105. /// <summary>
  106. /// Gets the video arguments.
  107. /// </summary>
  108. /// <param name="state">The state.</param>
  109. /// <returns>System.String.</returns>
  110. protected override string GetVideoArguments(StreamState state)
  111. {
  112. var codec = state.OutputVideoCodec;
  113. var args = "-codec:v:0 " + codec;
  114. if (state.EnableMpegtsM2TsMode)
  115. {
  116. args += " -mpegts_m2ts_mode 1";
  117. }
  118. // See if we can save come cpu cycles by avoiding encoding
  119. if (codec.Equals("copy", StringComparison.OrdinalIgnoreCase))
  120. {
  121. return state.VideoStream != null && IsH264(state.VideoStream) ?
  122. args + " -bsf:v h264_mp4toannexb" :
  123. args;
  124. }
  125. var keyFrameArg = string.Format(" -force_key_frames expr:gte(t,n_forced*{0})",
  126. state.SegmentLength.ToString(UsCulture));
  127. var hasGraphicalSubs = state.SubtitleStream != null && !state.SubtitleStream.IsTextSubtitleStream;
  128. args += " " + GetVideoQualityParam(state, H264Encoder, true) + keyFrameArg;
  129. // Add resolution params, if specified
  130. if (!hasGraphicalSubs)
  131. {
  132. args += GetOutputSizeParam(state, codec);
  133. }
  134. // This is for internal graphical subs
  135. if (hasGraphicalSubs)
  136. {
  137. args += GetGraphicalSubtitleParam(state, codec);
  138. }
  139. return args;
  140. }
  141. /// <summary>
  142. /// Gets the segment file extension.
  143. /// </summary>
  144. /// <param name="state">The state.</param>
  145. /// <returns>System.String.</returns>
  146. protected override string GetSegmentFileExtension(StreamState state)
  147. {
  148. return ".ts";
  149. }
  150. }
  151. }