MpegDashService.cs 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. using MediaBrowser.Common.IO;
  2. using MediaBrowser.Common.Net;
  3. using MediaBrowser.Controller.Channels;
  4. using MediaBrowser.Controller.Configuration;
  5. using MediaBrowser.Controller.Dlna;
  6. using MediaBrowser.Controller.Library;
  7. using MediaBrowser.Controller.LiveTv;
  8. using MediaBrowser.Controller.MediaEncoding;
  9. using MediaBrowser.Model.IO;
  10. using ServiceStack;
  11. using System;
  12. using System.Collections.Generic;
  13. using System.Globalization;
  14. using System.Text;
  15. using System.Threading;
  16. using System.Threading.Tasks;
  17. namespace MediaBrowser.Api.Playback.Hls
  18. {
  19. /// <summary>
  20. /// Options is needed for chromecast. Threw Head in there since it's related
  21. /// </summary>
  22. [Route("/Videos/{Id}/master.mpd", "GET", Summary = "Gets a video stream using Mpeg dash.")]
  23. [Route("/Videos/{Id}/master.mpd", "HEAD", Summary = "Gets a video stream using Mpeg dash.")]
  24. public class GetMasterManifest : VideoStreamRequest
  25. {
  26. public bool EnableAdaptiveBitrateStreaming { get; set; }
  27. public GetMasterManifest()
  28. {
  29. EnableAdaptiveBitrateStreaming = true;
  30. }
  31. }
  32. [Route("/Videos/{Id}/dash/{SegmentId}.ts", "GET")]
  33. public class GetDashSegment : VideoStreamRequest
  34. {
  35. /// <summary>
  36. /// Gets or sets the segment id.
  37. /// </summary>
  38. /// <value>The segment id.</value>
  39. public string SegmentId { get; set; }
  40. }
  41. public class MpegDashService : BaseHlsService
  42. {
  43. protected INetworkManager NetworkManager { get; private set; }
  44. public MpegDashService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, ILiveTvManager liveTvManager, IDlnaManager dlnaManager, IChannelManager channelManager, ISubtitleEncoder subtitleEncoder, INetworkManager networkManager)
  45. : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, fileSystem, liveTvManager, dlnaManager, channelManager, subtitleEncoder)
  46. {
  47. NetworkManager = networkManager;
  48. }
  49. public object Get(GetMasterManifest request)
  50. {
  51. var result = GetAsync(request, "GET").Result;
  52. return result;
  53. }
  54. public object Head(GetMasterManifest request)
  55. {
  56. var result = GetAsync(request, "HEAD").Result;
  57. return result;
  58. }
  59. private async Task<object> GetAsync(GetMasterManifest request, string method)
  60. {
  61. if (string.Equals(request.AudioCodec, "copy", StringComparison.OrdinalIgnoreCase))
  62. {
  63. throw new ArgumentException("Audio codec copy is not allowed here.");
  64. }
  65. if (string.Equals(request.VideoCodec, "copy", StringComparison.OrdinalIgnoreCase))
  66. {
  67. throw new ArgumentException("Video codec copy is not allowed here.");
  68. }
  69. if (string.IsNullOrEmpty(request.MediaSourceId))
  70. {
  71. throw new ArgumentException("MediaSourceId is required");
  72. }
  73. var state = await GetState(request, CancellationToken.None).ConfigureAwait(false);
  74. var playlistText = string.Empty;
  75. if (string.Equals(method, "GET", StringComparison.OrdinalIgnoreCase))
  76. {
  77. playlistText = GetManifestText(state);
  78. }
  79. return ResultFactory.GetResult(playlistText, Common.Net.MimeTypes.GetMimeType("playlist.mpd"), new Dictionary<string, string>());
  80. }
  81. private string GetManifestText(StreamState state)
  82. {
  83. var audioBitrate = state.OutputAudioBitrate ?? 0;
  84. var videoBitrate = state.OutputVideoBitrate ?? 0;
  85. var builder = new StringBuilder();
  86. var duration = "PT0H02M11.00S";
  87. builder.AppendFormat(
  88. "<MPD type=\"static\" minBufferTime=\"PT2S\" mediaPresentationDuration=\"{0}\" profiles=\"urn:mpeg:dash:profile:mp2t-simple:2011\" xmlns=\"urn:mpeg:DASH:schema:MPD:2011\" maxSegmentDuration=\"PT{1}S\">",
  89. duration,
  90. state.SegmentLength.ToString(CultureInfo.InvariantCulture));
  91. builder.Append("<ProgramInformation moreInformationURL=\"http://gpac.sourceforge.net\">");
  92. builder.Append("</ProgramInformation>");
  93. builder.AppendFormat("<Period start=\"PT0S\" duration=\"{0}\">", duration);
  94. builder.Append("<AdaptationSet segmentAlignment=\"true\">");
  95. builder.Append("<ContentComponent id=\"1\" contentType=\"video\"/>");
  96. builder.Append("<ContentComponent id=\"2\" contentType=\"audio\" lang=\"eng\"/>");
  97. builder.Append(GetRepresentationOpenElement(state));
  98. AppendSegmentList(state, builder);
  99. builder.Append("</Representation>");
  100. builder.Append("</AdaptationSet>");
  101. builder.Append("</Period>");
  102. builder.Append("</MPD>");
  103. return builder.ToString();
  104. }
  105. private string GetRepresentationOpenElement(StreamState state)
  106. {
  107. return
  108. "<Representation id=\"1\" mimeType=\"video/mp2t\" codecs=\"avc1.640028,mp4a.40.02\" width=\"1280\" height=\"1024\" sampleRate=\"44100\" numChannels=\"2\" lang=\"und\" startWithSAP=\"1\" bandwidth=\"317599\">";
  109. }
  110. public object Get(GetDashSegment request)
  111. {
  112. return null;
  113. }
  114. private void AppendSegmentList(StreamState state, StringBuilder builder)
  115. {
  116. var seconds = TimeSpan.FromTicks(state.RunTimeTicks ?? 0).TotalSeconds;
  117. builder.Append("<SegmentList timescale=\"1000\" duration=\"10000\">");
  118. var queryStringIndex = Request.RawUrl.IndexOf('?');
  119. var queryString = queryStringIndex == -1 ? string.Empty : Request.RawUrl.Substring(queryStringIndex);
  120. var index = 0;
  121. while (seconds > 0)
  122. {
  123. builder.AppendFormat("<SegmentURL media=\"{0}.ts{1}\"/>", index.ToString(UsCulture), queryString);
  124. seconds -= state.SegmentLength;
  125. index++;
  126. }
  127. builder.Append("</SegmentList>");
  128. }
  129. protected override string GetAudioArguments(StreamState state)
  130. {
  131. var codec = state.OutputAudioCodec;
  132. if (codec.Equals("copy", StringComparison.OrdinalIgnoreCase))
  133. {
  134. return "-codec:a:0 copy";
  135. }
  136. var args = "-codec:a:0 " + codec;
  137. var channels = state.OutputAudioChannels;
  138. if (channels.HasValue)
  139. {
  140. args += " -ac " + channels.Value;
  141. }
  142. var bitrate = state.OutputAudioBitrate;
  143. if (bitrate.HasValue)
  144. {
  145. args += " -ab " + bitrate.Value.ToString(UsCulture);
  146. }
  147. args += " " + GetAudioFilterParam(state, true);
  148. return args;
  149. }
  150. protected override string GetVideoArguments(StreamState state)
  151. {
  152. var codec = state.OutputVideoCodec;
  153. // See if we can save come cpu cycles by avoiding encoding
  154. if (codec.Equals("copy", StringComparison.OrdinalIgnoreCase))
  155. {
  156. return IsH264(state.VideoStream) ? "-codec:v:0 copy -bsf:v h264_mp4toannexb" : "-codec:v:0 copy";
  157. }
  158. var keyFrameArg = string.Format(" -force_key_frames expr:gte(t,n_forced*{0})",
  159. state.SegmentLength.ToString(UsCulture));
  160. var hasGraphicalSubs = state.SubtitleStream != null && !state.SubtitleStream.IsTextSubtitleStream;
  161. var args = "-codec:v:0 " + codec + " " + GetVideoQualityParam(state, "libx264", true) + keyFrameArg;
  162. // Add resolution params, if specified
  163. if (!hasGraphicalSubs)
  164. {
  165. args += GetOutputSizeParam(state, codec, false);
  166. }
  167. // This is for internal graphical subs
  168. if (hasGraphicalSubs)
  169. {
  170. args += GetGraphicalSubtitleParam(state, codec);
  171. }
  172. return args;
  173. }
  174. protected override string GetCommandLineArguments(string outputPath, string transcodingJobId, StreamState state, bool isEncoding)
  175. {
  176. var threads = GetNumberOfThreads(state, false);
  177. var inputModifier = GetInputModifier(state);
  178. // If isEncoding is true we're actually starting ffmpeg
  179. var startNumberParam = isEncoding ? GetStartNumber(state).ToString(UsCulture) : "0";
  180. var args = string.Format("{0} -i {1} -map_metadata -1 -threads {2} {3} {4} -copyts -flags -global_header {5} -hls_time {6} -start_number {7} -hls_list_size {8} -y \"{9}\"",
  181. inputModifier,
  182. GetInputArgument(transcodingJobId, state),
  183. threads,
  184. GetMapArgs(state),
  185. GetVideoArguments(state),
  186. GetAudioArguments(state),
  187. state.SegmentLength.ToString(UsCulture),
  188. startNumberParam,
  189. state.HlsListSize.ToString(UsCulture),
  190. outputPath
  191. ).Trim();
  192. return args;
  193. }
  194. /// <summary>
  195. /// Gets the segment file extension.
  196. /// </summary>
  197. /// <param name="state">The state.</param>
  198. /// <returns>System.String.</returns>
  199. protected override string GetSegmentFileExtension(StreamState state)
  200. {
  201. return ".ts";
  202. }
  203. protected override TranscodingJobType TranscodingJobType
  204. {
  205. get
  206. {
  207. return TranscodingJobType.Dash;
  208. }
  209. }
  210. }
  211. }