using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net; using System.Text.RegularExpressions; using NLog; using NzbDrone.Common; using NzbDrone.Core.MediaCover; using NzbDrone.Core.MetadataSource.Trakt; using NzbDrone.Core.Tv; using RestSharp; using Episode = NzbDrone.Core.Tv.Episode; using NzbDrone.Core.Rest; namespace NzbDrone.Core.MetadataSource { public class TraktProxy : ISearchForNewSeries, IProvideSeriesInfo { private readonly Logger _logger; private static readonly Regex InvalidSearchCharRegex = new Regex(@"[^a-zA-Z0-9\s-\.]", RegexOptions.Compiled); public TraktProxy(Logger logger) { _logger = logger; } public List SearchForNewSeries(string title) { try { var client = BuildClient("search", "shows"); var restRequest = new RestRequest(GetSearchTerm(title) +"/30/seasons"); var response = client.ExecuteAndValidate>(restRequest); return response.Select(MapSeries).ToList(); } catch (WebException ex) { throw new TraktException("Search for '{0}' failed. Unable to communicate with Trakt.", title); } catch (Exception ex) { _logger.WarnException(ex.Message, ex); throw new TraktException("Search for '{0}' failed. Invalid response received from Trakt.", title); } } public Tuple> GetSeriesInfo(int tvDbSeriesId) { var client = BuildClient("show", "summary"); var restRequest = new RestRequest(tvDbSeriesId.ToString() + "/extended"); var response = client.ExecuteAndValidate(restRequest); var episodes = response.seasons.SelectMany(c => c.episodes).Select(MapEpisode).ToList(); var series = MapSeries(response); return new Tuple>(series, episodes); } private static IRestClient BuildClient(string resource, string method) { return new RestClient(string.Format("http://api.trakt.tv/{0}/{1}.json/bc3c2c460f22cbb01c264022b540e191", resource, method)); } private static Series MapSeries(Show show) { var series = new Series(); series.TvdbId = show.tvdb_id; series.TvRageId = show.tvrage_id; series.ImdbId = show.imdb_id; series.Title = show.title; series.CleanTitle = Parser.Parser.CleanSeriesTitle(show.title); series.Year = show.year; series.FirstAired = FromIso(show.first_aired_iso); series.Overview = show.overview; series.Runtime = show.runtime; series.Network = show.network; series.AirTime = show.air_time_utc; series.TitleSlug = show.url.ToLower().Replace("http://trakt.tv/show/", ""); series.Status = GetSeriesStatus(show.status); series.Seasons = show.seasons.Select(s => new Tv.Season { SeasonNumber = s.season }).OrderByDescending(s => s.SeasonNumber).ToList(); series.Images.Add(new MediaCover.MediaCover { CoverType = MediaCoverTypes.Banner, Url = show.images.banner }); series.Images.Add(new MediaCover.MediaCover { CoverType = MediaCoverTypes.Poster, Url = GetPosterThumbnailUrl(show.images.poster) }); series.Images.Add(new MediaCover.MediaCover { CoverType = MediaCoverTypes.Fanart, Url = show.images.fanart }); return series; } private static Series MapSearchSeries(SearchShow show) { var series = new Series(); series.TvdbId = show.tvdb_id; series.TvRageId = show.tvrage_id; series.ImdbId = show.imdb_id; series.Title = show.title; series.CleanTitle = Parser.Parser.CleanSeriesTitle(show.title); series.FirstAired = FromIso(show.first_aired_iso); series.Overview = show.overview; series.Runtime = show.runtime; series.Network = show.network; series.AirTime = show.air_time_utc; series.TitleSlug = show.url.ToLower().Replace("http://trakt.tv/show/", ""); series.Status = GetSeriesStatus(show.status); series.Images.Add(new MediaCover.MediaCover { CoverType = MediaCoverTypes.Banner, Url = show.images.banner }); series.Images.Add(new MediaCover.MediaCover { CoverType = MediaCoverTypes.Poster, Url = GetPosterThumbnailUrl(show.images.poster) }); series.Images.Add(new MediaCover.MediaCover { CoverType = MediaCoverTypes.Fanart, Url = show.images.fanart }); return series; } private static Episode MapEpisode(Trakt.Episode traktEpisode) { var episode = new Episode(); episode.Overview = traktEpisode.overview; episode.SeasonNumber = traktEpisode.season; episode.EpisodeNumber = traktEpisode.episode; episode.EpisodeNumber = traktEpisode.number; episode.TvDbEpisodeId = traktEpisode.tvdb_id; episode.Title = traktEpisode.title; episode.AirDate = FromIsoToString(traktEpisode.first_aired_iso); episode.AirDateUtc = FromIso(traktEpisode.first_aired_iso); return episode; } private static string GetPosterThumbnailUrl(string posterUrl) { if (posterUrl.Contains("poster-small.jpg")) return posterUrl; var extension = Path.GetExtension(posterUrl); var withoutExtension = posterUrl.Substring(0, posterUrl.Length - extension.Length); return withoutExtension + "-300" + extension; } private static SeriesStatusType GetSeriesStatus(string status) { if (string.IsNullOrWhiteSpace(status)) return SeriesStatusType.Continuing; if (status.Equals("Ended", StringComparison.InvariantCultureIgnoreCase)) return SeriesStatusType.Ended; return SeriesStatusType.Continuing; } private static DateTime? FromEpoch(long ticks) { if (ticks == 0) return null; return new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Unspecified).AddSeconds(ticks); } private static DateTime? FromIso(string iso) { DateTime result; //Todo: Remove this when DST ends and/or trakt fixes DST airings in EST/EDT if (iso != null && iso.EndsWith("-05:00")) { iso = iso.Replace("-05:00", "-04:00"); } if (!DateTime.TryParse(iso, out result)) return null; return result.ToUniversalTime(); } private static string FromIsoToString(string iso) { if (String.IsNullOrWhiteSpace(iso)) return null; var match = Regex.Match(iso, @"^\d{4}\W\d{2}\W\d{2}"); if (!match.Success) return null; return match.Captures[0].Value; } private static string GetSearchTerm(string phrase) { phrase = phrase.RemoveAccent().ToLower(); phrase = phrase.Replace("&", "and"); phrase = InvalidSearchCharRegex.Replace(phrase, string.Empty); phrase = phrase.CleanSpaces().Replace(" ", "+"); return phrase; } } }