From 4ba50a7b96ae849438341e60004da4496093cd9c Mon Sep 17 00:00:00 2001 From: mediaminister Date: Tue, 23 Jan 2024 17:19:04 +0100 Subject: [PATCH] Fix VRT MAX Api --- resources/lib/api.py | 1266 +++++++++++++++------------------ resources/lib/favorites.py | 6 + resources/lib/resumepoints.py | 39 +- tests/test_language.py | 6 + tests/test_settings.py | 2 + tests/test_vrtplayer.py | 2 + 6 files changed, 638 insertions(+), 683 deletions(-) diff --git a/resources/lib/api.py b/resources/lib/api.py index a477fe60..7b6ac8f4 100644 --- a/resources/lib/api.py +++ b/resources/lib/api.py @@ -230,368 +230,314 @@ def get_next_info(episode_id): def get_stream_id_data(vrtmax_url): """Get stream_id from from GraphQL API""" - from tokenresolver import TokenResolver - access_token = TokenResolver().get_token('vrtnu-site_profile_at') - data_json = {} - if access_token: - headers = { - 'Authorization': 'Bearer ' + access_token, - 'Content-Type': 'application/json', - } - page_id = vrtmax_url.split('www.vrt.be')[1].replace('/vrtmax/', '/vrtnu/').rstrip('/') + '.model.json' - graphql_query = """ - query StreamId($pageId: ID!) { - page(id: $pageId) { - ... on IPage { - ... on LivestreamPage { - player { - watchAction { - ... on LiveWatchAction { - streamId - } - } - } - } - } - ... on EpisodePage { - episode { - watchAction { + page_id = vrtmax_url.split('www.vrt.be')[1].replace('/vrtmax/', '/vrtnu/').rstrip('/') + '.model.json' + graphql_query = """ + query StreamId($pageId: ID!) { + page(id: $pageId) { + ... on IPage { + ... on LivestreamPage { + player { + watchAction { + ... on LiveWatchAction { streamId } } } } } - """ - payload = { - 'operationName': 'StreamId', - 'variables': { - 'pageId': page_id - }, - 'query': graphql_query, + ... on EpisodePage { + episode { + watchAction { + streamId + } + } + } + } } - from json import dumps - data = dumps(payload).encode('utf-8') - data_json = get_url_json(url=GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') - return data_json + """ + operation_name = 'StreamId' + variables = { + 'pageId': page_id + } + return api_req(graphql_query, operation_name, variables) def get_single_episode_data(episode_id): """Get single episode data from GraphQL API""" - from tokenresolver import TokenResolver - access_token = TokenResolver().get_token('vrtnu-site_profile_at') - data_json = {} - if access_token: - headers = { - 'Authorization': 'Bearer ' + access_token, - 'Content-Type': 'application/json', + graphql_query = """ + query PlayerData($id: ID!) { + catalogMember(id: $id) { + __typename + ...episode + } } - graphql_query = """ - query PlayerData($id: ID!) { - catalogMember(id: $id) { - __typename - ...episode - } + fragment episode on Episode { + __typename + id + title + description + episodeNumberRaw + durationSeconds + offTimeRaw + onTimeRaw + image { + alt + templateUrl + } + analytics { + airDate + categories + } + program { + id + title + link + programType + image { + alt + templateUrl } - fragment episode on Episode { + posterImage { + alt + templateUrl + } + } + season { + titleRaw + } + watchAction { + avodUrl + completed + resumePoint + resumePointTotal + resumePointProgress + resumePointTitle + episodeId + videoId + publicationId + streamId + } + favoriteAction { + favorite + id + title + } + nextUp { + title + autoPlay + countdown + tile { __typename - id - title - description - episodeNumberRaw - durationSeconds - offTimeRaw - onTimeRaw - image { - alt - templateUrl - } - analytics { - airDate - categories - } - program { - id - title - link - programType - image { - alt - templateUrl - } - posterImage { - alt - templateUrl - } - } - season { - titleRaw - } - watchAction { - avodUrl - completed - resumePoint - resumePointTotal - resumePointProgress - resumePointTitle - episodeId - videoId - publicationId - streamId - } - favoriteAction { - favorite - id - title - } - nextUp { - title - autoPlay - countdown - tile { - __typename - ...episodeTile - } - } + ...episodeTile } - %s - """ % EPISODE_TILE - payload = { - 'operationName': 'PlayerData', - 'variables': { - 'id': episode_id, - }, - 'query': graphql_query, + } } - from json import dumps - data = dumps(payload).encode('utf-8') - data_json = get_url_json(url=GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') - return data_json + %s + """ % EPISODE_TILE + operation_name = 'PlayerData' + variables = { + 'id': episode_id, + } + return api_req(graphql_query, operation_name, variables) def get_latest_episode_data(program_name): """Get latest episode data from GraphQL API""" - from tokenresolver import TokenResolver - access_token = TokenResolver().get_token('vrtnu-site_profile_at') - data_json = {} - if access_token: - headers = { - 'Authorization': 'Bearer ' + access_token, - 'Content-Type': 'application/json', - } - graphql_query = """ - query VideoProgramPage($pageId: ID!, $lazyItemCount: Int = 500, $after: ID) { - page(id: $pageId) { - ... on ProgramPage { - components { + graphql_query = """ + query VideoProgramPage($pageId: ID!, $lazyItemCount: Int = 500, $after: ID) { + page(id: $pageId) { + ... on ProgramPage { + components { + __typename + ... on PageHeader { + mostRelevantEpisodeTile { __typename - ... on PageHeader { - mostRelevantEpisodeTile { - __typename - title - tile { - ...episodeTile - __typename - } - __typename - } + title + tile { + ...episodeTile __typename } - ... on ContainerNavigation { - items { - title - components { + __typename + } + __typename + } + ... on ContainerNavigation { + items { + title + components { + __typename + ... on PaginatedTileList { + __typename + paginatedItems(first: $lazyItemCount, after: $after) { __typename - ... on PaginatedTileList { + edges { __typename - paginatedItems(first: $lazyItemCount, after: $after) { + cursor + node { __typename - edges { - __typename - cursor - node { - __typename - ... on EpisodeTile { - id - description - ...episodeTile - } - } + ... on EpisodeTile { + id + description + ...episodeTile } } } - ... on ContainerNavigation { - items { - title - components { + } + } + ... on ContainerNavigation { + items { + title + components { + __typename + ... on PaginatedTileList { + __typename + paginatedItems(first: $lazyItemCount, after: $after) { __typename - ... on PaginatedTileList { + edges { __typename - paginatedItems(first: $lazyItemCount, after: $after) { + cursor + node { __typename - edges { - __typename - cursor - node { - __typename - ... on EpisodeTile { - id - description - ...episodeTile - } - } + ... on EpisodeTile { + id + description + ...episodeTile } } } } } - __typename } } __typename } - __typename } + __typename } __typename } - __typename } + __typename } - %s - """ % EPISODE_TILE - payload = { - 'operationName': 'VideoProgramPage', - 'variables': { - 'pageId': '/vrtnu/a-z/{}.model.json'.format(program_name), - }, - 'query': graphql_query, + __typename + } } - from json import dumps - data = dumps(payload).encode('utf-8') - data_json = get_url_json(url=GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') - return data_json + %s + """ % EPISODE_TILE + operation_name = 'VideoProgramPage' + variables = { + 'pageId': '/vrtnu/a-z/{}.model.json'.format(program_name), + } + return api_req(graphql_query, operation_name, variables) def get_seasons_data(program_name): """Get seasons data from GraphQL API""" - from tokenresolver import TokenResolver - access_token = TokenResolver().get_token('vrtnu-site_profile_at') - data_json = {} - if access_token: - headers = { - 'Authorization': 'Bearer ' + access_token, - 'Content-Type': 'application/json', - } - graphql_query = """ - query VideoProgramPage( - $pageId: ID!) { - page(id: $pageId) { - ... on ProgramPage { - id - permalink - components { + graphql_query = """ + query VideoProgramPage( + $pageId: ID!) { + page(id: $pageId) { + ... on ProgramPage { + id + permalink + components { + __typename + ... on PageHeader { + mostRelevantEpisodeTile { + __typename + title + tile { + ...episodeTile + __typename + } __typename - ... on PageHeader { - mostRelevantEpisodeTile { + } + __typename + } + ... on PaginatedTileList { + __typename + id: objectId + objectId + listId + title + tileContentType + } + ... on ContainerNavigation { + id: objectId + navigationType + items { + id: objectId + title + active + components { + __typename + ... on PaginatedTileList { __typename + id: objectId + objectId + listId title - tile { - ...episodeTile - __typename - } + tileContentType + } + ... on StaticTileList { __typename + id: objectId + objectId + listId + title + tileContentType } - __typename - } - ... on PaginatedTileList { - __typename - id: objectId - objectId - listId - title - tileContentType - } - ... on ContainerNavigation { - id: objectId - navigationType - items { + ... on LazyTileList { + __typename id: objectId + objectId + listId title - active - components { - __typename - ... on PaginatedTileList { - __typename - id: objectId - objectId - listId - title - tileContentType - } - ... on StaticTileList { - __typename - id: objectId - objectId - listId - title - tileContentType - } - ... on LazyTileList { - __typename + tileContentType + } + ... on IComponent { + ... on ContainerNavigation { + id: objectId + navigationType + items { id: objectId - objectId - listId title - tileContentType - } - ... on IComponent { - ... on ContainerNavigation { - id: objectId - navigationType - items { - id: objectId - title - components { + components { + __typename + ... on Component { + ... on PaginatedTileList { __typename - ... on Component { - ... on PaginatedTileList { - __typename - id: objectId - objectId - listId - title - tileContentType - } - ... on StaticTileList { - __typename - id: objectId - objectId - listId - title - tileContentType - } - ... on LazyTileList { - __typename - id: objectId - objectId - listId - title - tileContentType - } - __typename - } + id: objectId + objectId + listId + title + tileContentType + } + ... on StaticTileList { + __typename + id: objectId + objectId + listId + title + tileContentType + } + ... on LazyTileList { + __typename + id: objectId + objectId + listId + title + tileContentType } __typename } - __typename } __typename } + __typename } __typename } - __typename } __typename } @@ -599,20 +545,18 @@ def get_seasons_data(program_name): } __typename } + __typename } - %s - """ % EPISODE_TILE - payload = { - 'operationName': 'VideoProgramPage', - 'variables': { - 'pageId': '/vrtnu/a-z/{}.model.json'.format(program_name), - }, - 'query': graphql_query, + __typename + } } - from json import dumps - data = dumps(payload).encode('utf-8') - data_json = get_url_json(url=GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') - return data_json + %s + """ % EPISODE_TILE + operation_name = 'VideoProgramPage' + variables = { + 'pageId': '/vrtnu/a-z/{}.model.json'.format(program_name), + } + return api_req(graphql_query, operation_name, variables) def set_resumepoint(video_id, title, position, total): @@ -645,155 +589,126 @@ def set_resumepoint(video_id, title, position, total): def get_paginated_episodes(list_id, page_size, end_cursor=''): """Get paginated list of episodes from GraphQL API""" - from tokenresolver import TokenResolver - access_token = TokenResolver().get_token('vrtnu-site_profile_at') - data_json = {} - if access_token: - headers = { - 'Authorization': 'Bearer ' + access_token, - 'Content-Type': 'application/json', - } - graphql_query = """ - query ListedEpisodes( - $listId: ID! - $endCursor: ID! - $pageSize: Int! - ) { - list(listId: $listId) { - __typename - ... on PaginatedTileList { - paginated: paginatedItems(first: $pageSize, after: $endCursor) { - edges { - node { - __typename - ...episodeTile - } - } - pageInfo { - startCursor - endCursor - hasNextPage - hasPreviousPage - __typename - } + graphql_query = """ + query ListedEpisodes( + $listId: ID! + $endCursor: ID! + $pageSize: Int! + ) { + list(listId: $listId) { + __typename + ... on PaginatedTileList { + paginated: paginatedItems(first: $pageSize, after: $endCursor) { + edges { + node { + __typename + ...episodeTile } } + pageInfo { + startCursor + endCursor + hasNextPage + hasPreviousPage + __typename + } } } - %s - """ % EPISODE_TILE - # FIXME: Find a better way to change GraphQL typename - if list_id.startswith('static:/'): - graphql_query = graphql_query.replace('on PaginatedTileList', 'on StaticTileList') - - payload = { - 'operationName': 'ListedEpisodes', - 'variables': { - 'listId': list_id, - 'endCursor': end_cursor, - 'pageSize': page_size, - }, - 'query': graphql_query, + } } - from json import dumps - data = dumps(payload).encode('utf-8') - data_json = get_url_json(url=GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') - return data_json + %s + """ % EPISODE_TILE + # FIXME: Find a better way to change GraphQL typename + if list_id.startswith('static:/'): + graphql_query = graphql_query.replace('on PaginatedTileList', 'on StaticTileList') + + operation_name = 'ListedEpisodes' + variables = { + 'listId': list_id, + 'endCursor': end_cursor, + 'pageSize': page_size, + } + return api_req(graphql_query, operation_name, variables) -def get_paginated_programs(list_id, page_size, end_cursor=''): +def get_paginated_programs(list_id, page_size, end_cursor='', client='WEB'): """Get paginated list of episodes from GraphQL API""" - from tokenresolver import TokenResolver - access_token = TokenResolver().get_token('vrtnu-site_profile_at') - data_json = {} - if access_token: - headers = { - 'Authorization': 'Bearer ' + access_token, - 'Content-Type': 'application/json', - 'x-vrt-client-name': 'MobileAndroid', - } - graphql_query = """ - query PaginatedPrograms( - $listId: ID! - $endCursor: ID! - $pageSize: Int! - ) { - list(listId: $listId) { - __typename - ... on PaginatedTileList { - paginated: paginatedItems(first: $pageSize, after: $endCursor) { - edges { - node { - __typename - ...ep - } - } - pageInfo { - startCursor - endCursor - hasNextPage - hasPreviousPage - __typename - } + graphql_query = """ + query PaginatedPrograms( + $listId: ID! + $endCursor: ID! + $pageSize: Int! + ) { + list(listId: $listId) { + __typename + ... on PaginatedTileList { + paginated: paginatedItems(first: $pageSize, after: $endCursor) { + edges { + node { + __typename + ...ep } } + pageInfo { + startCursor + endCursor + hasNextPage + hasPreviousPage + __typename + } } } - fragment ep on ProgramTile { - __typename - objectId + } + } + fragment ep on ProgramTile { + __typename + objectId + id + link + tileType + image { + alt + templateUrl + } + title + program { + title + id + link + programType + description + shortDescription + subtitle + announcementType + announcementValue + whatsonId + image { + alt + templateUrl + } + posterImage { + alt + templateUrl + } + favoriteAction { + favorite id - link - tileType - image { - alt - templateUrl - } title - program { - title - id - link - programType - description - shortDescription - subtitle - announcementType - announcementValue - whatsonId - image { - alt - templateUrl - } - posterImage { - alt - templateUrl - } - favoriteAction { - favorite - id - title - } - } } - """ - # FIXME: Find a better way to change GraphQL typename - if list_id.startswith('static:/'): - graphql_query = graphql_query.replace('on PaginatedTileList', 'on StaticTileList') - - payload = { - 'operationName': 'PaginatedPrograms', - 'variables': { - 'listId': list_id, - 'endCursor': end_cursor, - 'pageSize': page_size, - }, - 'query': graphql_query, + } } - from json import dumps - data = dumps(payload).encode('utf-8') - data_json = get_url_json(url=GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') - return data_json + """ + # FIXME: Find a better way to change GraphQL typename + if list_id.startswith('static:/'): + graphql_query = graphql_query.replace('on PaginatedTileList', 'on StaticTileList') + + operation_name = 'PaginatedPrograms' + variables = { + 'listId': list_id, + 'endCursor': end_cursor, + 'pageSize': page_size, + } + return api_req(graphql_query, operation_name, variables, client) def convert_programs(api_data, destination, use_favorites=False, **kwargs): @@ -1122,7 +1037,7 @@ def get_programs(category=None, channel=None, keywords=None, end_cursor=''): encoded_search = base64.b64encode(dumps(search_dict).encode('utf-8')) list_id = 'uisearch:searchdata@{}'.format(encoded_search.decode('utf-8')) - api_data = get_paginated_programs(list_id=list_id, page_size=page_size, end_cursor=end_cursor) + api_data = get_paginated_programs(list_id=list_id, page_size=page_size, end_cursor=end_cursor, client='MobileAndroid') programs = convert_programs(api_data, destination=destination, category=category, channel=channel, keywords=keywords) return programs @@ -1262,88 +1177,97 @@ def get_seasons(program_name): return seasons -def get_featured_data(): - """Get featured data""" +def api_req(graphql_query, operation_name, variables, client='WEB'): + """GraphQL API Request""" + from json import dumps from tokenresolver import TokenResolver access_token = TokenResolver().get_token('vrtnu-site_profile_at') data_json = {} if access_token: + payload = { + 'operationName': operation_name, + 'query': graphql_query, + 'variables': variables, + } + data = dumps(payload).encode('utf-8') headers = { + 'Accept': 'application/json', 'Authorization': 'Bearer ' + access_token, 'Content-Type': 'application/json', - 'x-vrt-client-name': 'MobileAndroid', + 'x-vrt-client-name': client, + 'x-vrt-client-version': '1.5.0', } - graphql_query = """ - query Page( - $pageId: ID! - $lazyItemCount: Int = 10 - $after: ID - $before: ID - $componentCount: Int = 5 - $componentAfter: ID - ) { - page(id: $pageId) { - ... on IPage { - title - permalink - paginatedComponents(first: $componentCount, after: $componentAfter) { - __typename - edges { + data_json = get_url_json(url=GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') + return data_json + + +def get_featured_data(): + """Get featured data""" + graphql_query = """ + query Page( + $pageId: ID! + $lazyItemCount: Int = 10 + $after: ID + $before: ID + $componentCount: Int = 5 + $componentAfter: ID + ) { + page(id: $pageId) { + ... on IPage { + title + permalink + paginatedComponents(first: $componentCount, after: $componentAfter) { + __typename + edges { + __typename + node { + ... on PaginatedTileList { __typename - node { - ... on PaginatedTileList { + listId + componentType + paginatedItems(first: $lazyItemCount, after: $after, before: $before) { + __typename + edges { __typename - listId - componentType - paginatedItems(first: $lazyItemCount, after: $after, before: $before) { + node { __typename - edges { - __typename - node { - __typename - } - } } - tileContentType - title - __typename - } - ... on StaticTileList { - __typename - listId - title - header { - brand - ctaText - description - } - componentType - tileContentType } - __typename } + tileContentType + title + __typename } + ... on StaticTileList { + __typename + listId + title + header { + brand + ctaText + description + } + componentType + tileContentType + } + __typename } } } } - """ - payload = { - 'operationName': 'Page', - 'variables': { - 'pageId': '/vrtmax/', - 'pageContext': { - 'mediaType': 'watch' - }, - 'componentAfter': '', - 'componentCount': 50, - }, - 'query': graphql_query, + } } - from json import dumps - data = dumps(payload).encode('utf-8') - data_json = get_url_json(url=GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') - return data_json + """ + operation_name = 'Page' + variables = { + 'pageId': '/vrtmax/', + 'pageContext': { + 'mediaType': 'watch' + }, + 'componentAfter': '', + 'componentCount': 50, + } + return api_req(graphql_query, operation_name, variables) def get_featured(feature=None, end_cursor=''): @@ -1434,196 +1358,182 @@ def get_categories(): def get_online_categories(): """Return a list of categories from the VRT MAX website""" categories = [] - from tokenresolver import TokenResolver - access_token = TokenResolver().get_token('vrtnu-site_profile_at') - categories_json = {} - if access_token: - headers = { - 'Authorization': 'Bearer ' + access_token, - 'Content-Type': 'application/json', - 'x-vrt-client-name': 'MobileAndroid', + graphql_query = """ + query Search( + $q: String + $mediaType: MediaType + $facets: [SearchFacetInput] + ) { + uiSearch(input: { q: $q, mediaType: $mediaType, facets: $facets }) { + __typename + ... on IIdentifiable { + objectId + __typename + } + ... on IComponent { + componentType + __typename + } + ...staticTileListFragment + } } - graphql_query = """ - query Search( - $q: String - $mediaType: MediaType - $facets: [SearchFacetInput] - ) { - uiSearch(input: { q: $q, mediaType: $mediaType, facets: $facets }) { - __typename - ... on IIdentifiable { - objectId - __typename - } - ... on IComponent { - componentType - __typename - } - ...staticTileListFragment - } + fragment staticTileListFragment on StaticTileList { + __typename + id: objectId + objectId + listId + title + componentType + tileContentType + tileOrientation + displayType + expires + tileVariant + sort { + icon + order + title + __typename + } + actionItems { + ...actionItem + __typename + } + header { + action { + ...action + __typename } - fragment staticTileListFragment on StaticTileList { + brand + brandLogos { + height + mono + primary + type + width __typename - id: objectId - objectId - listId - title - componentType - tileContentType - tileOrientation - displayType - expires - tileVariant - sort { - icon - order - title - __typename - } - actionItems { - ...actionItem - __typename - } - header { - action { - ...action - __typename - } - brand - brandLogos { - height - mono - primary - type - width - __typename - } - ctaText - description - image { - ...imageFragment - __typename - } - type - compactLayout - backgroundColor - textTheme - __typename - } - bannerSize - items { - ...tileFragment - __typename - } - ... on IComponent { - __typename - } } - fragment tileFragment on Tile { - ... on IIdentifiable { - __typename - objectId - } - ... on IComponent { - title - componentType - __typename - } - ... on ITile { - title - action { - ...action - __typename - } - image { - ...imageFragment - __typename - } - __typename - } - ... on BannerTile { - id - backgroundColor - textTheme - active - description - __typename - } + ctaText + description + image { + ...imageFragment + __typename } - fragment actionItem on ActionItem { + type + compactLayout + backgroundColor + textTheme + __typename + } + bannerSize + items { + ...tileFragment + __typename + } + ... on IComponent { + __typename + } + } + fragment tileFragment on Tile { + ... on IIdentifiable { + __typename + objectId + } + ... on IComponent { + title + componentType + __typename + } + ... on ITile { + title + action { + ...action __typename - id: objectId - accessibilityLabel - action { - ...action - __typename - } - active - analytics { - __typename - eventId - interaction - interactionDetail - pageProgrambrand - } - icon - iconPosition - mode - objectId - title } - fragment action on Action { + image { + ...imageFragment __typename - ... on SearchAction { - facets { - name - values - __typename - } - mediaType - navigationType - q - __typename - } } - fragment imageFragment on Image { - id: objectId - alt - title - focalPoint - objectId - templateUrl + __typename + } + ... on BannerTile { + id + backgroundColor + textTheme + active + description + __typename + } + } + fragment actionItem on ActionItem { + __typename + id: objectId + accessibilityLabel + action { + ...action + __typename + } + active + analytics { + __typename + eventId + interaction + interactionDetail + pageProgrambrand + } + icon + iconPosition + mode + objectId + title + } + fragment action on Action { + __typename + ... on SearchAction { + facets { + name + values + __typename } - """ - payload = { - 'operationName': 'Search', - 'variables': { - 'facets': [], - 'mediaType': 'watch', - 'q': '', - }, - 'query': graphql_query, + mediaType + navigationType + q + __typename + } } - from json import dumps - data = dumps(payload).encode('utf-8') - categories_json = get_url_json(url=GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') - if categories_json is not None: - content_types = find_entry(categories_json.get('data').get('uiSearch'), 'listId', 'initialsearchcontenttypes').get('items') - genres = find_entry(categories_json.get('data').get('uiSearch'), 'listId', 'initialsearchgenres').get('items') - category_items = content_types + genres - for category in category_items: - # Don't add audio-only categories - if category.get('title') in ('Podcasts', 'Radio'): - continue - thumb = category.get('image') - if thumb: - thumb = thumb.get('templateUrl') - categories.append({ - 'id': category.get('action').get('facets')[0].get('values')[0], - 'thumbnail': thumb, - 'name': category.get('title'), - }) - categories.sort(key=lambda x: x.get('name')) + fragment imageFragment on Image { + id: objectId + alt + title + focalPoint + objectId + templateUrl + } + """ + operation_name = 'Search' + variables = { + 'facets': [], + 'mediaType': 'watch', + 'q': '', + } + categories_json = api_req(graphql_query, operation_name, variables, client='MobileAndroid') + if categories_json is not None: + content_types = find_entry(categories_json.get('data').get('uiSearch'), 'listId', 'initialsearchcontenttypes').get('items') + genres = find_entry(categories_json.get('data').get('uiSearch'), 'listId', 'initialsearchgenres').get('items') + category_items = content_types + genres + for category in category_items: + # Don't add audio-only categories + if category.get('title') in ('Podcasts', 'Radio'): + continue + thumb = category.get('image') + if thumb: + thumb = thumb.get('templateUrl') + categories.append({ + 'id': category.get('action').get('facets')[0].get('values')[0], + 'thumbnail': thumb, + 'name': category.get('title'), + }) + categories.sort(key=lambda x: x.get('name')) return categories diff --git a/resources/lib/favorites.py b/resources/lib/favorites.py index df47a502..afdbf1f8 100644 --- a/resources/lib/favorites.py +++ b/resources/lib/favorites.py @@ -83,6 +83,8 @@ def get_favorites(self): headers = { 'Authorization': 'Bearer ' + access_token, 'Content-Type': 'application/json', + 'x-vrt-client-name': 'WEB', + 'x-vrt-client-version': '1.5.0', } graphql = """ query Favs( @@ -145,6 +147,8 @@ def get_program_id_graphql(self, program_name): headers = { 'Authorization': 'Bearer ' + access_token, 'Content-Type': 'application/json', + 'x-vrt-client-name': 'WEB', + 'x-vrt-client-version': '1.5.0', } graphql = """ query Page($id: ID!) { @@ -176,6 +180,8 @@ def set_favorite_graphql(self, program_id, title, is_favorite=True): headers = { 'Authorization': 'Bearer ' + access_token, 'Content-Type': 'application/json', + 'x-vrt-client-name': 'WEB', + 'x-vrt-client-version': '1.5.0', } graphql_query = """ mutation setFavorite($input: FavoriteActionInput!) { diff --git a/resources/lib/resumepoints.py b/resources/lib/resumepoints.py index 3e590185..a6267141 100644 --- a/resources/lib/resumepoints.py +++ b/resources/lib/resumepoints.py @@ -74,6 +74,8 @@ def resumepoints_headers(): headers = { 'Authorization': 'Bearer ' + access_token, 'Content-Type': 'application/json', + 'x-vrt-client-name': 'WEB', + 'x-vrt-client-version': '1.5.0', } else: log_error('Failed to get access token from VRT MAX') @@ -175,29 +177,54 @@ def delete_continue(self, episode_id): def _delete_continue_graphql(self, episode_id): """Delete continue episode using GraphQL API""" from tokenresolver import TokenResolver + from json import dumps + import base64 access_token = TokenResolver().get_token('vrtnu-site_profile_at') result_json = {} if access_token: headers = { 'Authorization': 'Bearer ' + access_token, 'Content-Type': 'application/json', + 'x-vrt-client-name': 'WEB', + 'x-vrt-client-version': '1.5.0', } graphql_query = """ - mutation listDelete($input: ListDeleteInput!) { - listDelete(input: $input) + mutation listDelete($input: ListDeleteActionInput!) { + setListDeleteActionItem(input: $input) { + title + active + action { + __typename + ... on NoAction { + __typename + reason + } + ... on ListTileDeletedAction { + __typename + listId + listName + id + } + } + __typename + } } """ + list_name = { + 'listId': 'dynamic:/vrtnu.model.json@resume-list-video', + 'listType': 'verderkijken', + } + encoded_list_name = base64.b64encode(dumps(list_name).encode('utf-8')) payload = { 'operationName': 'listDelete', 'variables': { 'input': { 'id': episode_id, - 'listName': 'verderkijken', + 'listName': encoded_list_name.decode('utf-8'), }, }, 'query': graphql_query, } - from json import dumps data = dumps(payload).encode('utf-8') result_json = get_url_json(url=self.GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') return result_json @@ -205,12 +232,15 @@ def _delete_continue_graphql(self, episode_id): def get_continue(self): """Get continue using GraphQL API""" from tokenresolver import TokenResolver + from json import dumps access_token = TokenResolver().get_token('vrtnu-site_profile_at') continue_json = {} if access_token: headers = { 'Authorization': 'Bearer ' + access_token, 'Content-Type': 'application/json', + 'x-vrt-client-name': 'WEB', + 'x-vrt-client-version': '1.5.0', } graphql_query = """ query ContinueEpisodes( @@ -251,7 +281,6 @@ def get_continue(self): }, 'query': graphql_query, } - from json import dumps data = dumps(payload).encode('utf-8') continue_json = get_url_json(url=self.GRAPHQL_URL, cache=None, headers=headers, data=data, raise_errors='all') return continue_json diff --git a/tests/test_language.py b/tests/test_language.py index c9dcad1d..56298eda 100644 --- a/tests/test_language.py +++ b/tests/test_language.py @@ -23,6 +23,8 @@ def tearDown(self): xbmc.settings['locale.language'] = 'resource.language.nl_nl' @staticmethod + @unittest.skipUnless(addon.settings.get('username'), 'Skipping as VRT username is missing.') + @unittest.skipUnless(addon.settings.get('password'), 'Skipping as VRT password is missing.') def test_dutch(): """Test the principal add-on language""" xbmc.settings['locale.language'] = 'resource.language.nl_nl' @@ -31,6 +33,8 @@ def test_dutch(): plugin.run(['plugin://plugin.video.vrt.nu/tvguide', '0', '']) @staticmethod + @unittest.skipUnless(addon.settings.get('username'), 'Skipping as VRT username is missing.') + @unittest.skipUnless(addon.settings.get('password'), 'Skipping as VRT password is missing.') def test_german(): """Test an unsupported language""" xbmc.settings['locale.language'] = 'resource.language.de_de' @@ -39,6 +43,8 @@ def test_german(): plugin.run(['plugin://plugin.video.vrt.nu/tvguide', '0', '']) @staticmethod + @unittest.skipUnless(addon.settings.get('username'), 'Skipping as VRT username is missing.') + @unittest.skipUnless(addon.settings.get('password'), 'Skipping as VRT password is missing.') def test_english(): """Test the default Kodi language""" xbmc.settings['locale.language'] = 'resource.language.en_gb' diff --git a/tests/test_settings.py b/tests/test_settings.py index ddab0644..8be67703 100644 --- a/tests/test_settings.py +++ b/tests/test_settings.py @@ -99,6 +99,8 @@ def test_youtube_disabled(): plugin.run(['plugin://plugin.video.vrt.nu/channels/radio1', '0', '']) @staticmethod + @unittest.skipUnless(addon.settings.get('username'), 'Skipping as VRT username is missing.') + @unittest.skipUnless(addon.settings.get('password'), 'Skipping as VRT password is missing.') def test_showfanart_disabled(): """Test with showfanart disabled""" addon.settings['showfanart'] = False diff --git a/tests/test_vrtplayer.py b/tests/test_vrtplayer.py index 6294d630..b74ee92f 100644 --- a/tests/test_vrtplayer.py +++ b/tests/test_vrtplayer.py @@ -107,6 +107,8 @@ def test_random_tvshow_episodes(self): else: self.fail('We did not expect this, either we find episodes or it is a playable item') + @unittest.skipUnless(addon.settings.get('username'), 'Skipping as VRT username is missing.') + @unittest.skipUnless(addon.settings.get('password'), 'Skipping as VRT password is missing.') def test_categories(self): """Test to ensure our hardcoded categories conforms to scraped categories""" category_items = get_categories()