StreamingHelpers.cs 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.IO;
  5. using System.Linq;
  6. using System.Threading;
  7. using System.Threading.Tasks;
  8. using Jellyfin.Api.Models.StreamingDtos;
  9. using Jellyfin.Extensions;
  10. using MediaBrowser.Common.Configuration;
  11. using MediaBrowser.Common.Extensions;
  12. using MediaBrowser.Controller.Configuration;
  13. using MediaBrowser.Controller.Devices;
  14. using MediaBrowser.Controller.Dlna;
  15. using MediaBrowser.Controller.Library;
  16. using MediaBrowser.Controller.MediaEncoding;
  17. using MediaBrowser.Controller.Net;
  18. using MediaBrowser.Model.Dlna;
  19. using MediaBrowser.Model.Dto;
  20. using MediaBrowser.Model.Entities;
  21. using Microsoft.AspNetCore.Http;
  22. using Microsoft.Extensions.Primitives;
  23. using Microsoft.Net.Http.Headers;
  24. namespace Jellyfin.Api.Helpers
  25. {
  26. /// <summary>
  27. /// The streaming helpers.
  28. /// </summary>
  29. public static class StreamingHelpers
  30. {
  31. /// <summary>
  32. /// Gets the current streaming state.
  33. /// </summary>
  34. /// <param name="streamingRequest">The <see cref="StreamingRequestDto"/>.</param>
  35. /// <param name="httpRequest">The <see cref="HttpRequest"/>.</param>
  36. /// <param name="authorizationContext">Instance of the <see cref="IAuthorizationContext"/> interface.</param>
  37. /// <param name="mediaSourceManager">Instance of the <see cref="IMediaSourceManager"/> interface.</param>
  38. /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param>
  39. /// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>
  40. /// <param name="serverConfigurationManager">Instance of the <see cref="IServerConfigurationManager"/> interface.</param>
  41. /// <param name="mediaEncoder">Instance of the <see cref="IMediaEncoder"/> interface.</param>
  42. /// <param name="encodingHelper">Instance of <see cref="EncodingHelper"/>.</param>
  43. /// <param name="dlnaManager">Instance of the <see cref="IDlnaManager"/> interface.</param>
  44. /// <param name="deviceManager">Instance of the <see cref="IDeviceManager"/> interface.</param>
  45. /// <param name="transcodingJobHelper">Initialized <see cref="TranscodingJobHelper"/>.</param>
  46. /// <param name="transcodingJobType">The <see cref="TranscodingJobType"/>.</param>
  47. /// <param name="cancellationToken">The <see cref="CancellationToken"/>.</param>
  48. /// <returns>A <see cref="Task"/> containing the current <see cref="StreamState"/>.</returns>
  49. public static async Task<StreamState> GetStreamingState(
  50. StreamingRequestDto streamingRequest,
  51. HttpRequest httpRequest,
  52. IAuthorizationContext authorizationContext,
  53. IMediaSourceManager mediaSourceManager,
  54. IUserManager userManager,
  55. ILibraryManager libraryManager,
  56. IServerConfigurationManager serverConfigurationManager,
  57. IMediaEncoder mediaEncoder,
  58. EncodingHelper encodingHelper,
  59. IDlnaManager dlnaManager,
  60. IDeviceManager deviceManager,
  61. TranscodingJobHelper transcodingJobHelper,
  62. TranscodingJobType transcodingJobType,
  63. CancellationToken cancellationToken)
  64. {
  65. // Parse the DLNA time seek header
  66. if (!streamingRequest.StartTimeTicks.HasValue)
  67. {
  68. var timeSeek = httpRequest.Headers["TimeSeekRange.dlna.org"];
  69. streamingRequest.StartTimeTicks = ParseTimeSeekHeader(timeSeek.ToString());
  70. }
  71. if (!string.IsNullOrWhiteSpace(streamingRequest.Params))
  72. {
  73. ParseParams(streamingRequest);
  74. }
  75. streamingRequest.StreamOptions = ParseStreamOptions(httpRequest.Query);
  76. if (httpRequest.Path.Value == null)
  77. {
  78. throw new ResourceNotFoundException(nameof(httpRequest.Path));
  79. }
  80. var url = httpRequest.Path.Value.AsSpan().RightPart('.').ToString();
  81. if (string.IsNullOrEmpty(streamingRequest.AudioCodec))
  82. {
  83. streamingRequest.AudioCodec = encodingHelper.InferAudioCodec(url);
  84. }
  85. var enableDlnaHeaders = !string.IsNullOrWhiteSpace(streamingRequest.Params) ||
  86. streamingRequest.StreamOptions.ContainsKey("dlnaheaders") ||
  87. string.Equals(httpRequest.Headers["GetContentFeatures.DLNA.ORG"], "1", StringComparison.OrdinalIgnoreCase);
  88. var state = new StreamState(mediaSourceManager, transcodingJobType, transcodingJobHelper)
  89. {
  90. Request = streamingRequest,
  91. RequestedUrl = url,
  92. UserAgent = httpRequest.Headers[HeaderNames.UserAgent],
  93. EnableDlnaHeaders = enableDlnaHeaders
  94. };
  95. var auth = await authorizationContext.GetAuthorizationInfo(httpRequest).ConfigureAwait(false);
  96. if (!auth.UserId.Equals(default))
  97. {
  98. state.User = userManager.GetUserById(auth.UserId);
  99. }
  100. if (state.IsVideoRequest && !string.IsNullOrWhiteSpace(state.Request.VideoCodec))
  101. {
  102. state.SupportedVideoCodecs = state.Request.VideoCodec.Split(',', StringSplitOptions.RemoveEmptyEntries);
  103. state.Request.VideoCodec = state.SupportedVideoCodecs.FirstOrDefault();
  104. }
  105. if (!string.IsNullOrWhiteSpace(streamingRequest.AudioCodec))
  106. {
  107. state.SupportedAudioCodecs = streamingRequest.AudioCodec.Split(',', StringSplitOptions.RemoveEmptyEntries);
  108. state.Request.AudioCodec = state.SupportedAudioCodecs.FirstOrDefault(mediaEncoder.CanEncodeToAudioCodec)
  109. ?? state.SupportedAudioCodecs.FirstOrDefault();
  110. }
  111. if (!string.IsNullOrWhiteSpace(streamingRequest.SubtitleCodec))
  112. {
  113. state.SupportedSubtitleCodecs = streamingRequest.SubtitleCodec.Split(',', StringSplitOptions.RemoveEmptyEntries);
  114. state.Request.SubtitleCodec = state.SupportedSubtitleCodecs.FirstOrDefault(mediaEncoder.CanEncodeToSubtitleCodec)
  115. ?? state.SupportedSubtitleCodecs.FirstOrDefault();
  116. }
  117. var item = libraryManager.GetItemById(streamingRequest.Id);
  118. state.IsInputVideo = string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase);
  119. MediaSourceInfo? mediaSource = null;
  120. if (string.IsNullOrWhiteSpace(streamingRequest.LiveStreamId))
  121. {
  122. var currentJob = !string.IsNullOrWhiteSpace(streamingRequest.PlaySessionId)
  123. ? transcodingJobHelper.GetTranscodingJob(streamingRequest.PlaySessionId)
  124. : null;
  125. if (currentJob != null)
  126. {
  127. mediaSource = currentJob.MediaSource;
  128. }
  129. if (mediaSource == null)
  130. {
  131. var mediaSources = await mediaSourceManager.GetPlaybackMediaSources(libraryManager.GetItemById(streamingRequest.Id), null, false, false, cancellationToken).ConfigureAwait(false);
  132. mediaSource = string.IsNullOrEmpty(streamingRequest.MediaSourceId)
  133. ? mediaSources[0]
  134. : mediaSources.Find(i => string.Equals(i.Id, streamingRequest.MediaSourceId, StringComparison.Ordinal));
  135. if (mediaSource == null && Guid.Parse(streamingRequest.MediaSourceId).Equals(streamingRequest.Id))
  136. {
  137. mediaSource = mediaSources[0];
  138. }
  139. }
  140. }
  141. else
  142. {
  143. var liveStreamInfo = await mediaSourceManager.GetLiveStreamWithDirectStreamProvider(streamingRequest.LiveStreamId, cancellationToken).ConfigureAwait(false);
  144. mediaSource = liveStreamInfo.Item1;
  145. state.DirectStreamProvider = liveStreamInfo.Item2;
  146. }
  147. var encodingOptions = serverConfigurationManager.GetEncodingOptions();
  148. encodingHelper.AttachMediaSourceInfo(state, encodingOptions, mediaSource, url);
  149. string? containerInternal = Path.GetExtension(state.RequestedUrl);
  150. if (!string.IsNullOrEmpty(streamingRequest.Container))
  151. {
  152. containerInternal = streamingRequest.Container;
  153. }
  154. if (string.IsNullOrEmpty(containerInternal))
  155. {
  156. containerInternal = streamingRequest.Static ?
  157. StreamBuilder.NormalizeMediaSourceFormatIntoSingleContainer(state.InputContainer, null, DlnaProfileType.Audio)
  158. : GetOutputFileExtension(state);
  159. }
  160. state.OutputContainer = (containerInternal ?? string.Empty).TrimStart('.');
  161. state.OutputAudioBitrate = encodingHelper.GetAudioBitrateParam(streamingRequest.AudioBitRate, streamingRequest.AudioCodec, state.AudioStream);
  162. state.OutputAudioCodec = streamingRequest.AudioCodec;
  163. state.OutputAudioChannels = encodingHelper.GetNumAudioChannelsParam(state, state.AudioStream, state.OutputAudioCodec);
  164. if (state.VideoRequest != null)
  165. {
  166. state.OutputVideoCodec = state.Request.VideoCodec;
  167. state.OutputVideoBitrate = encodingHelper.GetVideoBitrateParamValue(state.VideoRequest, state.VideoStream, state.OutputVideoCodec);
  168. encodingHelper.TryStreamCopy(state);
  169. if (!EncodingHelper.IsCopyCodec(state.OutputVideoCodec) && state.OutputVideoBitrate.HasValue)
  170. {
  171. var isVideoResolutionNotRequested = !state.VideoRequest.Width.HasValue
  172. && !state.VideoRequest.Height.HasValue
  173. && !state.VideoRequest.MaxWidth.HasValue
  174. && !state.VideoRequest.MaxHeight.HasValue;
  175. if (isVideoResolutionNotRequested
  176. && state.VideoStream != null
  177. && state.VideoRequest.VideoBitRate.HasValue
  178. && state.VideoStream.BitRate.HasValue
  179. && state.VideoRequest.VideoBitRate.Value >= state.VideoStream.BitRate.Value)
  180. {
  181. // Don't downscale the resolution if the width/height/MaxWidth/MaxHeight is not requested,
  182. // and the requested video bitrate is higher than source video bitrate.
  183. if (state.VideoStream.Width.HasValue || state.VideoStream.Height.HasValue)
  184. {
  185. state.VideoRequest.MaxWidth = state.VideoStream?.Width;
  186. state.VideoRequest.MaxHeight = state.VideoStream?.Height;
  187. }
  188. }
  189. else
  190. {
  191. var resolution = ResolutionNormalizer.Normalize(
  192. state.VideoStream?.BitRate,
  193. state.OutputVideoBitrate.Value,
  194. state.VideoRequest.MaxWidth,
  195. state.VideoRequest.MaxHeight);
  196. state.VideoRequest.MaxWidth = resolution.MaxWidth;
  197. state.VideoRequest.MaxHeight = resolution.MaxHeight;
  198. }
  199. }
  200. }
  201. ApplyDeviceProfileSettings(state, dlnaManager, deviceManager, httpRequest, streamingRequest.DeviceProfileId, streamingRequest.Static);
  202. var ext = string.IsNullOrWhiteSpace(state.OutputContainer)
  203. ? GetOutputFileExtension(state)
  204. : ("." + state.OutputContainer);
  205. state.OutputFilePath = GetOutputFilePath(state, ext!, serverConfigurationManager, streamingRequest.DeviceId, streamingRequest.PlaySessionId);
  206. return state;
  207. }
  208. /// <summary>
  209. /// Adds the dlna headers.
  210. /// </summary>
  211. /// <param name="state">The state.</param>
  212. /// <param name="responseHeaders">The response headers.</param>
  213. /// <param name="isStaticallyStreamed">if set to <c>true</c> [is statically streamed].</param>
  214. /// <param name="startTimeTicks">The start time in ticks.</param>
  215. /// <param name="request">The <see cref="HttpRequest"/>.</param>
  216. /// <param name="dlnaManager">Instance of the <see cref="IDlnaManager"/> interface.</param>
  217. public static void AddDlnaHeaders(
  218. StreamState state,
  219. IHeaderDictionary responseHeaders,
  220. bool isStaticallyStreamed,
  221. long? startTimeTicks,
  222. HttpRequest request,
  223. IDlnaManager dlnaManager)
  224. {
  225. if (!state.EnableDlnaHeaders)
  226. {
  227. return;
  228. }
  229. var profile = state.DeviceProfile;
  230. StringValues transferMode = request.Headers["transferMode.dlna.org"];
  231. responseHeaders.Add("transferMode.dlna.org", string.IsNullOrEmpty(transferMode) ? "Streaming" : transferMode.ToString());
  232. responseHeaders.Add("realTimeInfo.dlna.org", "DLNA.ORG_TLAG=*");
  233. if (state.RunTimeTicks.HasValue)
  234. {
  235. if (string.Equals(request.Headers["getMediaInfo.sec"], "1", StringComparison.OrdinalIgnoreCase))
  236. {
  237. var ms = TimeSpan.FromTicks(state.RunTimeTicks.Value).TotalMilliseconds;
  238. responseHeaders.Add("MediaInfo.sec", string.Format(
  239. CultureInfo.InvariantCulture,
  240. "SEC_Duration={0};",
  241. Convert.ToInt32(ms)));
  242. }
  243. if (!isStaticallyStreamed && profile != null)
  244. {
  245. AddTimeSeekResponseHeaders(state, responseHeaders, startTimeTicks);
  246. }
  247. }
  248. profile ??= dlnaManager.GetDefaultProfile();
  249. var audioCodec = state.ActualOutputAudioCodec;
  250. if (!state.IsVideoRequest)
  251. {
  252. responseHeaders.Add("contentFeatures.dlna.org", ContentFeatureBuilder.BuildAudioHeader(
  253. profile,
  254. state.OutputContainer,
  255. audioCodec,
  256. state.OutputAudioBitrate,
  257. state.OutputAudioSampleRate,
  258. state.OutputAudioChannels,
  259. state.OutputAudioBitDepth,
  260. isStaticallyStreamed,
  261. state.RunTimeTicks,
  262. state.TranscodeSeekInfo));
  263. }
  264. else
  265. {
  266. var videoCodec = state.ActualOutputVideoCodec;
  267. responseHeaders.Add(
  268. "contentFeatures.dlna.org",
  269. ContentFeatureBuilder.BuildVideoHeader(profile, state.OutputContainer, videoCodec, audioCodec, state.OutputWidth, state.OutputHeight, state.TargetVideoBitDepth, state.OutputVideoBitrate, state.TargetTimestamp, isStaticallyStreamed, state.RunTimeTicks, state.TargetVideoProfile, state.TargetVideoLevel, state.TargetFramerate, state.TargetPacketLength, state.TranscodeSeekInfo, state.IsTargetAnamorphic, state.IsTargetInterlaced, state.TargetRefFrames, state.TargetVideoStreamCount, state.TargetAudioStreamCount, state.TargetVideoCodecTag, state.IsTargetAVC).FirstOrDefault() ?? string.Empty);
  270. }
  271. }
  272. /// <summary>
  273. /// Parses the time seek header.
  274. /// </summary>
  275. /// <param name="value">The time seek header string.</param>
  276. /// <returns>A nullable <see cref="long"/> representing the seek time in ticks.</returns>
  277. private static long? ParseTimeSeekHeader(ReadOnlySpan<char> value)
  278. {
  279. if (value.IsEmpty)
  280. {
  281. return null;
  282. }
  283. const string npt = "npt=";
  284. if (!value.StartsWith(npt, StringComparison.OrdinalIgnoreCase))
  285. {
  286. throw new ArgumentException("Invalid timeseek header");
  287. }
  288. var index = value.IndexOf('-');
  289. value = index == -1
  290. ? value.Slice(npt.Length)
  291. : value.Slice(npt.Length, index - npt.Length);
  292. if (value.IndexOf(':') == -1)
  293. {
  294. // Parses npt times in the format of '417.33'
  295. if (double.TryParse(value, NumberStyles.Any, CultureInfo.InvariantCulture, out var seconds))
  296. {
  297. return TimeSpan.FromSeconds(seconds).Ticks;
  298. }
  299. throw new ArgumentException("Invalid timeseek header");
  300. }
  301. try
  302. {
  303. // Parses npt times in the format of '10:19:25.7'
  304. return TimeSpan.Parse(value).Ticks;
  305. }
  306. catch
  307. {
  308. throw new ArgumentException("Invalid timeseek header");
  309. }
  310. }
  311. /// <summary>
  312. /// Parses query parameters as StreamOptions.
  313. /// </summary>
  314. /// <param name="queryString">The query string.</param>
  315. /// <returns>A <see cref="Dictionary{String,String}"/> containing the stream options.</returns>
  316. private static Dictionary<string, string> ParseStreamOptions(IQueryCollection queryString)
  317. {
  318. Dictionary<string, string> streamOptions = new Dictionary<string, string>();
  319. foreach (var param in queryString)
  320. {
  321. if (char.IsLower(param.Key[0]))
  322. {
  323. // This was probably not parsed initially and should be a StreamOptions
  324. // or the generated URL should correctly serialize it
  325. // TODO: This should be incorporated either in the lower framework for parsing requests
  326. streamOptions[param.Key] = param.Value;
  327. }
  328. }
  329. return streamOptions;
  330. }
  331. /// <summary>
  332. /// Adds the dlna time seek headers to the response.
  333. /// </summary>
  334. /// <param name="state">The current <see cref="StreamState"/>.</param>
  335. /// <param name="responseHeaders">The <see cref="IHeaderDictionary"/> of the response.</param>
  336. /// <param name="startTimeTicks">The start time in ticks.</param>
  337. private static void AddTimeSeekResponseHeaders(StreamState state, IHeaderDictionary responseHeaders, long? startTimeTicks)
  338. {
  339. var runtimeSeconds = TimeSpan.FromTicks(state.RunTimeTicks!.Value).TotalSeconds.ToString(CultureInfo.InvariantCulture);
  340. var startSeconds = TimeSpan.FromTicks(startTimeTicks ?? 0).TotalSeconds.ToString(CultureInfo.InvariantCulture);
  341. responseHeaders.Add("TimeSeekRange.dlna.org", string.Format(
  342. CultureInfo.InvariantCulture,
  343. "npt={0}-{1}/{1}",
  344. startSeconds,
  345. runtimeSeconds));
  346. responseHeaders.Add("X-AvailableSeekRange", string.Format(
  347. CultureInfo.InvariantCulture,
  348. "1 npt={0}-{1}",
  349. startSeconds,
  350. runtimeSeconds));
  351. }
  352. /// <summary>
  353. /// Gets the output file extension.
  354. /// </summary>
  355. /// <param name="state">The state.</param>
  356. /// <returns>System.String.</returns>
  357. private static string? GetOutputFileExtension(StreamState state)
  358. {
  359. var ext = Path.GetExtension(state.RequestedUrl);
  360. if (!string.IsNullOrEmpty(ext))
  361. {
  362. return ext;
  363. }
  364. // Try to infer based on the desired video codec
  365. if (state.IsVideoRequest)
  366. {
  367. var videoCodec = state.Request.VideoCodec;
  368. if (string.Equals(videoCodec, "h264", StringComparison.OrdinalIgnoreCase) ||
  369. string.Equals(videoCodec, "h265", StringComparison.OrdinalIgnoreCase))
  370. {
  371. return ".ts";
  372. }
  373. if (string.Equals(videoCodec, "theora", StringComparison.OrdinalIgnoreCase))
  374. {
  375. return ".ogv";
  376. }
  377. if (string.Equals(videoCodec, "vp8", StringComparison.OrdinalIgnoreCase)
  378. || string.Equals(videoCodec, "vp9", StringComparison.OrdinalIgnoreCase)
  379. || string.Equals(videoCodec, "vpx", StringComparison.OrdinalIgnoreCase))
  380. {
  381. return ".webm";
  382. }
  383. if (string.Equals(videoCodec, "wmv", StringComparison.OrdinalIgnoreCase))
  384. {
  385. return ".asf";
  386. }
  387. }
  388. // Try to infer based on the desired audio codec
  389. if (!state.IsVideoRequest)
  390. {
  391. var audioCodec = state.Request.AudioCodec;
  392. if (string.Equals("aac", audioCodec, StringComparison.OrdinalIgnoreCase))
  393. {
  394. return ".aac";
  395. }
  396. if (string.Equals("mp3", audioCodec, StringComparison.OrdinalIgnoreCase))
  397. {
  398. return ".mp3";
  399. }
  400. if (string.Equals("vorbis", audioCodec, StringComparison.OrdinalIgnoreCase))
  401. {
  402. return ".ogg";
  403. }
  404. if (string.Equals("wma", audioCodec, StringComparison.OrdinalIgnoreCase))
  405. {
  406. return ".wma";
  407. }
  408. }
  409. return null;
  410. }
  411. /// <summary>
  412. /// Gets the output file path for transcoding.
  413. /// </summary>
  414. /// <param name="state">The current <see cref="StreamState"/>.</param>
  415. /// <param name="outputFileExtension">The file extension of the output file.</param>
  416. /// <param name="serverConfigurationManager">Instance of the <see cref="IServerConfigurationManager"/> interface.</param>
  417. /// <param name="deviceId">The device id.</param>
  418. /// <param name="playSessionId">The play session id.</param>
  419. /// <returns>The complete file path, including the folder, for the transcoding file.</returns>
  420. private static string GetOutputFilePath(StreamState state, string outputFileExtension, IServerConfigurationManager serverConfigurationManager, string? deviceId, string? playSessionId)
  421. {
  422. var data = $"{state.MediaPath}-{state.UserAgent}-{deviceId!}-{playSessionId!}";
  423. var filename = data.GetMD5().ToString("N", CultureInfo.InvariantCulture);
  424. var ext = outputFileExtension?.ToLowerInvariant();
  425. var folder = serverConfigurationManager.GetTranscodePath();
  426. return Path.Combine(folder, filename + ext);
  427. }
  428. private static void ApplyDeviceProfileSettings(StreamState state, IDlnaManager dlnaManager, IDeviceManager deviceManager, HttpRequest request, string? deviceProfileId, bool? @static)
  429. {
  430. if (!string.IsNullOrWhiteSpace(deviceProfileId))
  431. {
  432. state.DeviceProfile = dlnaManager.GetProfile(deviceProfileId);
  433. if (state.DeviceProfile == null)
  434. {
  435. var caps = deviceManager.GetCapabilities(deviceProfileId);
  436. state.DeviceProfile = caps == null ? dlnaManager.GetProfile(request.Headers) : caps.DeviceProfile;
  437. }
  438. }
  439. var profile = state.DeviceProfile;
  440. if (profile == null)
  441. {
  442. // Don't use settings from the default profile.
  443. // Only use a specific profile if it was requested.
  444. return;
  445. }
  446. var audioCodec = state.ActualOutputAudioCodec;
  447. var videoCodec = state.ActualOutputVideoCodec;
  448. var mediaProfile = !state.IsVideoRequest
  449. ? profile.GetAudioMediaProfile(state.OutputContainer, audioCodec, state.OutputAudioChannels, state.OutputAudioBitrate, state.OutputAudioSampleRate, state.OutputAudioBitDepth)
  450. : profile.GetVideoMediaProfile(
  451. state.OutputContainer,
  452. audioCodec,
  453. videoCodec,
  454. state.OutputWidth,
  455. state.OutputHeight,
  456. state.TargetVideoBitDepth,
  457. state.OutputVideoBitrate,
  458. state.TargetVideoProfile,
  459. state.TargetVideoLevel,
  460. state.TargetFramerate,
  461. state.TargetPacketLength,
  462. state.TargetTimestamp,
  463. state.IsTargetAnamorphic,
  464. state.IsTargetInterlaced,
  465. state.TargetRefFrames,
  466. state.TargetVideoStreamCount,
  467. state.TargetAudioStreamCount,
  468. state.TargetVideoCodecTag,
  469. state.IsTargetAVC);
  470. if (mediaProfile != null)
  471. {
  472. state.MimeType = mediaProfile.MimeType;
  473. }
  474. if (!(@static.HasValue && @static.Value))
  475. {
  476. var transcodingProfile = !state.IsVideoRequest ? profile.GetAudioTranscodingProfile(state.OutputContainer, audioCodec) : profile.GetVideoTranscodingProfile(state.OutputContainer, audioCodec, videoCodec);
  477. if (transcodingProfile != null)
  478. {
  479. state.EstimateContentLength = transcodingProfile.EstimateContentLength;
  480. // state.EnableMpegtsM2TsMode = transcodingProfile.EnableMpegtsM2TsMode;
  481. state.TranscodeSeekInfo = transcodingProfile.TranscodeSeekInfo;
  482. if (state.VideoRequest != null)
  483. {
  484. state.VideoRequest.CopyTimestamps = transcodingProfile.CopyTimestamps;
  485. state.VideoRequest.EnableSubtitlesInManifest = transcodingProfile.EnableSubtitlesInManifest;
  486. }
  487. }
  488. }
  489. }
  490. /// <summary>
  491. /// Parses the parameters.
  492. /// </summary>
  493. /// <param name="request">The request.</param>
  494. private static void ParseParams(StreamingRequestDto request)
  495. {
  496. if (string.IsNullOrEmpty(request.Params))
  497. {
  498. return;
  499. }
  500. var vals = request.Params.Split(';');
  501. var videoRequest = request as VideoRequestDto;
  502. for (var i = 0; i < vals.Length; i++)
  503. {
  504. var val = vals[i];
  505. if (string.IsNullOrWhiteSpace(val))
  506. {
  507. continue;
  508. }
  509. switch (i)
  510. {
  511. case 0:
  512. request.DeviceProfileId = val;
  513. break;
  514. case 1:
  515. request.DeviceId = val;
  516. break;
  517. case 2:
  518. request.MediaSourceId = val;
  519. break;
  520. case 3:
  521. request.Static = string.Equals("true", val, StringComparison.OrdinalIgnoreCase);
  522. break;
  523. case 4:
  524. if (videoRequest != null)
  525. {
  526. videoRequest.VideoCodec = val;
  527. }
  528. break;
  529. case 5:
  530. request.AudioCodec = val;
  531. break;
  532. case 6:
  533. if (videoRequest != null)
  534. {
  535. videoRequest.AudioStreamIndex = int.Parse(val, CultureInfo.InvariantCulture);
  536. }
  537. break;
  538. case 7:
  539. if (videoRequest != null)
  540. {
  541. videoRequest.SubtitleStreamIndex = int.Parse(val, CultureInfo.InvariantCulture);
  542. }
  543. break;
  544. case 8:
  545. if (videoRequest != null)
  546. {
  547. videoRequest.VideoBitRate = int.Parse(val, CultureInfo.InvariantCulture);
  548. }
  549. break;
  550. case 9:
  551. request.AudioBitRate = int.Parse(val, CultureInfo.InvariantCulture);
  552. break;
  553. case 10:
  554. request.MaxAudioChannels = int.Parse(val, CultureInfo.InvariantCulture);
  555. break;
  556. case 11:
  557. if (videoRequest != null)
  558. {
  559. videoRequest.MaxFramerate = float.Parse(val, CultureInfo.InvariantCulture);
  560. }
  561. break;
  562. case 12:
  563. if (videoRequest != null)
  564. {
  565. videoRequest.MaxWidth = int.Parse(val, CultureInfo.InvariantCulture);
  566. }
  567. break;
  568. case 13:
  569. if (videoRequest != null)
  570. {
  571. videoRequest.MaxHeight = int.Parse(val, CultureInfo.InvariantCulture);
  572. }
  573. break;
  574. case 14:
  575. request.StartTimeTicks = long.Parse(val, CultureInfo.InvariantCulture);
  576. break;
  577. case 15:
  578. if (videoRequest != null)
  579. {
  580. videoRequest.Level = val;
  581. }
  582. break;
  583. case 16:
  584. if (videoRequest != null)
  585. {
  586. videoRequest.MaxRefFrames = int.Parse(val, CultureInfo.InvariantCulture);
  587. }
  588. break;
  589. case 17:
  590. if (videoRequest != null)
  591. {
  592. videoRequest.MaxVideoBitDepth = int.Parse(val, CultureInfo.InvariantCulture);
  593. }
  594. break;
  595. case 18:
  596. if (videoRequest != null)
  597. {
  598. videoRequest.Profile = val;
  599. }
  600. break;
  601. case 19:
  602. // cabac no longer used
  603. break;
  604. case 20:
  605. request.PlaySessionId = val;
  606. break;
  607. case 21:
  608. // api_key
  609. break;
  610. case 22:
  611. request.LiveStreamId = val;
  612. break;
  613. case 23:
  614. // Duplicating ItemId because of MediaMonkey
  615. break;
  616. case 24:
  617. if (videoRequest != null)
  618. {
  619. videoRequest.CopyTimestamps = string.Equals("true", val, StringComparison.OrdinalIgnoreCase);
  620. }
  621. break;
  622. case 25:
  623. if (!string.IsNullOrWhiteSpace(val) && videoRequest != null)
  624. {
  625. if (Enum.TryParse(val, out SubtitleDeliveryMethod method))
  626. {
  627. videoRequest.SubtitleMethod = method;
  628. }
  629. }
  630. break;
  631. case 26:
  632. request.TranscodingMaxAudioChannels = int.Parse(val, CultureInfo.InvariantCulture);
  633. break;
  634. case 27:
  635. if (videoRequest != null)
  636. {
  637. videoRequest.EnableSubtitlesInManifest = string.Equals("true", val, StringComparison.OrdinalIgnoreCase);
  638. }
  639. break;
  640. case 28:
  641. request.Tag = val;
  642. break;
  643. case 29:
  644. if (videoRequest != null)
  645. {
  646. videoRequest.RequireAvc = string.Equals("true", val, StringComparison.OrdinalIgnoreCase);
  647. }
  648. break;
  649. case 30:
  650. request.SubtitleCodec = val;
  651. break;
  652. case 31:
  653. if (videoRequest != null)
  654. {
  655. videoRequest.RequireNonAnamorphic = string.Equals("true", val, StringComparison.OrdinalIgnoreCase);
  656. }
  657. break;
  658. case 32:
  659. if (videoRequest != null)
  660. {
  661. videoRequest.DeInterlace = string.Equals("true", val, StringComparison.OrdinalIgnoreCase);
  662. }
  663. break;
  664. case 33:
  665. request.TranscodeReasons = val;
  666. break;
  667. }
  668. }
  669. }
  670. }
  671. }