2017-03-01 17:47:52 +00:00
|
|
|
package org.schabi.newpipe.extractor.stream_info;
|
|
|
|
|
|
|
|
import org.schabi.newpipe.extractor.AbstractStreamInfo;
|
|
|
|
import org.schabi.newpipe.extractor.DashMpdParser;
|
|
|
|
import org.schabi.newpipe.extractor.InfoItem;
|
|
|
|
import org.schabi.newpipe.extractor.UrlIdHandler;
|
|
|
|
import org.schabi.newpipe.extractor.exceptions.ExtractionException;
|
|
|
|
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.util.List;
|
|
|
|
import java.util.Vector;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Created by Christian Schabesberger on 26.08.15.
|
|
|
|
*
|
|
|
|
* Copyright (C) Christian Schabesberger 2016 <chris.schabesberger@mailbox.org>
|
|
|
|
* StreamInfo.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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**Info object for opened videos, ie the video ready to play.*/
|
|
|
|
@SuppressWarnings("ALL")
|
|
|
|
public class StreamInfo extends AbstractStreamInfo {
|
|
|
|
|
|
|
|
public static class StreamExctractException extends ExtractionException {
|
|
|
|
StreamExctractException(String message) {
|
|
|
|
super(message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public StreamInfo() {}
|
|
|
|
|
|
|
|
/**Creates a new StreamInfo object from an existing AbstractVideoInfo.
|
|
|
|
* All the shared properties are copied to the new StreamInfo.*/
|
|
|
|
@SuppressWarnings("WeakerAccess")
|
|
|
|
public StreamInfo(AbstractStreamInfo avi) {
|
|
|
|
this.id = avi.id;
|
|
|
|
this.title = avi.title;
|
|
|
|
this.uploader = avi.uploader;
|
|
|
|
this.thumbnail_url = avi.thumbnail_url;
|
|
|
|
this.webpage_url = avi.webpage_url;
|
|
|
|
this.upload_date = avi.upload_date;
|
|
|
|
this.upload_date = avi.upload_date;
|
|
|
|
this.view_count = avi.view_count;
|
|
|
|
|
|
|
|
//todo: better than this
|
|
|
|
if(avi instanceof StreamInfoItem) {
|
|
|
|
//shitty String to convert code
|
|
|
|
/*
|
|
|
|
String dur = ((StreamInfoItem)avi).duration;
|
|
|
|
int minutes = Integer.parseInt(dur.substring(0, dur.indexOf(":")));
|
|
|
|
int seconds = Integer.parseInt(dur.substring(dur.indexOf(":")+1, dur.length()));
|
|
|
|
*/
|
|
|
|
this.duration = ((StreamInfoItem)avi).duration;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void addException(Exception e) {
|
|
|
|
errors.add(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**Fills out the video info fields which are common to all services.
|
|
|
|
* Probably needs to be overridden by subclasses*/
|
|
|
|
public static StreamInfo getVideoInfo(StreamExtractor extractor)
|
2017-05-27 12:23:48 +00:00
|
|
|
throws ExtractionException, StreamExtractor.ContentNotAvailableException {
|
2017-03-01 17:47:52 +00:00
|
|
|
StreamInfo streamInfo = new StreamInfo();
|
|
|
|
|
2017-05-27 12:23:48 +00:00
|
|
|
try {
|
|
|
|
streamInfo = extractImportantData(streamInfo, extractor);
|
|
|
|
streamInfo = extractStreams(streamInfo, extractor);
|
|
|
|
streamInfo = extractOptionalData(streamInfo, extractor);
|
|
|
|
} catch (ExtractionException e) {
|
|
|
|
// Currently YouTube does not distinguish between age restricted videos and videos blocked
|
|
|
|
// by country. This means that during the initialisation of the extractor, the extractor
|
|
|
|
// will assume that a video is age restricted while in reality it it blocked by country.
|
|
|
|
//
|
|
|
|
// We will now detect whether the video is blocked by country or not.
|
|
|
|
String errorMsg = extractor.getErrorMessage();
|
|
|
|
|
|
|
|
if (errorMsg != null) {
|
|
|
|
throw new StreamExtractor.ContentNotAvailableException(errorMsg);
|
|
|
|
} else {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
2017-03-01 17:47:52 +00:00
|
|
|
|
|
|
|
return streamInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static StreamInfo extractImportantData(
|
|
|
|
StreamInfo streamInfo, StreamExtractor extractor)
|
2017-05-27 12:23:48 +00:00
|
|
|
throws ExtractionException {
|
2017-03-01 17:47:52 +00:00
|
|
|
/* ---- important data, withoug the video can't be displayed goes here: ---- */
|
|
|
|
// if one of these is not available an exception is meant to be thrown directly into the frontend.
|
|
|
|
|
|
|
|
UrlIdHandler uiconv = extractor.getUrlIdHandler();
|
|
|
|
|
|
|
|
streamInfo.service_id = extractor.getServiceId();
|
|
|
|
streamInfo.webpage_url = extractor.getPageUrl();
|
|
|
|
streamInfo.stream_type = extractor.getStreamType();
|
|
|
|
streamInfo.id = uiconv.getId(extractor.getPageUrl());
|
|
|
|
streamInfo.title = extractor.getTitle();
|
|
|
|
streamInfo.age_limit = extractor.getAgeLimit();
|
|
|
|
|
|
|
|
if((streamInfo.stream_type == StreamType.NONE)
|
|
|
|
|| (streamInfo.webpage_url == null || streamInfo.webpage_url.isEmpty())
|
|
|
|
|| (streamInfo.id == null || streamInfo.id.isEmpty())
|
|
|
|
|| (streamInfo.title == null /* streamInfo.title can be empty of course */)
|
|
|
|
|| (streamInfo.age_limit == -1)) {
|
2017-05-21 13:53:54 +00:00
|
|
|
throw new ExtractionException("Some important stream information was not given.");
|
2017-03-01 17:47:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return streamInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static StreamInfo extractStreams(
|
|
|
|
StreamInfo streamInfo, StreamExtractor extractor)
|
2017-05-27 12:23:48 +00:00
|
|
|
throws ExtractionException {
|
2017-03-01 17:47:52 +00:00
|
|
|
/* ---- stream extraction goes here ---- */
|
|
|
|
// At least one type of stream has to be available,
|
|
|
|
// otherwise an exception will be thrown directly into the frontend.
|
|
|
|
|
|
|
|
try {
|
|
|
|
streamInfo.dashMpdUrl = extractor.getDashMpdUrl();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(new ExtractionException("Couldn't get Dash manifest", e));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Load and extract audio */
|
|
|
|
try {
|
|
|
|
streamInfo.audio_streams = extractor.getAudioStreams();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(new ExtractionException("Couldn't get audio streams", e));
|
|
|
|
}
|
|
|
|
// also try to get streams from the dashMpd
|
|
|
|
if(streamInfo.dashMpdUrl != null && !streamInfo.dashMpdUrl.isEmpty()) {
|
|
|
|
if(streamInfo.audio_streams == null) {
|
|
|
|
streamInfo.audio_streams = new Vector<>();
|
|
|
|
}
|
|
|
|
//todo: make this quick and dirty solution a real fallback
|
|
|
|
// same as the quick and dirty above
|
|
|
|
try {
|
|
|
|
streamInfo.audio_streams.addAll(
|
|
|
|
DashMpdParser.getAudioStreams(streamInfo.dashMpdUrl));
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(
|
|
|
|
new ExtractionException("Couldn't get audio streams from dash mpd", e));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Extract video stream url*/
|
|
|
|
try {
|
|
|
|
streamInfo.video_streams = extractor.getVideoStreams();
|
|
|
|
} catch (Exception e) {
|
|
|
|
streamInfo.addException(
|
|
|
|
new ExtractionException("Couldn't get video streams", e));
|
|
|
|
}
|
|
|
|
/* Extract video only stream url*/
|
|
|
|
try {
|
|
|
|
streamInfo.video_only_streams = extractor.getVideoOnlyStreams();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(
|
|
|
|
new ExtractionException("Couldn't get video only streams", e));
|
|
|
|
}
|
|
|
|
|
|
|
|
// either dash_mpd audio_only or video has to be available, otherwise we didn't get a stream,
|
|
|
|
// and therefore failed. (Since video_only_streams are just optional they don't caunt).
|
|
|
|
if((streamInfo.video_streams == null || streamInfo.video_streams.isEmpty())
|
|
|
|
&& (streamInfo.audio_streams == null || streamInfo.audio_streams.isEmpty())
|
|
|
|
&& (streamInfo.dashMpdUrl == null || streamInfo.dashMpdUrl.isEmpty())) {
|
|
|
|
throw new StreamExctractException(
|
|
|
|
"Could not get any stream. See error variable to get further details.");
|
|
|
|
}
|
|
|
|
|
|
|
|
return streamInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static StreamInfo extractOptionalData(
|
|
|
|
StreamInfo streamInfo, StreamExtractor extractor) {
|
|
|
|
/* ---- optional data goes here: ---- */
|
|
|
|
// If one of these fails, the frontend needs to handle that they are not available.
|
|
|
|
// Exceptions are therefore not thrown into the frontend, but stored into the error List,
|
|
|
|
// so the frontend can afterwards check where errors happened.
|
|
|
|
|
|
|
|
try {
|
|
|
|
streamInfo.thumbnail_url = extractor.getThumbnailUrl();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.duration = extractor.getLength();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.uploader = extractor.getUploader();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.channel_url = extractor.getChannelUrl();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.description = extractor.getDescription();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.view_count = extractor.getViewCount();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.upload_date = extractor.getUploadDate();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.uploader_thumbnail_url = extractor.getUploaderThumbnailUrl();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.start_position = extractor.getTimeStamp();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.average_rating = extractor.getAverageRating();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.like_count = extractor.getLikeCount();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
streamInfo.dislike_count = extractor.getDislikeCount();
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
2017-04-11 20:45:00 +00:00
|
|
|
StreamInfoItemCollector c = new StreamInfoItemCollector(
|
|
|
|
extractor.getUrlIdHandler(), extractor.getServiceId());
|
|
|
|
StreamInfoItemExtractor nextVideo = extractor.getNextVideo();
|
|
|
|
c.commit(nextVideo);
|
|
|
|
if(c.getItemList().size() != 0) {
|
|
|
|
streamInfo.next_video = (StreamInfoItem) c.getItemList().get(0);
|
2017-03-01 17:47:52 +00:00
|
|
|
}
|
2017-04-11 20:45:00 +00:00
|
|
|
streamInfo.errors.addAll(c.getErrors());
|
2017-03-01 17:47:52 +00:00
|
|
|
}
|
|
|
|
catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
// get related videos
|
|
|
|
StreamInfoItemCollector c = extractor.getRelatedVideos();
|
|
|
|
streamInfo.related_streams = c.getItemList();
|
|
|
|
streamInfo.errors.addAll(c.getErrors());
|
|
|
|
} catch(Exception e) {
|
|
|
|
streamInfo.addException(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
return streamInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String uploader_thumbnail_url = "";
|
|
|
|
public String channel_url = "";
|
|
|
|
public String description = "";
|
|
|
|
|
|
|
|
public List<VideoStream> video_streams = null;
|
|
|
|
public List<AudioStream> audio_streams = null;
|
|
|
|
public List<VideoStream> video_only_streams = null;
|
|
|
|
// video streams provided by the dash mpd do not need to be provided as VideoStream.
|
|
|
|
// Later on this will also aplly to audio streams. Since dash mpd is standarized,
|
|
|
|
// crawling such a file is not service dependent. Therefore getting audio only streams by yust
|
|
|
|
// providing the dash mpd fille will be possible in the future.
|
|
|
|
public String dashMpdUrl = "";
|
|
|
|
public int duration = -1;
|
|
|
|
|
|
|
|
public int age_limit = -1;
|
|
|
|
public int like_count = -1;
|
|
|
|
public int dislike_count = -1;
|
|
|
|
public String average_rating = "";
|
|
|
|
public StreamInfoItem next_video = null;
|
|
|
|
public List<InfoItem> related_streams = null;
|
|
|
|
//in seconds. some metadata is not passed using a StreamInfo object!
|
|
|
|
public int start_position = 0;
|
|
|
|
|
|
|
|
public List<Throwable> errors = new Vector<>();
|
|
|
|
}
|