AudioHlsService.cs 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. using MediaBrowser.Common.IO;
  2. using MediaBrowser.Common.MediaInfo;
  3. using MediaBrowser.Controller;
  4. using MediaBrowser.Controller.Library;
  5. using MediaBrowser.Model.Dto;
  6. using ServiceStack.ServiceHost;
  7. using System;
  8. using System.IO;
  9. namespace MediaBrowser.Api.Playback.Hls
  10. {
  11. /// <summary>
  12. /// Class GetHlsAudioStream
  13. /// </summary>
  14. [Route("/Audio/{Id}/stream.m3u8", "GET")]
  15. [Api(Description = "Gets an audio stream using HTTP live streaming.")]
  16. public class GetHlsAudioStream : StreamRequest
  17. {
  18. }
  19. /// <summary>
  20. /// Class GetHlsAudioSegment
  21. /// </summary>
  22. [Route("/Audio/{Id}/segments/{SegmentId}/stream.mp3", "GET")]
  23. [Route("/Audio/{Id}/segments/{SegmentId}/stream.aac", "GET")]
  24. [Api(Description = "Gets an Http live streaming segment file. Internal use only.")]
  25. public class GetHlsAudioSegment
  26. {
  27. /// <summary>
  28. /// Gets or sets the id.
  29. /// </summary>
  30. /// <value>The id.</value>
  31. public string Id { get; set; }
  32. /// <summary>
  33. /// Gets or sets the segment id.
  34. /// </summary>
  35. /// <value>The segment id.</value>
  36. public string SegmentId { get; set; }
  37. }
  38. /// <summary>
  39. /// Class AudioHlsService
  40. /// </summary>
  41. public class AudioHlsService : BaseHlsService
  42. {
  43. /// <summary>
  44. /// Initializes a new instance of the <see cref="AudioHlsService" /> class.
  45. /// </summary>
  46. /// <param name="appPaths">The app paths.</param>
  47. /// <param name="userManager">The user manager.</param>
  48. /// <param name="libraryManager">The library manager.</param>
  49. /// <param name="isoManager">The iso manager.</param>
  50. /// <param name="mediaEncoder">The media encoder.</param>
  51. public AudioHlsService(IServerApplicationPaths appPaths, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder)
  52. : base(appPaths, userManager, libraryManager, isoManager, mediaEncoder)
  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(GetHlsAudioSegment request)
  61. {
  62. var file = SegmentFilePrefix + request.SegmentId + Path.GetExtension(RequestContext.PathInfo);
  63. file = Path.Combine(ApplicationPaths.EncodedMediaCachePath, file);
  64. return ResultFactory.GetStaticFileResult(RequestContext, 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(GetHlsAudioStream request)
  72. {
  73. return ProcessRequest(request);
  74. }
  75. /// <summary>
  76. /// Gets the audio arguments.
  77. /// </summary>
  78. /// <param name="state">The state.</param>
  79. /// <returns>System.String.</returns>
  80. protected override string GetAudioArguments(StreamState state)
  81. {
  82. var codec = GetAudioCodec(state.Request);
  83. var args = "-codec:a " + codec;
  84. if (string.Equals(codec, "aac", StringComparison.OrdinalIgnoreCase))
  85. {
  86. args += " -strict experimental";
  87. }
  88. var channels = GetNumAudioChannelsParam(state.Request, state.AudioStream);
  89. if (channels.HasValue)
  90. {
  91. args += " -ac " + channels.Value;
  92. }
  93. if (state.Request.AudioSampleRate.HasValue)
  94. {
  95. args += " -ar " + state.Request.AudioSampleRate.Value;
  96. }
  97. if (state.Request.AudioBitRate.HasValue)
  98. {
  99. args += " -ab " + state.Request.AudioBitRate.Value;
  100. }
  101. return args;
  102. }
  103. /// <summary>
  104. /// Gets the video arguments.
  105. /// </summary>
  106. /// <param name="state">The state.</param>
  107. /// <param name="performSubtitleConversion">if set to <c>true</c> [perform subtitle conversion].</param>
  108. /// <returns>System.String.</returns>
  109. protected override string GetVideoArguments(StreamState state, bool performSubtitleConversion)
  110. {
  111. // No video
  112. return string.Empty;
  113. }
  114. /// <summary>
  115. /// Gets the segment file extension.
  116. /// </summary>
  117. /// <param name="state">The state.</param>
  118. /// <returns>System.String.</returns>
  119. /// <exception cref="System.ArgumentException">Must specify either aac or mp3 audio codec.</exception>
  120. /// <exception cref="System.InvalidOperationException">Only aac and mp3 audio codecs are supported.</exception>
  121. protected override string GetSegmentFileExtension(StreamState state)
  122. {
  123. if (state.Request.AudioCodec == AudioCodecs.Aac)
  124. {
  125. return ".aac";
  126. }
  127. if (state.Request.AudioCodec == AudioCodecs.Mp3)
  128. {
  129. return ".mp3";
  130. }
  131. throw new ArgumentException("Must specify either aac or mp3 audio codec.");
  132. }
  133. /// <summary>
  134. /// Gets the map args.
  135. /// </summary>
  136. /// <param name="state">The state.</param>
  137. /// <returns>System.String.</returns>
  138. protected override string GetMapArgs(StreamState state)
  139. {
  140. return string.Format("-map 0:{0}", state.AudioStream.Index);
  141. }
  142. }
  143. }