123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251 |
- #nullable disable
- #pragma warning disable CS1591
- using System;
- using System.Collections.Generic;
- using System.Globalization;
- using System.IO;
- using System.IO.Compression;
- using System.Linq;
- using System.Net.Http;
- using System.Threading;
- using System.Threading.Tasks;
- using Jellyfin.Extensions;
- using Jellyfin.XmlTv;
- using Jellyfin.XmlTv.Entities;
- using MediaBrowser.Common.Extensions;
- using MediaBrowser.Common.Net;
- using MediaBrowser.Controller.Configuration;
- using MediaBrowser.Controller.LiveTv;
- using MediaBrowser.Model.Dto;
- using MediaBrowser.Model.IO;
- using MediaBrowser.Model.LiveTv;
- using Microsoft.Extensions.Logging;
- namespace Emby.Server.Implementations.LiveTv.Listings
- {
- public class XmlTvListingsProvider : IListingsProvider
- {
- private static readonly TimeSpan _maxCacheAge = TimeSpan.FromHours(1);
- private readonly IServerConfigurationManager _config;
- private readonly IHttpClientFactory _httpClientFactory;
- private readonly ILogger<XmlTvListingsProvider> _logger;
- public XmlTvListingsProvider(
- IServerConfigurationManager config,
- IHttpClientFactory httpClientFactory,
- ILogger<XmlTvListingsProvider> logger)
- {
- _config = config;
- _httpClientFactory = httpClientFactory;
- _logger = logger;
- }
- public string Name => "XmlTV";
- public string Type => "xmltv";
- private string GetLanguage(ListingsProviderInfo info)
- {
- if (!string.IsNullOrWhiteSpace(info.PreferredLanguage))
- {
- return info.PreferredLanguage;
- }
- return _config.Configuration.PreferredMetadataLanguage;
- }
- private async Task<string> GetXml(ListingsProviderInfo info, CancellationToken cancellationToken)
- {
- _logger.LogInformation("xmltv path: {Path}", info.Path);
- string cacheFilename = info.Id + ".xml";
- string cacheFile = Path.Combine(_config.ApplicationPaths.CachePath, "xmltv", cacheFilename);
- if (File.Exists(cacheFile) && File.GetLastWriteTimeUtc(cacheFile) >= DateTime.UtcNow.Subtract(_maxCacheAge))
- {
- return cacheFile;
- }
- // Must check if file exists as parent directory may not exist.
- if (File.Exists(cacheFile))
- {
- File.Delete(cacheFile);
- }
- else
- {
- Directory.CreateDirectory(Path.GetDirectoryName(cacheFile));
- }
- if (info.Path.StartsWith("http", StringComparison.OrdinalIgnoreCase))
- {
- _logger.LogInformation("Downloading xmltv listings from {Path}", info.Path);
- using var response = await _httpClientFactory.CreateClient(NamedClient.Default).GetAsync(info.Path, cancellationToken).ConfigureAwait(false);
- await using var stream = await response.Content.ReadAsStreamAsync(cancellationToken).ConfigureAwait(false);
- return await UnzipIfNeededAndCopy(info.Path, stream, cacheFile, cancellationToken).ConfigureAwait(false);
- }
- else
- {
- await using var stream = AsyncFile.OpenRead(info.Path);
- return await UnzipIfNeededAndCopy(info.Path, stream, cacheFile, cancellationToken).ConfigureAwait(false);
- }
- }
- private async Task<string> UnzipIfNeededAndCopy(string originalUrl, Stream stream, string file, CancellationToken cancellationToken)
- {
- await using var fileStream = new FileStream(file, FileMode.CreateNew, FileAccess.Write, FileShare.None, IODefaults.FileStreamBufferSize, FileOptions.Asynchronous);
- if (Path.GetExtension(originalUrl.AsSpan().LeftPart('?')).Equals(".gz", StringComparison.OrdinalIgnoreCase))
- {
- try
- {
- using var reader = new GZipStream(stream, CompressionMode.Decompress);
- await reader.CopyToAsync(fileStream, cancellationToken).ConfigureAwait(false);
- }
- catch (Exception ex)
- {
- _logger.LogError(ex, "Error extracting from gz file {File}", originalUrl);
- }
- }
- else
- {
- await stream.CopyToAsync(fileStream, cancellationToken).ConfigureAwait(false);
- }
- return file;
- }
- public async Task<IEnumerable<ProgramInfo>> GetProgramsAsync(ListingsProviderInfo info, string channelId, DateTime startDateUtc, DateTime endDateUtc, CancellationToken cancellationToken)
- {
- if (string.IsNullOrWhiteSpace(channelId))
- {
- throw new ArgumentNullException(nameof(channelId));
- }
- _logger.LogDebug("Getting xmltv programs for channel {Id}", channelId);
- string path = await GetXml(info, cancellationToken).ConfigureAwait(false);
- _logger.LogDebug("Opening XmlTvReader for {Path}", path);
- var reader = new XmlTvReader(path, GetLanguage(info));
- return reader.GetProgrammes(channelId, startDateUtc, endDateUtc, cancellationToken)
- .Select(p => GetProgramInfo(p, info));
- }
- private static ProgramInfo GetProgramInfo(XmlTvProgram program, ListingsProviderInfo info)
- {
- string episodeTitle = program.Episode?.Title;
- var programInfo = new ProgramInfo
- {
- ChannelId = program.ChannelId,
- EndDate = program.EndDate.UtcDateTime,
- EpisodeNumber = program.Episode?.Episode,
- EpisodeTitle = episodeTitle,
- Genres = program.Categories,
- StartDate = program.StartDate.UtcDateTime,
- Name = program.Title,
- Overview = program.Description,
- ProductionYear = program.CopyrightDate?.Year,
- SeasonNumber = program.Episode?.Series,
- IsSeries = program.Episode is not null,
- IsRepeat = program.IsPreviouslyShown && !program.IsNew,
- IsPremiere = program.Premiere is not null,
- IsKids = program.Categories.Any(c => info.KidsCategories.Contains(c, StringComparison.OrdinalIgnoreCase)),
- IsMovie = program.Categories.Any(c => info.MovieCategories.Contains(c, StringComparison.OrdinalIgnoreCase)),
- IsNews = program.Categories.Any(c => info.NewsCategories.Contains(c, StringComparison.OrdinalIgnoreCase)),
- IsSports = program.Categories.Any(c => info.SportsCategories.Contains(c, StringComparison.OrdinalIgnoreCase)),
- ImageUrl = string.IsNullOrEmpty(program.Icon?.Source) ? null : program.Icon.Source,
- HasImage = !string.IsNullOrEmpty(program.Icon?.Source),
- OfficialRating = string.IsNullOrEmpty(program.Rating?.Value) ? null : program.Rating.Value,
- CommunityRating = program.StarRating,
- SeriesId = program.Episode is null ? null : program.Title?.GetMD5().ToString("N", CultureInfo.InvariantCulture)
- };
- if (string.IsNullOrWhiteSpace(program.ProgramId))
- {
- string uniqueString = (program.Title ?? string.Empty) + (episodeTitle ?? string.Empty);
- if (programInfo.SeasonNumber.HasValue)
- {
- uniqueString = "-" + programInfo.SeasonNumber.Value.ToString(CultureInfo.InvariantCulture);
- }
- if (programInfo.EpisodeNumber.HasValue)
- {
- uniqueString = "-" + programInfo.EpisodeNumber.Value.ToString(CultureInfo.InvariantCulture);
- }
- programInfo.ShowId = uniqueString.GetMD5().ToString("N", CultureInfo.InvariantCulture);
- // If we don't have valid episode info, assume it's a unique program, otherwise recordings might be skipped
- if (programInfo.IsSeries
- && !programInfo.IsRepeat
- && (programInfo.EpisodeNumber ?? 0) == 0)
- {
- programInfo.ShowId += programInfo.StartDate.Ticks.ToString(CultureInfo.InvariantCulture);
- }
- }
- else
- {
- programInfo.ShowId = program.ProgramId;
- }
- // Construct an id from the channel and start date
- programInfo.Id = string.Format(CultureInfo.InvariantCulture, "{0}_{1:O}", program.ChannelId, program.StartDate);
- if (programInfo.IsMovie)
- {
- programInfo.IsSeries = false;
- programInfo.EpisodeNumber = null;
- programInfo.EpisodeTitle = null;
- }
- return programInfo;
- }
- public Task Validate(ListingsProviderInfo info, bool validateLogin, bool validateListings)
- {
- // Assume all urls are valid. check files for existence
- if (!info.Path.StartsWith("http", StringComparison.OrdinalIgnoreCase) && !File.Exists(info.Path))
- {
- throw new FileNotFoundException("Could not find the XmlTv file specified:", info.Path);
- }
- return Task.CompletedTask;
- }
- public async Task<List<NameIdPair>> GetLineups(ListingsProviderInfo info, string country, string location)
- {
- // In theory this should never be called because there is always only one lineup
- string path = await GetXml(info, CancellationToken.None).ConfigureAwait(false);
- _logger.LogDebug("Opening XmlTvReader for {Path}", path);
- var reader = new XmlTvReader(path, GetLanguage(info));
- IEnumerable<XmlTvChannel> results = reader.GetChannels();
- // Should this method be async?
- return results.Select(c => new NameIdPair() { Id = c.Id, Name = c.DisplayName }).ToList();
- }
- public async Task<List<ChannelInfo>> GetChannels(ListingsProviderInfo info, CancellationToken cancellationToken)
- {
- // In theory this should never be called because there is always only one lineup
- string path = await GetXml(info, cancellationToken).ConfigureAwait(false);
- _logger.LogDebug("Opening XmlTvReader for {Path}", path);
- var reader = new XmlTvReader(path, GetLanguage(info));
- var results = reader.GetChannels();
- // Should this method be async?
- return results.Select(c => new ChannelInfo
- {
- Id = c.Id,
- Name = c.DisplayName,
- ImageUrl = string.IsNullOrEmpty(c.Icon.Source) ? null : c.Icon.Source,
- Number = string.IsNullOrWhiteSpace(c.Number) ? c.Id : c.Number
- }).ToList();
- }
- }
- }
|