using System.ServiceModel.Syndication; using NLog; using NzbDrone.Core.Model; using NzbDrone.Core.Providers.Core; using NzbDrone.Core.Repository; using SubSonic.Repository; namespace NzbDrone.Core.Providers.Indexer { public abstract class IndexerProviderBase { protected static readonly Logger Logger = LogManager.GetCurrentClassLogger(); protected readonly ConfigProvider _configProvider; protected readonly EpisodeProvider _episodeProvider; private readonly HttpProvider _httpProvider; protected readonly IRepository _repository; private readonly IndexerProvider _indexerProvider; protected readonly SeasonProvider _seasonProvider; protected readonly SeriesProvider _seriesProvider; public IndexerProviderBase(SeriesProvider seriesProvider, SeasonProvider seasonProvider, EpisodeProvider episodeProvider, ConfigProvider configProvider, HttpProvider httpProvider, IRepository repository, IndexerProvider indexerProvider) { _seriesProvider = seriesProvider; _seasonProvider = seasonProvider; _episodeProvider = episodeProvider; _configProvider = configProvider; _httpProvider = httpProvider; _repository = repository; _indexerProvider = indexerProvider; } /// /// Gets the source URL for the feed /// protected abstract string[] Url { get; } /// /// Gets the name for the feed /// public abstract string Name { get; } /// /// Generates direct link to download an NZB /// /// RSS Feed item to generate the link for /// Download link URL protected abstract string NzbDownloadUrl(SyndicationItem item); /// /// Parses the RSS feed item and. /// /// RSS feed item to parse /// Detailed episode info protected EpisodeParseResult ParseFeed(SyndicationItem item) { var episodeParseResult = Parser.ParseEpisodeInfo(item.Title.Text); if (episodeParseResult == null) return null; var seriesInfo = _seriesProvider.FindSeries(episodeParseResult.SeriesTitle); if (seriesInfo != null) { episodeParseResult.SeriesId = seriesInfo.SeriesId; episodeParseResult.SeriesTitle = seriesInfo.Title; return episodeParseResult; } Logger.Debug("Unable to map {0} to any of series in database", episodeParseResult.SeriesTitle); return null; } /// /// Fetches RSS feed and process each news item. /// public void Fetch() { Logger.Info("Fetching feeds from " + Settings.Name); foreach (var url in Url) { Logger.Debug("Downloading RSS " + url); var feed = SyndicationFeed.Load(_httpProvider.DownloadXml(url)).Items; foreach (var item in feed) { ProcessItem(item); } } Logger.Info("Finished processing feeds from " + Settings.Name); } private void ProcessItem(SyndicationItem feedItem) { Logger.Info("Processing RSS feed item " + feedItem.Title.Text); var parseResult = ParseFeed(feedItem); if (parseResult != null) { if (!_seriesProvider.IsMonitored(parseResult.SeriesId)) { Logger.Debug("{0} is present in the DB but not tracked. skipping.", parseResult.SeriesTitle); return; } if (!_seriesProvider.QualityWanted(parseResult.SeriesId, parseResult.Quality)) { Logger.Debug("Post doesn't meet the quality requirements [{0}]. skipping.", parseResult.Quality); return; } if (_seasonProvider.IsIgnored(parseResult.SeriesId, parseResult.SeasonNumber)) { Logger.Debug("Season {0} is currently set to ignore. skipping.", parseResult.SeasonNumber); return; } if (!_episodeProvider.IsNeeded(parseResult)) { Logger.Debug("Episode {0} is not needed. skipping.", parseResult); return; } //Should probably queue item to download } } protected IndexerSetting Settings { get { return _indexerProvider.GetSettings(GetType()); } } } }