UniversalAudioService.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Linq;
  5. using System.Threading.Tasks;
  6. using MediaBrowser.Api.Playback.Hls;
  7. using MediaBrowser.Api.Playback.Progressive;
  8. using MediaBrowser.Common.Net;
  9. using MediaBrowser.Controller.Configuration;
  10. using MediaBrowser.Controller.Devices;
  11. using MediaBrowser.Controller.Dlna;
  12. using MediaBrowser.Controller.Drawing;
  13. using MediaBrowser.Controller.Library;
  14. using MediaBrowser.Controller.MediaEncoding;
  15. using MediaBrowser.Controller.Net;
  16. using MediaBrowser.Model.Dlna;
  17. using MediaBrowser.Model.IO;
  18. using MediaBrowser.Model.MediaInfo;
  19. using MediaBrowser.Model.Serialization;
  20. using MediaBrowser.Model.Services;
  21. using MediaBrowser.Model.System;
  22. using Microsoft.Extensions.Logging;
  23. namespace MediaBrowser.Api.Playback
  24. {
  25. public class BaseUniversalRequest
  26. {
  27. /// <summary>
  28. /// Gets or sets the id.
  29. /// </summary>
  30. /// <value>The id.</value>
  31. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  32. public Guid Id { get; set; }
  33. [ApiMember(Name = "MediaSourceId", Description = "The media version id, if playing an alternate version", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  34. public string MediaSourceId { get; set; }
  35. [ApiMember(Name = "DeviceId", Description = "The device id of the client requesting. Used to stop encoding processes when needed.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  36. public string DeviceId { get; set; }
  37. public Guid UserId { get; set; }
  38. public string AudioCodec { get; set; }
  39. public string Container { get; set; }
  40. public int? MaxAudioChannels { get; set; }
  41. public int? TranscodingAudioChannels { get; set; }
  42. public long? MaxStreamingBitrate { get; set; }
  43. [ApiMember(Name = "StartTimeTicks", Description = "Optional. Specify a starting offset, in ticks. 1 tick = 10000 ms", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  44. public long? StartTimeTicks { get; set; }
  45. public string TranscodingContainer { get; set; }
  46. public string TranscodingProtocol { get; set; }
  47. public int? MaxAudioSampleRate { get; set; }
  48. public int? MaxAudioBitDepth { get; set; }
  49. public bool EnableRedirection { get; set; }
  50. public bool EnableRemoteMedia { get; set; }
  51. public bool BreakOnNonKeyFrames { get; set; }
  52. public BaseUniversalRequest()
  53. {
  54. EnableRedirection = true;
  55. }
  56. }
  57. [Route("/Audio/{Id}/universal.{Container}", "GET", Summary = "Gets an audio stream")]
  58. [Route("/Audio/{Id}/universal", "GET", Summary = "Gets an audio stream")]
  59. [Route("/Audio/{Id}/universal.{Container}", "HEAD", Summary = "Gets an audio stream")]
  60. [Route("/Audio/{Id}/universal", "HEAD", Summary = "Gets an audio stream")]
  61. public class GetUniversalAudioStream : BaseUniversalRequest
  62. {
  63. }
  64. [Authenticated]
  65. public class UniversalAudioService : BaseApiService
  66. {
  67. public UniversalAudioService(
  68. IHttpClient httpClient,
  69. IServerConfigurationManager serverConfigurationManager,
  70. IUserManager userManager,
  71. ILibraryManager libraryManager,
  72. IIsoManager isoManager,
  73. IMediaEncoder mediaEncoder,
  74. IFileSystem fileSystem,
  75. IDlnaManager dlnaManager,
  76. IDeviceManager deviceManager,
  77. ISubtitleEncoder subtitleEncoder,
  78. IMediaSourceManager mediaSourceManager,
  79. IZipClient zipClient,
  80. IJsonSerializer jsonSerializer,
  81. IAuthorizationContext authorizationContext,
  82. IImageProcessor imageProcessor,
  83. INetworkManager networkManager,
  84. IEnvironmentInfo environmentInfo,
  85. ILoggerFactory loggerFactory)
  86. {
  87. HttpClient = httpClient;
  88. ServerConfigurationManager = serverConfigurationManager;
  89. UserManager = userManager;
  90. LibraryManager = libraryManager;
  91. IsoManager = isoManager;
  92. MediaEncoder = mediaEncoder;
  93. FileSystem = fileSystem;
  94. DlnaManager = dlnaManager;
  95. DeviceManager = deviceManager;
  96. SubtitleEncoder = subtitleEncoder;
  97. MediaSourceManager = mediaSourceManager;
  98. ZipClient = zipClient;
  99. JsonSerializer = jsonSerializer;
  100. AuthorizationContext = authorizationContext;
  101. ImageProcessor = imageProcessor;
  102. NetworkManager = networkManager;
  103. EnvironmentInfo = environmentInfo;
  104. _loggerFactory = loggerFactory;
  105. _logger = loggerFactory.CreateLogger(nameof(UniversalAudioService));
  106. }
  107. protected IHttpClient HttpClient { get; private set; }
  108. protected IServerConfigurationManager ServerConfigurationManager { get; private set; }
  109. protected IUserManager UserManager { get; private set; }
  110. protected ILibraryManager LibraryManager { get; private set; }
  111. protected IIsoManager IsoManager { get; private set; }
  112. protected IMediaEncoder MediaEncoder { get; private set; }
  113. protected IFileSystem FileSystem { get; private set; }
  114. protected IDlnaManager DlnaManager { get; private set; }
  115. protected IDeviceManager DeviceManager { get; private set; }
  116. protected ISubtitleEncoder SubtitleEncoder { get; private set; }
  117. protected IMediaSourceManager MediaSourceManager { get; private set; }
  118. protected IZipClient ZipClient { get; private set; }
  119. protected IJsonSerializer JsonSerializer { get; private set; }
  120. protected IAuthorizationContext AuthorizationContext { get; private set; }
  121. protected IImageProcessor ImageProcessor { get; private set; }
  122. protected INetworkManager NetworkManager { get; private set; }
  123. protected IEnvironmentInfo EnvironmentInfo { get; private set; }
  124. private ILoggerFactory _loggerFactory;
  125. private ILogger _logger;
  126. public Task<object> Get(GetUniversalAudioStream request)
  127. {
  128. return GetUniversalStream(request, false);
  129. }
  130. public Task<object> Head(GetUniversalAudioStream request)
  131. {
  132. return GetUniversalStream(request, true);
  133. }
  134. private DeviceProfile GetDeviceProfile(GetUniversalAudioStream request)
  135. {
  136. var deviceProfile = new DeviceProfile();
  137. var directPlayProfiles = new List<DirectPlayProfile>();
  138. var containers = (request.Container ?? string.Empty).Split(new[] { ',' }, StringSplitOptions.RemoveEmptyEntries);
  139. foreach (var container in containers)
  140. {
  141. var parts = container.Split(new[] { '|' }, StringSplitOptions.RemoveEmptyEntries);
  142. var audioCodecs = parts.Length == 1 ? null : string.Join(",", parts.Skip(1).ToArray());
  143. directPlayProfiles.Add(new DirectPlayProfile
  144. {
  145. Type = DlnaProfileType.Audio,
  146. Container = parts[0],
  147. AudioCodec = audioCodecs
  148. });
  149. }
  150. deviceProfile.DirectPlayProfiles = directPlayProfiles.ToArray();
  151. deviceProfile.TranscodingProfiles = new[]
  152. {
  153. new TranscodingProfile
  154. {
  155. Type = DlnaProfileType.Audio,
  156. Context = EncodingContext.Streaming,
  157. Container = request.TranscodingContainer,
  158. AudioCodec = request.AudioCodec,
  159. Protocol = request.TranscodingProtocol,
  160. BreakOnNonKeyFrames = request.BreakOnNonKeyFrames,
  161. MaxAudioChannels = request.TranscodingAudioChannels.HasValue ? request.TranscodingAudioChannels.Value.ToString(CultureInfo.InvariantCulture) : null
  162. }
  163. };
  164. var codecProfiles = new List<CodecProfile>();
  165. var conditions = new List<ProfileCondition>();
  166. if (request.MaxAudioSampleRate.HasValue)
  167. {
  168. // codec profile
  169. conditions.Add(new ProfileCondition
  170. {
  171. Condition = ProfileConditionType.LessThanEqual,
  172. IsRequired = false,
  173. Property = ProfileConditionValue.AudioSampleRate,
  174. Value = request.MaxAudioSampleRate.Value.ToString(CultureInfo.InvariantCulture)
  175. });
  176. }
  177. if (request.MaxAudioBitDepth.HasValue)
  178. {
  179. // codec profile
  180. conditions.Add(new ProfileCondition
  181. {
  182. Condition = ProfileConditionType.LessThanEqual,
  183. IsRequired = false,
  184. Property = ProfileConditionValue.AudioBitDepth,
  185. Value = request.MaxAudioBitDepth.Value.ToString(CultureInfo.InvariantCulture)
  186. });
  187. }
  188. if (request.MaxAudioChannels.HasValue)
  189. {
  190. // codec profile
  191. conditions.Add(new ProfileCondition
  192. {
  193. Condition = ProfileConditionType.LessThanEqual,
  194. IsRequired = false,
  195. Property = ProfileConditionValue.AudioChannels,
  196. Value = request.MaxAudioChannels.Value.ToString(CultureInfo.InvariantCulture)
  197. });
  198. }
  199. if (conditions.Count > 0)
  200. {
  201. // codec profile
  202. codecProfiles.Add(new CodecProfile
  203. {
  204. Type = CodecType.Audio,
  205. Container = request.Container,
  206. Conditions = conditions.ToArray()
  207. });
  208. }
  209. deviceProfile.CodecProfiles = codecProfiles.ToArray();
  210. return deviceProfile;
  211. }
  212. private async Task<object> GetUniversalStream(GetUniversalAudioStream request, bool isHeadRequest)
  213. {
  214. var deviceProfile = GetDeviceProfile(request);
  215. AuthorizationContext.GetAuthorizationInfo(Request).DeviceId = request.DeviceId;
  216. var mediaInfoService = new MediaInfoService(MediaSourceManager, DeviceManager, LibraryManager, ServerConfigurationManager, NetworkManager, MediaEncoder, UserManager, JsonSerializer, AuthorizationContext, _loggerFactory)
  217. {
  218. Request = Request
  219. };
  220. var playbackInfoResult = await mediaInfoService.GetPlaybackInfo(new GetPostedPlaybackInfo
  221. {
  222. Id = request.Id,
  223. MaxAudioChannels = request.MaxAudioChannels,
  224. MaxStreamingBitrate = request.MaxStreamingBitrate,
  225. StartTimeTicks = request.StartTimeTicks,
  226. UserId = request.UserId,
  227. DeviceProfile = deviceProfile,
  228. MediaSourceId = request.MediaSourceId
  229. }).ConfigureAwait(false);
  230. var mediaSource = playbackInfoResult.MediaSources[0];
  231. if (mediaSource.SupportsDirectPlay && mediaSource.Protocol == MediaProtocol.Http)
  232. {
  233. if (request.EnableRedirection)
  234. {
  235. if (mediaSource.IsRemote && request.EnableRemoteMedia)
  236. {
  237. return ResultFactory.GetRedirectResult(mediaSource.Path);
  238. }
  239. }
  240. }
  241. var isStatic = mediaSource.SupportsDirectStream;
  242. if (!isStatic && string.Equals(mediaSource.TranscodingSubProtocol, "hls", StringComparison.OrdinalIgnoreCase))
  243. {
  244. var service = new DynamicHlsService(ServerConfigurationManager,
  245. UserManager,
  246. LibraryManager,
  247. IsoManager,
  248. MediaEncoder,
  249. FileSystem,
  250. DlnaManager,
  251. SubtitleEncoder,
  252. DeviceManager,
  253. MediaSourceManager,
  254. JsonSerializer,
  255. AuthorizationContext,
  256. NetworkManager)
  257. {
  258. Request = Request
  259. };
  260. var transcodingProfile = deviceProfile.TranscodingProfiles[0];
  261. var newRequest = new GetMasterHlsAudioPlaylist
  262. {
  263. AudioBitRate = isStatic ? (int?)null : Convert.ToInt32(Math.Min(request.MaxStreamingBitrate ?? 192000, int.MaxValue)),
  264. AudioCodec = transcodingProfile.AudioCodec,
  265. Container = ".m3u8",
  266. DeviceId = request.DeviceId,
  267. Id = request.Id,
  268. MaxAudioChannels = request.MaxAudioChannels,
  269. MediaSourceId = mediaSource.Id,
  270. PlaySessionId = playbackInfoResult.PlaySessionId,
  271. StartTimeTicks = request.StartTimeTicks,
  272. Static = isStatic,
  273. SegmentContainer = request.TranscodingContainer,
  274. AudioSampleRate = request.MaxAudioSampleRate,
  275. MaxAudioBitDepth = request.MaxAudioBitDepth,
  276. BreakOnNonKeyFrames = transcodingProfile.BreakOnNonKeyFrames,
  277. TranscodeReasons = mediaSource.TranscodeReasons == null ? null : string.Join(",", mediaSource.TranscodeReasons.Select(i => i.ToString()).ToArray())
  278. };
  279. if (isHeadRequest)
  280. {
  281. return await service.Head(newRequest).ConfigureAwait(false);
  282. }
  283. return await service.Get(newRequest).ConfigureAwait(false);
  284. }
  285. else
  286. {
  287. var service = new AudioService(HttpClient,
  288. ServerConfigurationManager,
  289. UserManager,
  290. LibraryManager,
  291. IsoManager,
  292. MediaEncoder,
  293. FileSystem,
  294. DlnaManager,
  295. SubtitleEncoder,
  296. DeviceManager,
  297. MediaSourceManager,
  298. JsonSerializer,
  299. AuthorizationContext,
  300. EnvironmentInfo)
  301. {
  302. Request = Request
  303. };
  304. var newRequest = new GetAudioStream
  305. {
  306. AudioBitRate = isStatic ? (int?)null : Convert.ToInt32(Math.Min(request.MaxStreamingBitrate ?? 192000, int.MaxValue)),
  307. AudioCodec = request.AudioCodec,
  308. Container = isStatic ? null : ("." + mediaSource.TranscodingContainer),
  309. DeviceId = request.DeviceId,
  310. Id = request.Id,
  311. MaxAudioChannels = request.MaxAudioChannels,
  312. MediaSourceId = mediaSource.Id,
  313. PlaySessionId = playbackInfoResult.PlaySessionId,
  314. StartTimeTicks = request.StartTimeTicks,
  315. Static = isStatic,
  316. AudioSampleRate = request.MaxAudioSampleRate,
  317. MaxAudioBitDepth = request.MaxAudioBitDepth,
  318. TranscodeReasons = mediaSource.TranscodeReasons == null ? null : string.Join(",", mediaSource.TranscodeReasons.Select(i => i.ToString()).ToArray())
  319. };
  320. if (isHeadRequest)
  321. {
  322. return await service.Head(newRequest).ConfigureAwait(false);
  323. }
  324. return await service.Get(newRequest).ConfigureAwait(false);
  325. }
  326. }
  327. }
  328. }