12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304 |
- #nullable disable
- #pragma warning disable CS1591
- using System;
- using System.Collections.Generic;
- using System.Globalization;
- using System.Linq;
- using System.Net.Http;
- using System.Security;
- using System.Threading;
- using System.Threading.Tasks;
- using System.Xml;
- using System.Xml.Linq;
- using Emby.Dlna.Common;
- using Emby.Dlna.Ssdp;
- using Microsoft.Extensions.Logging;
- namespace Emby.Dlna.PlayTo
- {
- public class Device : IDisposable
- {
- private readonly IHttpClientFactory _httpClientFactory;
- private readonly ILogger _logger;
- private readonly object _timerLock = new object();
- private Timer _timer;
- private int _muteVol;
- private int _volume;
- private DateTime _lastVolumeRefresh;
- private bool _volumeRefreshActive;
- private int _connectFailureCount;
- private bool _disposed;
- public Device(DeviceInfo deviceProperties, IHttpClientFactory httpClientFactory, ILogger logger)
- {
- Properties = deviceProperties;
- _httpClientFactory = httpClientFactory;
- _logger = logger;
- }
- public event EventHandler<PlaybackStartEventArgs> PlaybackStart;
- public event EventHandler<PlaybackProgressEventArgs> PlaybackProgress;
- public event EventHandler<PlaybackStoppedEventArgs> PlaybackStopped;
- public event EventHandler<MediaChangedEventArgs> MediaChanged;
- public DeviceInfo Properties { get; set; }
- public bool IsMuted { get; set; }
- public int Volume
- {
- get
- {
- RefreshVolumeIfNeeded().GetAwaiter().GetResult();
- return _volume;
- }
- set => _volume = value;
- }
- public TimeSpan? Duration { get; set; }
- public TimeSpan Position { get; set; } = TimeSpan.FromSeconds(0);
- public TransportState TransportState { get; private set; }
- public bool IsPlaying => TransportState == TransportState.PLAYING;
- public bool IsPaused => TransportState == TransportState.PAUSED_PLAYBACK;
- public bool IsStopped => TransportState == TransportState.STOPPED;
- public Action OnDeviceUnavailable { get; set; }
- private TransportCommands AvCommands { get; set; }
- private TransportCommands RendererCommands { get; set; }
- public UBaseObject CurrentMediaInfo { get; private set; }
- public void Start()
- {
- _logger.LogDebug("Dlna Device.Start");
- _timer = new Timer(TimerCallback, null, 1000, Timeout.Infinite);
- }
- private Task RefreshVolumeIfNeeded()
- {
- if (_volumeRefreshActive
- && DateTime.UtcNow >= _lastVolumeRefresh.AddSeconds(5))
- {
- _lastVolumeRefresh = DateTime.UtcNow;
- return RefreshVolume();
- }
- return Task.CompletedTask;
- }
- private async Task RefreshVolume(CancellationToken cancellationToken = default)
- {
- if (_disposed)
- {
- return;
- }
- try
- {
- await GetVolume(cancellationToken).ConfigureAwait(false);
- await GetMute(cancellationToken).ConfigureAwait(false);
- }
- catch (Exception ex)
- {
- _logger.LogError(ex, "Error updating device volume info for {DeviceName}", Properties.Name);
- }
- }
- private void RestartTimer(bool immediate = false)
- {
- lock (_timerLock)
- {
- if (_disposed)
- {
- return;
- }
- _volumeRefreshActive = true;
- var time = immediate ? 100 : 10000;
- _timer.Change(time, Timeout.Infinite);
- }
- }
- /// <summary>
- /// Restarts the timer in inactive mode.
- /// </summary>
- private void RestartTimerInactive()
- {
- lock (_timerLock)
- {
- if (_disposed)
- {
- return;
- }
- _volumeRefreshActive = false;
- _timer.Change(Timeout.Infinite, Timeout.Infinite);
- }
- }
- public Task VolumeDown(CancellationToken cancellationToken)
- {
- var sendVolume = Math.Max(Volume - 5, 0);
- return SetVolume(sendVolume, cancellationToken);
- }
- public Task VolumeUp(CancellationToken cancellationToken)
- {
- var sendVolume = Math.Min(Volume + 5, 100);
- return SetVolume(sendVolume, cancellationToken);
- }
- public Task ToggleMute(CancellationToken cancellationToken)
- {
- if (IsMuted)
- {
- return Unmute(cancellationToken);
- }
- return Mute(cancellationToken);
- }
- public async Task Mute(CancellationToken cancellationToken)
- {
- var success = await SetMute(true, cancellationToken).ConfigureAwait(true);
- if (!success)
- {
- await SetVolume(0, cancellationToken).ConfigureAwait(false);
- }
- }
- public async Task Unmute(CancellationToken cancellationToken)
- {
- var success = await SetMute(false, cancellationToken).ConfigureAwait(true);
- if (!success)
- {
- var sendVolume = _muteVol <= 0 ? 20 : _muteVol;
- await SetVolume(sendVolume, cancellationToken).ConfigureAwait(false);
- }
- }
- private DeviceService GetServiceRenderingControl()
- {
- var services = Properties.Services;
- return services.FirstOrDefault(s => string.Equals(s.ServiceType, "urn:schemas-upnp-org:service:RenderingControl:1", StringComparison.OrdinalIgnoreCase)) ??
- services.FirstOrDefault(s => (s.ServiceType ?? string.Empty).StartsWith("urn:schemas-upnp-org:service:RenderingControl", StringComparison.OrdinalIgnoreCase));
- }
- private DeviceService GetAvTransportService()
- {
- var services = Properties.Services;
- return services.FirstOrDefault(s => string.Equals(s.ServiceType, "urn:schemas-upnp-org:service:AVTransport:1", StringComparison.OrdinalIgnoreCase)) ??
- services.FirstOrDefault(s => (s.ServiceType ?? string.Empty).StartsWith("urn:schemas-upnp-org:service:AVTransport", StringComparison.OrdinalIgnoreCase));
- }
- private async Task<bool> SetMute(bool mute, CancellationToken cancellationToken)
- {
- var rendererCommands = await GetRenderingProtocolAsync(cancellationToken).ConfigureAwait(false);
- var command = rendererCommands?.ServiceActions.FirstOrDefault(c => c.Name == "SetMute");
- if (command is null)
- {
- return false;
- }
- var service = GetServiceRenderingControl();
- if (service is null)
- {
- return false;
- }
- _logger.LogDebug("Setting mute");
- var value = mute ? 1 : 0;
- await new DlnaHttpClient(_logger, _httpClientFactory)
- .SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- rendererCommands.BuildPost(command, service.ServiceType, value),
- cancellationToken: cancellationToken)
- .ConfigureAwait(false);
- IsMuted = mute;
- return true;
- }
- /// <summary>
- /// Sets volume on a scale of 0-100.
- /// </summary>
- /// <param name="value">The volume on a scale of 0-100.</param>
- /// <param name="cancellationToken">The cancellation token to cancel operation.</param>
- /// <returns>A <see cref="Task"/> representing the asynchronous operation.</returns>
- public async Task SetVolume(int value, CancellationToken cancellationToken)
- {
- var rendererCommands = await GetRenderingProtocolAsync(cancellationToken).ConfigureAwait(false);
- var command = rendererCommands?.ServiceActions.FirstOrDefault(c => c.Name == "SetVolume");
- if (command is null)
- {
- return;
- }
- var service = GetServiceRenderingControl();
- if (service is null)
- {
- throw new InvalidOperationException("Unable to find service");
- }
- // Set it early and assume it will succeed
- // Remote control will perform better
- Volume = value;
- await new DlnaHttpClient(_logger, _httpClientFactory)
- .SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- rendererCommands.BuildPost(command, service.ServiceType, value),
- cancellationToken: cancellationToken)
- .ConfigureAwait(false);
- }
- public async Task Seek(TimeSpan value, CancellationToken cancellationToken)
- {
- var avCommands = await GetAVProtocolAsync(cancellationToken).ConfigureAwait(false);
- var command = avCommands?.ServiceActions.FirstOrDefault(c => c.Name == "Seek");
- if (command is null)
- {
- return;
- }
- var service = GetAvTransportService();
- if (service is null)
- {
- throw new InvalidOperationException("Unable to find service");
- }
- await new DlnaHttpClient(_logger, _httpClientFactory)
- .SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- avCommands.BuildPost(command, service.ServiceType, string.Format(CultureInfo.InvariantCulture, "{0:hh}:{0:mm}:{0:ss}", value), "REL_TIME"),
- cancellationToken: cancellationToken)
- .ConfigureAwait(false);
- RestartTimer(true);
- }
- public async Task SetAvTransport(string url, string header, string metaData, CancellationToken cancellationToken)
- {
- var avCommands = await GetAVProtocolAsync(cancellationToken).ConfigureAwait(false);
- url = url.Replace("&", "&", StringComparison.Ordinal);
- _logger.LogDebug("{0} - SetAvTransport Uri: {1} DlnaHeaders: {2}", Properties.Name, url, header);
- var command = avCommands?.ServiceActions.FirstOrDefault(c => c.Name == "SetAVTransportURI");
- if (command is null)
- {
- return;
- }
- var dictionary = new Dictionary<string, string>
- {
- { "CurrentURI", url },
- { "CurrentURIMetaData", CreateDidlMeta(metaData) }
- };
- var service = GetAvTransportService();
- if (service is null)
- {
- throw new InvalidOperationException("Unable to find service");
- }
- var post = avCommands.BuildPost(command, service.ServiceType, url, dictionary);
- await new DlnaHttpClient(_logger, _httpClientFactory)
- .SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- post,
- header: header,
- cancellationToken: cancellationToken)
- .ConfigureAwait(false);
- await Task.Delay(50, cancellationToken).ConfigureAwait(false);
- try
- {
- await SetPlay(avCommands, cancellationToken).ConfigureAwait(false);
- }
- catch
- {
- // Some devices will throw an error if you tell it to play when it's already playing
- // Others won't
- }
- RestartTimer(true);
- }
- /*
- * SetNextAvTransport is used to specify to the DLNA device what is the next track to play.
- * Without that information, the next track command on the device does not work.
- */
- public async Task SetNextAvTransport(string url, string header, string metaData, CancellationToken cancellationToken = default)
- {
- var avCommands = await GetAVProtocolAsync(cancellationToken).ConfigureAwait(false);
- url = url.Replace("&", "&", StringComparison.Ordinal);
- _logger.LogDebug("{PropertyName} - SetNextAvTransport Uri: {Url} DlnaHeaders: {Header}", Properties.Name, url, header);
- var command = avCommands.ServiceActions.FirstOrDefault(c => string.Equals(c.Name, "SetNextAVTransportURI", StringComparison.OrdinalIgnoreCase));
- if (command is null)
- {
- return;
- }
- var dictionary = new Dictionary<string, string>
- {
- { "NextURI", url },
- { "NextURIMetaData", CreateDidlMeta(metaData) }
- };
- var service = GetAvTransportService();
- if (service is null)
- {
- throw new InvalidOperationException("Unable to find service");
- }
- var post = avCommands.BuildPost(command, service.ServiceType, url, dictionary);
- await new DlnaHttpClient(_logger, _httpClientFactory)
- .SendCommandAsync(Properties.BaseUrl, service, command.Name, post, header, cancellationToken)
- .ConfigureAwait(false);
- }
- private static string CreateDidlMeta(string value)
- {
- if (string.IsNullOrEmpty(value))
- {
- return string.Empty;
- }
- return SecurityElement.Escape(value);
- }
- private Task SetPlay(TransportCommands avCommands, CancellationToken cancellationToken)
- {
- var command = avCommands.ServiceActions.FirstOrDefault(c => c.Name == "Play");
- if (command is null)
- {
- return Task.CompletedTask;
- }
- var service = GetAvTransportService();
- if (service is null)
- {
- throw new InvalidOperationException("Unable to find service");
- }
- return new DlnaHttpClient(_logger, _httpClientFactory).SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- avCommands.BuildPost(command, service.ServiceType, 1),
- cancellationToken: cancellationToken);
- }
- public async Task SetPlay(CancellationToken cancellationToken)
- {
- var avCommands = await GetAVProtocolAsync(cancellationToken).ConfigureAwait(false);
- if (avCommands is null)
- {
- return;
- }
- await SetPlay(avCommands, cancellationToken).ConfigureAwait(false);
- RestartTimer(true);
- }
- public async Task SetStop(CancellationToken cancellationToken)
- {
- var avCommands = await GetAVProtocolAsync(cancellationToken).ConfigureAwait(false);
- var command = avCommands?.ServiceActions.FirstOrDefault(c => c.Name == "Stop");
- if (command is null)
- {
- return;
- }
- var service = GetAvTransportService();
- await new DlnaHttpClient(_logger, _httpClientFactory)
- .SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- avCommands.BuildPost(command, service.ServiceType, 1),
- cancellationToken: cancellationToken)
- .ConfigureAwait(false);
- RestartTimer(true);
- }
- public async Task SetPause(CancellationToken cancellationToken)
- {
- var avCommands = await GetAVProtocolAsync(cancellationToken).ConfigureAwait(false);
- var command = avCommands?.ServiceActions.FirstOrDefault(c => c.Name == "Pause");
- if (command is null)
- {
- return;
- }
- var service = GetAvTransportService();
- await new DlnaHttpClient(_logger, _httpClientFactory)
- .SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- avCommands.BuildPost(command, service.ServiceType, 1),
- cancellationToken: cancellationToken)
- .ConfigureAwait(false);
- TransportState = TransportState.PAUSED_PLAYBACK;
- RestartTimer(true);
- }
- private async void TimerCallback(object sender)
- {
- if (_disposed)
- {
- return;
- }
- try
- {
- var cancellationToken = CancellationToken.None;
- var avCommands = await GetAVProtocolAsync(cancellationToken).ConfigureAwait(false);
- if (avCommands is null)
- {
- return;
- }
- var transportState = await GetTransportInfo(avCommands, cancellationToken).ConfigureAwait(false);
- if (_disposed)
- {
- return;
- }
- if (transportState.HasValue)
- {
- // If we're not playing anything no need to get additional data
- if (transportState.Value == TransportState.STOPPED)
- {
- UpdateMediaInfo(null, transportState.Value);
- }
- else
- {
- var tuple = await GetPositionInfo(avCommands, cancellationToken).ConfigureAwait(false);
- var currentObject = tuple.Track;
- if (tuple.Success && currentObject is null)
- {
- currentObject = await GetMediaInfo(avCommands, cancellationToken).ConfigureAwait(false);
- }
- if (currentObject is not null)
- {
- UpdateMediaInfo(currentObject, transportState.Value);
- }
- }
- _connectFailureCount = 0;
- if (_disposed)
- {
- return;
- }
- // If we're not playing anything make sure we don't get data more often than necessary to keep the Session alive
- if (transportState.Value == TransportState.STOPPED)
- {
- RestartTimerInactive();
- }
- else
- {
- RestartTimer();
- }
- }
- else
- {
- RestartTimerInactive();
- }
- }
- catch (Exception ex)
- {
- if (_disposed)
- {
- return;
- }
- _logger.LogError(ex, "Error updating device info for {DeviceName}", Properties.Name);
- _connectFailureCount++;
- if (_connectFailureCount >= 3)
- {
- var action = OnDeviceUnavailable;
- if (action is not null)
- {
- _logger.LogDebug("Disposing device due to loss of connection");
- action();
- return;
- }
- }
- RestartTimerInactive();
- }
- }
- private async Task GetVolume(CancellationToken cancellationToken)
- {
- if (_disposed)
- {
- return;
- }
- var rendererCommands = await GetRenderingProtocolAsync(cancellationToken).ConfigureAwait(false);
- var command = rendererCommands?.ServiceActions.FirstOrDefault(c => c.Name == "GetVolume");
- if (command is null)
- {
- return;
- }
- var service = GetServiceRenderingControl();
- if (service is null)
- {
- return;
- }
- var result = await new DlnaHttpClient(_logger, _httpClientFactory).SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- rendererCommands.BuildPost(command, service.ServiceType),
- cancellationToken: cancellationToken).ConfigureAwait(false);
- if (result is null || result.Document is null)
- {
- return;
- }
- var volume = result.Document.Descendants(UPnpNamespaces.RenderingControl + "GetVolumeResponse").Select(i => i.Element("CurrentVolume")).FirstOrDefault(i => i is not null);
- var volumeValue = volume?.Value;
- if (string.IsNullOrWhiteSpace(volumeValue))
- {
- return;
- }
- Volume = int.Parse(volumeValue, CultureInfo.InvariantCulture);
- if (Volume > 0)
- {
- _muteVol = Volume;
- }
- }
- private async Task GetMute(CancellationToken cancellationToken)
- {
- if (_disposed)
- {
- return;
- }
- var rendererCommands = await GetRenderingProtocolAsync(cancellationToken).ConfigureAwait(false);
- var command = rendererCommands?.ServiceActions.FirstOrDefault(c => c.Name == "GetMute");
- if (command is null)
- {
- return;
- }
- var service = GetServiceRenderingControl();
- if (service is null)
- {
- return;
- }
- var result = await new DlnaHttpClient(_logger, _httpClientFactory).SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- rendererCommands.BuildPost(command, service.ServiceType),
- cancellationToken: cancellationToken).ConfigureAwait(false);
- if (result is null || result.Document is null)
- {
- return;
- }
- var valueNode = result.Document.Descendants(UPnpNamespaces.RenderingControl + "GetMuteResponse")
- .Select(i => i.Element("CurrentMute"))
- .FirstOrDefault(i => i is not null);
- IsMuted = string.Equals(valueNode?.Value, "1", StringComparison.OrdinalIgnoreCase);
- }
- private async Task<TransportState?> GetTransportInfo(TransportCommands avCommands, CancellationToken cancellationToken)
- {
- var command = avCommands.ServiceActions.FirstOrDefault(c => c.Name == "GetTransportInfo");
- if (command is null)
- {
- return null;
- }
- var service = GetAvTransportService();
- if (service is null)
- {
- return null;
- }
- var result = await new DlnaHttpClient(_logger, _httpClientFactory).SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- avCommands.BuildPost(command, service.ServiceType),
- cancellationToken: cancellationToken).ConfigureAwait(false);
- if (result is null || result.Document is null)
- {
- return null;
- }
- var transportState =
- result.Document.Descendants(UPnpNamespaces.AvTransport + "GetTransportInfoResponse").Select(i => i.Element("CurrentTransportState")).FirstOrDefault(i => i is not null);
- var transportStateValue = transportState?.Value;
- if (transportStateValue is not null
- && Enum.TryParse(transportStateValue, true, out TransportState state))
- {
- return state;
- }
- return null;
- }
- private async Task<UBaseObject> GetMediaInfo(TransportCommands avCommands, CancellationToken cancellationToken)
- {
- var command = avCommands.ServiceActions.FirstOrDefault(c => c.Name == "GetMediaInfo");
- if (command is null)
- {
- return null;
- }
- var service = GetAvTransportService();
- if (service is null)
- {
- throw new InvalidOperationException("Unable to find service");
- }
- var rendererCommands = await GetRenderingProtocolAsync(cancellationToken).ConfigureAwait(false);
- if (rendererCommands is null)
- {
- return null;
- }
- var result = await new DlnaHttpClient(_logger, _httpClientFactory).SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- rendererCommands.BuildPost(command, service.ServiceType),
- cancellationToken: cancellationToken).ConfigureAwait(false);
- if (result is null || result.Document is null)
- {
- return null;
- }
- var track = result.Document.Descendants("CurrentURIMetaData").FirstOrDefault();
- if (track is null)
- {
- return null;
- }
- var e = track.Element(UPnpNamespaces.Items) ?? track;
- var elementString = (string)e;
- if (!string.IsNullOrWhiteSpace(elementString))
- {
- return UpnpContainer.Create(e);
- }
- track = result.Document.Descendants("CurrentURI").FirstOrDefault();
- if (track is null)
- {
- return null;
- }
- e = track.Element(UPnpNamespaces.Items) ?? track;
- elementString = (string)e;
- if (!string.IsNullOrWhiteSpace(elementString))
- {
- return new UBaseObject
- {
- Url = elementString
- };
- }
- return null;
- }
- private async Task<(bool Success, UBaseObject Track)> GetPositionInfo(TransportCommands avCommands, CancellationToken cancellationToken)
- {
- var command = avCommands.ServiceActions.FirstOrDefault(c => c.Name == "GetPositionInfo");
- if (command is null)
- {
- return (false, null);
- }
- var service = GetAvTransportService();
- if (service is null)
- {
- throw new InvalidOperationException("Unable to find service");
- }
- var rendererCommands = await GetRenderingProtocolAsync(cancellationToken).ConfigureAwait(false);
- if (rendererCommands is null)
- {
- return (false, null);
- }
- var result = await new DlnaHttpClient(_logger, _httpClientFactory).SendCommandAsync(
- Properties.BaseUrl,
- service,
- command.Name,
- rendererCommands.BuildPost(command, service.ServiceType),
- cancellationToken: cancellationToken).ConfigureAwait(false);
- if (result is null || result.Document is null)
- {
- return (false, null);
- }
- var trackUriElem = result.Document.Descendants(UPnpNamespaces.AvTransport + "GetPositionInfoResponse").Select(i => i.Element("TrackURI")).FirstOrDefault(i => i is not null);
- var trackUri = trackUriElem?.Value;
- var durationElem = result.Document.Descendants(UPnpNamespaces.AvTransport + "GetPositionInfoResponse").Select(i => i.Element("TrackDuration")).FirstOrDefault(i => i is not null);
- var duration = durationElem?.Value;
- if (!string.IsNullOrWhiteSpace(duration)
- && !string.Equals(duration, "NOT_IMPLEMENTED", StringComparison.OrdinalIgnoreCase))
- {
- Duration = TimeSpan.Parse(duration, CultureInfo.InvariantCulture);
- }
- else
- {
- Duration = null;
- }
- var positionElem = result.Document.Descendants(UPnpNamespaces.AvTransport + "GetPositionInfoResponse").Select(i => i.Element("RelTime")).FirstOrDefault(i => i is not null);
- var position = positionElem?.Value;
- if (!string.IsNullOrWhiteSpace(position) && !string.Equals(position, "NOT_IMPLEMENTED", StringComparison.OrdinalIgnoreCase))
- {
- Position = TimeSpan.Parse(position, CultureInfo.InvariantCulture);
- }
- var track = result.Document.Descendants("TrackMetaData").FirstOrDefault();
- if (track is null)
- {
- // If track is null, some vendors do this, use GetMediaInfo instead.
- return (true, null);
- }
- var trackString = (string)track;
- if (string.IsNullOrWhiteSpace(trackString) || string.Equals(trackString, "NOT_IMPLEMENTED", StringComparison.OrdinalIgnoreCase))
- {
- return (true, null);
- }
- XElement uPnpResponse = null;
- try
- {
- uPnpResponse = ParseResponse(trackString);
- }
- catch (Exception ex)
- {
- _logger.LogError(ex, "Uncaught exception while parsing xml");
- }
- if (uPnpResponse is null)
- {
- _logger.LogError("Failed to parse xml: \n {Xml}", trackString);
- return (true, null);
- }
- var e = uPnpResponse.Element(UPnpNamespaces.Items);
- var uTrack = CreateUBaseObject(e, trackUri);
- return (true, uTrack);
- }
- private XElement ParseResponse(string xml)
- {
- // Handle different variations sent back by devices.
- try
- {
- return XElement.Parse(xml);
- }
- catch (XmlException)
- {
- }
- // first try to add a root node with a dlna namespace.
- try
- {
- return XElement.Parse("<data xmlns:dlna=\"urn:schemas-dlna-org:device-1-0\">" + xml + "</data>")
- .Descendants()
- .First();
- }
- catch (XmlException)
- {
- }
- // some devices send back invalid xml
- try
- {
- return XElement.Parse(xml.Replace("&", "&", StringComparison.Ordinal));
- }
- catch (XmlException)
- {
- }
- return null;
- }
- private static UBaseObject CreateUBaseObject(XElement container, string trackUri)
- {
- ArgumentNullException.ThrowIfNull(container);
- var url = container.GetValue(UPnpNamespaces.Res);
- if (string.IsNullOrWhiteSpace(url))
- {
- url = trackUri;
- }
- return new UBaseObject
- {
- Id = container.GetAttributeValue(UPnpNamespaces.Id),
- ParentId = container.GetAttributeValue(UPnpNamespaces.ParentId),
- Title = container.GetValue(UPnpNamespaces.Title),
- IconUrl = container.GetValue(UPnpNamespaces.Artwork),
- SecondText = string.Empty,
- Url = url,
- ProtocolInfo = GetProtocolInfo(container),
- MetaData = container.ToString()
- };
- }
- private static string[] GetProtocolInfo(XElement container)
- {
- ArgumentNullException.ThrowIfNull(container);
- var resElement = container.Element(UPnpNamespaces.Res);
- if (resElement is not null)
- {
- var info = resElement.Attribute(UPnpNamespaces.ProtocolInfo);
- if (info is not null && !string.IsNullOrWhiteSpace(info.Value))
- {
- return info.Value.Split(':');
- }
- }
- return new string[4];
- }
- private async Task<TransportCommands> GetAVProtocolAsync(CancellationToken cancellationToken)
- {
- if (AvCommands is not null)
- {
- return AvCommands;
- }
- if (_disposed)
- {
- throw new ObjectDisposedException(GetType().Name);
- }
- var avService = GetAvTransportService();
- if (avService is null)
- {
- return null;
- }
- string url = NormalizeUrl(Properties.BaseUrl, avService.ScpdUrl);
- var httpClient = new DlnaHttpClient(_logger, _httpClientFactory);
- var document = await httpClient.GetDataAsync(url, cancellationToken).ConfigureAwait(false);
- if (document is null)
- {
- return null;
- }
- AvCommands = TransportCommands.Create(document);
- return AvCommands;
- }
- private async Task<TransportCommands> GetRenderingProtocolAsync(CancellationToken cancellationToken)
- {
- if (RendererCommands is not null)
- {
- return RendererCommands;
- }
- if (_disposed)
- {
- throw new ObjectDisposedException(GetType().Name);
- }
- var avService = GetServiceRenderingControl();
- if (avService is null)
- {
- throw new ArgumentException("Device AvService is null");
- }
- string url = NormalizeUrl(Properties.BaseUrl, avService.ScpdUrl);
- var httpClient = new DlnaHttpClient(_logger, _httpClientFactory);
- _logger.LogDebug("Dlna Device.GetRenderingProtocolAsync");
- var document = await httpClient.GetDataAsync(url, cancellationToken).ConfigureAwait(false);
- if (document is null)
- {
- return null;
- }
- RendererCommands = TransportCommands.Create(document);
- return RendererCommands;
- }
- private string NormalizeUrl(string baseUrl, string url)
- {
- // If it's already a complete url, don't stick anything onto the front of it
- if (url.StartsWith("http", StringComparison.OrdinalIgnoreCase))
- {
- return url;
- }
- if (!url.Contains('/', StringComparison.Ordinal))
- {
- url = "/dmr/" + url;
- }
- if (!url.StartsWith('/'))
- {
- url = "/" + url;
- }
- return baseUrl + url;
- }
- public static async Task<Device> CreateuPnpDeviceAsync(Uri url, IHttpClientFactory httpClientFactory, ILogger logger, CancellationToken cancellationToken)
- {
- var ssdpHttpClient = new DlnaHttpClient(logger, httpClientFactory);
- var document = await ssdpHttpClient.GetDataAsync(url.ToString(), cancellationToken).ConfigureAwait(false);
- if (document is null)
- {
- return null;
- }
- var friendlyNames = new List<string>();
- var name = document.Descendants(UPnpNamespaces.Ud.GetName("friendlyName")).FirstOrDefault();
- if (name is not null && !string.IsNullOrWhiteSpace(name.Value))
- {
- friendlyNames.Add(name.Value);
- }
- var room = document.Descendants(UPnpNamespaces.Ud.GetName("roomName")).FirstOrDefault();
- if (room is not null && !string.IsNullOrWhiteSpace(room.Value))
- {
- friendlyNames.Add(room.Value);
- }
- var deviceProperties = new DeviceInfo()
- {
- Name = string.Join(' ', friendlyNames),
- BaseUrl = string.Format(CultureInfo.InvariantCulture, "http://{0}:{1}", url.Host, url.Port)
- };
- var model = document.Descendants(UPnpNamespaces.Ud.GetName("modelName")).FirstOrDefault();
- if (model is not null)
- {
- deviceProperties.ModelName = model.Value;
- }
- var modelNumber = document.Descendants(UPnpNamespaces.Ud.GetName("modelNumber")).FirstOrDefault();
- if (modelNumber is not null)
- {
- deviceProperties.ModelNumber = modelNumber.Value;
- }
- var uuid = document.Descendants(UPnpNamespaces.Ud.GetName("UDN")).FirstOrDefault();
- if (uuid is not null)
- {
- deviceProperties.UUID = uuid.Value;
- }
- var manufacturer = document.Descendants(UPnpNamespaces.Ud.GetName("manufacturer")).FirstOrDefault();
- if (manufacturer is not null)
- {
- deviceProperties.Manufacturer = manufacturer.Value;
- }
- var manufacturerUrl = document.Descendants(UPnpNamespaces.Ud.GetName("manufacturerURL")).FirstOrDefault();
- if (manufacturerUrl is not null)
- {
- deviceProperties.ManufacturerUrl = manufacturerUrl.Value;
- }
- var presentationUrl = document.Descendants(UPnpNamespaces.Ud.GetName("presentationURL")).FirstOrDefault();
- if (presentationUrl is not null)
- {
- deviceProperties.PresentationUrl = presentationUrl.Value;
- }
- var modelUrl = document.Descendants(UPnpNamespaces.Ud.GetName("modelURL")).FirstOrDefault();
- if (modelUrl is not null)
- {
- deviceProperties.ModelUrl = modelUrl.Value;
- }
- var serialNumber = document.Descendants(UPnpNamespaces.Ud.GetName("serialNumber")).FirstOrDefault();
- if (serialNumber is not null)
- {
- deviceProperties.SerialNumber = serialNumber.Value;
- }
- var modelDescription = document.Descendants(UPnpNamespaces.Ud.GetName("modelDescription")).FirstOrDefault();
- if (modelDescription is not null)
- {
- deviceProperties.ModelDescription = modelDescription.Value;
- }
- var icon = document.Descendants(UPnpNamespaces.Ud.GetName("icon")).FirstOrDefault();
- if (icon is not null)
- {
- deviceProperties.Icon = CreateIcon(icon);
- }
- foreach (var services in document.Descendants(UPnpNamespaces.Ud.GetName("serviceList")))
- {
- if (services is null)
- {
- continue;
- }
- var servicesList = services.Descendants(UPnpNamespaces.Ud.GetName("service"));
- if (servicesList is null)
- {
- continue;
- }
- foreach (var element in servicesList)
- {
- var service = Create(element);
- if (service is not null)
- {
- deviceProperties.Services.Add(service);
- }
- }
- }
- return new Device(deviceProperties, httpClientFactory, logger);
- }
- #nullable enable
- private static DeviceIcon CreateIcon(XElement element)
- {
- ArgumentNullException.ThrowIfNull(element);
- var width = element.GetDescendantValue(UPnpNamespaces.Ud.GetName("width"));
- var height = element.GetDescendantValue(UPnpNamespaces.Ud.GetName("height"));
- _ = int.TryParse(width, NumberStyles.Integer, CultureInfo.InvariantCulture, out var widthValue);
- _ = int.TryParse(height, NumberStyles.Integer, CultureInfo.InvariantCulture, out var heightValue);
- return new DeviceIcon
- {
- Depth = element.GetDescendantValue(UPnpNamespaces.Ud.GetName("depth")) ?? string.Empty,
- Height = heightValue,
- MimeType = element.GetDescendantValue(UPnpNamespaces.Ud.GetName("mimetype")) ?? string.Empty,
- Url = element.GetDescendantValue(UPnpNamespaces.Ud.GetName("url")) ?? string.Empty,
- Width = widthValue
- };
- }
- private static DeviceService Create(XElement element)
- => new DeviceService()
- {
- ControlUrl = element.GetDescendantValue(UPnpNamespaces.Ud.GetName("controlURL")) ?? string.Empty,
- EventSubUrl = element.GetDescendantValue(UPnpNamespaces.Ud.GetName("eventSubURL")) ?? string.Empty,
- ScpdUrl = element.GetDescendantValue(UPnpNamespaces.Ud.GetName("SCPDURL")) ?? string.Empty,
- ServiceId = element.GetDescendantValue(UPnpNamespaces.Ud.GetName("serviceId")) ?? string.Empty,
- ServiceType = element.GetDescendantValue(UPnpNamespaces.Ud.GetName("serviceType")) ?? string.Empty
- };
- private void UpdateMediaInfo(UBaseObject? mediaInfo, TransportState state)
- {
- TransportState = state;
- var previousMediaInfo = CurrentMediaInfo;
- CurrentMediaInfo = mediaInfo;
- if (mediaInfo is null)
- {
- if (previousMediaInfo is not null)
- {
- OnPlaybackStop(previousMediaInfo);
- }
- }
- else if (previousMediaInfo is null)
- {
- if (state != TransportState.STOPPED)
- {
- OnPlaybackStart(mediaInfo);
- }
- }
- else if (mediaInfo.Equals(previousMediaInfo))
- {
- OnPlaybackProgress(mediaInfo);
- }
- else
- {
- OnMediaChanged(previousMediaInfo, mediaInfo);
- }
- }
- private void OnPlaybackStart(UBaseObject mediaInfo)
- {
- if (string.IsNullOrWhiteSpace(mediaInfo.Url))
- {
- return;
- }
- PlaybackStart?.Invoke(this, new PlaybackStartEventArgs(mediaInfo));
- }
- private void OnPlaybackProgress(UBaseObject mediaInfo)
- {
- if (string.IsNullOrWhiteSpace(mediaInfo.Url))
- {
- return;
- }
- PlaybackProgress?.Invoke(this, new PlaybackProgressEventArgs(mediaInfo));
- }
- private void OnPlaybackStop(UBaseObject mediaInfo)
- {
- PlaybackStopped?.Invoke(this, new PlaybackStoppedEventArgs(mediaInfo));
- }
- private void OnMediaChanged(UBaseObject old, UBaseObject newMedia)
- {
- MediaChanged?.Invoke(this, new MediaChangedEventArgs(old, newMedia));
- }
- /// <inheritdoc />
- public void Dispose()
- {
- Dispose(true);
- GC.SuppressFinalize(this);
- }
- /// <summary>
- /// Releases unmanaged and optionally managed resources.
- /// </summary>
- /// <param name="disposing"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
- protected virtual void Dispose(bool disposing)
- {
- if (_disposed)
- {
- return;
- }
- if (disposing)
- {
- _timer?.Dispose();
- }
- _timer = null;
- Properties = null;
- _disposed = true;
- }
- /// <inheritdoc />
- public override string ToString()
- {
- return string.Format(CultureInfo.InvariantCulture, "{0} - {1}", Properties.Name, Properties.BaseUrl);
- }
- }
- }
|