|
|
@ -7,8 +7,9 @@ from retrying import retry
|
|
|
|
logger = logging.getLogger("Plex Meta Manager")
|
|
|
|
logger = logging.getLogger("Plex Meta Manager")
|
|
|
|
|
|
|
|
|
|
|
|
class LetterboxdAPI:
|
|
|
|
class LetterboxdAPI:
|
|
|
|
def __init__(self):
|
|
|
|
def __init__(self, Cache=None):
|
|
|
|
self.url = "https://letterboxd.com"
|
|
|
|
self.url = "https://letterboxd.com"
|
|
|
|
|
|
|
|
self.Cache = Cache
|
|
|
|
|
|
|
|
|
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000)
|
|
|
|
@retry(stop_max_attempt_number=6, wait_fixed=10000)
|
|
|
|
def send_request(self, url, language):
|
|
|
|
def send_request(self, url, language):
|
|
|
@ -18,22 +19,28 @@ class LetterboxdAPI:
|
|
|
|
descriptions = self.send_request(list_url, language).xpath("//meta[@property='og:description']/@content")
|
|
|
|
descriptions = self.send_request(list_url, language).xpath("//meta[@property='og:description']/@content")
|
|
|
|
return descriptions[0] if len(descriptions) > 0 and len(descriptions[0]) > 0 else None
|
|
|
|
return descriptions[0] if len(descriptions) > 0 and len(descriptions[0]) > 0 else None
|
|
|
|
|
|
|
|
|
|
|
|
def parse_list_for_slugs(self, list_url, language):
|
|
|
|
def parse_list(self, list_url, language):
|
|
|
|
response = self.send_request(list_url, language)
|
|
|
|
response = self.send_request(list_url, language)
|
|
|
|
slugs = response.xpath("//div[@class='poster film-poster really-lazy-load']/@data-film-slug")
|
|
|
|
letterboxd_ids = response.xpath("//div[@class='poster film-poster really-lazy-load']/@data-film-id")
|
|
|
|
|
|
|
|
items = []
|
|
|
|
|
|
|
|
for letterboxd_id in letterboxd_ids:
|
|
|
|
|
|
|
|
slugs = response.xpath(f"//div[@data-film-id='{letterboxd_id}']/@data-film-slug")
|
|
|
|
|
|
|
|
items.append((letterboxd_id, slugs[0]))
|
|
|
|
next_url = response.xpath("//a[@class='next']/@href")
|
|
|
|
next_url = response.xpath("//a[@class='next']/@href")
|
|
|
|
if len(next_url) > 0:
|
|
|
|
if len(next_url) > 0:
|
|
|
|
slugs.extend(self.parse_list_for_slugs(f"{self.url}{next_url[0]}", language))
|
|
|
|
items.extend(self.parse_list(f"{self.url}{next_url[0]}", language))
|
|
|
|
return slugs
|
|
|
|
return items
|
|
|
|
|
|
|
|
|
|
|
|
def get_tmdb_from_slug(self, slug, language):
|
|
|
|
def get_tmdb_from_slug(self, slug, language):
|
|
|
|
return self.get_tmdb(f"{self.url}{slug}", language)
|
|
|
|
return self.get_tmdb(f"{self.url}{slug}", language)
|
|
|
|
|
|
|
|
|
|
|
|
def get_tmdb(self, letterboxd_url, language):
|
|
|
|
def get_tmdb(self, letterboxd_url, language):
|
|
|
|
response = self.send_request(letterboxd_url, language)
|
|
|
|
response = self.send_request(letterboxd_url, language)
|
|
|
|
ids = response.xpath("//body/@data-tmdb-id")
|
|
|
|
ids = response.xpath("//a[@data-track-action='TMDb']/@href")
|
|
|
|
if len(ids) > 0 and ids[0]:
|
|
|
|
if len(ids) > 0 and ids[0]:
|
|
|
|
return int(ids[0])
|
|
|
|
if "themoviedb.org/movie" in ids[0]:
|
|
|
|
|
|
|
|
return util.regex_first_int(ids[0], "TMDB Movie ID")
|
|
|
|
|
|
|
|
raise Failed(f"Letterboxd Error: TMDb Movie ID not found in {ids[0]}")
|
|
|
|
raise Failed(f"Letterboxd Error: TMDb Movie ID not found at {letterboxd_url}")
|
|
|
|
raise Failed(f"Letterboxd Error: TMDb Movie ID not found at {letterboxd_url}")
|
|
|
|
|
|
|
|
|
|
|
|
def get_items(self, method, data, language, status_message=True):
|
|
|
|
def get_items(self, method, data, language, status_message=True):
|
|
|
@ -41,18 +48,26 @@ class LetterboxdAPI:
|
|
|
|
movie_ids = []
|
|
|
|
movie_ids = []
|
|
|
|
if status_message:
|
|
|
|
if status_message:
|
|
|
|
logger.info(f"Processing {pretty}: {data}")
|
|
|
|
logger.info(f"Processing {pretty}: {data}")
|
|
|
|
slugs = self.parse_list_for_slugs(data, language)
|
|
|
|
items = self.parse_list(data, language)
|
|
|
|
total_slugs = len(slugs)
|
|
|
|
total_items = len(items)
|
|
|
|
if total_slugs == 0:
|
|
|
|
if total_items == 0:
|
|
|
|
raise Failed(f"Letterboxd Error: No List Items found in {data}")
|
|
|
|
raise Failed(f"Letterboxd Error: No List Items found in {data}")
|
|
|
|
length = 0
|
|
|
|
length = 0
|
|
|
|
for i, slug in enumerate(slugs, 1):
|
|
|
|
for i, item in enumerate(items, 1):
|
|
|
|
length = util.print_return(length, f"Finding TMDb ID {i}/{total_slugs}")
|
|
|
|
length = util.print_return(length, f"Finding TMDb ID {i}/{total_items}")
|
|
|
|
|
|
|
|
tmdb_id = None
|
|
|
|
|
|
|
|
expired = None
|
|
|
|
|
|
|
|
if self.Cache:
|
|
|
|
|
|
|
|
tmdb_id, expired = self.Cache.query_letterboxd_map(item[0])
|
|
|
|
|
|
|
|
if not tmdb_id or expired is not False:
|
|
|
|
try:
|
|
|
|
try:
|
|
|
|
movie_ids.append(self.get_tmdb_from_slug(slug, language))
|
|
|
|
tmdb_id = self.get_tmdb_from_slug(item[1], language)
|
|
|
|
|
|
|
|
if self.Cache:
|
|
|
|
|
|
|
|
self.Cache.update_letterboxd(expired, item[0], tmdb_id)
|
|
|
|
except Failed as e:
|
|
|
|
except Failed as e:
|
|
|
|
logger.error(e)
|
|
|
|
logger.error(e)
|
|
|
|
util.print_end(length, f"Processed {total_slugs} TMDb IDs")
|
|
|
|
movie_ids.append(tmdb_id)
|
|
|
|
|
|
|
|
util.print_end(length, f"Processed {total_items} TMDb IDs")
|
|
|
|
if status_message:
|
|
|
|
if status_message:
|
|
|
|
logger.debug(f"TMDb IDs Found: {movie_ids}")
|
|
|
|
logger.debug(f"TMDb IDs Found: {movie_ids}")
|
|
|
|
return movie_ids, []
|
|
|
|
return movie_ids, []
|
|
|
|