2
0

UniversalAudioService.cs 16 KB

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