jellyfin/MediaBrowser.Dlna/PlayTo/DlnaController.cs

645 lines
22 KiB
C#
Raw Normal View History

2014-04-15 19:17:48 -07:00
using MediaBrowser.Common.Net;
2014-03-16 21:25:11 -07:00
using MediaBrowser.Controller;
2014-03-13 12:08:02 -07:00
using MediaBrowser.Controller.Dlna;
2014-04-17 22:03:01 -07:00
using MediaBrowser.Controller.Dto;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Persistence;
using MediaBrowser.Controller.Session;
2014-04-01 15:23:07 -07:00
using MediaBrowser.Model.Dlna;
2014-04-17 22:03:01 -07:00
using MediaBrowser.Model.Dto;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Logging;
using MediaBrowser.Model.Session;
using System;
using System.Collections.Generic;
2014-04-15 19:17:48 -07:00
using System.Globalization;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
namespace MediaBrowser.Dlna.PlayTo
{
public class PlayToController : ISessionController, IDisposable
{
private Device _device;
private readonly SessionInfo _session;
private readonly ISessionManager _sessionManager;
private readonly IItemRepository _itemRepository;
private readonly ILibraryManager _libraryManager;
private readonly INetworkManager _networkManager;
private readonly ILogger _logger;
2014-03-13 12:08:02 -07:00
private readonly IDlnaManager _dlnaManager;
private readonly IUserManager _userManager;
2014-03-16 21:25:11 -07:00
private readonly IServerApplicationHost _appHost;
2014-04-17 22:03:01 -07:00
private readonly IDtoService _dtoService;
2014-03-16 21:25:11 -07:00
public bool SupportsMediaRemoteControl
{
get { return true; }
}
public bool IsSessionActive
{
get
{
if (_device == null || _device.UpdateTime == default(DateTime))
return false;
return DateTime.UtcNow <= _device.UpdateTime.AddSeconds(30);
}
}
2014-04-17 22:03:01 -07:00
public PlayToController(SessionInfo session, ISessionManager sessionManager, IItemRepository itemRepository, ILibraryManager libraryManager, ILogger logger, INetworkManager networkManager, IDlnaManager dlnaManager, IUserManager userManager, IServerApplicationHost appHost, IDtoService dtoService)
{
_session = session;
_itemRepository = itemRepository;
_sessionManager = sessionManager;
_libraryManager = libraryManager;
_networkManager = networkManager;
2014-03-13 14:00:17 -07:00
_dlnaManager = dlnaManager;
_userManager = userManager;
2014-03-16 21:25:11 -07:00
_appHost = appHost;
2014-04-17 22:03:01 -07:00
_dtoService = dtoService;
_logger = logger;
}
2014-03-13 12:08:02 -07:00
public void Init(Device device)
{
_device = device;
2014-04-22 10:25:54 -07:00
_device.PlaybackStart += _device_PlaybackStart;
_device.PlaybackProgress += _device_PlaybackProgress;
_device.PlaybackStopped += _device_PlaybackStopped;
_device.Start();
2014-04-22 10:25:54 -07:00
_updateTimer = new Timer(updateTimer_Elapsed, null, 30000, 30000);
}
2014-04-22 10:25:54 -07:00
async void _device_PlaybackStopped(object sender, PlaybackStoppedEventArgs e)
{
2014-04-22 10:25:54 -07:00
try
{
2014-04-15 19:17:48 -07:00
await _sessionManager.OnPlaybackStopped(new PlaybackStopInfo
{
2014-04-22 10:25:54 -07:00
ItemId = e.MediaInfo.Id,
SessionId = _session.Id,
PositionTicks = _device.Position.Ticks
}).ConfigureAwait(false);
}
2014-04-22 10:25:54 -07:00
catch (Exception ex)
{
_logger.ErrorException("Error reporting progress", ex);
}
await SetNext().ConfigureAwait(false);
}
2014-04-22 10:25:54 -07:00
async void _device_PlaybackStart(object sender, PlaybackStartEventArgs e)
{
2014-04-22 10:25:54 -07:00
var playlistItem = Playlist.FirstOrDefault(p => p.PlayState == 1);
2014-04-22 10:25:54 -07:00
if (playlistItem != null)
{
var streamInfo = playlistItem.StreamInfo;
2014-04-22 10:25:54 -07:00
var info = GetProgressInfo(streamInfo, e.MediaInfo);
2014-04-22 10:25:54 -07:00
try
{
await _sessionManager.OnPlaybackStart(info).ConfigureAwait(false);
}
catch (Exception ex)
{
_logger.ErrorException("Error reporting progress", ex);
}
}
}
2014-04-22 10:25:54 -07:00
async void _device_PlaybackProgress(object sender, PlaybackProgressEventArgs e)
{
2014-04-22 10:25:54 -07:00
var playlistItem = Playlist.FirstOrDefault(p => p.PlayState == 1);
2014-04-22 10:25:54 -07:00
if (playlistItem != null)
2014-03-16 21:25:11 -07:00
{
2014-04-22 10:25:54 -07:00
var streamInfo = playlistItem.StreamInfo;
var info = GetProgressInfo(streamInfo, e.MediaInfo);
2014-03-24 22:25:03 -07:00
try
{
2014-04-22 10:25:54 -07:00
await _sessionManager.OnPlaybackProgress(info).ConfigureAwait(false);
2014-03-24 22:25:03 -07:00
}
catch (Exception ex)
{
2014-04-22 10:25:54 -07:00
_logger.ErrorException("Error reporting progress", ex);
2014-03-24 22:25:03 -07:00
}
2014-03-16 21:25:11 -07:00
}
}
2014-04-22 10:25:54 -07:00
private PlaybackStartInfo GetProgressInfo(StreamInfo streamInfo, uBaseObject mediaInfo)
{
2014-04-22 10:25:54 -07:00
var ticks = _device.Position.Ticks;
2014-04-22 10:25:54 -07:00
if (!streamInfo.IsDirectStream)
{
ticks += streamInfo.StartPositionTicks;
}
2014-04-15 19:17:48 -07:00
2014-04-22 10:25:54 -07:00
return new PlaybackStartInfo
{
2014-04-22 10:25:54 -07:00
ItemId = mediaInfo.Id,
SessionId = _session.Id,
PositionTicks = ticks,
IsMuted = _device.IsMuted,
IsPaused = _device.IsPaused,
MediaSourceId = streamInfo.MediaSourceId,
AudioStreamIndex = streamInfo.AudioStreamIndex,
SubtitleStreamIndex = streamInfo.SubtitleStreamIndex,
VolumeLevel = _device.Volume,
CanSeek = streamInfo.RunTimeTicks.HasValue,
PlayMethod = streamInfo.IsDirectStream ? PlayMethod.DirectStream : PlayMethod.Transcode,
QueueableMediaTypes = new List<string> { mediaInfo.MediaType }
};
}
2014-04-17 22:03:01 -07:00
2014-04-22 10:25:54 -07:00
#region Device EventHandlers & Update Timer
2014-04-06 10:53:23 -07:00
2014-04-22 10:25:54 -07:00
Timer _updateTimer;
2014-04-06 10:53:23 -07:00
2014-04-22 10:25:54 -07:00
/// <summary>
/// Handles the Elapsed event of the updateTimer control.
/// </summary>
/// <param name="state">The state.</param>
private async void updateTimer_Elapsed(object state)
{
if (!IsSessionActive)
{
_updateTimer.Change(Timeout.Infinite, Timeout.Infinite);
2014-04-22 10:25:54 -07:00
try
{
// Session is inactive, mark it for Disposal and don't start the elapsed timer.
await _sessionManager.ReportSessionEnded(_session.Id);
}
catch (Exception ex)
{
_logger.ErrorException("Error in ReportSessionEnded", ex);
}
}
}
#endregion
#region SendCommands
public async Task SendPlayCommand(PlayRequest command, CancellationToken cancellationToken)
{
_logger.Debug("{0} - Received PlayRequest: {1}", this._session.DeviceName, command.PlayCommand);
var items = new List<BaseItem>();
foreach (string id in command.ItemIds)
{
AddItemFromId(Guid.Parse(id), items);
}
var playlist = new List<PlaylistItem>();
var isFirst = true;
var serverAddress = GetServerAddress();
foreach (var item in items)
{
if (isFirst && command.StartPositionTicks.HasValue)
{
playlist.Add(CreatePlaylistItem(item, command.StartPositionTicks.Value, serverAddress));
isFirst = false;
}
else
{
playlist.Add(CreatePlaylistItem(item, 0, serverAddress));
}
}
_logger.Debug("{0} - Playlist created", _session.DeviceName);
if (command.PlayCommand == PlayCommand.PlayLast)
{
2014-04-22 10:25:54 -07:00
Playlist.AddRange(playlist);
}
if (command.PlayCommand == PlayCommand.PlayNext)
{
2014-04-22 10:25:54 -07:00
Playlist.AddRange(playlist);
}
_logger.Debug("{0} - Playing {1} items", _session.DeviceName, playlist.Count);
if (!string.IsNullOrWhiteSpace(command.ControllingUserId))
{
var userId = new Guid(command.ControllingUserId);
var user = _userManager.GetUserById(userId);
await _sessionManager.LogSessionActivity(_session.Client, _session.ApplicationVersion, _session.DeviceId,
_session.DeviceName, _session.RemoteEndPoint, user).ConfigureAwait(false);
}
await PlayItems(playlist).ConfigureAwait(false);
}
public Task SendPlaystateCommand(PlaystateRequest command, CancellationToken cancellationToken)
{
switch (command.Command)
{
case PlaystateCommand.Stop:
Playlist.Clear();
return _device.SetStop();
case PlaystateCommand.Pause:
return _device.SetPause();
case PlaystateCommand.Unpause:
return _device.SetPlay();
case PlaystateCommand.Seek:
2014-04-06 10:53:23 -07:00
//var playlistItem = Playlist.FirstOrDefault(p => p.PlayState == 1);
//if (playlistItem != null && playlistItem.Transcode && _currentItem != null)
//{
// var newItem = CreatePlaylistItem(_currentItem, command.SeekPositionTicks ?? 0, GetServerAddress());
// playlistItem.StartPositionTicks = newItem.StartPositionTicks;
// playlistItem.StreamUrl = newItem.StreamUrl;
// playlistItem.Didl = newItem.Didl;
// return _device.SetAvTransport(playlistItem.StreamUrl, GetDlnaHeaders(playlistItem), playlistItem.Didl);
//}
return _device.Seek(TimeSpan.FromTicks(command.SeekPositionTicks ?? 0));
case PlaystateCommand.NextTrack:
return SetNext();
case PlaystateCommand.PreviousTrack:
return SetPrevious();
}
return Task.FromResult(true);
}
public Task SendUserDataChangeInfo(UserDataChangeInfo info, CancellationToken cancellationToken)
{
return Task.FromResult(true);
}
public Task SendRestartRequiredNotification(CancellationToken cancellationToken)
{
return Task.FromResult(true);
}
public Task SendServerRestartNotification(CancellationToken cancellationToken)
{
return Task.FromResult(true);
}
2014-04-06 10:53:23 -07:00
public Task SendSessionEndedNotification(SessionInfoDto sessionInfo, CancellationToken cancellationToken)
{
return Task.FromResult(true);
}
public Task SendPlaybackStartNotification(SessionInfoDto sessionInfo, CancellationToken cancellationToken)
{
return Task.FromResult(true);
}
public Task SendPlaybackStoppedNotification(SessionInfoDto sessionInfo, CancellationToken cancellationToken)
{
return Task.FromResult(true);
}
public Task SendServerShutdownNotification(CancellationToken cancellationToken)
{
return Task.FromResult(true);
}
public Task SendLibraryUpdateInfo(LibraryUpdateInfo info, CancellationToken cancellationToken)
{
return Task.FromResult(true);
}
public Task SendMessageCommand(MessageCommand command, CancellationToken cancellationToken)
{
return Task.FromResult(true);
}
#endregion
#region Playlist
2014-04-22 10:25:54 -07:00
private readonly List<PlaylistItem> _playlist = new List<PlaylistItem>();
private List<PlaylistItem> Playlist
{
get
{
return _playlist;
}
}
private void AddItemFromId(Guid id, List<BaseItem> list)
{
var item = _libraryManager.GetItemById(id);
if (item.IsFolder)
{
foreach (var childId in _itemRepository.GetChildren(item.Id))
{
AddItemFromId(childId, list);
}
}
else
{
if (item.MediaType == MediaType.Audio || item.MediaType == MediaType.Video)
{
list.Add(item);
}
}
}
private string GetServerAddress()
{
return string.Format("{0}://{1}:{2}/mediabrowser",
"http",
_networkManager.GetLocalIpAddresses().FirstOrDefault() ?? "localhost",
2014-03-16 21:25:11 -07:00
_appHost.HttpServerPort
);
}
private PlaylistItem CreatePlaylistItem(BaseItem item, long startPostionTicks, string serverAddress)
{
2014-03-13 12:08:02 -07:00
var deviceInfo = _device.Properties;
2014-03-26 08:17:36 -07:00
var profile = _dlnaManager.GetProfile(deviceInfo.ToDeviceIdentification()) ??
_dlnaManager.GetDefaultProfile();
2014-03-23 09:42:02 -07:00
2014-04-17 22:03:01 -07:00
var mediaSources = item is Audio || item is Video
? _dtoService.GetMediaSources(item)
: new List<MediaSourceInfo>();
2014-04-17 22:03:01 -07:00
var playlistItem = GetPlaylistItem(item, mediaSources, profile, _session.DeviceId);
playlistItem.StreamInfo.StartPositionTicks = startPostionTicks;
playlistItem.StreamUrl = playlistItem.StreamInfo.ToUrl(serverAddress);
2014-04-17 22:03:01 -07:00
var mediaStreams = mediaSources
.Where(i => string.Equals(i.Id, playlistItem.StreamInfo.MediaSourceId))
.SelectMany(i => i.MediaStreams)
.ToList();
playlistItem.Didl = DidlBuilder.Build(item, _session.UserId.ToString(), serverAddress, playlistItem.StreamUrl, mediaStreams, profile.EnableAlbumArtInDidl);
return playlistItem;
}
2014-03-22 22:10:33 -07:00
private string GetDlnaHeaders(PlaylistItem item)
{
2014-04-17 22:03:01 -07:00
var streamInfo = item.StreamInfo;
var orgOp = !streamInfo.IsDirectStream ? ";DLNA.ORG_OP=00" : ";DLNA.ORG_OP=01";
2014-03-22 22:10:33 -07:00
2014-04-17 22:03:01 -07:00
var orgCi = !streamInfo.IsDirectStream ? ";DLNA.ORG_CI=0" : ";DLNA.ORG_CI=1";
2014-03-22 22:10:33 -07:00
const string dlnaflags = ";DLNA.ORG_FLAGS=01500000000000000000000000000000";
string contentFeatures;
2014-04-17 22:03:01 -07:00
var container = streamInfo.Container.TrimStart('.');
2014-03-22 22:10:33 -07:00
if (string.Equals(container, "mp3", StringComparison.OrdinalIgnoreCase))
{
contentFeatures = "DLNA.ORG_PN=MP3";
}
else if (string.Equals(container, "wma", StringComparison.OrdinalIgnoreCase))
{
contentFeatures = "DLNA.ORG_PN=WMABASE";
}
else if (string.Equals(container, "wmw", StringComparison.OrdinalIgnoreCase))
{
contentFeatures = "DLNA.ORG_PN=WMVMED_BASE";
}
else if (string.Equals(container, "asf", StringComparison.OrdinalIgnoreCase))
{
contentFeatures = "DLNA.ORG_PN=WMVMED_BASE";
}
else if (string.Equals(container, "avi", StringComparison.OrdinalIgnoreCase))
{
contentFeatures = "DLNA.ORG_PN=AVI";
}
else if (string.Equals(container, "mkv", StringComparison.OrdinalIgnoreCase))
{
contentFeatures = "DLNA.ORG_PN=MATROSKA";
}
else if (string.Equals(container, "mp4", StringComparison.OrdinalIgnoreCase))
{
contentFeatures = "DLNA.ORG_PN=AVC_MP4_MP_HD_720p_AAC";
}
else if (string.Equals(container, "mpeg", StringComparison.OrdinalIgnoreCase))
{
contentFeatures = "DLNA.ORG_PN=MPEG_PS_PAL";
}
else if (string.Equals(container, "ts", StringComparison.OrdinalIgnoreCase))
{
contentFeatures = "DLNA.ORG_PN=MPEG_PS_PAL";
}
2014-04-17 22:03:01 -07:00
else if (streamInfo.MediaType == DlnaProfileType.Video)
2014-03-22 22:10:33 -07:00
{
// Default to AVI for video
contentFeatures = "DLNA.ORG_PN=AVI";
}
else
{
// Default to MP3 for audio
contentFeatures = "DLNA.ORG_PN=MP3";
}
return (contentFeatures + orgOp + orgCi + dlnaflags).Trim(';');
}
2014-04-06 11:34:47 -07:00
2014-04-17 22:03:01 -07:00
private PlaylistItem GetPlaylistItem(BaseItem item, List<MediaSourceInfo> mediaSources, DeviceProfile profile, string deviceId)
{
var video = item as Video;
if (video != null)
{
2014-04-17 22:03:01 -07:00
return new PlaylistItem
{
StreamInfo = new StreamBuilder().BuildVideoItem(new VideoOptions
{
ItemId = item.Id.ToString("N"),
MediaSources = mediaSources,
Profile = profile,
DeviceId = deviceId
})
};
}
var audio = item as Audio;
if (audio != null)
{
2014-04-17 22:03:01 -07:00
return new PlaylistItem
{
StreamInfo = new StreamBuilder().BuildAudioItem(new AudioOptions
{
ItemId = item.Id.ToString("N"),
MediaSources = mediaSources,
Profile = profile,
DeviceId = deviceId
})
};
}
var photo = item as Photo;
if (photo != null)
{
2014-03-22 22:10:33 -07:00
return new PlaylistItemFactory().Create(photo, profile);
}
throw new ArgumentException("Unrecognized item type.");
}
/// <summary>
/// Plays the items.
/// </summary>
/// <param name="items">The items.</param>
/// <returns></returns>
private async Task<bool> PlayItems(IEnumerable<PlaylistItem> items)
{
Playlist.Clear();
Playlist.AddRange(items);
await SetNext();
return true;
}
private async Task<bool> SetNext()
{
if (!Playlist.Any() || Playlist.All(i => i.PlayState != 0))
{
return true;
}
var currentitem = Playlist.FirstOrDefault(i => i.PlayState == 1);
if (currentitem != null)
{
currentitem.PlayState = 2;
}
var nextTrack = Playlist.FirstOrDefault(i => i.PlayState == 0);
if (nextTrack == null)
{
await _device.SetStop();
return true;
}
2014-03-22 22:10:33 -07:00
nextTrack.PlayState = 1;
2014-03-22 22:10:33 -07:00
var dlnaheaders = GetDlnaHeaders(nextTrack);
_logger.Debug("{0} - SetAvTransport Uri: {1} DlnaHeaders: {2}", _device.Properties.Name, nextTrack.StreamUrl, dlnaheaders);
await _device.SetAvTransport(nextTrack.StreamUrl, dlnaheaders, nextTrack.Didl);
2014-04-06 11:34:47 -07:00
2014-04-17 22:03:01 -07:00
var streamInfo = nextTrack.StreamInfo;
if (streamInfo.StartPositionTicks > 0 && streamInfo.IsDirectStream)
await _device.Seek(TimeSpan.FromTicks(streamInfo.StartPositionTicks));
2014-04-06 11:34:47 -07:00
return true;
}
2014-04-22 10:25:54 -07:00
public Task SetPrevious()
{
if (!Playlist.Any() || Playlist.All(i => i.PlayState != 2))
return Task.FromResult(false);
var currentitem = Playlist.FirstOrDefault(i => i.PlayState == 1);
var prevTrack = Playlist.LastOrDefault(i => i.PlayState == 2);
if (currentitem != null)
{
currentitem.PlayState = 0;
}
if (prevTrack == null)
return Task.FromResult(false);
prevTrack.PlayState = 1;
2014-04-17 22:03:01 -07:00
return _device.SetAvTransport(prevTrack.StreamInfo.ToDlnaUrl(GetServerAddress()), GetDlnaHeaders(prevTrack), prevTrack.Didl);
}
#endregion
private bool _disposed;
public void Dispose()
{
if (!_disposed)
{
_disposed = true;
2014-04-22 10:25:54 -07:00
_device.PlaybackStart -= _device_PlaybackStart;
_device.PlaybackProgress -= _device_PlaybackProgress;
_device.PlaybackStopped -= _device_PlaybackStopped;
2014-03-16 21:25:11 -07:00
_updateTimer.Dispose();
_device.Dispose();
}
}
2014-03-28 12:58:18 -07:00
2014-04-14 20:54:52 -07:00
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
2014-04-15 19:17:48 -07:00
2014-03-31 14:04:22 -07:00
public Task SendGeneralCommand(GeneralCommand command, CancellationToken cancellationToken)
2014-03-28 12:58:18 -07:00
{
2014-03-31 14:04:22 -07:00
GeneralCommandType commandType;
2014-04-13 23:42:51 -07:00
if (Enum.TryParse(command.Name, true, out commandType))
2014-03-31 14:04:22 -07:00
{
switch (commandType)
{
case GeneralCommandType.VolumeDown:
return _device.VolumeDown();
case GeneralCommandType.VolumeUp:
return _device.VolumeUp();
case GeneralCommandType.Mute:
2014-04-22 10:25:54 -07:00
return _device.Mute();
2014-03-31 14:04:22 -07:00
case GeneralCommandType.Unmute:
2014-04-22 10:25:54 -07:00
return _device.Unmute();
2014-03-31 14:04:22 -07:00
case GeneralCommandType.ToggleMute:
return _device.ToggleMute();
2014-04-14 20:54:52 -07:00
case GeneralCommandType.SetVolume:
{
2014-04-15 19:17:48 -07:00
string volumeArg;
2014-04-14 20:54:52 -07:00
2014-04-15 19:17:48 -07:00
if (command.Arguments.TryGetValue("Volume", out volumeArg))
2014-04-14 20:54:52 -07:00
{
2014-04-15 19:17:48 -07:00
int volume;
if (int.TryParse(volumeArg, NumberStyles.Any, _usCulture, out volume))
{
return _device.SetVolume(volume);
}
throw new ArgumentException("Unsupported volume value supplied.");
2014-04-14 20:54:52 -07:00
}
2014-04-15 19:17:48 -07:00
throw new ArgumentException("Volume argument cannot be null");
2014-04-14 20:54:52 -07:00
}
2014-03-31 14:04:22 -07:00
default:
return Task.FromResult(true);
}
}
return Task.FromResult(true);
2014-03-28 12:58:18 -07:00
}
}
}