|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Net;
|
|
|
|
|
using NLog;
|
|
|
|
|
using NzbDrone.Common.Cloud;
|
|
|
|
|
using NzbDrone.Common.Extensions;
|
|
|
|
|
using NzbDrone.Common.Http;
|
|
|
|
|
using NzbDrone.Core.Exceptions;
|
|
|
|
|
using NzbDrone.Core.MediaCover;
|
|
|
|
|
using NzbDrone.Core.MetadataSource.SkyHook.Resource;
|
|
|
|
|
using NzbDrone.Core.Tv;
|
|
|
|
|
using Newtonsoft.Json;
|
|
|
|
|
|
|
|
|
|
namespace NzbDrone.Core.MetadataSource.SkyHook
|
|
|
|
|
{
|
|
|
|
|
public class SkyHookProxy : IProvideSeriesInfo, ISearchForNewSeries, IProvideMovieInfo, ISearchForNewMovie
|
|
|
|
|
{
|
|
|
|
|
private readonly IHttpClient _httpClient;
|
|
|
|
|
private readonly Logger _logger;
|
|
|
|
|
|
|
|
|
|
private readonly IHttpRequestBuilderFactory _requestBuilder;
|
|
|
|
|
|
|
|
|
|
public SkyHookProxy(IHttpClient httpClient, ISonarrCloudRequestBuilder requestBuilder, Logger logger)
|
|
|
|
|
{
|
|
|
|
|
_httpClient = httpClient;
|
|
|
|
|
_requestBuilder = requestBuilder.SkyHookTvdb;
|
|
|
|
|
_logger = logger;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Tuple<Series, List<Episode>> GetSeriesInfo(int tvdbSeriesId)
|
|
|
|
|
{
|
|
|
|
|
var httpRequest = _requestBuilder.Create()
|
|
|
|
|
.SetSegment("route", "shows")
|
|
|
|
|
.Resource(tvdbSeriesId.ToString())
|
|
|
|
|
.Build();
|
|
|
|
|
|
|
|
|
|
httpRequest.AllowAutoRedirect = true;
|
|
|
|
|
httpRequest.SuppressHttpError = true;
|
|
|
|
|
|
|
|
|
|
var httpResponse = _httpClient.Get<ShowResource>(httpRequest);
|
|
|
|
|
|
|
|
|
|
if (httpResponse.HasHttpError)
|
|
|
|
|
{
|
|
|
|
|
if (httpResponse.StatusCode == HttpStatusCode.NotFound)
|
|
|
|
|
{
|
|
|
|
|
throw new SeriesNotFoundException(tvdbSeriesId);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
throw new HttpException(httpRequest, httpResponse);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var episodes = httpResponse.Resource.Episodes.Select(MapEpisode);
|
|
|
|
|
var series = MapSeries(httpResponse.Resource);
|
|
|
|
|
|
|
|
|
|
return new Tuple<Series, List<Episode>>(series, episodes.ToList());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Movie GetMovieInfo(string ImdbId)
|
|
|
|
|
{
|
|
|
|
|
var imdbRequest = new HttpRequest("http://www.omdbapi.com/?i=" + ImdbId + "&plot=full&r=json");
|
|
|
|
|
|
|
|
|
|
var httpResponse = _httpClient.Get(imdbRequest);
|
|
|
|
|
|
|
|
|
|
if (httpResponse.HasHttpError)
|
|
|
|
|
{
|
|
|
|
|
if (httpResponse.StatusCode == HttpStatusCode.NotFound)
|
|
|
|
|
{
|
|
|
|
|
throw new MovieNotFoundException(ImdbId);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
throw new HttpException(imdbRequest, httpResponse);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var response = httpResponse.Content;
|
|
|
|
|
|
|
|
|
|
dynamic json = JsonConvert.DeserializeObject(response);
|
|
|
|
|
|
|
|
|
|
var movie = new Movie();
|
|
|
|
|
|
|
|
|
|
movie.Title = json.Title;
|
|
|
|
|
movie.TitleSlug = movie.Title.ToLower().Replace(" ", "-");
|
|
|
|
|
movie.Overview = json.Plot;
|
|
|
|
|
movie.CleanTitle = Parser.Parser.CleanSeriesTitle(movie.Title);
|
|
|
|
|
string airDateStr = json.Released;
|
|
|
|
|
DateTime airDate = DateTime.Parse(airDateStr);
|
|
|
|
|
movie.InCinemas = airDate;
|
|
|
|
|
movie.Year = airDate.Year;
|
|
|
|
|
movie.ImdbId = ImdbId;
|
|
|
|
|
string imdbRating = json.imdbVotes;
|
|
|
|
|
if (imdbRating == "N/A")
|
|
|
|
|
{
|
|
|
|
|
movie.Status = MovieStatusType.Announced;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
movie.Status = MovieStatusType.Released;
|
|
|
|
|
}
|
|
|
|
|
string url = json.Poster;
|
|
|
|
|
var imdbPoster = new MediaCover.MediaCover(MediaCoverTypes.Poster, url);
|
|
|
|
|
movie.Images.Add(imdbPoster);
|
|
|
|
|
string runtime = json.Runtime;
|
|
|
|
|
int runtimeNum = 0;
|
|
|
|
|
int.TryParse(runtime.Replace("min", "").Trim(), out runtimeNum);
|
|
|
|
|
movie.Runtime = runtimeNum;
|
|
|
|
|
|
|
|
|
|
return movie;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public List<Movie> SearchForNewMovie(string title)
|
|
|
|
|
{
|
|
|
|
|
var lowerTitle = title.ToLower();
|
|
|
|
|
|
|
|
|
|
if (lowerTitle.StartsWith("imdb:") || lowerTitle.StartsWith("imdbid:"))
|
|
|
|
|
{
|
|
|
|
|
var slug = lowerTitle.Split(':')[1].Trim();
|
|
|
|
|
|
|
|
|
|
string imdbid = slug;
|
|
|
|
|
|
|
|
|
|
if (slug.IsNullOrWhiteSpace() || slug.Any(char.IsWhiteSpace))
|
|
|
|
|
{
|
|
|
|
|
return new List<Movie>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
return new List<Movie> { GetMovieInfo(imdbid) };
|
|
|
|
|
}
|
|
|
|
|
catch (SeriesNotFoundException)
|
|
|
|
|
{
|
|
|
|
|
return new List<Movie>();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var searchTerm = lowerTitle.Replace("+", "_").Replace(" ", "_");
|
|
|
|
|
|
|
|
|
|
var firstChar = searchTerm.First();
|
|
|
|
|
|
|
|
|
|
var imdbRequest = new HttpRequest("https://v2.sg.media-imdb.com/suggests/" + firstChar + "/" + searchTerm + ".json");
|
|
|
|
|
|
|
|
|
|
var response = _httpClient.Get(imdbRequest);
|
|
|
|
|
|
|
|
|
|
var imdbCallback = "imdb$" + searchTerm + "(";
|
|
|
|
|
|
|
|
|
|
var responseCleaned = response.Content.Replace(imdbCallback, "").TrimEnd(")");
|
|
|
|
|
|
|
|
|
|
dynamic json = JsonConvert.DeserializeObject(responseCleaned);
|
|
|
|
|
|
|
|
|
|
var imdbMovies = new List<Movie>();
|
|
|
|
|
|
|
|
|
|
foreach (dynamic entry in json.d)
|
|
|
|
|
{
|
|
|
|
|
var imdbMovie = new Movie();
|
|
|
|
|
imdbMovie.ImdbId = entry.id;
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
imdbMovie.SortTitle = entry.l;
|
|
|
|
|
imdbMovie.Title = entry.l;
|
|
|
|
|
string titleSlug = entry.l;
|
|
|
|
|
imdbMovie.TitleSlug = titleSlug.ToLower().Replace(" ", "-");
|
|
|
|
|
imdbMovie.Year = entry.y;
|
|
|
|
|
imdbMovie.Images = new List<MediaCover.MediaCover>();
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
string url = entry.i[0];
|
|
|
|
|
var imdbPoster = new MediaCover.MediaCover(MediaCoverTypes.Poster, url);
|
|
|
|
|
imdbMovie.Images.Add(imdbPoster);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
_logger.Debug(entry);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
imdbMovies.Add(imdbMovie);
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return imdbMovies;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public List<Series> SearchForNewSeries(string title)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var lowerTitle = title.ToLowerInvariant();
|
|
|
|
|
|
|
|
|
|
if (lowerTitle.StartsWith("tvdb:") || lowerTitle.StartsWith("tvdbid:"))
|
|
|
|
|
{
|
|
|
|
|
var slug = lowerTitle.Split(':')[1].Trim();
|
|
|
|
|
|
|
|
|
|
int tvdbId;
|
|
|
|
|
|
|
|
|
|
if (slug.IsNullOrWhiteSpace() || slug.Any(char.IsWhiteSpace) || !int.TryParse(slug, out tvdbId) || tvdbId <= 0)
|
|
|
|
|
{
|
|
|
|
|
return new List<Series>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
return new List<Series> { GetSeriesInfo(tvdbId).Item1 };
|
|
|
|
|
}
|
|
|
|
|
catch (SeriesNotFoundException)
|
|
|
|
|
{
|
|
|
|
|
return new List<Series>();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var httpRequest = _requestBuilder.Create()
|
|
|
|
|
.SetSegment("route", "search")
|
|
|
|
|
.AddQueryParam("term", title.ToLower().Trim())
|
|
|
|
|
.Build();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var httpResponse = _httpClient.Get<List<ShowResource>>(httpRequest);
|
|
|
|
|
|
|
|
|
|
return httpResponse.Resource.SelectList(MapSeries);
|
|
|
|
|
}
|
|
|
|
|
catch (HttpException)
|
|
|
|
|
{
|
|
|
|
|
throw new SkyHookException("Search for '{0}' failed. Unable to communicate with SkyHook.", title);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.Warn(ex, ex.Message);
|
|
|
|
|
throw new SkyHookException("Search for '{0}' failed. Invalid response received from SkyHook.", title);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static Series MapSeries(ShowResource show)
|
|
|
|
|
{
|
|
|
|
|
var series = new Series();
|
|
|
|
|
series.TvdbId = show.TvdbId;
|
|
|
|
|
|
|
|
|
|
if (show.TvRageId.HasValue)
|
|
|
|
|
{
|
|
|
|
|
series.TvRageId = show.TvRageId.Value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (show.TvMazeId.HasValue)
|
|
|
|
|
{
|
|
|
|
|
series.TvMazeId = show.TvMazeId.Value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
series.ImdbId = show.ImdbId;
|
|
|
|
|
series.Title = show.Title;
|
|
|
|
|
series.CleanTitle = Parser.Parser.CleanSeriesTitle(show.Title);
|
|
|
|
|
series.SortTitle = SeriesTitleNormalizer.Normalize(show.Title, show.TvdbId);
|
|
|
|
|
|
|
|
|
|
if (show.FirstAired != null)
|
|
|
|
|
{
|
|
|
|
|
series.FirstAired = DateTime.Parse(show.FirstAired).ToUniversalTime();
|
|
|
|
|
series.Year = series.FirstAired.Value.Year;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
series.Overview = show.Overview;
|
|
|
|
|
|
|
|
|
|
if (show.Runtime != null)
|
|
|
|
|
{
|
|
|
|
|
series.Runtime = show.Runtime.Value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
series.Network = show.Network;
|
|
|
|
|
|
|
|
|
|
if (show.TimeOfDay != null)
|
|
|
|
|
{
|
|
|
|
|
series.AirTime = string.Format("{0:00}:{1:00}", show.TimeOfDay.Hours, show.TimeOfDay.Minutes);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
series.TitleSlug = show.Slug;
|
|
|
|
|
series.Status = MapSeriesStatus(show.Status);
|
|
|
|
|
series.Ratings = MapRatings(show.Rating);
|
|
|
|
|
series.Genres = show.Genres;
|
|
|
|
|
|
|
|
|
|
if (show.ContentRating.IsNotNullOrWhiteSpace())
|
|
|
|
|
{
|
|
|
|
|
series.Certification = show.ContentRating.ToUpper();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
series.Actors = show.Actors.Select(MapActors).ToList();
|
|
|
|
|
series.Seasons = show.Seasons.Select(MapSeason).ToList();
|
|
|
|
|
series.Images = show.Images.Select(MapImage).ToList();
|
|
|
|
|
|
|
|
|
|
return series;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static Actor MapActors(ActorResource arg)
|
|
|
|
|
{
|
|
|
|
|
var newActor = new Actor
|
|
|
|
|
{
|
|
|
|
|
Name = arg.Name,
|
|
|
|
|
Character = arg.Character
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (arg.Image != null)
|
|
|
|
|
{
|
|
|
|
|
newActor.Images = new List<MediaCover.MediaCover>
|
|
|
|
|
{
|
|
|
|
|
new MediaCover.MediaCover(MediaCoverTypes.Headshot, arg.Image)
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return newActor;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static Episode MapEpisode(EpisodeResource oracleEpisode)
|
|
|
|
|
{
|
|
|
|
|
var episode = new Episode();
|
|
|
|
|
episode.Overview = oracleEpisode.Overview;
|
|
|
|
|
episode.SeasonNumber = oracleEpisode.SeasonNumber;
|
|
|
|
|
episode.EpisodeNumber = oracleEpisode.EpisodeNumber;
|
|
|
|
|
episode.AbsoluteEpisodeNumber = oracleEpisode.AbsoluteEpisodeNumber;
|
|
|
|
|
episode.Title = oracleEpisode.Title;
|
|
|
|
|
|
|
|
|
|
episode.AirDate = oracleEpisode.AirDate;
|
|
|
|
|
episode.AirDateUtc = oracleEpisode.AirDateUtc;
|
|
|
|
|
|
|
|
|
|
episode.Ratings = MapRatings(oracleEpisode.Rating);
|
|
|
|
|
|
|
|
|
|
//Don't include series fanart images as episode screenshot
|
|
|
|
|
if (oracleEpisode.Image != null)
|
|
|
|
|
{
|
|
|
|
|
episode.Images.Add(new MediaCover.MediaCover(MediaCoverTypes.Screenshot, oracleEpisode.Image));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return episode;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static Season MapSeason(SeasonResource seasonResource)
|
|
|
|
|
{
|
|
|
|
|
return new Season
|
|
|
|
|
{
|
|
|
|
|
SeasonNumber = seasonResource.SeasonNumber,
|
|
|
|
|
Images = seasonResource.Images.Select(MapImage).ToList()
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static SeriesStatusType MapSeriesStatus(string status)
|
|
|
|
|
{
|
|
|
|
|
if (status.Equals("ended", StringComparison.InvariantCultureIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
return SeriesStatusType.Ended;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return SeriesStatusType.Continuing;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static Ratings MapRatings(RatingResource rating)
|
|
|
|
|
{
|
|
|
|
|
if (rating == null)
|
|
|
|
|
{
|
|
|
|
|
return new Ratings();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return new Ratings
|
|
|
|
|
{
|
|
|
|
|
Votes = rating.Count,
|
|
|
|
|
Value = rating.Value
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static MediaCover.MediaCover MapImage(ImageResource arg)
|
|
|
|
|
{
|
|
|
|
|
return new MediaCover.MediaCover
|
|
|
|
|
{
|
|
|
|
|
Url = arg.Url,
|
|
|
|
|
CoverType = MapCoverType(arg.CoverType)
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static MediaCoverTypes MapCoverType(string coverType)
|
|
|
|
|
{
|
|
|
|
|
switch (coverType.ToLower())
|
|
|
|
|
{
|
|
|
|
|
case "poster":
|
|
|
|
|
return MediaCoverTypes.Poster;
|
|
|
|
|
case "banner":
|
|
|
|
|
return MediaCoverTypes.Banner;
|
|
|
|
|
case "fanart":
|
|
|
|
|
return MediaCoverTypes.Fanart;
|
|
|
|
|
default:
|
|
|
|
|
return MediaCoverTypes.Unknown;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|