diff --git a/.github/.wordlist.txt b/.github/.wordlist.txt index 6e15b69e9..3a51508f9 100644 --- a/.github/.wordlist.txt +++ b/.github/.wordlist.txt @@ -252,6 +252,7 @@ SciFi ScudLee SDTV SemVer +setuptools ShawShank Skywalker Sohjiro diff --git a/CHANGELOG b/CHANGELOG index 51f78660b..a44708abc 100644 --- a/CHANGELOG +++ b/CHANGELOG @@ -1,5 +1,9 @@ # Requirements Update (requirements will need to be reinstalled) Updated PlexAPI requirement to 4.15.13 +Update lxml requirement to 5.2.2 +Update requests requirement to 2.32.3 +Update schedule requirement to 1.2.2 +Update setuptools requirement to 70.0.0 # Removed Features @@ -7,6 +11,8 @@ Updated PlexAPI requirement to 4.15.13 Checks requirement versions to print a message if one needs to be updated Added the `mass_added_at_update` operation to mass set the Added At field of Movies and Shows. Add automated Anime Aggregations for AniDB matching +Added `top_tamil`, `top_telugu`, `top_malayalam`, `trending_india`, `trending_tamil`, and `trending_telugu` as options for `imdb_chart` +Adds the `sort_by` attribute to `imdb_list` # Updates Changed the `overlay_artwork_filetype` Setting to accept `webp_lossy` and `webp_lossless` while the old attribute `webp` will be treated as `webp_lossy`. @@ -19,5 +25,7 @@ Fixes #2034 `anilist_userlist` `score` attribute wasn't being validated correctl Fixes #1367 Error when trying to symlink the logs folder Fixes #2028 TMDb IDs were being ignored on the report Fixes a bug when parsing a comma-separated string of ints +Fixes `imdb_chart` only getting 25 results +Fixes `imdb_list` not returning items Various other Minor Fixes \ No newline at end of file diff --git a/VERSION b/VERSION index e4a5d4c2b..9db4b4be6 100644 --- a/VERSION +++ b/VERSION @@ -1 +1 @@ -2.0.1-build36 +2.0.1-build37 diff --git a/docs/files/builders/imdb.md b/docs/files/builders/imdb.md index e800e2517..794bfd432 100644 --- a/docs/files/builders/imdb.md +++ b/docs/files/builders/imdb.md @@ -31,16 +31,22 @@ The expected input are the options below. Multiple values are supported as eithe The `sync_mode: sync` and `collection_order: custom` Setting are recommended since the lists are continuously updated and in a specific order. -| Name | Attribute | Works with Movies | Works with Shows | -|:-------------------------------------------------------------------------------|:-----------------|:------------------------------------------:|:------------------------------------------:| -| [Box Office](https://www.imdb.com/chart/boxoffice) | `box_office` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | -| [Most Popular Movies](https://www.imdb.com/chart/moviemeter) | `popular_movies` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | -| [Top 250 Movies](https://www.imdb.com/chart/top) | `top_movies` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | -| [Top Rated English Movies](https://www.imdb.com/chart/top-english-movies) | `top_english` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | -| [Most Popular TV Shows](https://www.imdb.com/chart/tvmeter) | `popular_shows` | :fontawesome-solid-circle-xmark:{ .red } | :fontawesome-solid-circle-check:{ .green } | -| [Top 250 TV Shows](https://www.imdb.com/chart/toptv) | `top_shows` | :fontawesome-solid-circle-xmark:{ .red } | :fontawesome-solid-circle-check:{ .green } | -| [Top Rated Indian Movies](https://www.imdb.com/india/top-rated-indian-movies/) | `top_indian` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | -| [Lowest Rated Movies](https://www.imdb.com/chart/bottom) | `lowest_rated` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| Name | Attribute | Works with Movies | Works with Shows | +|:-------------------------------------------------------------------------------------|:------------------|:------------------------------------------:|:------------------------------------------:| +| [Box Office](https://www.imdb.com/chart/boxoffice) | `box_office` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| [Most Popular Movies](https://www.imdb.com/chart/moviemeter) | `popular_movies` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| [Top 250 Movies](https://www.imdb.com/chart/top) | `top_movies` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| [Top Rated English Movies](https://www.imdb.com/chart/top-english-movies) | `top_english` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| [Most Popular TV Shows](https://www.imdb.com/chart/tvmeter) | `popular_shows` | :fontawesome-solid-circle-xmark:{ .red } | :fontawesome-solid-circle-check:{ .green } | +| [Top 250 TV Shows](https://www.imdb.com/chart/toptv) | `top_shows` | :fontawesome-solid-circle-xmark:{ .red } | :fontawesome-solid-circle-check:{ .green } | +| [Lowest Rated Movies](https://www.imdb.com/chart/bottom) | `lowest_rated` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| [Top Rated Indian Movies](https://www.imdb.com/india/top-rated-indian-movies/) | `top_indian` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| [Top Rated Tamil Movies](https://www.imdb.com/india/top-rated-tamil-movies/) | `top_tamil` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| [Top Rated Telugu Movies](https://www.imdb.com/india/top-rated-telugu-movies/) | `top_telugu` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| [Top Rated Malayalam Movies](https://www.imdb.com/india/top-rated-malayalam-movies/) | `top_malayalam` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| [Trending Indian Movies & Shows](https://www.imdb.com/india/upcoming/) | `trending_india` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-check:{ .green } | +| [Trending Tamil Movies](https://www.imdb.com/india/tamil/) | `trending_tamil` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | +| [Trending Telugu Movies](https://www.imdb.com/india/telugu/) | `trending_telugu` | :fontawesome-solid-circle-check:{ .green } | :fontawesome-solid-circle-xmark:{ .red } | ```yaml collections: @@ -62,34 +68,48 @@ collections: Finds every item in an IMDb List. -The expected input is an IMDb List URL. Multiple values are supported as a list only a comma-separated string will not work. +| List Parameter | Description | +|:---------------|:----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| `list_id` | Specify the IMDb List ID. **This attribute is required.**
**Options:** The ID that starts with `ls` found in the URL of the list. (ex. `ls005526372`) | +| `limit` | Specify how items you want returned by the query.
**Options:** Any Integer `0` or greater where `0` get all items.
**Default:** `0` | +| `sort_by` | Choose from one of the many available sort options.
**Options:** `custom.asc`, `custom.desc`, `title.asc`, `title.desc`, `rating.asc`, `rating.desc`, `popularity.asc`, `popularity.desc`, `votes.asc`, `votes.desc`, `release.asc`, `release.desc`, `runtime.asc`, `runtime.desc`, `added.asc`, `added.desc`
**Default:** `custom.asc` | + +Multiple values are supported as a list only a comma-separated string will not work. The `sync_mode: sync` and `collection_order: custom` Setting are recommended since the lists are continuously updated and in a specific order. ```yaml collections: James Bonds: - imdb_list: https://www.imdb.com/list/ls006405458 + imdb_list: + list_id: ls006405458 + limit: 100 + sort_by: rating.asc collection_order: custom sync_mode: sync ``` -You can also limit the number of items to search for by using the `limit` and `url` parameters under `imdb_list`. +You can search multiple lists in one collection by using a list. ```yaml collections: Christmas: imdb_list: - - url: https://www.imdb.com/list/ls025976544/ + - list_id: ls025976544 limit: 10 - - url: https://www.imdb.com/list/ls003863000/ + sort_by: rating.asc + - list_id: ls003863000 limit: 10 - - url: https://www.imdb.com/list/ls027454200/ + sort_by: rating.asc + - list_id: ls027454200 limit: 10 - - url: https://www.imdb.com/list/ls027886673/ + sort_by: rating.asc + - list_id: ls027886673 limit: 10 - - url: https://www.imdb.com/list/ls097998599/ + sort_by: rating.asc + - list_id: ls097998599 limit: 10 + sort_by: rating.asc sync_mode: sync collection_order: alpha ``` @@ -184,7 +204,7 @@ The `sync_mode: sync` and `collection_order: custom` Setting are recommended sin | Search Parameter | Description | |:------------------------|:--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| -| `limit` | Specify how items you want returned by the query.
**Options:** Any Integer greater than `0`
**Default:** `100` | +| `limit` | Specify how items you want returned by the query.
**Options:** Any Integer `0` or greater where `0` get all items.
**Default:** `100` | | `sort_by` | Choose from one of the many available sort options.
**Options:** `popularity.asc`, `popularity.desc`, `title.asc`, `title.desc`, `rating.asc`, `rating.desc`, `votes.asc`, `votes.desc`, `box_office.asc`, `box_office.desc`, `runtime.asc`, `runtime.desc`, `year.asc`, `year.desc`, `release.asc`, `release.desc`
**Default:** `popularity.asc` | | `title` | Search by title name.
**Options:** Any String | | `type` | Item must match at least one given type. Can be a comma-separated list.
**Options:** `movie`, `tv_series`, `short`, `tv_episode`, `tv_mini_series`, `tv_movie`, `tv_special`, `tv_short`, `video_game`, `video`, `music_video`, `podcast_series`, `podcast_episode` | diff --git a/docs/files/filters.md b/docs/files/filters.md index dd9f67ded..38ce9dde0 100644 --- a/docs/files/filters.md +++ b/docs/files/filters.md @@ -241,7 +241,8 @@ collections: ```yaml collections: Daniel Craig only James Bonds: - imdb_list: https://www.imdb.com/list/ls006405458/ + imdb_list: + list_id: ls006405458 filters: actor: Daniel Craig ``` diff --git a/modules/builder.py b/modules/builder.py index 9f102d455..8f41f0e8d 100644 --- a/modules/builder.py +++ b/modules/builder.py @@ -1479,7 +1479,7 @@ def _imdb(self, method_name, method_data): raise Failed(f"{self.Type} Error: imdb_id {value} must begin with tt") elif method_name == "imdb_list": try: - for imdb_dict in self.config.IMDb.validate_imdb_lists(self.Type, method_data, self.language): + for imdb_dict in self.config.IMDb.validate_imdb_lists(self.Type, method_data): self.builders.append((method_name, imdb_dict)) except Failed as e: logger.error(e) @@ -1739,9 +1739,8 @@ def _mal(self, method_name, method_data): final_attributes["letter"] = util.parse(self.Type, "prefix", dict_data, methods=dict_methods, parent=method_name) final_text += f"\nPrefix: {final_attributes['letter']}" if "type" in dict_methods: - type_list = util.parse(self.Type, "type", dict_data, datatype="commalist", methods=dict_methods, parent=method_name, options=mal.search_types) - final_attributes["type"] = ",".join(type_list) - final_text += f"\nType: {' or '.join(type_list)}" + final_attributes["type"] = util.parse(self.Type, "type", dict_data, methods=dict_methods, parent=method_name, options=mal.search_types) + final_text += f"\nType: {final_attributes['type']}" if "status" in dict_methods: final_attributes["status"] = util.parse(self.Type, "status", dict_data, methods=dict_methods, parent=method_name, options=mal.search_status) final_text += f"\nStatus: {final_attributes['status']}" diff --git a/modules/imdb.py b/modules/imdb.py index 2878900a4..d833fd4ed 100644 --- a/modules/imdb.py +++ b/modules/imdb.py @@ -6,8 +6,11 @@ logger = util.logger builders = ["imdb_list", "imdb_id", "imdb_chart", "imdb_watchlist", "imdb_search", "imdb_award"] -movie_charts = ["box_office", "popular_movies", "top_movies", "top_english", "top_indian", "lowest_rated"] -show_charts = ["popular_shows", "top_shows"] +movie_charts = [ + "box_office", "popular_movies", "top_movies", "top_english", "lowest_rated", + "top_indian", "top_tamil", "top_telugu", "top_malayalam", "trending_india", "trending_tamil", "trending_telugu" +] +show_charts = ["popular_shows", "top_shows", "trending_india"] charts = { "box_office": "Box Office", "popular_movies": "Most Popular Movies", @@ -15,8 +18,30 @@ "top_movies": "Top 250 Movies", "top_shows": "Top 250 TV Shows", "top_english": "Top Rated English Movies", + "lowest_rated": "Lowest Rated Movies", + "top_tamil": "Top Rated Tamil Movies", + "top_telugu": "Top Rated Telugu Movies", + "top_malayalam": "Top Rated Malayalam Movies", + "trending_india": "Trending Indian Movies & Shows", + "trending_tamil": "Trending Tamil Movies", + "trending_telugu": "Trending Telugu Movies", "top_indian": "Top Rated Indian Movies", - "lowest_rated": "Lowest Rated Movies" +} +chart_urls = { + "box_office": "chart/boxoffice", + "popular_movies": "chart/moviemeter", + "popular_shows": "chart/tvmeter", + "top_movies": "chart/top", + "top_shows": "chart/toptv", + "top_english": "chart/top-english-movies", + "lowest_rated": "chart/bottom", + "top_indian": "india/top-rated-indian-movies", + "top_tamil": "india/top-rated-tamil-movies", + "top_telugu": "india/top-rated-telugu-movies", + "top_malayalam": "india/top-rated-malayalam-movies", + "trending_india": "india/upcoming", + "trending_tamil": "india/tamil", + "trending_telugu": "india/telugu", } imdb_search_attributes = [ "limit", "sort_by", "title", "type", "type.not", "release.after", "release.before", "rating.gte", "rating.lte", @@ -40,6 +65,17 @@ "release": "RELEASE_DATE", } sort_options = [f"{a}.{d}"for a in sort_by_options for d in ["asc", "desc"]] +list_sort_by_options = { + "custom": "LIST_ORDER", + "popularity": "POPULARITY", + "title": "TITLE_REGIONAL", + "rating": "USER_RATING", + "votes": "USER_RATING_COUNT", + "runtime": "RUNTIME", + "added": "DATE_ADDED", + "release": "RELEASE_DATE", +} +list_sort_options = [f"{a}.{d}"for a in sort_by_options for d in ["asc", "desc"]] title_type_options = { "movie": "movie", "tv_series": "tvSeries", "short": "short", "tv_episode": "tvEpisode", "tv_mini_series": "tvMiniSeries", "tv_movie": "tvMovie", "tv_special": "tvSpecial", "tv_short": "tvShort", "video_game": "videoGame", "video": "video", @@ -89,7 +125,8 @@ } base_url = "https://www.imdb.com" git_base = "https://raw.githubusercontent.com/Kometa-Team/IMDb-Awards/master" -hash_url = "https://raw.githubusercontent.com/Kometa-Team/IMDb-Hash/master/HASH" +search_hash_url = "https://raw.githubusercontent.com/Kometa-Team/IMDb-Hash/master/HASH" +list_hash_url = "https://raw.githubusercontent.com/Kometa-Team/IMDb-Hash/master/LIST_HASH" graphql_url = "https://api.graphql.imdb.com/" list_url = f"{base_url}/list/ls" @@ -103,7 +140,8 @@ def __init__(self, requests, cache, default_dir): self._episode_ratings = None self._events_validation = None self._events = {} - self._hash = None + self._search_hash = None + self._list_hash = None self.event_url_validation = {} def _request(self, url, language=None, xpath=None, params=None): @@ -117,10 +155,16 @@ def _graph_request(self, json_data): return self.requests.post_json(graphql_url, headers={"content-type": "application/json"}, json=json_data) @property - def hash(self): - if self._hash is None: - self._hash = self.requests.get(hash_url).text.strip() - return self._hash + def search_hash(self): + if self._search_hash is None: + self._search_hash = self.requests.get(search_hash_url).text.strip() + return self._search_hash + + @property + def list_hash(self): + if self._list_hash is None: + self._list_hash = self.requests.get(list_hash_url).text.strip() + return self._list_hash @property def events_validation(self): @@ -133,26 +177,29 @@ def get_event(self, event_id): self._events[event_id] = self.requests.get_yaml(f"{git_base}/events/{event_id}.yml").data return self._events[event_id] - def validate_imdb_lists(self, err_type, imdb_lists, language): + def validate_imdb_lists(self, err_type, imdb_lists): valid_lists = [] for imdb_dict in util.get_list(imdb_lists, split=False): if not isinstance(imdb_dict, dict): - imdb_dict = {"url": imdb_dict} + imdb_dict = {"list_id": imdb_dict} + if "url" in imdb_dict and "list_id" not in imdb_dict: + imdb_dict["list_id"] = imdb_dict["url"] dict_methods = {dm.lower(): dm for dm in imdb_dict} - if "url" not in dict_methods: - raise Failed(f"{err_type} Error: imdb_list url attribute not found") - elif imdb_dict[dict_methods["url"]] is None: - raise Failed(f"{err_type} Error: imdb_list url attribute is blank") + if "list_id" not in dict_methods: + raise Failed(f"{err_type} Error: imdb_list list_id attribute not found") + elif imdb_dict[dict_methods["list_id"]] is None: + raise Failed(f"{err_type} Error: imdb_list list_id attribute is blank") else: - imdb_url = imdb_dict[dict_methods["url"]].strip() - if imdb_url.startswith(f"{base_url}/search/"): - raise Failed("IMDb Error: URLs with https://www.imdb.com/search/ no longer works with imdb_list use imdb_search.") - if imdb_url.startswith(f"{base_url}/filmosearch/"): - raise Failed("IMDb Error: URLs with https://www.imdb.com/filmosearch/ no longer works with imdb_list use imdb_search.") - if not imdb_url.startswith(list_url): - raise Failed(f"IMDb Error: imdb_list URLs must begin with {list_url}") - self._total(imdb_url, language) - list_count = None + imdb_url = imdb_dict[dict_methods["list_id"]].strip() + if imdb_url.startswith(f"{base_url}/search/"): + raise Failed("IMDb Error: URLs with https://www.imdb.com/search/ no longer works with imdb_list use imdb_search.") + if imdb_url.startswith(f"{base_url}/filmosearch/"): + raise Failed("IMDb Error: URLs with https://www.imdb.com/filmosearch/ no longer works with imdb_list use imdb_search.") + search = re.search(r"(ls\d+)", imdb_url) + if not search: + raise Failed("IMDb Error: imdb_list list_id must begin with ls (ex. ls005526372)") + new_dict = {"list_id": search.group(1)} + if "limit" in dict_methods: if imdb_dict[dict_methods["limit"]] is None: logger.warning(f"{err_type} Warning: imdb_list limit attribute is blank using 0 as default") @@ -160,14 +207,18 @@ def validate_imdb_lists(self, err_type, imdb_lists, language): try: value = int(str(imdb_dict[dict_methods["limit"]])) if 0 <= value: - list_count = value + new_dict["limit"] = value except ValueError: pass - if list_count is None: - logger.warning(f"{err_type} Warning: imdb_list limit attribute must be an integer 0 or greater using 0 as default") - if list_count is None: - list_count = 0 - valid_lists.append({"url": imdb_url, "limit": list_count}) + if "limit" not in new_dict: + logger.warning(f"{err_type} Warning: imdb_list limit attribute: {imdb_dict[dict_methods['limit']]} must be an integer 0 or greater using 0 as default") + if "limit" not in new_dict: + new_dict["limit"] = 0 + + if "sort_by" in dict_methods: + new_dict["sort_by"] = util.parse(err_type, dict_methods, imdb_dict, parent="imdb_list", default="custom.asc", options=list_sort_options) + + valid_lists.append(new_dict) return valid_lists def validate_imdb_watchlists(self, err_type, users, language): @@ -220,63 +271,12 @@ def _watchlist(self, user, language): return [f for f in json.loads(jsonline[jsonline.find('{'):-2])["starbars"]] raise Failed(f"IMDb Error: Failed to parse URL: {imdb_url}") - def _total(self, imdb_url, language): - xpath_total = "//div[@class='desc lister-total-num-results']/text()" - per_page = 100 - results = self._request(imdb_url, language=language, xpath=xpath_total) - total = 0 - for result in results: - if "title" in result: - try: - total = int(re.findall("(\\d+) title", result.replace(",", ""))[0]) - break - except IndexError: - pass - if total > 0: - return total, per_page - raise Failed(f"IMDb Error: Failed to parse URL: {imdb_url}") - - def _ids_from_url(self, imdb_url, language, limit): - total, item_count = self._total(imdb_url, language) - imdb_ids = [] - parsed_url = urlparse(imdb_url) - params = parse_qs(parsed_url.query) - imdb_base = parsed_url._replace(query=None).geturl() # noqa - params.pop("start", None) # noqa - params.pop("count", None) # noqa - params.pop("page", None) # noqa - logger.trace(f"URL: {imdb_base}") - logger.trace(f"Params: {params}") - if limit < 1 or total < limit: - limit = total - remainder = limit % item_count - if remainder == 0: - remainder = item_count - num_of_pages = math.ceil(int(limit) / item_count) - for i in range(1, num_of_pages + 1): - start_num = (i - 1) * item_count + 1 - logger.ghost(f"Parsing Page {i}/{num_of_pages} {start_num}-{limit if i == num_of_pages else i * item_count}") - params["page"] = i # noqa - ids_found = self._request(imdb_base, language=language, xpath="//div[contains(@class, 'lister-item-image')]//a/img//@data-tconst", params=params) - if i == num_of_pages: - ids_found = ids_found[:remainder] - imdb_ids.extend(ids_found) - time.sleep(2) - logger.exorcise() - if len(imdb_ids) > 0: - return imdb_ids - raise Failed(f"IMDb Error: No IMDb IDs Found at {imdb_url}") - - def _search_json(self, data): + def _graphql_json(self, data, search=True): + page_limit = 250 if search else 100 out = { "locale": "en-US", - "first": data["limit"] if "limit" in data and 0 < data["limit"] < 250 else 250, - "titleTypeConstraint": {"anyTitleTypeIds": [title_type_options[t] for t in data["type"]] if "type" in data else []}, + "first": data["limit"] if "limit" in data and 0 < data["limit"] < page_limit else page_limit, } - sort = data["sort_by"] if "sort_by" in data else "popularity.asc" - sort_by, sort_order = sort.split(".") - out["sortBy"] = sort_by_options[sort_by] - out["sortOrder"] = sort_order.upper() def check_constraint(bases, mods, constraint, lower="", translation=None, range_name=None): if not isinstance(bases, list): @@ -302,84 +302,96 @@ def check_constraint(bases, mods, constraint, lower="", translation=None, range_ if range_data: out[constraint][range_name[i]] = range_data - check_constraint("type", [("not", "excludeTitleTypeIds")], "titleTypeConstraint", translation=title_type_options) - check_constraint("release", [("after", "start"), ("before", "end")], "releaseDateConstraint", range_name="releaseDateRange") - check_constraint("title", [("", "searchTerm")], "titleTextConstraint") - check_constraint(["rating", "votes"], [("gte", "min"), ("lte", "max")], "userRatingsConstraint", range_name=["aggregateRatingRange", "ratingsCountRange"]) - check_constraint("genre", [("", "all"), ("any", "any"), ("not", "exclude")], "genreConstraint", lower="GenreIds", translation=genre_options) - check_constraint("topic", [("", "all"), ("any", "any"), ("not", "no")], "withTitleDataConstraint", lower="DataAvailable", translation=topic_options) - check_constraint("alternate_version", [("", "all"), ("any", "any")], "alternateVersionMatchingConstraint", lower="AlternateVersionTextTerms") - check_constraint("crazy_credit", [("", "all"), ("any", "any")], "crazyCreditMatchingConstraint", lower="CrazyCreditTextTerms") - check_constraint("location", [("", "all"), ("any", "any")], "filmingLocationConstraint", lower="Locations") - check_constraint("goof", [("", "all"), ("any", "any")], "goofMatchingConstraint", lower="GoofTextTerms") - check_constraint("plot", [("", "all"), ("any", "any")], "plotMatchingConstraint", lower="PlotTextTerms") - check_constraint("quote", [("", "all"), ("any", "any")], "quoteMatchingConstraint", lower="QuoteTextTerms") - check_constraint("soundtrack", [("", "all"), ("any", "any")], "soundtrackMatchingConstraint", lower="SoundtrackTextTerms") - check_constraint("trivia", [("", "all"), ("any", "any")], "triviaMatchingConstraint", lower="TriviaTextTerms") - - if "event" in data or "event.winning" in data: - input_list = [] - if "event" in data: - input_list.extend([event_options[a] if a in event_options else {"eventId": a} for a in data["event"]]) - if "event.winning" in data: - for a in data["event.winning"]: - award_dict = event_options[a] if a in event_options else {"eventId": a} - award_dict["winnerFilter"] = "WINNER_ONLY" - input_list.append(award_dict) - out["awardConstraint"] = {"allEventNominations": input_list} - - if any([a in data for a in ["imdb_top", "imdb_bottom", "popularity.gte", "popularity.lte"]]): - ranges = [] - if "imdb_top" in data: - ranges.append({"rankRange": {"max": data["imdb_top"]}, "rankedTitleListType": "TOP_RATED_MOVIES"}) - if "imdb_bottom" in data: - ranges.append({"rankRange": {"max": data["imdb_bottom"]}, "rankedTitleListType": "LOWEST_RATED_MOVIES"}) - if "popularity.gte" in data or "popularity.lte" in data: - num_range = {} - if "popularity.lte" in data: - num_range["max"] = data["popularity.lte"] - if "popularity.gte" in data: - num_range["min"] = data["popularity.gte"] - ranges.append({"rankRange": num_range, "rankedTitleListType": "TITLE_METER"}) - out["rankedTitleListConstraint"] = {"allRankedTitleLists": ranges} - - check_constraint("series", [("", "any"), ("not", "exclude")], "episodicConstraint", lower="SeriesIds") - check_constraint("list", [("", "inAllLists"), ("any", "inAnyList"), ("not", "notInAnyList")], "listConstraint") - - if "company" in data: - company_ids = [] - for c in data["company"]: - if c in company_options: - company_ids.extend(company_options[c]) - else: - company_ids.append(c) - out["creditedCompanyConstraint"] = {"anyCompanyIds": company_ids} - - check_constraint("content_rating", [("", "anyRegionCertificateRatings")], "certificateConstraint") - check_constraint("country", [("", "all"), ("any", "any"), ("not", "exclude"), ("origin", "anyPrimary")], "originCountryConstraint", lower="Countries") - check_constraint("keyword", [("", "all"), ("any", "any"), ("not", "exclude")], "keywordConstraint", lower="Keywords", translation=(" ", "-")) - check_constraint("language", [("", "all"), ("any", "any"), ("not", "exclude"), ("primary", "anyPrimary")], "languageConstraint", lower="Languages") - check_constraint("cast", [("", "all"), ("any", "any"), ("not", "exclude")], "creditedNameConstraint", lower="NameIds") - check_constraint("runtime", [("gte", "min"), ("lte", "max")], "runtimeConstraint", range_name="runtimeRangeMinutes") + sort = data["sort_by"] if "sort_by" in data else "popularity.asc" if search else "custom.asc" + sort_by, sort_order = sort.split(".") - if "adult" in data and data["adult"]: - out["explicitContentConstraint"] = {"explicitContentFilter": "INCLUDE_ADULT"} + if search: + out["titleTypeConstraint"] = {"anyTitleTypeIds": [title_type_options[t] for t in data["type"]] if "type" in data else []} + out["sortBy"] = sort_by_options[sort_by] + out["sortOrder"] = sort_order.upper() + + check_constraint("type", [("not", "excludeTitleTypeIds")], "titleTypeConstraint", translation=title_type_options) + check_constraint("release", [("after", "start"), ("before", "end")], "releaseDateConstraint", range_name="releaseDateRange") + check_constraint("title", [("", "searchTerm")], "titleTextConstraint") + check_constraint(["rating", "votes"], [("gte", "min"), ("lte", "max")], "userRatingsConstraint", range_name=["aggregateRatingRange", "ratingsCountRange"]) + check_constraint("genre", [("", "all"), ("any", "any"), ("not", "exclude")], "genreConstraint", lower="GenreIds", translation=genre_options) + check_constraint("topic", [("", "all"), ("any", "any"), ("not", "no")], "withTitleDataConstraint", lower="DataAvailable", translation=topic_options) + check_constraint("alternate_version", [("", "all"), ("any", "any")], "alternateVersionMatchingConstraint", lower="AlternateVersionTextTerms") + check_constraint("crazy_credit", [("", "all"), ("any", "any")], "crazyCreditMatchingConstraint", lower="CrazyCreditTextTerms") + check_constraint("location", [("", "all"), ("any", "any")], "filmingLocationConstraint", lower="Locations") + check_constraint("goof", [("", "all"), ("any", "any")], "goofMatchingConstraint", lower="GoofTextTerms") + check_constraint("plot", [("", "all"), ("any", "any")], "plotMatchingConstraint", lower="PlotTextTerms") + check_constraint("quote", [("", "all"), ("any", "any")], "quoteMatchingConstraint", lower="QuoteTextTerms") + check_constraint("soundtrack", [("", "all"), ("any", "any")], "soundtrackMatchingConstraint", lower="SoundtrackTextTerms") + check_constraint("trivia", [("", "all"), ("any", "any")], "triviaMatchingConstraint", lower="TriviaTextTerms") + + if "event" in data or "event.winning" in data: + input_list = [] + if "event" in data: + input_list.extend([event_options[a] if a in event_options else {"eventId": a} for a in data["event"]]) + if "event.winning" in data: + for a in data["event.winning"]: + award_dict = event_options[a] if a in event_options else {"eventId": a} + award_dict["winnerFilter"] = "WINNER_ONLY" + input_list.append(award_dict) + out["awardConstraint"] = {"allEventNominations": input_list} + + if any([a in data for a in ["imdb_top", "imdb_bottom", "popularity.gte", "popularity.lte"]]): + ranges = [] + if "imdb_top" in data: + ranges.append({"rankRange": {"max": data["imdb_top"]}, "rankedTitleListType": "TOP_RATED_MOVIES"}) + if "imdb_bottom" in data: + ranges.append({"rankRange": {"max": data["imdb_bottom"]}, "rankedTitleListType": "LOWEST_RATED_MOVIES"}) + if "popularity.gte" in data or "popularity.lte" in data: + num_range = {} + if "popularity.lte" in data: + num_range["max"] = data["popularity.lte"] + if "popularity.gte" in data: + num_range["min"] = data["popularity.gte"] + ranges.append({"rankRange": num_range, "rankedTitleListType": "TITLE_METER"}) + out["rankedTitleListConstraint"] = {"allRankedTitleLists": ranges} + + check_constraint("series", [("", "any"), ("not", "exclude")], "episodicConstraint", lower="SeriesIds") + check_constraint("list", [("", "inAllLists"), ("any", "inAnyList"), ("not", "notInAnyList")], "listConstraint") + + if "company" in data: + company_ids = [] + for c in data["company"]: + if c in company_options: + company_ids.extend(company_options[c]) + else: + company_ids.append(c) + out["creditedCompanyConstraint"] = {"anyCompanyIds": company_ids} + + check_constraint("content_rating", [("", "anyRegionCertificateRatings")], "certificateConstraint") + check_constraint("country", [("", "all"), ("any", "any"), ("not", "exclude"), ("origin", "anyPrimary")], "originCountryConstraint", lower="Countries") + check_constraint("keyword", [("", "all"), ("any", "any"), ("not", "exclude")], "keywordConstraint", lower="Keywords", translation=(" ", "-")) + check_constraint("language", [("", "all"), ("any", "any"), ("not", "exclude"), ("primary", "anyPrimary")], "languageConstraint", lower="Languages") + check_constraint("cast", [("", "all"), ("any", "any"), ("not", "exclude")], "creditedNameConstraint", lower="NameIds") + check_constraint("runtime", [("gte", "min"), ("lte", "max")], "runtimeConstraint", range_name="runtimeRangeMinutes") + + if "adult" in data and data["adult"]: + out["explicitContentConstraint"] = {"explicitContentFilter": "INCLUDE_ADULT"} + else: + out["lsConst"] = data["list_id"] + out["sort"] = {"by": list_sort_by_options[sort_by], "order": sort_order.upper()} logger.trace(out) return { - "operationName": "AdvancedTitleSearch", + "operationName": "AdvancedTitleSearch" if search else "TitleListMainPage", "variables": out, - "extensions": {"persistedQuery": {"version": 1, "sha256Hash": self.hash}} + "extensions": {"persistedQuery": {"version": 1, "sha256Hash": self.search_hash if search else self.list_hash}} } - def _search(self, data): - json_obj = self._search_json(data) - item_count = 250 + def _pagination(self, data, search=True): + json_obj = self._graphql_json(data, search=search) + item_count = 250 if search else 100 imdb_ids = [] logger.ghost("Parsing Page 1") response_json = self._graph_request(json_obj) try: - total = response_json["data"]["advancedTitleSearch"]["total"] + search_data = response_json["data"]["advancedTitleSearch"] if search else response_json["data"]["list"]["titleListItemSearch"] + total = search_data["total"] limit = data["limit"] if limit < 1 or total < limit: limit = total @@ -387,16 +399,17 @@ def _search(self, data): if remainder == 0: remainder = item_count num_of_pages = math.ceil(int(limit) / item_count) - end_cursor = response_json["data"]["advancedTitleSearch"]["pageInfo"]["endCursor"] - imdb_ids.extend([n["node"]["title"]["id"] for n in response_json["data"]["advancedTitleSearch"]["edges"]]) + end_cursor = search_data["pageInfo"]["endCursor"] + imdb_ids.extend([n["node"]["title"]["id"] if search else n["listItem"]["id"] for n in search_data["edges"]]) if num_of_pages > 1: for i in range(2, num_of_pages + 1): start_num = (i - 1) * item_count + 1 logger.ghost(f"Parsing Page {i}/{num_of_pages} {start_num}-{limit if i == num_of_pages else i * item_count}") json_obj["variables"]["after"] = end_cursor response_json = self._graph_request(json_obj) - end_cursor = response_json["data"]["advancedTitleSearch"]["pageInfo"]["endCursor"] - ids_found = [n["node"]["title"]["id"] for n in response_json["data"]["advancedTitleSearch"]["edges"]] + search_data = response_json["data"]["advancedTitleSearch"] if search else response_json["data"]["list"]["titleListItemSearch"] + end_cursor = search_data["pageInfo"]["endCursor"] + ids_found = [n["node"]["title"]["id"] if search else n["listItem"]["id"] for n in search_data["edges"]] if i == num_of_pages: ids_found = ids_found[:remainder] imdb_ids.extend(ids_found) @@ -489,35 +502,22 @@ def parental_guide(self, imdb_id, ignore_cache=False): return parental_dict def _ids_from_chart(self, chart, language): - if chart == "box_office": - url = "chart/boxoffice" - elif chart == "popular_movies": - url = "chart/moviemeter" - elif chart == "popular_shows": - url = "chart/tvmeter" - elif chart == "top_movies": - url = "chart/top" - elif chart == "top_shows": - url = "chart/toptv" - elif chart == "top_english": - url = "chart/top-english-movies" - elif chart == "top_indian": - url = "india/top-rated-indian-movies" - elif chart == "lowest_rated": - url = "chart/bottom" - else: + if chart not in chart_urls: raise Failed(f"IMDb Error: chart: {chart} not ") - links = self._request(f"{base_url}/{url}", language=language, xpath="//li//a[@class='ipc-title-link-wrapper']/@href") - return [re.search("(tt\\d+)", link).group(1) for link in links] + script_data = self._request(f"{base_url}/{chart_urls[chart]}", language=language, xpath="//script[@id='__NEXT_DATA__']/text()")[0] + return [x.group(1) for x in re.finditer(r'"(tt\d+)"', script_data)] def get_imdb_ids(self, method, data, language): if method == "imdb_id": logger.info(f"Processing IMDb ID: {data}") return [(data, "imdb")] elif method == "imdb_list": - status = f"{data['limit']} Items at " if data['limit'] > 0 else '' - logger.info(f"Processing IMDb List: {status}{data['url']}") - return [(i, "imdb") for i in self._ids_from_url(data["url"], language, data["limit"])] + logger.info(f"Processing IMDb List: {data['list_id']}") + if data["limit"] > 0: + logger.info(f" Limit: {data['limit']}") + if "sort_by" in data: + logger.info(f" Sort By: {data['sort_by']}") + return [(i, "imdb") for i in self._pagination(data, search=False)] elif method == "imdb_chart": logger.info(f"Processing IMDb Chart: {charts[data]}") return [(_i, "imdb") for _i in self._ids_from_chart(data, language)] @@ -538,7 +538,7 @@ def get_imdb_ids(self, method, data, language): logger.info(f"Processing IMDb Search:") for k, v in data.items(): logger.info(f" {k}: {v}") - return [(_i, "imdb") for _i in self._search(data)] + return [(_i, "imdb") for _i in self._pagination(data)] else: raise Failed(f"IMDb Error: Method {method} not supported") diff --git a/modules/letterboxd.py b/modules/letterboxd.py index 7410680fc..17a711ff2 100644 --- a/modules/letterboxd.py +++ b/modules/letterboxd.py @@ -56,7 +56,7 @@ def _tmdb(self, letterboxd_url, language): def get_list_description(self, list_url, language): logger.trace(f"URL: {list_url}") response = self.requests.get_html(list_url, language=language) - descriptions = response.xpath("//meta[@property='og:description']/@content") + descriptions = response.xpath("//meta[@name='description']/@content") if len(descriptions) > 0 and len(descriptions[0]) > 0 and "About this list: " in descriptions[0]: return str(descriptions[0]).split("About this list: ")[1] return None