Merge pull request #996 from TeamNewPipe/feat/peeertube-playlists
[PeerTube] Support searching for playlists and channels
This commit is contained in:
		
						commit
						ce15f7cc50
					
				
					 7 changed files with 175 additions and 7 deletions
				
			
		|  | @ -2,10 +2,14 @@ package org.schabi.newpipe.extractor.services.peertube; | ||||||
| 
 | 
 | ||||||
| import com.grack.nanojson.JsonArray; | import com.grack.nanojson.JsonArray; | ||||||
| import com.grack.nanojson.JsonObject; | import com.grack.nanojson.JsonObject; | ||||||
|  | 
 | ||||||
|  | import org.schabi.newpipe.extractor.InfoItemExtractor; | ||||||
| import org.schabi.newpipe.extractor.InfoItemsCollector; | import org.schabi.newpipe.extractor.InfoItemsCollector; | ||||||
| import org.schabi.newpipe.extractor.Page; | import org.schabi.newpipe.extractor.Page; | ||||||
| import org.schabi.newpipe.extractor.exceptions.ContentNotAvailableException; | import org.schabi.newpipe.extractor.exceptions.ContentNotAvailableException; | ||||||
| import org.schabi.newpipe.extractor.exceptions.ParsingException; | import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||||
|  | import org.schabi.newpipe.extractor.services.peertube.extractors.PeertubeChannelInfoItemExtractor; | ||||||
|  | import org.schabi.newpipe.extractor.services.peertube.extractors.PeertubePlaylistInfoItemExtractor; | ||||||
| import org.schabi.newpipe.extractor.services.peertube.extractors.PeertubeSepiaStreamInfoItemExtractor; | import org.schabi.newpipe.extractor.services.peertube.extractors.PeertubeSepiaStreamInfoItemExtractor; | ||||||
| import org.schabi.newpipe.extractor.services.peertube.extractors.PeertubeStreamInfoItemExtractor; | import org.schabi.newpipe.extractor.services.peertube.extractors.PeertubeStreamInfoItemExtractor; | ||||||
| import org.schabi.newpipe.extractor.utils.JsonUtils; | import org.schabi.newpipe.extractor.utils.JsonUtils; | ||||||
|  | @ -101,10 +105,16 @@ public final class PeertubeParsingHelper { | ||||||
|                 if (item.has("video")) { |                 if (item.has("video")) { | ||||||
|                     item = item.getObject("video"); |                     item = item.getObject("video"); | ||||||
|                 } |                 } | ||||||
|  |                 final boolean isPlaylistInfoItem = item.has("videosLength"); | ||||||
|  |                 final boolean isChannelInfoItem = item.has("followersCount"); | ||||||
| 
 | 
 | ||||||
|                 final PeertubeStreamInfoItemExtractor extractor; |                 final InfoItemExtractor extractor; | ||||||
|                 if (sepia) { |                 if (sepia) { | ||||||
|                     extractor = new PeertubeSepiaStreamInfoItemExtractor(item, baseUrl); |                     extractor = new PeertubeSepiaStreamInfoItemExtractor(item, baseUrl); | ||||||
|  |                 } else if (isPlaylistInfoItem) { | ||||||
|  |                     extractor = new PeertubePlaylistInfoItemExtractor(item, baseUrl); | ||||||
|  |                 } else if (isChannelInfoItem) { | ||||||
|  |                     extractor = new PeertubeChannelInfoItemExtractor(item, baseUrl); | ||||||
|                 } else { |                 } else { | ||||||
|                     extractor = new PeertubeStreamInfoItemExtractor(item, baseUrl); |                     extractor = new PeertubeStreamInfoItemExtractor(item, baseUrl); | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,62 @@ | ||||||
|  | package org.schabi.newpipe.extractor.services.peertube.extractors; | ||||||
|  | 
 | ||||||
|  | import com.grack.nanojson.JsonObject; | ||||||
|  | import org.schabi.newpipe.extractor.channel.ChannelExtractor; | ||||||
|  | import org.schabi.newpipe.extractor.channel.ChannelInfoItemExtractor; | ||||||
|  | import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||||
|  | 
 | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | import java.util.Comparator; | ||||||
|  | 
 | ||||||
|  | public class PeertubeChannelInfoItemExtractor implements ChannelInfoItemExtractor { | ||||||
|  | 
 | ||||||
|  |     final JsonObject item; | ||||||
|  |     final JsonObject uploader; | ||||||
|  |     final String baseUrl; | ||||||
|  |     public PeertubeChannelInfoItemExtractor(@Nonnull final JsonObject item, | ||||||
|  |                                             @Nonnull final String baseUrl) { | ||||||
|  |         this.item = item; | ||||||
|  |         this.uploader = item.getObject("uploader"); | ||||||
|  |         this.baseUrl = baseUrl; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String getName() throws ParsingException { | ||||||
|  |         return item.getString("displayName"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String getUrl() throws ParsingException { | ||||||
|  |         return item.getString("url"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String getThumbnailUrl() throws ParsingException { | ||||||
|  |         return item.getArray("avatars").stream() | ||||||
|  |                 .filter(JsonObject.class::isInstance) | ||||||
|  |                 .map(JsonObject.class::cast) | ||||||
|  |                 .max(Comparator.comparingInt(avatar -> avatar.getInt("width"))) | ||||||
|  |                 .map(avatar -> baseUrl + avatar.getString("path")) | ||||||
|  |                 .orElse(null); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String getDescription() throws ParsingException { | ||||||
|  |         return item.getString("description"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public long getSubscriberCount() throws ParsingException { | ||||||
|  |         return item.getInt("followersCount"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public long getStreamCount() throws ParsingException { | ||||||
|  |         return ChannelExtractor.ITEM_COUNT_UNKNOWN; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public boolean isVerified() throws ParsingException { | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,57 @@ | ||||||
|  | package org.schabi.newpipe.extractor.services.peertube.extractors; | ||||||
|  | 
 | ||||||
|  | import com.grack.nanojson.JsonObject; | ||||||
|  | 
 | ||||||
|  | import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||||
|  | import org.schabi.newpipe.extractor.playlist.PlaylistInfoItemExtractor; | ||||||
|  | 
 | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | 
 | ||||||
|  | public class PeertubePlaylistInfoItemExtractor implements PlaylistInfoItemExtractor  { | ||||||
|  | 
 | ||||||
|  |     final JsonObject item; | ||||||
|  |     final JsonObject uploader; | ||||||
|  |     final String baseUrl; | ||||||
|  | 
 | ||||||
|  |     public PeertubePlaylistInfoItemExtractor(@Nonnull final JsonObject item, | ||||||
|  |                                              @Nonnull final String baseUrl) { | ||||||
|  |         this.item = item; | ||||||
|  |         this.uploader = item.getObject("uploader"); | ||||||
|  |         this.baseUrl = baseUrl; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String getName() throws ParsingException { | ||||||
|  |         return item.getString("displayName"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String getUrl() throws ParsingException { | ||||||
|  |         return item.getString("url"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String getThumbnailUrl() throws ParsingException { | ||||||
|  |         return baseUrl + item.getString("thumbnailPath"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String getUploaderName() throws ParsingException { | ||||||
|  |         return uploader.getString("displayName"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String getUploaderUrl() throws ParsingException { | ||||||
|  |         return uploader.getString("url"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public boolean isUploaderVerified() throws ParsingException { | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public long getStreamCount() throws ParsingException { | ||||||
|  |         return item.getInt("videosLength"); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -329,7 +329,7 @@ public class PeertubeStreamExtractor extends StreamExtractor { | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     private String getRelatedItemsUrl(@Nonnull final List<String> tags) |     private String getRelatedItemsUrl(@Nonnull final List<String> tags) | ||||||
|             throws UnsupportedEncodingException { |             throws UnsupportedEncodingException { | ||||||
|         final String url = baseUrl + PeertubeSearchQueryHandlerFactory.SEARCH_ENDPOINT; |         final String url = baseUrl + PeertubeSearchQueryHandlerFactory.SEARCH_ENDPOINT_VIDEOS; | ||||||
|         final StringBuilder params = new StringBuilder(); |         final StringBuilder params = new StringBuilder(); | ||||||
|         params.append("start=0&count=8&sort=-createdAt"); |         params.append("start=0&count=8&sort=-createdAt"); | ||||||
|         for (final String tag : tags) { |         for (final String tag : tags) { | ||||||
|  |  | ||||||
|  | @ -13,6 +13,7 @@ public final class PeertubePlaylistLinkHandlerFactory extends ListLinkHandlerFac | ||||||
|     private static final PeertubePlaylistLinkHandlerFactory INSTANCE |     private static final PeertubePlaylistLinkHandlerFactory INSTANCE | ||||||
|             = new PeertubePlaylistLinkHandlerFactory(); |             = new PeertubePlaylistLinkHandlerFactory(); | ||||||
|     private static final String ID_PATTERN = "(/videos/watch/playlist/|/w/p/)([^/?&#]*)"; |     private static final String ID_PATTERN = "(/videos/watch/playlist/|/w/p/)([^/?&#]*)"; | ||||||
|  |     private static final String API_ID_PATTERN = "/video-playlists/([^/?&#]*)"; | ||||||
| 
 | 
 | ||||||
|     private PeertubePlaylistLinkHandlerFactory() { |     private PeertubePlaylistLinkHandlerFactory() { | ||||||
|     } |     } | ||||||
|  | @ -38,7 +39,12 @@ public final class PeertubePlaylistLinkHandlerFactory extends ListLinkHandlerFac | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|     public String getId(final String url) throws ParsingException { |     public String getId(final String url) throws ParsingException { | ||||||
|         return Parser.matchGroup(ID_PATTERN, url, 2); |         try { | ||||||
|  |             return Parser.matchGroup(ID_PATTERN, url, 2); | ||||||
|  |         } catch (final ParsingException ignored) { | ||||||
|  |             // might also be an API url, no reason to throw an exception here | ||||||
|  |         } | ||||||
|  |         return Parser.matchGroup1(API_ID_PATTERN, url); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|  |  | ||||||
|  | @ -12,8 +12,12 @@ public final class PeertubeSearchQueryHandlerFactory extends SearchQueryHandlerF | ||||||
| 
 | 
 | ||||||
|     public static final String VIDEOS = "videos"; |     public static final String VIDEOS = "videos"; | ||||||
|     public static final String SEPIA_VIDEOS = "sepia_videos"; // sepia is the global index |     public static final String SEPIA_VIDEOS = "sepia_videos"; // sepia is the global index | ||||||
|  |     public static final String PLAYLISTS = "playlists"; | ||||||
|  |     public static final String CHANNELS = "channels"; | ||||||
|     public static final String SEPIA_BASE_URL = "https://sepiasearch.org"; |     public static final String SEPIA_BASE_URL = "https://sepiasearch.org"; | ||||||
|     public static final String SEARCH_ENDPOINT = "/api/v1/search/videos"; |     public static final String SEARCH_ENDPOINT_PLAYLISTS = "/api/v1/search/video-playlists"; | ||||||
|  |     public static final String SEARCH_ENDPOINT_VIDEOS = "/api/v1/search/videos"; | ||||||
|  |     public static final String SEARCH_ENDPOINT_CHANNELS = "/api/v1/search/video-channels"; | ||||||
| 
 | 
 | ||||||
|     private PeertubeSearchQueryHandlerFactory() { |     private PeertubeSearchQueryHandlerFactory() { | ||||||
|     } |     } | ||||||
|  | @ -41,7 +45,17 @@ public final class PeertubeSearchQueryHandlerFactory extends SearchQueryHandlerF | ||||||
|                          final String sortFilter, |                          final String sortFilter, | ||||||
|                          final String baseUrl) throws ParsingException { |                          final String baseUrl) throws ParsingException { | ||||||
|         try { |         try { | ||||||
|             return baseUrl + SEARCH_ENDPOINT + "?search=" + Utils.encodeUrlUtf8(searchString); |             final String endpoint; | ||||||
|  |             if (contentFilters.isEmpty() | ||||||
|  |                     || contentFilters.get(0).equals(VIDEOS) | ||||||
|  |                     || contentFilters.get(0).equals(SEPIA_VIDEOS)) { | ||||||
|  |                 endpoint = SEARCH_ENDPOINT_VIDEOS; | ||||||
|  |             } else if (contentFilters.get(0).equals(CHANNELS)) { | ||||||
|  |                 endpoint = SEARCH_ENDPOINT_CHANNELS; | ||||||
|  |             } else { | ||||||
|  |                 endpoint = SEARCH_ENDPOINT_PLAYLISTS; | ||||||
|  |             } | ||||||
|  |             return baseUrl + endpoint + "?search=" + Utils.encodeUrlUtf8(searchString); | ||||||
|         } catch (final UnsupportedEncodingException e) { |         } catch (final UnsupportedEncodingException e) { | ||||||
|             throw new ParsingException("Could not encode query", e); |             throw new ParsingException("Could not encode query", e); | ||||||
|         } |         } | ||||||
|  | @ -51,7 +65,9 @@ public final class PeertubeSearchQueryHandlerFactory extends SearchQueryHandlerF | ||||||
|     public String[] getAvailableContentFilter() { |     public String[] getAvailableContentFilter() { | ||||||
|         return new String[]{ |         return new String[]{ | ||||||
|                 VIDEOS, |                 VIDEOS, | ||||||
|                 SEPIA_VIDEOS |                 PLAYLISTS, | ||||||
|  |                 CHANNELS, | ||||||
|  |                 SEPIA_VIDEOS, | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -18,13 +18,30 @@ public class PeertubeSearchQHTest { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     public void testRegularValues() throws Exception { |     void testVideoSearch() throws Exception { | ||||||
|         assertEquals("https://peertube.mastodon.host/api/v1/search/videos?search=asdf", PeerTube.getSearchQHFactory().fromQuery("asdf").getUrl()); |         assertEquals("https://peertube.mastodon.host/api/v1/search/videos?search=asdf", PeerTube.getSearchQHFactory().fromQuery("asdf").getUrl()); | ||||||
|         assertEquals("https://peertube.mastodon.host/api/v1/search/videos?search=hans", PeerTube.getSearchQHFactory().fromQuery("hans").getUrl()); |         assertEquals("https://peertube.mastodon.host/api/v1/search/videos?search=hans", PeerTube.getSearchQHFactory().fromQuery("hans").getUrl()); | ||||||
|         assertEquals("https://peertube.mastodon.host/api/v1/search/videos?search=Poifj%26jaijf", PeerTube.getSearchQHFactory().fromQuery("Poifj&jaijf").getUrl()); |         assertEquals("https://peertube.mastodon.host/api/v1/search/videos?search=Poifj%26jaijf", PeerTube.getSearchQHFactory().fromQuery("Poifj&jaijf").getUrl()); | ||||||
|         assertEquals("https://peertube.mastodon.host/api/v1/search/videos?search=G%C3%BCl%C3%BCm", PeerTube.getSearchQHFactory().fromQuery("Gülüm").getUrl()); |         assertEquals("https://peertube.mastodon.host/api/v1/search/videos?search=G%C3%BCl%C3%BCm", PeerTube.getSearchQHFactory().fromQuery("Gülüm").getUrl()); | ||||||
|         assertEquals("https://peertube.mastodon.host/api/v1/search/videos?search=%3Fj%24%29H%C2%A7B", PeerTube.getSearchQHFactory().fromQuery("?j$)H§B").getUrl()); |         assertEquals("https://peertube.mastodon.host/api/v1/search/videos?search=%3Fj%24%29H%C2%A7B", PeerTube.getSearchQHFactory().fromQuery("?j$)H§B").getUrl()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     void testSepiaVideoSearch() throws Exception { | ||||||
|         assertEquals("https://sepiasearch.org/api/v1/search/videos?search=%3Fj%24%29H%C2%A7B", PeerTube.getSearchQHFactory().fromQuery("?j$)H§B", singletonList(PeertubeSearchQueryHandlerFactory.SEPIA_VIDEOS), "").getUrl()); |         assertEquals("https://sepiasearch.org/api/v1/search/videos?search=%3Fj%24%29H%C2%A7B", PeerTube.getSearchQHFactory().fromQuery("?j$)H§B", singletonList(PeertubeSearchQueryHandlerFactory.SEPIA_VIDEOS), "").getUrl()); | ||||||
|         assertEquals("https://anotherpeertubeindex.com/api/v1/search/videos?search=%3Fj%24%29H%C2%A7B", PeerTube.getSearchQHFactory().fromQuery("?j$)H§B", singletonList(PeertubeSearchQueryHandlerFactory.SEPIA_VIDEOS), "", "https://anotherpeertubeindex.com").getUrl()); |         assertEquals("https://anotherpeertubeindex.com/api/v1/search/videos?search=%3Fj%24%29H%C2%A7B", PeerTube.getSearchQHFactory().fromQuery("?j$)H§B", singletonList(PeertubeSearchQueryHandlerFactory.SEPIA_VIDEOS), "", "https://anotherpeertubeindex.com").getUrl()); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     void testPlaylistSearch() throws Exception { | ||||||
|  |         assertEquals("https://peertube.mastodon.host/api/v1/search/video-playlists?search=asdf", PeerTube.getSearchQHFactory().fromQuery("asdf", singletonList(PeertubeSearchQueryHandlerFactory.PLAYLISTS), "").getUrl()); | ||||||
|  |         assertEquals("https://peertube.mastodon.host/api/v1/search/video-playlists?search=hans", PeerTube.getSearchQHFactory().fromQuery("hans", singletonList(PeertubeSearchQueryHandlerFactory.PLAYLISTS), "").getUrl()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     void testChannelSearch() throws Exception { | ||||||
|  |         assertEquals("https://peertube.mastodon.host/api/v1/search/video-channels?search=asdf", PeerTube.getSearchQHFactory().fromQuery("asdf", singletonList(PeertubeSearchQueryHandlerFactory.CHANNELS), "").getUrl()); | ||||||
|  |         assertEquals("https://peertube.mastodon.host/api/v1/search/video-channels?search=hans", PeerTube.getSearchQHFactory().fromQuery("hans", singletonList(PeertubeSearchQueryHandlerFactory.CHANNELS), "").getUrl()); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
| } | } | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue