mirror of
https://github.com/TeamNewPipe/NewPipeExtractor.git
synced 2025-04-29 00:10:35 +05:30
[SoundCloud] Fix extractors built from next playlist pages
They didn't have the information to calculate another next page url. So now `nextPageUrl` contains a full link with all video ids, and `getPage` takes the first part of the url (containing 15 streams) and produces another `nextPageUrl` with the remaining streams. Also add a test for this.
This commit is contained in:
parent
0e1b4bbf17
commit
5e4ddb368f
@ -30,8 +30,6 @@ public class SoundcloudPlaylistExtractor extends PlaylistExtractor {
|
|||||||
private JsonObject playlist;
|
private JsonObject playlist;
|
||||||
|
|
||||||
private StreamInfoItemsCollector streamInfoItemsCollector;
|
private StreamInfoItemsCollector streamInfoItemsCollector;
|
||||||
private List<Integer> nextTrackIds;
|
|
||||||
private int nextTrackIdsIndex;
|
|
||||||
private String nextPageUrl;
|
private String nextPageUrl;
|
||||||
|
|
||||||
public SoundcloudPlaylistExtractor(StreamingService service, ListLinkHandler linkHandler) {
|
public SoundcloudPlaylistExtractor(StreamingService service, ListLinkHandler linkHandler) {
|
||||||
@ -121,15 +119,16 @@ public class SoundcloudPlaylistExtractor extends PlaylistExtractor {
|
|||||||
@Override
|
@Override
|
||||||
public InfoItemsPage<StreamInfoItem> getInitialPage() throws IOException, ExtractionException {
|
public InfoItemsPage<StreamInfoItem> getInitialPage() throws IOException, ExtractionException {
|
||||||
if (streamInfoItemsCollector == null) {
|
if (streamInfoItemsCollector == null) {
|
||||||
computeInitialTracksAndNextIds();
|
computeInitialTracksAndNextPageUrl();
|
||||||
}
|
}
|
||||||
return new InfoItemsPage<>(streamInfoItemsCollector, getNextPageUrl());
|
return new InfoItemsPage<>(streamInfoItemsCollector, nextPageUrl);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void computeInitialTracksAndNextIds() {
|
private void computeInitialTracksAndNextPageUrl() throws IOException, ExtractionException {
|
||||||
streamInfoItemsCollector = new StreamInfoItemsCollector(getServiceId());
|
streamInfoItemsCollector = new StreamInfoItemsCollector(getServiceId());
|
||||||
nextTrackIds = new ArrayList<>();
|
StringBuilder nextPageUrlBuilder = new StringBuilder("https://api-v2.soundcloud.com/tracks?client_id=");
|
||||||
nextTrackIdsIndex = 0;
|
nextPageUrlBuilder.append(SoundcloudParsingHelper.clientId());
|
||||||
|
nextPageUrlBuilder.append("&ids=");
|
||||||
|
|
||||||
JsonArray tracks = playlist.getArray("tracks");
|
JsonArray tracks = playlist.getArray("tracks");
|
||||||
for (Object o : tracks) {
|
for (Object o : tracks) {
|
||||||
@ -138,39 +137,23 @@ public class SoundcloudPlaylistExtractor extends PlaylistExtractor {
|
|||||||
if (track.has("title")) { // i.e. if full info is available
|
if (track.has("title")) { // i.e. if full info is available
|
||||||
streamInfoItemsCollector.commit(new SoundcloudStreamInfoItemExtractor(track));
|
streamInfoItemsCollector.commit(new SoundcloudStreamInfoItemExtractor(track));
|
||||||
} else {
|
} else {
|
||||||
nextTrackIds.add(track.getInt("id"));
|
// %09d would be enough, but a 0 before the number does not create problems, so let's be sure
|
||||||
}
|
nextPageUrlBuilder.append(String.format("%010d,", track.getInt("id")));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void computeAnotherNextPageUrl() throws IOException, ExtractionException {
|
nextPageUrl = nextPageUrlBuilder.toString();
|
||||||
if (nextTrackIds == null || nextTrackIdsIndex >= nextTrackIds.size()) {
|
if (nextPageUrl.endsWith("&ids=")) {
|
||||||
nextPageUrl = ""; // there are no more tracks
|
// there are no other videos
|
||||||
return;
|
nextPageUrl = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
StringBuilder urlBuilder = new StringBuilder("https://api-v2.soundcloud.com/tracks?client_id=");
|
|
||||||
urlBuilder.append(SoundcloudParsingHelper.clientId());
|
|
||||||
urlBuilder.append("&ids=");
|
|
||||||
|
|
||||||
int upperIndex = Math.min(nextTrackIdsIndex + streamsPerRequestedPage, nextTrackIds.size());
|
|
||||||
for (int i = nextTrackIdsIndex; i < upperIndex; ++i) {
|
|
||||||
urlBuilder.append(nextTrackIds.get(i));
|
|
||||||
urlBuilder.append(","); // a , at the end is ok
|
|
||||||
}
|
|
||||||
|
|
||||||
nextTrackIdsIndex = upperIndex;
|
|
||||||
nextPageUrl = urlBuilder.toString();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getNextPageUrl() throws IOException, ExtractionException {
|
public String getNextPageUrl() throws IOException, ExtractionException {
|
||||||
if (nextPageUrl == null) {
|
if (nextPageUrl == null) {
|
||||||
if (nextTrackIds == null) {
|
computeInitialTracksAndNextPageUrl();
|
||||||
computeInitialTracksAndNextIds();
|
|
||||||
}
|
|
||||||
computeAnotherNextPageUrl();
|
|
||||||
}
|
}
|
||||||
return nextPageUrl;
|
return nextPageUrl;
|
||||||
}
|
}
|
||||||
@ -181,8 +164,24 @@ public class SoundcloudPlaylistExtractor extends PlaylistExtractor {
|
|||||||
throw new ExtractionException(new IllegalArgumentException("Page url is empty or null"));
|
throw new ExtractionException(new IllegalArgumentException("Page url is empty or null"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// see computeInitialTracksAndNextPageUrl
|
||||||
|
final int lengthFirstPartOfUrl = ("https://api-v2.soundcloud.com/tracks?client_id="
|
||||||
|
+ SoundcloudParsingHelper.clientId()
|
||||||
|
+ "&ids=").length();
|
||||||
|
final int lengthOfEveryStream = 11;
|
||||||
|
|
||||||
|
String currentPageUrl;
|
||||||
|
int lengthMaxStreams = lengthFirstPartOfUrl + lengthOfEveryStream * streamsPerRequestedPage;
|
||||||
|
if (pageUrl.length() <= lengthMaxStreams) {
|
||||||
|
currentPageUrl = pageUrl; // fetch every remaining video, there are less than the max
|
||||||
|
nextPageUrl = ""; // afterwards the list is complete
|
||||||
|
} else {
|
||||||
|
currentPageUrl = pageUrl.substring(0, lengthMaxStreams);
|
||||||
|
nextPageUrl = pageUrl.substring(0, lengthFirstPartOfUrl) + pageUrl.substring(lengthMaxStreams);
|
||||||
|
}
|
||||||
|
|
||||||
StreamInfoItemsCollector collector = new StreamInfoItemsCollector(getServiceId());
|
StreamInfoItemsCollector collector = new StreamInfoItemsCollector(getServiceId());
|
||||||
String response = NewPipe.getDownloader().get(pageUrl, getExtractorLocalization()).responseBody();
|
String response = NewPipe.getDownloader().get(currentPageUrl, getExtractorLocalization()).responseBody();
|
||||||
|
|
||||||
try {
|
try {
|
||||||
JsonArray tracks = JsonParser.array().from(response);
|
JsonArray tracks = JsonParser.array().from(response);
|
||||||
@ -195,7 +194,6 @@ public class SoundcloudPlaylistExtractor extends PlaylistExtractor {
|
|||||||
throw new ParsingException("Could not parse json response", e);
|
throw new ParsingException("Could not parse json response", e);
|
||||||
}
|
}
|
||||||
|
|
||||||
computeAnotherNextPageUrl();
|
|
||||||
return new InfoItemsPage<>(collector, nextPageUrl);
|
return new InfoItemsPage<>(collector, nextPageUrl);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -234,6 +234,15 @@ public class SoundcloudPlaylistExtractorTest {
|
|||||||
public void testGetPageInNewExtractor() throws Exception {
|
public void testGetPageInNewExtractor() throws Exception {
|
||||||
final PlaylistExtractor newExtractor = SoundCloud.getPlaylistExtractor(extractor.getUrl());
|
final PlaylistExtractor newExtractor = SoundCloud.getPlaylistExtractor(extractor.getUrl());
|
||||||
defaultTestGetPageInNewExtractor(extractor, newExtractor);
|
defaultTestGetPageInNewExtractor(extractor, newExtractor);
|
||||||
|
String page1 = newExtractor.getNextPageUrl();
|
||||||
|
defaultTestMoreItems(newExtractor); // there has to be another page
|
||||||
|
String page2 = newExtractor.getNextPageUrl();
|
||||||
|
defaultTestMoreItems(newExtractor); // and another one
|
||||||
|
String page3 = newExtractor.getNextPageUrl();
|
||||||
|
|
||||||
|
assertNotEquals("Same pages", page1, page2);
|
||||||
|
assertNotEquals("Same pages", page2, page3);
|
||||||
|
assertNotEquals("Same pages", page3, page1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*//////////////////////////////////////////////////////////////////////////
|
/*//////////////////////////////////////////////////////////////////////////
|
||||||
|
Loading…
x
Reference in New Issue
Block a user