import logging, requests, webbrowser from modules import util from modules.util import Failed, TimeoutExpired from retrying import retry from ruamel import yaml from trakt import Trakt from trakt.objects.episode import Episode from trakt.objects.movie import Movie from trakt.objects.season import Season from trakt.objects.show import Show logger = logging.getLogger("Plex Meta Manager") class TraktAPI: def __init__(self, params, authorization=None): self.base_url = "https://api.trakt.tv" self.redirect_uri = "urn:ietf:wg:oauth:2.0:oob" self.aliases = { "trakt_trending": "Trakt Trending", "trakt_watchlist": "Trakt Watchlist", "trakt_list": "Trakt List" } self.client_id = params["client_id"] self.client_secret = params["client_secret"] self.config_path = params["config_path"] self.authorization = authorization Trakt.configuration.defaults.client(self.client_id, self.client_secret) if not self.save_authorization(self.authorization): if not self.refresh_authorization(): self.get_authorization() def get_authorization(self): url = Trakt["oauth"].authorize_url(self.redirect_uri) logger.info(f"Navigate to: {url}") logger.info("If you get an OAuth error your client_id or client_secret is invalid") webbrowser.open(url, new=2) try: pin = util.logger_input("Trakt pin (case insensitive)", timeout=300).strip() except TimeoutExpired: raise Failed("Input Timeout: Trakt pin required.") if not pin: raise Failed("Trakt Error: No input Trakt pin required.") new_authorization = Trakt["oauth"].token(pin, self.redirect_uri) if not new_authorization: raise Failed("Trakt Error: Invalid trakt pin. If you're sure you typed it in correctly your client_id or client_secret may be invalid") if not self.save_authorization(new_authorization): raise Failed("Trakt Error: New Authorization Failed") def check_authorization(self, authorization): try: with Trakt.configuration.oauth.from_response(authorization, refresh=True): if Trakt["users/settings"].get(): return True except ValueError: pass return False def refresh_authorization(self): if self.authorization and "refresh_token" in self.authorization and self.authorization["refresh_token"]: logger.info("Refreshing Access Token...") refreshed_authorization = Trakt["oauth"].token_refresh(self.authorization["refresh_token"], self.redirect_uri) return self.save_authorization(refreshed_authorization) return False def save_authorization(self, authorization): if authorization and self.check_authorization(authorization): if self.authorization != authorization: yaml.YAML().allow_duplicate_keys = True config, ind, bsi = yaml.util.load_yaml_guess_indent(open(self.config_path)) config["trakt"]["authorization"] = { "access_token": authorization["access_token"], "token_type": authorization["token_type"], "expires_in": authorization["expires_in"], "refresh_token": authorization["refresh_token"], "scope": authorization["scope"], "created_at": authorization["created_at"] } logger.info(f"Saving authorization information to {self.config_path}") yaml.round_trip_dump(config, open(self.config_path, "w"), indent=ind, block_seq_indent=bsi) self.authorization = authorization Trakt.configuration.defaults.oauth.from_response(self.authorization) return True return False def convert_tmdb_to_imdb(self, tmdb_id, is_movie=True): return self.convert_id(tmdb_id, "tmdb", "imdb", "movie" if is_movie else "show") def convert_imdb_to_tmdb(self, imdb_id, is_movie=True): return self.convert_id(imdb_id, "imdb", "tmdb", "movie" if is_movie else "show") def convert_tmdb_to_tvdb(self, tmdb_id): return self.convert_id(tmdb_id, "tmdb", "tvdb", "show") def convert_tvdb_to_tmdb(self, tvdb_id): return self.convert_id(tvdb_id, "tvdb", "tmdb", "show") def convert_tvdb_to_imdb(self, tvdb_id): return self.convert_id(tvdb_id, "tvdb", "imdb", "show") def convert_imdb_to_tvdb(self, imdb_id): return self.convert_id(imdb_id, "imdb", "tvdb", "show") @retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed) def convert_id(self, external_id, from_source, to_source, media_type): lookup = Trakt["search"].lookup(external_id, from_source, media_type) if lookup: lookup = lookup[0] if isinstance(lookup, list) else lookup if lookup.get_key(to_source): return lookup.get_key(to_source) if to_source == "imdb" else int(lookup.get_key(to_source)) raise Failed(f"No {to_source.upper().replace('B', 'b')} ID found for {from_source.upper().replace('B', 'b')} ID {external_id}") @retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed) def watchlist(self, data, is_movie): items = Trakt[f"users/{data}/watchlist"].movies() if is_movie else Trakt[f"users/{data}/watchlist"].shows() if items is None: raise Failed("Trakt Error: No List found") else: return [i for i in items] @retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed) def standard_list(self, data): try: trakt_list = Trakt[requests.utils.urlparse(data).path].get() except AttributeError: trakt_list = None if trakt_list is None: raise Failed("Trakt Error: No List found") else: return trakt_list @retry(stop_max_attempt_number=6, wait_fixed=10000) def send_request(self, url): return requests.get(url, headers={"Content-Type": "application/json", "trakt-api-version": "2", "trakt-api-key": self.client_id}).json() def get_pagenation(self, pagenation, amount, is_movie): items = self.send_request(f"{self.base_url}/{'movies' if is_movie else 'shows'}/{pagenation}?limit={amount}") if is_movie: return [item["ids"]["tmdb" if is_movie else "tvdb"] for item in items], [] else: return [], [item["ids"]["tmdb" if is_movie else "tvdb"] for item in items] def validate_trakt_list(self, values): trakt_values = [] for value in values: try: self.standard_list(value) trakt_values.append(value) except Failed as e: logger.error(e) if len(trakt_values) == 0: raise Failed(f"Trakt Error: No valid Trakt Lists in {values}") return trakt_values def validate_trakt_watchlist(self, values, is_movie): trakt_values = [] for value in values: try: self.watchlist(value, is_movie) trakt_values.append(value) except Failed as e: logger.error(e) if len(trakt_values) == 0: raise Failed(f"Trakt Error: No valid Trakt Watchlists in {values}") return trakt_values def get_items(self, method, data, is_movie, status_message=True): if status_message: logger.debug(f"Data: {data}") pretty = self.aliases[method] if method in self.aliases else method media_type = "Movie" if is_movie else "Show" if method in ["trakt_trending", "trakt_popular", "trakt_recommended", "trakt_played", "trakt_watched", "trakt_collected"]: movie_ids, show_ids = self.get_pagenation(method[6:], data, is_movie) if status_message: logger.info(f"Processing {pretty}: {data} {media_type}{'' if data == 1 else 's'}") else: show_ids = [] movie_ids = [] if method == "trakt_watchlist": trakt_items = self.watchlist(data, is_movie) elif method == "trakt_list": trakt_items = self.standard_list(data).items() else: raise Failed(f"Trakt Error: Method {method} not supported") if status_message: logger.info(f"Processing {pretty}: {data}") for trakt_item in trakt_items: if isinstance(trakt_item, Movie): movie_ids.append(int(trakt_item.get_key("tmdb"))) elif isinstance(trakt_item, Show) and trakt_item.pk[1] not in show_ids: show_ids.append(int(trakt_item.pk[1])) elif (isinstance(trakt_item, (Season, Episode))) and trakt_item.show.pk[1] not in show_ids: show_ids.append(int(trakt_item.show.pk[1])) if status_message: logger.debug(f"Trakt {media_type} Found: {trakt_items}") if status_message: logger.debug(f"TMDb IDs Found: {movie_ids}") logger.debug(f"TVDb IDs Found: {show_ids}") return movie_ids, show_ids