Skip to content

Commit

Permalink
Respect pagination limits for related videos
Browse files Browse the repository at this point in the history
- Also partially fixes Youtube bug where related video details are no longer returned in first request to next endpoint
- Fix anxdpanic#572
  • Loading branch information
MoojMidge committed Feb 1, 2024
1 parent 3716d53 commit bd13ff4
Show file tree
Hide file tree
Showing 3 changed files with 65 additions and 31 deletions.
92 changes: 61 additions & 31 deletions resources/lib/youtube_plugin/youtube/client/youtube.py
Original file line number Diff line number Diff line change
Expand Up @@ -694,9 +694,10 @@ def index_items(items, index,

# Fetch related videos. Use threads for faster execution.
def threaded_get_related(video_id, func, *args, **kwargs):
related_videos = self.get_related_videos(video_id).get('items')
if related_videos:
func(related_videos[:items_per_page], *args, **kwargs)
related = self.get_related_videos(video_id,
max_results=items_per_page)
if related and 'items' in related:
func(related['items'][:items_per_page], *args, **kwargs)

running = 0
threads = []
Expand Down Expand Up @@ -1024,6 +1025,7 @@ def get_related_videos(self,
video_id,
page_token='',
max_results=0,
offset=0,
**kwargs):
# TODO: Improve handling of InnerTube requests, including automatic
# continuation processing to retrieve max_results number of results
Expand Down Expand Up @@ -1058,7 +1060,7 @@ def get_related_videos(self,
'results',
)
) + (
slice(None),
slice(offset, None, None),
(
(
'compactVideoRenderer',
Expand Down Expand Up @@ -1097,40 +1099,68 @@ def get_related_videos(self,
)
)

items = [{
'kind': "youtube#video",
'id': video['videoId'],
'related_video_id': video_id,
'related_channel_id': channel_id,
'partial': True,
'snippet': {
'title': video['title']['simpleText'],
'thumbnails': dict(zip(
('default', 'high'),
video['thumbnail']['thumbnails'],
)),
'channelId': self.json_traverse(video, (
('longBylineText', 'shortBylineText'),
'runs',
0,
'navigationEndpoint',
'browseEndpoint',
'browseId',
)),
}
} for video in related_videos if video and 'videoId' in video]

v3_response = {
'kind': 'youtube#videoListResponse',
'items': [
{
'kind': "youtube#video",
'id': video['videoId'],
'related_video_id': video_id,
'related_channel_id': channel_id,
'partial': True,
'snippet': {
'title': video['title']['simpleText'],
'thumbnails': dict(zip(
('default', 'high'),
video['thumbnail']['thumbnails'],
)),
'channelId': self.json_traverse(video, (
('longBylineText', 'shortBylineText'),
'runs',
0,
'navigationEndpoint',
'browseEndpoint',
'browseId',
)),
}
}
for video in related_videos
if video and 'videoId' in video
]
'items': [],
}

last_item = related_videos[-1]
if last_item and 'token' in last_item:
v3_response['nextPageToken'] = last_item['token']
page_token = last_item['token']

while 1:
remaining = max_results - len(items)
if remaining < 0:
items = items[:max_results]
if page_token:
v3_response['nextPageToken'] = page_token
v3_response['offset'] = remaining
break

if not page_token:
break

if not remaining:
v3_response['nextPageToken'] = page_token
break

continuation = self.get_related_videos(
video_id,
page_token=page_token,
max_results=remaining,
**kwargs
)
if 'nextPageToken' in continuation:
page_token = continuation['nextPageToken']
else:
page_token = ''
if 'items' in continuation:
items.extend(continuation['items'])

v3_response['items'] = items
return v3_response

def get_parent_comments(self,
Expand Down
3 changes: 3 additions & 0 deletions resources/lib/youtube_plugin/youtube/helper/v3.py
Original file line number Diff line number Diff line change
Expand Up @@ -382,6 +382,7 @@ def response_to_items(provider,
yt_total_results = int(page_info.get('totalResults', 0))
yt_results_per_page = int(page_info.get('resultsPerPage', 0))
page = int(context.get_param('page', 1))
offset = int(json_data.get('offset', 0))
yt_visitor_data = json_data.get('visitorData', '')
yt_next_page_token = json_data.get('nextPageToken', '')
yt_click_tracking = json_data.get('clickTracking', '')
Expand All @@ -398,6 +399,8 @@ def response_to_items(provider,
new_params['visitor'] = yt_visitor_data
if yt_click_tracking:
new_params['click_tracking'] = yt_click_tracking
if offset:
new_params['offset'] = offset
new_context = context.clone(new_params=new_params)
current_page = new_context.get_param('page', 1)
next_page_item = NextPageItem(new_context, current_page)
Expand Down
1 change: 1 addition & 0 deletions resources/lib/youtube_plugin/youtube/helper/yt_specials.py
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ def _process_related_videos(provider, context):
_refresh=params.get('refresh'),
video_id=video_id,
page_token=params.get('page_token', ''),
offset=params.get('offset', 0),
)
else:
json_data = function_cache.run(
Expand Down

0 comments on commit bd13ff4

Please sign in to comment.