mirror of
https://github.com/TeamNewPipe/NewPipeExtractor
synced 2025-09-01 06:45:53 +00:00
Basic implementation of `YoutubeStreamInfoItemLockupExtractor
`
This commit is contained in:
@@ -759,10 +759,13 @@ public class YoutubeStreamExtractor extends StreamExtractor {
|
|||||||
result.getObject("compactPlaylistRenderer"));
|
result.getObject("compactPlaylistRenderer"));
|
||||||
} else if (result.has("lockupViewModel")) {
|
} else if (result.has("lockupViewModel")) {
|
||||||
final JsonObject lockupViewModel = result.getObject("lockupViewModel");
|
final JsonObject lockupViewModel = result.getObject("lockupViewModel");
|
||||||
if ("LOCKUP_CONTENT_TYPE_PLAYLIST".equals(
|
final String contentType = lockupViewModel.getString("contentType");
|
||||||
lockupViewModel.getString("contentType"))) {
|
if ("LOCKUP_CONTENT_TYPE_PLAYLIST".equals(contentType)) {
|
||||||
return new YoutubeMixOrPlaylistLockupInfoItemExtractor(
|
return new YoutubeMixOrPlaylistLockupInfoItemExtractor(
|
||||||
lockupViewModel);
|
lockupViewModel);
|
||||||
|
} else if ("LOCKUP_CONTENT_TYPE_VIDEO".equals(contentType)) {
|
||||||
|
return new YoutubeStreamInfoItemLockupExtractor(
|
||||||
|
lockupViewModel, timeAgoParser);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
@@ -0,0 +1,278 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2016 Christian Schabesberger <chris.schabesberger@mailbox.org>
|
||||||
|
* YoutubeStreamInfoItemExtractor.java is part of NewPipe Extractor.
|
||||||
|
*
|
||||||
|
* NewPipe Extractor is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* NewPipe Extractor is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with NewPipe Extractor. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.schabi.newpipe.extractor.services.youtube.extractors;
|
||||||
|
|
||||||
|
import static org.schabi.newpipe.extractor.utils.Utils.isNullOrEmpty;
|
||||||
|
|
||||||
|
import com.grack.nanojson.JsonObject;
|
||||||
|
|
||||||
|
import org.schabi.newpipe.extractor.Image;
|
||||||
|
import org.schabi.newpipe.extractor.exceptions.ParsingException;
|
||||||
|
import org.schabi.newpipe.extractor.localization.DateWrapper;
|
||||||
|
import org.schabi.newpipe.extractor.localization.TimeAgoParser;
|
||||||
|
import org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper;
|
||||||
|
import org.schabi.newpipe.extractor.services.youtube.linkHandler.YoutubeChannelLinkHandlerFactory;
|
||||||
|
import org.schabi.newpipe.extractor.services.youtube.linkHandler.YoutubeStreamLinkHandlerFactory;
|
||||||
|
import org.schabi.newpipe.extractor.stream.StreamInfoItemExtractor;
|
||||||
|
import org.schabi.newpipe.extractor.stream.StreamType;
|
||||||
|
import org.schabi.newpipe.extractor.utils.JsonUtils;
|
||||||
|
import org.schabi.newpipe.extractor.utils.Utils;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import javax.annotation.Nonnull;
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
|
public class YoutubeStreamInfoItemLockupExtractor implements StreamInfoItemExtractor {
|
||||||
|
|
||||||
|
private static final String NO_VIEWS_LOWERCASE = "no views";
|
||||||
|
|
||||||
|
private final JsonObject lockupViewModel;
|
||||||
|
private final TimeAgoParser timeAgoParser;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an extractor of StreamInfoItems from a YouTube page.
|
||||||
|
*
|
||||||
|
* @param lockupViewModel The JSON page element
|
||||||
|
* @param timeAgoParser A parser of the textual dates or {@code null}.
|
||||||
|
*/
|
||||||
|
public YoutubeStreamInfoItemLockupExtractor(final JsonObject lockupViewModel,
|
||||||
|
@Nullable final TimeAgoParser timeAgoParser) {
|
||||||
|
this.lockupViewModel = lockupViewModel;
|
||||||
|
this.timeAgoParser = timeAgoParser;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public StreamType getStreamType() {
|
||||||
|
// TODO only encountered video streams so far... Are there more types?
|
||||||
|
return StreamType.VIDEO_STREAM;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isAd() throws ParsingException {
|
||||||
|
if (isPremium()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
final String name = getName(); // only get it once
|
||||||
|
return "[Private video]".equals(name)
|
||||||
|
|| "[Deleted video]".equals(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getUrl() throws ParsingException {
|
||||||
|
try {
|
||||||
|
final String videoId = lockupViewModel.getString("contentId");
|
||||||
|
return YoutubeStreamLinkHandlerFactory.getInstance().getUrl(videoId);
|
||||||
|
} catch (final Exception e) {
|
||||||
|
throw new ParsingException("Could not get url", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getName() throws ParsingException {
|
||||||
|
// TODO Is there Formatting?
|
||||||
|
final String name = JsonUtils.getString(lockupViewModel,
|
||||||
|
"metadata.lockupMetadataViewModel.title.content");
|
||||||
|
if (!isNullOrEmpty(name)) {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
throw new ParsingException("Could not get name");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long getDuration() throws ParsingException {
|
||||||
|
final List<String> potentialDurations = lockupViewModel
|
||||||
|
.getObject("contentImage")
|
||||||
|
.getObject("thumbnailViewModel")
|
||||||
|
.getArray("overlays")
|
||||||
|
.streamAsJsonObjects()
|
||||||
|
.flatMap(jsonObject -> jsonObject
|
||||||
|
.getObject("thumbnailOverlayBadgeViewModel")
|
||||||
|
.getArray("thumbnailBadges")
|
||||||
|
.streamAsJsonObjects())
|
||||||
|
.map(jsonObject -> jsonObject
|
||||||
|
.getObject("thumbnailBadgeViewModel")
|
||||||
|
.getString("text"))
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
|
||||||
|
if (potentialDurations.isEmpty()) {
|
||||||
|
throw new ParsingException("Could not get duration: No parsable durations detected");
|
||||||
|
}
|
||||||
|
|
||||||
|
ParsingException parsingException = null;
|
||||||
|
for (final String potentialDuration : potentialDurations) {
|
||||||
|
try {
|
||||||
|
return YoutubeParsingHelper.parseDurationString(potentialDuration);
|
||||||
|
} catch (final ParsingException ex) {
|
||||||
|
parsingException = ex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new ParsingException("Could not get duration", parsingException);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getUploaderName() throws ParsingException {
|
||||||
|
return metadataPart(0, 0)
|
||||||
|
.map(this::getTextContentFromMetadataPart)
|
||||||
|
.filter(s -> !isNullOrEmpty(s))
|
||||||
|
.orElseThrow(() -> new ParsingException("Could not get uploader name"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getUploaderUrl() throws ParsingException {
|
||||||
|
final String channelId = JsonUtils.getString(lockupViewModel,
|
||||||
|
"metadata.lockupMetadataViewModel.image.decoratedAvatarViewModel"
|
||||||
|
+ ".rendererContext.commandContext.onTap"
|
||||||
|
+ ".innertubeCommand.browseEndpoint.browseId");
|
||||||
|
if (isNullOrEmpty(channelId)) {
|
||||||
|
throw new ParsingException("Could not get uploader url");
|
||||||
|
}
|
||||||
|
return YoutubeChannelLinkHandlerFactory.getInstance().getUrl(channelId);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nonnull
|
||||||
|
@Override
|
||||||
|
public List<Image> getUploaderAvatars() throws ParsingException {
|
||||||
|
return YoutubeParsingHelper.getImagesFromThumbnailsArray(
|
||||||
|
JsonUtils.getArray(lockupViewModel,
|
||||||
|
"metadata.lockupMetadataViewModel.image.decoratedAvatarViewModel"
|
||||||
|
+ ".avatar.avatarViewModel.image.sources"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isUploaderVerified() throws ParsingException {
|
||||||
|
return metadataPart(0, 0)
|
||||||
|
.stream()
|
||||||
|
.flatMap(jsonObject -> jsonObject
|
||||||
|
.getObject("text")
|
||||||
|
.getArray("attachmentRuns")
|
||||||
|
.streamAsJsonObjects())
|
||||||
|
.flatMap(jsonObject -> jsonObject
|
||||||
|
.getObject("element")
|
||||||
|
.getObject("type")
|
||||||
|
.getObject("imageType")
|
||||||
|
.getObject("image")
|
||||||
|
.getArray("sources")
|
||||||
|
.streamAsJsonObjects())
|
||||||
|
.map(jsonObject -> jsonObject
|
||||||
|
.getObject("clientResource")
|
||||||
|
.getString("imageName"))
|
||||||
|
.map("CHECK_CIRCLE_FILLED"::equals)
|
||||||
|
.findFirst()
|
||||||
|
.orElse(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
@Override
|
||||||
|
public String getTextualUploadDate() throws ParsingException {
|
||||||
|
return metadataPart(1, 1)
|
||||||
|
.map(this::getTextContentFromMetadataPart)
|
||||||
|
.orElse(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
@Override
|
||||||
|
public DateWrapper getUploadDate() throws ParsingException {
|
||||||
|
if (timeAgoParser == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return timeAgoParser.parse(getTextualUploadDate());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long getViewCount() throws ParsingException {
|
||||||
|
if (isPremium() || isPremiere()) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO Check if this is the same for shorts
|
||||||
|
final Optional<String> optTextContent = metadataPart(1, 0)
|
||||||
|
.map(this::getTextContentFromMetadataPart);
|
||||||
|
// We could do this inline if the ParsingException would be a RuntimeException -.-
|
||||||
|
if (optTextContent.isPresent()) {
|
||||||
|
return getViewCountFromViewCountText(optTextContent.get());
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected long getViewCountFromViewCountText(@Nonnull final String viewCountText)
|
||||||
|
throws NumberFormatException, ParsingException {
|
||||||
|
// These approaches are language dependent
|
||||||
|
if (viewCountText.toLowerCase().contains(NO_VIEWS_LOWERCASE)) {
|
||||||
|
return 0;
|
||||||
|
} else if (viewCountText.toLowerCase().contains("recommended")) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Utils.mixedNumberWordToLong(viewCountText);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nonnull
|
||||||
|
@Override
|
||||||
|
public List<Image> getThumbnails() throws ParsingException {
|
||||||
|
return YoutubeParsingHelper.getImagesFromThumbnailsArray(
|
||||||
|
JsonUtils.getArray(lockupViewModel,
|
||||||
|
"contentImage.thumbnailViewModel.image.sources"));
|
||||||
|
}
|
||||||
|
|
||||||
|
protected boolean isPremium() {
|
||||||
|
// TODO Detect with samples
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected boolean isPremiere() {
|
||||||
|
// TODO Detect with samples
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Optional<JsonObject> metadataPart(final int rowIndex, final int partIndex)
|
||||||
|
throws ParsingException {
|
||||||
|
return JsonUtils.getArray(lockupViewModel,
|
||||||
|
"metadata.lockupMetadataViewModel.metadata"
|
||||||
|
+ ".contentMetadataViewModel.metadataRows")
|
||||||
|
.streamAsJsonObjects()
|
||||||
|
.skip(rowIndex)
|
||||||
|
.limit(1)
|
||||||
|
.flatMap(jsonObject -> jsonObject.getArray("metadataParts")
|
||||||
|
.streamAsJsonObjects()
|
||||||
|
.skip(partIndex)
|
||||||
|
.limit(1))
|
||||||
|
.findFirst();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected String getTextContentFromMetadataPart(final JsonObject metadataPart) {
|
||||||
|
return metadataPart.getObject("text").getString("content");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
@Override
|
||||||
|
public String getShortDescription() throws ParsingException {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isShortFormContent() throws ParsingException {
|
||||||
|
// TODO Detect with samples
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
Reference in New Issue
Block a user