2015-07-20 11:32:55 -07: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 12:02:04 -07:00
|
|
|
|
using MediaBrowser.Model.IO;
|
2016-02-24 11:45:11 -07:00
|
|
|
|
using MediaBrowser.Common.Extensions;
|
2016-09-25 11:39:13 -07:00
|
|
|
|
using MediaBrowser.Controller;
|
|
|
|
|
using MediaBrowser.Controller.Configuration;
|
2015-10-13 19:41:46 -07:00
|
|
|
|
using MediaBrowser.Controller.MediaEncoding;
|
2015-10-30 09:40:12 -07:00
|
|
|
|
using MediaBrowser.Model.Configuration;
|
2016-07-07 20:22:16 -07:00
|
|
|
|
using MediaBrowser.Model.Net;
|
2015-07-20 11:32:55 -07:00
|
|
|
|
|
2016-11-03 16:35:19 -07:00
|
|
|
|
namespace Emby.Server.Implementations.LiveTv.TunerHosts.HdHomerun
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
2016-02-18 23:20:18 -07:00
|
|
|
|
public class HdHomerunHost : BaseTunerHost, ITunerHost, IConfigurableTunerHost
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
|
|
|
|
private readonly IHttpClient _httpClient;
|
2016-09-25 11:39:13 -07:00
|
|
|
|
private readonly IFileSystem _fileSystem;
|
|
|
|
|
private readonly IServerApplicationHost _appHost;
|
2017-03-02 13:50:09 -07:00
|
|
|
|
private readonly ISocketFactory _socketFactory;
|
2017-03-02 22:53:21 -07:00
|
|
|
|
private readonly INetworkManager _networkManager;
|
2015-07-20 11:32:55 -07:00
|
|
|
|
|
2017-03-02 22:53:21 -07:00
|
|
|
|
public HdHomerunHost(IServerConfigurationManager config, ILogger logger, IJsonSerializer jsonSerializer, IMediaEncoder mediaEncoder, IHttpClient httpClient, IFileSystem fileSystem, IServerApplicationHost appHost, ISocketFactory socketFactory, INetworkManager networkManager)
|
2015-10-30 09:40:12 -07:00
|
|
|
|
: base(config, logger, jsonSerializer, mediaEncoder)
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
|
|
|
|
_httpClient = httpClient;
|
2016-09-25 11:39:13 -07:00
|
|
|
|
_fileSystem = fileSystem;
|
|
|
|
|
_appHost = appHost;
|
2017-03-02 13:50:09 -07:00
|
|
|
|
_socketFactory = socketFactory;
|
2017-03-02 22:53:21 -07:00
|
|
|
|
_networkManager = networkManager;
|
2015-07-20 11:32:55 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string Name
|
|
|
|
|
{
|
|
|
|
|
get { return "HD Homerun"; }
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-19 10:58:41 -07:00
|
|
|
|
public override string Type
|
2015-07-23 09:32:34 -07:00
|
|
|
|
{
|
|
|
|
|
get { return DeviceType; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static string DeviceType
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
|
|
|
|
get { return "hdhomerun"; }
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-16 11:37:53 -07:00
|
|
|
|
private const string ChannelIdPrefix = "hdhr_";
|
|
|
|
|
|
2016-02-25 13:29:38 -07:00
|
|
|
|
private string GetChannelId(TunerHostInfo info, Channels i)
|
|
|
|
|
{
|
2016-11-03 16:35:19 -07:00
|
|
|
|
var id = ChannelIdPrefix + i.GuideNumber;
|
2016-02-25 13:29:38 -07:00
|
|
|
|
|
2017-01-23 14:51:23 -07:00
|
|
|
|
id += '_' + (i.GuideName ?? string.Empty).GetMD5().ToString("N");
|
2016-02-25 13:29:38 -07:00
|
|
|
|
|
|
|
|
|
return id;
|
|
|
|
|
}
|
|
|
|
|
|
2017-02-18 20:46:09 -07:00
|
|
|
|
private async Task<List<Channels>> GetLineup(TunerHostInfo info, CancellationToken cancellationToken)
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
2017-03-05 19:32:56 -07:00
|
|
|
|
var model = await GetModelInfo(info, false, cancellationToken).ConfigureAwait(false);
|
|
|
|
|
|
2015-07-20 11:32:55 -07:00
|
|
|
|
var options = new HttpRequestOptions
|
|
|
|
|
{
|
2017-03-05 19:32:56 -07:00
|
|
|
|
Url = model.LineupURL,
|
2016-10-06 11:55:01 -07:00
|
|
|
|
CancellationToken = cancellationToken,
|
|
|
|
|
BufferContent = false
|
2015-07-20 11:32:55 -07:00
|
|
|
|
};
|
2017-02-18 20:46:09 -07:00
|
|
|
|
using (var stream = await _httpClient.Get(options).ConfigureAwait(false))
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
2016-04-03 16:20:43 -07:00
|
|
|
|
var lineup = JsonSerializer.DeserializeFromStream<List<Channels>>(stream) ?? new List<Channels>();
|
2015-07-20 11:32:55 -07:00
|
|
|
|
|
2016-04-03 16:20:43 -07:00
|
|
|
|
if (info.ImportFavoritesOnly)
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
2016-04-03 16:20:43 -07:00
|
|
|
|
lineup = lineup.Where(i => i.Favorite).ToList();
|
|
|
|
|
}
|
2015-07-20 11:32:55 -07:00
|
|
|
|
|
2016-04-05 19:18:56 -07:00
|
|
|
|
return lineup.Where(i => !i.DRM).ToList();
|
2016-04-03 16:20:43 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-07-25 11:11:46 -07:00
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
private class HdHomerunChannelInfo : ChannelInfo
|
|
|
|
|
{
|
2017-03-02 14:27:46 -07:00
|
|
|
|
public bool IsLegacyTuner { get; set; }
|
2017-03-02 22:53:21 -07:00
|
|
|
|
public string Url { get; set; }
|
2017-03-02 14:24:46 -07:00
|
|
|
|
}
|
|
|
|
|
|
2017-02-23 12:13:26 -07:00
|
|
|
|
protected override async Task<List<ChannelInfo>> GetChannelsInternal(TunerHostInfo info, CancellationToken cancellationToken)
|
2016-04-03 16:20:43 -07:00
|
|
|
|
{
|
|
|
|
|
var lineup = await GetLineup(info, cancellationToken).ConfigureAwait(false);
|
2015-07-25 11:11:46 -07:00
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
return lineup.Select(i => new HdHomerunChannelInfo
|
2016-04-03 16:20:43 -07:00
|
|
|
|
{
|
|
|
|
|
Name = i.GuideName,
|
2016-11-03 16:35:19 -07:00
|
|
|
|
Number = i.GuideNumber,
|
2016-04-03 16:20:43 -07:00
|
|
|
|
Id = GetChannelId(info, i),
|
|
|
|
|
IsFavorite = i.Favorite,
|
2016-04-03 17:01:03 -07:00
|
|
|
|
TunerHostId = info.Id,
|
|
|
|
|
IsHD = i.HD == 1,
|
|
|
|
|
AudioCodec = i.AudioCodec,
|
2017-01-01 13:47:54 -07:00
|
|
|
|
VideoCodec = i.VideoCodec,
|
2017-03-02 14:24:46 -07:00
|
|
|
|
ChannelType = ChannelType.TV,
|
2017-03-02 22:53:21 -07:00
|
|
|
|
IsLegacyTuner = (i.URL ?? string.Empty).StartsWith("hdhomerun", StringComparison.OrdinalIgnoreCase),
|
|
|
|
|
Url = i.URL
|
2017-02-23 12:13:26 -07:00
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
}).Cast<ChannelInfo>().ToList();
|
2015-07-20 11:32:55 -07:00
|
|
|
|
}
|
|
|
|
|
|
2016-10-07 08:08:13 -07:00
|
|
|
|
private readonly Dictionary<string, DiscoverResponse> _modelCache = new Dictionary<string, DiscoverResponse>();
|
2017-03-02 13:50:09 -07:00
|
|
|
|
private async Task<DiscoverResponse> GetModelInfo(TunerHostInfo info, bool throwAllExceptions, CancellationToken cancellationToken)
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
2016-09-29 23:50:06 -07:00
|
|
|
|
lock (_modelCache)
|
|
|
|
|
{
|
|
|
|
|
DiscoverResponse response;
|
|
|
|
|
if (_modelCache.TryGetValue(info.Url, out response))
|
|
|
|
|
{
|
2017-03-02 13:50:09 -07:00
|
|
|
|
return response;
|
2016-09-29 23:50:06 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-07 20:22:16 -07:00
|
|
|
|
try
|
2015-07-20 21:22:46 -07:00
|
|
|
|
{
|
2016-07-07 20:22:16 -07:00
|
|
|
|
using (var stream = await _httpClient.Get(new HttpRequestOptions()
|
|
|
|
|
{
|
|
|
|
|
Url = string.Format("{0}/discover.json", GetApiUrl(info, false)),
|
|
|
|
|
CancellationToken = cancellationToken,
|
|
|
|
|
CacheLength = TimeSpan.FromDays(1),
|
|
|
|
|
CacheMode = CacheMode.Unconditional,
|
2016-10-06 11:55:01 -07:00
|
|
|
|
TimeoutMs = Convert.ToInt32(TimeSpan.FromSeconds(5).TotalMilliseconds),
|
|
|
|
|
BufferContent = false
|
2017-02-18 20:46:09 -07:00
|
|
|
|
|
|
|
|
|
}).ConfigureAwait(false))
|
2016-07-07 20:22:16 -07:00
|
|
|
|
{
|
|
|
|
|
var response = JsonSerializer.DeserializeFromStream<DiscoverResponse>(stream);
|
|
|
|
|
|
2017-03-02 23:38:05 -07:00
|
|
|
|
if (!string.IsNullOrWhiteSpace(info.Id))
|
2016-09-29 23:50:06 -07:00
|
|
|
|
{
|
2017-03-02 23:38:05 -07:00
|
|
|
|
lock (_modelCache)
|
|
|
|
|
{
|
|
|
|
|
_modelCache[info.Id] = response;
|
|
|
|
|
}
|
2016-09-29 23:50:06 -07:00
|
|
|
|
}
|
|
|
|
|
|
2017-03-02 13:50:09 -07:00
|
|
|
|
return response;
|
2016-07-07 20:22:16 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (HttpException ex)
|
2015-07-20 21:22:46 -07:00
|
|
|
|
{
|
2017-03-02 13:50:09 -07:00
|
|
|
|
if (!throwAllExceptions && ex.StatusCode.HasValue && ex.StatusCode.Value == System.Net.HttpStatusCode.NotFound)
|
2016-07-07 20:22:16 -07:00
|
|
|
|
{
|
2016-09-29 23:50:06 -07:00
|
|
|
|
var defaultValue = "HDHR";
|
2017-03-02 13:50:09 -07:00
|
|
|
|
var response = new DiscoverResponse
|
|
|
|
|
{
|
|
|
|
|
ModelNumber = defaultValue
|
|
|
|
|
};
|
2017-03-02 23:38:05 -07:00
|
|
|
|
if (!string.IsNullOrWhiteSpace(info.Id))
|
2016-09-29 23:50:06 -07:00
|
|
|
|
{
|
2017-03-02 23:38:05 -07:00
|
|
|
|
// HDHR4 doesn't have this api
|
|
|
|
|
lock (_modelCache)
|
|
|
|
|
{
|
|
|
|
|
_modelCache[info.Id] = response;
|
|
|
|
|
}
|
2016-09-29 23:50:06 -07:00
|
|
|
|
}
|
2017-03-02 13:50:09 -07:00
|
|
|
|
return response;
|
2016-07-07 20:22:16 -07:00
|
|
|
|
}
|
2015-07-20 21:22:46 -07:00
|
|
|
|
|
2016-07-07 20:22:16 -07:00
|
|
|
|
throw;
|
2016-02-28 14:31:54 -07:00
|
|
|
|
}
|
2015-07-23 19:48:10 -07:00
|
|
|
|
}
|
|
|
|
|
|
2017-03-02 13:50:09 -07:00
|
|
|
|
private async Task<List<LiveTvTunerInfo>> GetTunerInfos(TunerHostInfo info, CancellationToken cancellationToken)
|
2015-07-23 19:48:10 -07:00
|
|
|
|
{
|
2017-03-02 13:50:09 -07:00
|
|
|
|
var model = await GetModelInfo(info, false, cancellationToken).ConfigureAwait(false);
|
2015-07-23 19:48:10 -07:00
|
|
|
|
|
2017-03-02 13:50:09 -07:00
|
|
|
|
var tuners = new List<LiveTvTunerInfo>();
|
2017-02-18 20:46:09 -07:00
|
|
|
|
|
2017-03-02 21:36:20 -07:00
|
|
|
|
var uri = new Uri(GetApiUrl(info, false));
|
2017-03-02 13:56:25 -07:00
|
|
|
|
|
2017-03-02 13:50:09 -07:00
|
|
|
|
using (var manager = new HdHomerunManager(_socketFactory))
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
2017-03-02 13:50:09 -07:00
|
|
|
|
// Legacy HdHomeruns are IPv4 only
|
2017-03-03 13:16:43 -07:00
|
|
|
|
var ipInfo = _networkManager.ParseIpAddress(uri.Host);
|
2017-03-02 13:50:09 -07:00
|
|
|
|
|
|
|
|
|
for (int i = 0; i < model.TunerCount; ++i)
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
2017-03-02 13:50:09 -07: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 11:32:55 -07:00
|
|
|
|
{
|
2017-03-02 13:50:09 -07:00
|
|
|
|
Name = name,
|
|
|
|
|
SourceType = string.IsNullOrWhiteSpace(model.ModelNumber) ? Name : model.ModelNumber,
|
|
|
|
|
ProgramName = currentChannel,
|
|
|
|
|
Status = status
|
|
|
|
|
});
|
2015-07-20 11:32:55 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
2017-03-02 13:50:09 -07:00
|
|
|
|
return tuners;
|
2015-07-20 11:32:55 -07:00
|
|
|
|
}
|
|
|
|
|
|
2015-08-19 09:43:23 -07:00
|
|
|
|
public async Task<List<LiveTvTunerInfo>> GetTunerInfos(CancellationToken cancellationToken)
|
|
|
|
|
{
|
|
|
|
|
var list = new List<LiveTvTunerInfo>();
|
|
|
|
|
|
|
|
|
|
foreach (var host in GetConfiguration().TunerHosts
|
|
|
|
|
.Where(i => i.IsEnabled && string.Equals(i.Type, Type, StringComparison.OrdinalIgnoreCase)))
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
list.AddRange(await GetTunerInfos(host, cancellationToken).ConfigureAwait(false));
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
2015-08-19 10:58:41 -07:00
|
|
|
|
Logger.ErrorException("Error getting tuner info", ex);
|
2015-08-19 09:43:23 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return list;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 16:40:54 -07:00
|
|
|
|
private string GetApiUrl(TunerHostInfo info, bool isPlayback)
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
|
|
|
|
var url = info.Url;
|
|
|
|
|
|
2015-08-23 19:08:20 -07:00
|
|
|
|
if (string.IsNullOrWhiteSpace(url))
|
|
|
|
|
{
|
|
|
|
|
throw new ArgumentException("Invalid tuner info");
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-20 11:32:55 -07:00
|
|
|
|
if (!url.StartsWith("http", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
url = "http://" + url;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 16:40:54 -07: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 11:32:55 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private class Channels
|
|
|
|
|
{
|
|
|
|
|
public string GuideNumber { get; set; }
|
|
|
|
|
public string GuideName { get; set; }
|
2016-04-03 16:20:43 -07:00
|
|
|
|
public string VideoCodec { get; set; }
|
|
|
|
|
public string AudioCodec { get; set; }
|
2015-07-20 11:32:55 -07:00
|
|
|
|
public string URL { get; set; }
|
|
|
|
|
public bool Favorite { get; set; }
|
|
|
|
|
public bool DRM { get; set; }
|
2016-04-03 16:20:43 -07:00
|
|
|
|
public int HD { get; set; }
|
2015-07-20 11:32:55 -07:00
|
|
|
|
}
|
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
private MediaSourceInfo GetMediaSource(TunerHostInfo info, string channelId, ChannelInfo channelInfo, string profile)
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
2015-07-23 19:48:10 -07:00
|
|
|
|
int? width = null;
|
|
|
|
|
int? height = null;
|
|
|
|
|
bool isInterlaced = true;
|
2016-04-03 16:20:43 -07:00
|
|
|
|
string videoCodec = null;
|
2017-03-02 14:24:46 -07:00
|
|
|
|
string audioCodec = null;
|
2015-10-30 09:40:12 -07:00
|
|
|
|
|
2015-07-23 19:48:10 -07:00
|
|
|
|
int? videoBitrate = null;
|
2016-04-14 12:12:00 -07:00
|
|
|
|
int? audioBitrate = null;
|
2015-07-23 19:48:10 -07: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 09:12:33 -07:00
|
|
|
|
videoBitrate = 15000000;
|
2015-07-23 19:48:10 -07:00
|
|
|
|
}
|
|
|
|
|
else if (string.Equals(profile, "internet540", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
2016-04-03 23:39:40 -07:00
|
|
|
|
width = 960;
|
|
|
|
|
height = 546;
|
2015-07-23 19:48:10 -07: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 14:24:46 -07:00
|
|
|
|
if (channelInfo != null)
|
2016-04-03 16:20:43 -07:00
|
|
|
|
{
|
2016-09-18 13:38:38 -07:00
|
|
|
|
if (string.IsNullOrWhiteSpace(videoCodec))
|
2016-04-03 16:20:43 -07:00
|
|
|
|
{
|
2017-03-02 14:24:46 -07:00
|
|
|
|
videoCodec = channelInfo.VideoCodec;
|
2016-09-18 13:38:38 -07:00
|
|
|
|
}
|
2017-03-02 14:24:46 -07:00
|
|
|
|
audioCodec = channelInfo.AudioCodec;
|
2016-04-03 16:20:43 -07:00
|
|
|
|
|
2016-09-18 13:38:38 -07:00
|
|
|
|
if (!videoBitrate.HasValue)
|
|
|
|
|
{
|
2017-03-02 14:24:46 -07:00
|
|
|
|
videoBitrate = (channelInfo.IsHD ?? true) ? 15000000 : 2000000;
|
2016-04-03 16:20:43 -07:00
|
|
|
|
}
|
2017-03-02 14:24:46 -07:00
|
|
|
|
audioBitrate = (channelInfo.IsHD ?? true) ? 448000 : 192000;
|
2016-04-03 16:20:43 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// normalize
|
|
|
|
|
if (string.Equals(videoCodec, "mpeg2", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
videoCodec = "mpeg2video";
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-18 10:43:00 -07:00
|
|
|
|
string nal = null;
|
|
|
|
|
if (string.Equals(videoCodec, "h264", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
nal = "0";
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 19:48:10 -07:00
|
|
|
|
var url = GetApiUrl(info, true) + "/auto/v" + channelId;
|
|
|
|
|
|
2016-12-07 23:53:46 -07:00
|
|
|
|
// If raw was used, the tuner doesn't support params
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(profile)
|
|
|
|
|
&& !string.Equals(profile, "native", StringComparison.OrdinalIgnoreCase))
|
2015-07-23 19:48:10 -07:00
|
|
|
|
{
|
|
|
|
|
url += "?transcode=" + profile;
|
|
|
|
|
}
|
|
|
|
|
|
2016-09-25 11:39:13 -07:00
|
|
|
|
var id = profile;
|
|
|
|
|
if (string.IsNullOrWhiteSpace(id))
|
|
|
|
|
{
|
|
|
|
|
id = "native";
|
|
|
|
|
}
|
|
|
|
|
id += "_" + url.GetMD5().ToString("N");
|
|
|
|
|
|
2015-07-23 16:40:54 -07:00
|
|
|
|
var mediaSource = new MediaSourceInfo
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
2015-07-23 19:48:10 -07:00
|
|
|
|
Path = url,
|
2015-07-23 16:40:54 -07:00
|
|
|
|
Protocol = MediaProtocol.Http,
|
|
|
|
|
MediaStreams = new List<MediaStream>
|
2015-07-20 11:32:55 -07:00
|
|
|
|
{
|
|
|
|
|
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,
|
2015-07-23 19:48:10 -07:00
|
|
|
|
IsInterlaced = isInterlaced,
|
|
|
|
|
Codec = videoCodec,
|
|
|
|
|
Width = width,
|
|
|
|
|
Height = height,
|
2016-04-18 10:43:00 -07:00
|
|
|
|
BitRate = videoBitrate,
|
|
|
|
|
NalLengthSize = nal
|
2016-04-03 17:01:03 -07:00
|
|
|
|
|
2015-07-20 11:32:55 -07:00
|
|
|
|
},
|
|
|
|
|
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
|
2015-07-23 19:48:10 -07:00
|
|
|
|
Index = -1,
|
2016-04-03 16:20:43 -07:00
|
|
|
|
Codec = audioCodec,
|
2016-04-14 12:12:00 -07:00
|
|
|
|
BitRate = audioBitrate
|
2015-07-20 11:32:55 -07:00
|
|
|
|
}
|
2015-07-23 16:40:54 -07:00
|
|
|
|
},
|
2016-09-18 13:38:38 -07:00
|
|
|
|
RequiresOpening = true,
|
2015-10-30 09:40:12 -07:00
|
|
|
|
RequiresClosing = false,
|
2015-12-29 09:12:33 -07:00
|
|
|
|
BufferMs = 0,
|
2015-07-25 11:42:39 -07:00
|
|
|
|
Container = "ts",
|
2016-09-25 11:39:13 -07:00
|
|
|
|
Id = id,
|
2016-10-12 11:23:09 -07:00
|
|
|
|
SupportsDirectPlay = false,
|
|
|
|
|
SupportsDirectStream = true,
|
2016-09-29 19:21:24 -07:00
|
|
|
|
SupportsTranscoding = true,
|
|
|
|
|
IsInfiniteStream = true
|
2015-07-23 16:40:54 -07:00
|
|
|
|
};
|
2015-07-20 11:32:55 -07:00
|
|
|
|
|
2017-01-21 13:27:07 -07:00
|
|
|
|
mediaSource.InferTotalBitrate();
|
|
|
|
|
|
2015-07-23 16:40:54 -07:00
|
|
|
|
return mediaSource;
|
2015-07-20 11:32:55 -07:00
|
|
|
|
}
|
2015-07-23 06:23:22 -07:00
|
|
|
|
|
2015-10-30 09:40:12 -07:00
|
|
|
|
protected EncodingOptions GetEncodingOptions()
|
|
|
|
|
{
|
|
|
|
|
return Config.GetConfiguration<EncodingOptions>("encoding");
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-24 11:45:11 -07:00
|
|
|
|
private string GetHdHrIdFromChannelId(string channelId)
|
|
|
|
|
{
|
|
|
|
|
return channelId.Split('_')[1];
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
private MediaSourceInfo GetLegacyMediaSource(TunerHostInfo info, string channelId, ChannelInfo channel)
|
|
|
|
|
{
|
|
|
|
|
int? width = null;
|
|
|
|
|
int? height = null;
|
|
|
|
|
bool isInterlaced = true;
|
|
|
|
|
string videoCodec = null;
|
|
|
|
|
string audioCodec = null;
|
|
|
|
|
|
|
|
|
|
int? videoBitrate = null;
|
|
|
|
|
int? audioBitrate = null;
|
|
|
|
|
|
|
|
|
|
if (channel != null)
|
|
|
|
|
{
|
|
|
|
|
if (string.IsNullOrWhiteSpace(videoCodec))
|
|
|
|
|
{
|
|
|
|
|
videoCodec = channel.VideoCodec;
|
|
|
|
|
}
|
|
|
|
|
audioCodec = channel.AudioCodec;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// normalize
|
|
|
|
|
if (string.Equals(videoCodec, "mpeg2", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
videoCodec = "mpeg2video";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
string nal = null;
|
|
|
|
|
|
2017-03-05 19:32:56 -07:00
|
|
|
|
var url = GetApiUrl(info, false);
|
2017-03-02 14:24:46 -07:00
|
|
|
|
var id = channelId;
|
|
|
|
|
id += "_" + url.GetMD5().ToString("N");
|
|
|
|
|
|
|
|
|
|
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,
|
|
|
|
|
IsInfiniteStream = true
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
mediaSource.InferTotalBitrate();
|
|
|
|
|
|
|
|
|
|
return mediaSource;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-19 12:25:18 -07:00
|
|
|
|
protected override async Task<List<MediaSourceInfo>> GetChannelStreamMediaSources(TunerHostInfo info, string channelId, CancellationToken cancellationToken)
|
2015-07-23 16:40:54 -07:00
|
|
|
|
{
|
|
|
|
|
var list = new List<MediaSourceInfo>();
|
|
|
|
|
|
2015-08-16 11:37:53 -07:00
|
|
|
|
if (!channelId.StartsWith(ChannelIdPrefix, StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
return list;
|
|
|
|
|
}
|
2016-02-24 11:45:11 -07:00
|
|
|
|
var hdhrId = GetHdHrIdFromChannelId(channelId);
|
2015-08-16 11:37:53 -07:00
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
var channels = await GetChannels(info, true, CancellationToken.None).ConfigureAwait(false);
|
2017-03-02 22:53:21 -07:00
|
|
|
|
var channelInfo = channels.FirstOrDefault(i => string.Equals(i.Id, channelId, StringComparison.OrdinalIgnoreCase));
|
2017-03-02 14:24:46 -07:00
|
|
|
|
|
|
|
|
|
var hdHomerunChannelInfo = channelInfo as HdHomerunChannelInfo;
|
|
|
|
|
|
2017-03-02 14:27:46 -07:00
|
|
|
|
var isLegacyTuner = hdHomerunChannelInfo != null && hdHomerunChannelInfo.IsLegacyTuner;
|
2016-12-07 23:53:46 -07:00
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
if (isLegacyTuner)
|
|
|
|
|
{
|
|
|
|
|
list.Add(GetLegacyMediaSource(info, hdhrId, channelInfo));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
try
|
2015-07-23 19:48:10 -07:00
|
|
|
|
{
|
2017-03-02 14:24:46 -07:00
|
|
|
|
var modelInfo = await GetModelInfo(info, false, cancellationToken).ConfigureAwait(false);
|
|
|
|
|
var model = modelInfo == null ? string.Empty : (modelInfo.ModelNumber ?? string.Empty);
|
2016-09-29 23:50:06 -07:00
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
if ((model.IndexOf("hdtc", StringComparison.OrdinalIgnoreCase) != -1))
|
2016-09-29 23:50:06 -07:00
|
|
|
|
{
|
2017-03-02 14:24:46 -07:00
|
|
|
|
list.Add(GetMediaSource(info, hdhrId, channelInfo, "native"));
|
2015-07-24 08:20:11 -07:00
|
|
|
|
|
2017-03-02 14:24:46 -07: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-29 23:50:06 -07:00
|
|
|
|
}
|
2015-07-23 19:48:10 -07:00
|
|
|
|
}
|
2017-03-02 14:24:46 -07:00
|
|
|
|
catch
|
|
|
|
|
{
|
2015-08-16 11:37:53 -07:00
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
}
|
2015-07-23 19:48:10 -07:00
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
if (list.Count == 0)
|
|
|
|
|
{
|
|
|
|
|
list.Add(GetMediaSource(info, hdhrId, channelInfo, "native"));
|
|
|
|
|
}
|
2016-12-07 23:53:46 -07:00
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 19:48:10 -07:00
|
|
|
|
return list;
|
2015-07-23 16:40:54 -07:00
|
|
|
|
}
|
|
|
|
|
|
2015-08-19 12:25:18 -07:00
|
|
|
|
protected override bool IsValidChannelId(string channelId)
|
|
|
|
|
{
|
2015-10-04 20:24:24 -07:00
|
|
|
|
if (string.IsNullOrWhiteSpace(channelId))
|
|
|
|
|
{
|
|
|
|
|
throw new ArgumentNullException("channelId");
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-19 12:25:18 -07:00
|
|
|
|
return channelId.StartsWith(ChannelIdPrefix, StringComparison.OrdinalIgnoreCase);
|
|
|
|
|
}
|
|
|
|
|
|
2016-09-25 11:39:13 -07:00
|
|
|
|
protected override async Task<LiveStream> GetChannelStream(TunerHostInfo info, string channelId, string streamId, CancellationToken cancellationToken)
|
2015-07-23 16:40:54 -07:00
|
|
|
|
{
|
2016-09-25 11:39:13 -07:00
|
|
|
|
var profile = streamId.Split('_')[0];
|
|
|
|
|
|
|
|
|
|
Logger.Info("GetChannelStream: channel id: {0}. stream id: {1} profile: {2}", channelId, streamId, profile);
|
2015-09-01 12:18:25 -07:00
|
|
|
|
|
2015-08-16 11:37:53 -07:00
|
|
|
|
if (!channelId.StartsWith(ChannelIdPrefix, StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
2015-10-13 19:41:46 -07:00
|
|
|
|
throw new ArgumentException("Channel not found");
|
2015-08-16 11:37:53 -07:00
|
|
|
|
}
|
2016-02-24 11:45:11 -07:00
|
|
|
|
var hdhrId = GetHdHrIdFromChannelId(channelId);
|
2015-08-16 11:37:53 -07:00
|
|
|
|
|
2017-03-02 14:24:46 -07:00
|
|
|
|
var channels = await GetChannels(info, true, CancellationToken.None).ConfigureAwait(false);
|
2017-03-02 22:53:21 -07:00
|
|
|
|
var channelInfo = channels.FirstOrDefault(i => string.Equals(i.Id, channelId, StringComparison.OrdinalIgnoreCase));
|
2017-03-02 14:24:46 -07:00
|
|
|
|
|
2017-03-02 14:27:46 -07:00
|
|
|
|
var hdhomerunChannel = channelInfo as HdHomerunChannelInfo;
|
2016-09-25 11:39:13 -07:00
|
|
|
|
|
2017-03-02 14:27:46 -07:00
|
|
|
|
if (hdhomerunChannel != null && hdhomerunChannel.IsLegacyTuner)
|
2017-03-05 19:32:56 -07:00
|
|
|
|
{
|
2017-03-02 14:27:46 -07:00
|
|
|
|
var mediaSource = GetLegacyMediaSource(info, hdhrId, channelInfo);
|
2017-03-05 19:32:56 -07:00
|
|
|
|
var modelInfo = await GetModelInfo(info, false, cancellationToken).ConfigureAwait(false);
|
2017-03-02 14:27:46 -07:00
|
|
|
|
|
2017-03-05 19:32:56 -07:00
|
|
|
|
return new HdHomerunUdpStream(mediaSource, streamId, new LegacyHdHomerunChannelCommands(hdhomerunChannel.Url), modelInfo.TunerCount, _fileSystem, _httpClient, Logger, Config.ApplicationPaths, _appHost, _socketFactory, _networkManager);
|
2017-03-02 14:27:46 -07:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
var mediaSource = GetMediaSource(info, hdhrId, channelInfo, profile);
|
2017-03-07 11:27:56 -07:00
|
|
|
|
//var modelInfo = await GetModelInfo(info, false, cancellationToken).ConfigureAwait(false);
|
2017-03-02 14:27:46 -07:00
|
|
|
|
|
2017-03-05 19:32:56 -07:00
|
|
|
|
return new HdHomerunHttpStream(mediaSource, streamId, _fileSystem, _httpClient, Logger, Config.ApplicationPaths, _appHost);
|
|
|
|
|
//return new HdHomerunUdpStream(mediaSource, streamId, new HdHomerunChannelCommands(hdhomerunChannel.Number), modelInfo.TunerCount, _fileSystem, _httpClient, Logger, Config.ApplicationPaths, _appHost, _socketFactory, _networkManager);
|
2017-03-02 14:27:46 -07:00
|
|
|
|
}
|
2015-07-23 16:40:54 -07:00
|
|
|
|
}
|
2015-07-23 06:23:22 -07:00
|
|
|
|
|
|
|
|
|
public async Task Validate(TunerHostInfo info)
|
|
|
|
|
{
|
2016-03-07 22:00:03 -07:00
|
|
|
|
if (!info.IsEnabled)
|
2015-08-23 19:08:20 -07:00
|
|
|
|
{
|
2016-03-07 22:00:03 -07:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2016-09-29 23:50:06 -07:00
|
|
|
|
lock (_modelCache)
|
|
|
|
|
{
|
|
|
|
|
_modelCache.Clear();
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-07 20:22:16 -07:00
|
|
|
|
try
|
2016-03-07 22:00:03 -07:00
|
|
|
|
{
|
2016-07-07 20:22:16 -07:00
|
|
|
|
// Test it by pulling down the lineup
|
2017-03-02 13:50:09 -07:00
|
|
|
|
var modelInfo = await GetModelInfo(info, true, CancellationToken.None).ConfigureAwait(false);
|
|
|
|
|
info.DeviceId = modelInfo.DeviceID;
|
2016-07-07 20:22:16 -07:00
|
|
|
|
}
|
|
|
|
|
catch (HttpException ex)
|
2016-03-07 22:00:03 -07:00
|
|
|
|
{
|
2016-07-07 20:22:16 -07:00
|
|
|
|
if (ex.StatusCode.HasValue && ex.StatusCode.Value == System.Net.HttpStatusCode.NotFound)
|
|
|
|
|
{
|
|
|
|
|
// HDHR4 doesn't have this api
|
|
|
|
|
return;
|
|
|
|
|
}
|
2016-03-07 22:00:03 -07:00
|
|
|
|
|
2016-07-07 20:22:16 -07:00
|
|
|
|
throw;
|
2015-08-23 19:08:20 -07:00
|
|
|
|
}
|
2015-07-23 06:23:22 -07:00
|
|
|
|
}
|
2015-10-30 09:40:12 -07: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 14:31:54 -07: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 13:50:09 -07:00
|
|
|
|
public int TunerCount { get; set; }
|
2016-02-28 14:31:54 -07:00
|
|
|
|
}
|
2015-07-20 11:32:55 -07:00
|
|
|
|
}
|
|
|
|
|
}
|