merged master
This commit is contained in:
		
						commit
						df0db8468d
					
				
					 40 changed files with 763 additions and 321 deletions
				
			
		|  | @ -1,6 +1,6 @@ | |||
| # NewPipe Extractor | ||||
| 
 | ||||
| [](https://travis-ci.org/TeamNewPipe/NewPipeExtractor) [](https://jitpack.io/#TeamNewPipe/NewPipeExtractor) [Documentation](https://teamnewpipe.github.io/NewPipeExtractor/javadoc/) | ||||
| [](https://travis-ci.org/TeamNewPipe/NewPipeExtractor) [](https://jitpack.io/#TeamNewPipe/NewPipeExtractor) [Documentation](https://teamnewpipe.github.io/documentation/) | ||||
| 
 | ||||
| NewPipe Extractor is a library for extracting things from streaming sites. It is a core component of [NewPipe](https://github.com/TeamNewPipe/NewPipe), but could be used independently. | ||||
| 
 | ||||
|  |  | |||
|  | @ -36,7 +36,14 @@ public enum MediaFormat { | |||
|     M4A         (0x3,   "m4a",   "m4a",  "audio/mp4"), | ||||
|     WEBMA       (0x4,   "WebM",  "webm", "audio/webm"), | ||||
|     MP3         (0x5,   "MP3",   "mp3",  "audio/mpeg"), | ||||
|     OPUS        (0x6,   "opus",  "opus", "audio/opus"); | ||||
|     OPUS        (0x6,   "opus",  "opus", "audio/opus"), | ||||
|     // subtitles formats | ||||
|     VTT         (0x7,   "WebVTT",                      "vtt",   "text/vtt"), | ||||
|     TTML        (0x8,   "Timed Text Markup Language",  "ttml",  "application/ttml+xml"), | ||||
|     TRANSCRIPT1 (0x9,   "TranScript v1",               "srv1",  "text/xml"), | ||||
|     TRANSCRIPT2 (0xA,   "TranScript v2",               "srv2",  "text/xml"), | ||||
|     TRANSCRIPT3 (0xB,   "TranScript v3",               "srv3",  "text/xml"), | ||||
|     SRT         (0xC,   "SubRip file format",          "srt",   "text/srt"); | ||||
| 
 | ||||
|     public final int id; | ||||
|     public final String name; | ||||
|  |  | |||
|  | @ -103,7 +103,7 @@ public class NewPipe { | |||
|         NewPipe.localization = localization; | ||||
|     } | ||||
| 
 | ||||
|     public static Localization getLocalization() { | ||||
|     public static Localization getPreferredLocalization() { | ||||
|         return localization; | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -9,6 +9,24 @@ import java.util.List; | |||
| import static java.util.Arrays.asList; | ||||
| import static java.util.Collections.unmodifiableList; | ||||
| 
 | ||||
| /* | ||||
|  * Copyright (C) Christian Schabesberger 2018 <chris.schabesberger@mailbox.org> | ||||
|  * ServiceList.java is part of NewPipe. | ||||
|  * | ||||
|  * NewPipe 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 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.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| 
 | ||||
| /** | ||||
|  * A list of supported services. | ||||
|  */ | ||||
|  | @ -21,6 +39,10 @@ public final class ServiceList { | |||
|     public static final SoundcloudService SoundCloud; | ||||
|     public static final PeertubeService PeerTube;  | ||||
| 
 | ||||
|     /** | ||||
|      * When creating a new service, put this service in the end of this list, | ||||
|      * and give it the next free id. | ||||
|      */ | ||||
|     private static final List<StreamingService> SERVICES = unmodifiableList( | ||||
|             asList( | ||||
|                     YouTube = new YoutubeService(0), | ||||
|  |  | |||
|  | @ -20,12 +20,39 @@ import org.schabi.newpipe.extractor.stream.StreamExtractor; | |||
| import org.schabi.newpipe.extractor.subscription.SubscriptionExtractor; | ||||
| import org.schabi.newpipe.extractor.utils.Localization; | ||||
| 
 | ||||
| /* | ||||
|  * Copyright (C) Christian Schabesberger 2018 <chris.schabesberger@mailbox.org> | ||||
|  * StreamingService.java is part of NewPipe. | ||||
|  * | ||||
|  * NewPipe 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 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.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| 
 | ||||
| public abstract class StreamingService { | ||||
| 
 | ||||
|     /** | ||||
|      * This class holds meta information about the service implementation. | ||||
|      */ | ||||
|     public static class ServiceInfo { | ||||
|         private String name; | ||||
| 
 | ||||
|         private final List<MediaCapability> mediaCapabilities; | ||||
| 
 | ||||
|         /** | ||||
|          * Creates a new instance of a ServiceInfo | ||||
|          * @param name the name of the service | ||||
|          * @param mediaCapabilities the type of media this service can handle | ||||
|          */ | ||||
|         public ServiceInfo(String name, List<MediaCapability> mediaCapabilities) { | ||||
|             this.name = name; | ||||
|             this.mediaCapabilities = Collections.unmodifiableList(mediaCapabilities); | ||||
|  | @ -48,6 +75,10 @@ public abstract class StreamingService { | |||
|         } | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * LinkType will be used to determine which type of URL you are handling, and therefore which part | ||||
|      * of NewPipe should handle a certain URL. | ||||
|      */ | ||||
|     public enum LinkType { | ||||
|         NONE, | ||||
|         STREAM, | ||||
|  | @ -58,6 +89,16 @@ public abstract class StreamingService { | |||
|     private final int serviceId; | ||||
|     private final ServiceInfo serviceInfo; | ||||
| 
 | ||||
|     /** | ||||
|      * Creates a new Streaming service. | ||||
|      * If you Implement one do not set id within your implementation of this extractor, instead | ||||
|      * set the id when you put the extractor into | ||||
|      * <a href="https://teamnewpipe.github.io/NewPipeExtractor/javadoc/org/schabi/newpipe/extractor/ServiceList.html">ServiceList</a>. | ||||
|      * All other parameters can be set directly from the overriding constructor. | ||||
|      * @param id the number of the service to identify him within the NewPipe frontend | ||||
|      * @param name the name of the service | ||||
|      * @param capabilities the type of media this service can handle | ||||
|      */ | ||||
|     public StreamingService(int id, String name, List<ServiceInfo.MediaCapability> capabilities) { | ||||
|         this.serviceId = id; | ||||
|         this.serviceInfo = new ServiceInfo(name, capabilities); | ||||
|  | @ -79,9 +120,31 @@ public abstract class StreamingService { | |||
|     //////////////////////////////////////////// | ||||
|     // Url Id handler | ||||
|     //////////////////////////////////////////// | ||||
| 
 | ||||
|     /** | ||||
|      * Must return a new instance of an implementation of LinkHandlerFactory for streams. | ||||
|      * @return an instance of a LinkHandlerFactory for streams | ||||
|      */ | ||||
|     public abstract LinkHandlerFactory getStreamLHFactory(); | ||||
| 
 | ||||
|     /** | ||||
|      * Must return a new instance of an implementation of ListLinkHandlerFactory for channels. | ||||
|      * If support for channels is not given null must be returned. | ||||
|      * @return an instance of a ListLinkHandlerFactory for channels or null | ||||
|      */ | ||||
|     public abstract ListLinkHandlerFactory getChannelLHFactory(); | ||||
| 
 | ||||
|     /** | ||||
|      * Must return a new instance of an implementation of ListLinkHandlerFactory for playlists. | ||||
|      * If support for playlists is not given null must be returned. | ||||
|      * @return an instance of a ListLinkHandlerFactory for playlists or null | ||||
|      */ | ||||
|     public abstract ListLinkHandlerFactory getPlaylistLHFactory(); | ||||
| 
 | ||||
|     /** | ||||
|      * Must return an instance of an implementation of SearchQueryHandlerFactory. | ||||
|      * @return an instance of a SearchQueryHandlerFactory | ||||
|      */ | ||||
|     public abstract SearchQueryHandlerFactory getSearchQHFactory(); | ||||
|     public abstract ListLinkHandlerFactory getCommentsLHFactory(); | ||||
| 
 | ||||
|  | @ -89,15 +152,62 @@ public abstract class StreamingService { | |||
|     //////////////////////////////////////////// | ||||
|     // Extractor | ||||
|     //////////////////////////////////////////// | ||||
|     public abstract SearchExtractor getSearchExtractor(SearchQueryHandler queryHandler, Localization localization); | ||||
|     public abstract SuggestionExtractor getSuggestionExtractor(Localization localization); | ||||
|     public abstract SubscriptionExtractor getSubscriptionExtractor(); | ||||
|     public abstract KioskList getKioskList(Localization localization) throws ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * Must create a new instance of a SearchExtractor implementation. | ||||
|      * @param queryHandler specifies the keyword lock for, and the filters which should be applied. | ||||
|      * @param localization specifies the language/country for the extractor. | ||||
|      * @return a new SearchExtractor instance | ||||
|      */ | ||||
|     public abstract SearchExtractor getSearchExtractor(SearchQueryHandler queryHandler, Localization localization); | ||||
| 
 | ||||
|     /** | ||||
|      * Must create a new instance of a SuggestionExtractor implementation. | ||||
|      * @param localization specifies the language/country for the extractor. | ||||
|      * @return a new SuggestionExtractor instance | ||||
|      */ | ||||
|     public abstract SuggestionExtractor getSuggestionExtractor(Localization localization); | ||||
| 
 | ||||
|     /** | ||||
|      * Outdated or obsolete. null can be returned. | ||||
|      * @return just null | ||||
|      */ | ||||
|     public abstract SubscriptionExtractor getSubscriptionExtractor(); | ||||
| 
 | ||||
|     /** | ||||
|      * Must create a new instance of a KioskList implementation. | ||||
|      * @return a new KioskList instance | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     public abstract KioskList getKioskList() throws ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * Must create a new instance of a ChannelExtractor implementation. | ||||
|      * @param linkHandler is pointing to the channel which should be handled by this new instance. | ||||
|      * @param localization specifies the language used for the request. | ||||
|      * @return a new ChannelExtractor | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     public abstract ChannelExtractor getChannelExtractor(ListLinkHandler linkHandler, | ||||
|                                                          Localization localization) throws ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * Must crete a new instance of a PlaylistExtractor implementation. | ||||
|      * @param linkHandler is pointing to the playlist which should be handled by this new instance. | ||||
|      * @param localization specifies the language used for the request. | ||||
|      * @return a new PlaylistExtractor | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     public abstract PlaylistExtractor getPlaylistExtractor(ListLinkHandler linkHandler, | ||||
|                                                            Localization localization) throws ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * Must create a new instance of a StreamExtractor implementation. | ||||
|      * @param linkHandler is pointing to the stream which should be handled by this new instance. | ||||
|      * @param localization specifies the language used for the request. | ||||
|      * @return a new StreamExtractor | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     public abstract StreamExtractor getStreamExtractor(LinkHandler linkHandler, | ||||
|                                                        Localization localization) throws ExtractionException; | ||||
|     public abstract CommentsExtractor getCommentsExtractor(ListLinkHandler linkHandler, | ||||
|  | @ -107,31 +217,27 @@ public abstract class StreamingService { | |||
|     //////////////////////////////////////////// | ||||
| 
 | ||||
|     public SearchExtractor getSearchExtractor(SearchQueryHandler queryHandler) { | ||||
|         return getSearchExtractor(queryHandler, NewPipe.getLocalization()); | ||||
|         return getSearchExtractor(queryHandler, NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public SuggestionExtractor getSuggestionExtractor() { | ||||
|         return getSuggestionExtractor(NewPipe.getLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public KioskList getKioskList() throws ExtractionException { | ||||
|         return getKioskList(NewPipe.getLocalization()); | ||||
|         return getSuggestionExtractor(NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public ChannelExtractor getChannelExtractor(ListLinkHandler linkHandler) throws ExtractionException { | ||||
|         return getChannelExtractor(linkHandler, NewPipe.getLocalization()); | ||||
|         return getChannelExtractor(linkHandler, NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
|   | ||||
|     public PlaylistExtractor getPlaylistExtractor(ListLinkHandler linkHandler) throws ExtractionException { | ||||
|         return getPlaylistExtractor(linkHandler, NewPipe.getLocalization()); | ||||
|         return getPlaylistExtractor(linkHandler, NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public StreamExtractor getStreamExtractor(LinkHandler linkHandler) throws ExtractionException { | ||||
|         return getStreamExtractor(linkHandler, NewPipe.getLocalization()); | ||||
|         return getStreamExtractor(linkHandler, NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
|      | ||||
|     public CommentsExtractor getCommentsExtractor(ListLinkHandler urlIdHandler) throws ExtractionException { | ||||
|         return getCommentsExtractor(urlIdHandler, NewPipe.getLocalization()); | ||||
|         return getCommentsExtractor(urlIdHandler, NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     //////////////////////////////////////////// | ||||
|  | @ -172,19 +278,19 @@ public abstract class StreamingService { | |||
|     //////////////////////////////////////////// | ||||
| 
 | ||||
|     public SearchExtractor getSearchExtractor(String query) throws ExtractionException { | ||||
|         return getSearchExtractor(getSearchQHFactory().fromQuery(query), NewPipe.getLocalization()); | ||||
|         return getSearchExtractor(getSearchQHFactory().fromQuery(query), NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public ChannelExtractor getChannelExtractor(String url) throws ExtractionException { | ||||
|         return getChannelExtractor(getChannelLHFactory().fromUrl(url), NewPipe.getLocalization()); | ||||
|         return getChannelExtractor(getChannelLHFactory().fromUrl(url), NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public PlaylistExtractor getPlaylistExtractor(String url) throws ExtractionException { | ||||
|         return getPlaylistExtractor(getPlaylistLHFactory().fromUrl(url), NewPipe.getLocalization()); | ||||
|         return getPlaylistExtractor(getPlaylistLHFactory().fromUrl(url), NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public StreamExtractor getStreamExtractor(String url) throws ExtractionException { | ||||
|         return getStreamExtractor(getStreamLHFactory().fromUrl(url), NewPipe.getLocalization()); | ||||
|         return getStreamExtractor(getStreamLHFactory().fromUrl(url), NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
|      | ||||
|     public CommentsExtractor getCommentsExtractor(String url) throws ExtractionException { | ||||
|  | @ -192,7 +298,7 @@ public abstract class StreamingService { | |||
|         if(null == llhf) { | ||||
|             return null; | ||||
|         } | ||||
|         return getCommentsExtractor(llhf.fromUrl(url), NewPipe.getLocalization()); | ||||
|         return getCommentsExtractor(llhf.fromUrl(url), NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public abstract boolean isCommentsSupported(); | ||||
|  | @ -203,6 +309,11 @@ public abstract class StreamingService { | |||
| 
 | ||||
| 	/** | ||||
|      * figure out where the link is pointing to (a channel, video, playlist, etc.) | ||||
|     /** | ||||
|      * Figures out where the link is pointing to (a channel, a video, a playlist, etc.) | ||||
|      * @param url the url on which it should be decided of which link type it is | ||||
|      * @return the link type of url | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     public final LinkType getLinkTypeByUrl(String url) throws ParsingException { | ||||
|         LinkHandlerFactory sH = getStreamLHFactory(); | ||||
|  |  | |||
|  | @ -1,34 +0,0 @@ | |||
| package org.schabi.newpipe.extractor; | ||||
| 
 | ||||
| import org.schabi.newpipe.extractor.stream.SubtitlesFormat; | ||||
| 
 | ||||
| import java.io.Serializable; | ||||
| import java.util.Locale; | ||||
| 
 | ||||
| public class Subtitles implements Serializable { | ||||
|     private final SubtitlesFormat format; | ||||
|     private final Locale locale; | ||||
|     private final String URL; | ||||
|     private final boolean autoGenerated; | ||||
| 
 | ||||
|     public Subtitles(SubtitlesFormat format, Locale locale, String URL, boolean autoGenerated) { | ||||
|         this.format = format; | ||||
|         this.locale = locale; | ||||
|         this.URL = URL; | ||||
|         this.autoGenerated = autoGenerated; | ||||
|     } | ||||
| 
 | ||||
|     public SubtitlesFormat getFileType() { return format; } | ||||
| 
 | ||||
|     public Locale getLocale() { | ||||
|         return locale; | ||||
|     } | ||||
| 
 | ||||
|     public String getURL() { | ||||
|         return URL; | ||||
|     } | ||||
| 
 | ||||
|     public boolean isAutoGenerated() { | ||||
|         return autoGenerated; | ||||
|     } | ||||
| } | ||||
|  | @ -51,8 +51,7 @@ public class ChannelInfo extends ListInfo<StreamInfoItem> { | |||
| 
 | ||||
|     public static InfoItemsPage<StreamInfoItem> getMoreItems(StreamingService service, | ||||
|                                                              String url, | ||||
|                                                              String pageUrl, | ||||
|                                                              Localization localization) throws IOException, ExtractionException { | ||||
|                                                              String pageUrl) throws IOException, ExtractionException { | ||||
|         return service.getChannelExtractor(url).getPage(pageUrl); | ||||
|     } | ||||
| 
 | ||||
|  |  | |||
|  | @ -30,7 +30,6 @@ import org.schabi.newpipe.extractor.utils.Localization; | |||
| import javax.annotation.Nonnull; | ||||
| 
 | ||||
| public abstract class KioskExtractor extends ListExtractor<StreamInfoItem> { | ||||
|     private String contentCountry = null; | ||||
|     private final String id; | ||||
| 
 | ||||
|     public KioskExtractor(StreamingService streamingService, | ||||
|  | @ -41,17 +40,6 @@ public abstract class KioskExtractor extends ListExtractor<StreamInfoItem> { | |||
|         this.id = kioskId; | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * For certain Websites the content of a kiosk will be different depending | ||||
|      * on the country you want to poen the website in. Therefore you should | ||||
|      * set the contentCountry. | ||||
|      * @param contentCountry Set the country decoded as Country Code: http://www.1728.org/countries.htm | ||||
|      */ | ||||
|     public void setContentCountry(String contentCountry) { | ||||
|         this.contentCountry = contentCountry; | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     @Nonnull | ||||
|     @Override | ||||
|     public String getId() { | ||||
|  | @ -69,9 +57,4 @@ public abstract class KioskExtractor extends ListExtractor<StreamInfoItem> { | |||
|     @Nonnull | ||||
|     @Override | ||||
|     public abstract String getName() throws ParsingException; | ||||
| 
 | ||||
| 
 | ||||
|     public String getContentCountry() { | ||||
|         return contentCountry; | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -37,25 +37,20 @@ public class KioskInfo extends ListInfo<StreamInfoItem> { | |||
| 
 | ||||
|     public static ListExtractor.InfoItemsPage<StreamInfoItem> getMoreItems(StreamingService service, | ||||
|                                                                            String url, | ||||
|                                                                            String pageUrl, | ||||
|                                                                            String contentCountry) throws IOException, ExtractionException { | ||||
|                                                                            String pageUrl) throws IOException, ExtractionException { | ||||
|         KioskList kl = service.getKioskList(); | ||||
|         KioskExtractor extractor = kl.getExtractorByUrl(url, pageUrl); | ||||
|         extractor.setContentCountry(contentCountry); | ||||
|         return extractor.getPage(pageUrl); | ||||
|     } | ||||
| 
 | ||||
|     public static KioskInfo getInfo(String url, | ||||
|                                     String contentCountry) throws IOException, ExtractionException { | ||||
|         return getInfo(NewPipe.getServiceByUrl(url), url, contentCountry); | ||||
|     public static KioskInfo getInfo(String url) throws IOException, ExtractionException { | ||||
|         return getInfo(NewPipe.getServiceByUrl(url), url); | ||||
|     } | ||||
| 
 | ||||
|     public static KioskInfo getInfo(StreamingService service, | ||||
|                                     String url, | ||||
|                                     String contentCountry) throws IOException, ExtractionException { | ||||
|                                     String url) throws IOException, ExtractionException { | ||||
|         KioskList kl = service.getKioskList(); | ||||
|         KioskExtractor extractor = kl.getExtractorByUrl(url, null); | ||||
|         extractor.setContentCountry(contentCountry); | ||||
|         extractor.fetchPage(); | ||||
|         return getInfo(extractor); | ||||
|     } | ||||
|  |  | |||
|  | @ -23,7 +23,6 @@ public  class KioskList { | |||
|     private final int service_id; | ||||
|     private final HashMap<String, KioskEntry> kioskList = new HashMap<>(); | ||||
|     private String defaultKiosk = null; | ||||
|     private final Localization localization; | ||||
| 
 | ||||
|     private class KioskEntry { | ||||
|         public KioskEntry(KioskExtractorFactory ef, ListLinkHandlerFactory h) { | ||||
|  | @ -34,9 +33,8 @@ public  class KioskList { | |||
|         final ListLinkHandlerFactory handlerFactory; | ||||
|     } | ||||
| 
 | ||||
|     public KioskList(int service_id, Localization localization) { | ||||
|     public KioskList(int service_id) { | ||||
|         this.service_id = service_id; | ||||
|         this.localization = localization; | ||||
|     } | ||||
| 
 | ||||
|     public void addKioskEntry(KioskExtractorFactory extractorFactory, ListLinkHandlerFactory handlerFactory, String id) | ||||
|  | @ -53,13 +51,18 @@ public  class KioskList { | |||
| 
 | ||||
|     public KioskExtractor getDefaultKioskExtractor(String nextPageUrl) | ||||
|             throws ExtractionException, IOException { | ||||
|         return getDefaultKioskExtractor(nextPageUrl, NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public KioskExtractor getDefaultKioskExtractor(String nextPageUrl, Localization localization) | ||||
|             throws ExtractionException, IOException { | ||||
|         if(defaultKiosk != null && !defaultKiosk.equals("")) { | ||||
|             return getExtractorById(defaultKiosk, nextPageUrl); | ||||
|             return getExtractorById(defaultKiosk, nextPageUrl, localization); | ||||
|         } else { | ||||
|             if(!kioskList.isEmpty()) { | ||||
|                 // if not set get any entry | ||||
|                 Object[] keySet = kioskList.keySet().toArray(); | ||||
|                 return getExtractorById(keySet[0].toString(), nextPageUrl); | ||||
|                 return getExtractorById(keySet[0].toString(), nextPageUrl, localization); | ||||
|             } else { | ||||
|                 return null; | ||||
|             } | ||||
|  | @ -72,6 +75,11 @@ public  class KioskList { | |||
| 
 | ||||
|     public KioskExtractor getExtractorById(String kioskId, String nextPageUrl) | ||||
|             throws ExtractionException, IOException { | ||||
|         return getExtractorById(kioskId, nextPageUrl, NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public KioskExtractor getExtractorById(String kioskId, String nextPageUrl, Localization localization) | ||||
|             throws ExtractionException, IOException { | ||||
|         KioskEntry ke = kioskList.get(kioskId); | ||||
|         if(ke == null) { | ||||
|             throw new ExtractionException("No kiosk found with the type: " + kioskId); | ||||
|  | @ -86,11 +94,16 @@ public  class KioskList { | |||
|     } | ||||
| 
 | ||||
|     public KioskExtractor getExtractorByUrl(String url, String nextPageUrl) | ||||
|             throws ExtractionException, IOException{ | ||||
|         return getExtractorByUrl(url, nextPageUrl, NewPipe.getPreferredLocalization()); | ||||
|     } | ||||
| 
 | ||||
|     public KioskExtractor getExtractorByUrl(String url, String nextPageUrl, Localization localization) | ||||
|             throws ExtractionException, IOException { | ||||
|         for(Map.Entry<String, KioskEntry> e : kioskList.entrySet()) { | ||||
|             KioskEntry ke = e.getValue(); | ||||
|             if(ke.handlerFactory.acceptUrl(url)) { | ||||
|                 return getExtractorById(e.getKey(), nextPageUrl); | ||||
|                 return getExtractorById(e.getKey(), nextPageUrl, localization); | ||||
|             } | ||||
|         } | ||||
|         throw new ExtractionException("Could not find a kiosk that fits to the url: " + url); | ||||
|  |  | |||
|  | @ -97,36 +97,6 @@ public class PeertubeService extends StreamingService { | |||
|         return null; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public KioskList getKioskList(Localization localization) throws ExtractionException { | ||||
|         KioskList.KioskExtractorFactory kioskFactory = new KioskList.KioskExtractorFactory() { | ||||
|             @Override | ||||
|             public KioskExtractor createNewKiosk(StreamingService streamingService, | ||||
|                                                  String url, | ||||
|                                                  String id, | ||||
|                                                  Localization local) | ||||
|                     throws ExtractionException { | ||||
|                 return new PeertubeTrendingExtractor(PeertubeService.this, | ||||
|                         new PeertubeTrendingLinkHandlerFactory().fromId(id), id, local); | ||||
|             } | ||||
|         }; | ||||
| 
 | ||||
|         KioskList list = new KioskList(getServiceId(), localization); | ||||
| 
 | ||||
|         // add kiosks here e.g.: | ||||
|         final PeertubeTrendingLinkHandlerFactory h = new PeertubeTrendingLinkHandlerFactory(); | ||||
|         try { | ||||
|             list.addKioskEntry(kioskFactory, h, PeertubeTrendingLinkHandlerFactory.KIOSK_TRENDING); | ||||
|             list.addKioskEntry(kioskFactory, h, PeertubeTrendingLinkHandlerFactory.KIOSK_RECENT); | ||||
|             list.addKioskEntry(kioskFactory, h, PeertubeTrendingLinkHandlerFactory.KIOSK_LOCAL); | ||||
|             list.setDefaultKiosk(PeertubeTrendingLinkHandlerFactory.KIOSK_TRENDING); | ||||
|         } catch (Exception e) { | ||||
|             throw new ExtractionException(e); | ||||
|         } | ||||
| 
 | ||||
|         return list; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public ChannelExtractor getChannelExtractor(ListLinkHandler linkHandler, Localization localization) | ||||
|             throws ExtractionException { | ||||
|  | @ -170,6 +140,36 @@ public class PeertubeService extends StreamingService { | |||
|             this.getServiceInfo().setName("PeerTube"); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public KioskList getKioskList() throws ExtractionException { | ||||
|         KioskList.KioskExtractorFactory kioskFactory = new KioskList.KioskExtractorFactory() { | ||||
|             @Override | ||||
|             public KioskExtractor createNewKiosk(StreamingService streamingService, | ||||
|                                                  String url, | ||||
|                                                  String id, | ||||
|                                                  Localization local) | ||||
|                     throws ExtractionException { | ||||
|                 return new PeertubeTrendingExtractor(PeertubeService.this, | ||||
|                         new PeertubeTrendingLinkHandlerFactory().fromId(id), id, local); | ||||
|             } | ||||
|         }; | ||||
| 
 | ||||
|         KioskList list = new KioskList(getServiceId()); | ||||
| 
 | ||||
|         // add kiosks here e.g.: | ||||
|         final PeertubeTrendingLinkHandlerFactory h = new PeertubeTrendingLinkHandlerFactory(); | ||||
|         try { | ||||
|             list.addKioskEntry(kioskFactory, h, PeertubeTrendingLinkHandlerFactory.KIOSK_TRENDING); | ||||
|             list.addKioskEntry(kioskFactory, h, PeertubeTrendingLinkHandlerFactory.KIOSK_RECENT); | ||||
|             list.addKioskEntry(kioskFactory, h, PeertubeTrendingLinkHandlerFactory.KIOSK_LOCAL); | ||||
|             list.setDefaultKiosk(PeertubeTrendingLinkHandlerFactory.KIOSK_TRENDING); | ||||
|         } catch (Exception e) { | ||||
|             throw new ExtractionException(e); | ||||
|         } | ||||
| 
 | ||||
|         return list; | ||||
|     } | ||||
|      | ||||
| 
 | ||||
| } | ||||
|  |  | |||
|  | @ -11,7 +11,6 @@ import org.schabi.newpipe.extractor.Downloader; | |||
| import org.schabi.newpipe.extractor.MediaFormat; | ||||
| import org.schabi.newpipe.extractor.ServiceList; | ||||
| import org.schabi.newpipe.extractor.StreamingService; | ||||
| import org.schabi.newpipe.extractor.Subtitles; | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||
| import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||
| import org.schabi.newpipe.extractor.exceptions.ReCaptchaException; | ||||
|  | @ -25,7 +24,7 @@ import org.schabi.newpipe.extractor.stream.StreamExtractor; | |||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItemsCollector; | ||||
| import org.schabi.newpipe.extractor.stream.StreamType; | ||||
| import org.schabi.newpipe.extractor.stream.SubtitlesFormat; | ||||
| import org.schabi.newpipe.extractor.stream.SubtitlesStream; | ||||
| import org.schabi.newpipe.extractor.stream.VideoStream; | ||||
| import org.schabi.newpipe.extractor.utils.JsonUtils; | ||||
| import org.schabi.newpipe.extractor.utils.Localization; | ||||
|  | @ -170,12 +169,12 @@ public class PeertubeStreamExtractor extends StreamExtractor { | |||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public List<Subtitles> getSubtitlesDefault() throws IOException, ExtractionException { | ||||
|     public List<SubtitlesStream> getSubtitlesDefault() throws IOException, ExtractionException { | ||||
|         return Collections.emptyList(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public List<Subtitles> getSubtitles(SubtitlesFormat format) throws IOException, ExtractionException { | ||||
|     public List<SubtitlesStream> getSubtitles(final MediaFormat format) throws IOException, ExtractionException { | ||||
|         return Collections.emptyList(); | ||||
|     } | ||||
| 
 | ||||
|  | @ -185,12 +184,12 @@ public class PeertubeStreamExtractor extends StreamExtractor { | |||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public StreamInfoItem getNextVideo() throws IOException, ExtractionException { | ||||
|     public StreamInfoItem getNextStream() throws IOException, ExtractionException { | ||||
|         return null; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public StreamInfoItemsCollector getRelatedVideos() throws IOException, ExtractionException { | ||||
|     public StreamInfoItemsCollector getRelatedStreams() throws IOException, ExtractionException { | ||||
|         StreamInfoItemsCollector collector = new StreamInfoItemsCollector(getServiceId()); | ||||
| 
 | ||||
|         //TODO fetch related videos not trending | ||||
|  |  | |||
|  | @ -69,7 +69,7 @@ public class SoundcloudService extends StreamingService { | |||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public KioskList getKioskList(Localization localization) throws ExtractionException { | ||||
|     public KioskList getKioskList() throws ExtractionException { | ||||
|         KioskList.KioskExtractorFactory chartsFactory = new KioskList.KioskExtractorFactory() { | ||||
|             @Override | ||||
|             public KioskExtractor createNewKiosk(StreamingService streamingService, | ||||
|  | @ -82,7 +82,7 @@ public class SoundcloudService extends StreamingService { | |||
|             } | ||||
|         }; | ||||
| 
 | ||||
|         KioskList list = new KioskList(getServiceId(), localization); | ||||
|         KioskList list = new KioskList(getServiceId()); | ||||
| 
 | ||||
|         // add kiosks here e.g.: | ||||
|         final SoundcloudChartsLinkHandlerFactory h = new SoundcloudChartsLinkHandlerFactory(); | ||||
|  |  | |||
|  | @ -172,13 +172,13 @@ public class SoundcloudStreamExtractor extends StreamExtractor { | |||
| 
 | ||||
|     @Override | ||||
|     @Nonnull | ||||
|     public List<Subtitles> getSubtitlesDefault() throws IOException, ExtractionException { | ||||
|     public List<SubtitlesStream> getSubtitlesDefault() throws IOException, ExtractionException { | ||||
|         return Collections.emptyList(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     @Nonnull | ||||
|     public List<Subtitles> getSubtitles(SubtitlesFormat format) throws IOException, ExtractionException { | ||||
|     public List<SubtitlesStream> getSubtitles(MediaFormat format) throws IOException, ExtractionException { | ||||
|         return Collections.emptyList(); | ||||
|     } | ||||
| 
 | ||||
|  | @ -188,12 +188,12 @@ public class SoundcloudStreamExtractor extends StreamExtractor { | |||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public StreamInfoItem getNextVideo() throws IOException, ExtractionException { | ||||
|     public StreamInfoItem getNextStream() throws IOException, ExtractionException { | ||||
|         return null; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public StreamInfoItemsCollector getRelatedVideos() throws IOException, ExtractionException { | ||||
|     public StreamInfoItemsCollector getRelatedStreams() throws IOException, ExtractionException { | ||||
|         StreamInfoItemsCollector collector = new StreamInfoItemsCollector(getServiceId()); | ||||
| 
 | ||||
|         String apiUrl = "https://api-v2.soundcloud.com/tracks/" + urlEncode(getId()) + "/related" | ||||
|  |  | |||
|  | @ -111,8 +111,8 @@ public class YoutubeService extends StreamingService { | |||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public KioskList getKioskList(final Localization localization) throws ExtractionException { | ||||
|         KioskList list = new KioskList(getServiceId(), localization); | ||||
|     public KioskList getKioskList() throws ExtractionException { | ||||
|         KioskList list = new KioskList(getServiceId()); | ||||
| 
 | ||||
|         // add kiosks here e.g.: | ||||
|         try { | ||||
|  |  | |||
|  | @ -7,6 +7,7 @@ import java.util.Arrays; | |||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.regex.Pattern; | ||||
| 
 | ||||
| import javax.annotation.Nonnull; | ||||
| 
 | ||||
|  | @ -25,6 +26,7 @@ import org.schabi.newpipe.extractor.exceptions.ReCaptchaException; | |||
| import org.schabi.newpipe.extractor.linkhandler.ListLinkHandler; | ||||
| import org.schabi.newpipe.extractor.utils.JsonUtils; | ||||
| import org.schabi.newpipe.extractor.utils.Localization; | ||||
| import org.schabi.newpipe.extractor.utils.Parser; | ||||
| 
 | ||||
| import com.grack.nanojson.JsonArray; | ||||
| import com.grack.nanojson.JsonObject; | ||||
|  | @ -34,6 +36,7 @@ import com.grack.nanojson.JsonParser; | |||
| public class YoutubeCommentsExtractor extends CommentsExtractor { | ||||
| 
 | ||||
|     private static final String USER_AGENT = "Mozilla/5.0 (Android 8.1.0; Mobile; rv:62.0) Gecko/62.0 Firefox/62.0"; | ||||
|     private static final Pattern YT_CLIENT_NAME_PATTERN = Pattern.compile("INNERTUBE_CONTEXT_CLIENT_NAME\\\":(.*?)[,}]"); | ||||
| 
 | ||||
|     private String ytClientVersion; | ||||
|     private String ytClientName; | ||||
|  | @ -62,7 +65,7 @@ public class YoutubeCommentsExtractor extends CommentsExtractor { | |||
|          | ||||
|         JsonArray arr; | ||||
|         try { | ||||
|             arr = (JsonArray) JsonUtils.getValue(ajaxJson, "response.continuationContents.commentSectionContinuation.continuations"); | ||||
|             arr = JsonUtils.getArray(ajaxJson, "response.continuationContents.commentSectionContinuation.continuations"); | ||||
|         } catch (Exception e) { | ||||
|             return ""; | ||||
|         } | ||||
|  | @ -71,7 +74,7 @@ public class YoutubeCommentsExtractor extends CommentsExtractor { | |||
|         } | ||||
|         String continuation; | ||||
|         try { | ||||
|             continuation = (String) JsonUtils.getValue(arr.getObject(0), "nextContinuationData.continuation"); | ||||
|             continuation = JsonUtils.getString(arr.getObject(0), "nextContinuationData.continuation"); | ||||
|         } catch (Exception e) { | ||||
|             return ""; | ||||
|         } | ||||
|  | @ -111,7 +114,7 @@ public class YoutubeCommentsExtractor extends CommentsExtractor { | |||
|          | ||||
|         JsonArray contents; | ||||
|         try { | ||||
|             contents = (JsonArray) JsonUtils.getValue(ajaxJson, "response.continuationContents.commentSectionContinuation.items"); | ||||
|             contents = JsonUtils.getArray(ajaxJson, "response.continuationContents.commentSectionContinuation.items"); | ||||
|         }catch(Exception e) { | ||||
|             //no comments | ||||
|             return; | ||||
|  | @ -135,7 +138,7 @@ public class YoutubeCommentsExtractor extends CommentsExtractor { | |||
|     private void fetchTitle(JsonArray contents) { | ||||
|         if(null == title) { | ||||
|             try { | ||||
|                 title = getYoutubeText((JsonObject) JsonUtils.getValue(contents.getObject(0), "commentThreadRenderer.commentTargetTitle")); | ||||
|                 title = getYoutubeText(JsonUtils.getObject(contents.getObject(0), "commentThreadRenderer.commentTargetTitle")); | ||||
|             } catch (Exception e) { | ||||
|                 title = "Youtube Comments"; | ||||
|             } | ||||
|  | @ -150,7 +153,7 @@ public class YoutubeCommentsExtractor extends CommentsExtractor { | |||
|         DownloadResponse response = downloader.get(getUrl(), request); | ||||
|         String responseBody = response.getResponseBody(); | ||||
|         ytClientVersion = findValue(responseBody, "INNERTUBE_CONTEXT_CLIENT_VERSION\":\"", "\""); | ||||
|         ytClientName = findValue(responseBody, "INNERTUBE_CONTEXT_CLIENT_NAME\":", ","); | ||||
|         ytClientName = Parser.matchGroup1(YT_CLIENT_NAME_PATTERN, responseBody); | ||||
|         String commentsTokenInside = findValue(responseBody, "commentSectionRenderer", "}"); | ||||
|         String commentsToken = findValue(commentsTokenInside, "continuation\":\"", "\""); | ||||
|         initPage = getPage(getNextPageUrl(commentsToken)); | ||||
|  | @ -196,17 +199,17 @@ public class YoutubeCommentsExtractor extends CommentsExtractor { | |||
|      | ||||
|     public static String getYoutubeText(@Nonnull JsonObject object) throws ParsingException { | ||||
|         try { | ||||
|             return (String) JsonUtils.getValue(object, "simpleText"); | ||||
|             return JsonUtils.getString(object, "simpleText"); | ||||
|         } catch (Exception e1) { | ||||
|             try { | ||||
|                 JsonArray arr = (JsonArray) JsonUtils.getValue(object, "runs"); | ||||
|                 JsonArray arr = JsonUtils.getArray(object, "runs"); | ||||
|                 String result = ""; | ||||
|                 for(int i=0; i<arr.size();i++) { | ||||
|                     result = result + (String) JsonUtils.getValue(arr.getObject(i), "text"); | ||||
|                     result = result + JsonUtils.getString(arr.getObject(i), "text"); | ||||
|                 } | ||||
|                 return result; | ||||
|             } catch (Exception e2) { | ||||
|                 throw new ParsingException("Could not get text", e2); | ||||
|                 return ""; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  |  | |||
|  | @ -7,7 +7,6 @@ import org.schabi.newpipe.extractor.utils.JsonUtils; | |||
| import com.grack.nanojson.JsonArray; | ||||
| import com.grack.nanojson.JsonObject; | ||||
| 
 | ||||
| 
 | ||||
| public class YoutubeCommentsInfoItemExtractor implements CommentsInfoItemExtractor { | ||||
| 
 | ||||
|     private final JsonObject json; | ||||
|  | @ -26,8 +25,8 @@ public class YoutubeCommentsInfoItemExtractor implements CommentsInfoItemExtract | |||
|     @Override | ||||
|     public String getThumbnailUrl() throws ParsingException { | ||||
|         try { | ||||
|             JsonArray arr = (JsonArray) JsonUtils.getValue(json, "authorThumbnail.thumbnails"); | ||||
|             return (String) JsonUtils.getValue(arr.getObject(2), "url"); | ||||
|             JsonArray arr = JsonUtils.getArray(json, "authorThumbnail.thumbnails"); | ||||
|             return JsonUtils.getString(arr.getObject(2), "url"); | ||||
|         } catch (Exception e) { | ||||
|             throw new ParsingException("Could not get thumbnail url", e); | ||||
|         } | ||||
|  | @ -36,7 +35,7 @@ public class YoutubeCommentsInfoItemExtractor implements CommentsInfoItemExtract | |||
|     @Override | ||||
|     public String getName() throws ParsingException { | ||||
|         try { | ||||
|             return YoutubeCommentsExtractor.getYoutubeText((JsonObject) JsonUtils.getValue(json, "authorText")); | ||||
|             return YoutubeCommentsExtractor.getYoutubeText(JsonUtils.getObject(json, "authorText")); | ||||
|         } catch (Exception e) { | ||||
|             throw new ParsingException("Could not get author name", e); | ||||
|         } | ||||
|  | @ -45,7 +44,7 @@ public class YoutubeCommentsInfoItemExtractor implements CommentsInfoItemExtract | |||
|     @Override | ||||
|     public String getPublishedTime() throws ParsingException { | ||||
|         try { | ||||
|             return YoutubeCommentsExtractor.getYoutubeText((JsonObject) JsonUtils.getValue(json, "publishedTimeText")); | ||||
|             return YoutubeCommentsExtractor.getYoutubeText(JsonUtils.getObject(json, "publishedTimeText")); | ||||
|         } catch (Exception e) { | ||||
|             throw new ParsingException("Could not get publishedTimeText", e); | ||||
|         } | ||||
|  | @ -54,7 +53,7 @@ public class YoutubeCommentsInfoItemExtractor implements CommentsInfoItemExtract | |||
|     @Override | ||||
|     public Integer getLikeCount() throws ParsingException { | ||||
|         try { | ||||
|             return (Integer) JsonUtils.getValue(json, "likeCount"); | ||||
|             return JsonUtils.getNumber(json, "likeCount").intValue(); | ||||
|         } catch (Exception e) { | ||||
|             throw new ParsingException("Could not get like count", e); | ||||
|         } | ||||
|  | @ -63,7 +62,7 @@ public class YoutubeCommentsInfoItemExtractor implements CommentsInfoItemExtract | |||
|     @Override | ||||
|     public String getCommentText() throws ParsingException { | ||||
|         try { | ||||
|             return YoutubeCommentsExtractor.getYoutubeText((JsonObject) JsonUtils.getValue(json, "contentText")); | ||||
|             return YoutubeCommentsExtractor.getYoutubeText(JsonUtils.getObject(json, "contentText")); | ||||
|         } catch (Exception e) { | ||||
|             throw new ParsingException("Could not get comment text", e); | ||||
|         } | ||||
|  | @ -72,7 +71,7 @@ public class YoutubeCommentsInfoItemExtractor implements CommentsInfoItemExtract | |||
|     @Override | ||||
|     public String getCommentId() throws ParsingException { | ||||
|         try { | ||||
|             return (String) JsonUtils.getValue(json, "commentId"); | ||||
|             return JsonUtils.getString(json, "commentId"); | ||||
|         } catch (Exception e) { | ||||
|             throw new ParsingException("Could not get comment id", e); | ||||
|         } | ||||
|  | @ -81,8 +80,8 @@ public class YoutubeCommentsInfoItemExtractor implements CommentsInfoItemExtract | |||
|     @Override | ||||
|     public String getAuthorThumbnail() throws ParsingException { | ||||
|         try { | ||||
|             JsonArray arr = (JsonArray) JsonUtils.getValue(json, "authorThumbnail.thumbnails"); | ||||
|             return (String) JsonUtils.getValue(arr.getObject(2), "url"); | ||||
|             JsonArray arr = JsonUtils.getArray(json, "authorThumbnail.thumbnails"); | ||||
|             return JsonUtils.getString(arr.getObject(2), "url"); | ||||
|         } catch (Exception e) { | ||||
|             throw new ParsingException("Could not get author thumbnail", e); | ||||
|         } | ||||
|  | @ -91,7 +90,7 @@ public class YoutubeCommentsInfoItemExtractor implements CommentsInfoItemExtract | |||
|     @Override | ||||
|     public String getAuthorName() throws ParsingException { | ||||
|         try { | ||||
|             return YoutubeCommentsExtractor.getYoutubeText((JsonObject) JsonUtils.getValue(json, "authorText")); | ||||
|             return YoutubeCommentsExtractor.getYoutubeText(JsonUtils.getObject(json, "authorText")); | ||||
|         } catch (Exception e) { | ||||
|             throw new ParsingException("Could not get author name", e); | ||||
|         } | ||||
|  | @ -100,8 +99,7 @@ public class YoutubeCommentsInfoItemExtractor implements CommentsInfoItemExtract | |||
|     @Override | ||||
|     public String getAuthorEndpoint() throws ParsingException { | ||||
|         try { | ||||
|             return "https://youtube.com" | ||||
|                     + (String) JsonUtils.getValue(json, "authorEndpoint.browseEndpoint.canonicalBaseUrl"); | ||||
|             return "https://youtube.com" + JsonUtils.getString(json, "authorEndpoint.browseEndpoint.canonicalBaseUrl"); | ||||
|         } catch (Exception e) { | ||||
|             throw new ParsingException("Could not get author endpoint", e); | ||||
|         } | ||||
|  |  | |||
|  | @ -54,21 +54,20 @@ public class YoutubeSearchExtractor extends SearchExtractor { | |||
|     public void onFetchPage(@Nonnull Downloader downloader) throws IOException, ExtractionException { | ||||
|         final String site; | ||||
|         final String url = getUrl(); | ||||
|         final String contentCountry = getLocalization().getCountry(); | ||||
|         //String url = builder.build().toString(); | ||||
|         //if we've been passed a valid language code, append it to the URL | ||||
|         if (!contentCountry.isEmpty()) { | ||||
|             //assert Pattern.matches("[a-z]{2}(-([A-Z]{2}|[0-9]{1,3}))?", languageCode); | ||||
|             site = downloader.download(url, getLocalization()); | ||||
|         } else { | ||||
|             site = downloader.download(url); | ||||
|         } | ||||
|         site = downloader.download(url, getLocalization()); | ||||
| 
 | ||||
|         doc = Jsoup.parse(site, url); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String getSearchSuggestion() throws ParsingException { | ||||
|     public String getUrl() throws ParsingException { | ||||
|         return super.getUrl() + "&gl="+ getLocalization().getCountry(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String getSearchSuggestion() { | ||||
|         final Element el = doc.select("div[class*=\"spell-correction\"]").first(); | ||||
|         if (el != null) { | ||||
|             return el.select("a").first().text(); | ||||
|  | @ -79,13 +78,13 @@ public class YoutubeSearchExtractor extends SearchExtractor { | |||
| 
 | ||||
|     @Nonnull | ||||
|     @Override | ||||
|     public InfoItemsPage<InfoItem> getInitialPage() throws IOException, ExtractionException { | ||||
|     public InfoItemsPage<InfoItem> getInitialPage() throws ExtractionException { | ||||
|         return new InfoItemsPage<>(collectItems(doc), getNextPageUrl()); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String getNextPageUrl() throws ExtractionException { | ||||
|         return getUrl() + "&page=" + Integer.toString( 2); | ||||
|         return getUrl() + "&page=" + 2; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|  | @ -104,7 +103,7 @@ public class YoutubeSearchExtractor extends SearchExtractor { | |||
|                                 .getQuery()) | ||||
|                         .get("page")); | ||||
| 
 | ||||
|         return currentUrl.replace("&page=" + Integer.toString( pageNr), | ||||
|         return currentUrl.replace("&page=" + pageNr, | ||||
|                 "&page=" + Integer.toString(pageNr + 1)); | ||||
|     } | ||||
| 
 | ||||
|  |  | |||
|  | @ -7,6 +7,7 @@ import com.grack.nanojson.JsonParserException; | |||
| import org.jsoup.Jsoup; | ||||
| import org.jsoup.nodes.Document; | ||||
| import org.jsoup.nodes.Element; | ||||
| import org.jsoup.select.Elements; | ||||
| import org.mozilla.javascript.Context; | ||||
| import org.mozilla.javascript.Function; | ||||
| import org.mozilla.javascript.ScriptableObject; | ||||
|  | @ -460,15 +461,15 @@ public class YoutubeStreamExtractor extends StreamExtractor { | |||
| 
 | ||||
|     @Override | ||||
|     @Nonnull | ||||
|     public List<Subtitles> getSubtitlesDefault() throws IOException, ExtractionException { | ||||
|         return getSubtitles(SubtitlesFormat.TTML); | ||||
|     public List<SubtitlesStream> getSubtitlesDefault() throws IOException, ExtractionException { | ||||
|         return getSubtitles(MediaFormat.TTML); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     @Nonnull | ||||
|     public List<Subtitles> getSubtitles(final SubtitlesFormat format) throws IOException, ExtractionException { | ||||
|     public List<SubtitlesStream> getSubtitles(final MediaFormat format) throws IOException, ExtractionException { | ||||
|         assertPageFetched(); | ||||
|         List<Subtitles> subtitles = new ArrayList<>(); | ||||
|         List<SubtitlesStream> subtitles = new ArrayList<>(); | ||||
|         for (final SubtitlesInfo subtitlesInfo : subtitlesInfos) { | ||||
|             subtitles.add(subtitlesInfo.getSubtitle(format)); | ||||
|         } | ||||
|  | @ -490,13 +491,17 @@ public class YoutubeStreamExtractor extends StreamExtractor { | |||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public StreamInfoItem getNextVideo() throws IOException, ExtractionException { | ||||
|     public StreamInfoItem getNextStream() throws IOException, ExtractionException { | ||||
|         assertPageFetched(); | ||||
|         try { | ||||
|             StreamInfoItemsCollector collector = new StreamInfoItemsCollector(getServiceId()); | ||||
|             collector.commit(extractVideoPreviewInfo(doc.select("div[class=\"watch-sidebar-section\"]") | ||||
|                     .first().select("li").first())); | ||||
| 
 | ||||
|             Elements watch = doc.select("div[class=\"watch-sidebar-section\"]"); | ||||
|             if (watch.size() < 1) { | ||||
|                 return null;// prevent the snackbar notification "report error" on age-restricted videos | ||||
|             } | ||||
|              | ||||
|             collector.commit(extractVideoPreviewInfo(watch.first().select("li").first())); | ||||
|             return collector.getItems().get(0); | ||||
|         } catch (Exception e) { | ||||
|             throw new ParsingException("Could not get next video", e); | ||||
|  | @ -504,7 +509,7 @@ public class YoutubeStreamExtractor extends StreamExtractor { | |||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public StreamInfoItemsCollector getRelatedVideos() throws IOException, ExtractionException { | ||||
|     public StreamInfoItemsCollector getRelatedStreams() throws IOException, ExtractionException { | ||||
|         assertPageFetched(); | ||||
|         try { | ||||
|             StreamInfoItemsCollector collector = new StreamInfoItemsCollector(getServiceId()); | ||||
|  | @ -815,21 +820,16 @@ public class YoutubeStreamExtractor extends StreamExtractor { | |||
|         final String languageCode; | ||||
|         final boolean isGenerated; | ||||
| 
 | ||||
|         final Locale locale; | ||||
| 
 | ||||
|         public SubtitlesInfo(final String baseUrl, final String languageCode, final boolean isGenerated) { | ||||
|             this.cleanUrl = baseUrl | ||||
|                     .replaceAll("&fmt=[^&]*", "") // Remove preexisting format if exists | ||||
|                     .replaceAll("&tlang=[^&]*", ""); // Remove translation language | ||||
|             this.languageCode = languageCode; | ||||
|             this.isGenerated = isGenerated; | ||||
| 
 | ||||
|             final String[] splits = languageCode.split("-"); | ||||
|             this.locale = splits.length == 2 ? new Locale(splits[0], splits[1]) : new Locale(languageCode); | ||||
|         } | ||||
| 
 | ||||
|         public Subtitles getSubtitle(final SubtitlesFormat format) { | ||||
|             return new Subtitles(format, locale, cleanUrl + "&fmt=" + format.getExtension(), isGenerated); | ||||
|         public SubtitlesStream getSubtitle(final MediaFormat format) { | ||||
|             return new SubtitlesStream(format, languageCode, cleanUrl + "&fmt=" + format.getSuffix(), isGenerated); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|  |  | |||
|  | @ -50,7 +50,7 @@ public class YoutubeTrendingExtractor extends KioskExtractor { | |||
| 
 | ||||
|     @Override | ||||
|     public void onFetchPage(@Nonnull Downloader downloader) throws IOException, ExtractionException { | ||||
|         final String contentCountry = getContentCountry(); | ||||
|         final String contentCountry = getLocalization().getCountry(); | ||||
|         String url = getUrl(); | ||||
|         if(contentCountry != null && !contentCountry.isEmpty()) { | ||||
|             url += "?gl=" + contentCountry; | ||||
|  |  | |||
|  | @ -1,7 +1,7 @@ | |||
| package org.schabi.newpipe.extractor.stream; | ||||
| 
 | ||||
| /* | ||||
|  * Created by Christian Schabesberger on 10.08.15. | ||||
|  * Created by Christian Schabesberger on 10.08.18. | ||||
|  * | ||||
|  * Copyright (C) Christian Schabesberger 2016 <chris.schabesberger@mailbox.org> | ||||
|  * StreamExtractor.java is part of NewPipe. | ||||
|  | @ -21,8 +21,8 @@ package org.schabi.newpipe.extractor.stream; | |||
|  */ | ||||
| 
 | ||||
| import org.schabi.newpipe.extractor.Extractor; | ||||
| import org.schabi.newpipe.extractor.MediaFormat; | ||||
| import org.schabi.newpipe.extractor.StreamingService; | ||||
| import org.schabi.newpipe.extractor.Subtitles; | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||
| import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||
| import org.schabi.newpipe.extractor.linkhandler.LinkHandler; | ||||
|  | @ -34,7 +34,7 @@ import java.io.IOException; | |||
| import java.util.List; | ||||
| 
 | ||||
| /** | ||||
|  * Scrapes information from a video streaming service (eg, YouTube). | ||||
|  * Scrapes information from a video/audio streaming service (eg, YouTube). | ||||
|  */ | ||||
| public abstract class StreamExtractor extends Extractor { | ||||
| 
 | ||||
|  | @ -44,22 +44,235 @@ public abstract class StreamExtractor extends Extractor { | |||
|         super(service, linkHandler, localization); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * The day on which the stream got uploaded/created. The return information should be in the format | ||||
|      * dd.mm.yyyy, however it NewPipe will not crash if its not. | ||||
|      * @return The day on which the stream got uploaded. | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     @Nonnull | ||||
|     public abstract String getUploadDate() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * This will return the url to the thumbnail of the stream. Try to return the medium resolution here. | ||||
|      * @return The url of the thumbnail. | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     @Nonnull | ||||
|     public abstract String getThumbnailUrl() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * This is the stream description. On YouTube this is the video description. You can return simple HTML here. | ||||
|      * @return The description of the stream/video. | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     @Nonnull | ||||
|     public abstract String getDescription() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * Get the age limit | ||||
|      * Get the age limit. | ||||
|      * @return The age which limits the content or {@value NO_AGE_LIMIT} if there is no limit | ||||
|      * @throws ParsingException if an error occurs while parsing | ||||
|      */ | ||||
|     public abstract int getAgeLimit() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * This should return the length of a video in seconds. | ||||
|      * @return The length of the stream in seconds. | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     public abstract long getLength() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * If the url you are currently handling contains a time stamp/seek, you can return the | ||||
|      * position it represents here. | ||||
|      * If the url has no time stamp simply return zero. | ||||
|      * @return the timestamp in seconds | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     public abstract long getTimeStamp() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * The count of how many people have watched the video/listened to the audio stream. | ||||
|      * If the current stream has no view count or its not available simply return -1 | ||||
|      * @return amount of views. | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     public abstract long getViewCount() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * The Amount of likes a video/audio stream got. | ||||
|      * If the current stream has no likes or its not available simply return -1 | ||||
|      * @return the amount of likes the stream got | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     public abstract long getLikeCount() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * The Amount of dislikes a video/audio stream got. | ||||
|      * If the current stream has no dislikes or its not available simply return -1 | ||||
|      * @return the amount of likes the stream got | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     public abstract long getDislikeCount() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * The Url to the page of the creator/uploader of the stream. This must not be a homepage, | ||||
|      * but the page offered by the service the extractor handles. This url will be handled by the | ||||
|      * <a href="https://teamnewpipe.github.io/documentation/03_Implement_a_service/#channel">ChannelExtractor</a>, | ||||
|      * so be sure to implement that one before you return a value here, otherwise NewPipe will crash if one selects | ||||
|      * this url. | ||||
|      * @return the url to the page of the creator/uploader of the stream or an empty String | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     @Nonnull | ||||
|     public abstract String getUploaderUrl() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * The name of the creator/uploader of the stream. | ||||
|      * If the name is not available you can simply return an empty string. | ||||
|      * @return the name of the creator/uploader of the stream or an empty String | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     @Nonnull | ||||
|     public abstract String getUploaderName() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * The url to the image file/profile picture/avatar of the creator/uploader of the stream. | ||||
|      * If the url is not available you can return an empty String. | ||||
|      * @return The url of the image file of the uploader or an empty String | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     @Nonnull | ||||
|     public abstract String getUploaderAvatarUrl() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * Get the dash mpd url. If you don't know what a dash MPD is you can read about it | ||||
|      * <a href="https://www.brendanlong.com/the-structure-of-an-mpeg-dash-mpd.html">here</a>. | ||||
|      * @return the url as a string or an empty string | ||||
|      * @throws ParsingException if an error occurs while reading | ||||
|      */ | ||||
|     @Nonnull public abstract String getDashMpdUrl() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * I am not sure if this is in use, and how this is used. However the frontend is missing support | ||||
|      * for HLS streams. Prove me if I am wrong. Please open an | ||||
|      * <a href="https://github.com/teamnewpipe/newpipe/issues">issue</a>, | ||||
|      * or fix this description if you know whats up with this. | ||||
|      * @return The Url to the hls stream. | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     @Nonnull public abstract String getHlsUrl() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * This should return a list of available | ||||
|      * <a href="https://teamnewpipe.github.io/NewPipeExtractor/javadoc/org/schabi/newpipe/extractor/stream/AudioStream.html">AudioStream</a>s | ||||
|      * You can also return null or an empty list, however be aware that if you don't return anything | ||||
|      * in getVideoStreams(), getVideoOnlyStreams() and getDashMpdUrl() either the Collector will handle this as | ||||
|      * a failed extraction procedure. | ||||
|      * @return a list of audio only streams in the format of AudioStream | ||||
|      * @throws IOException | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     public abstract List<AudioStream> getAudioStreams() throws IOException, ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * This should return a list of available | ||||
|      * <a href="https://teamnewpipe.github.io/NewPipeExtractor/javadoc/org/schabi/newpipe/extractor/stream/VideoStream.html">VideoStream</a>s | ||||
|      * Be aware this is the list of video streams which do contain an audio stream. | ||||
|      * You can also return null or an empty list, however be aware that if you don't return anything | ||||
|      * in getAudioStreams(), getVideoOnlyStreams() and getDashMpdUrl() either the Collector will handle this as | ||||
|      * a failed extraction procedure. | ||||
|      * @return a list of combined video and streams in the format of AudioStream | ||||
|      * @throws IOException | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     public abstract List<VideoStream> getVideoStreams() throws IOException, ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * This should return a list of available | ||||
|      * <a href="https://teamnewpipe.github.io/NewPipeExtractor/javadoc/org/schabi/newpipe/extractor/stream/VideoStream.html">VideoStream</a>s. | ||||
|      * Be aware this is the list of video streams which do NOT contain an audio stream. | ||||
|      * You can also return null or an empty list, however be aware that if you don't return anything | ||||
|      * in getAudioStreams(), getVideoStreams() and getDashMpdUrl() either the Collector will handle this as | ||||
|      * a failed extraction procedure. | ||||
|      * @return a list of video and streams in the format of AudioStream | ||||
|      * @throws IOException | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     public abstract List<VideoStream> getVideoOnlyStreams() throws IOException, ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * This will return a list of available | ||||
|      * <a href="https://teamnewpipe.github.io/NewPipeExtractor/javadoc/org/schabi/newpipe/extractor/stream/Subtitles.html">Subtitles</a>s. | ||||
|      * If no subtitles are available an empty list can returned. | ||||
|      * @return a list of available subtitles or an empty list | ||||
|      * @throws IOException | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     @Nonnull | ||||
|     public abstract List<SubtitlesStream> getSubtitlesDefault() throws IOException, ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * This will return a list of available | ||||
|      * <a href="https://teamnewpipe.github.io/NewPipeExtractor/javadoc/org/schabi/newpipe/extractor/stream/Subtitles.html">Subtitles</a>s. | ||||
|      * given by a specific type. | ||||
|      * If no subtitles in that specific format are available an empty list can returned. | ||||
|      * @param format the media format by which the subtitles should be filtered | ||||
|      * @return a list of available subtitles or an empty list | ||||
|      * @throws IOException | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     @Nonnull | ||||
|     public abstract List<SubtitlesStream> getSubtitles(MediaFormat format) throws IOException, ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * Get the <a href="https://teamnewpipe.github.io/NewPipeExtractor/javadoc/">StreamType</a>. | ||||
|      * @return the type of the stream | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     public abstract StreamType getStreamType() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * should return the url of the next stream. NewPipe will automatically play | ||||
|      * the next stream if the user wants that. | ||||
|      * If the next stream is is not available simply return null | ||||
|      * @return the InfoItem of the next stream | ||||
|      * @throws IOException | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     public abstract StreamInfoItem getNextStream() throws IOException, ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * Should return a list of streams related to the current handled. Many services show suggested | ||||
|      * streams. If you don't like suggested streams you should implement them anyway since they can | ||||
|      * be disabled by the user later in the frontend. | ||||
|      * This list MUST NOT contain the next available video as this should be return through getNextStream() | ||||
|      * If  is is not available simply return null | ||||
|      * @return a list of InfoItems showing the related videos/streams | ||||
|      * @throws IOException | ||||
|      * @throws ExtractionException | ||||
|      */ | ||||
|     public abstract StreamInfoItemsCollector getRelatedStreams() throws IOException, ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * Should analyse the webpage's document and extracts any error message there might be. (e.g. GEMA block) | ||||
|      * | ||||
|      * @return Error message; null if there is no error message. | ||||
|      */ | ||||
|     public abstract String getErrorMessage(); | ||||
| 
 | ||||
|     ////////////////////////////////////////////////////////////////// | ||||
|     ///  Helper | ||||
|     ////////////////////////////////////////////////////////////////// | ||||
| 
 | ||||
|     /** | ||||
|      * Override this function if the format of time stamp in the url is not the same format as that form youtube. | ||||
|      * Honestly I don't even know the time stamp fromat of youtube. | ||||
|      * @param regexPattern | ||||
|      * @return the sime stamp/seek for the video in seconds | ||||
|      * @throws ParsingException | ||||
|      */ | ||||
|     protected long getTimestampSeconds(String regexPattern) throws ParsingException { | ||||
|         String timeStamp; | ||||
|         try { | ||||
|  | @ -103,43 +316,6 @@ public abstract class StreamExtractor extends Extractor { | |||
|             } | ||||
|         } else { | ||||
|             return 0; | ||||
|         }}; | ||||
| 
 | ||||
|     public abstract long getViewCount() throws ParsingException; | ||||
|     public abstract long getLikeCount() throws ParsingException; | ||||
|     public abstract long getDislikeCount() throws ParsingException; | ||||
| 
 | ||||
|     @Nonnull | ||||
|     public abstract String getUploaderUrl() throws ParsingException; | ||||
|     @Nonnull | ||||
|     public abstract String getUploaderName() throws ParsingException; | ||||
|     @Nonnull | ||||
|     public abstract String getUploaderAvatarUrl() throws ParsingException; | ||||
| 
 | ||||
|     /** | ||||
|      * Get the dash mpd url | ||||
|      * @return the url as a string or an empty string | ||||
|      * @throws ParsingException if an error occurs while reading | ||||
|      */ | ||||
|     @Nonnull public abstract String getDashMpdUrl() throws ParsingException; | ||||
|     @Nonnull public abstract String getHlsUrl() throws ParsingException; | ||||
|     public abstract List<AudioStream> getAudioStreams() throws IOException, ExtractionException; | ||||
|     public abstract List<VideoStream> getVideoStreams() throws IOException, ExtractionException; | ||||
|     public abstract List<VideoStream> getVideoOnlyStreams() throws IOException, ExtractionException; | ||||
| 
 | ||||
|     @Nonnull | ||||
|     public abstract List<Subtitles> getSubtitlesDefault() throws IOException, ExtractionException; | ||||
|     @Nonnull | ||||
|     public abstract List<Subtitles> getSubtitles(SubtitlesFormat format) throws IOException, ExtractionException; | ||||
| 
 | ||||
|     public abstract StreamType getStreamType() throws ParsingException; | ||||
|     public abstract StreamInfoItem getNextVideo() throws IOException, ExtractionException; | ||||
|     public abstract StreamInfoItemsCollector getRelatedVideos() throws IOException, ExtractionException; | ||||
| 
 | ||||
|     /** | ||||
|      * Analyses the webpage's document and extracts any error message there might be. | ||||
|      * | ||||
|      * @return Error message; null if there is no error message. | ||||
|      */ | ||||
|     public abstract String getErrorMessage(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -8,13 +8,10 @@ import org.schabi.newpipe.extractor.Info; | |||
| import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.NewPipe; | ||||
| import org.schabi.newpipe.extractor.StreamingService; | ||||
| import org.schabi.newpipe.extractor.Subtitles; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfo; | ||||
| import org.schabi.newpipe.extractor.exceptions.ContentNotAvailableException; | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||
| import org.schabi.newpipe.extractor.utils.DashMpdParser; | ||||
| import org.schabi.newpipe.extractor.utils.ExtractorHelper; | ||||
| import org.schabi.newpipe.extractor.utils.Localization; | ||||
| 
 | ||||
| /* | ||||
|  * Created by Christian Schabesberger on 26.08.15. | ||||
|  | @ -163,6 +160,9 @@ public class StreamInfo extends Info { | |||
|                 streamInfo.getVideoOnlyStreams().addAll(result.getVideoOnlyStreams()); | ||||
|                 streamInfo.getAudioStreams().addAll(result.getAudioStreams()); | ||||
|                 streamInfo.getVideoStreams().addAll(result.getVideoStreams()); | ||||
|                 streamInfo.segmentedVideoOnlyStreams = result.getSegmentedVideoOnlyStreams(); | ||||
|                 streamInfo.segmentedAudioStreams = result.getSegmentedAudioStreams(); | ||||
|                 streamInfo.segmentedVideoStreams = result.getSegmentedVideoStreams(); | ||||
|             } catch (Exception e) { | ||||
|                 // Sometimes we receive 403 (forbidden) error when trying to download the | ||||
|                 // manifest (similar to what happens with youtube-dl), | ||||
|  | @ -254,7 +254,7 @@ public class StreamInfo extends Info { | |||
|             streamInfo.addError(e); | ||||
|         } | ||||
|         try { | ||||
|             streamInfo.setNextVideo(extractor.getNextVideo()); | ||||
|             streamInfo.setNextVideo(extractor.getNextStream()); | ||||
|         } catch (Exception e) { | ||||
|             streamInfo.addError(e); | ||||
|         } | ||||
|  | @ -289,12 +289,17 @@ public class StreamInfo extends Info { | |||
|     private List<VideoStream> videoOnlyStreams; | ||||
| 
 | ||||
|     private String dashMpdUrl; | ||||
|     private List<VideoStream> segmentedVideoStreams; | ||||
|     private List<AudioStream> segmentedAudioStreams; | ||||
|     private List<VideoStream> segmentedVideoOnlyStreams; | ||||
| 
 | ||||
| 
 | ||||
|     private String hlsUrl; | ||||
|     private StreamInfoItem nextVideo; | ||||
|     private List<InfoItem> relatedStreams; | ||||
| 
 | ||||
|     private long startPosition = 0; | ||||
|     private List<Subtitles> subtitles; | ||||
|     private List<SubtitlesStream> subtitles; | ||||
| 
 | ||||
|     /** | ||||
|      * Get the stream type | ||||
|  | @ -449,6 +454,30 @@ public class StreamInfo extends Info { | |||
|         this.dashMpdUrl = dashMpdUrl; | ||||
|     } | ||||
| 
 | ||||
|     public List<VideoStream> getSegmentedVideoStreams() { | ||||
|         return segmentedVideoStreams; | ||||
|     } | ||||
| 
 | ||||
|     public void setSegmentedVideoStreams(List<VideoStream> segmentedVideoStreams) { | ||||
|         this.segmentedVideoStreams = segmentedVideoStreams; | ||||
|     } | ||||
| 
 | ||||
|     public List<AudioStream> getSegmentedAudioStreams() { | ||||
|         return segmentedAudioStreams; | ||||
|     } | ||||
| 
 | ||||
|     public void setSegmentedAudioStreams(List<AudioStream> segmentedAudioStreams) { | ||||
|         this.segmentedAudioStreams = segmentedAudioStreams; | ||||
|     } | ||||
| 
 | ||||
|     public List<VideoStream> getSegmentedVideoOnlyStreams() { | ||||
|         return segmentedVideoOnlyStreams; | ||||
|     } | ||||
| 
 | ||||
|     public void setSegmentedVideoOnlyStreams(List<VideoStream> segmentedVideoOnlyStreams) { | ||||
|         this.segmentedVideoOnlyStreams = segmentedVideoOnlyStreams; | ||||
|     } | ||||
| 
 | ||||
|     public String getHlsUrl() { | ||||
|         return hlsUrl; | ||||
|     } | ||||
|  | @ -481,11 +510,11 @@ public class StreamInfo extends Info { | |||
|         this.startPosition = startPosition; | ||||
|     } | ||||
| 
 | ||||
|     public List<Subtitles> getSubtitles() { | ||||
|     public List<SubtitlesStream> getSubtitles() { | ||||
|         return subtitles; | ||||
|     } | ||||
| 
 | ||||
|     public void setSubtitles(List<Subtitles> subtitles) { | ||||
|     public void setSubtitles(List<SubtitlesStream> subtitles) { | ||||
|         this.subtitles = subtitles; | ||||
|     } | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,25 +0,0 @@ | |||
| package org.schabi.newpipe.extractor.stream; | ||||
| 
 | ||||
| public enum SubtitlesFormat { | ||||
|     // YouTube subtitles formats | ||||
|     // TRANSCRIPT(3) is default YT format based on TTML, | ||||
|     // but unlike VTT or TTML, it is NOT W3 standard | ||||
|     // TRANSCRIPT subtitles are NOT supported by ExoPlayer, only VTT and TTML | ||||
|     VTT (0x0, "vtt"), | ||||
|     TTML (0x1, "ttml"), | ||||
|     TRANSCRIPT1 (0x2, "srv1"), | ||||
|     TRANSCRIPT2 (0x3, "srv2"), | ||||
|     TRANSCRIPT3 (0x4, "srv3"); | ||||
| 
 | ||||
|     private final int id; | ||||
|     private final String extension; | ||||
| 
 | ||||
|     SubtitlesFormat(int id, String extension) { | ||||
|         this.id = id; | ||||
|         this.extension = extension; | ||||
|     } | ||||
| 
 | ||||
|     public String getExtension() { | ||||
|         return extension; | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,73 @@ | |||
| package org.schabi.newpipe.extractor.stream; | ||||
| 
 | ||||
| import org.schabi.newpipe.extractor.MediaFormat; | ||||
| 
 | ||||
| import java.io.Serializable; | ||||
| import java.util.Locale; | ||||
| 
 | ||||
| public class SubtitlesStream extends Stream implements Serializable { | ||||
|     private final MediaFormat format; | ||||
|     private final Locale locale; | ||||
|     private final String url; | ||||
|     private final boolean autoGenerated; | ||||
|     private final String code; | ||||
| 
 | ||||
|     public SubtitlesStream(MediaFormat format, String languageCode, String url, boolean autoGenerated) { | ||||
|         super(url, format); | ||||
| 
 | ||||
|         /* | ||||
|         * Locale.forLanguageTag only for API >= 21 | ||||
|         * Locale.Builder only for API >= 21 | ||||
|         * Country codes doesn't work well without | ||||
|         */ | ||||
|         final String[] splits = languageCode.split("-"); | ||||
|         switch (splits.length) { | ||||
|             default: | ||||
|                 this.locale = new Locale(splits[0]); | ||||
|                 break; | ||||
|             case 3: | ||||
|                 this.locale = new Locale(splits[0], splits[1], splits[2]);// complex variants doesn't work! | ||||
|                 break; | ||||
|             case 2: | ||||
|                 this.locale = new Locale(splits[0], splits[1]); | ||||
|                 break; | ||||
|         } | ||||
|         this.code = languageCode; | ||||
|         this.format = format; | ||||
|         this.url = url; | ||||
|         this.autoGenerated = autoGenerated; | ||||
|     } | ||||
| 
 | ||||
|     public String getExtension() { | ||||
|         return format.suffix; | ||||
|     } | ||||
| 
 | ||||
|     public String getURL() { | ||||
|         return url; | ||||
|     } | ||||
| 
 | ||||
|     public boolean isAutoGenerated() { | ||||
|         return autoGenerated; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean equalStats(Stream cmp) { | ||||
|         return super.equalStats(cmp)&& | ||||
|                 cmp instanceof SubtitlesStream && | ||||
|                 code.equals(((SubtitlesStream) cmp).code) && | ||||
|                 autoGenerated == ((SubtitlesStream) cmp).autoGenerated; | ||||
|     } | ||||
| 
 | ||||
|     public String getDisplayLanguageName() { | ||||
|         return locale.getDisplayName(locale); | ||||
|     } | ||||
| 
 | ||||
|     public String getLanguageTag() { | ||||
|         return code; | ||||
|     } | ||||
| 
 | ||||
|     public Locale getLocale() { | ||||
|         return  locale; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -59,10 +59,23 @@ public class DashMpdParser { | |||
|         private final List<AudioStream> audioStreams; | ||||
|         private final List<VideoStream> videoOnlyStreams; | ||||
| 
 | ||||
|         public ParserResult(List<VideoStream> videoStreams, List<AudioStream> audioStreams, List<VideoStream> videoOnlyStreams) { | ||||
|         private final List<VideoStream> segmentedVideoStreams; | ||||
|         private final List<AudioStream> segmentedAudioStreams; | ||||
|         private final List<VideoStream> segmentedVideoOnlyStreams; | ||||
| 
 | ||||
| 
 | ||||
|         public ParserResult(List<VideoStream> videoStreams, | ||||
|                             List<AudioStream> audioStreams, | ||||
|                             List<VideoStream> videoOnlyStreams, | ||||
|                             List<VideoStream> segmentedVideoStreams, | ||||
|                             List<AudioStream> segmentedAudioStreams, | ||||
|                             List<VideoStream> segmentedVideoOnlyStreams) { | ||||
|             this.videoStreams = videoStreams; | ||||
|             this.audioStreams = audioStreams; | ||||
|             this.videoOnlyStreams = videoOnlyStreams; | ||||
|             this.segmentedVideoStreams = segmentedVideoStreams; | ||||
|             this.segmentedAudioStreams = segmentedAudioStreams; | ||||
|             this.segmentedVideoOnlyStreams = segmentedVideoOnlyStreams; | ||||
|         } | ||||
| 
 | ||||
|         public List<VideoStream> getVideoStreams() { | ||||
|  | @ -76,10 +89,22 @@ public class DashMpdParser { | |||
|         public List<VideoStream> getVideoOnlyStreams() { | ||||
|             return videoOnlyStreams; | ||||
|         } | ||||
| 
 | ||||
|         public List<VideoStream> getSegmentedVideoStreams() { | ||||
|             return segmentedVideoStreams; | ||||
|         } | ||||
| 
 | ||||
|         public List<AudioStream> getSegmentedAudioStreams() { | ||||
|             return segmentedAudioStreams; | ||||
|         } | ||||
| 
 | ||||
|         public List<VideoStream> getSegmentedVideoOnlyStreams() { | ||||
|             return segmentedVideoOnlyStreams; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Will try to download (using {@link StreamInfo#dashMpdUrl}) and parse the dash manifest, | ||||
|      * Will try to download (using {@link StreamInfo#getDashMpdUrl()}) and parse the dash manifest, | ||||
|      * then it will search for any stream that the ItagItem has (by the id). | ||||
|      * <p> | ||||
|      * It has video, video only and audio streams and will only add to the list if it don't | ||||
|  | @ -90,7 +115,8 @@ public class DashMpdParser { | |||
|      * | ||||
|      * @param streamInfo where the parsed streams will be added | ||||
|      */ | ||||
|     public static ParserResult getStreams(final StreamInfo streamInfo) throws DashMpdParsingException, ReCaptchaException { | ||||
|     public static ParserResult getStreams(final StreamInfo streamInfo) | ||||
|             throws DashMpdParsingException, ReCaptchaException { | ||||
|         String dashDoc; | ||||
|         Downloader downloader = NewPipe.getDownloader(); | ||||
|         try { | ||||
|  | @ -113,6 +139,10 @@ public class DashMpdParser { | |||
|             final List<AudioStream> audioStreams = new ArrayList<>(); | ||||
|             final List<VideoStream> videoOnlyStreams = new ArrayList<>(); | ||||
| 
 | ||||
|             final List<VideoStream> segmentedVideoStreams = new ArrayList<>(); | ||||
|             final List<AudioStream> segmentedAudioStreams = new ArrayList<>(); | ||||
|             final List<VideoStream> segmentedVideoOnlyStreams = new ArrayList<>(); | ||||
| 
 | ||||
|             for (int i = 0; i < representationList.getLength(); i++) { | ||||
|                 final Element representation = (Element) representationList.item(i); | ||||
|                 try { | ||||
|  | @ -126,34 +156,61 @@ public class DashMpdParser { | |||
|                     // instead we need to add the "media=" value from the <SegementURL/> tags inside the <SegmentList/> | ||||
|                     // tag in order to get a full working url. However each of these is just pointing to a part of the | ||||
|                     // video, so we can not return a URL with a working stream here. | ||||
|                     // We decided not to ignore such streams for the moment. | ||||
|                     if (itag != null && segmentationList == null) { | ||||
|                     // Instead of putting those streams into the list of regular stream urls wie put them in a | ||||
|                     // for example "segmentedVideoStreams" list. | ||||
|                     if (itag != null) { | ||||
|                         final MediaFormat mediaFormat = MediaFormat.getFromMimeType(mimeType); | ||||
| 
 | ||||
|                         if (itag.itagType.equals(ItagItem.ItagType.AUDIO)) { | ||||
|                             final AudioStream audioStream = new AudioStream(url, mediaFormat, itag.avgBitrate); | ||||
| 
 | ||||
|                             if (!Stream.containSimilarStream(audioStream, streamInfo.getAudioStreams())) { | ||||
|                                 audioStreams.add(audioStream); | ||||
|                             if(segmentationList == null) { | ||||
|                                 final AudioStream audioStream = new AudioStream(url, mediaFormat, itag.avgBitrate); | ||||
|                                 if (!Stream.containSimilarStream(audioStream, streamInfo.getAudioStreams())) { | ||||
|                                     audioStreams.add(audioStream); | ||||
|                                 } | ||||
|                             } else { | ||||
|                                 segmentedAudioStreams.add( | ||||
|                                         new AudioStream(id, mediaFormat, itag.avgBitrate)); | ||||
|                             } | ||||
|                         } else { | ||||
|                             boolean isVideoOnly = itag.itagType.equals(ItagItem.ItagType.VIDEO_ONLY); | ||||
|                             final VideoStream videoStream = new VideoStream(url, mediaFormat, itag.resolutionString, isVideoOnly); | ||||
| 
 | ||||
|                             if (isVideoOnly) { | ||||
|                                 if (!Stream.containSimilarStream(videoStream, streamInfo.getVideoOnlyStreams())) { | ||||
|                                     streamInfo.getVideoOnlyStreams().add(videoStream); | ||||
|                                     videoOnlyStreams.add(videoStream); | ||||
|                             if(segmentationList == null) { | ||||
|                                 final VideoStream videoStream = new VideoStream(url, | ||||
|                                         mediaFormat, | ||||
|                                         itag.resolutionString, | ||||
|                                         isVideoOnly); | ||||
| 
 | ||||
|                                 if (isVideoOnly) { | ||||
|                                     if (!Stream.containSimilarStream(videoStream, streamInfo.getVideoOnlyStreams())) { | ||||
|                                         videoOnlyStreams.add(videoStream); | ||||
|                                     } | ||||
|                                 } else if (!Stream.containSimilarStream(videoStream, streamInfo.getVideoStreams())) { | ||||
|                                     videoStreams.add(videoStream); | ||||
|                                 } | ||||
|                             } else { | ||||
|                                 final VideoStream videoStream = new VideoStream(id, | ||||
|                                         mediaFormat, | ||||
|                                         itag.resolutionString, | ||||
|                                         isVideoOnly); | ||||
| 
 | ||||
|                                 if(isVideoOnly) { | ||||
|                                     segmentedVideoOnlyStreams.add(videoStream); | ||||
|                                 } else { | ||||
|                                     segmentedVideoStreams.add(videoStream); | ||||
|                                 } | ||||
|                             } else if (!Stream.containSimilarStream(videoStream, streamInfo.getVideoStreams())) { | ||||
|                                 videoStreams.add(videoStream); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } catch (Exception ignored) { | ||||
|                 } | ||||
|             } | ||||
|             return new ParserResult(videoStreams, audioStreams, videoOnlyStreams); | ||||
|             return new ParserResult( | ||||
|                     videoStreams, | ||||
|                     audioStreams, | ||||
|                     videoOnlyStreams, | ||||
|                     segmentedVideoStreams, | ||||
|                     segmentedAudioStreams, | ||||
|                     segmentedVideoOnlyStreams); | ||||
|         } catch (Exception e) { | ||||
|             throw new DashMpdParsingException("Could not parse Dash mpd", e); | ||||
|         } | ||||
|  |  | |||
|  | @ -30,7 +30,7 @@ public class ExtractorHelper { | |||
| 
 | ||||
|     public static List<InfoItem> getRelatedVideosOrLogError(StreamInfo info, StreamExtractor extractor) { | ||||
|         try { | ||||
|             InfoItemsCollector<? extends InfoItem, ?> collector = extractor.getRelatedVideos(); | ||||
|             InfoItemsCollector<? extends InfoItem, ?> collector = extractor.getRelatedStreams(); | ||||
|             info.addAllErrors(collector.getErrors()); | ||||
| 
 | ||||
|             //noinspection unchecked | ||||
|  |  | |||
|  | @ -32,7 +32,7 @@ public class JsonUtils { | |||
|     public static String getString(@Nonnull JsonObject object, @Nonnull String path) throws ParsingException{ | ||||
|         Object value = getValue(object, path); | ||||
|         if(value instanceof String) { | ||||
|             return (String) getValue(object, path); | ||||
|             return (String) value; | ||||
|         }else { | ||||
|             throw new ParsingException("Unable to get " + path); | ||||
|         } | ||||
|  | @ -42,12 +42,31 @@ public class JsonUtils { | |||
|     public static Number getNumber(@Nonnull JsonObject object, @Nonnull String path) throws ParsingException{ | ||||
|         Object value = getValue(object, path); | ||||
|         if(value instanceof Number) { | ||||
|             return (Number) getValue(object, path); | ||||
|             return (Number) value; | ||||
|         }else { | ||||
|             throw new ParsingException("Unable to get " + path); | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     @Nonnull | ||||
|     public static JsonObject getObject(@Nonnull JsonObject object, @Nonnull String path) throws ParsingException{ | ||||
|         Object value = getValue(object, path); | ||||
|         if(value instanceof JsonObject) { | ||||
|             return (JsonObject) value; | ||||
|         }else { | ||||
|             throw new ParsingException("Unable to get " + path); | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     @Nonnull | ||||
|     public static JsonArray getArray(@Nonnull JsonObject object, @Nonnull String path) throws ParsingException{ | ||||
|         Object value = getValue(object, path); | ||||
|         if(value instanceof JsonArray) { | ||||
|             return (JsonArray) value; | ||||
|         }else { | ||||
|             throw new ParsingException("Unable to get " + path); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     @Nonnull | ||||
|     public static List<Object> getValues(@Nonnull JsonArray array, @Nonnull String path) throws ParsingException { | ||||
|  |  | |||
|  | @ -1,10 +1,5 @@ | |||
| package org.schabi.newpipe.extractor.utils; | ||||
| 
 | ||||
| import org.nibor.autolink.LinkExtractor; | ||||
| import org.nibor.autolink.LinkSpan; | ||||
| import org.nibor.autolink.LinkType; | ||||
| import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||
| 
 | ||||
| import java.io.UnsupportedEncodingException; | ||||
| import java.net.URLDecoder; | ||||
| import java.util.ArrayList; | ||||
|  | @ -14,6 +9,11 @@ import java.util.Map; | |||
| import java.util.regex.Matcher; | ||||
| import java.util.regex.Pattern; | ||||
| 
 | ||||
| import org.nibor.autolink.LinkExtractor; | ||||
| import org.nibor.autolink.LinkSpan; | ||||
| import org.nibor.autolink.LinkType; | ||||
| import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||
| 
 | ||||
| /* | ||||
|  * Created by Christian Schabesberger on 02.02.16. | ||||
|  * | ||||
|  | @ -51,18 +51,26 @@ public class Parser { | |||
|     public static String matchGroup1(String pattern, String input) throws RegexException { | ||||
|         return matchGroup(pattern, input, 1); | ||||
|     } | ||||
|      | ||||
|     public static String matchGroup1(Pattern pattern, String input) throws RegexException { | ||||
|         return matchGroup(pattern, input, 1); | ||||
|     } | ||||
| 
 | ||||
|     public static String matchGroup(String pattern, String input, int group) throws RegexException { | ||||
|         Pattern pat = Pattern.compile(pattern); | ||||
|         return matchGroup(pat, input, group); | ||||
|     } | ||||
|      | ||||
|     public static String matchGroup(Pattern pat, String input, int group) throws RegexException { | ||||
|         Matcher mat = pat.matcher(input); | ||||
|         boolean foundMatch = mat.find(); | ||||
|         if (foundMatch) { | ||||
|             return mat.group(group); | ||||
|         } else { | ||||
|             if (input.length() > 1024) { | ||||
|                 throw new RegexException("failed to find pattern \"" + pattern); | ||||
|                 throw new RegexException("failed to find pattern \"" + pat.pattern()); | ||||
|             } else { | ||||
|                 throw new RegexException("failed to find pattern \"" + pattern + " inside of " + input + "\""); | ||||
|                 throw new RegexException("failed to find pattern \"" + pat.pattern() + " inside of " + input + "\""); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  |  | |||
|  | @ -1,5 +1,13 @@ | |||
| package org.schabi.newpipe.extractor.services.peertube; | ||||
| 
 | ||||
| import static org.junit.Assert.assertEquals; | ||||
| import static org.junit.Assert.assertFalse; | ||||
| import static org.junit.Assert.assertTrue; | ||||
| import static org.schabi.newpipe.extractor.ExtractorAsserts.assertIsSecureUrl; | ||||
| import static org.schabi.newpipe.extractor.ServiceList.PeerTube; | ||||
| 
 | ||||
| import java.io.IOException; | ||||
| 
 | ||||
| import org.junit.BeforeClass; | ||||
| import org.junit.Test; | ||||
| import org.schabi.newpipe.Downloader; | ||||
|  | @ -12,12 +20,6 @@ import org.schabi.newpipe.extractor.stream.StreamInfoItemsCollector; | |||
| import org.schabi.newpipe.extractor.stream.StreamType; | ||||
| import org.schabi.newpipe.extractor.utils.Localization; | ||||
| 
 | ||||
| import java.io.IOException; | ||||
| 
 | ||||
| import static org.junit.Assert.*; | ||||
| import static org.schabi.newpipe.extractor.ExtractorAsserts.assertIsSecureUrl; | ||||
| import static org.schabi.newpipe.extractor.ServiceList.PeerTube; | ||||
| 
 | ||||
| /** | ||||
|  * Test for {@link StreamExtractor} | ||||
|  */ | ||||
|  | @ -27,7 +29,7 @@ public class PeertubeStreamExtractorDefaultTest { | |||
|     @BeforeClass | ||||
|     public static void setUp() throws Exception { | ||||
|         NewPipe.init(Downloader.getInstance(), new Localization("GB", "en")); | ||||
|         extractor = (PeertubeStreamExtractor) PeerTube.getStreamExtractor("https://peertube.mastodon.host/videos/watch/86fe4f24-64c3-4ab4-9e7e-66177219ed21"); | ||||
|         extractor = (PeertubeStreamExtractor) PeerTube.getStreamExtractor("https://peertube.mastodon.host/videos/watch/04af977f-4201-4697-be67-a8d8cae6fa7a"); | ||||
|         extractor.fetchPage(); | ||||
|     } | ||||
| 
 | ||||
|  | @ -39,22 +41,22 @@ public class PeertubeStreamExtractorDefaultTest { | |||
| 
 | ||||
|     @Test | ||||
|     public void testGetTitle() throws ParsingException { | ||||
|         assertEquals(extractor.getName(), "Pažadėtoji 1 Sezonas 1050 Serija - Ziuri.me"); | ||||
|         assertEquals(extractor.getName(), "The Internet's Own Boy"); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testGetDescription() throws ParsingException { | ||||
|         assertEquals(extractor.getDescription(), "Serialo veiksmas vyksta Radžastane. kur vis dar gyvos liaudies tradicijos. o jo centre - Anandi gimusi varguolių šeimoje. Mergaitė privalės ištekėti už turtingo paveldėtojo vos tik sulaukusi aštuonerių metų ir priprasti prie naujojo nuotakos vaidm..."); | ||||
|         assertEquals(extractor.getDescription(), "The story of programming prodigy and information activist Aaron Swartz, who took his own life at the age of 26."); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testGetUploaderName() throws ParsingException { | ||||
|         assertEquals(extractor.getUploaderName(), "djsets"); | ||||
|         assertEquals(extractor.getUploaderName(), "root"); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testGetLength() throws ParsingException { | ||||
|         assertEquals(extractor.getLength(), 1238); | ||||
|         assertEquals(extractor.getLength(), 6299); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|  | @ -65,13 +67,13 @@ public class PeertubeStreamExtractorDefaultTest { | |||
| 
 | ||||
|     @Test | ||||
|     public void testGetUploadDate() throws ParsingException { | ||||
|         assertEquals("2018-10-06", extractor.getUploadDate()); | ||||
|         assertEquals("2017-10-17", extractor.getUploadDate()); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testGetUploaderUrl() throws ParsingException { | ||||
|         assertIsSecureUrl(extractor.getUploaderUrl()); | ||||
|         assertEquals("https://peertube.mastodon.host/api/v1/accounts/djsets@hostyour.tv", extractor.getUploaderUrl()); | ||||
|         assertEquals("https://peertube.mastodon.host/api/v1/accounts/root@peertube2.cpy.re", extractor.getUploaderUrl()); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|  | @ -96,7 +98,7 @@ public class PeertubeStreamExtractorDefaultTest { | |||
| 
 | ||||
|     @Test | ||||
|     public void testGetRelatedVideos() throws ExtractionException, IOException { | ||||
|         StreamInfoItemsCollector relatedVideos = extractor.getRelatedVideos(); | ||||
|         StreamInfoItemsCollector relatedVideos = extractor.getRelatedStreams(); | ||||
|         assertFalse(relatedVideos.getItems().isEmpty()); | ||||
|         assertTrue(relatedVideos.getErrors().isEmpty()); | ||||
|     } | ||||
|  |  | |||
|  | @ -101,7 +101,7 @@ public class SoundcloudStreamExtractorDefaultTest { | |||
| 
 | ||||
|     @Test | ||||
|     public void testGetRelatedVideos() throws ExtractionException, IOException { | ||||
|         StreamInfoItemsCollector relatedVideos = extractor.getRelatedVideos(); | ||||
|         StreamInfoItemsCollector relatedVideos = extractor.getRelatedStreams(); | ||||
|         assertFalse(relatedVideos.getItems().isEmpty()); | ||||
|         assertTrue(relatedVideos.getErrors().isEmpty()); | ||||
|     } | ||||
|  |  | |||
|  | @ -219,7 +219,7 @@ public class YoutubePlaylistExtractorTest { | |||
| 
 | ||||
|         @Test | ||||
|         public void testUploaderName() throws Exception { | ||||
|             assertEquals("Tomas Nilsson", extractor.getUploaderName()); | ||||
|             assertEquals("Tomas Nilsson TOMPA571", extractor.getUploaderName()); | ||||
|         } | ||||
| 
 | ||||
|         @Test | ||||
|  |  | |||
|  | @ -4,13 +4,13 @@ import org.junit.BeforeClass; | |||
| import org.junit.Ignore; | ||||
| import org.junit.Test; | ||||
| import org.schabi.newpipe.Downloader; | ||||
| import org.schabi.newpipe.extractor.MediaFormat; | ||||
| import org.schabi.newpipe.extractor.NewPipe; | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||
| import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||
| import org.schabi.newpipe.extractor.services.youtube.extractors.YoutubeStreamExtractor; | ||||
| import org.schabi.newpipe.extractor.services.youtube.linkHandler.YoutubeStreamLinkHandlerFactory; | ||||
| import org.schabi.newpipe.extractor.stream.StreamExtractor; | ||||
| import org.schabi.newpipe.extractor.stream.SubtitlesFormat; | ||||
| import org.schabi.newpipe.extractor.stream.VideoStream; | ||||
| import org.schabi.newpipe.extractor.utils.Localization; | ||||
| 
 | ||||
|  | @ -131,6 +131,6 @@ public class YoutubeStreamExtractorAgeRestrictedTest { | |||
|     @Test | ||||
|     public void testGetSubtitlesList() throws IOException, ExtractionException { | ||||
|         // Video (/view?v=YQHsXMglC9A) set in the setUp() method has no captions => null | ||||
|         assertTrue(extractor.getSubtitles(SubtitlesFormat.TTML).isEmpty()); | ||||
|         assertTrue(extractor.getSubtitles(MediaFormat.TTML).isEmpty()); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -4,13 +4,13 @@ import org.junit.BeforeClass; | |||
| import org.junit.Ignore; | ||||
| import org.junit.Test; | ||||
| import org.schabi.newpipe.Downloader; | ||||
| import org.schabi.newpipe.extractor.MediaFormat; | ||||
| import org.schabi.newpipe.extractor.NewPipe; | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||
| import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||
| import org.schabi.newpipe.extractor.services.youtube.extractors.YoutubeStreamExtractor; | ||||
| import org.schabi.newpipe.extractor.services.youtube.linkHandler.YoutubeStreamLinkHandlerFactory; | ||||
| import org.schabi.newpipe.extractor.stream.StreamExtractor; | ||||
| import org.schabi.newpipe.extractor.stream.SubtitlesFormat; | ||||
| import org.schabi.newpipe.extractor.stream.VideoStream; | ||||
| import org.schabi.newpipe.extractor.utils.Localization; | ||||
| 
 | ||||
|  | @ -124,6 +124,6 @@ public class YoutubeStreamExtractorControversialTest { | |||
|     @Test | ||||
|     public void testGetSubtitlesList() throws IOException, ExtractionException { | ||||
|         // Video (/view?v=YQHsXMglC9A) set in the setUp() method has no captions => null | ||||
|         assertTrue(!extractor.getSubtitles(SubtitlesFormat.TTML).isEmpty()); | ||||
|         assertTrue(!extractor.getSubtitles(MediaFormat.TTML).isEmpty()); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -47,15 +47,21 @@ public class YoutubeStreamExtractorDASHTest { | |||
| 
 | ||||
|     @Test | ||||
|     public void testGetDashMpd() { | ||||
|         System.out.println(info.getDashMpdUrl()); | ||||
|         assertTrue(info.getDashMpdUrl(), | ||||
|                 info.getDashMpdUrl() != null && !info.getDashMpdUrl().isEmpty()); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testDashMpdParser() { | ||||
|     public void testRegularStreams() { | ||||
|         assertEquals(0, info.getAudioStreams().size()); | ||||
|         assertEquals(0, info.getVideoOnlyStreams().size()); | ||||
|         assertEquals(4, info.getVideoStreams().size()); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testSegmentedStreams() { | ||||
|         assertEquals(2, info.getSegmentedAudioStreams().size()); | ||||
|         assertEquals(3, info.getSegmentedVideoOnlyStreams().size()); | ||||
|         assertEquals(0, info.getSegmentedVideoStreams().size()); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -3,12 +3,12 @@ package org.schabi.newpipe.extractor.services.youtube; | |||
| import org.junit.BeforeClass; | ||||
| import org.junit.Test; | ||||
| import org.schabi.newpipe.Downloader; | ||||
| import org.schabi.newpipe.extractor.MediaFormat; | ||||
| import org.schabi.newpipe.extractor.NewPipe; | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||
| import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||
| import org.schabi.newpipe.extractor.services.youtube.extractors.YoutubeStreamExtractor; | ||||
| import org.schabi.newpipe.extractor.stream.*; | ||||
| import org.schabi.newpipe.extractor.utils.DashMpdParser; | ||||
| import org.schabi.newpipe.extractor.utils.Localization; | ||||
| import org.schabi.newpipe.extractor.utils.Utils; | ||||
| 
 | ||||
|  | @ -17,7 +17,6 @@ import java.io.IOException; | |||
| import static org.junit.Assert.*; | ||||
| import static org.schabi.newpipe.extractor.ExtractorAsserts.assertIsSecureUrl; | ||||
| import static org.schabi.newpipe.extractor.ServiceList.YouTube; | ||||
| import static org.schabi.newpipe.extractor.services.youtube.YoutubeTrendingExtractorTest.extractor; | ||||
| 
 | ||||
| /* | ||||
|  * Created by Christian Schabesberger on 30.12.15. | ||||
|  | @ -151,7 +150,7 @@ public class YoutubeStreamExtractorDefaultTest { | |||
| 
 | ||||
|         @Test | ||||
|         public void testGetRelatedVideos() throws ExtractionException, IOException { | ||||
|             StreamInfoItemsCollector relatedVideos = extractor.getRelatedVideos(); | ||||
|             StreamInfoItemsCollector relatedVideos = extractor.getRelatedStreams(); | ||||
|             Utils.printErrors(relatedVideos.getErrors()); | ||||
|             assertFalse(relatedVideos.getItems().isEmpty()); | ||||
|             assertTrue(relatedVideos.getErrors().isEmpty()); | ||||
|  | @ -166,7 +165,7 @@ public class YoutubeStreamExtractorDefaultTest { | |||
|         @Test | ||||
|         public void testGetSubtitlesList() throws IOException, ExtractionException { | ||||
|             // Video (/view?v=YQHsXMglC9A) set in the setUp() method has no captions => null | ||||
|             assertTrue(extractor.getSubtitles(SubtitlesFormat.TTML).isEmpty()); | ||||
|             assertTrue(extractor.getSubtitles(MediaFormat.TTML).isEmpty()); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|  | @ -177,7 +176,7 @@ public class YoutubeStreamExtractorDefaultTest { | |||
|         public static void setUp() throws Exception { | ||||
|             NewPipe.init(Downloader.getInstance(), new Localization("GB", "en")); | ||||
|             extractor = (YoutubeStreamExtractor) YouTube | ||||
|                     .getStreamExtractor("https://www.youtube.com/watch?v=LzR8Sf5PK2Q"); | ||||
|                     .getStreamExtractor("https://www.youtube.com/watch?v=fBc4Q_htqPg"); | ||||
|             extractor.fetchPage(); | ||||
|         } | ||||
| 
 | ||||
|  |  | |||
|  | @ -51,7 +51,6 @@ public class YoutubeTrendingExtractorTest { | |||
|                 .getKioskList() | ||||
|                 .getExtractorById("Trending", null); | ||||
|         extractor.fetchPage(); | ||||
|         extractor.setContentCountry("de"); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|  |  | |||
|  | @ -46,7 +46,7 @@ public class YoutubeTrendingKioskInfoTest { | |||
|         StreamingService service = YouTube; | ||||
|         LinkHandlerFactory LinkHandlerFactory = service.getKioskList().getListLinkHandlerFactoryByType("Trending"); | ||||
| 
 | ||||
|         kioskInfo = KioskInfo.getInfo(YouTube, LinkHandlerFactory.fromId("Trending").getUrl(), null); | ||||
|         kioskInfo = KioskInfo.getInfo(YouTube, LinkHandlerFactory.fromId("Trending").getUrl()); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|  |  | |||
|  | @ -28,7 +28,7 @@ public class YoutubeSearchCountTest { | |||
|         public void testViewCount() { | ||||
|             ChannelInfoItem ci = (ChannelInfoItem) itemsPage.getItems().get(0); | ||||
|             assertTrue("Count does not fit: " + Long.toString(ci.getSubscriberCount()), | ||||
|                     65043316 < ci.getSubscriberCount() && ci.getSubscriberCount() < 68043316); | ||||
|                     69043316 < ci.getSubscriberCount() && ci.getSubscriberCount() < 73043316); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -45,12 +45,12 @@ public class YoutubeSearchExtractorChannelOnlyTest extends YoutubeSearchExtracto | |||
|         } | ||||
|         assertFalse("First and second page are equal", equals); | ||||
| 
 | ||||
|         assertEquals("https://www.youtube.com/results?q=pewdiepie&sp=EgIQAlAU&page=3", secondPage.getNextPageUrl()); | ||||
|         assertEquals("https://www.youtube.com/results?q=pewdiepie&sp=EgIQAlAU&gl=GB&page=3", secondPage.getNextPageUrl()); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testGetSecondPageUrl() throws Exception { | ||||
|         assertEquals("https://www.youtube.com/results?q=pewdiepie&sp=EgIQAlAU&page=2", extractor.getNextPageUrl()); | ||||
|         assertEquals("https://www.youtube.com/results?q=pewdiepie&sp=EgIQAlAU&gl=GB&page=2", extractor.getNextPageUrl()); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|  |  | |||
|  | @ -49,18 +49,22 @@ public class YoutubeSearchExtractorDefaultTest extends YoutubeSearchExtractorBas | |||
|         itemsPage = extractor.getInitialPage(); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testGetUrl() throws Exception { | ||||
|         assertEquals("https://www.youtube.com/results?q=pewdiepie&gl=GB", extractor.getUrl()); | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     @Test | ||||
|     public void testGetSecondPageUrl() throws Exception { | ||||
|         assertEquals("https://www.youtube.com/results?q=pewdiepie&page=2", extractor.getNextPageUrl()); | ||||
|         assertEquals("https://www.youtube.com/results?q=pewdiepie&gl=GB&page=2", extractor.getNextPageUrl()); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testResultList_FirstElement() { | ||||
|         InfoItem firstInfoItem = itemsPage.getItems().get(0); | ||||
|         InfoItem firstInfoItem = itemsPage.getItems().get(1); | ||||
| 
 | ||||
|         // THe channel should be the first item | ||||
|         // The channel should be the first item | ||||
|         assertTrue(firstInfoItem instanceof ChannelInfoItem); | ||||
|         assertEquals("name", "PewDiePie", firstInfoItem.getName()); | ||||
|         assertEquals("url","https://www.youtube.com/user/PewDiePie", firstInfoItem.getUrl()); | ||||
|  | @ -96,7 +100,7 @@ public class YoutubeSearchExtractorDefaultTest extends YoutubeSearchExtractorBas | |||
|         } | ||||
|         assertFalse("First and second page are equal", equals); | ||||
| 
 | ||||
|         assertEquals("https://www.youtube.com/results?q=pewdiepie&page=3", secondPage.getNextPageUrl()); | ||||
|         assertEquals("https://www.youtube.com/results?q=pewdiepie&gl=GB&page=3", secondPage.getNextPageUrl()); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue