@ -3271,37 +3271,50 @@ class YoutubeIE(YoutubeBaseInfoExtractor):
if not comment_id :
if not comment_id :
return
return
text = self . _get_text ( comment_renderer , ' contentText ' )
info = {
' id ' : comment_id ,
' text ' : self . _get_text ( comment_renderer , ' contentText ' ) ,
' like_count ' : self . _get_count ( comment_renderer , ' voteCount ' ) ,
' author_id ' : traverse_obj ( comment_renderer , ( ' authorEndpoint ' , ' browseEndpoint ' , ' browseId ' , { self . ucid_or_none } ) ) ,
' author ' : self . _get_text ( comment_renderer , ' authorText ' ) ,
' author_thumbnail ' : traverse_obj ( comment_renderer , ( ' authorThumbnail ' , ' thumbnails ' , - 1 , ' url ' , { url_or_none } ) ) ,
' parent ' : parent or ' root ' ,
}
# Timestamp is an estimate calculated from the current time and time_text
# Timestamp is an estimate calculated from the current time and time_text
time_text = self . _get_text ( comment_renderer , ' publishedTimeText ' ) or ' '
time_text = self . _get_text ( comment_renderer , ' publishedTimeText ' ) or ' '
timestamp = self . _parse_time_text ( time_text )
timestamp = self . _parse_time_text ( time_text )
author = self . _get_text ( comment_renderer , ' authorText ' )
info . update ( {
author_id = try_get ( comment_renderer ,
# FIXME: non-standard, but we need a way of showing that it is an estimate.
lambda x : x [ ' authorEndpoint ' ] [ ' browseEndpoint ' ] [ ' browseId ' ] , str )
' _time_text ' : time_text ,
' timestamp ' : timestamp ,
} )
votes = parse_count ( try_get ( comment_renderer , ( lambda x : x [ ' voteCount ' ] [ ' simpleText ' ] ,
info[ ' author_url ' ] = urljoin (
lambda x : x [ ' likeCount ' ] ) , str ) ) or 0
' https://www.youtube.com ' , traverse_obj ( comment_renderer , ( ' authorEndpoint ' , (
author_thumbnail = try_get ( comment_renderer ,
( ' browseEndpoint ' , ' canonicalBaseUrl ' ) , ( ' commandMetadata ' , ' webCommandMetadata ' , ' url ' ) ) ) ,
lambda x : x [ ' authorThumbnail ' ] [ ' thumbnails ' ] [ - 1 ] [ ' url ' ] , str )
expected_type = str , get_all = False ) )
author_is_uploader = try_get ( comment_renderer , lambda x : x [ ' authorIsChannelOwner ' ] , bool )
author_is_uploader = traverse_obj ( comment_renderer , ' authorIsChannelOwner ' )
is_favorited = ' creatorHeart ' in ( try_get (
if author_is_uploader is not None :
comment_renderer , lambda x : x [ ' actionButtons ' ] [ ' commentActionButtonsRenderer ' ] , dict ) or { } )
info [ ' author_is_uploader ' ] = author_is_uploader
return {
' id ' : comment_id ,
comment_abr = traverse_obj (
' text ' : text ,
comment_renderer , ( ' actionsButtons ' , ' commentActionButtonsRenderer ' ) , expected_type = dict )
' timestamp ' : timestamp ,
if comment_abr is not None :
' time_text ' : time_text ,
info [ ' is_favorited ' ] = ' creatorHeart ' in comment_abr
' like_count ' : votes ,
' is_favorited ' : is_favorited ,
comment_ab_icontype = traverse_obj (
' author ' : author ,
comment_renderer , ( ' authorCommentBadge ' , ' authorCommentBadgeRenderer ' , ' icon ' , ' iconType ' ) )
' author_id ' : author_id ,
if comment_ab_icontype is not None :
' author_thumbnail ' : author_thumbnail ,
info [ ' author_is_verified ' ] = comment_ab_icontype in ( ' CHECK_CIRCLE_THICK ' , ' OFFICIAL_ARTIST_BADGE ' )
' author_is_uploader ' : author_is_uploader ,
' parent ' : parent or ' root '
is_pinned = traverse_obj ( comment_renderer , ' pinnedCommentBadge ' )
}
if is_pinned :
info [ ' is_pinned ' ] = True
return info
def _comment_entries ( self , root_continuation_data , ytcfg , video_id , parent = None , tracker = None ) :
def _comment_entries ( self , root_continuation_data , ytcfg , video_id , parent = None , tracker = None ) :
@ -3349,14 +3362,13 @@ class YoutubeIE(YoutubeBaseInfoExtractor):
comment = self . _extract_comment ( comment_renderer , parent )
comment = self . _extract_comment ( comment_renderer , parent )
if not comment :
if not comment :
continue
continue
is_pinned = bool ( traverse_obj ( comment_renderer , ' pinnedCommentBadge ' ) )
comment_id = comment [ ' id ' ]
comment_id = comment [ ' id ' ]
if is_pinned:
if comment. get ( ' is_pinned' ) :
tracker [ ' pinned_comment_ids ' ] . add ( comment_id )
tracker [ ' pinned_comment_ids ' ] . add ( comment_id )
# Sometimes YouTube may break and give us infinite looping comments.
# Sometimes YouTube may break and give us infinite looping comments.
# See: https://github.com/yt-dlp/yt-dlp/issues/6290
# See: https://github.com/yt-dlp/yt-dlp/issues/6290
if comment_id in tracker [ ' seen_comment_ids ' ] :
if comment_id in tracker [ ' seen_comment_ids ' ] :
if comment_id in tracker [ ' pinned_comment_ids ' ] and not is_pinned:
if comment_id in tracker [ ' pinned_comment_ids ' ] and not comment. get ( ' is_pinned' ) :
# Pinned comments may appear a second time in newest first sort
# Pinned comments may appear a second time in newest first sort
# See: https://github.com/yt-dlp/yt-dlp/issues/6712
# See: https://github.com/yt-dlp/yt-dlp/issues/6712
continue
continue