MediaInfoService.cs 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679
  1. #pragma warning disable CS1591
  2. #pragma warning disable SA1402
  3. #pragma warning disable SA1649
  4. using System;
  5. using System.Buffers;
  6. using System.Globalization;
  7. using System.Linq;
  8. using System.Text.Json;
  9. using System.Threading;
  10. using System.Threading.Tasks;
  11. using Jellyfin.Data.Enums;
  12. using MediaBrowser.Common.Net;
  13. using MediaBrowser.Controller.Configuration;
  14. using MediaBrowser.Controller.Devices;
  15. using MediaBrowser.Controller.Entities;
  16. using MediaBrowser.Controller.Entities.Audio;
  17. using MediaBrowser.Controller.Library;
  18. using MediaBrowser.Controller.MediaEncoding;
  19. using MediaBrowser.Controller.Net;
  20. using MediaBrowser.Model.Dlna;
  21. using MediaBrowser.Model.Dto;
  22. using MediaBrowser.Model.Entities;
  23. using MediaBrowser.Model.MediaInfo;
  24. using MediaBrowser.Model.Services;
  25. using MediaBrowser.Model.Session;
  26. using Microsoft.Extensions.Logging;
  27. namespace MediaBrowser.Api.Playback
  28. {
  29. [Route("/Items/{Id}/PlaybackInfo", "GET", Summary = "Gets live playback media info for an item")]
  30. public class GetPlaybackInfo : IReturn<PlaybackInfoResponse>
  31. {
  32. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  33. public Guid Id { get; set; }
  34. [ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "query", Verb = "GET")]
  35. public Guid UserId { get; set; }
  36. }
  37. [Route("/Items/{Id}/PlaybackInfo", "POST", Summary = "Gets live playback media info for an item")]
  38. public class GetPostedPlaybackInfo : PlaybackInfoRequest, IReturn<PlaybackInfoResponse>
  39. {
  40. }
  41. [Route("/LiveStreams/Open", "POST", Summary = "Opens a media source")]
  42. public class OpenMediaSource : LiveStreamRequest, IReturn<LiveStreamResponse>
  43. {
  44. }
  45. [Route("/LiveStreams/Close", "POST", Summary = "Closes a media source")]
  46. public class CloseMediaSource : IReturnVoid
  47. {
  48. [ApiMember(Name = "LiveStreamId", Description = "LiveStreamId", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
  49. public string LiveStreamId { get; set; }
  50. }
  51. [Route("/Playback/BitrateTest", "GET")]
  52. public class GetBitrateTestBytes
  53. {
  54. [ApiMember(Name = "Size", Description = "Size", IsRequired = true, DataType = "int", ParameterType = "query", Verb = "GET")]
  55. public int Size { get; set; }
  56. public GetBitrateTestBytes()
  57. {
  58. // 100k
  59. Size = 102400;
  60. }
  61. }
  62. [Authenticated]
  63. public class MediaInfoService : BaseApiService
  64. {
  65. private readonly IMediaSourceManager _mediaSourceManager;
  66. private readonly IDeviceManager _deviceManager;
  67. private readonly ILibraryManager _libraryManager;
  68. private readonly INetworkManager _networkManager;
  69. private readonly IMediaEncoder _mediaEncoder;
  70. private readonly IUserManager _userManager;
  71. private readonly IAuthorizationContext _authContext;
  72. public MediaInfoService(
  73. ILogger<MediaInfoService> logger,
  74. IServerConfigurationManager serverConfigurationManager,
  75. IHttpResultFactory httpResultFactory,
  76. IMediaSourceManager mediaSourceManager,
  77. IDeviceManager deviceManager,
  78. ILibraryManager libraryManager,
  79. INetworkManager networkManager,
  80. IMediaEncoder mediaEncoder,
  81. IUserManager userManager,
  82. IAuthorizationContext authContext)
  83. : base(logger, serverConfigurationManager, httpResultFactory)
  84. {
  85. _mediaSourceManager = mediaSourceManager;
  86. _deviceManager = deviceManager;
  87. _libraryManager = libraryManager;
  88. _networkManager = networkManager;
  89. _mediaEncoder = mediaEncoder;
  90. _userManager = userManager;
  91. _authContext = authContext;
  92. }
  93. public object Get(GetBitrateTestBytes request)
  94. {
  95. const int MaxSize = 10_000_000;
  96. var size = request.Size;
  97. if (size <= 0)
  98. {
  99. throw new ArgumentException($"The requested size ({size}) is equal to or smaller than 0.", nameof(request));
  100. }
  101. if (size > MaxSize)
  102. {
  103. throw new ArgumentException($"The requested size ({size}) is larger than the max allowed value ({MaxSize}).", nameof(request));
  104. }
  105. byte[] buffer = ArrayPool<byte>.Shared.Rent(size);
  106. try
  107. {
  108. new Random().NextBytes(buffer);
  109. return ResultFactory.GetResult(null, buffer, "application/octet-stream");
  110. }
  111. finally
  112. {
  113. ArrayPool<byte>.Shared.Return(buffer);
  114. }
  115. }
  116. public async Task<object> Get(GetPlaybackInfo request)
  117. {
  118. var result = await GetPlaybackInfo(request.Id, request.UserId, new[] { MediaType.Audio, MediaType.Video }).ConfigureAwait(false);
  119. return ToOptimizedResult(result);
  120. }
  121. public async Task<object> Post(OpenMediaSource request)
  122. {
  123. var result = await OpenMediaSource(request).ConfigureAwait(false);
  124. return ToOptimizedResult(result);
  125. }
  126. private async Task<LiveStreamResponse> OpenMediaSource(OpenMediaSource request)
  127. {
  128. var authInfo = _authContext.GetAuthorizationInfo(Request);
  129. var result = await _mediaSourceManager.OpenLiveStream(request, CancellationToken.None).ConfigureAwait(false);
  130. var profile = request.DeviceProfile;
  131. if (profile == null)
  132. {
  133. var caps = _deviceManager.GetCapabilities(authInfo.DeviceId);
  134. if (caps != null)
  135. {
  136. profile = caps.DeviceProfile;
  137. }
  138. }
  139. if (profile != null)
  140. {
  141. var item = _libraryManager.GetItemById(request.ItemId);
  142. SetDeviceSpecificData(item, result.MediaSource, profile, authInfo, request.MaxStreamingBitrate,
  143. request.StartTimeTicks ?? 0, result.MediaSource.Id, request.AudioStreamIndex,
  144. request.SubtitleStreamIndex, request.MaxAudioChannels, request.PlaySessionId, request.UserId, request.EnableDirectPlay, true, request.EnableDirectStream, true, true, true);
  145. }
  146. else
  147. {
  148. if (!string.IsNullOrWhiteSpace(result.MediaSource.TranscodingUrl))
  149. {
  150. result.MediaSource.TranscodingUrl += "&LiveStreamId=" + result.MediaSource.LiveStreamId;
  151. }
  152. }
  153. if (result.MediaSource != null)
  154. {
  155. NormalizeMediaSourceContainer(result.MediaSource, profile, DlnaProfileType.Video);
  156. }
  157. return result;
  158. }
  159. public void Post(CloseMediaSource request)
  160. {
  161. _mediaSourceManager.CloseLiveStream(request.LiveStreamId).GetAwaiter().GetResult();
  162. }
  163. public async Task<PlaybackInfoResponse> GetPlaybackInfo(GetPostedPlaybackInfo request)
  164. {
  165. var authInfo = _authContext.GetAuthorizationInfo(Request);
  166. var profile = request.DeviceProfile;
  167. Logger.LogInformation("GetPostedPlaybackInfo profile: {@Profile}", profile);
  168. if (profile == null)
  169. {
  170. var caps = _deviceManager.GetCapabilities(authInfo.DeviceId);
  171. if (caps != null)
  172. {
  173. profile = caps.DeviceProfile;
  174. }
  175. }
  176. var info = await GetPlaybackInfo(request.Id, request.UserId, new[] { MediaType.Audio, MediaType.Video }, request.MediaSourceId, request.LiveStreamId).ConfigureAwait(false);
  177. if (profile != null)
  178. {
  179. var mediaSourceId = request.MediaSourceId;
  180. SetDeviceSpecificData(request.Id, info, profile, authInfo, request.MaxStreamingBitrate ?? profile.MaxStreamingBitrate, request.StartTimeTicks ?? 0, mediaSourceId, request.AudioStreamIndex, request.SubtitleStreamIndex, request.MaxAudioChannels, request.UserId, request.EnableDirectPlay, true, request.EnableDirectStream, request.EnableTranscoding, request.AllowVideoStreamCopy, request.AllowAudioStreamCopy);
  181. }
  182. if (request.AutoOpenLiveStream)
  183. {
  184. var mediaSource = string.IsNullOrWhiteSpace(request.MediaSourceId) ? info.MediaSources.FirstOrDefault() : info.MediaSources.FirstOrDefault(i => string.Equals(i.Id, request.MediaSourceId, StringComparison.Ordinal));
  185. if (mediaSource != null && mediaSource.RequiresOpening && string.IsNullOrWhiteSpace(mediaSource.LiveStreamId))
  186. {
  187. var openStreamResult = await OpenMediaSource(new OpenMediaSource
  188. {
  189. AudioStreamIndex = request.AudioStreamIndex,
  190. DeviceProfile = request.DeviceProfile,
  191. EnableDirectPlay = request.EnableDirectPlay,
  192. EnableDirectStream = request.EnableDirectStream,
  193. ItemId = request.Id,
  194. MaxAudioChannels = request.MaxAudioChannels,
  195. MaxStreamingBitrate = request.MaxStreamingBitrate,
  196. PlaySessionId = info.PlaySessionId,
  197. StartTimeTicks = request.StartTimeTicks,
  198. SubtitleStreamIndex = request.SubtitleStreamIndex,
  199. UserId = request.UserId,
  200. OpenToken = mediaSource.OpenToken
  201. }).ConfigureAwait(false);
  202. info.MediaSources = new[] { openStreamResult.MediaSource };
  203. }
  204. }
  205. if (info.MediaSources != null)
  206. {
  207. foreach (var mediaSource in info.MediaSources)
  208. {
  209. NormalizeMediaSourceContainer(mediaSource, profile, DlnaProfileType.Video);
  210. }
  211. }
  212. return info;
  213. }
  214. private void NormalizeMediaSourceContainer(MediaSourceInfo mediaSource, DeviceProfile profile, DlnaProfileType type)
  215. {
  216. mediaSource.Container = StreamBuilder.NormalizeMediaSourceFormatIntoSingleContainer(mediaSource.Container, mediaSource.Path, profile, type);
  217. }
  218. public async Task<object> Post(GetPostedPlaybackInfo request)
  219. {
  220. var result = await GetPlaybackInfo(request).ConfigureAwait(false);
  221. return ToOptimizedResult(result);
  222. }
  223. private async Task<PlaybackInfoResponse> GetPlaybackInfo(Guid id, Guid userId, string[] supportedLiveMediaTypes, string mediaSourceId = null, string liveStreamId = null)
  224. {
  225. var user = _userManager.GetUserById(userId);
  226. var item = _libraryManager.GetItemById(id);
  227. var result = new PlaybackInfoResponse();
  228. MediaSourceInfo[] mediaSources;
  229. if (string.IsNullOrWhiteSpace(liveStreamId))
  230. {
  231. // TODO handle supportedLiveMediaTypes?
  232. var mediaSourcesList = await _mediaSourceManager.GetPlaybackMediaSources(item, user, true, true, CancellationToken.None).ConfigureAwait(false);
  233. if (string.IsNullOrWhiteSpace(mediaSourceId))
  234. {
  235. mediaSources = mediaSourcesList.ToArray();
  236. }
  237. else
  238. {
  239. mediaSources = mediaSourcesList
  240. .Where(i => string.Equals(i.Id, mediaSourceId, StringComparison.OrdinalIgnoreCase))
  241. .ToArray();
  242. }
  243. }
  244. else
  245. {
  246. var mediaSource = await _mediaSourceManager.GetLiveStream(liveStreamId, CancellationToken.None).ConfigureAwait(false);
  247. mediaSources = new[] { mediaSource };
  248. }
  249. if (mediaSources.Length == 0)
  250. {
  251. result.MediaSources = Array.Empty<MediaSourceInfo>();
  252. if (!result.ErrorCode.HasValue)
  253. {
  254. result.ErrorCode = PlaybackErrorCode.NoCompatibleStream;
  255. }
  256. }
  257. else
  258. {
  259. // Since we're going to be setting properties on MediaSourceInfos that come out of _mediaSourceManager, we should clone it
  260. // Should we move this directly into MediaSourceManager?
  261. result.MediaSources = JsonSerializer.Deserialize<MediaSourceInfo[]>(JsonSerializer.SerializeToUtf8Bytes(mediaSources));
  262. result.PlaySessionId = Guid.NewGuid().ToString("N", CultureInfo.InvariantCulture);
  263. }
  264. return result;
  265. }
  266. private void SetDeviceSpecificData(
  267. Guid itemId,
  268. PlaybackInfoResponse result,
  269. DeviceProfile profile,
  270. AuthorizationInfo auth,
  271. long? maxBitrate,
  272. long startTimeTicks,
  273. string mediaSourceId,
  274. int? audioStreamIndex,
  275. int? subtitleStreamIndex,
  276. int? maxAudioChannels,
  277. Guid userId,
  278. bool enableDirectPlay,
  279. bool forceDirectPlayRemoteMediaSource,
  280. bool enableDirectStream,
  281. bool enableTranscoding,
  282. bool allowVideoStreamCopy,
  283. bool allowAudioStreamCopy)
  284. {
  285. var item = _libraryManager.GetItemById(itemId);
  286. foreach (var mediaSource in result.MediaSources)
  287. {
  288. SetDeviceSpecificData(item, mediaSource, profile, auth, maxBitrate, startTimeTicks, mediaSourceId, audioStreamIndex, subtitleStreamIndex, maxAudioChannels, result.PlaySessionId, userId, enableDirectPlay, forceDirectPlayRemoteMediaSource, enableDirectStream, enableTranscoding, allowVideoStreamCopy, allowAudioStreamCopy);
  289. }
  290. SortMediaSources(result, maxBitrate);
  291. }
  292. private void SetDeviceSpecificData(
  293. BaseItem item,
  294. MediaSourceInfo mediaSource,
  295. DeviceProfile profile,
  296. AuthorizationInfo auth,
  297. long? maxBitrate,
  298. long startTimeTicks,
  299. string mediaSourceId,
  300. int? audioStreamIndex,
  301. int? subtitleStreamIndex,
  302. int? maxAudioChannels,
  303. string playSessionId,
  304. Guid userId,
  305. bool enableDirectPlay,
  306. bool forceDirectPlayRemoteMediaSource,
  307. bool enableDirectStream,
  308. bool enableTranscoding,
  309. bool allowVideoStreamCopy,
  310. bool allowAudioStreamCopy)
  311. {
  312. var streamBuilder = new StreamBuilder(_mediaEncoder, Logger);
  313. var options = new VideoOptions
  314. {
  315. MediaSources = new[] { mediaSource },
  316. Context = EncodingContext.Streaming,
  317. DeviceId = auth.DeviceId,
  318. ItemId = item.Id,
  319. Profile = profile,
  320. MaxAudioChannels = maxAudioChannels
  321. };
  322. if (string.Equals(mediaSourceId, mediaSource.Id, StringComparison.OrdinalIgnoreCase))
  323. {
  324. options.MediaSourceId = mediaSourceId;
  325. options.AudioStreamIndex = audioStreamIndex;
  326. options.SubtitleStreamIndex = subtitleStreamIndex;
  327. }
  328. var user = _userManager.GetUserById(userId);
  329. if (!enableDirectPlay)
  330. {
  331. mediaSource.SupportsDirectPlay = false;
  332. }
  333. if (!enableDirectStream)
  334. {
  335. mediaSource.SupportsDirectStream = false;
  336. }
  337. if (!enableTranscoding)
  338. {
  339. mediaSource.SupportsTranscoding = false;
  340. }
  341. if (item is Audio)
  342. {
  343. Logger.LogInformation(
  344. "User policy for {0}. EnableAudioPlaybackTranscoding: {1}",
  345. user.Username,
  346. user.HasPermission(PermissionKind.EnableAudioPlaybackTranscoding));
  347. }
  348. else
  349. {
  350. Logger.LogInformation("User policy for {0}. EnablePlaybackRemuxing: {1} EnableVideoPlaybackTranscoding: {2} EnableAudioPlaybackTranscoding: {3}",
  351. user.Username,
  352. user.HasPermission(PermissionKind.EnablePlaybackRemuxing),
  353. user.HasPermission(PermissionKind.EnableVideoPlaybackTranscoding),
  354. user.HasPermission(PermissionKind.EnableAudioPlaybackTranscoding));
  355. }
  356. // Beginning of Playback Determination: Attempt DirectPlay first
  357. if (mediaSource.SupportsDirectPlay)
  358. {
  359. if (mediaSource.IsRemote && user.HasPermission(PermissionKind.ForceRemoteSourceTranscoding))
  360. {
  361. mediaSource.SupportsDirectPlay = false;
  362. }
  363. else
  364. {
  365. var supportsDirectStream = mediaSource.SupportsDirectStream;
  366. // Dummy this up to fool StreamBuilder
  367. mediaSource.SupportsDirectStream = true;
  368. options.MaxBitrate = maxBitrate;
  369. if (item is Audio)
  370. {
  371. if (!user.HasPermission(PermissionKind.EnableAudioPlaybackTranscoding))
  372. {
  373. options.ForceDirectPlay = true;
  374. }
  375. }
  376. else if (item is Video)
  377. {
  378. if (!user.HasPermission(PermissionKind.EnableAudioPlaybackTranscoding)
  379. && !user.HasPermission(PermissionKind.EnableVideoPlaybackTranscoding)
  380. && !user.HasPermission(PermissionKind.EnablePlaybackRemuxing))
  381. {
  382. options.ForceDirectPlay = true;
  383. }
  384. }
  385. // The MediaSource supports direct stream, now test to see if the client supports it
  386. var streamInfo = string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase)
  387. ? streamBuilder.BuildAudioItem(options)
  388. : streamBuilder.BuildVideoItem(options);
  389. if (streamInfo == null || !streamInfo.IsDirectStream)
  390. {
  391. mediaSource.SupportsDirectPlay = false;
  392. }
  393. // Set this back to what it was
  394. mediaSource.SupportsDirectStream = supportsDirectStream;
  395. if (streamInfo != null)
  396. {
  397. SetDeviceSpecificSubtitleInfo(streamInfo, mediaSource, auth.Token);
  398. }
  399. }
  400. }
  401. if (mediaSource.SupportsDirectStream)
  402. {
  403. if (mediaSource.IsRemote && user.HasPermission(PermissionKind.ForceRemoteSourceTranscoding))
  404. {
  405. mediaSource.SupportsDirectStream = false;
  406. }
  407. else
  408. {
  409. options.MaxBitrate = GetMaxBitrate(maxBitrate, user);
  410. if (item is Audio)
  411. {
  412. if (!user.HasPermission(PermissionKind.EnableAudioPlaybackTranscoding))
  413. {
  414. options.ForceDirectStream = true;
  415. }
  416. }
  417. else if (item is Video)
  418. {
  419. if (!user.HasPermission(PermissionKind.EnableAudioPlaybackTranscoding)
  420. && !user.HasPermission(PermissionKind.EnableVideoPlaybackTranscoding)
  421. && !user.HasPermission(PermissionKind.EnablePlaybackRemuxing))
  422. {
  423. options.ForceDirectStream = true;
  424. }
  425. }
  426. // The MediaSource supports direct stream, now test to see if the client supports it
  427. var streamInfo = string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase)
  428. ? streamBuilder.BuildAudioItem(options)
  429. : streamBuilder.BuildVideoItem(options);
  430. if (streamInfo == null || !streamInfo.IsDirectStream)
  431. {
  432. mediaSource.SupportsDirectStream = false;
  433. }
  434. if (streamInfo != null)
  435. {
  436. SetDeviceSpecificSubtitleInfo(streamInfo, mediaSource, auth.Token);
  437. }
  438. }
  439. }
  440. if (mediaSource.SupportsTranscoding)
  441. {
  442. options.MaxBitrate = GetMaxBitrate(maxBitrate, user);
  443. // The MediaSource supports direct stream, now test to see if the client supports it
  444. var streamInfo = string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase)
  445. ? streamBuilder.BuildAudioItem(options)
  446. : streamBuilder.BuildVideoItem(options);
  447. if (mediaSource.IsRemote && user.HasPermission(PermissionKind.ForceRemoteSourceTranscoding))
  448. {
  449. if (streamInfo != null)
  450. {
  451. streamInfo.PlaySessionId = playSessionId;
  452. streamInfo.StartPositionTicks = startTimeTicks;
  453. mediaSource.TranscodingUrl = streamInfo.ToUrl("-", auth.Token).TrimStart('-');
  454. mediaSource.TranscodingUrl += "&allowVideoStreamCopy=false";
  455. mediaSource.TranscodingUrl += "&allowAudioStreamCopy=false";
  456. mediaSource.TranscodingContainer = streamInfo.Container;
  457. mediaSource.TranscodingSubProtocol = streamInfo.SubProtocol;
  458. // Do this after the above so that StartPositionTicks is set
  459. SetDeviceSpecificSubtitleInfo(streamInfo, mediaSource, auth.Token);
  460. }
  461. }
  462. else
  463. {
  464. if (streamInfo != null)
  465. {
  466. streamInfo.PlaySessionId = playSessionId;
  467. if (streamInfo.PlayMethod == PlayMethod.Transcode)
  468. {
  469. streamInfo.StartPositionTicks = startTimeTicks;
  470. mediaSource.TranscodingUrl = streamInfo.ToUrl("-", auth.Token).TrimStart('-');
  471. if (!allowVideoStreamCopy)
  472. {
  473. mediaSource.TranscodingUrl += "&allowVideoStreamCopy=false";
  474. }
  475. if (!allowAudioStreamCopy)
  476. {
  477. mediaSource.TranscodingUrl += "&allowAudioStreamCopy=false";
  478. }
  479. mediaSource.TranscodingContainer = streamInfo.Container;
  480. mediaSource.TranscodingSubProtocol = streamInfo.SubProtocol;
  481. }
  482. if (!allowAudioStreamCopy)
  483. {
  484. mediaSource.TranscodingUrl += "&allowAudioStreamCopy=false";
  485. }
  486. mediaSource.TranscodingContainer = streamInfo.Container;
  487. mediaSource.TranscodingSubProtocol = streamInfo.SubProtocol;
  488. // Do this after the above so that StartPositionTicks is set
  489. SetDeviceSpecificSubtitleInfo(streamInfo, mediaSource, auth.Token);
  490. }
  491. }
  492. }
  493. foreach (var attachment in mediaSource.MediaAttachments)
  494. {
  495. attachment.DeliveryUrl = string.Format(
  496. CultureInfo.InvariantCulture,
  497. "/Videos/{0}/{1}/Attachments/{2}",
  498. item.Id,
  499. mediaSource.Id,
  500. attachment.Index);
  501. }
  502. }
  503. private long? GetMaxBitrate(long? clientMaxBitrate, Jellyfin.Data.Entities.User user)
  504. {
  505. var maxBitrate = clientMaxBitrate;
  506. var remoteClientMaxBitrate = user?.RemoteClientBitrateLimit ?? 0;
  507. if (remoteClientMaxBitrate <= 0)
  508. {
  509. remoteClientMaxBitrate = ServerConfigurationManager.Configuration.RemoteClientBitrateLimit;
  510. }
  511. if (remoteClientMaxBitrate > 0)
  512. {
  513. var isInLocalNetwork = _networkManager.IsInLocalNetwork(Request.RemoteIp);
  514. Logger.LogInformation("RemoteClientBitrateLimit: {0}, RemoteIp: {1}, IsInLocalNetwork: {2}", remoteClientMaxBitrate, Request.RemoteIp, isInLocalNetwork);
  515. if (!isInLocalNetwork)
  516. {
  517. maxBitrate = Math.Min(maxBitrate ?? remoteClientMaxBitrate, remoteClientMaxBitrate);
  518. }
  519. }
  520. return maxBitrate;
  521. }
  522. private void SetDeviceSpecificSubtitleInfo(StreamInfo info, MediaSourceInfo mediaSource, string accessToken)
  523. {
  524. var profiles = info.GetSubtitleProfiles(_mediaEncoder, false, "-", accessToken);
  525. mediaSource.DefaultSubtitleStreamIndex = info.SubtitleStreamIndex;
  526. mediaSource.TranscodeReasons = info.TranscodeReasons;
  527. foreach (var profile in profiles)
  528. {
  529. foreach (var stream in mediaSource.MediaStreams)
  530. {
  531. if (stream.Type == MediaStreamType.Subtitle && stream.Index == profile.Index)
  532. {
  533. stream.DeliveryMethod = profile.DeliveryMethod;
  534. if (profile.DeliveryMethod == SubtitleDeliveryMethod.External)
  535. {
  536. stream.DeliveryUrl = profile.Url.TrimStart('-');
  537. stream.IsExternalUrl = profile.IsExternalUrl;
  538. }
  539. }
  540. }
  541. }
  542. }
  543. private void SortMediaSources(PlaybackInfoResponse result, long? maxBitrate)
  544. {
  545. var originalList = result.MediaSources.ToList();
  546. result.MediaSources = result.MediaSources.OrderBy(i =>
  547. {
  548. // Nothing beats direct playing a file
  549. if (i.SupportsDirectPlay && i.Protocol == MediaProtocol.File)
  550. {
  551. return 0;
  552. }
  553. return 1;
  554. }).ThenBy(i =>
  555. {
  556. // Let's assume direct streaming a file is just as desirable as direct playing a remote url
  557. if (i.SupportsDirectPlay || i.SupportsDirectStream)
  558. {
  559. return 0;
  560. }
  561. return 1;
  562. }).ThenBy(i =>
  563. {
  564. return i.Protocol switch
  565. {
  566. MediaProtocol.File => 0,
  567. _ => 1,
  568. };
  569. }).ThenBy(i =>
  570. {
  571. if (maxBitrate.HasValue && i.Bitrate.HasValue)
  572. {
  573. return i.Bitrate.Value <= maxBitrate.Value ? 0 : 2;
  574. }
  575. return 1;
  576. }).ThenBy(originalList.IndexOf)
  577. .ToArray();
  578. }
  579. }
  580. }