2022-10-24 06:17:49 +00:00
|
|
|
import re
|
2021-01-20 21:37:59 +00:00
|
|
|
from modules import util
|
|
|
|
from modules.util import Failed
|
2022-05-05 19:22:43 +00:00
|
|
|
from retrying import retry
|
2022-04-26 03:10:13 +00:00
|
|
|
from tmdbapis import TMDbAPIs, TMDbException, NotFound, Movie
|
2021-01-20 21:37:59 +00:00
|
|
|
|
2022-02-13 16:33:57 +00:00
|
|
|
logger = util.logger
|
2021-01-20 21:37:59 +00:00
|
|
|
|
2022-03-22 18:16:30 +00:00
|
|
|
int_builders = [
|
|
|
|
"tmdb_airing_today", "tmdb_popular", "tmdb_top_rated", "tmdb_now_playing", "tmdb_on_the_air",
|
|
|
|
"tmdb_trending_daily", "tmdb_trending_weekly", "tmdb_upcoming"
|
2021-03-30 05:50:53 +00:00
|
|
|
]
|
2022-03-22 18:16:30 +00:00
|
|
|
info_builders = [
|
|
|
|
"tmdb_actor", "tmdb_collection", "tmdb_crew", "tmdb_director", "tmdb_list",
|
|
|
|
"tmdb_movie", "tmdb_producer", "tmdb_show", "tmdb_writer"
|
|
|
|
]
|
|
|
|
details_builders = [f"{d}_details" for d in info_builders]
|
|
|
|
builders = ["tmdb_company", "tmdb_discover", "tmdb_keyword", "tmdb_network"] \
|
|
|
|
+ int_builders + info_builders + details_builders
|
2021-03-30 05:50:53 +00:00
|
|
|
type_map = {
|
2021-07-23 19:44:21 +00:00
|
|
|
"tmdb_actor": "Person", "tmdb_actor_details": "Person", "tmdb_crew": "Person", "tmdb_crew_details": "Person",
|
|
|
|
"tmdb_collection": "Collection", "tmdb_collection_details": "Collection", "tmdb_company": "Company",
|
|
|
|
"tmdb_director": "Person", "tmdb_director_details": "Person", "tmdb_keyword": "Keyword",
|
|
|
|
"tmdb_list": "List", "tmdb_list_details": "List", "tmdb_movie": "Movie", "tmdb_movie_details": "Movie",
|
|
|
|
"tmdb_network": "Network", "tmdb_person": "Person", "tmdb_producer": "Person", "tmdb_producer_details": "Person",
|
|
|
|
"tmdb_show": "Show", "tmdb_show_details": "Show", "tmdb_writer": "Person", "tmdb_writer_details": "Person"
|
2021-03-30 05:50:53 +00:00
|
|
|
}
|
2021-08-14 22:59:35 +00:00
|
|
|
discover_movie_only = [
|
2022-02-19 16:26:45 +00:00
|
|
|
"region", "with_cast", "with_crew", "with_people", "certification_country", "certification", "include_video", "year",
|
|
|
|
"primary_release_year", "primary_release_date", "release_date", "include_adult", "with_release_type", "with_title_translation"
|
2021-08-14 22:59:35 +00:00
|
|
|
]
|
|
|
|
discover_tv_only = [
|
2022-02-19 16:26:45 +00:00
|
|
|
"timezone", "screened_theatrically", "include_null_first_air_dates", "air_date", "first_air_date",
|
|
|
|
"first_air_date_year", "with_networks", "with_status", "with_type", "with_name_translation"
|
2021-03-30 05:50:53 +00:00
|
|
|
]
|
2021-11-29 00:27:15 +00:00
|
|
|
discover_strings = [
|
2022-03-21 17:40:32 +00:00
|
|
|
"with_cast", "with_crew", "with_people", "with_companies", "without_companies", "with_networks", "with_genres",
|
|
|
|
"without_genres", "with_release_type", "with_keywords", "without_keywords", "with_original_language", "timezone",
|
|
|
|
"with_watch_providers", "without_watch_providers", "with_overview_translation", "with_title_translation", "with_name_translation"
|
2021-11-29 00:27:15 +00:00
|
|
|
]
|
|
|
|
discover_ints = ["vote_count", "with_runtime"]
|
2022-03-21 17:40:32 +00:00
|
|
|
modifiers = [".gte", ".lte"]
|
2021-11-29 00:27:15 +00:00
|
|
|
discover_years = ["primary_release_year", "year", "first_air_date_year"]
|
2021-11-29 16:02:16 +00:00
|
|
|
discover_booleans = ["include_adult", "include_video", "include_null_first_air_dates", "screened_theatrically"]
|
2022-03-21 17:40:32 +00:00
|
|
|
discover_dates = ["primary_release_date", "release_date", "air_date", "first_air_date"]
|
|
|
|
date_methods = [f"{f}{m}" for f in discover_dates for m in modifiers]
|
|
|
|
discover_numbers = ["vote_average"]
|
|
|
|
discover_special = [
|
|
|
|
"region", "sort_by", "certification_country", "certification", "certification.lte", "certification.gte",
|
|
|
|
"watch_region", "with_watch_monetization_types", "with_status", "limit", "with_type"
|
2021-03-30 05:50:53 +00:00
|
|
|
]
|
2022-03-21 17:40:32 +00:00
|
|
|
discover_all = discover_special + discover_strings + discover_years + discover_booleans + date_methods + \
|
|
|
|
[f"{f}{m}" for f in discover_ints for m in modifiers] + \
|
|
|
|
[f"{f}{m}" for f in discover_numbers for m in modifiers]
|
2021-03-30 05:50:53 +00:00
|
|
|
discover_movie_sort = [
|
2021-07-23 19:44:21 +00:00
|
|
|
"popularity.asc", "popularity.desc", "release_date.asc", "release_date.desc", "revenue.asc", "revenue.desc",
|
|
|
|
"primary_release_date.asc", "primary_release_date.desc", "original_title.asc", "original_title.desc",
|
|
|
|
"vote_average.asc", "vote_average.desc", "vote_count.asc", "vote_count.desc"
|
2021-03-30 05:50:53 +00:00
|
|
|
]
|
2021-07-23 19:44:21 +00:00
|
|
|
discover_tv_sort = ["vote_average.desc", "vote_average.asc", "first_air_date.desc", "first_air_date.asc", "popularity.desc", "popularity.asc"]
|
2021-11-26 08:24:36 +00:00
|
|
|
discover_monetization_types = ["flatrate", "free", "ads", "rent", "buy"]
|
2022-10-24 06:17:49 +00:00
|
|
|
discover_types = {
|
|
|
|
"Documentary": "documentary", "News": "news", "Miniseries": "miniseries",
|
|
|
|
"Reality": "reality", "Scripted": "scripted", "Talk Show": "talk_show", "Video": "video"
|
|
|
|
}
|
|
|
|
discover_status = {
|
|
|
|
"Returning Series": "returning", "Planned": "planned", "In Production": "production",
|
|
|
|
"Ended": "ended", "Canceled": "canceled", "Pilot": "pilot"
|
|
|
|
}
|
2022-03-13 19:39:34 +00:00
|
|
|
|
|
|
|
class TMDbCountry:
|
|
|
|
def __init__(self, data):
|
|
|
|
self.iso_3166_1 = data.split(":")[0] if isinstance(data, str) else data.iso_3166_1
|
|
|
|
self.name = data.split(":")[1] if isinstance(data, str) else data.name
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
return f"{self.iso_3166_1}:{self.name}"
|
|
|
|
|
|
|
|
|
|
|
|
class TMDbSeason:
|
|
|
|
def __init__(self, data):
|
2024-03-28 20:45:55 +00:00
|
|
|
self.season_number = int(data.split("%:%")[0]) if isinstance(data, str) else data.season_number
|
2024-03-20 15:35:15 +00:00
|
|
|
self.name = data.split("%:%")[1] if isinstance(data, str) else data.name
|
2024-03-28 20:45:55 +00:00
|
|
|
self.average = float(data.split("%:%")[2]) if isinstance(data, str) else data.vote_average
|
2022-03-13 19:39:34 +00:00
|
|
|
|
|
|
|
def __repr__(self):
|
2024-03-28 20:45:55 +00:00
|
|
|
return f"{self.season_number}%:%{self.name}%:%{self.average}"
|
2022-03-13 19:39:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TMDBObj:
|
|
|
|
def __init__(self, tmdb, tmdb_id, ignore_cache=False):
|
|
|
|
self._tmdb = tmdb
|
|
|
|
self.tmdb_id = tmdb_id
|
|
|
|
self.ignore_cache = ignore_cache
|
|
|
|
|
|
|
|
def _load(self, data):
|
|
|
|
self.title = data["title"] if isinstance(data, dict) else data.title
|
|
|
|
self.tagline = data["tagline"] if isinstance(data, dict) else data.tagline
|
|
|
|
self.overview = data["overview"] if isinstance(data, dict) else data.overview
|
|
|
|
self.imdb_id = data["imdb_id"] if isinstance(data, dict) else data.imdb_id
|
|
|
|
self.poster_url = data["poster_url"] if isinstance(data, dict) else data.poster_url
|
|
|
|
self.backdrop_url = data["backdrop_url"] if isinstance(data, dict) else data.backdrop_url
|
|
|
|
self.vote_count = data["vote_count"] if isinstance(data, dict) else data.vote_count
|
|
|
|
self.vote_average = data["vote_average"] if isinstance(data, dict) else data.vote_average
|
|
|
|
self.language_iso = data["language_iso"] if isinstance(data, dict) else data.original_language.iso_639_1 if data.original_language else None
|
|
|
|
self.language_name = data["language_name"] if isinstance(data, dict) else data.original_language.english_name if data.original_language else None
|
2023-12-02 17:20:35 +00:00
|
|
|
self.genres = [g for g in data["genres"].split("|") if g] if isinstance(data, dict) else [g.name for g in data.genres if g]
|
|
|
|
self.keywords = [k for k in data["keywords"].split("|") if k] if isinstance(data, dict) else [k.name for k in data.keywords if k]
|
2022-03-13 19:39:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TMDbMovie(TMDBObj):
|
|
|
|
def __init__(self, tmdb, tmdb_id, ignore_cache=False):
|
|
|
|
super().__init__(tmdb, tmdb_id, ignore_cache=ignore_cache)
|
|
|
|
expired = None
|
|
|
|
data = None
|
|
|
|
if self._tmdb.config.Cache and not ignore_cache:
|
|
|
|
data, expired = self._tmdb.config.Cache.query_tmdb_movie(tmdb_id, self._tmdb.expiration)
|
|
|
|
if expired or not data:
|
2022-05-05 19:22:43 +00:00
|
|
|
data = self.load_movie()
|
2022-03-13 19:39:34 +00:00
|
|
|
super()._load(data)
|
|
|
|
|
|
|
|
self.original_title = data["original_title"] if isinstance(data, dict) else data.original_title
|
|
|
|
self.release_date = data["release_date"] if isinstance(data, dict) else data.release_date
|
2022-03-14 12:52:55 +00:00
|
|
|
self.studio = data["studio"] if isinstance(data, dict) else data.companies[0].name if data.companies else None
|
2022-03-13 19:39:34 +00:00
|
|
|
self.collection_id = data["collection_id"] if isinstance(data, dict) else data.collection.id if data.collection else None
|
|
|
|
self.collection_name = data["collection_name"] if isinstance(data, dict) else data.collection.name if data.collection else None
|
|
|
|
|
|
|
|
if self._tmdb.config.Cache and not ignore_cache:
|
|
|
|
self._tmdb.config.Cache.update_tmdb_movie(expired, self, self._tmdb.expiration)
|
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
|
|
|
def load_movie(self):
|
|
|
|
try:
|
|
|
|
return self._tmdb.TMDb.movie(self.tmdb_id, partial="external_ids,keywords")
|
|
|
|
except NotFound:
|
2024-01-04 20:24:53 +00:00
|
|
|
raise Failed(f"TMDb Error: No Movie found for TMDb ID: {self.tmdb_id}")
|
2023-12-01 21:15:10 +00:00
|
|
|
except TMDbException as e:
|
|
|
|
logger.stacktrace()
|
2024-01-04 20:24:53 +00:00
|
|
|
raise TMDbException(f"TMDb Error: Unexpected Error with TMDb ID: {self.tmdb_id}: {e}")
|
2022-05-05 19:22:43 +00:00
|
|
|
|
2022-03-13 19:39:34 +00:00
|
|
|
|
|
|
|
class TMDbShow(TMDBObj):
|
|
|
|
def __init__(self, tmdb, tmdb_id, ignore_cache=False):
|
|
|
|
super().__init__(tmdb, tmdb_id, ignore_cache=ignore_cache)
|
|
|
|
expired = None
|
|
|
|
data = None
|
|
|
|
if self._tmdb.config.Cache and not ignore_cache:
|
|
|
|
data, expired = self._tmdb.config.Cache.query_tmdb_show(tmdb_id, self._tmdb.expiration)
|
|
|
|
if expired or not data:
|
2022-05-05 19:22:43 +00:00
|
|
|
data = self.load_show()
|
2022-03-13 19:39:34 +00:00
|
|
|
super()._load(data)
|
|
|
|
|
|
|
|
self.original_title = data["original_title"] if isinstance(data, dict) else data.original_name
|
|
|
|
self.first_air_date = data["first_air_date"] if isinstance(data, dict) else data.first_air_date
|
|
|
|
self.last_air_date = data["last_air_date"] if isinstance(data, dict) else data.last_air_date
|
|
|
|
self.status = data["status"] if isinstance(data, dict) else data.status
|
|
|
|
self.type = data["type"] if isinstance(data, dict) else data.type
|
2022-03-14 12:52:55 +00:00
|
|
|
self.studio = data["studio"] if isinstance(data, dict) else data.networks[0].name if data.networks else None
|
2022-03-13 19:39:34 +00:00
|
|
|
self.tvdb_id = data["tvdb_id"] if isinstance(data, dict) else data.tvdb_id
|
2023-04-28 03:43:26 +00:00
|
|
|
loop = data.origin_countries if not isinstance(data, dict) else data["countries"].split("|") if data["countries"] else [] # noqa
|
2022-03-13 19:39:34 +00:00
|
|
|
self.countries = [TMDbCountry(c) for c in loop]
|
2024-03-20 15:35:15 +00:00
|
|
|
loop = data.seasons if not isinstance(data, dict) else data["seasons"].split("%|%") if data["seasons"] else [] # noqa
|
2022-03-13 19:39:34 +00:00
|
|
|
self.seasons = [TMDbSeason(s) for s in loop]
|
|
|
|
|
|
|
|
if self._tmdb.config.Cache and not ignore_cache:
|
|
|
|
self._tmdb.config.Cache.update_tmdb_show(expired, self, self._tmdb.expiration)
|
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
|
|
|
def load_show(self):
|
|
|
|
try:
|
|
|
|
return self._tmdb.TMDb.tv_show(self.tmdb_id, partial="external_ids,keywords")
|
|
|
|
except NotFound:
|
2024-01-04 20:24:53 +00:00
|
|
|
raise Failed(f"TMDb Error: No Show found for TMDb ID: {self.tmdb_id}")
|
2023-12-01 21:15:10 +00:00
|
|
|
except TMDbException as e:
|
|
|
|
logger.stacktrace()
|
2024-01-04 20:24:53 +00:00
|
|
|
raise TMDbException(f"TMDb Error: Unexpected Error with TMDb ID: {self.tmdb_id}: {e}")
|
2022-03-13 19:39:34 +00:00
|
|
|
|
2024-04-02 14:56:47 +00:00
|
|
|
class TMDbEpisode:
|
|
|
|
def __init__(self, tmdb, tmdb_id, season_number, episode_number, ignore_cache=False):
|
|
|
|
self._tmdb = tmdb
|
|
|
|
self.tmdb_id = tmdb_id
|
|
|
|
self.season_number = season_number
|
|
|
|
self.episode_number = episode_number
|
|
|
|
self.ignore_cache = ignore_cache
|
|
|
|
expired = None
|
|
|
|
data = None
|
|
|
|
if self._tmdb.config.Cache and not ignore_cache:
|
|
|
|
data, expired = self._tmdb.config.Cache.query_tmdb_episode(self.tmdb_id, self.season_number, self.episode_number, self._tmdb.expiration)
|
|
|
|
if expired or not data:
|
|
|
|
data = self.load_episode()
|
|
|
|
|
|
|
|
self.title = data["title"] if isinstance(data, dict) else data.title
|
|
|
|
self.air_date = data["air_date"] if isinstance(data, dict) else data.air_date
|
|
|
|
self.overview = data["overview"] if isinstance(data, dict) else data.overview
|
|
|
|
self.still_url = data["still_url"] if isinstance(data, dict) else data.still_url
|
|
|
|
self.vote_count = data["vote_count"] if isinstance(data, dict) else data.vote_count
|
|
|
|
self.vote_average = data["vote_average"] if isinstance(data, dict) else data.vote_average
|
|
|
|
self.imdb_id = data["imdb_id"] if isinstance(data, dict) else data.imdb_id
|
|
|
|
self.tvdb_id = data["tvdb_id"] if isinstance(data, dict) else data.tvdb_id
|
|
|
|
|
|
|
|
if self._tmdb.config.Cache and not ignore_cache:
|
|
|
|
self._tmdb.config.Cache.update_tmdb_episode(expired, self, self._tmdb.expiration)
|
|
|
|
|
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
|
|
|
def load_episode(self):
|
|
|
|
try:
|
|
|
|
return self._tmdb.TMDb.tv_episode(self.tmdb_id, self.season_number, self.episode_number)
|
|
|
|
except NotFound as e:
|
|
|
|
raise Failed(f"TMDb Error: No Episode found for TMDb ID {self.tmdb_id} Season {self.season_number} Episode {self.episode_number}: {e}")
|
|
|
|
except TMDbException as e:
|
|
|
|
logger.stacktrace()
|
|
|
|
raise TMDbException(f"TMDb Error: Unexpected Error with TMDb ID: {self.tmdb_id}: {e}")
|
|
|
|
|
|
|
|
|
2021-06-14 15:24:11 +00:00
|
|
|
class TMDb:
|
2021-05-08 00:40:07 +00:00
|
|
|
def __init__(self, config, params):
|
|
|
|
self.config = config
|
2022-01-21 16:34:19 +00:00
|
|
|
self.apikey = params["apikey"]
|
|
|
|
self.language = params["language"]
|
2022-03-23 18:43:50 +00:00
|
|
|
self.region = None
|
2022-03-13 19:39:34 +00:00
|
|
|
self.expiration = params["expiration"]
|
2022-02-13 16:33:57 +00:00
|
|
|
logger.secret(self.apikey)
|
2021-10-04 17:51:32 +00:00
|
|
|
try:
|
2022-01-21 16:34:19 +00:00
|
|
|
self.TMDb = TMDbAPIs(self.apikey, language=self.language, session=self.config.session)
|
2021-10-04 17:51:32 +00:00
|
|
|
except TMDbException as e:
|
|
|
|
raise Failed(f"TMDb Error: {e}")
|
2023-04-28 03:43:26 +00:00
|
|
|
self.iso_3166_1 = {iso: i.name for iso, i in self.TMDb._iso_3166_1.items()} # noqa
|
2021-01-20 21:37:59 +00:00
|
|
|
|
2021-05-08 00:40:07 +00:00
|
|
|
def convert_from(self, tmdb_id, convert_to, is_movie):
|
2022-03-13 19:39:34 +00:00
|
|
|
item = self.get_movie(tmdb_id) if is_movie else self.get_show(tmdb_id)
|
2022-01-21 16:34:19 +00:00
|
|
|
check_id = item.tvdb_id if convert_to == "tvdb_id" and not is_movie else item.imdb_id
|
|
|
|
if not check_id:
|
|
|
|
raise Failed(f"TMDb Error: No {convert_to.upper().replace('B_', 'b ')} found for TMDb ID {tmdb_id}")
|
|
|
|
return check_id
|
2021-08-07 06:01:21 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2021-08-07 06:01:21 +00:00
|
|
|
def convert_tvdb_to(self, tvdb_id):
|
2022-01-23 19:06:43 +00:00
|
|
|
try:
|
|
|
|
results = self.TMDb.find_by_id(tvdb_id=tvdb_id)
|
|
|
|
if results.tv_results:
|
|
|
|
return results.tv_results[0].id
|
|
|
|
except NotFound:
|
|
|
|
pass
|
|
|
|
raise Failed(f"TMDb Error: No TMDb ID found for TVDb ID {tvdb_id}")
|
2021-08-07 06:01:21 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2021-08-07 06:01:21 +00:00
|
|
|
def convert_imdb_to(self, imdb_id):
|
2022-01-23 00:09:06 +00:00
|
|
|
try:
|
|
|
|
results = self.TMDb.find_by_id(imdb_id=imdb_id)
|
|
|
|
if results.movie_results:
|
|
|
|
return results.movie_results[0].id, "movie"
|
|
|
|
elif results.tv_results:
|
|
|
|
return results.tv_results[0].id, "show"
|
|
|
|
elif results.tv_episode_results:
|
|
|
|
item = results.tv_episode_results[0]
|
|
|
|
return f"{item.tv_id}_{item.season_number}_{item.episode_number}", "episode"
|
|
|
|
except NotFound:
|
2022-01-23 19:06:43 +00:00
|
|
|
pass
|
|
|
|
raise Failed(f"TMDb Error: No TMDb ID found for IMDb ID {imdb_id}")
|
2021-01-20 21:37:59 +00:00
|
|
|
|
|
|
|
def get_movie_show_or_collection(self, tmdb_id, is_movie):
|
|
|
|
if is_movie:
|
|
|
|
try: return self.get_collection(tmdb_id)
|
|
|
|
except Failed:
|
|
|
|
try: return self.get_movie(tmdb_id)
|
2021-02-24 06:44:06 +00:00
|
|
|
except Failed: raise Failed(f"TMDb Error: No Movie or Collection found for TMDb ID {tmdb_id}")
|
2021-01-20 21:37:59 +00:00
|
|
|
else: return self.get_show(tmdb_id)
|
|
|
|
|
2022-05-28 05:32:02 +00:00
|
|
|
def get_movie(self, tmdb_id, ignore_cache=False):
|
|
|
|
return TMDbMovie(self, tmdb_id, ignore_cache=ignore_cache)
|
2021-01-20 21:37:59 +00:00
|
|
|
|
2022-05-28 05:32:02 +00:00
|
|
|
def get_show(self, tmdb_id, ignore_cache=False):
|
|
|
|
return TMDbShow(self, tmdb_id, ignore_cache=ignore_cache)
|
2021-01-20 21:37:59 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2022-04-18 18:16:39 +00:00
|
|
|
def get_season(self, tmdb_id, season_number, partial=None):
|
|
|
|
try: return self.TMDb.tv_season(tmdb_id, season_number, partial=partial)
|
2022-05-05 19:22:43 +00:00
|
|
|
except NotFound as e: raise Failed(f"TMDb Error: No Season found for TMDb ID {tmdb_id} Season {season_number}: {e}")
|
2022-04-18 18:16:39 +00:00
|
|
|
|
2024-04-02 14:56:47 +00:00
|
|
|
def get_episode(self, tmdb_id, season_number, episode_number, ignore_cache=False):
|
|
|
|
return TMDbEpisode(self, tmdb_id, season_number, episode_number, ignore_cache=ignore_cache)
|
2022-04-18 18:16:39 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2022-01-27 20:53:32 +00:00
|
|
|
def get_collection(self, tmdb_id, partial=None):
|
|
|
|
try: return self.TMDb.collection(tmdb_id, partial=partial)
|
2022-05-05 19:22:43 +00:00
|
|
|
except NotFound as e: raise Failed(f"TMDb Error: No Collection found for TMDb ID {tmdb_id}: {e}")
|
2021-01-20 21:37:59 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2022-01-27 20:53:32 +00:00
|
|
|
def get_person(self, tmdb_id, partial=None):
|
|
|
|
try: return self.TMDb.person(tmdb_id, partial=partial)
|
2022-05-05 19:22:43 +00:00
|
|
|
except NotFound as e: raise Failed(f"TMDb Error: No Person found for TMDb ID {tmdb_id}: {e}")
|
2021-02-28 02:47:41 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2022-01-27 20:53:32 +00:00
|
|
|
def _company(self, tmdb_id, partial=None):
|
|
|
|
try: return self.TMDb.company(tmdb_id, partial=partial)
|
2022-05-05 19:22:43 +00:00
|
|
|
except NotFound as e: raise Failed(f"TMDb Error: No Company found for TMDb ID {tmdb_id}: {e}")
|
2021-01-20 21:37:59 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2022-01-27 20:53:32 +00:00
|
|
|
def _network(self, tmdb_id, partial=None):
|
|
|
|
try: return self.TMDb.network(tmdb_id, partial=partial)
|
2022-05-05 19:22:43 +00:00
|
|
|
except NotFound as e: raise Failed(f"TMDb Error: No Network found for TMDb ID {tmdb_id}: {e}")
|
2021-01-20 21:37:59 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2021-05-07 19:53:54 +00:00
|
|
|
def _keyword(self, tmdb_id):
|
2022-01-21 16:34:19 +00:00
|
|
|
try: return self.TMDb.keyword(tmdb_id)
|
2022-05-05 19:22:43 +00:00
|
|
|
except NotFound as e: raise Failed(f"TMDb Error: No Keyword found for TMDb ID {tmdb_id}: {e}")
|
2021-01-20 21:37:59 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2021-01-20 21:37:59 +00:00
|
|
|
def get_list(self, tmdb_id):
|
2022-01-21 16:34:19 +00:00
|
|
|
try: return self.TMDb.list(tmdb_id)
|
2022-05-05 19:22:43 +00:00
|
|
|
except NotFound as e: raise Failed(f"TMDb Error: No List found for TMDb ID {tmdb_id}: {e}")
|
2021-01-20 21:37:59 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2022-01-30 07:48:56 +00:00
|
|
|
def get_popular_people(self, limit):
|
2022-03-12 15:34:43 +00:00
|
|
|
return {str(p.id): p.name for p in self.TMDb.popular_people().get_results(limit)}
|
2022-01-30 07:48:56 +00:00
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
2022-02-01 14:55:07 +00:00
|
|
|
def search_people(self, name):
|
2022-05-05 19:22:43 +00:00
|
|
|
try: return self.TMDb.people_search(name)
|
|
|
|
except NotFound: raise Failed(f"TMDb Error: Actor {name} Not Found")
|
2022-02-01 14:55:07 +00:00
|
|
|
|
2021-07-21 17:40:05 +00:00
|
|
|
def validate_tmdb_ids(self, tmdb_ids, tmdb_method):
|
|
|
|
tmdb_list = util.get_int_list(tmdb_ids, f"TMDb {type_map[tmdb_method]} ID")
|
2021-05-07 19:53:54 +00:00
|
|
|
tmdb_values = []
|
|
|
|
for tmdb_id in tmdb_list:
|
2021-07-21 17:40:05 +00:00
|
|
|
try: tmdb_values.append(self.validate_tmdb(tmdb_id, tmdb_method))
|
2021-05-07 19:53:54 +00:00
|
|
|
except Failed as e: logger.error(e)
|
|
|
|
if len(tmdb_values) == 0: raise Failed(f"TMDb Error: No valid TMDb IDs in {tmdb_list}")
|
|
|
|
return tmdb_values
|
|
|
|
|
2021-07-21 17:40:05 +00:00
|
|
|
def validate_tmdb(self, tmdb_id, tmdb_method):
|
|
|
|
tmdb_type = type_map[tmdb_method]
|
2021-05-07 19:53:54 +00:00
|
|
|
if tmdb_type == "Movie": self.get_movie(tmdb_id)
|
|
|
|
elif tmdb_type == "Show": self.get_show(tmdb_id)
|
|
|
|
elif tmdb_type == "Collection": self.get_collection(tmdb_id)
|
|
|
|
elif tmdb_type == "Person": self.get_person(tmdb_id)
|
|
|
|
elif tmdb_type == "Company": self._company(tmdb_id)
|
|
|
|
elif tmdb_type == "Network": self._network(tmdb_id)
|
2022-01-21 16:34:19 +00:00
|
|
|
elif tmdb_type == "Keyword": self._keyword(tmdb_id)
|
2021-05-07 19:53:54 +00:00
|
|
|
elif tmdb_type == "List": self.get_list(tmdb_id)
|
|
|
|
return tmdb_id
|
|
|
|
|
2022-05-05 19:22:43 +00:00
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000, retry_on_exception=util.retry_if_not_failed)
|
|
|
|
def get_items(self, method, data, region, is_movie, result_type):
|
|
|
|
if method == "tmdb_popular":
|
|
|
|
results = self.TMDb.popular_movies(region=region) if is_movie else self.TMDb.popular_tv()
|
|
|
|
elif method == "tmdb_top_rated":
|
|
|
|
results = self.TMDb.top_rated_movies(region=region) if is_movie else self.TMDb.top_rated_tv()
|
|
|
|
elif method == "tmdb_now_playing":
|
|
|
|
results = self.TMDb.now_playing_movies(region=region)
|
|
|
|
elif method == "tmdb_upcoming":
|
|
|
|
results = self.TMDb.upcoming_movies(region=region)
|
|
|
|
elif method == "tmdb_airing_today":
|
|
|
|
results = self.TMDb.tv_airing_today()
|
|
|
|
elif method == "tmdb_on_the_air":
|
|
|
|
results = self.TMDb.tv_on_the_air()
|
|
|
|
else:
|
|
|
|
results = self.TMDb.trending("movie" if is_movie else "tv", "day" if method == "tmdb_trending_daily" else "week")
|
|
|
|
return [(i.id, result_type) for i in results.get_results(data)]
|
|
|
|
|
2022-03-23 18:43:50 +00:00
|
|
|
def get_tmdb_ids(self, method, data, is_movie, region):
|
|
|
|
if not region and self.region:
|
|
|
|
region = self.region
|
2021-08-01 04:35:42 +00:00
|
|
|
pretty = method.replace("_", " ").title().replace("Tmdb", "TMDb")
|
2021-01-20 21:37:59 +00:00
|
|
|
media_type = "Movie" if is_movie else "Show"
|
2022-01-21 16:34:19 +00:00
|
|
|
result_type = "tmdb" if is_movie else "tmdb_show"
|
2021-08-07 06:01:21 +00:00
|
|
|
ids = []
|
2022-01-21 16:34:19 +00:00
|
|
|
if method in ["tmdb_network", "tmdb_company", "tmdb_keyword"]:
|
|
|
|
if method == "tmdb_company":
|
|
|
|
item = self._company(int(data))
|
|
|
|
elif method == "tmdb_network":
|
|
|
|
item = self._network(int(data))
|
2021-01-20 21:37:59 +00:00
|
|
|
else:
|
2022-01-21 16:34:19 +00:00
|
|
|
item = self._keyword(int(data))
|
|
|
|
results = item.movies if is_movie else item.tv_shows
|
|
|
|
ids = [(i.id, result_type) for i in results.get_results(results.total_results)]
|
|
|
|
logger.info(f"Processing {pretty}: ({data}) {item.name} ({len(results)} {media_type}{'' if len(results) == 1 else 's'})")
|
|
|
|
elif method == "tmdb_discover":
|
|
|
|
attrs = data.copy()
|
|
|
|
limit = int(attrs.pop("limit"))
|
2022-03-21 17:40:32 +00:00
|
|
|
for date_attr in date_methods:
|
2022-01-21 16:34:19 +00:00
|
|
|
if date_attr in attrs:
|
2023-12-07 19:49:32 +00:00
|
|
|
try:
|
|
|
|
attrs[date_attr] = util.validate_date(attrs[date_attr], return_as="%Y-%m-%d")
|
|
|
|
except Failed as e:
|
|
|
|
raise Failed(f"Collection Error: tmdb_discover attribute {date_attr}: {e}")
|
2022-11-10 18:44:38 +00:00
|
|
|
if is_movie and region and "region" not in attrs:
|
|
|
|
attrs["region"] = region
|
2022-09-07 19:32:52 +00:00
|
|
|
logger.trace(f"Params: {attrs}")
|
2022-01-21 16:34:19 +00:00
|
|
|
results = self.TMDb.discover_movies(**attrs) if is_movie else self.TMDb.discover_tv_shows(**attrs)
|
|
|
|
amount = results.total_results if limit == 0 or results.total_results < limit else limit
|
|
|
|
ids = [(i.id, result_type) for i in results.get_results(amount)]
|
|
|
|
logger.info(f"Processing {pretty}: {amount} {media_type}{'' if amount == 1 else 's'}")
|
|
|
|
for attr, value in attrs.items():
|
|
|
|
logger.info(f" {attr}: {value}")
|
2022-03-22 18:16:30 +00:00
|
|
|
elif method in int_builders:
|
2022-05-05 19:22:43 +00:00
|
|
|
ids = self.get_items(method, data, region, is_movie, result_type)
|
2021-05-09 05:37:45 +00:00
|
|
|
logger.info(f"Processing {pretty}: {data} {media_type}{'' if data == 1 else 's'}")
|
2021-01-20 21:37:59 +00:00
|
|
|
else:
|
|
|
|
tmdb_id = int(data)
|
|
|
|
if method == "tmdb_list":
|
2022-01-21 16:34:19 +00:00
|
|
|
results = self.get_list(tmdb_id)
|
|
|
|
tmdb_name = results.name
|
2023-10-25 20:56:03 +00:00
|
|
|
ids = [(i.id, "tmdb" if isinstance(i, Movie) else "tmdb_show") for i in results.get_results()]
|
2021-01-20 21:37:59 +00:00
|
|
|
elif method == "tmdb_movie":
|
2022-01-21 16:34:19 +00:00
|
|
|
tmdb_name = self.get_movie(tmdb_id).title
|
2021-08-07 06:01:21 +00:00
|
|
|
ids.append((tmdb_id, "tmdb"))
|
2021-01-20 21:37:59 +00:00
|
|
|
elif method == "tmdb_collection":
|
2022-01-21 16:34:19 +00:00
|
|
|
collection = self.get_collection(tmdb_id)
|
|
|
|
tmdb_name = collection.name
|
|
|
|
ids = [(t.id, "tmdb") for t in collection.movies]
|
2021-01-20 21:37:59 +00:00
|
|
|
elif method == "tmdb_show":
|
2022-03-13 19:39:34 +00:00
|
|
|
tmdb_name = self.get_show(tmdb_id).title
|
2021-08-07 06:01:21 +00:00
|
|
|
ids.append((tmdb_id, "tmdb_show"))
|
2021-01-20 21:37:59 +00:00
|
|
|
else:
|
2022-01-27 20:53:32 +00:00
|
|
|
person = self.get_person(tmdb_id, partial="movie_credits,tv_credits")
|
2022-01-21 16:34:19 +00:00
|
|
|
tmdb_name = person.name
|
|
|
|
if method == "tmdb_actor":
|
2022-01-23 01:53:47 +00:00
|
|
|
ids = [(i.movie.id, "tmdb") for i in person.movie_cast]
|
|
|
|
ids.extend([(i.tv_show.id, "tmdb_show") for i in person.tv_cast])
|
2022-01-21 16:34:19 +00:00
|
|
|
elif method == "tmdb_crew":
|
2022-01-23 01:53:47 +00:00
|
|
|
ids = [(i.movie.id, "tmdb") for i in person.movie_crew]
|
|
|
|
ids.extend([(i.tv_show.id, "tmdb_show") for i in person.tv_crew])
|
2022-01-21 16:34:19 +00:00
|
|
|
elif method == "tmdb_director":
|
2022-01-23 01:53:47 +00:00
|
|
|
ids = [(i.movie.id, "tmdb") for i in person.movie_crew if i.department == "Directing"]
|
|
|
|
ids.extend([(i.tv_show.id, "tmdb_show") for i in person.tv_crew if i.department == "Directing"])
|
2022-01-21 16:34:19 +00:00
|
|
|
elif method == "tmdb_writer":
|
2022-01-23 01:53:47 +00:00
|
|
|
ids = [(i.movie.id, "tmdb") for i in person.movie_crew if i.department == "Writing"]
|
|
|
|
ids.extend([(i.tv_show.id, "tmdb_show") for i in person.tv_crew if i.department == "Writing"])
|
2022-01-21 16:34:19 +00:00
|
|
|
elif method == "tmdb_producer":
|
2022-01-23 01:53:47 +00:00
|
|
|
ids = [(i.movie.id, "tmdb") for i in person.movie_crew if i.department == "Production"]
|
|
|
|
ids.extend([(i.tv_show.id, "tmdb_show") for i in person.tv_crew if i.department == "Production"])
|
2022-01-21 16:34:19 +00:00
|
|
|
else:
|
|
|
|
raise Failed(f"TMDb Error: Method {method} not supported")
|
2021-08-07 06:01:21 +00:00
|
|
|
if len(ids) > 0:
|
|
|
|
logger.info(f"Processing {pretty}: ({tmdb_id}) {tmdb_name} ({len(ids)} Item{'' if len(ids) == 1 else 's'})")
|
|
|
|
return ids
|
2022-01-27 15:22:58 +00:00
|
|
|
|
|
|
|
def get_item(self, item, tmdb_id, tvdb_id, imdb_id, is_movie=True):
|
|
|
|
tmdb_item = None
|
|
|
|
if tvdb_id and not tmdb_id:
|
|
|
|
tmdb_id = self.config.Convert.tvdb_to_tmdb(tvdb_id)
|
|
|
|
if imdb_id and not tmdb_id:
|
|
|
|
_id, _type = self.config.Convert.imdb_to_tmdb(imdb_id)
|
|
|
|
if _id and ((_type == "movie" and is_movie) or (_type == "show" and not is_movie)):
|
|
|
|
tmdb_id = _id
|
|
|
|
if tmdb_id:
|
|
|
|
try:
|
|
|
|
tmdb_item = self.get_movie(tmdb_id) if is_movie else self.get_show(tmdb_id)
|
|
|
|
except Failed as e:
|
2022-02-13 16:33:57 +00:00
|
|
|
logger.error(str(e))
|
2022-03-08 21:06:13 +00:00
|
|
|
elif tvdb_id and not is_movie:
|
|
|
|
logger.info(f"{item.title[:25]:<25} | No TMDb ID for TVDb ID: {tvdb_id}")
|
|
|
|
elif imdb_id:
|
|
|
|
logger.info(f"{item.title[:25]:<25} | No TMDb ID for IMDb ID: {imdb_id}")
|
2022-01-27 15:22:58 +00:00
|
|
|
else:
|
2022-02-13 16:33:57 +00:00
|
|
|
logger.info(f"{item.title[:25]:<25} | No TMDb ID for Guid: {item.guid}")
|
2022-01-27 15:22:58 +00:00
|
|
|
return tmdb_item
|
2022-10-24 06:17:49 +00:00
|
|
|
|
|
|
|
def item_filter(self, item, filter_attr, modifier, filter_final, filter_data, is_movie, current_time):
|
|
|
|
if filter_attr in ["tmdb_status", "tmdb_type", "original_language"]:
|
|
|
|
if filter_attr == "tmdb_status":
|
|
|
|
check_value = discover_status[item.status]
|
|
|
|
elif filter_attr == "tmdb_type":
|
|
|
|
check_value = discover_types[item.type]
|
|
|
|
elif filter_attr == "original_language":
|
|
|
|
check_value = item.language_iso
|
|
|
|
else:
|
|
|
|
raise Failed
|
|
|
|
if (modifier == ".not" and check_value in filter_data) or (modifier == "" and check_value not in filter_data):
|
|
|
|
return False
|
|
|
|
elif filter_attr in ["first_episode_aired", "last_episode_aired", "last_episode_aired_or_never"]:
|
|
|
|
tmdb_date = None
|
|
|
|
if filter_attr == "first_episode_aired":
|
|
|
|
tmdb_date = item.first_air_date
|
|
|
|
elif filter_attr in ["last_episode_aired", "last_episode_aired_or_never"]:
|
|
|
|
tmdb_date = item.last_air_date
|
|
|
|
|
|
|
|
# tmdb_date is empty if never aired yet
|
|
|
|
if tmdb_date is None and filter_attr == "last_episode_aired_or_never":
|
|
|
|
return True
|
|
|
|
if util.is_date_filter(tmdb_date, modifier, filter_data, filter_final, current_time):
|
|
|
|
return False
|
|
|
|
elif modifier in [".gt", ".gte", ".lt", ".lte"]:
|
|
|
|
attr = None
|
|
|
|
if filter_attr == "tmdb_vote_count":
|
|
|
|
attr = item.vote_count
|
2023-04-18 14:50:26 +00:00
|
|
|
elif filter_attr == "tmdb_vote_average":
|
|
|
|
attr = item.vote_average
|
2022-10-24 06:17:49 +00:00
|
|
|
elif filter_attr == "tmdb_year":
|
|
|
|
attr = item.release_date.year if is_movie else item.first_air_date.year
|
|
|
|
if util.is_number_filter(attr, modifier, filter_data):
|
|
|
|
return False
|
|
|
|
elif filter_attr in ["tmdb_genre", "tmdb_keyword", "origin_country"]:
|
|
|
|
if filter_attr == "tmdb_genre":
|
|
|
|
attrs = item.genres
|
|
|
|
elif filter_attr == "tmdb_keyword":
|
|
|
|
attrs = item.keywords
|
|
|
|
elif filter_attr == "origin_country":
|
|
|
|
attrs = [c.iso_3166_1 for c in item.countries]
|
|
|
|
else:
|
|
|
|
raise Failed
|
|
|
|
if modifier == ".regex":
|
|
|
|
has_match = False
|
|
|
|
for reg in filter_data:
|
|
|
|
for name in attrs:
|
|
|
|
if re.compile(reg).search(name):
|
|
|
|
has_match = True
|
|
|
|
if has_match is False:
|
|
|
|
return False
|
2023-01-27 15:16:00 +00:00
|
|
|
elif modifier in [".count_gt", ".count_gte", ".count_lt", ".count_lte"]:
|
|
|
|
test_number = len(attrs) if attrs else 0
|
|
|
|
modifier = f".{modifier[7:]}"
|
|
|
|
if test_number is None or util.is_number_filter(test_number, modifier, filter_data):
|
|
|
|
return False
|
2022-10-24 06:17:49 +00:00
|
|
|
elif (not list(set(filter_data) & set(attrs)) and modifier == "") \
|
|
|
|
or (list(set(filter_data) & set(attrs)) and modifier == ".not"):
|
|
|
|
return False
|
|
|
|
elif filter_attr == "tmdb_title":
|
|
|
|
if util.is_string_filter([item.title], modifier, filter_data):
|
|
|
|
return False
|
|
|
|
return True
|