Skip to content

Commit 14ef430

Browse files
authored
Merge pull request #920 from AudricV/fix-yt-trending-extraction
[YouTube] Fix extraction of the Trending kiosk with the new data model returned
2 parents 884da40 + d8ddeb5 commit 14ef430

1 file changed

Lines changed: 63 additions & 23 deletions

File tree

extractor/src/main/java/org/schabi/newpipe/extractor/services/youtube/extractors/YoutubeTrendingExtractor.java

Lines changed: 63 additions & 23 deletions
Original file line numberDiff line numberDiff line change
@@ -1,26 +1,25 @@
1-
package org.schabi.newpipe.extractor.services.youtube.extractors;
2-
31
/*
42
* Created by Christian Schabesberger on 12.08.17.
53
*
64
* Copyright (C) Christian Schabesberger 2018 <chris.schabesberger@mailbox.org>
7-
* YoutubeTrendingExtractor.java is part of NewPipe.
5+
* YoutubeTrendingExtractor.java is part of NewPipe Extractor.
86
*
9-
* NewPipe is free software: you can redistribute it and/or modify
7+
* NewPipe Extractor is free software: you can redistribute it and/or modify
108
* it under the terms of the GNU General Public License as published by
119
* the Free Software Foundation, either version 3 of the License, or
1210
* (at your option) any later version.
1311
*
14-
* NewPipe is distributed in the hope that it will be useful,
12+
* NewPipe Extractor is distributed in the hope that it will be useful,
1513
* but WITHOUT ANY WARRANTY; without even the implied warranty of
16-
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
1715
* GNU General Public License for more details.
1816
*
1917
* You should have received a copy of the GNU General Public License
20-
* along with NewPipe. If not, see <http://www.gnu.org/licenses/>.
18+
* along with NewPipe Extractor. If not, see <https://www.gnu.org/licenses/>.
2119
*/
2220

23-
import com.grack.nanojson.JsonArray;
21+
package org.schabi.newpipe.extractor.services.youtube.extractors;
22+
2423
import com.grack.nanojson.JsonObject;
2524
import com.grack.nanojson.JsonWriter;
2625

@@ -92,25 +91,66 @@ public String getName() throws ParsingException {
9291

9392
@Nonnull
9493
@Override
95-
public InfoItemsPage<StreamInfoItem> getInitialPage() {
94+
public InfoItemsPage<StreamInfoItem> getInitialPage() throws ParsingException {
9695
final StreamInfoItemsCollector collector = new StreamInfoItemsCollector(getServiceId());
9796
final TimeAgoParser timeAgoParser = getTimeAgoParser();
98-
final JsonArray itemSectionRenderers = initialData.getObject("contents")
99-
.getObject("twoColumnBrowseResultsRenderer").getArray("tabs").getObject(0)
100-
.getObject("tabRenderer").getObject("content").getObject("sectionListRenderer")
101-
.getArray("contents");
102-
103-
for (final Object itemSectionRenderer : itemSectionRenderers) {
104-
final JsonObject expandedShelfContentsRenderer = ((JsonObject) itemSectionRenderer)
105-
.getObject("itemSectionRenderer").getArray("contents").getObject(0)
106-
.getObject("shelfRenderer").getObject("content")
107-
.getObject("expandedShelfContentsRenderer");
108-
for (final Object ul : expandedShelfContentsRenderer.getArray("items")) {
109-
final JsonObject videoInfo = ((JsonObject) ul).getObject("videoRenderer");
110-
collector.commit(new YoutubeStreamInfoItemExtractor(videoInfo, timeAgoParser));
111-
}
97+
final JsonObject tabContent = getTrendingTabContent();
98+
99+
if (tabContent.has("richGridRenderer")) {
100+
tabContent.getObject("richGridRenderer")
101+
.getArray("contents")
102+
.stream()
103+
.filter(JsonObject.class::isInstance)
104+
.map(JsonObject.class::cast)
105+
// Filter Trending shorts and Recently trending sections
106+
.filter(content -> content.has("richItemRenderer"))
107+
.map(content -> content.getObject("richItemRenderer")
108+
.getObject("content")
109+
.getObject("videoRenderer"))
110+
.forEachOrdered(videoRenderer -> collector.commit(
111+
new YoutubeStreamInfoItemExtractor(videoRenderer, timeAgoParser)));
112+
} else if (tabContent.has("sectionListRenderer")) {
113+
tabContent.getObject("sectionListRenderer")
114+
.getArray("contents")
115+
.stream()
116+
.filter(JsonObject.class::isInstance)
117+
.map(JsonObject.class::cast)
118+
.flatMap(content -> content.getObject("itemSectionRenderer")
119+
.getArray("contents")
120+
.stream())
121+
.filter(JsonObject.class::isInstance)
122+
.map(JsonObject.class::cast)
123+
.map(content -> content.getObject("shelfRenderer"))
124+
// Filter Trending shorts and Recently trending sections which have a title,
125+
// contrary to normal trends
126+
.filter(shelfRenderer -> !shelfRenderer.has("title"))
127+
.flatMap(shelfRenderer -> shelfRenderer.getObject("content")
128+
.getObject("expandedShelfContentsRenderer")
129+
.getArray("items")
130+
.stream())
131+
.filter(JsonObject.class::isInstance)
132+
.map(JsonObject.class::cast)
133+
.map(item -> item.getObject("videoRenderer"))
134+
.forEachOrdered(videoRenderer -> collector.commit(
135+
new YoutubeStreamInfoItemExtractor(videoRenderer, timeAgoParser)));
112136
}
113137

114138
return new InfoItemsPage<>(collector, null);
115139
}
140+
141+
private JsonObject getTrendingTabContent() throws ParsingException {
142+
return initialData.getObject("contents")
143+
.getObject("twoColumnBrowseResultsRenderer")
144+
.getArray("tabs")
145+
.stream()
146+
.filter(JsonObject.class::isInstance)
147+
.map(JsonObject.class::cast)
148+
.map(tab -> tab.getObject("tabRenderer"))
149+
.filter(tabRenderer -> tabRenderer.getBoolean("selected"))
150+
.filter(tabRenderer -> tabRenderer.has("content"))
151+
// There should be at most one tab selected
152+
.findFirst()
153+
.orElseThrow(() -> new ParsingException("Could not get \"Now\" trending tab"))
154+
.getObject("content");
155+
}
116156
}

0 commit comments

Comments
 (0)