From 4992e07aa489c7b2da627ab57769c4b0b03c6e62 Mon Sep 17 00:00:00 2001 From: Sebastian Golasch Date: Fri, 17 Mar 2017 10:10:52 +0100 Subject: [PATCH] chore(performance): Speeds up display of main menu, genre and recommendation list --- addon.xml | 6 +- resources/language/English/strings.po | 2 +- resources/language/German/strings.po | 2 +- resources/language/Slovak/strings.po | 2 +- resources/language/Spanish/strings.po | 2 +- resources/lib/Navigation.py | 18 +---- .../lib/NetflixHttpSubRessourceHandler.py | 19 +----- resources/lib/NetflixSession.py | 65 ++++++++----------- 8 files changed, 36 insertions(+), 80 deletions(-) diff --git a/addon.xml b/addon.xml index a725770cc..44635ecc9 100644 --- a/addon.xml +++ b/addon.xml @@ -1,5 +1,5 @@ - + @@ -28,7 +28,7 @@ MIT http://www.kodinerds.net/index.php/Thread/55607-Inputstream-Agile-Betatest-Netflix/ https://github.com/asciidisco/plugin.video.netflix - v0.11.5 (2017-3-16) - - Remove flawed debugging code + v0.11.5 (2017-3-17) + - Speed up display of main menu, genre and recommendation list diff --git a/resources/language/English/strings.po b/resources/language/English/strings.po index b682e9f70..98150a6d0 100644 --- a/resources/language/English/strings.po +++ b/resources/language/English/strings.po @@ -1,7 +1,7 @@ # Kodi Media Center language file # Addon Name: Netflix # Addon id: plugin.video.netflix -# Addon version: 0.11.5 +# Addon version: 0.11.6 # Addon Provider: libdev + jojo + asciidisco msgid "" msgstr "" diff --git a/resources/language/German/strings.po b/resources/language/German/strings.po index e0821560a..64662b3b6 100644 --- a/resources/language/German/strings.po +++ b/resources/language/German/strings.po @@ -1,7 +1,7 @@ # Kodi Media Center language file # Addon Name: Netflix # Addon id: plugin.video.netflix -# Addon version: 0.11.5 +# Addon version: 0.11.6 # Addon Provider: libdev + jojo + asciidisco msgid "" msgstr "" diff --git a/resources/language/Slovak/strings.po b/resources/language/Slovak/strings.po index f16be5c9a..243b72e42 100644 --- a/resources/language/Slovak/strings.po +++ b/resources/language/Slovak/strings.po @@ -1,7 +1,7 @@ # Kodi Media Center language file # Addon Name: Netflix # Addon id: plugin.video.netflix -# Addon version: 0.11.5 +# Addon version: 0.11.6 # Addon Provider: libdev + jojo + asciidisco msgid "" msgstr "" diff --git a/resources/language/Spanish/strings.po b/resources/language/Spanish/strings.po index 4b1cae95c..0fb62f6c9 100644 --- a/resources/language/Spanish/strings.po +++ b/resources/language/Spanish/strings.po @@ -1,7 +1,7 @@ # Kodi Media Center language file # Addon Name: Netflix # Addon id: plugin.video.netflix -# Addon version: 0.11.5 +# Addon version: 0.11.6 # Addon Provider: libdev + jojo + asciidisco msgid "" msgstr "" diff --git a/resources/lib/Navigation.py b/resources/lib/Navigation.py index 1c9a9ab97..894fc81e9 100644 --- a/resources/lib/Navigation.py +++ b/resources/lib/Navigation.py @@ -163,13 +163,7 @@ def show_user_list (self, type): """ # determine if we´re in kids mode user_data = self.call_netflix_service({'method': 'get_user_data'}) - profiles = self.call_netflix_service({'method': 'list_profiles'}) - is_kids = profiles.get(user_data['guid']).get('isKids', False) - # fetch video lists - if is_kids == True: - video_list_ids = self.call_netflix_service({'method': 'fetch_video_list_ids_for_kids'}) - else: - video_list_ids = self.call_netflix_service({'method': 'fetch_video_list_ids', 'type': type}) + video_list_ids = self.call_netflix_service({'method': 'fetch_video_list_ids', 'guid': user_data['guid'], 'cache': True}) # check for any errors if self._is_dirty_response(response=video_list_ids): return False @@ -246,16 +240,8 @@ def show_video_list (self, video_list_id, type): def show_video_lists (self): """List the users video lists (recommendations, my list, etc.)""" - # determine if we´re in Kids profile mode user_data = self.call_netflix_service({'method': 'get_user_data'}) - profiles = self.call_netflix_service({'method': 'list_profiles'}) - is_kids = profiles.get(user_data['guid']).get('isKids', False) - # fetch video lists - if is_kids == True: - video_list_ids = self.call_netflix_service({'method': 'fetch_video_list_ids_for_kids', 'guid': user_data['guid'], 'cache': True}) - else: - video_list_ids = self.call_netflix_service({'method': 'fetch_video_list_ids', 'guid': user_data['guid'], 'cache': True}) - + video_list_ids = self.call_netflix_service({'method': 'fetch_video_list_ids', 'guid': user_data['guid'], 'cache': True}) # check for any errors if self._is_dirty_response(response=video_list_ids): return False diff --git a/resources/lib/NetflixHttpSubRessourceHandler.py b/resources/lib/NetflixHttpSubRessourceHandler.py index 5123ceb5d..b1bad7a51 100644 --- a/resources/lib/NetflixHttpSubRessourceHandler.py +++ b/resources/lib/NetflixHttpSubRessourceHandler.py @@ -154,28 +154,11 @@ def fetch_video_list_ids (self, params): self.kodi_helper.log('Serving cached list for user: ' + self.netflix_session.user_data['guid']) return cached_list video_list_ids_raw = self.netflix_session.fetch_video_list_ids() + if 'error' in video_list_ids_raw: return video_list_ids_raw return self.netflix_session.parse_video_list_ids(response_data=video_list_ids_raw) - def fetch_video_list_ids_for_kids (self, params): - """Video list ids proxy function (thanks to Netflix that we need to use a different API for Kids profiles) - - Parameters - ---------- - params : :obj:`dict` of :obj:`str` - Request params - - Returns - ------- - :obj:`list` - Transformed response of the remote call - """ - if self.lolomo == None: - self.lolomo = self.netflix_session.get_lolomo_for_kids() - response = self.netflix_session.fetch_lists_for_kids(lolomo=self.lolomo) - return response - def fetch_video_list (self, params): """Video list proxy function diff --git a/resources/lib/NetflixSession.py b/resources/lib/NetflixSession.py index 377866765..f4c9be3fd 100644 --- a/resources/lib/NetflixSession.py +++ b/resources/lib/NetflixSession.py @@ -446,6 +446,11 @@ def parse_video_list_ids (self, response_data): for key in self.video_list_keys: video_list_ids[key] = {} + # check if the list items are hidden behind a `value` sub key + # this is the case when we fetch the lists via POST, not via a GET preflight request + if 'value' in response_data.keys(): + response_data = response_data['value'] + # subcatogorize the lists by their context video_lists = response_data['lists'] for video_list_id in video_lists.keys(): @@ -1270,8 +1275,9 @@ def fetch_browse_list_contents (self): response = self._session_get(component='browse') return BeautifulSoup(response.text, 'html.parser') - def fetch_video_list_ids (self, list_from=0, list_to=50): + def fetch_video_list_ids_via_preflight (self, list_from=0, list_to=50): """Fetches the JSON with detailed information based on the lists on the landing page (browse page) of Netflix + via the preflight (GET) request Parameters ---------- @@ -1298,14 +1304,11 @@ def fetch_video_list_ids (self, list_from=0, list_to=50): response = self._session_get(component='video_list_ids', params=payload, type='api') return self._process_response(response=response, component=self._get_api_url_for(component='video_list_ids')) - def fetch_search_results (self, search_str, list_from=0, list_to=10): - """Fetches the JSON which contains the results for the given search query + def fetch_video_list_ids (self, list_from=0, list_to=50): + """Fetches the JSON with detailed information based on the lists on the landing page (browse page) of Netflix Parameters ---------- - search_str : :obj:`str` - String to query Netflix search for - list_from : :obj:`int` Start entry for pagination @@ -1317,42 +1320,20 @@ def fetch_search_results (self, search_str, list_from=0, list_to=10): :obj:`dict` of :obj:`dict` of :obj:`str` Raw Netflix API call response or api call error """ - # properly encode the search string - encoded_search_string = quote(search_str) - paths = [ - ['search', encoded_search_string, 'titles', {'from': list_from, 'to': list_to}, ['summary', 'title']], - ['search', encoded_search_string, 'titles', {'from': list_from, 'to': list_to}, 'boxarts', '_342x192', 'jpg'], - ['search', encoded_search_string, 'titles', ['id', 'length', 'name', 'trackIds', 'requestId']], - ['search', encoded_search_string, 'suggestions', 0, 'relatedvideos', {'from': list_from, 'to': list_to}, ['summary', 'title']], - ['search', encoded_search_string, 'suggestions', 0, 'relatedvideos', {'from': list_from, 'to': list_to}, 'boxarts', '_342x192', 'jpg'], - ['search', encoded_search_string, 'suggestions', 0, 'relatedvideos', ['id', 'length', 'name', 'trackIds', 'requestId']] + ['lolomo', {'from': list_from, 'to': list_to}, ['displayName', 'context', 'id', 'index', 'length']] ] - response = self._path_request(paths=paths) - return self._process_response(response=response, component='Search results') - - def get_lolomo_for_kids (self): - """Fetches the lolomo ID for Kids profiles - Returns - ------- - :obj:`str` - Kids Lolomo ID - """ - response = self._session_get(component='kids') - for cookie in response.cookies: - if cookie.name.find('lhpuuidh-browse-' + self.user_data['guid']) != -1 and cookie.name.rfind('-T') == -1: - start = unquote(cookie.value).rfind(':') - return unquote(cookie.value)[start+1:] - return None + response = self._path_request(paths=paths) + return self._process_response(response=response, component='Video list ids') - def fetch_lists_for_kids (self, lolomo, list_from=0, list_to=50): - """Fetches the JSON which contains the contents of a the video list for kids users + def fetch_search_results (self, search_str, list_from=0, list_to=10): + """Fetches the JSON which contains the results for the given search query Parameters ---------- - lolomo : :obj:`str` - Lolomo ID for the Kids profile + search_str : :obj:`str` + String to query Netflix search for list_from : :obj:`int` Start entry for pagination @@ -1365,13 +1346,19 @@ def fetch_lists_for_kids (self, lolomo, list_from=0, list_to=50): :obj:`dict` of :obj:`dict` of :obj:`str` Raw Netflix API call response or api call error """ + # properly encode the search string + encoded_search_string = quote(search_str) + paths = [ - ['lists', lolomo, {'from': list_from, 'to': list_to}, ['displayName', 'context', 'genreId', 'id', 'index', 'length']] + ['search', encoded_search_string, 'titles', {'from': list_from, 'to': list_to}, ['summary', 'title']], + ['search', encoded_search_string, 'titles', {'from': list_from, 'to': list_to}, 'boxarts', '_342x192', 'jpg'], + ['search', encoded_search_string, 'titles', ['id', 'length', 'name', 'trackIds', 'requestId']], + ['search', encoded_search_string, 'suggestions', 0, 'relatedvideos', {'from': list_from, 'to': list_to}, ['summary', 'title']], + ['search', encoded_search_string, 'suggestions', 0, 'relatedvideos', {'from': list_from, 'to': list_to}, 'boxarts', '_342x192', 'jpg'], + ['search', encoded_search_string, 'suggestions', 0, 'relatedvideos', ['id', 'length', 'name', 'trackIds', 'requestId']] ] - response = self._path_request(paths=paths) - res = self._process_response(response=response, component='Kids lists') - return self.parse_video_list_ids(response_data=res['value']) + return self._process_response(response=response, component='Search results') def fetch_video_list (self, list_id, list_from=0, list_to=20): """Fetches the JSON which contains the contents of a given video list