123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517 |
- 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.Globalization;
- using System.IO;
- using System.Linq;
- using System.Threading;
- using System.Threading.Tasks;
- using MediaBrowser.Common.Extensions;
- using MediaBrowser.Controller.MediaEncoding;
- using MediaBrowser.Model.Configuration;
- using MediaBrowser.Model.Net;
- namespace MediaBrowser.Server.Implementations.LiveTv.TunerHosts.HdHomerun
- {
- public class HdHomerunHost : BaseTunerHost, ITunerHost, IConfigurableTunerHost
- {
- private readonly IHttpClient _httpClient;
- public HdHomerunHost(IConfigurationManager config, ILogger logger, IJsonSerializer jsonSerializer, IMediaEncoder mediaEncoder, IHttpClient httpClient)
- : base(config, logger, jsonSerializer, mediaEncoder)
- {
- _httpClient = httpClient;
- }
- public string Name
- {
- get { return "HD Homerun"; }
- }
- public override string Type
- {
- get { return DeviceType; }
- }
- public static string DeviceType
- {
- get { return "hdhomerun"; }
- }
- private const string ChannelIdPrefix = "hdhr_";
- private string GetChannelId(TunerHostInfo info, Channels i)
- {
- var id = ChannelIdPrefix + i.GuideNumber.ToString(CultureInfo.InvariantCulture);
- if (info.DataVersion >= 1)
- {
- id += '_' + (i.GuideName ?? string.Empty).GetMD5().ToString("N");
- }
- return id;
- }
- public string ApplyDuration(string streamPath, TimeSpan duration)
- {
- streamPath += streamPath.IndexOf('?') == -1 ? "?" : "&";
- streamPath += "duration=" + Convert.ToInt32(duration.TotalSeconds).ToString(CultureInfo.InvariantCulture);
- return streamPath;
- }
- private async Task<IEnumerable<Channels>> GetLineup(TunerHostInfo info, CancellationToken cancellationToken)
- {
- var options = new HttpRequestOptions
- {
- Url = string.Format("{0}/lineup.json", GetApiUrl(info, false)),
- CancellationToken = cancellationToken
- };
- using (var stream = await _httpClient.Get(options))
- {
- var lineup = JsonSerializer.DeserializeFromStream<List<Channels>>(stream) ?? new List<Channels>();
- if (info.ImportFavoritesOnly)
- {
- lineup = lineup.Where(i => i.Favorite).ToList();
- }
- return lineup.Where(i => !i.DRM).ToList();
- }
- }
- protected override async Task<IEnumerable<ChannelInfo>> GetChannelsInternal(TunerHostInfo info, CancellationToken cancellationToken)
- {
- var lineup = await GetLineup(info, cancellationToken).ConfigureAwait(false);
- return lineup.Select(i => new ChannelInfo
- {
- Name = i.GuideName,
- Number = i.GuideNumber.ToString(CultureInfo.InvariantCulture),
- Id = GetChannelId(info, i),
- IsFavorite = i.Favorite,
- TunerHostId = info.Id,
- IsHD = i.HD == 1,
- AudioCodec = i.AudioCodec,
- VideoCodec = i.VideoCodec
- });
- }
- private async Task<string> GetModelInfo(TunerHostInfo info, CancellationToken cancellationToken)
- {
- try
- {
- 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,
- TimeoutMs = Convert.ToInt32(TimeSpan.FromSeconds(5).TotalMilliseconds)
- }))
- {
- var response = JsonSerializer.DeserializeFromStream<DiscoverResponse>(stream);
- return response.ModelNumber;
- }
- }
- catch (HttpException ex)
- {
- if (ex.StatusCode.HasValue && ex.StatusCode.Value == System.Net.HttpStatusCode.NotFound)
- {
- // HDHR4 doesn't have this api
- return "HDHR";
- }
- throw;
- }
- }
- public async Task<List<LiveTvTunerInfo>> GetTunerInfos(TunerHostInfo info, CancellationToken cancellationToken)
- {
- var model = await GetModelInfo(info, cancellationToken).ConfigureAwait(false);
- using (var stream = await _httpClient.Get(new HttpRequestOptions()
- {
- Url = string.Format("{0}/tuners.html", GetApiUrl(info, false)),
- CancellationToken = cancellationToken,
- TimeoutMs = Convert.ToInt32(TimeSpan.FromSeconds(5).TotalMilliseconds)
- }))
- {
- var tuners = new List<LiveTvTunerInfo>();
- using (var sr = new StreamReader(stream, System.Text.Encoding.UTF8))
- {
- while (!sr.EndOfStream)
- {
- string line = StripXML(sr.ReadLine());
- if (line.Contains("Channel"))
- {
- LiveTvTunerStatus status;
- var index = line.IndexOf("Channel", StringComparison.OrdinalIgnoreCase);
- var name = line.Substring(0, index - 1);
- var currentChannel = line.Substring(index + 7);
- if (currentChannel != "none") { status = LiveTvTunerStatus.LiveTv; } else { status = LiveTvTunerStatus.Available; }
- tuners.Add(new LiveTvTunerInfo
- {
- Name = name,
- SourceType = string.IsNullOrWhiteSpace(model) ? Name : model,
- ProgramName = currentChannel,
- Status = status
- });
- }
- }
- }
- return tuners;
- }
- }
- 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)
- {
- Logger.ErrorException("Error getting tuner info", ex);
- }
- }
- return list;
- }
- private string GetApiUrl(TunerHostInfo info, bool isPlayback)
- {
- var url = info.Url;
- if (string.IsNullOrWhiteSpace(url))
- {
- throw new ArgumentException("Invalid tuner info");
- }
- if (!url.StartsWith("http", StringComparison.OrdinalIgnoreCase))
- {
- url = "http://" + url;
- }
- var uri = new Uri(url);
- if (isPlayback)
- {
- var builder = new UriBuilder(uri);
- builder.Port = 5004;
- uri = builder.Uri;
- }
- return uri.AbsoluteUri.TrimEnd('/');
- }
- private static string StripXML(string source)
- {
- char[] buffer = new char[source.Length];
- int bufferIndex = 0;
- bool inside = false;
- for (int i = 0; i < source.Length; i++)
- {
- char let = source[i];
- if (let == '<')
- {
- inside = true;
- continue;
- }
- if (let == '>')
- {
- inside = false;
- continue;
- }
- if (!inside)
- {
- buffer[bufferIndex] = let;
- bufferIndex++;
- }
- }
- return new string(buffer, 0, bufferIndex);
- }
- private class Channels
- {
- public string GuideNumber { get; set; }
- public string GuideName { get; set; }
- public string VideoCodec { get; set; }
- public string AudioCodec { get; set; }
- public string URL { get; set; }
- public bool Favorite { get; set; }
- public bool DRM { get; set; }
- public int HD { get; set; }
- }
- private async Task<MediaSourceInfo> GetMediaSource(TunerHostInfo info, string channelId, string profile)
- {
- int? width = null;
- int? height = null;
- bool isInterlaced = true;
- string videoCodec = null;
- string audioCodec = "ac3";
- int? videoBitrate = null;
- int? audioBitrate = null;
- 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";
- videoBitrate = 15000000;
- }
- else if (string.Equals(profile, "internet540", StringComparison.OrdinalIgnoreCase))
- {
- width = 960;
- height = 546;
- 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;
- }
- if (string.IsNullOrWhiteSpace(videoCodec))
- {
- var channels = await GetChannels(info, true, CancellationToken.None).ConfigureAwait(false);
- var channel = channels.FirstOrDefault(i => string.Equals(i.Number, channelId, StringComparison.OrdinalIgnoreCase));
- if (channel != null)
- {
- videoCodec = channel.VideoCodec;
- audioCodec = channel.AudioCodec;
- videoBitrate = (channel.IsHD ?? true) ? 15000000 : 2000000;
- audioBitrate = (channel.IsHD ?? true) ? 448000 : 192000;
- }
- }
- // normalize
- if (string.Equals(videoCodec, "mpeg2", StringComparison.OrdinalIgnoreCase))
- {
- videoCodec = "mpeg2video";
- }
- string nal = null;
- if (string.Equals(videoCodec, "h264", StringComparison.OrdinalIgnoreCase))
- {
- nal = "0";
- }
- var url = GetApiUrl(info, true) + "/auto/v" + channelId;
- if (!string.IsNullOrWhiteSpace(profile) && !string.Equals(profile, "native", StringComparison.OrdinalIgnoreCase))
- {
- url += "?transcode=" + profile;
- }
- var mediaSource = new MediaSourceInfo
- {
- Path = url,
- Protocol = MediaProtocol.Http,
- 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 = false,
- RequiresClosing = false,
- BufferMs = 0,
- Container = "ts",
- Id = profile,
- SupportsDirectPlay = true,
- SupportsDirectStream = false,
- SupportsTranscoding = true
- };
- return mediaSource;
- }
- protected EncodingOptions GetEncodingOptions()
- {
- return Config.GetConfiguration<EncodingOptions>("encoding");
- }
- private string GetHdHrIdFromChannelId(string channelId)
- {
- return channelId.Split('_')[1];
- }
- protected override async Task<List<MediaSourceInfo>> GetChannelStreamMediaSources(TunerHostInfo info, string channelId, CancellationToken cancellationToken)
- {
- var list = new List<MediaSourceInfo>();
- if (!channelId.StartsWith(ChannelIdPrefix, StringComparison.OrdinalIgnoreCase))
- {
- return list;
- }
- var hdhrId = GetHdHrIdFromChannelId(channelId);
- list.Add(await GetMediaSource(info, hdhrId, "native").ConfigureAwait(false));
- try
- {
- string model = await GetModelInfo(info, cancellationToken).ConfigureAwait(false);
- model = model ?? string.Empty;
- if (info.AllowHWTranscoding && (model.IndexOf("hdtc", StringComparison.OrdinalIgnoreCase) != -1))
- {
- list.Add(await GetMediaSource(info, hdhrId, "heavy").ConfigureAwait(false));
- list.Add(await GetMediaSource(info, hdhrId, "internet540").ConfigureAwait(false));
- list.Add(await GetMediaSource(info, hdhrId, "internet480").ConfigureAwait(false));
- list.Add(await GetMediaSource(info, hdhrId, "internet360").ConfigureAwait(false));
- list.Add(await GetMediaSource(info, hdhrId, "internet240").ConfigureAwait(false));
- list.Add(await GetMediaSource(info, hdhrId, "mobile").ConfigureAwait(false));
- }
- }
- catch (Exception ex)
- {
- }
- return list;
- }
- protected override bool IsValidChannelId(string channelId)
- {
- if (string.IsNullOrWhiteSpace(channelId))
- {
- throw new ArgumentNullException("channelId");
- }
- return channelId.StartsWith(ChannelIdPrefix, StringComparison.OrdinalIgnoreCase);
- }
- protected override async Task<MediaSourceInfo> GetChannelStream(TunerHostInfo info, string channelId, string streamId, CancellationToken cancellationToken)
- {
- Logger.Info("GetChannelStream: channel id: {0}. stream id: {1}", channelId, streamId ?? string.Empty);
- if (!channelId.StartsWith(ChannelIdPrefix, StringComparison.OrdinalIgnoreCase))
- {
- throw new ArgumentException("Channel not found");
- }
- var hdhrId = GetHdHrIdFromChannelId(channelId);
- return await GetMediaSource(info, hdhrId, streamId).ConfigureAwait(false);
- }
- public async Task Validate(TunerHostInfo info)
- {
- if (!info.IsEnabled)
- {
- return;
- }
- try
- {
- // Test it by pulling down the lineup
- using (var stream = await _httpClient.Get(new HttpRequestOptions
- {
- Url = string.Format("{0}/discover.json", GetApiUrl(info, false)),
- CancellationToken = CancellationToken.None
- }))
- {
- var response = JsonSerializer.DeserializeFromStream<DiscoverResponse>(stream);
- info.DeviceId = response.DeviceID;
- }
- }
- catch (HttpException ex)
- {
- if (ex.StatusCode.HasValue && ex.StatusCode.Value == System.Net.HttpStatusCode.NotFound)
- {
- // HDHR4 doesn't have this api
- return;
- }
- throw;
- }
- }
- 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);
- }
- 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; }
- }
- }
- }
|