BaseTunerHost.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. using MediaBrowser.Common.Configuration;
  2. using MediaBrowser.Controller.LiveTv;
  3. using MediaBrowser.Model.Dto;
  4. using MediaBrowser.Model.LiveTv;
  5. using MediaBrowser.Model.Logging;
  6. using System;
  7. using System.Collections.Concurrent;
  8. using System.Collections.Generic;
  9. using System.IO;
  10. using System.Linq;
  11. using System.Threading;
  12. using System.Threading.Tasks;
  13. using MediaBrowser.Common.Extensions;
  14. using MediaBrowser.Controller.Configuration;
  15. using MediaBrowser.Controller.MediaEncoding;
  16. using MediaBrowser.Model.Dlna;
  17. using MediaBrowser.Model.IO;
  18. using MediaBrowser.Model.Serialization;
  19. namespace Emby.Server.Implementations.LiveTv.TunerHosts
  20. {
  21. public abstract class BaseTunerHost
  22. {
  23. protected readonly IServerConfigurationManager Config;
  24. protected readonly ILogger Logger;
  25. protected IJsonSerializer JsonSerializer;
  26. protected readonly IMediaEncoder MediaEncoder;
  27. protected readonly IFileSystem FileSystem;
  28. private readonly ConcurrentDictionary<string, ChannelCache> _channelCache =
  29. new ConcurrentDictionary<string, ChannelCache>(StringComparer.OrdinalIgnoreCase);
  30. protected BaseTunerHost(IServerConfigurationManager config, ILogger logger, IJsonSerializer jsonSerializer, IMediaEncoder mediaEncoder, IFileSystem fileSystem)
  31. {
  32. Config = config;
  33. Logger = logger;
  34. JsonSerializer = jsonSerializer;
  35. MediaEncoder = mediaEncoder;
  36. FileSystem = fileSystem;
  37. }
  38. protected abstract Task<List<ChannelInfo>> GetChannelsInternal(TunerHostInfo tuner, CancellationToken cancellationToken);
  39. public abstract string Type { get; }
  40. public async Task<List<ChannelInfo>> GetChannels(TunerHostInfo tuner, bool enableCache, CancellationToken cancellationToken)
  41. {
  42. ChannelCache cache = null;
  43. var key = tuner.Id;
  44. if (enableCache && !string.IsNullOrWhiteSpace(key) && _channelCache.TryGetValue(key, out cache))
  45. {
  46. if (DateTime.UtcNow - cache.Date < TimeSpan.FromMinutes(60))
  47. {
  48. return cache.Channels.ToList();
  49. }
  50. }
  51. var result = await GetChannelsInternal(tuner, cancellationToken).ConfigureAwait(false);
  52. var list = result.ToList();
  53. Logger.Info("Channels from {0}: {1}", tuner.Url, JsonSerializer.SerializeToString(list));
  54. if (!string.IsNullOrWhiteSpace(key) && list.Count > 0)
  55. {
  56. cache = cache ?? new ChannelCache();
  57. cache.Date = DateTime.UtcNow;
  58. cache.Channels = list;
  59. _channelCache.AddOrUpdate(key, cache, (k, v) => cache);
  60. }
  61. return list;
  62. }
  63. protected virtual List<TunerHostInfo> GetTunerHosts()
  64. {
  65. return GetConfiguration().TunerHosts
  66. .Where(i => string.Equals(i.Type, Type, StringComparison.OrdinalIgnoreCase))
  67. .ToList();
  68. }
  69. public async Task<List<ChannelInfo>> GetChannels(bool enableCache, CancellationToken cancellationToken)
  70. {
  71. var list = new List<ChannelInfo>();
  72. var hosts = GetTunerHosts();
  73. foreach (var host in hosts)
  74. {
  75. var channelCacheFile = Path.Combine(Config.ApplicationPaths.CachePath, host.Id + "_channels");
  76. try
  77. {
  78. var channels = await GetChannels(host, enableCache, cancellationToken).ConfigureAwait(false);
  79. var newChannels = channels.Where(i => !list.Any(l => string.Equals(i.Id, l.Id, StringComparison.OrdinalIgnoreCase))).ToList();
  80. list.AddRange(newChannels);
  81. if (!enableCache)
  82. {
  83. try
  84. {
  85. FileSystem.CreateDirectory(FileSystem.GetDirectoryName(channelCacheFile));
  86. JsonSerializer.SerializeToFile(channels, channelCacheFile);
  87. }
  88. catch (IOException)
  89. {
  90. }
  91. }
  92. }
  93. catch (Exception ex)
  94. {
  95. Logger.ErrorException("Error getting channel list", ex);
  96. if (enableCache)
  97. {
  98. try
  99. {
  100. var channels = JsonSerializer.DeserializeFromFile<List<ChannelInfo>>(channelCacheFile);
  101. list.AddRange(channels);
  102. }
  103. catch (IOException)
  104. {
  105. }
  106. }
  107. }
  108. }
  109. return list;
  110. }
  111. protected abstract Task<List<MediaSourceInfo>> GetChannelStreamMediaSources(TunerHostInfo tuner, string channelId, CancellationToken cancellationToken);
  112. public async Task<List<MediaSourceInfo>> GetChannelStreamMediaSources(string channelId, CancellationToken cancellationToken)
  113. {
  114. if (string.IsNullOrWhiteSpace(channelId))
  115. {
  116. throw new ArgumentNullException("channelId");
  117. }
  118. if (IsValidChannelId(channelId))
  119. {
  120. var hosts = GetTunerHosts();
  121. var hostsWithChannel = new List<TunerHostInfo>();
  122. foreach (var host in hosts)
  123. {
  124. try
  125. {
  126. var channels = await GetChannels(host, true, cancellationToken).ConfigureAwait(false);
  127. if (channels.Any(i => string.Equals(i.Id, channelId, StringComparison.OrdinalIgnoreCase)))
  128. {
  129. hostsWithChannel.Add(host);
  130. }
  131. }
  132. catch (Exception ex)
  133. {
  134. Logger.Error("Error getting channels", ex);
  135. }
  136. }
  137. foreach (var host in hostsWithChannel)
  138. {
  139. try
  140. {
  141. // Check to make sure the tuner is available
  142. // If there's only one tuner, don't bother with the check and just let the tuner be the one to throw an error
  143. if (hostsWithChannel.Count > 1 && !await IsAvailable(host, channelId, cancellationToken).ConfigureAwait(false))
  144. {
  145. Logger.Error("Tuner is not currently available");
  146. continue;
  147. }
  148. var mediaSources = await GetChannelStreamMediaSources(host, channelId, cancellationToken).ConfigureAwait(false);
  149. // Prefix the id with the host Id so that we can easily find it
  150. foreach (var mediaSource in mediaSources)
  151. {
  152. mediaSource.Id = host.Id + mediaSource.Id;
  153. }
  154. return mediaSources;
  155. }
  156. catch (Exception ex)
  157. {
  158. Logger.Error("Error opening tuner", ex);
  159. }
  160. }
  161. }
  162. return new List<MediaSourceInfo>();
  163. }
  164. protected abstract Task<LiveStream> GetChannelStream(TunerHostInfo tuner, string channelId, string streamId, CancellationToken cancellationToken);
  165. public async Task<LiveStream> GetChannelStream(string channelId, string streamId, CancellationToken cancellationToken)
  166. {
  167. if (string.IsNullOrWhiteSpace(channelId))
  168. {
  169. throw new ArgumentNullException("channelId");
  170. }
  171. if (!IsValidChannelId(channelId))
  172. {
  173. throw new FileNotFoundException();
  174. }
  175. var hosts = GetTunerHosts();
  176. var hostsWithChannel = new List<TunerHostInfo>();
  177. foreach (var host in hosts)
  178. {
  179. if (string.IsNullOrWhiteSpace(streamId))
  180. {
  181. try
  182. {
  183. var channels = await GetChannels(host, true, cancellationToken).ConfigureAwait(false);
  184. if (channels.Any(i => string.Equals(i.Id, channelId, StringComparison.OrdinalIgnoreCase)))
  185. {
  186. hostsWithChannel.Add(host);
  187. }
  188. }
  189. catch (Exception ex)
  190. {
  191. Logger.Error("Error getting channels", ex);
  192. }
  193. }
  194. else if (streamId.StartsWith(host.Id, StringComparison.OrdinalIgnoreCase))
  195. {
  196. hostsWithChannel = new List<TunerHostInfo> { host };
  197. streamId = streamId.Substring(host.Id.Length);
  198. break;
  199. }
  200. }
  201. foreach (var host in hostsWithChannel)
  202. {
  203. if (!channelId.StartsWith(ChannelIdPrefix, StringComparison.OrdinalIgnoreCase))
  204. {
  205. continue;
  206. }
  207. try
  208. {
  209. var liveStream = await GetChannelStream(host, channelId, streamId, cancellationToken).ConfigureAwait(false);
  210. var startTime = DateTime.UtcNow;
  211. await liveStream.Open(cancellationToken).ConfigureAwait(false);
  212. var endTime = DateTime.UtcNow;
  213. Logger.Info("Live stream opened after {0}ms", (endTime - startTime).TotalMilliseconds);
  214. return liveStream;
  215. }
  216. catch (Exception ex)
  217. {
  218. Logger.Error("Error opening tuner", ex);
  219. }
  220. }
  221. throw new LiveTvConflictException();
  222. }
  223. protected async Task<bool> IsAvailable(TunerHostInfo tuner, string channelId, CancellationToken cancellationToken)
  224. {
  225. try
  226. {
  227. return await IsAvailableInternal(tuner, channelId, cancellationToken).ConfigureAwait(false);
  228. }
  229. catch (Exception ex)
  230. {
  231. Logger.ErrorException("Error checking tuner availability", ex);
  232. return false;
  233. }
  234. }
  235. protected abstract Task<bool> IsAvailableInternal(TunerHostInfo tuner, string channelId, CancellationToken cancellationToken);
  236. protected virtual string ChannelIdPrefix
  237. {
  238. get
  239. {
  240. return Type + "_";
  241. }
  242. }
  243. protected virtual bool IsValidChannelId(string channelId)
  244. {
  245. if (string.IsNullOrWhiteSpace(channelId))
  246. {
  247. throw new ArgumentNullException("channelId");
  248. }
  249. return channelId.StartsWith(ChannelIdPrefix, StringComparison.OrdinalIgnoreCase);
  250. }
  251. protected LiveTvOptions GetConfiguration()
  252. {
  253. return Config.GetConfiguration<LiveTvOptions>("livetv");
  254. }
  255. private class ChannelCache
  256. {
  257. public DateTime Date;
  258. public List<ChannelInfo> Channels;
  259. }
  260. }
  261. }