Plex-Meta-Manager/modules/trakt.py

413 lines
22 KiB
Python
Raw Normal View History

import requests, webbrowser
2021-01-20 21:37:59 +00:00
from modules import util
from modules.util import Failed, TimeoutExpired
from ruamel import yaml
logger = util.logger
2021-01-20 21:37:59 +00:00
2021-07-15 17:42:28 +00:00
redirect_uri = "urn:ietf:wg:oauth:2.0:oob"
base_url = "https://api.trakt.tv"
2021-03-30 05:50:53 +00:00
builders = [
2022-03-27 06:26:08 +00:00
"trakt_list", "trakt_list_details", "trakt_chart", "trakt_userlist", "trakt_boxoffice", "trakt_recommendations",
"trakt_collected_daily", "trakt_collected_weekly", "trakt_collected_monthly", "trakt_collected_yearly", "trakt_collected_all",
2022-03-27 06:26:08 +00:00
"trakt_recommended_daily", "trakt_recommended_weekly", "trakt_recommended_monthly", "trakt_recommended_yearly", "trakt_recommended_all",
"trakt_watched_daily", "trakt_watched_weekly", "trakt_watched_monthly", "trakt_watched_yearly", "trakt_watched_all",
2022-03-27 06:26:08 +00:00
"trakt_collection", "trakt_popular", "trakt_trending", "trakt_watchlist"
2021-03-30 05:50:53 +00:00
]
2021-07-29 21:36:26 +00:00
sorts = [
"rank", "added", "title", "released", "runtime", "popularity",
"percentage", "votes", "random", "my_rating", "watched", "collected"
]
2022-03-28 16:15:34 +00:00
status = ["returning", "production", "planned", "canceled", "ended"]
status_translation = {
"returning": "returning series", "production": "in production",
"planned": "planned", "canceled": "canceled", "ended": "ended"
}
2022-03-27 06:26:08 +00:00
periods = ["daily", "weekly", "monthly", "yearly", "all"]
2022-01-31 08:37:49 +00:00
id_translation = {"movie": "movie", "show": "show", "season": "show", "episode": "show", "person": "person", "list": "list"}
id_types = {
"movie": ("tmdb", "TMDb ID"),
"person": ("tmdb", "TMDb ID"),
"show": ("tvdb", "TVDb ID"),
"season": ("tvdb", "TVDb ID"),
"episode": ("tvdb", "TVDb ID"),
"list": ("slug", "Trakt Slug")
}
2021-03-30 05:50:53 +00:00
2021-06-14 15:24:11 +00:00
class Trakt:
2021-07-26 19:03:17 +00:00
def __init__(self, config, params):
2021-07-14 14:47:20 +00:00
self.config = config
2021-01-20 21:37:59 +00:00
self.client_id = params["client_id"]
self.client_secret = params["client_secret"]
2022-03-24 14:10:27 +00:00
self.pin = params["pin"]
2021-01-20 21:37:59 +00:00
self.config_path = params["config_path"]
2021-07-26 19:03:17 +00:00
self.authorization = params["authorization"]
logger.secret(self.client_secret)
2021-05-07 19:53:54 +00:00
if not self._save(self.authorization):
if not self._refresh():
self._authorization()
2022-03-28 16:15:34 +00:00
self._movie_genres = None
self._show_genres = None
self._movie_languages = None
self._show_languages = None
self._movie_countries = None
self._show_countries = None
self._movie_certifications = None
self._show_certifications = None
@property
def movie_genres(self):
if not self._movie_genres:
self._movie_genres = [g["slug"] for g in self._request("/genres/movies")]
return self._movie_genres
@property
def show_genres(self):
if not self._show_genres:
self._show_genres = [g["slug"] for g in self._request("/genres/shows")]
return self._show_genres
@property
def movie_languages(self):
if not self._movie_languages:
self._movie_languages = [g["code"] for g in self._request("/languages/movies")]
return self._movie_languages
@property
def show_languages(self):
if not self._show_languages:
self._show_languages = [g["code"] for g in self._request("/languages/shows")]
return self._show_languages
@property
def movie_countries(self):
if not self._movie_countries:
self._movie_countries = [g["code"] for g in self._request("/countries/movies")]
return self._movie_countries
@property
def show_countries(self):
if not self._show_countries:
self._show_countries = [g["code"] for g in self._request("/countries/shows")]
return self._show_countries
@property
def movie_certifications(self):
if not self._movie_certifications:
self._movie_certifications = [g["slug"] for g in self._request("/certifications/movies")["us"]]
return self._movie_certifications
@property
def show_certifications(self):
if not self._show_certifications:
self._show_certifications = [g["slug"] for g in self._request("/certifications/shows")["us"]]
return self._show_certifications
2021-01-20 21:37:59 +00:00
2021-05-07 19:53:54 +00:00
def _authorization(self):
2022-03-24 14:10:27 +00:00
if self.pin:
pin = self.pin
else:
url = f"https://trakt.tv/oauth/authorize?response_type=code&redirect_uri={redirect_uri}&client_id={self.client_id}"
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: Trakt pin required.")
2021-07-15 17:42:28 +00:00
json = {
"code": pin,
"client_id": self.client_id,
"client_secret": self.client_secret,
"redirect_uri": redirect_uri,
"grant_type": "authorization_code"
}
response = self.config.post(f"{base_url}/oauth/token", json=json, headers={"Content-Type": "application/json"})
if response.status_code != 200:
2021-01-20 21:37:59 +00:00
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")
2021-07-15 17:42:28 +00:00
elif not self._save(response.json()):
2021-01-20 21:37:59 +00:00
raise Failed("Trakt Error: New Authorization Failed")
2021-07-15 17:42:28 +00:00
def _check(self, authorization=None):
token = self.authorization['access_token'] if authorization is None else authorization['access_token']
2021-07-15 17:42:28 +00:00
headers = {
"Content-Type": "application/json",
"Authorization": f"Bearer {token}",
2021-07-15 17:42:28 +00:00
"trakt-api-version": "2",
"trakt-api-key": self.client_id
}
logger.secret(token)
2021-07-15 17:42:28 +00:00
response = self.config.get(f"{base_url}/users/settings", headers=headers)
return response.status_code == 200
2021-01-20 21:37:59 +00:00
2021-05-07 19:53:54 +00:00
def _refresh(self):
2021-01-20 21:37:59 +00:00
if self.authorization and "refresh_token" in self.authorization and self.authorization["refresh_token"]:
logger.info("Refreshing Access Token...")
2021-07-15 17:42:28 +00:00
json = {
"refresh_token": self.authorization["refresh_token"],
"client_id": self.client_id,
"client_secret": self.client_secret,
"redirect_uri": redirect_uri,
"grant_type": "refresh_token"
}
response = self.config.post(f"{base_url}/oauth/token", json=json, headers={"Content-Type": "application/json"})
if response.status_code != 200:
return False
return self._save(response.json())
2021-01-20 21:37:59 +00:00
return False
2021-05-07 19:53:54 +00:00
def _save(self, authorization):
if authorization and self._check(authorization):
2021-12-28 00:01:43 +00:00
if self.authorization != authorization and not self.config.read_only:
2021-01-20 21:37:59 +00:00
yaml.YAML().allow_duplicate_keys = True
config, ind, bsi = yaml.util.load_yaml_guess_indent(open(self.config_path))
2022-03-24 14:10:27 +00:00
config["trakt"]["pin"] = None
2021-01-20 21:37:59 +00:00
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"]
}
2021-02-24 06:44:06 +00:00
logger.info(f"Saving authorization information to {self.config_path}")
2021-01-20 21:37:59 +00:00
yaml.round_trip_dump(config, open(self.config_path, "w"), indent=ind, block_seq_indent=bsi)
2021-07-15 17:42:28 +00:00
self.authorization = authorization
2021-01-20 21:37:59 +00:00
return True
return False
2022-03-28 16:15:34 +00:00
def _request(self, url, params=None):
2021-07-15 17:42:28 +00:00
headers = {
"Content-Type": "application/json",
"Authorization": f"Bearer {self.authorization['access_token']}",
"trakt-api-version": "2",
"trakt-api-key": self.client_id
}
output_json = []
2022-03-28 16:15:34 +00:00
if params is None:
params = {}
pages = 1
current = 1
2021-11-03 14:38:43 +00:00
if self.config.trace_mode:
logger.debug(f"URL: {base_url}{url}")
while current <= pages:
if pages == 1:
2022-03-28 16:15:34 +00:00
response = self.config.get(f"{base_url}{url}", headers=headers, params=params)
if "X-Pagination-Page-Count" in response.headers and not params:
pages = int(response.headers["X-Pagination-Page-Count"])
else:
2022-03-28 16:15:34 +00:00
params["page"] = current
response = self.config.get(f"{base_url}{url}", headers=headers, params=params)
if response.status_code == 200:
2021-08-03 13:09:00 +00:00
json_data = response.json()
2021-11-03 14:38:43 +00:00
if self.config.trace_mode:
logger.debug(f"Response: {json_data}")
2021-08-03 13:09:00 +00:00
if isinstance(json_data, dict):
return json_data
else:
output_json.extend(response.json())
else:
raise Failed(f"({response.status_code}) {response.reason}")
current += 1
return output_json
def user_ratings(self, is_movie):
media = "movie" if is_movie else "show"
id_type = "tmdb" if is_movie else "tvdb"
return {int(i[media]["ids"][id_type]): i["rating"] for i in self._request(f"/users/me/ratings/{media}s")}
2021-07-15 17:42:28 +00:00
2021-05-08 00:40:07 +00:00
def convert(self, external_id, from_source, to_source, media_type):
2021-07-15 17:42:28 +00:00
path = f"/search/{from_source}/{external_id}"
2022-03-28 16:15:34 +00:00
params = {"type": media_type} if from_source in ["tmdb", "tvdb"] else None
lookup = self._request(path, params=params)
2021-07-15 17:42:28 +00:00
if lookup and media_type in lookup[0] and to_source in lookup[0][media_type]["ids"]:
return lookup[0][media_type]["ids"][to_source]
2021-05-08 04:05:10 +00:00
raise Failed(f"Trakt Error: No {to_source.upper().replace('B', 'b')} ID found for {from_source.upper().replace('B', 'b')} ID: {external_id}")
2021-01-20 21:37:59 +00:00
2021-07-15 17:42:28 +00:00
def list_description(self, data):
try:
return self._request(requests.utils.urlparse(data).path)["description"]
2021-07-15 17:42:28 +00:00
except Failed:
raise Failed(f"Trakt Error: List {data} not found")
2021-03-30 05:49:10 +00:00
2021-08-22 15:54:33 +00:00
def _parse(self, items, typeless=False, item_type=None):
2021-08-04 13:42:21 +00:00
ids = []
for item in items:
2021-08-22 15:54:33 +00:00
if typeless:
2021-08-07 06:01:21 +00:00
data = item
2021-11-15 21:12:06 +00:00
current_type = item_type
2021-08-22 15:54:33 +00:00
elif item_type:
data = item[item_type]
current_type = item_type
elif "type" in item and item["type"] in id_translation:
data = item[id_translation[item["type"]]]
2021-08-22 15:54:33 +00:00
current_type = item["type"]
2021-08-07 06:01:21 +00:00
else:
2021-08-22 15:54:33 +00:00
continue
2022-01-31 08:37:49 +00:00
id_type, id_display = id_types[current_type]
2021-09-18 02:07:30 +00:00
if id_type in data["ids"] and data["ids"][id_type]:
2021-08-22 15:54:33 +00:00
final_id = data["ids"][id_type]
if current_type == "episode":
final_id = f"{final_id}_{item[current_type]['season']}"
if current_type in ["episode", "season"]:
final_id = f"{final_id}_{item[current_type]['number']}"
2022-01-31 08:37:49 +00:00
if current_type in ["person", "list"]:
final_id = (final_id, data["name"])
final_type = f"{id_type}_{current_type}" if current_type in ["episode", "season", "person"] else id_type
2021-08-22 15:54:33 +00:00
ids.append((final_id, final_type))
2021-08-04 13:42:21 +00:00
else:
2022-01-31 08:37:49 +00:00
name = data["name"] if current_type in ["person", "list"] else f"{data['title']} ({data['year']})"
logger.error(f"Trakt Error: No {id_display} found for {name}")
2021-08-07 06:01:21 +00:00
return ids
2021-08-04 13:42:21 +00:00
2022-03-27 06:26:08 +00:00
def all_user_lists(self, user):
try:
2022-03-27 06:26:08 +00:00
items = self._request(f"/users/{user}/lists")
except Failed:
2022-03-27 06:26:08 +00:00
raise Failed(f"Trakt Error: User {user} not found")
if len(items) == 0:
2022-03-27 06:26:08 +00:00
raise Failed(f"Trakt Error: User {user} has no lists")
return {self.build_user_url(user, i["ids"]["slug"]): i["name"] for i in items}
2022-01-31 08:37:49 +00:00
2022-03-27 06:26:08 +00:00
def all_liked_lists(self):
2022-01-31 08:37:49 +00:00
items = self._request(f"/users/likes/lists")
if len(items) == 0:
raise Failed(f"Trakt Error: No Liked lists found")
return {self.build_user_url(i['list']['user']['ids']['slug'], i['list']['ids']['slug']): i["list"]["name"] for i in items}
def build_user_url(self, user, name):
return f"{base_url.replace('api.', '')}/users/{user}/lists/{name}"
2022-03-27 06:26:08 +00:00
def _list(self, data):
2021-07-15 17:42:28 +00:00
try:
2021-08-07 06:01:21 +00:00
items = self._request(f"{requests.utils.urlparse(data).path}/items")
2021-07-15 17:42:28 +00:00
except Failed:
2021-08-07 06:01:21 +00:00
raise Failed(f"Trakt Error: List {data} not found")
2021-07-15 17:42:28 +00:00
if len(items) == 0:
2021-08-07 06:01:21 +00:00
raise Failed(f"Trakt Error: List {data} is empty")
return self._parse(items)
2022-03-27 06:26:08 +00:00
def _userlist(self, list_type, user, is_movie, sort_by=None):
2021-08-07 06:01:21 +00:00
try:
2022-03-27 06:26:08 +00:00
url_end = "movies" if is_movie else "shows"
if sort_by:
url_end = f"{url_end}/{sort_by}"
items = self._request(f"/users/{user}/{list_type}/{url_end}")
2021-08-07 06:01:21 +00:00
except Failed:
2022-03-27 06:26:08 +00:00
raise Failed(f"Trakt Error: User {user} not found")
2021-08-07 06:01:21 +00:00
if len(items) == 0:
2022-03-27 06:26:08 +00:00
raise Failed(f"Trakt Error: {user}'s {list_type.capitalize()} is empty")
2021-08-07 06:01:21 +00:00
return self._parse(items, item_type="movie" if is_movie else "show")
2021-03-27 07:30:07 +00:00
2022-03-27 06:26:08 +00:00
def _recommendations(self, limit, is_movie):
media_type = "Movie" if is_movie else "Show"
try:
2022-03-28 16:15:34 +00:00
items = self._request(f"/recommendations/{'movies' if is_movie else 'shows'}", params={"limit": limit})
except Failed:
raise Failed(f"Trakt Error: failed to fetch {media_type} Recommendations")
if len(items) == 0:
raise Failed(f"Trakt Error: no {media_type} Recommendations were found")
return self._parse(items, typeless=True, item_type="movie" if is_movie else "show")
2022-03-28 16:15:34 +00:00
def _charts(self, chart_type, is_movie, params, time_period=None):
2022-03-27 06:26:08 +00:00
chart_url = f"{chart_type}/{time_period}" if time_period else chart_type
2022-03-28 16:15:34 +00:00
items = self._request(f"/{'movies' if is_movie else 'shows'}/{chart_url}", params=params)
2022-03-27 06:26:08 +00:00
return self._parse(items, typeless=chart_type == "popular", item_type="movie" if is_movie else "show")
def get_people(self, data):
2022-03-27 06:26:08 +00:00
return {str(i[0][0]): i[0][1] for i in self._list(data) if i[1] == "tmdb_person"}
2022-03-27 06:26:08 +00:00
def validate_list(self, trakt_lists):
2021-07-29 21:36:26 +00:00
values = util.get_list(trakt_lists, split=False)
2021-01-20 21:37:59 +00:00
trakt_values = []
for value in values:
2021-12-02 06:39:46 +00:00
if isinstance(value, dict):
raise Failed("Trakt Error: List cannot be a dictionary")
2021-01-20 21:37:59 +00:00
try:
2022-03-27 06:26:08 +00:00
self._list(value)
2021-01-20 21:37:59 +00:00
trakt_values.append(value)
except Failed as e:
logger.error(e)
if len(trakt_values) == 0:
2022-03-27 06:26:08 +00:00
raise Failed(f"Trakt Error: No valid Trakt Lists in {values}")
2021-01-20 21:37:59 +00:00
return trakt_values
2022-03-28 16:15:34 +00:00
def validate_chart(self, err_type, method_name, data, is_movie):
2022-03-27 06:26:08 +00:00
valid_dicts = []
for trakt_dict in util.get_list(data, split=False):
if not isinstance(trakt_dict, dict):
raise Failed(f"{err_type} Error: {method_name} must be a dictionary")
dict_methods = {dm.lower(): dm for dm in trakt_dict}
try:
if method_name == "trakt_chart":
2022-03-28 16:15:34 +00:00
final_dict = {}
final_dict["chart"] = util.parse(err_type, "chart", trakt_dict, methods=dict_methods, parent=method_name, options=["recommended", "watched", "collected", "trending", "popular"])
final_dict["limit"] = util.parse(err_type, "limit", trakt_dict, methods=dict_methods, parent=method_name, datatype="int", default=10)
final_dict["time_period"] = None
if final_dict["chart"] in ["recommended", "watched", "collected"] and "time_period" in dict_methods:
final_dict["time_period"] = util.parse(err_type, "time_period", trakt_dict, methods=dict_methods, parent=method_name, default="weekly", options=periods)
if "query" in dict_methods:
final_dict["query"] = util.parse(err_type, "query", trakt_dict, methods=dict_methods, parent=method_name)
if "year" in dict_methods:
try:
if trakt_dict[dict_methods["year"]] and len(str(trakt_dict[dict_methods["year"]])) == 4:
final_dict["year"] = util.parse(err_type, "year", trakt_dict, methods=dict_methods, parent=method_name, datatype="int", minimum=1000, maximum=3000)
else:
final_dict["year"] = util.parse(err_type, "year", trakt_dict, methods=dict_methods, parent=method_name, datatype="int", minimum=1000, maximum=3000, range_split="-")
except Failed:
raise Failed(f"{err_type} Error: trakt_chart year attribute must be either a 4 digit year or a range of two 4 digit year with a '-' i.e. 1950 or 1950-1959")
if "runtimes" in dict_methods:
final_dict["runtimes"] = util.parse(err_type, "runtimes", trakt_dict, methods=dict_methods, parent=method_name, datatype="int", range_split="-")
if "ratings" in dict_methods:
final_dict["ratings"] = util.parse(err_type, "ratings", trakt_dict, methods=dict_methods, parent=method_name, datatype="int", minimum=0, maximum=100, range_split="-")
if "genres" in dict_methods:
final_dict["genres"] = util.parse(err_type, "genres", trakt_dict, methods=dict_methods, parent=method_name, datatype="list", options=self.movie_genres if is_movie else self.show_genres)
if "languages" in dict_methods:
final_dict["languages"] = util.parse(err_type, "languages", trakt_dict, methods=dict_methods, parent=method_name, datatype="list", options=self.movie_languages if is_movie else self.show_languages)
if "countries" in dict_methods:
final_dict["countries"] = util.parse(err_type, "countries", trakt_dict, methods=dict_methods, parent=method_name, datatype="list", options=self.movie_countries if is_movie else self.show_countries)
if "certifications" in dict_methods:
final_dict["certifications"] = util.parse(err_type, "certifications", trakt_dict, methods=dict_methods, parent=method_name, datatype="list", options=self.movie_certifications if is_movie else self.show_certifications)
if "networks" in dict_methods and not is_movie:
final_dict["networks"] = util.parse(err_type, "networks", trakt_dict, methods=dict_methods, parent=method_name, datatype="list")
if "status" in dict_methods and not is_movie:
final_dict["status"] = util.parse(err_type, "status", trakt_dict, methods=dict_methods, parent=method_name, datatype="list", options=status)
valid_dicts.append(final_dict)
2022-03-27 06:26:08 +00:00
else:
userlist = util.parse(err_type, "userlist", trakt_dict, methods=dict_methods, parent=method_name, options=["recommended", "watched", "collected", "watchlist"])
user = util.parse(err_type, "user", trakt_dict, methods=dict_methods, parent=method_name, default="me")
sort_by = None
if userlist in ["recommended", "watchlist"] and "sort" in dict_methods:
sort_by = util.parse(err_type, "sort_by", trakt_dict, methods=dict_methods, parent=method_name, default="rank", options=["rank", "added", "released", "title"])
self._userlist("collection" if userlist == "collected" else userlist, user, is_movie, sort_by=sort_by)
valid_dicts.append({"userlist": userlist, "user": user, "sort_by": sort_by})
except Failed as e:
logger.error(e)
if len(valid_dicts) == 0:
raise Failed(f"Trakt Error: No valid Trakt {method_name[6:].capitalize()}")
return valid_dicts
2021-08-07 06:01:21 +00:00
def get_trakt_ids(self, method, data, is_movie):
2021-08-01 04:35:42 +00:00
pretty = method.replace("_", " ").title()
2021-01-20 21:37:59 +00:00
media_type = "Movie" if is_movie else "Show"
2022-03-27 06:26:08 +00:00
if method == "trakt_list":
2021-05-09 05:37:45 +00:00
logger.info(f"Processing {pretty}: {data}")
2022-03-27 06:26:08 +00:00
return self._list(data)
elif method == "trakt_recommendations":
2021-11-13 23:51:12 +00:00
logger.info(f"Processing {pretty}: {data} {media_type}{'' if data == 1 else 's'}")
2022-03-27 06:26:08 +00:00
return self._recommendations(data, is_movie)
elif method == "trakt_chart":
2022-03-28 16:15:34 +00:00
params = {"limit": data["limit"]}
chart_limit = f"{data['limit']} {data['time_period'].capitalize()}" if data["time_period"] else data["limit"]
logger.info(f"Processing {pretty}: {chart_limit} {data['chart'].capitalize()} {media_type}{'' if data == 1 else 's'}")
for attr in ["query", "year", "runtimes", "ratings", "genres", "languages", "countries", "certifications", "networks", "status"]:
if attr in data:
logger.info(f"{attr:>22}: {','.join(data[attr]) if isinstance(data[attr], list) else data[attr]}")
values = [status_translation[v] for v in data[attr]] if attr == "status" else data[attr]
params[attr] = ",".join(values) if isinstance(values, list) else values
return self._charts(data["chart"], is_movie, params, time_period=data["time_period"])
2022-03-27 06:26:08 +00:00
elif method == "trakt_userlist":
logger.info(f"Processing {pretty} {media_type}s from {data['user']}'s {data['userlist'].capitalize()}")
return self._userlist(data["userlist"], data["user"], is_movie, sort_by=data["sort_by"])
2021-07-15 17:42:28 +00:00
else:
raise Failed(f"Trakt Error: Method {method} not supported")