[YouTube] Support new A/B tested like data and avoid like count conversion from integer to long

Also make minor improvements to current like data extraction and remove
previous like count data support, as it is not returned anymore.
This commit is contained in:
AudricV 2023-10-28 18:26:52 +02:00
parent b71ce1123f
commit 3782d9a02a
No known key found for this signature in database
GPG Key ID: DA92EC7905614198

View File

@ -388,19 +388,33 @@ public class YoutubeStreamExtractor extends StreamExtractor {
// If ratings are not allowed, there is no like count available // If ratings are not allowed, there is no like count available
if (!playerResponse.getObject("videoDetails").getBoolean("allowRatings")) { if (!playerResponse.getObject("videoDetails").getBoolean("allowRatings")) {
return -1; return -1L;
} }
String likesString = "";
try {
final JsonArray topLevelButtons = getVideoPrimaryInfoRenderer() final JsonArray topLevelButtons = getVideoPrimaryInfoRenderer()
.getObject("videoActions") .getObject("videoActions")
.getObject("menuRenderer") .getObject("menuRenderer")
.getArray("topLevelButtons"); .getArray("topLevelButtons");
// Try first with the new video actions buttons data structure try {
JsonObject likeToggleButtonRenderer = topLevelButtons.stream() return parseLikeCountFromLikeButtonViewModel(topLevelButtons);
} catch (final ParsingException ignored) {
// A segmentedLikeDislikeButtonRenderer could be returned instead of a
// segmentedLikeDislikeButtonViewModel, so ignore extraction errors relative to
// segmentedLikeDislikeButtonViewModel object
}
try {
return parseLikeCountFromLikeButtonRenderer(topLevelButtons);
} catch (final ParsingException e) {
throw new ParsingException("Could not get like count", e);
}
}
private static long parseLikeCountFromLikeButtonRenderer(
@Nonnull final JsonArray topLevelButtons) throws ParsingException {
String likesString = null;
final JsonObject likeToggleButtonRenderer = topLevelButtons.stream()
.filter(JsonObject.class::isInstance) .filter(JsonObject.class::isInstance)
.map(JsonObject.class::cast) .map(JsonObject.class::cast)
.map(button -> button.getObject("segmentedLikeDislikeButtonRenderer") .map(button -> button.getObject("segmentedLikeDislikeButtonRenderer")
@ -410,35 +424,7 @@ public class YoutubeStreamExtractor extends StreamExtractor {
.findFirst() .findFirst()
.orElse(null); .orElse(null);
// Use the old video actions buttons data structure if the new one isn't returned if (likeToggleButtonRenderer != null) {
if (likeToggleButtonRenderer == null) {
/*
In the old video actions buttons data structure, there are 3 ways to detect whether
a button is the like button, using its toggleButtonRenderer:
- checking whether toggleButtonRenderer.targetId is equal to watch-like;
- checking whether toggleButtonRenderer.defaultIcon.iconType is equal to LIKE;
- checking whether
toggleButtonRenderer.toggleButtonSupportedData.toggleButtonIdData.id
is equal to TOGGLE_BUTTON_ID_TYPE_LIKE.
*/
likeToggleButtonRenderer = topLevelButtons.stream()
.filter(JsonObject.class::isInstance)
.map(JsonObject.class::cast)
.map(topLevelButton -> topLevelButton.getObject("toggleButtonRenderer"))
.filter(toggleButtonRenderer -> toggleButtonRenderer.getString("targetId")
.equalsIgnoreCase("watch-like")
|| toggleButtonRenderer.getObject("defaultIcon")
.getString("iconType")
.equalsIgnoreCase("LIKE")
|| toggleButtonRenderer.getObject("toggleButtonSupportedData")
.getObject("toggleButtonIdData")
.getString("id")
.equalsIgnoreCase("TOGGLE_BUTTON_ID_TYPE_LIKE"))
.findFirst()
.orElseThrow(() -> new ParsingException(
"The like button is missing even though ratings are enabled"));
}
// Use one of the accessibility strings available (this one has the same path as the // Use one of the accessibility strings available (this one has the same path as the
// one used for comments' like count extraction) // one used for comments' like count extraction)
likesString = likeToggleButtonRenderer.getObject("accessibilityData") likesString = likeToggleButtonRenderer.getObject("accessibilityData")
@ -460,23 +446,58 @@ public class YoutubeStreamExtractor extends StreamExtractor {
.getString("label"); .getString("label");
} }
// If ratings are allowed and the likes string is null, it means that we couldn't // This check only works with English localizations!
// extract the (real) like count from accessibility data if (likesString != null && likesString.toLowerCase().contains("no likes")) {
return 0;
}
}
// If ratings are allowed and the likes string is null, it means that we couldn't extract
// the full like count from accessibility data
if (likesString == null) { if (likesString == null) {
throw new ParsingException("Could not get like count from accessibility data"); throw new ParsingException("Could not get like count from accessibility data");
} }
// This check only works with English localizations! try {
if (likesString.toLowerCase().contains("no likes")) { return Long.parseLong(Utils.removeNonDigitCharacters(likesString));
return 0; } catch (final NumberFormatException e) {
throw new ParsingException("Could not parse \"" + likesString + "\" as a long", e);
}
} }
return Integer.parseInt(Utils.removeNonDigitCharacters(likesString)); private static long parseLikeCountFromLikeButtonViewModel(
} catch (final NumberFormatException nfe) { @Nonnull final JsonArray topLevelButtons) throws ParsingException {
throw new ParsingException("Could not parse \"" + likesString + "\" as an Integer", // Try first with the current video actions buttons data structure
nfe); final JsonObject likeToggleButtonViewModel = topLevelButtons.stream()
} catch (final Exception e) { .filter(JsonObject.class::isInstance)
throw new ParsingException("Could not get like count", e); .map(JsonObject.class::cast)
.map(button -> button.getObject("segmentedLikeDislikeButtonViewModel")
.getObject("likeButtonViewModel")
.getObject("likeButtonViewModel")
.getObject("toggleButtonViewModel")
.getObject("toggleButtonViewModel")
.getObject("defaultButtonViewModel")
.getObject("buttonViewModel"))
.filter(buttonViewModel -> !isNullOrEmpty(buttonViewModel))
.findFirst()
.orElse(null);
if (likeToggleButtonViewModel == null) {
throw new ParsingException("Could not find buttonViewModel object");
}
final String accessibilityText = likeToggleButtonViewModel.getString("accessibilityText");
if (accessibilityText == null) {
throw new ParsingException("Could not find buttonViewModel's accessibilityText string");
}
// The like count is always returned as a number in this element, even for videos with no
// likes
try {
return Long.parseLong(Utils.removeNonDigitCharacters(accessibilityText));
} catch (final NumberFormatException e) {
throw new ParsingException(
"Could not parse \"" + accessibilityText + "\" as a long", e);
} }
} }