VideoHlsService.cs 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  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.MediaEncoding;
  7. using MediaBrowser.Model.IO;
  8. using MediaBrowser.Model.Serialization;
  9. using ServiceStack;
  10. using System;
  11. namespace MediaBrowser.Api.Playback.Hls
  12. {
  13. /// <summary>
  14. /// Class GetHlsVideoStream
  15. /// </summary>
  16. [Route("/Videos/{Id}/stream.m3u8", "GET")]
  17. [Api(Description = "Gets a video stream using HTTP live streaming.")]
  18. public class GetHlsVideoStream : VideoStreamRequest
  19. {
  20. // TODO: Deprecate with new iOS app
  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 VideoHlsService
  35. /// </summary>
  36. public class VideoHlsService : BaseHlsService
  37. {
  38. public VideoHlsService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, IDlnaManager dlnaManager, ISubtitleEncoder subtitleEncoder, IDeviceManager deviceManager, IMediaSourceManager mediaSourceManager, IZipClient zipClient, IJsonSerializer jsonSerializer) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, fileSystem, dlnaManager, subtitleEncoder, deviceManager, mediaSourceManager, zipClient, jsonSerializer)
  39. {
  40. }
  41. /// <summary>
  42. /// Gets the specified request.
  43. /// </summary>
  44. /// <param name="request">The request.</param>
  45. /// <returns>System.Object.</returns>
  46. public object Get(GetHlsVideoStream request)
  47. {
  48. return ProcessRequest(request, false);
  49. }
  50. public object Get(GetLiveHlsStream request)
  51. {
  52. return ProcessRequest(request, true);
  53. }
  54. /// <summary>
  55. /// Gets the audio arguments.
  56. /// </summary>
  57. /// <param name="state">The state.</param>
  58. /// <returns>System.String.</returns>
  59. protected override string GetAudioArguments(StreamState state)
  60. {
  61. var codec = state.OutputAudioCodec;
  62. if (codec.Equals("copy", StringComparison.OrdinalIgnoreCase))
  63. {
  64. return "-codec:a:0 copy";
  65. }
  66. var args = "-codec:a:0 " + codec;
  67. var channels = state.OutputAudioChannels;
  68. if (channels.HasValue)
  69. {
  70. args += " -ac " + channels.Value;
  71. }
  72. var bitrate = state.OutputAudioBitrate;
  73. if (bitrate.HasValue)
  74. {
  75. args += " -ab " + bitrate.Value.ToString(UsCulture);
  76. }
  77. args += " " + GetAudioFilterParam(state, true);
  78. return args;
  79. }
  80. /// <summary>
  81. /// Gets the video arguments.
  82. /// </summary>
  83. /// <param name="state">The state.</param>
  84. /// <returns>System.String.</returns>
  85. protected override string GetVideoArguments(StreamState state)
  86. {
  87. var codec = state.OutputVideoCodec;
  88. var args = "-codec:v:0 " + codec;
  89. if (state.EnableMpegtsM2TsMode)
  90. {
  91. args += " -mpegts_m2ts_mode 1";
  92. }
  93. // See if we can save come cpu cycles by avoiding encoding
  94. if (codec.Equals("copy", StringComparison.OrdinalIgnoreCase))
  95. {
  96. return state.VideoStream != null && IsH264(state.VideoStream) ?
  97. args + " -bsf:v h264_mp4toannexb" :
  98. args;
  99. }
  100. var keyFrameArg = string.Format(" -force_key_frames expr:gte(t,n_forced*{0})",
  101. state.SegmentLength.ToString(UsCulture));
  102. var hasGraphicalSubs = state.SubtitleStream != null && !state.SubtitleStream.IsTextSubtitleStream;
  103. args += " " + GetVideoQualityParam(state, H264Encoder, true) + keyFrameArg;
  104. // Add resolution params, if specified
  105. if (!hasGraphicalSubs)
  106. {
  107. args += GetOutputSizeParam(state, codec);
  108. }
  109. // This is for internal graphical subs
  110. if (hasGraphicalSubs)
  111. {
  112. args += GetGraphicalSubtitleParam(state, codec);
  113. }
  114. return args;
  115. }
  116. /// <summary>
  117. /// Gets the segment file extension.
  118. /// </summary>
  119. /// <param name="state">The state.</param>
  120. /// <returns>System.String.</returns>
  121. protected override string GetSegmentFileExtension(StreamState state)
  122. {
  123. return ".ts";
  124. }
  125. }
  126. }