jellyfin-server/Emby.Server.Implementations/LiveTv/TunerHosts/HdHomerun/HdHomerunHost.cs

661 lines
25 KiB
C#
Raw Normal View History

2015-07-20 18:32:55 +00:00
using MediaBrowser.Common.Configuration;
using MediaBrowser.Common.Net;
using MediaBrowser.Controller.LiveTv;
using MediaBrowser.Model.Dto;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.LiveTv;
using MediaBrowser.Model.Logging;
using MediaBrowser.Model.MediaInfo;
using MediaBrowser.Model.Serialization;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
2016-10-25 19:02:04 +00:00
using MediaBrowser.Model.IO;
2016-02-24 18:45:11 +00:00
using MediaBrowser.Common.Extensions;
2016-09-25 18:39:13 +00:00
using MediaBrowser.Controller;
using MediaBrowser.Controller.Configuration;
2015-10-14 02:41:46 +00:00
using MediaBrowser.Controller.MediaEncoding;
2015-10-30 16:40:12 +00:00
using MediaBrowser.Model.Configuration;
2016-07-08 03:22:16 +00:00
using MediaBrowser.Model.Net;
2017-03-27 00:27:29 +00:00
using MediaBrowser.Model.System;
2015-07-20 18:32:55 +00:00
2016-11-03 23:35:19 +00:00
namespace Emby.Server.Implementations.LiveTv.TunerHosts.HdHomerun
2015-07-20 18:32:55 +00:00
{
2016-02-19 06:20:18 +00:00
public class HdHomerunHost : BaseTunerHost, ITunerHost, IConfigurableTunerHost
2015-07-20 18:32:55 +00:00
{
private readonly IHttpClient _httpClient;
2016-09-25 18:39:13 +00:00
private readonly IFileSystem _fileSystem;
private readonly IServerApplicationHost _appHost;
2017-03-02 20:50:09 +00:00
private readonly ISocketFactory _socketFactory;
2017-03-03 05:53:21 +00:00
private readonly INetworkManager _networkManager;
2017-03-27 00:27:29 +00:00
private readonly IEnvironmentInfo _environment;
2015-07-20 18:32:55 +00:00
2017-03-27 00:27:29 +00:00
public HdHomerunHost(IServerConfigurationManager config, ILogger logger, IJsonSerializer jsonSerializer, IMediaEncoder mediaEncoder, IHttpClient httpClient, IFileSystem fileSystem, IServerApplicationHost appHost, ISocketFactory socketFactory, INetworkManager networkManager, IEnvironmentInfo environment)
2015-10-30 16:40:12 +00:00
: base(config, logger, jsonSerializer, mediaEncoder)
2015-07-20 18:32:55 +00:00
{
_httpClient = httpClient;
2016-09-25 18:39:13 +00:00
_fileSystem = fileSystem;
_appHost = appHost;
2017-03-02 20:50:09 +00:00
_socketFactory = socketFactory;
2017-03-03 05:53:21 +00:00
_networkManager = networkManager;
2017-03-27 00:27:29 +00:00
_environment = environment;
2015-07-20 18:32:55 +00:00
}
public string Name
{
get { return "HD Homerun"; }
}
2015-08-19 17:58:41 +00:00
public override string Type
2015-07-23 16:32:34 +00:00
{
get { return DeviceType; }
}
public static string DeviceType
2015-07-20 18:32:55 +00:00
{
get { return "hdhomerun"; }
}
2017-03-14 19:44:54 +00:00
protected override string ChannelIdPrefix
{
get
{
return "hdhr_";
}
}
2015-08-16 18:37:53 +00:00
2016-02-25 20:29:38 +00:00
private string GetChannelId(TunerHostInfo info, Channels i)
{
2016-11-03 23:35:19 +00:00
var id = ChannelIdPrefix + i.GuideNumber;
2016-02-25 20:29:38 +00:00
2017-01-23 21:51:23 +00:00
id += '_' + (i.GuideName ?? string.Empty).GetMD5().ToString("N");
2016-02-25 20:29:38 +00:00
return id;
}
private async Task<List<Channels>> GetLineup(TunerHostInfo info, CancellationToken cancellationToken)
2015-07-20 18:32:55 +00:00
{
2017-03-06 02:32:56 +00:00
var model = await GetModelInfo(info, false, cancellationToken).ConfigureAwait(false);
2015-07-20 18:32:55 +00:00
var options = new HttpRequestOptions
{
2017-03-06 02:32:56 +00:00
Url = model.LineupURL,
2016-10-06 18:55:01 +00:00
CancellationToken = cancellationToken,
BufferContent = false
2015-07-20 18:32:55 +00:00
};
using (var stream = await _httpClient.Get(options).ConfigureAwait(false))
2015-07-20 18:32:55 +00:00
{
2016-04-03 23:20:43 +00:00
var lineup = JsonSerializer.DeserializeFromStream<List<Channels>>(stream) ?? new List<Channels>();
2015-07-20 18:32:55 +00:00
2016-04-03 23:20:43 +00:00
if (info.ImportFavoritesOnly)
2015-07-20 18:32:55 +00:00
{
2016-04-03 23:20:43 +00:00
lineup = lineup.Where(i => i.Favorite).ToList();
}
2015-07-20 18:32:55 +00:00
2016-04-06 02:18:56 +00:00
return lineup.Where(i => !i.DRM).ToList();
2016-04-03 23:20:43 +00:00
}
}
2015-07-25 18:11:46 +00:00
2017-03-02 21:24:46 +00:00
private class HdHomerunChannelInfo : ChannelInfo
{
2017-03-02 21:27:46 +00:00
public bool IsLegacyTuner { get; set; }
2017-03-03 05:53:21 +00:00
public string Url { get; set; }
2017-03-02 21:24:46 +00:00
}
2017-02-23 19:13:26 +00:00
protected override async Task<List<ChannelInfo>> GetChannelsInternal(TunerHostInfo info, CancellationToken cancellationToken)
2016-04-03 23:20:43 +00:00
{
var lineup = await GetLineup(info, cancellationToken).ConfigureAwait(false);
2015-07-25 18:11:46 +00:00
2017-03-02 21:24:46 +00:00
return lineup.Select(i => new HdHomerunChannelInfo
2016-04-03 23:20:43 +00:00
{
Name = i.GuideName,
2016-11-03 23:35:19 +00:00
Number = i.GuideNumber,
2016-04-03 23:20:43 +00:00
Id = GetChannelId(info, i),
IsFavorite = i.Favorite,
2016-04-04 00:01:03 +00:00
TunerHostId = info.Id,
IsHD = i.HD == 1,
AudioCodec = i.AudioCodec,
2017-01-01 20:47:54 +00:00
VideoCodec = i.VideoCodec,
2017-03-02 21:24:46 +00:00
ChannelType = ChannelType.TV,
2017-03-03 05:53:21 +00:00
IsLegacyTuner = (i.URL ?? string.Empty).StartsWith("hdhomerun", StringComparison.OrdinalIgnoreCase),
Url = i.URL
2017-02-23 19:13:26 +00:00
2017-03-02 21:24:46 +00:00
}).Cast<ChannelInfo>().ToList();
2015-07-20 18:32:55 +00:00
}
2016-10-07 15:08:13 +00:00
private readonly Dictionary<string, DiscoverResponse> _modelCache = new Dictionary<string, DiscoverResponse>();
2017-03-02 20:50:09 +00:00
private async Task<DiscoverResponse> GetModelInfo(TunerHostInfo info, bool throwAllExceptions, CancellationToken cancellationToken)
2015-07-20 18:32:55 +00:00
{
2016-09-30 06:50:06 +00:00
lock (_modelCache)
{
DiscoverResponse response;
if (_modelCache.TryGetValue(info.Url, out response))
{
2017-03-16 17:21:24 +00:00
if ((DateTime.UtcNow - response.DateQueried).TotalHours <= 12)
{
return response;
}
2016-09-30 06:50:06 +00:00
}
}
2016-07-08 03:22:16 +00:00
try
2015-07-21 04:22:46 +00:00
{
2016-07-08 03:22:16 +00:00
using (var stream = await _httpClient.Get(new HttpRequestOptions()
{
Url = string.Format("{0}/discover.json", GetApiUrl(info, false)),
CancellationToken = cancellationToken,
2016-10-06 18:55:01 +00:00
TimeoutMs = Convert.ToInt32(TimeSpan.FromSeconds(5).TotalMilliseconds),
BufferContent = false
}).ConfigureAwait(false))
2016-07-08 03:22:16 +00:00
{
var response = JsonSerializer.DeserializeFromStream<DiscoverResponse>(stream);
2017-03-03 06:38:05 +00:00
if (!string.IsNullOrWhiteSpace(info.Id))
2016-09-30 06:50:06 +00:00
{
2017-03-03 06:38:05 +00:00
lock (_modelCache)
{
_modelCache[info.Id] = response;
}
2016-09-30 06:50:06 +00:00
}
2017-03-02 20:50:09 +00:00
return response;
2016-07-08 03:22:16 +00:00
}
}
catch (HttpException ex)
2015-07-21 04:22:46 +00:00
{
2017-03-02 20:50:09 +00:00
if (!throwAllExceptions && ex.StatusCode.HasValue && ex.StatusCode.Value == System.Net.HttpStatusCode.NotFound)
2016-07-08 03:22:16 +00:00
{
2016-09-30 06:50:06 +00:00
var defaultValue = "HDHR";
2017-03-02 20:50:09 +00:00
var response = new DiscoverResponse
{
ModelNumber = defaultValue
};
2017-03-03 06:38:05 +00:00
if (!string.IsNullOrWhiteSpace(info.Id))
2016-09-30 06:50:06 +00:00
{
2017-03-03 06:38:05 +00:00
// HDHR4 doesn't have this api
lock (_modelCache)
{
_modelCache[info.Id] = response;
}
2016-09-30 06:50:06 +00:00
}
2017-03-02 20:50:09 +00:00
return response;
2016-07-08 03:22:16 +00:00
}
2015-07-21 04:22:46 +00:00
2016-07-08 03:22:16 +00:00
throw;
2016-02-28 21:31:54 +00:00
}
2015-07-24 02:48:10 +00:00
}
2017-03-02 20:50:09 +00:00
private async Task<List<LiveTvTunerInfo>> GetTunerInfos(TunerHostInfo info, CancellationToken cancellationToken)
2015-07-24 02:48:10 +00:00
{
2017-03-02 20:50:09 +00:00
var model = await GetModelInfo(info, false, cancellationToken).ConfigureAwait(false);
2015-07-24 02:48:10 +00:00
2017-03-02 20:50:09 +00:00
var tuners = new List<LiveTvTunerInfo>();
2017-03-03 04:36:20 +00:00
var uri = new Uri(GetApiUrl(info, false));
2017-03-02 20:56:25 +00:00
2017-03-02 20:50:09 +00:00
using (var manager = new HdHomerunManager(_socketFactory))
2015-07-20 18:32:55 +00:00
{
2017-03-02 20:50:09 +00:00
// Legacy HdHomeruns are IPv4 only
2017-03-03 20:16:43 +00:00
var ipInfo = _networkManager.ParseIpAddress(uri.Host);
2017-03-02 20:50:09 +00:00
for (int i = 0; i < model.TunerCount; ++i)
2015-07-20 18:32:55 +00:00
{
2017-03-02 20:50:09 +00:00
var name = String.Format("Tuner {0}", i + 1);
var currentChannel = "none"; /// @todo Get current channel and map back to Station Id
var isAvailable = await manager.CheckTunerAvailability(ipInfo, i, cancellationToken).ConfigureAwait(false);
LiveTvTunerStatus status = isAvailable ? LiveTvTunerStatus.Available : LiveTvTunerStatus.LiveTv;
tuners.Add(new LiveTvTunerInfo
2015-07-20 18:32:55 +00:00
{
2017-03-02 20:50:09 +00:00
Name = name,
SourceType = string.IsNullOrWhiteSpace(model.ModelNumber) ? Name : model.ModelNumber,
ProgramName = currentChannel,
Status = status
});
2015-07-20 18:32:55 +00:00
}
}
2017-03-02 20:50:09 +00:00
return tuners;
2015-07-20 18:32:55 +00:00
}
2015-08-19 16:43:23 +00:00
public async Task<List<LiveTvTunerInfo>> GetTunerInfos(CancellationToken cancellationToken)
{
var list = new List<LiveTvTunerInfo>();
foreach (var host in GetConfiguration().TunerHosts
2017-03-13 04:56:41 +00:00
.Where(i => string.Equals(i.Type, Type, StringComparison.OrdinalIgnoreCase)))
2015-08-19 16:43:23 +00:00
{
try
{
list.AddRange(await GetTunerInfos(host, cancellationToken).ConfigureAwait(false));
}
catch (Exception ex)
{
2015-08-19 17:58:41 +00:00
Logger.ErrorException("Error getting tuner info", ex);
2015-08-19 16:43:23 +00:00
}
}
return list;
}
2015-07-23 23:40:54 +00:00
private string GetApiUrl(TunerHostInfo info, bool isPlayback)
2015-07-20 18:32:55 +00:00
{
var url = info.Url;
2015-08-24 02:08:20 +00:00
if (string.IsNullOrWhiteSpace(url))
{
throw new ArgumentException("Invalid tuner info");
}
2015-07-20 18:32:55 +00:00
if (!url.StartsWith("http", StringComparison.OrdinalIgnoreCase))
{
url = "http://" + url;
}
2015-07-23 23:40:54 +00:00
var uri = new Uri(url);
if (isPlayback)
{
var builder = new UriBuilder(uri);
builder.Port = 5004;
uri = builder.Uri;
}
return uri.AbsoluteUri.TrimEnd('/');
2015-07-20 18:32:55 +00:00
}
private class Channels
{
public string GuideNumber { get; set; }
public string GuideName { get; set; }
2016-04-03 23:20:43 +00:00
public string VideoCodec { get; set; }
public string AudioCodec { get; set; }
2015-07-20 18:32:55 +00:00
public string URL { get; set; }
public bool Favorite { get; set; }
public bool DRM { get; set; }
2016-04-03 23:20:43 +00:00
public int HD { get; set; }
2015-07-20 18:32:55 +00:00
}
2017-03-26 16:26:52 +00:00
protected EncodingOptions GetEncodingOptions()
{
return Config.GetConfiguration<EncodingOptions>("encoding");
}
private string GetHdHrIdFromChannelId(string channelId)
{
return channelId.Split('_')[1];
}
2017-03-02 21:24:46 +00:00
private MediaSourceInfo GetMediaSource(TunerHostInfo info, string channelId, ChannelInfo channelInfo, string profile)
2015-07-20 18:32:55 +00:00
{
2015-07-24 02:48:10 +00:00
int? width = null;
int? height = null;
bool isInterlaced = true;
2016-04-03 23:20:43 +00:00
string videoCodec = null;
2017-03-02 21:24:46 +00:00
string audioCodec = null;
2015-10-30 16:40:12 +00:00
2015-07-24 02:48:10 +00:00
int? videoBitrate = null;
2016-04-14 19:12:00 +00:00
int? audioBitrate = null;
2015-07-24 02:48:10 +00:00
if (string.Equals(profile, "mobile", StringComparison.OrdinalIgnoreCase))
{
width = 1280;
height = 720;
isInterlaced = false;
videoCodec = "h264";
videoBitrate = 2000000;
}
else if (string.Equals(profile, "heavy", StringComparison.OrdinalIgnoreCase))
{
width = 1920;
height = 1080;
isInterlaced = false;
videoCodec = "h264";
2015-12-29 16:12:33 +00:00
videoBitrate = 15000000;
2015-07-24 02:48:10 +00:00
}
else if (string.Equals(profile, "internet540", StringComparison.OrdinalIgnoreCase))
{
width = 960;
height = 546;
2015-07-24 02:48:10 +00:00
isInterlaced = false;
videoCodec = "h264";
videoBitrate = 2500000;
}
else if (string.Equals(profile, "internet480", StringComparison.OrdinalIgnoreCase))
{
width = 848;
height = 480;
isInterlaced = false;
videoCodec = "h264";
videoBitrate = 2000000;
}
else if (string.Equals(profile, "internet360", StringComparison.OrdinalIgnoreCase))
{
width = 640;
height = 360;
isInterlaced = false;
videoCodec = "h264";
videoBitrate = 1500000;
}
else if (string.Equals(profile, "internet240", StringComparison.OrdinalIgnoreCase))
{
width = 432;
height = 240;
isInterlaced = false;
videoCodec = "h264";
videoBitrate = 1000000;
}
2017-03-02 21:24:46 +00:00
if (channelInfo != null)
2016-04-03 23:20:43 +00:00
{
2016-09-18 20:38:38 +00:00
if (string.IsNullOrWhiteSpace(videoCodec))
2016-04-03 23:20:43 +00:00
{
2017-03-02 21:24:46 +00:00
videoCodec = channelInfo.VideoCodec;
2016-09-18 20:38:38 +00:00
}
2017-03-02 21:24:46 +00:00
audioCodec = channelInfo.AudioCodec;
2016-04-03 23:20:43 +00:00
2016-09-18 20:38:38 +00:00
if (!videoBitrate.HasValue)
{
2017-03-02 21:24:46 +00:00
videoBitrate = (channelInfo.IsHD ?? true) ? 15000000 : 2000000;
2016-04-03 23:20:43 +00:00
}
2017-03-02 21:24:46 +00:00
audioBitrate = (channelInfo.IsHD ?? true) ? 448000 : 192000;
2016-04-03 23:20:43 +00:00
}
// normalize
if (string.Equals(videoCodec, "mpeg2", StringComparison.OrdinalIgnoreCase))
{
videoCodec = "mpeg2video";
}
2016-04-18 17:43:00 +00:00
string nal = null;
if (string.Equals(videoCodec, "h264", StringComparison.OrdinalIgnoreCase))
{
nal = "0";
}
2017-03-26 16:26:52 +00:00
var url = GetApiUrl(info, false);
2015-07-24 02:48:10 +00:00
2016-09-25 18:39:13 +00:00
var id = profile;
if (string.IsNullOrWhiteSpace(id))
{
id = "native";
}
2017-03-30 17:58:38 +00:00
id += "_" + channelId.GetMD5().ToString("N") + "_" + url.GetMD5().ToString("N");
2016-09-25 18:39:13 +00:00
2017-03-02 21:24:46 +00:00
var mediaSource = new MediaSourceInfo
{
Path = url,
Protocol = MediaProtocol.Udp,
MediaStreams = new List<MediaStream>
{
new MediaStream
{
Type = MediaStreamType.Video,
// Set the index to -1 because we don't know the exact index of the video stream within the container
Index = -1,
IsInterlaced = isInterlaced,
Codec = videoCodec,
Width = width,
Height = height,
BitRate = videoBitrate,
NalLengthSize = nal
},
new MediaStream
{
Type = MediaStreamType.Audio,
// Set the index to -1 because we don't know the exact index of the audio stream within the container
Index = -1,
Codec = audioCodec,
BitRate = audioBitrate
}
},
RequiresOpening = true,
RequiresClosing = true,
BufferMs = 0,
Container = "ts",
Id = id,
SupportsDirectPlay = false,
SupportsDirectStream = true,
SupportsTranscoding = true,
2017-04-30 20:03:28 +00:00
IsInfiniteStream = true,
2017-05-19 16:39:40 +00:00
IgnoreDts = true,
2017-05-23 16:44:11 +00:00
//IgnoreIndex = true,
//ReadAtNativeFramerate = true
2017-03-02 21:24:46 +00:00
};
mediaSource.InferTotalBitrate();
return mediaSource;
}
2015-08-19 19:25:18 +00:00
protected override async Task<List<MediaSourceInfo>> GetChannelStreamMediaSources(TunerHostInfo info, string channelId, CancellationToken cancellationToken)
2015-07-23 23:40:54 +00:00
{
var list = new List<MediaSourceInfo>();
2015-08-16 18:37:53 +00:00
if (!channelId.StartsWith(ChannelIdPrefix, StringComparison.OrdinalIgnoreCase))
{
return list;
}
2016-02-24 18:45:11 +00:00
var hdhrId = GetHdHrIdFromChannelId(channelId);
2015-08-16 18:37:53 +00:00
2017-03-02 21:24:46 +00:00
var channels = await GetChannels(info, true, CancellationToken.None).ConfigureAwait(false);
2017-03-03 05:53:21 +00:00
var channelInfo = channels.FirstOrDefault(i => string.Equals(i.Id, channelId, StringComparison.OrdinalIgnoreCase));
2017-03-02 21:24:46 +00:00
var hdHomerunChannelInfo = channelInfo as HdHomerunChannelInfo;
2017-03-02 21:27:46 +00:00
var isLegacyTuner = hdHomerunChannelInfo != null && hdHomerunChannelInfo.IsLegacyTuner;
2016-12-08 06:53:46 +00:00
2017-03-02 21:24:46 +00:00
if (isLegacyTuner)
{
2017-03-26 16:26:52 +00:00
list.Add(GetMediaSource(info, hdhrId, channelInfo, "native"));
2017-03-02 21:24:46 +00:00
}
else
{
try
2015-07-24 02:48:10 +00:00
{
2017-03-02 21:24:46 +00:00
var modelInfo = await GetModelInfo(info, false, cancellationToken).ConfigureAwait(false);
var model = modelInfo == null ? string.Empty : (modelInfo.ModelNumber ?? string.Empty);
2016-09-30 06:50:06 +00:00
2017-03-02 21:24:46 +00:00
if ((model.IndexOf("hdtc", StringComparison.OrdinalIgnoreCase) != -1))
2016-09-30 06:50:06 +00:00
{
2017-03-02 21:24:46 +00:00
list.Add(GetMediaSource(info, hdhrId, channelInfo, "native"));
2015-07-24 15:20:11 +00:00
2017-03-02 21:24:46 +00:00
if (info.AllowHWTranscoding)
{
list.Add(GetMediaSource(info, hdhrId, channelInfo, "heavy"));
list.Add(GetMediaSource(info, hdhrId, channelInfo, "internet540"));
list.Add(GetMediaSource(info, hdhrId, channelInfo, "internet480"));
list.Add(GetMediaSource(info, hdhrId, channelInfo, "internet360"));
list.Add(GetMediaSource(info, hdhrId, channelInfo, "internet240"));
list.Add(GetMediaSource(info, hdhrId, channelInfo, "mobile"));
}
2016-09-30 06:50:06 +00:00
}
2015-07-24 02:48:10 +00:00
}
2017-03-02 21:24:46 +00:00
catch
{
2015-08-16 18:37:53 +00:00
2017-03-02 21:24:46 +00:00
}
2015-07-24 02:48:10 +00:00
2017-03-02 21:24:46 +00:00
if (list.Count == 0)
{
list.Add(GetMediaSource(info, hdhrId, channelInfo, "native"));
}
2016-12-08 06:53:46 +00:00
}
2015-07-24 02:48:10 +00:00
return list;
2015-07-23 23:40:54 +00:00
}
2016-09-25 18:39:13 +00:00
protected override async Task<LiveStream> GetChannelStream(TunerHostInfo info, string channelId, string streamId, CancellationToken cancellationToken)
2015-07-23 23:40:54 +00:00
{
2016-09-25 18:39:13 +00:00
var profile = streamId.Split('_')[0];
Logger.Info("GetChannelStream: channel id: {0}. stream id: {1} profile: {2}", channelId, streamId, profile);
2015-09-01 19:18:25 +00:00
2016-02-24 18:45:11 +00:00
var hdhrId = GetHdHrIdFromChannelId(channelId);
2015-08-16 18:37:53 +00:00
2017-03-02 21:24:46 +00:00
var channels = await GetChannels(info, true, CancellationToken.None).ConfigureAwait(false);
2017-03-03 05:53:21 +00:00
var channelInfo = channels.FirstOrDefault(i => string.Equals(i.Id, channelId, StringComparison.OrdinalIgnoreCase));
2017-03-02 21:24:46 +00:00
2017-03-02 21:27:46 +00:00
var hdhomerunChannel = channelInfo as HdHomerunChannelInfo;
2016-09-25 18:39:13 +00:00
2017-03-26 16:26:52 +00:00
var mediaSource = GetMediaSource(info, hdhrId, channelInfo, profile);
var modelInfo = await GetModelInfo(info, false, cancellationToken).ConfigureAwait(false);
2017-03-02 21:27:46 +00:00
if (hdhomerunChannel != null && hdhomerunChannel.IsLegacyTuner)
2017-03-13 04:08:49 +00:00
{
2017-05-22 04:54:02 +00:00
return new HdHomerunUdpStream(mediaSource, streamId, new LegacyHdHomerunChannelCommands(hdhomerunChannel.Url), modelInfo.TunerCount, _fileSystem, _httpClient, Logger, Config.ApplicationPaths, _appHost, _socketFactory, _networkManager, _environment);
2017-03-02 21:27:46 +00:00
}
2017-03-27 00:27:29 +00:00
// The UDP method is not working reliably on OSX, and on BSD it hasn't been tested yet
2017-05-24 19:12:55 +00:00
var enableHttpStream = _environment.OperatingSystem == MediaBrowser.Model.System.OperatingSystem.OSX
|| _environment.OperatingSystem == MediaBrowser.Model.System.OperatingSystem.BSD;
2017-03-27 19:31:24 +00:00
enableHttpStream = true;
2017-03-27 00:27:29 +00:00
if (enableHttpStream)
2017-03-02 21:27:46 +00:00
{
2017-03-27 00:27:29 +00:00
mediaSource.Protocol = MediaProtocol.Http;
2017-03-26 19:54:50 +00:00
2017-03-27 00:27:29 +00:00
var httpUrl = GetApiUrl(info, true) + "/auto/v" + hdhrId;
// If raw was used, the tuner doesn't support params
2017-05-22 04:54:02 +00:00
if (!string.IsNullOrWhiteSpace(profile) && !string.Equals(profile, "native", StringComparison.OrdinalIgnoreCase))
2017-03-26 19:54:50 +00:00
{
2017-03-27 00:27:29 +00:00
httpUrl += "?transcode=" + profile;
2017-03-26 19:54:50 +00:00
}
2017-03-27 00:27:29 +00:00
mediaSource.Path = httpUrl;
2017-03-26 19:54:50 +00:00
2017-05-22 04:54:02 +00:00
return new HdHomerunHttpStream(mediaSource, streamId, _fileSystem, _httpClient, Logger, Config.ApplicationPaths, _appHost, _environment);
2017-03-02 21:27:46 +00:00
}
2017-03-27 00:27:29 +00:00
2017-05-22 04:54:02 +00:00
return new HdHomerunUdpStream(mediaSource, streamId, new HdHomerunChannelCommands(hdhomerunChannel.Number, profile), modelInfo.TunerCount, _fileSystem, _httpClient, Logger, Config.ApplicationPaths, _appHost, _socketFactory, _networkManager, _environment);
2015-07-23 23:40:54 +00:00
}
2015-07-23 13:23:22 +00:00
public async Task Validate(TunerHostInfo info)
{
2016-09-30 06:50:06 +00:00
lock (_modelCache)
{
_modelCache.Clear();
}
2016-07-08 03:22:16 +00:00
try
2016-03-08 05:00:03 +00:00
{
2016-07-08 03:22:16 +00:00
// Test it by pulling down the lineup
2017-03-02 20:50:09 +00:00
var modelInfo = await GetModelInfo(info, true, CancellationToken.None).ConfigureAwait(false);
info.DeviceId = modelInfo.DeviceID;
2016-07-08 03:22:16 +00:00
}
catch (HttpException ex)
2016-03-08 05:00:03 +00:00
{
2016-07-08 03:22:16 +00:00
if (ex.StatusCode.HasValue && ex.StatusCode.Value == System.Net.HttpStatusCode.NotFound)
{
// HDHR4 doesn't have this api
return;
}
2016-03-08 05:00:03 +00:00
2016-07-08 03:22:16 +00:00
throw;
2015-08-24 02:08:20 +00:00
}
2015-07-23 13:23:22 +00:00
}
2015-10-30 16:40:12 +00:00
protected override async Task<bool> IsAvailableInternal(TunerHostInfo tuner, string channelId, CancellationToken cancellationToken)
{
var info = await GetTunerInfos(tuner, cancellationToken).ConfigureAwait(false);
return info.Any(i => i.Status == LiveTvTunerStatus.Available);
}
2016-02-28 21:31:54 +00:00
public class DiscoverResponse
{
public string FriendlyName { get; set; }
public string ModelNumber { get; set; }
public string FirmwareName { get; set; }
public string FirmwareVersion { get; set; }
public string DeviceID { get; set; }
public string DeviceAuth { get; set; }
public string BaseURL { get; set; }
public string LineupURL { get; set; }
2017-03-02 20:50:09 +00:00
public int TunerCount { get; set; }
2017-03-16 17:21:24 +00:00
public DateTime DateQueried { get; set; }
public DiscoverResponse()
{
DateQueried = DateTime.UtcNow;
}
2016-02-28 21:31:54 +00:00
}
2017-03-13 04:08:49 +00:00
2017-03-13 04:49:10 +00:00
public async Task<List<TunerHostInfo>> DiscoverDevices(int discoveryDurationMs, CancellationToken cancellationToken)
2017-03-13 04:08:49 +00:00
{
2017-03-13 04:49:10 +00:00
cancellationToken = CancellationTokenSource.CreateLinkedTokenSource(new CancellationTokenSource(discoveryDurationMs).Token, cancellationToken).Token;
2017-03-13 04:08:49 +00:00
var list = new List<TunerHostInfo>();
// Create udp broadcast discovery message
byte[] discBytes = { 0, 2, 0, 12, 1, 4, 255, 255, 255, 255, 2, 4, 255, 255, 255, 255, 115, 204, 125, 143 };
using (var udpClient = _socketFactory.CreateUdpBroadcastSocket(0))
{
// Need a way to set the Receive timeout on the socket otherwise this might never timeout?
try
{
2017-05-24 19:12:55 +00:00
await udpClient.SendToAsync(discBytes, 0, discBytes.Length, new IpEndPointInfo(new IpAddressInfo("255.255.255.255", IpAddressFamily.InterNetwork), 65001), cancellationToken);
var receiveBuffer = new byte[8192];
2017-03-13 04:08:49 +00:00
while (!cancellationToken.IsCancellationRequested)
{
2017-05-24 19:12:55 +00:00
var response = await udpClient.ReceiveAsync(receiveBuffer, 0, receiveBuffer.Length, cancellationToken).ConfigureAwait(false);
2017-03-13 04:08:49 +00:00
var deviceIp = response.RemoteEndPoint.IpAddress.Address;
// check to make sure we have enough bytes received to be a valid message and make sure the 2nd byte is the discover reply byte
if (response.ReceivedBytes > 13 && response.Buffer[1] == 3)
{
var deviceAddress = "http://" + deviceIp;
var info = await TryGetTunerHostInfo(deviceAddress, cancellationToken).ConfigureAwait(false);
if (info != null)
{
list.Add(info);
}
}
}
}
catch (OperationCanceledException)
{
}
catch
{
// Socket timeout indicates all messages have been received.
}
}
return list;
}
private async Task<TunerHostInfo> TryGetTunerHostInfo(string url, CancellationToken cancellationToken)
{
var hostInfo = new TunerHostInfo
{
Type = Type,
Url = url
};
try
{
var modelInfo = await GetModelInfo(hostInfo, false, cancellationToken).ConfigureAwait(false);
hostInfo.DeviceId = modelInfo.DeviceID;
2017-03-13 20:42:21 +00:00
hostInfo.FriendlyName = modelInfo.FriendlyName;
2017-03-13 04:08:49 +00:00
return hostInfo;
}
catch
{
// logged at lower levels
}
return null;
}
2015-07-20 18:32:55 +00:00
}
}