|
@@ -1,10 +1,13 @@
|
|
using System;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Collections.Generic;
|
|
|
|
+using System.IO;
|
|
using System.Linq;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Text;
|
|
using System.Threading;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using System.Threading.Tasks;
|
|
|
|
+using System.Xml;
|
|
using MediaBrowser.Common.Configuration;
|
|
using MediaBrowser.Common.Configuration;
|
|
|
|
+using MediaBrowser.Common.Extensions;
|
|
using MediaBrowser.Common.Net;
|
|
using MediaBrowser.Common.Net;
|
|
using MediaBrowser.Controller.Configuration;
|
|
using MediaBrowser.Controller.Configuration;
|
|
using MediaBrowser.Controller.Dlna;
|
|
using MediaBrowser.Controller.Dlna;
|
|
@@ -13,6 +16,7 @@ using MediaBrowser.Controller.Plugins;
|
|
using MediaBrowser.Model.Extensions;
|
|
using MediaBrowser.Model.Extensions;
|
|
using MediaBrowser.Model.LiveTv;
|
|
using MediaBrowser.Model.LiveTv;
|
|
using MediaBrowser.Model.Logging;
|
|
using MediaBrowser.Model.Logging;
|
|
|
|
+using MediaBrowser.Model.Serialization;
|
|
|
|
|
|
namespace MediaBrowser.Server.Implementations.LiveTv.TunerHosts.SatIp
|
|
namespace MediaBrowser.Server.Implementations.LiveTv.TunerHosts.SatIp
|
|
{
|
|
{
|
|
@@ -24,14 +28,26 @@ namespace MediaBrowser.Server.Implementations.LiveTv.TunerHosts.SatIp
|
|
private readonly ILiveTvManager _liveTvManager;
|
|
private readonly ILiveTvManager _liveTvManager;
|
|
private readonly SemaphoreSlim _semaphore = new SemaphoreSlim(1, 1);
|
|
private readonly SemaphoreSlim _semaphore = new SemaphoreSlim(1, 1);
|
|
private readonly IHttpClient _httpClient;
|
|
private readonly IHttpClient _httpClient;
|
|
|
|
+ private readonly IJsonSerializer _json;
|
|
|
|
|
|
- public SatIpDiscovery(IDeviceDiscovery deviceDiscovery, IServerConfigurationManager config, ILogger logger, ILiveTvManager liveTvManager, IHttpClient httpClient)
|
|
|
|
|
|
+ public static SatIpDiscovery Current;
|
|
|
|
+
|
|
|
|
+ private readonly List<TunerHostInfo> _discoveredHosts = new List<TunerHostInfo>();
|
|
|
|
+
|
|
|
|
+ public List<TunerHostInfo> DiscoveredHosts
|
|
|
|
+ {
|
|
|
|
+ get { return _discoveredHosts.ToList(); }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ public SatIpDiscovery(IDeviceDiscovery deviceDiscovery, IServerConfigurationManager config, ILogger logger, ILiveTvManager liveTvManager, IHttpClient httpClient, IJsonSerializer json)
|
|
{
|
|
{
|
|
_deviceDiscovery = deviceDiscovery;
|
|
_deviceDiscovery = deviceDiscovery;
|
|
_config = config;
|
|
_config = config;
|
|
_logger = logger;
|
|
_logger = logger;
|
|
_liveTvManager = liveTvManager;
|
|
_liveTvManager = liveTvManager;
|
|
_httpClient = httpClient;
|
|
_httpClient = httpClient;
|
|
|
|
+ _json = json;
|
|
|
|
+ Current = this;
|
|
}
|
|
}
|
|
|
|
|
|
public void Run()
|
|
public void Run()
|
|
@@ -66,26 +82,23 @@ namespace MediaBrowser.Server.Implementations.LiveTv.TunerHosts.SatIp
|
|
|
|
|
|
try
|
|
try
|
|
{
|
|
{
|
|
- var options = GetConfiguration();
|
|
|
|
-
|
|
|
|
- //if (options.TunerHosts.Any(i =>
|
|
|
|
- // string.Equals(i.Type, SatIpHost.DeviceType, StringComparison.OrdinalIgnoreCase) &&
|
|
|
|
- // UriEquals(i.Url, url)))
|
|
|
|
- //{
|
|
|
|
- // return;
|
|
|
|
- //}
|
|
|
|
-
|
|
|
|
- //// Strip off the port
|
|
|
|
- //url = new Uri(url).GetComponents(UriComponents.AbsoluteUri & ~UriComponents.Port, UriFormat.UriEscaped).TrimEnd('/');
|
|
|
|
|
|
+ if (_discoveredHosts.Any(i => string.Equals(i.Type, SatIpHost.DeviceType, StringComparison.OrdinalIgnoreCase) && string.Equals(location, i.Url, StringComparison.OrdinalIgnoreCase)))
|
|
|
|
+ {
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
|
|
- //await TestUrl(url).ConfigureAwait(false);
|
|
|
|
|
|
+ _logger.Debug("Will attempt to add SAT device {0}", location);
|
|
|
|
+ var info = await GetInfo(location, CancellationToken.None).ConfigureAwait(false);
|
|
|
|
|
|
- //await _liveTvManager.SaveTunerHost(new TunerHostInfo
|
|
|
|
- //{
|
|
|
|
- // Type = SatIpHost.DeviceType,
|
|
|
|
- // Url = url
|
|
|
|
|
|
+ _discoveredHosts.Add(info);
|
|
|
|
+ }
|
|
|
|
+ catch (OperationCanceledException)
|
|
|
|
+ {
|
|
|
|
|
|
- //}).ConfigureAwait(false);
|
|
|
|
|
|
+ }
|
|
|
|
+ catch (NotImplementedException)
|
|
|
|
+ {
|
|
|
|
+
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
catch (Exception ex)
|
|
{
|
|
{
|
|
@@ -97,43 +110,116 @@ namespace MediaBrowser.Server.Implementations.LiveTv.TunerHosts.SatIp
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- private async Task TestUrl(string url)
|
|
|
|
|
|
+ public void Dispose()
|
|
{
|
|
{
|
|
- // Test it by pulling down the lineup
|
|
|
|
- using (await _httpClient.Get(new HttpRequestOptions
|
|
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ public async Task<SatIpTunerHostInfo> GetInfo(string url, CancellationToken cancellationToken)
|
|
|
|
+ {
|
|
|
|
+ var result = new SatIpTunerHostInfo
|
|
{
|
|
{
|
|
- Url = string.Format("{0}/lineup.json", url),
|
|
|
|
- CancellationToken = CancellationToken.None
|
|
|
|
- }))
|
|
|
|
|
|
+ Url = url,
|
|
|
|
+ IsEnabled = true,
|
|
|
|
+ Type = SatIpHost.DeviceType,
|
|
|
|
+ Tuners = 1,
|
|
|
|
+ TunersAvailable = 1
|
|
|
|
+ };
|
|
|
|
+
|
|
|
|
+ using (var stream = await _httpClient.Get(url, cancellationToken).ConfigureAwait(false))
|
|
|
|
+ {
|
|
|
|
+ using (var streamReader = new StreamReader(stream))
|
|
|
|
+ {
|
|
|
|
+ // Use XmlReader for best performance
|
|
|
|
+ using (var reader = XmlReader.Create(streamReader))
|
|
|
|
+ {
|
|
|
|
+ reader.MoveToContent();
|
|
|
|
+
|
|
|
|
+ // Loop through each element
|
|
|
|
+ while (reader.Read())
|
|
|
|
+ {
|
|
|
|
+ if (reader.NodeType == XmlNodeType.Element)
|
|
|
|
+ {
|
|
|
|
+ switch (reader.Name)
|
|
|
|
+ {
|
|
|
|
+ case "device":
|
|
|
|
+ using (var subtree = reader.ReadSubtree())
|
|
|
|
+ {
|
|
|
|
+ FillFromDeviceNode(result, subtree);
|
|
|
|
+ }
|
|
|
|
+ break;
|
|
|
|
+ default:
|
|
|
|
+ reader.Skip();
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (string.IsNullOrWhiteSpace(result.Id))
|
|
{
|
|
{
|
|
|
|
+ throw new NotImplementedException();
|
|
}
|
|
}
|
|
- }
|
|
|
|
|
|
|
|
- private bool UriEquals(string savedUri, string location)
|
|
|
|
- {
|
|
|
|
- return string.Equals(NormalizeUrl(location), NormalizeUrl(savedUri), StringComparison.OrdinalIgnoreCase);
|
|
|
|
- }
|
|
|
|
|
|
+ if (string.IsNullOrWhiteSpace(result.M3UUrl))
|
|
|
|
+ {
|
|
|
|
+ throw new NotImplementedException();
|
|
|
|
+ }
|
|
|
|
|
|
- private string NormalizeUrl(string url)
|
|
|
|
- {
|
|
|
|
- if (!url.StartsWith("http", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
|
+ if (!result.M3UUrl.StartsWith("http", StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
{
|
|
- url = "http://" + url;
|
|
|
|
|
|
+ var fullM3uUrl = url.Substring(0, url.LastIndexOf('/'));
|
|
|
|
+ result.M3UUrl = fullM3uUrl + "/" + result.M3UUrl.TrimStart('/');
|
|
}
|
|
}
|
|
|
|
|
|
- url = url.TrimEnd('/');
|
|
|
|
|
|
+ _logger.Debug("SAT device result: {0}", _json.SerializeToString(result));
|
|
|
|
|
|
- // Strip off the port
|
|
|
|
- return new Uri(url).GetComponents(UriComponents.AbsoluteUri & ~UriComponents.Port, UriFormat.UriEscaped);
|
|
|
|
|
|
+ return result;
|
|
}
|
|
}
|
|
|
|
|
|
- private LiveTvOptions GetConfiguration()
|
|
|
|
|
|
+ private void FillFromDeviceNode(SatIpTunerHostInfo info, XmlReader reader)
|
|
{
|
|
{
|
|
- return _config.GetConfiguration<LiveTvOptions>("livetv");
|
|
|
|
- }
|
|
|
|
|
|
+ reader.MoveToContent();
|
|
|
|
|
|
- public void Dispose()
|
|
|
|
- {
|
|
|
|
|
|
+ while (reader.Read())
|
|
|
|
+ {
|
|
|
|
+ if (reader.NodeType == XmlNodeType.Element)
|
|
|
|
+ {
|
|
|
|
+ switch (reader.Name)
|
|
|
|
+ {
|
|
|
|
+ case "UDN":
|
|
|
|
+ {
|
|
|
|
+ info.Id = reader.ReadElementContentAsString();
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ case "X_SATIPCAP":
|
|
|
|
+ {
|
|
|
|
+ var value = reader.ReadElementContentAsString();
|
|
|
|
+ // TODO
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ case "X_SATIPM3U":
|
|
|
|
+ {
|
|
|
|
+ info.M3UUrl = reader.ReadElementContentAsString();
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ default:
|
|
|
|
+ reader.Skip();
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+ public class SatIpTunerHostInfo : TunerHostInfo
|
|
|
|
+ {
|
|
|
|
+ public int Tuners { get; set; }
|
|
|
|
+ public int TunersAvailable { get; set; }
|
|
|
|
+ public string M3UUrl { get; set; }
|
|
|
|
+ }
|
|
}
|
|
}
|