PlayToController.cs 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861
  1. using MediaBrowser.Controller.Dlna;
  2. using MediaBrowser.Controller.Drawing;
  3. using MediaBrowser.Controller.Entities;
  4. using MediaBrowser.Controller.Entities.Audio;
  5. using MediaBrowser.Controller.Library;
  6. using MediaBrowser.Controller.Persistence;
  7. using MediaBrowser.Controller.Session;
  8. using MediaBrowser.Dlna.Didl;
  9. using MediaBrowser.Dlna.Ssdp;
  10. using MediaBrowser.Model.Dlna;
  11. using MediaBrowser.Model.Dto;
  12. using MediaBrowser.Model.Entities;
  13. using MediaBrowser.Model.Logging;
  14. using MediaBrowser.Model.Session;
  15. using MediaBrowser.Model.System;
  16. using System;
  17. using System.Collections.Generic;
  18. using System.Globalization;
  19. using System.Linq;
  20. using System.Threading;
  21. using System.Threading.Tasks;
  22. namespace MediaBrowser.Dlna.PlayTo
  23. {
  24. public class PlayToController : ISessionController, IDisposable
  25. {
  26. private Device _device;
  27. private readonly SessionInfo _session;
  28. private readonly ISessionManager _sessionManager;
  29. private readonly IItemRepository _itemRepository;
  30. private readonly ILibraryManager _libraryManager;
  31. private readonly ILogger _logger;
  32. private readonly IDlnaManager _dlnaManager;
  33. private readonly IUserManager _userManager;
  34. private readonly IImageProcessor _imageProcessor;
  35. private readonly SsdpHandler _ssdpHandler;
  36. private readonly string _serverAddress;
  37. public bool IsSessionActive
  38. {
  39. get
  40. {
  41. return _device != null;
  42. }
  43. }
  44. public bool SupportsMediaControl
  45. {
  46. get { return IsSessionActive; }
  47. }
  48. private Timer _updateTimer;
  49. public PlayToController(SessionInfo session, ISessionManager sessionManager, IItemRepository itemRepository, ILibraryManager libraryManager, ILogger logger, IDlnaManager dlnaManager, IUserManager userManager, IImageProcessor imageProcessor, SsdpHandler ssdpHandler, string serverAddress)
  50. {
  51. _session = session;
  52. _itemRepository = itemRepository;
  53. _sessionManager = sessionManager;
  54. _libraryManager = libraryManager;
  55. _dlnaManager = dlnaManager;
  56. _userManager = userManager;
  57. _imageProcessor = imageProcessor;
  58. _ssdpHandler = ssdpHandler;
  59. _serverAddress = serverAddress;
  60. _logger = logger;
  61. }
  62. public void Init(Device device)
  63. {
  64. _device = device;
  65. _device.PlaybackStart += _device_PlaybackStart;
  66. _device.PlaybackProgress += _device_PlaybackProgress;
  67. _device.PlaybackStopped += _device_PlaybackStopped;
  68. _device.Start();
  69. _ssdpHandler.MessageReceived += _SsdpHandler_MessageReceived;
  70. _updateTimer = new Timer(updateTimer_Elapsed, null, 60000, 60000);
  71. }
  72. private async void updateTimer_Elapsed(object state)
  73. {
  74. if (DateTime.UtcNow >= _device.DateLastActivity.AddSeconds(120))
  75. {
  76. try
  77. {
  78. // Session is inactive, mark it for Disposal and don't start the elapsed timer.
  79. _sessionManager.ReportSessionEnded(_session.Id);
  80. }
  81. catch (Exception ex)
  82. {
  83. _logger.ErrorException("Error in ReportSessionEnded", ex);
  84. }
  85. }
  86. }
  87. private string GetServerAddress()
  88. {
  89. return _serverAddress;
  90. }
  91. async void _SsdpHandler_MessageReceived(object sender, SsdpMessageEventArgs e)
  92. {
  93. string nts;
  94. e.Headers.TryGetValue("NTS", out nts);
  95. string usn;
  96. if (!e.Headers.TryGetValue("USN", out usn)) usn = String.Empty;
  97. string nt;
  98. if (!e.Headers.TryGetValue("NT", out nt)) nt = String.Empty;
  99. if (String.Equals(e.Method, "NOTIFY", StringComparison.OrdinalIgnoreCase) &&
  100. String.Equals(nts, "ssdp:byebye", StringComparison.OrdinalIgnoreCase) &&
  101. usn.IndexOf(_device.Properties.UUID, StringComparison.OrdinalIgnoreCase) != -1 &&
  102. !_disposed)
  103. {
  104. if (usn.IndexOf("MediaRenderer:", StringComparison.OrdinalIgnoreCase) != -1 ||
  105. nt.IndexOf("MediaRenderer:", StringComparison.OrdinalIgnoreCase) != -1)
  106. {
  107. try
  108. {
  109. _sessionManager.ReportSessionEnded(_session.Id);
  110. }
  111. catch
  112. {
  113. // Could throw if the session is already gone
  114. }
  115. }
  116. }
  117. }
  118. async void _device_PlaybackStopped(object sender, PlaybackStoppedEventArgs e)
  119. {
  120. try
  121. {
  122. await _sessionManager.OnPlaybackStopped(new PlaybackStopInfo
  123. {
  124. ItemId = e.MediaInfo.Id,
  125. SessionId = _session.Id,
  126. PositionTicks = _device.Position.Ticks
  127. }).ConfigureAwait(false);
  128. }
  129. catch (Exception ex)
  130. {
  131. _logger.ErrorException("Error reporting progress", ex);
  132. }
  133. //await SetNext().ConfigureAwait(false);
  134. }
  135. async void _device_PlaybackStart(object sender, PlaybackStartEventArgs e)
  136. {
  137. var info = GetProgressInfo(e.MediaInfo);
  138. try
  139. {
  140. await _sessionManager.OnPlaybackStart(info).ConfigureAwait(false);
  141. }
  142. catch (Exception ex)
  143. {
  144. _logger.ErrorException("Error reporting progress", ex);
  145. }
  146. }
  147. async void _device_PlaybackProgress(object sender, PlaybackProgressEventArgs e)
  148. {
  149. var info = GetProgressInfo(e.MediaInfo);
  150. try
  151. {
  152. await _sessionManager.OnPlaybackProgress(info).ConfigureAwait(false);
  153. }
  154. catch (Exception ex)
  155. {
  156. _logger.ErrorException("Error reporting progress", ex);
  157. }
  158. }
  159. private PlaybackStartInfo GetProgressInfo(uBaseObject mediaInfo)
  160. {
  161. var ticks = _device.Position.Ticks;
  162. var info = StreamParams.ParseFromUrl(mediaInfo.Url, _libraryManager);
  163. if (!info.IsDirectStream)
  164. {
  165. ticks += info.StartPositionTicks;
  166. }
  167. return new PlaybackStartInfo
  168. {
  169. ItemId = mediaInfo.Id,
  170. SessionId = _session.Id,
  171. PositionTicks = ticks,
  172. IsMuted = _device.IsMuted,
  173. IsPaused = _device.IsPaused,
  174. MediaSourceId = info.MediaSourceId,
  175. AudioStreamIndex = info.AudioStreamIndex,
  176. SubtitleStreamIndex = info.SubtitleStreamIndex,
  177. VolumeLevel = _device.Volume,
  178. CanSeek = info.MediaSource == null ? _device.Duration.HasValue : info.MediaSource.RunTimeTicks.HasValue,
  179. PlayMethod = info.IsDirectStream ? PlayMethod.DirectStream : PlayMethod.Transcode,
  180. QueueableMediaTypes = new List<string> { mediaInfo.MediaType }
  181. };
  182. }
  183. #region SendCommands
  184. public async Task SendPlayCommand(PlayRequest command, CancellationToken cancellationToken)
  185. {
  186. _logger.Debug("{0} - Received PlayRequest: {1}", this._session.DeviceName, command.PlayCommand);
  187. var user = String.IsNullOrEmpty(command.ControllingUserId) ? null : _userManager.GetUserById(new Guid(command.ControllingUserId));
  188. var items = new List<BaseItem>();
  189. foreach (string id in command.ItemIds)
  190. {
  191. AddItemFromId(Guid.Parse(id), items);
  192. }
  193. var playlist = new List<PlaylistItem>();
  194. var isFirst = true;
  195. var serverAddress = GetServerAddress();
  196. foreach (var item in items)
  197. {
  198. if (isFirst && command.StartPositionTicks.HasValue)
  199. {
  200. playlist.Add(CreatePlaylistItem(item, user, command.StartPositionTicks.Value, serverAddress));
  201. isFirst = false;
  202. }
  203. else
  204. {
  205. playlist.Add(CreatePlaylistItem(item, user, 0, serverAddress));
  206. }
  207. }
  208. _logger.Debug("{0} - Playlist created", _session.DeviceName);
  209. if (command.PlayCommand == PlayCommand.PlayLast)
  210. {
  211. Playlist.AddRange(playlist);
  212. }
  213. if (command.PlayCommand == PlayCommand.PlayNext)
  214. {
  215. Playlist.AddRange(playlist);
  216. }
  217. _logger.Debug("{0} - Playing {1} items", _session.DeviceName, playlist.Count);
  218. if (!String.IsNullOrWhiteSpace(command.ControllingUserId))
  219. {
  220. await _sessionManager.LogSessionActivity(_session.Client, _session.ApplicationVersion, _session.DeviceId,
  221. _session.DeviceName, _session.RemoteEndPoint, user).ConfigureAwait(false);
  222. }
  223. await PlayItems(playlist).ConfigureAwait(false);
  224. }
  225. public Task SendPlaystateCommand(PlaystateRequest command, CancellationToken cancellationToken)
  226. {
  227. switch (command.Command)
  228. {
  229. case PlaystateCommand.Stop:
  230. Playlist.Clear();
  231. return _device.SetStop();
  232. case PlaystateCommand.Pause:
  233. return _device.SetPause();
  234. case PlaystateCommand.Unpause:
  235. return _device.SetPlay();
  236. case PlaystateCommand.Seek:
  237. {
  238. return Seek(command.SeekPositionTicks ?? 0);
  239. }
  240. case PlaystateCommand.NextTrack:
  241. return SetNext();
  242. case PlaystateCommand.PreviousTrack:
  243. return SetPrevious();
  244. }
  245. return Task.FromResult(true);
  246. }
  247. private async Task Seek(long newPosition)
  248. {
  249. var media = _device.CurrentMediaInfo;
  250. if (media != null)
  251. {
  252. var info = StreamParams.ParseFromUrl(media.Url, _libraryManager);
  253. if (info.Item != null && !info.IsDirectStream)
  254. {
  255. var user = _session.UserId.HasValue ? _userManager.GetUserById(_session.UserId.Value) : null;
  256. var newItem = CreatePlaylistItem(info.Item, user, newPosition, GetServerAddress(), info.MediaSourceId, info.AudioStreamIndex, info.SubtitleStreamIndex);
  257. await _device.SetAvTransport(newItem.StreamUrl, GetDlnaHeaders(newItem), newItem.Didl).ConfigureAwait(false);
  258. if (newItem.StreamInfo.IsDirectStream)
  259. {
  260. await _device.Seek(TimeSpan.FromTicks(newPosition)).ConfigureAwait(false);
  261. }
  262. return;
  263. }
  264. await _device.Seek(TimeSpan.FromTicks(newPosition)).ConfigureAwait(false);
  265. }
  266. }
  267. public Task SendUserDataChangeInfo(UserDataChangeInfo info, CancellationToken cancellationToken)
  268. {
  269. return Task.FromResult(true);
  270. }
  271. public Task SendRestartRequiredNotification(SystemInfo info, CancellationToken cancellationToken)
  272. {
  273. return Task.FromResult(true);
  274. }
  275. public Task SendServerRestartNotification(CancellationToken cancellationToken)
  276. {
  277. return Task.FromResult(true);
  278. }
  279. public Task SendSessionEndedNotification(SessionInfoDto sessionInfo, CancellationToken cancellationToken)
  280. {
  281. return Task.FromResult(true);
  282. }
  283. public Task SendPlaybackStartNotification(SessionInfoDto sessionInfo, CancellationToken cancellationToken)
  284. {
  285. return Task.FromResult(true);
  286. }
  287. public Task SendPlaybackStoppedNotification(SessionInfoDto sessionInfo, CancellationToken cancellationToken)
  288. {
  289. return Task.FromResult(true);
  290. }
  291. public Task SendServerShutdownNotification(CancellationToken cancellationToken)
  292. {
  293. return Task.FromResult(true);
  294. }
  295. public Task SendLibraryUpdateInfo(LibraryUpdateInfo info, CancellationToken cancellationToken)
  296. {
  297. return Task.FromResult(true);
  298. }
  299. #endregion
  300. #region Playlist
  301. private readonly List<PlaylistItem> _playlist = new List<PlaylistItem>();
  302. private List<PlaylistItem> Playlist
  303. {
  304. get
  305. {
  306. return _playlist;
  307. }
  308. }
  309. private void AddItemFromId(Guid id, List<BaseItem> list)
  310. {
  311. var item = _libraryManager.GetItemById(id);
  312. if (item.IsFolder)
  313. {
  314. foreach (var childId in _itemRepository.GetChildren(item.Id))
  315. {
  316. AddItemFromId(childId, list);
  317. }
  318. }
  319. else
  320. {
  321. if (item.MediaType == MediaType.Audio || item.MediaType == MediaType.Video)
  322. {
  323. list.Add(item);
  324. }
  325. }
  326. }
  327. private PlaylistItem CreatePlaylistItem(BaseItem item, User user, long startPostionTicks, string serverAddress)
  328. {
  329. return CreatePlaylistItem(item, user, startPostionTicks, serverAddress, null, null, null);
  330. }
  331. private PlaylistItem CreatePlaylistItem(BaseItem item, User user, long startPostionTicks, string serverAddress, string mediaSourceId, int? audioStreamIndex, int? subtitleStreamIndex)
  332. {
  333. var deviceInfo = _device.Properties;
  334. var profile = _dlnaManager.GetProfile(deviceInfo.ToDeviceIdentification()) ??
  335. _dlnaManager.GetDefaultProfile();
  336. var hasMediaSources = item as IHasMediaSources;
  337. var mediaSources = hasMediaSources != null
  338. ? (user == null ? hasMediaSources.GetMediaSources(true) : hasMediaSources.GetMediaSources(true, user)).ToList()
  339. : new List<MediaSourceInfo>();
  340. var playlistItem = GetPlaylistItem(item, mediaSources, profile, _session.DeviceId, mediaSourceId, audioStreamIndex, subtitleStreamIndex);
  341. playlistItem.StreamInfo.StartPositionTicks = startPostionTicks;
  342. playlistItem.StreamUrl = playlistItem.StreamInfo.ToUrl(serverAddress);
  343. var itemXml = new DidlBuilder(profile, user, _imageProcessor, serverAddress).GetItemDidl(item, _session.DeviceId, new Filter(), playlistItem.StreamInfo);
  344. playlistItem.Didl = itemXml;
  345. return playlistItem;
  346. }
  347. private string GetDlnaHeaders(PlaylistItem item)
  348. {
  349. var profile = item.Profile;
  350. var streamInfo = item.StreamInfo;
  351. if (streamInfo.MediaType == DlnaProfileType.Audio)
  352. {
  353. return new ContentFeatureBuilder(profile)
  354. .BuildAudioHeader(streamInfo.Container,
  355. streamInfo.AudioCodec,
  356. streamInfo.TargetAudioBitrate,
  357. streamInfo.TargetAudioSampleRate,
  358. streamInfo.TargetAudioChannels,
  359. streamInfo.IsDirectStream,
  360. streamInfo.RunTimeTicks,
  361. streamInfo.TranscodeSeekInfo);
  362. }
  363. if (streamInfo.MediaType == DlnaProfileType.Video)
  364. {
  365. return new ContentFeatureBuilder(profile)
  366. .BuildVideoHeader(streamInfo.Container,
  367. streamInfo.VideoCodec,
  368. streamInfo.AudioCodec,
  369. streamInfo.TargetWidth,
  370. streamInfo.TargetHeight,
  371. streamInfo.TargetVideoBitDepth,
  372. streamInfo.TargetVideoBitrate,
  373. streamInfo.TargetAudioChannels,
  374. streamInfo.TargetAudioBitrate,
  375. streamInfo.TargetTimestamp,
  376. streamInfo.IsDirectStream,
  377. streamInfo.RunTimeTicks,
  378. streamInfo.TargetVideoProfile,
  379. streamInfo.TargetVideoLevel,
  380. streamInfo.TargetFramerate,
  381. streamInfo.TargetPacketLength,
  382. streamInfo.TranscodeSeekInfo);
  383. }
  384. return null;
  385. }
  386. private PlaylistItem GetPlaylistItem(BaseItem item, List<MediaSourceInfo> mediaSources, DeviceProfile profile, string deviceId, string mediaSourceId, int? audioStreamIndex, int? subtitleStreamIndex)
  387. {
  388. var video = item as Video;
  389. if (video != null)
  390. {
  391. return new PlaylistItem
  392. {
  393. StreamInfo = new StreamBuilder().BuildVideoItem(new VideoOptions
  394. {
  395. ItemId = item.Id.ToString("N"),
  396. MediaSources = mediaSources,
  397. Profile = profile,
  398. DeviceId = deviceId,
  399. MaxBitrate = profile.MaxBitrate,
  400. MediaSourceId = mediaSourceId,
  401. AudioStreamIndex = audioStreamIndex,
  402. SubtitleStreamIndex = subtitleStreamIndex
  403. }),
  404. Profile = profile
  405. };
  406. }
  407. var audio = item as Audio;
  408. if (audio != null)
  409. {
  410. return new PlaylistItem
  411. {
  412. StreamInfo = new StreamBuilder().BuildAudioItem(new AudioOptions
  413. {
  414. ItemId = item.Id.ToString("N"),
  415. MediaSources = mediaSources,
  416. Profile = profile,
  417. DeviceId = deviceId,
  418. MaxBitrate = profile.MaxBitrate,
  419. MediaSourceId = mediaSourceId
  420. }),
  421. Profile = profile
  422. };
  423. }
  424. var photo = item as Photo;
  425. if (photo != null)
  426. {
  427. return new PlaylistItemFactory().Create(photo, profile);
  428. }
  429. throw new ArgumentException("Unrecognized item type.");
  430. }
  431. /// <summary>
  432. /// Plays the items.
  433. /// </summary>
  434. /// <param name="items">The items.</param>
  435. /// <returns></returns>
  436. private async Task<bool> PlayItems(IEnumerable<PlaylistItem> items)
  437. {
  438. Playlist.Clear();
  439. Playlist.AddRange(items);
  440. await SetNext();
  441. return true;
  442. }
  443. private async Task SetNext()
  444. {
  445. if (!Playlist.Any() || Playlist.All(i => i.PlayState != 0))
  446. {
  447. return;
  448. }
  449. var currentitem = Playlist.FirstOrDefault(i => i.PlayState == 1);
  450. if (currentitem != null)
  451. {
  452. currentitem.PlayState = 2;
  453. }
  454. var nextTrack = Playlist.FirstOrDefault(i => i.PlayState == 0);
  455. if (nextTrack == null)
  456. {
  457. await _device.SetStop();
  458. return;
  459. }
  460. nextTrack.PlayState = 1;
  461. var dlnaheaders = GetDlnaHeaders(nextTrack);
  462. await _device.SetAvTransport(nextTrack.StreamUrl, dlnaheaders, nextTrack.Didl);
  463. var streamInfo = nextTrack.StreamInfo;
  464. if (streamInfo.StartPositionTicks > 0 && streamInfo.IsDirectStream)
  465. await _device.Seek(TimeSpan.FromTicks(streamInfo.StartPositionTicks));
  466. }
  467. public Task SetPrevious()
  468. {
  469. if (!Playlist.Any() || Playlist.All(i => i.PlayState != 2))
  470. return Task.FromResult(false);
  471. var currentitem = Playlist.FirstOrDefault(i => i.PlayState == 1);
  472. var prevTrack = Playlist.LastOrDefault(i => i.PlayState == 2);
  473. if (currentitem != null)
  474. {
  475. currentitem.PlayState = 0;
  476. }
  477. if (prevTrack == null)
  478. return Task.FromResult(false);
  479. prevTrack.PlayState = 1;
  480. return _device.SetAvTransport(prevTrack.StreamInfo.ToDlnaUrl(GetServerAddress()), GetDlnaHeaders(prevTrack), prevTrack.Didl);
  481. }
  482. #endregion
  483. private bool _disposed;
  484. public void Dispose()
  485. {
  486. if (!_disposed)
  487. {
  488. _disposed = true;
  489. _device.PlaybackStart -= _device_PlaybackStart;
  490. _device.PlaybackProgress -= _device_PlaybackProgress;
  491. _device.PlaybackStopped -= _device_PlaybackStopped;
  492. _ssdpHandler.MessageReceived -= _SsdpHandler_MessageReceived;
  493. DisposeUpdateTimer();
  494. _device.Dispose();
  495. }
  496. }
  497. private void DisposeUpdateTimer()
  498. {
  499. if (_updateTimer != null)
  500. {
  501. _updateTimer.Dispose();
  502. _updateTimer = null;
  503. }
  504. }
  505. private readonly CultureInfo _usCulture = new CultureInfo("en-US");
  506. public Task SendGeneralCommand(GeneralCommand command, CancellationToken cancellationToken)
  507. {
  508. GeneralCommandType commandType;
  509. if (Enum.TryParse(command.Name, true, out commandType))
  510. {
  511. switch (commandType)
  512. {
  513. case GeneralCommandType.VolumeDown:
  514. return _device.VolumeDown();
  515. case GeneralCommandType.VolumeUp:
  516. return _device.VolumeUp();
  517. case GeneralCommandType.Mute:
  518. return _device.Mute();
  519. case GeneralCommandType.Unmute:
  520. return _device.Unmute();
  521. case GeneralCommandType.ToggleMute:
  522. return _device.ToggleMute();
  523. case GeneralCommandType.SetAudioStreamIndex:
  524. {
  525. string arg;
  526. if (command.Arguments.TryGetValue("Index", out arg))
  527. {
  528. int val;
  529. if (Int32.TryParse(arg, NumberStyles.Any, _usCulture, out val))
  530. {
  531. return SetAudioStreamIndex(val);
  532. }
  533. throw new ArgumentException("Unsupported SetAudioStreamIndex value supplied.");
  534. }
  535. throw new ArgumentException("SetAudioStreamIndex argument cannot be null");
  536. }
  537. case GeneralCommandType.SetSubtitleStreamIndex:
  538. {
  539. string arg;
  540. if (command.Arguments.TryGetValue("Index", out arg))
  541. {
  542. int val;
  543. if (Int32.TryParse(arg, NumberStyles.Any, _usCulture, out val))
  544. {
  545. return SetSubtitleStreamIndex(val);
  546. }
  547. throw new ArgumentException("Unsupported SetSubtitleStreamIndex value supplied.");
  548. }
  549. throw new ArgumentException("SetSubtitleStreamIndex argument cannot be null");
  550. }
  551. case GeneralCommandType.SetVolume:
  552. {
  553. string arg;
  554. if (command.Arguments.TryGetValue("Volume", out arg))
  555. {
  556. int volume;
  557. if (Int32.TryParse(arg, NumberStyles.Any, _usCulture, out volume))
  558. {
  559. return _device.SetVolume(volume);
  560. }
  561. throw new ArgumentException("Unsupported volume value supplied.");
  562. }
  563. throw new ArgumentException("Volume argument cannot be null");
  564. }
  565. default:
  566. return Task.FromResult(true);
  567. }
  568. }
  569. return Task.FromResult(true);
  570. }
  571. private async Task SetAudioStreamIndex(int? newIndex)
  572. {
  573. var media = _device.CurrentMediaInfo;
  574. if (media != null)
  575. {
  576. var info = StreamParams.ParseFromUrl(media.Url, _libraryManager);
  577. if (info.Item != null && !info.IsDirectStream)
  578. {
  579. var newPosition = _device.Position.Ticks;
  580. var user = _session.UserId.HasValue ? _userManager.GetUserById(_session.UserId.Value) : null;
  581. var newItem = CreatePlaylistItem(info.Item, user, newPosition, GetServerAddress(), info.MediaSourceId, newIndex, info.SubtitleStreamIndex);
  582. await _device.SetAvTransport(newItem.StreamUrl, GetDlnaHeaders(newItem), newItem.Didl).ConfigureAwait(false);
  583. if (newItem.StreamInfo.IsDirectStream)
  584. {
  585. await _device.Seek(TimeSpan.FromTicks(newPosition)).ConfigureAwait(false);
  586. }
  587. }
  588. }
  589. }
  590. private async Task SetSubtitleStreamIndex(int? newIndex)
  591. {
  592. var media = _device.CurrentMediaInfo;
  593. if (media != null)
  594. {
  595. var info = StreamParams.ParseFromUrl(media.Url, _libraryManager);
  596. if (info.Item != null && !info.IsDirectStream)
  597. {
  598. var newPosition = _device.Position.Ticks;
  599. var user = _session.UserId.HasValue ? _userManager.GetUserById(_session.UserId.Value) : null;
  600. var newItem = CreatePlaylistItem(info.Item, user, newPosition, GetServerAddress(), info.MediaSourceId, info.AudioStreamIndex, newIndex);
  601. await _device.SetAvTransport(newItem.StreamUrl, GetDlnaHeaders(newItem), newItem.Didl).ConfigureAwait(false);
  602. if (newItem.StreamInfo.IsDirectStream)
  603. {
  604. await _device.Seek(TimeSpan.FromTicks(newPosition)).ConfigureAwait(false);
  605. }
  606. }
  607. }
  608. }
  609. private class StreamParams
  610. {
  611. public string ItemId { get; set; }
  612. public bool IsDirectStream { get; set; }
  613. public long StartPositionTicks { get; set; }
  614. public int? AudioStreamIndex { get; set; }
  615. public int? SubtitleStreamIndex { get; set; }
  616. public string DeviceProfileId { get; set; }
  617. public string DeviceId { get; set; }
  618. public string MediaSourceId { get; set; }
  619. public BaseItem Item { get; set; }
  620. public MediaSourceInfo MediaSource { get; set; }
  621. private static string GetItemId(string url)
  622. {
  623. var parts = url.Split('/');
  624. for (var i = 0; i < parts.Length; i++)
  625. {
  626. var part = parts[i];
  627. if (string.Equals(part, "audio", StringComparison.OrdinalIgnoreCase) ||
  628. string.Equals(part, "videos", StringComparison.OrdinalIgnoreCase))
  629. {
  630. if (parts.Length > i + 1)
  631. {
  632. return parts[i + 1];
  633. }
  634. }
  635. }
  636. return null;
  637. }
  638. public static StreamParams ParseFromUrl(string url, ILibraryManager libraryManager)
  639. {
  640. var request = new StreamParams
  641. {
  642. ItemId = GetItemId(url)
  643. };
  644. if (string.IsNullOrWhiteSpace(request.ItemId))
  645. {
  646. return request;
  647. }
  648. const string srch = "params=";
  649. var index = url.IndexOf(srch, StringComparison.OrdinalIgnoreCase);
  650. if (index == -1) return request;
  651. var vals = url.Substring(index + srch.Length).Split(';');
  652. for (var i = 0; i < vals.Length; i++)
  653. {
  654. var val = vals[i];
  655. if (string.IsNullOrWhiteSpace(val))
  656. {
  657. continue;
  658. }
  659. if (i == 0)
  660. {
  661. request.DeviceProfileId = val;
  662. }
  663. else if (i == 1)
  664. {
  665. request.DeviceId = val;
  666. }
  667. else if (i == 2)
  668. {
  669. request.MediaSourceId = val;
  670. }
  671. else if (i == 3)
  672. {
  673. request.IsDirectStream = string.Equals("true", val, StringComparison.OrdinalIgnoreCase);
  674. }
  675. else if (i == 6)
  676. {
  677. request.AudioStreamIndex = int.Parse(val, CultureInfo.InvariantCulture);
  678. }
  679. else if (i == 7)
  680. {
  681. request.SubtitleStreamIndex = int.Parse(val, CultureInfo.InvariantCulture);
  682. }
  683. else if (i == 14)
  684. {
  685. request.StartPositionTicks = long.Parse(val, CultureInfo.InvariantCulture);
  686. }
  687. }
  688. request.Item = string.IsNullOrWhiteSpace(request.ItemId)
  689. ? null
  690. : libraryManager.GetItemById(new Guid(request.ItemId));
  691. var hasMediaSources = request.Item as IHasMediaSources;
  692. request.MediaSource = hasMediaSources == null ?
  693. null :
  694. hasMediaSources.GetMediaSources(false).FirstOrDefault(i => string.Equals(i.Id, request.MediaSourceId, StringComparison.OrdinalIgnoreCase));
  695. return request;
  696. }
  697. }
  698. }
  699. }