Improve documentation and adress most of the requested changes

Also fix some issues in several places, in the code and the documentation.
This commit is contained in:
TiA4f8R 2022-03-15 11:19:13 +01:00
parent 6985167e63
commit aa4c10e751
No known key found for this signature in database
GPG key ID: E6D3E7F5949450DD
20 changed files with 759 additions and 522 deletions

View file

@ -120,9 +120,9 @@ public class BandcampStreamExtractor extends StreamExtractor {
public String getThumbnailUrl() throws ParsingException { public String getThumbnailUrl() throws ParsingException {
if (albumJson.isNull("art_id")) { if (albumJson.isNull("art_id")) {
return EMPTY_STRING; return EMPTY_STRING;
} else {
return getImageUrl(albumJson.getLong("art_id"), true);
} }
return getImageUrl(albumJson.getLong("art_id"), true);
} }
@Nonnull @Nonnull

View file

@ -12,15 +12,16 @@ import org.schabi.newpipe.extractor.linkhandler.LinkHandler;
import org.schabi.newpipe.extractor.stream.AudioStream; import org.schabi.newpipe.extractor.stream.AudioStream;
import org.schabi.newpipe.extractor.stream.DeliveryMethod; import org.schabi.newpipe.extractor.stream.DeliveryMethod;
import org.schabi.newpipe.extractor.stream.Description; import org.schabi.newpipe.extractor.stream.Description;
import org.schabi.newpipe.extractor.stream.Stream;
import org.schabi.newpipe.extractor.stream.StreamExtractor; import org.schabi.newpipe.extractor.stream.StreamExtractor;
import org.schabi.newpipe.extractor.stream.StreamType; import org.schabi.newpipe.extractor.stream.StreamType;
import org.schabi.newpipe.extractor.stream.VideoStream; import org.schabi.newpipe.extractor.stream.VideoStream;
import java.io.IOException; import java.io.IOException;
import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.stream.IntStream; import java.util.function.Function;
import java.util.stream.Collectors;
import javax.annotation.Nonnull; import javax.annotation.Nonnull;
@ -58,9 +59,9 @@ public class MediaCCCLiveStreamExtractor extends StreamExtractor {
final JsonObject roomObject = rooms.getObject(r); final JsonObject roomObject = rooms.getObject(r);
if (getId().equals(conferenceObject.getString("slug") + "/" if (getId().equals(conferenceObject.getString("slug") + "/"
+ roomObject.getString("slug"))) { + roomObject.getString("slug"))) {
this.conference = conferenceObject; conference = conferenceObject;
this.group = groupObject; group = groupObject;
this.room = roomObject; room = roomObject;
return; return;
} }
} }
@ -109,122 +110,120 @@ public class MediaCCCLiveStreamExtractor extends StreamExtractor {
* Get the URL of the first DASH stream found. * Get the URL of the first DASH stream found.
* *
* <p> * <p>
* There can be several DASH streams, so the URL of the first found is returned by this method. * There can be several DASH streams, so the URL of the first one found is returned by this
* method.
* </p> * </p>
* *
* <p> * <p>
* You can find the other video DASH streams by using {@link #getVideoStreams()} * You can find the other DASH video streams by using {@link #getVideoStreams()}
* </p> * </p>
*/ */
@Nonnull @Nonnull
@Override @Override
public String getDashMpdUrl() throws ParsingException { public String getDashMpdUrl() throws ParsingException {
return getManifestOfDeliveryMethodWanted("dash");
for (int s = 0; s < room.getArray(STREAMS).size(); s++) {
final JsonObject stream = room.getArray(STREAMS).getObject(s);
final JsonObject urls = stream.getObject(URLS);
if (urls.has("dash")) {
return urls.getObject("dash").getString(URL, EMPTY_STRING);
}
}
return EMPTY_STRING;
} }
/** /**
* Get the URL of the first HLS stream found. * Get the URL of the first HLS stream found.
* *
* <p> * <p>
* There can be several HLS streams, so the URL of the first found is returned by this method. * There can be several HLS streams, so the URL of the first one found is returned by this
* method.
* </p> * </p>
* *
* <p> * <p>
* You can find the other video HLS streams by using {@link #getVideoStreams()} * You can find the other HLS video streams by using {@link #getVideoStreams()}
* </p> * </p>
*/ */
@Nonnull @Nonnull
@Override @Override
public String getHlsUrl() { public String getHlsUrl() {
for (int s = 0; s < room.getArray(STREAMS).size(); s++) { return getManifestOfDeliveryMethodWanted("hls");
final JsonObject stream = room.getArray(STREAMS).getObject(s); }
final JsonObject urls = stream.getObject(URLS);
if (urls.has("hls")) { @Nonnull
return urls.getObject("hls").getString(URL, EMPTY_STRING); private String getManifestOfDeliveryMethodWanted(@Nonnull final String deliveryMethod) {
} return room.getArray(STREAMS).stream()
} .filter(JsonObject.class::isInstance)
return EMPTY_STRING; .map(JsonObject.class::cast)
.map(streamObject -> streamObject.getObject(URLS))
.filter(urls -> urls.has(deliveryMethod))
.map(urls -> urls.getObject(deliveryMethod).getString(URL, EMPTY_STRING))
.findFirst()
.orElse(EMPTY_STRING);
} }
@Override @Override
public List<AudioStream> getAudioStreams() throws IOException, ExtractionException { public List<AudioStream> getAudioStreams() throws IOException, ExtractionException {
final List<AudioStream> audioStreams = new ArrayList<>(); return getStreams("audio",
IntStream.range(0, room.getArray(STREAMS).size()) dto -> {
.mapToObj(s -> room.getArray(STREAMS).getObject(s)) final AudioStream.Builder builder = new AudioStream.Builder()
.filter(streamJsonObject -> streamJsonObject.getString("type").equals("audio")) .setId(dto.getUrlValue().getString("tech", ID_UNKNOWN))
.forEachOrdered(streamJsonObject -> streamJsonObject.getObject(URLS).keySet() .setContent(dto.getUrlValue().getString(URL), true)
.forEach(type -> { .setAverageBitrate(UNKNOWN_BITRATE);
final JsonObject urlObject = streamJsonObject.getObject(URLS)
.getObject(type);
// The DASH manifest will be extracted with getDashMpdUrl
if (!type.equals("dash")) {
final AudioStream.Builder builder = new AudioStream.Builder()
.setId(urlObject.getString("tech", ID_UNKNOWN))
.setContent(urlObject.getString(URL), true)
.setAverageBitrate(UNKNOWN_BITRATE);
if (type.equals("hls")) {
// We don't know with the type string what media format will
// have HLS streams.
// However, the tech string may contain some information
// about the media format used.
builder.setDeliveryMethod(DeliveryMethod.HLS);
} else {
builder.setMediaFormat(MediaFormat.getFromSuffix(type));
}
audioStreams.add(builder.build()); if ("hls".equals(dto.getUrlKey())) {
} // We don't know with the type string what media format will
})); // have HLS streams.
// However, the tech string may contain some information
// about the media format used.
return builder.setDeliveryMethod(DeliveryMethod.HLS)
.build();
}
return audioStreams; return builder.setMediaFormat(MediaFormat.getFromSuffix(dto.getUrlKey()))
.build();
});
} }
@Override @Override
public List<VideoStream> getVideoStreams() throws IOException, ExtractionException { public List<VideoStream> getVideoStreams() throws IOException, ExtractionException {
final List<VideoStream> videoStreams = new ArrayList<>(); return getStreams("video",
IntStream.range(0, room.getArray(STREAMS).size()) dto -> {
.mapToObj(s -> room.getArray(STREAMS).getObject(s)) final JsonArray videoSize = dto.getStreamJsonObj().getArray("videoSize");
.filter(stream -> stream.getString("type").equals("video"))
.forEachOrdered(streamJsonObject -> streamJsonObject.getObject(URLS).keySet()
.forEach(type -> {
final String resolution =
streamJsonObject.getArray("videoSize").getInt(0)
+ "x"
+ streamJsonObject.getArray("videoSize").getInt(1);
final JsonObject urlObject = streamJsonObject.getObject(URLS)
.getObject(type);
// The DASH manifest will be extracted with getDashMpdUrl
if (!type.equals("dash")) {
final VideoStream.Builder builder = new VideoStream.Builder()
.setId(urlObject.getString("tech", ID_UNKNOWN))
.setContent(urlObject.getString(URL), true)
.setIsVideoOnly(false)
.setResolution(resolution);
if (type.equals("hls")) { final VideoStream.Builder builder = new VideoStream.Builder()
// We don't know with the type string what media format will .setId(dto.getUrlValue().getString("tech", ID_UNKNOWN))
// have HLS streams. .setContent(dto.getUrlValue().getString(URL), true)
// However, the tech string may contain some information .setIsVideoOnly(false)
// about the media format used. .setResolution(videoSize.getInt(0) + "x" + videoSize.getInt(1));
builder.setDeliveryMethod(DeliveryMethod.HLS);
} else {
builder.setMediaFormat(MediaFormat.getFromSuffix(type));
}
videoStreams.add(builder.build()); if ("hls".equals(dto.getUrlKey())) {
} // We don't know with the type string what media format will
})); // have HLS streams.
// However, the tech string may contain some information
// about the media format used.
return builder.setDeliveryMethod(DeliveryMethod.HLS)
.build();
}
return videoStreams; return builder.setMediaFormat(MediaFormat.getFromSuffix(dto.getUrlKey()))
.build();
});
}
private <T extends Stream> List<T> getStreams(
@Nonnull final String streamType,
@Nonnull final Function<MediaCCCLiveStreamMapperDTO, T> converter) {
return room.getArray(STREAMS).stream()
// Ensure that we use only process JsonObjects
.filter(JsonObject.class::isInstance)
.map(JsonObject.class::cast)
// Only process audio streams
.filter(streamJsonObj -> streamType.equals(streamJsonObj.getString("type")))
// Flatmap Urls and ensure that we use only process JsonObjects
.flatMap(streamJsonObj -> streamJsonObj.getObject(URLS).entrySet().stream()
.filter(e -> e.getValue() instanceof JsonObject)
.map(e -> new MediaCCCLiveStreamMapperDTO(
streamJsonObj,
e.getKey(),
(JsonObject) e.getValue())))
// The DASH manifest will be extracted with getDashMpdUrl
.filter(dto -> !"dash".equals(dto.getUrlKey()))
// Convert
.map(converter)
.collect(Collectors.toList());
} }
@Override @Override

View file

@ -0,0 +1,29 @@
package org.schabi.newpipe.extractor.services.media_ccc.extractors;
import com.grack.nanojson.JsonObject;
final class MediaCCCLiveStreamMapperDTO {
private final JsonObject streamJsonObj;
private final String urlKey;
private final JsonObject urlValue;
MediaCCCLiveStreamMapperDTO(final JsonObject streamJsonObj,
final String urlKey,
final JsonObject urlValue) {
this.streamJsonObj = streamJsonObj;
this.urlKey = urlKey;
this.urlValue = urlValue;
}
JsonObject getStreamJsonObj() {
return streamJsonObj;
}
String getUrlKey() {
return urlKey;
}
JsonObject getUrlValue() {
return urlValue;
}
}

View file

@ -102,7 +102,7 @@ public class MediaCCCStreamExtractor extends StreamExtractor {
final JsonObject recording = recordings.getObject(i); final JsonObject recording = recordings.getObject(i);
final String mimeType = recording.getString("mime_type"); final String mimeType = recording.getString("mime_type");
if (mimeType.startsWith("audio")) { if (mimeType.startsWith("audio")) {
// First we need to resolve the actual video data from CDN // First we need to resolve the actual video data from the CDN
final MediaFormat mediaFormat; final MediaFormat mediaFormat;
if (mimeType.endsWith("opus")) { if (mimeType.endsWith("opus")) {
mediaFormat = MediaFormat.OPUS; mediaFormat = MediaFormat.OPUS;
@ -115,7 +115,7 @@ public class MediaCCCStreamExtractor extends StreamExtractor {
} }
// Don't use the containsSimilarStream method because it will always return // Don't use the containsSimilarStream method because it will always return
// false so if there are multiples audio streams available, only the first will // false. So if there are multiple audio streams available, only the first one will
// be extracted in this case. // be extracted in this case.
audioStreams.add(new AudioStream.Builder() audioStreams.add(new AudioStream.Builder()
.setId(recording.getString("filename", ID_UNKNOWN)) .setId(recording.getString("filename", ID_UNKNOWN))
@ -136,7 +136,7 @@ public class MediaCCCStreamExtractor extends StreamExtractor {
final JsonObject recording = recordings.getObject(i); final JsonObject recording = recordings.getObject(i);
final String mimeType = recording.getString("mime_type"); final String mimeType = recording.getString("mime_type");
if (mimeType.startsWith("video")) { if (mimeType.startsWith("video")) {
// First we need to resolve the actual video data from CDN // First we need to resolve the actual video data from the CDN
final MediaFormat mediaFormat; final MediaFormat mediaFormat;
if (mimeType.endsWith("webm")) { if (mimeType.endsWith("webm")) {
@ -148,7 +148,8 @@ public class MediaCCCStreamExtractor extends StreamExtractor {
} }
// Don't use the containsSimilarStream method because it will remove the // Don't use the containsSimilarStream method because it will remove the
// extraction of some video versions (mostly languages) // extraction of some video versions (mostly languages). So if there are multiple
// video streams available, only the first one will be extracted in this case.
videoStreams.add(new VideoStream.Builder() videoStreams.add(new VideoStream.Builder()
.setId(recording.getString("filename", ID_UNKNOWN)) .setId(recording.getString("filename", ID_UNKNOWN))
.setContent(recording.getString("recording_url"), true) .setContent(recording.getString("recording_url"), true)

View file

@ -220,10 +220,10 @@ public class PeertubeStreamExtractor extends StreamExtractor {
if (getStreamType() == StreamType.VIDEO_STREAM if (getStreamType() == StreamType.VIDEO_STREAM
&& !isNullOrEmpty(json.getObject(FILES))) { && !isNullOrEmpty(json.getObject(FILES))) {
return json.getObject(FILES).getString(PLAYLIST_URL, EMPTY_STRING); return json.getObject(FILES).getString(PLAYLIST_URL, EMPTY_STRING);
} else {
return json.getArray(STREAMING_PLAYLISTS).getObject(0).getString(PLAYLIST_URL,
EMPTY_STRING);
} }
return json.getArray(STREAMING_PLAYLISTS).getObject(0).getString(PLAYLIST_URL,
EMPTY_STRING);
} }
@Override @Override
@ -231,7 +231,7 @@ public class PeertubeStreamExtractor extends StreamExtractor {
assertPageFetched(); assertPageFetched();
/* /*
Some videos have audio streams, some videos don't have audio streams. Some videos have audio streams; others don't.
So an audio stream may be available if a video stream is available. So an audio stream may be available if a video stream is available.
Audio streams are also not returned as separated streams for livestreams. Audio streams are also not returned as separated streams for livestreams.
That's why the extraction of audio streams is only run when there are video streams That's why the extraction of audio streams is only run when there are video streams
@ -435,23 +435,21 @@ public class PeertubeStreamExtractor extends StreamExtractor {
private void extractLiveVideoStreams() throws ParsingException { private void extractLiveVideoStreams() throws ParsingException {
try { try {
final JsonArray streamingPlaylists = json.getArray(STREAMING_PLAYLISTS); final JsonArray streamingPlaylists = json.getArray(STREAMING_PLAYLISTS);
for (final Object s : streamingPlaylists) { streamingPlaylists.stream()
if (!(s instanceof JsonObject)) { .filter(JsonObject.class::isInstance)
continue; .map(JsonObject.class::cast)
} .map(stream -> new VideoStream.Builder()
final JsonObject stream = (JsonObject) s; .setId(String.valueOf(stream.getInt("id", -1)))
// Don't use the containsSimilarStream method because it will always return false .setContent(stream.getString(PLAYLIST_URL, EMPTY_STRING), true)
// so if there are multiples HLS URLs returned, only the first will be extracted in .setIsVideoOnly(false)
// this case. .setResolution(EMPTY_STRING)
videoStreams.add(new VideoStream.Builder() .setMediaFormat(MediaFormat.MPEG_4)
.setId(String.valueOf(stream.getInt("id", -1))) .setDeliveryMethod(DeliveryMethod.HLS)
.setContent(stream.getString(PLAYLIST_URL, EMPTY_STRING), true) .build())
.setIsVideoOnly(false) // Don't use the containsSimilarStream method because it will always return
.setResolution(EMPTY_STRING) // false so if there are multiples HLS URLs returned, only the first will be
.setMediaFormat(MediaFormat.MPEG_4) // extracted in this case.
.setDeliveryMethod(DeliveryMethod.HLS) .forEachOrdered(videoStreams::add);
.build());
}
} catch (final Exception e) { } catch (final Exception e) {
throw new ParsingException("Could not get video streams", e); throw new ParsingException("Could not get video streams", e);
} }
@ -463,14 +461,11 @@ public class PeertubeStreamExtractor extends StreamExtractor {
// HLS streams // HLS streams
try { try {
final JsonArray streamingPlaylists = json.getArray(STREAMING_PLAYLISTS); for (final JsonObject playlist : json.getArray(STREAMING_PLAYLISTS).stream()
for (final Object p : streamingPlaylists) { .filter(JsonObject.class::isInstance)
if (!(p instanceof JsonObject)) { .map(JsonObject.class::cast)
continue; .collect(Collectors.toList())) {
} getStreamsFromArray(playlist.getArray(FILES), playlist.getString(PLAYLIST_URL));
final JsonObject playlist = (JsonObject) p;
final String playlistUrl = playlist.getString(PLAYLIST_URL);
getStreamsFromArray(playlist.getArray(FILES), playlistUrl);
} }
} catch (final Exception e) { } catch (final Exception e) {
throw new ParsingException("Could not get streams", e); throw new ParsingException("Could not get streams", e);
@ -481,39 +476,31 @@ public class PeertubeStreamExtractor extends StreamExtractor {
final String playlistUrl) throws ParsingException { final String playlistUrl) throws ParsingException {
try { try {
/* /*
Starting with version 3.4.0 of PeerTube, HLS playlist of stream resolutions contain the Starting with version 3.4.0 of PeerTube, the HLS playlist of stream resolutions
UUID of the stream, so we can't use the same system to get HLS playlist URL of streams contains the UUID of the streams, so we can't use the same method to get the URL of
without fetching the master playlist. the HLS playlist without fetching the master playlist.
These UUIDs are the same that the ones returned into the fileUrl and fileDownloadUrl These UUIDs are the same as the ones returned into the fileUrl and fileDownloadUrl
strings. strings.
*/ */
final boolean isInstanceUsingRandomUuidsForHlsStreams = !isNullOrEmpty(playlistUrl) final boolean isInstanceUsingRandomUuidsForHlsStreams = !isNullOrEmpty(playlistUrl)
&& playlistUrl.endsWith("-master.m3u8"); && playlistUrl.endsWith("-master.m3u8");
for (final Object s : streams) { for (final JsonObject stream : streams.stream()
if (!(s instanceof JsonObject)) { .filter(JsonObject.class::isInstance)
continue; .map(JsonObject.class::cast)
} .collect(Collectors.toList())) {
final JsonObject stream = (JsonObject) s;
final String resolution = JsonUtils.getString(stream, "resolution.label");
final String url;
final String idSuffix;
// Extract stream version of streams first // Extract stream version of streams first
if (stream.has(FILE_URL)) { final String url = JsonUtils.getString(stream,
url = JsonUtils.getString(stream, FILE_URL); stream.has(FILE_URL) ? FILE_URL : FILE_DOWNLOAD_URL);
idSuffix = FILE_URL;
} else {
url = JsonUtils.getString(stream, FILE_DOWNLOAD_URL);
idSuffix = FILE_DOWNLOAD_URL;
}
if (isNullOrEmpty(url)) { if (isNullOrEmpty(url)) {
// Not a valid stream URL // Not a valid stream URL
return; return;
} }
final String resolution = JsonUtils.getString(stream, "resolution.label");
final String idSuffix = stream.has(FILE_URL) ? FILE_URL : FILE_DOWNLOAD_URL;
if (resolution.toLowerCase().contains("audio")) { if (resolution.toLowerCase().contains("audio")) {
// An audio stream // An audio stream
addNewAudioStream(stream, isInstanceUsingRandomUuidsForHlsStreams, resolution, addNewAudioStream(stream, isInstanceUsingRandomUuidsForHlsStreams, resolution,
@ -535,12 +522,9 @@ public class PeertubeStreamExtractor extends StreamExtractor {
@Nonnull final String idSuffix, @Nonnull final String idSuffix,
@Nonnull final String format, @Nonnull final String format,
@Nonnull final String url) throws ParsingException { @Nonnull final String url) throws ParsingException {
final String streamUrl; final String streamUrl = FILE_DOWNLOAD_URL.equals(idSuffix)
if (FILE_DOWNLOAD_URL.equals(idSuffix)) { ? JsonUtils.getString(streamJsonObject, FILE_URL)
streamUrl = JsonUtils.getString(streamJsonObject, FILE_URL); : url;
} else {
streamUrl = url;
}
return streamUrl.replace("-fragmented." + format, ".m3u8"); return streamUrl.replace("-fragmented." + format, ".m3u8");
} }
@ -593,7 +577,7 @@ public class PeertubeStreamExtractor extends StreamExtractor {
} }
} }
// Add finally torrent URLs // Finally, add torrent URLs
final String torrentUrl = JsonUtils.getString(streamJsonObject, "torrentUrl"); final String torrentUrl = JsonUtils.getString(streamJsonObject, "torrentUrl");
if (!isNullOrEmpty(torrentUrl)) { if (!isNullOrEmpty(torrentUrl)) {
audioStreams.add(new AudioStream.Builder() audioStreams.add(new AudioStream.Builder()
@ -627,14 +611,10 @@ public class PeertubeStreamExtractor extends StreamExtractor {
// Then add HLS streams // Then add HLS streams
if (!isNullOrEmpty(playlistUrl)) { if (!isNullOrEmpty(playlistUrl)) {
final String hlsStreamUrl; final String hlsStreamUrl = isInstanceUsingRandomUuidsForHlsStreams
if (isInstanceUsingRandomUuidsForHlsStreams) { ? getHlsPlaylistUrlFromFragmentedFileUrl(streamJsonObject, idSuffix, extension,
hlsStreamUrl = getHlsPlaylistUrlFromFragmentedFileUrl(streamJsonObject, idSuffix, url)
extension, url); : getHlsPlaylistUrlFromMasterPlaylist(streamJsonObject, playlistUrl);
} else {
hlsStreamUrl = playlistUrl.replace("master", JsonUtils.getNumber(
streamJsonObject, RESOLUTION_ID).toString());
}
final VideoStream videoStream = new VideoStream.Builder() final VideoStream videoStream = new VideoStream.Builder()
.setId(id + "-" + DeliveryMethod.HLS) .setId(id + "-" + DeliveryMethod.HLS)

View file

@ -233,11 +233,10 @@ public class SoundcloudStreamExtractor extends StreamExtractor {
@Nullable @Nullable
private String getDownloadUrl(@Nonnull final String trackId) private String getDownloadUrl(@Nonnull final String trackId)
throws IOException, ExtractionException { throws IOException, ExtractionException {
final Downloader dl = NewPipe.getDownloader(); final String response = NewPipe.getDownloader().get(SOUNDCLOUD_API_V2_URL + "tracks/"
final JsonObject downloadJsonObject; + trackId + "/download" + "?client_id=" + clientId()).responseBody();
final String response = dl.get(SOUNDCLOUD_API_V2_URL + "tracks/" + trackId final JsonObject downloadJsonObject;
+ "/download" + "?client_id=" + clientId()).responseBody();
try { try {
downloadJsonObject = JsonParser.object().from(response); downloadJsonObject = JsonParser.object().from(response);
} catch (final JsonParserException e) { } catch (final JsonParserException e) {
@ -293,7 +292,7 @@ public class SoundcloudStreamExtractor extends StreamExtractor {
} }
} }
} catch (final Exception ignored) { } catch (final Exception ignored) {
// Something went wrong when parsing this transcoding, don't add it to the // Something went wrong when parsing this transcoding URL, so don't add it to the
// audioStreams // audioStreams
} }
} }
@ -304,11 +303,15 @@ public class SoundcloudStreamExtractor extends StreamExtractor {
* *
* <p> * <p>
* A track can have the {@code downloadable} boolean set to {@code true}, but it doesn't mean * A track can have the {@code downloadable} boolean set to {@code true}, but it doesn't mean
* we can download it: if the value of the {@code has_download_left} boolean is true, the track * we can download it.
* can be downloaded; otherwise not.
* </p> * </p>
* *
* @param audioStreams the audio streams to which add the downloadable file * <p>
* If the value of the {@code has_download_left} boolean is {@code true}, the track can be
* downloaded, and not otherwise.
* </p>
*
* @param audioStreams the audio streams to which the downloadable file is added
*/ */
public void extractDownloadableFileIfAvailable(final List<AudioStream> audioStreams) { public void extractDownloadableFileIfAvailable(final List<AudioStream> audioStreams) {
if (track.getBoolean("downloadable") && track.getBoolean("has_downloads_left")) { if (track.getBoolean("downloadable") && track.getBoolean("has_downloads_left")) {
@ -332,9 +335,9 @@ public class SoundcloudStreamExtractor extends StreamExtractor {
* Parses a SoundCloud HLS manifest to get a single URL of HLS streams. * Parses a SoundCloud HLS manifest to get a single URL of HLS streams.
* *
* <p> * <p>
* This method downloads the provided manifest URL, find all web occurrences in the manifest, * This method downloads the provided manifest URL, finds all web occurrences in the manifest,
* get the last segment URL, changes its segment range to {@code 0/track-length} and return * gets the last segment URL, changes its segment range to {@code 0/track-length}, and return
* this string. * this as a string.
* </p> * </p>
* *
* @param hlsManifestUrl the URL of the manifest to be parsed * @param hlsManifestUrl the URL of the manifest to be parsed

View file

@ -27,7 +27,6 @@ import org.schabi.newpipe.extractor.exceptions.ParsingException;
import org.schabi.newpipe.extractor.exceptions.ReCaptchaException; import org.schabi.newpipe.extractor.exceptions.ReCaptchaException;
import org.schabi.newpipe.extractor.stream.AudioStream; import org.schabi.newpipe.extractor.stream.AudioStream;
import org.schabi.newpipe.extractor.stream.DeliveryMethod; import org.schabi.newpipe.extractor.stream.DeliveryMethod;
import org.schabi.newpipe.extractor.stream.StreamInfo;
import org.schabi.newpipe.extractor.stream.VideoStream; import org.schabi.newpipe.extractor.stream.VideoStream;
import org.w3c.dom.Document; import org.w3c.dom.Document;
import org.w3c.dom.Element; import org.w3c.dom.Element;
@ -52,10 +51,26 @@ import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource; import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult; import javax.xml.transform.stream.StreamResult;
/**
* Class to extract streams from a DASH manifest.
*
* <p>
* Note that this class relies on the YouTube's {@link ItagItem} class and should be made generic
* in order to be used on other services.
* </p>
*
* <p>
* This class is not used by the extractor itself, as all streams are supported by the extractor.
* </p>
*/
public final class DashMpdParser { public final class DashMpdParser {
private DashMpdParser() { private DashMpdParser() {
} }
/**
* Exception class which is thrown when something went wrong when using
* {@link DashMpdParser#getStreams(String)}.
*/
public static class DashMpdParsingException extends ParsingException { public static class DashMpdParsingException extends ParsingException {
DashMpdParsingException(final String message, final Exception e) { DashMpdParsingException(final String message, final Exception e) {
@ -63,15 +78,21 @@ public final class DashMpdParser {
} }
} }
/**
* Class which represents the result of a DASH MPD file parsing by {@link DashMpdParser}.
*
* <p>
* The result contains video, video-only and audio streams.
* </p>
*/
public static class Result { public static class Result {
private final List<VideoStream> videoStreams; private final List<VideoStream> videoStreams;
private final List<VideoStream> videoOnlyStreams; private final List<VideoStream> videoOnlyStreams;
private final List<AudioStream> audioStreams; private final List<AudioStream> audioStreams;
Result(final List<VideoStream> videoStreams,
public Result(final List<VideoStream> videoStreams, final List<VideoStream> videoOnlyStreams,
final List<VideoStream> videoOnlyStreams, final List<AudioStream> audioStreams) {
final List<AudioStream> audioStreams) {
this.videoStreams = videoStreams; this.videoStreams = videoStreams;
this.videoOnlyStreams = videoOnlyStreams; this.videoOnlyStreams = videoOnlyStreams;
this.audioStreams = audioStreams; this.audioStreams = audioStreams;
@ -90,19 +111,22 @@ public final class DashMpdParser {
} }
} }
// TODO: Make this class generic and decouple from YouTube's ItagItem class.
/** /**
* Will try to download and parse the DASH manifest (using {@link StreamInfo#getDashMpdUrl()}), * This method will try to download and parse the YouTube DASH MPD manifest URL provided to get
* adding items that are listed in the {@link ItagItem} class. * supported {@link AudioStream}s and {@link VideoStream}s.
* <p>
* It has video, video only and audio streams.
* <p>
* Info about DASH MPD can be found here
* *
* @param dashMpdUrl URL to the DASH MPD * <p>
* The parser supports video, video-only and audio streams.
* </p>
*
* @param dashMpdUrl the URL of the DASH MPD manifest
* @return a {@link Result} which contains all video, video-only and audio streams extracted
* and supported by the extractor (so the ones for which {@link ItagItem#isSupported(int)}
* returns {@code true}).
* @throws DashMpdParsingException if something went wrong when downloading or parsing the
* manifest
* @see <a href="https://www.brendanlong.com/the-structure-of-an-mpeg-dash-mpd.html"> * @see <a href="https://www.brendanlong.com/the-structure-of-an-mpeg-dash-mpd.html">
* www.brendanlog.com</a> * www.brendanlong.com's page about the structure of an MPEG-DASH MPD manifest</a>
*/ */
@Nonnull @Nonnull
public static Result getStreams(final String dashMpdUrl) public static Result getStreams(final String dashMpdUrl)
@ -188,7 +212,7 @@ public final class DashMpdParser {
throws TransformerException { throws TransformerException {
final Element mpdElement = (Element) document.getElementsByTagName("MPD").item(0); final Element mpdElement = (Element) document.getElementsByTagName("MPD").item(0);
// Clone element so we can freely modify it // Clone the element so we can freely modify it
final Element adaptationSet = (Element) representation.getParentNode(); final Element adaptationSet = (Element) representation.getParentNode();
final Element adaptationSetClone = (Element) adaptationSet.cloneNode(true); final Element adaptationSetClone = (Element) adaptationSet.cloneNode(true);

View file

@ -1,37 +0,0 @@
package org.schabi.newpipe.extractor.services.youtube;
import javax.annotation.Nonnull;
import java.io.Serializable;
public final class ItagInfo implements Serializable {
@Nonnull
private final String content;
@Nonnull
private final ItagItem itagItem;
private boolean isUrl;
public ItagInfo(@Nonnull final String content,
@Nonnull final ItagItem itagItem) {
this.content = content;
this.itagItem = itagItem;
}
public void setIsUrl(final boolean isUrl) {
this.isUrl = isUrl;
}
@Nonnull
public String getContent() {
return content;
}
@Nonnull
public ItagItem getItagItem() {
return itagItem;
}
public boolean getIsUrl() {
return isUrl;
}
}

View file

@ -237,11 +237,15 @@ public class ItagItem implements Serializable {
} }
/** /**
* Get the frame rate per second. * Get the frame rate.
* *
* <p> * <p>
* It defaults to the standard value associated with this itag and is set to the {@code fps} * It is set to the {@code fps} value returned in the corresponding itag in the YouTube player
* value returned in the corresponding itag in the YouTube player response. * response.
* </p>
*
* <p>
* It defaults to the standard value associated with this itag.
* </p> * </p>
* *
* <p> * <p>
@ -249,28 +253,24 @@ public class ItagItem implements Serializable {
* #FPS_NOT_APPLICABLE_OR_UNKNOWN} is returned for non video itags. * #FPS_NOT_APPLICABLE_OR_UNKNOWN} is returned for non video itags.
* </p> * </p>
* *
* @return the frame rate per second or {@link #FPS_NOT_APPLICABLE_OR_UNKNOWN} * @return the frame rate or {@link #FPS_NOT_APPLICABLE_OR_UNKNOWN}
*/ */
public int getFps() { public int getFps() {
return fps; return fps;
} }
/** /**
* Set the frame rate per second. * Set the frame rate.
* *
* <p> * <p>
* It is only known for video itags, so {@link #FPS_NOT_APPLICABLE_OR_UNKNOWN} is set/used for * It is only known for video itags, so {@link #FPS_NOT_APPLICABLE_OR_UNKNOWN} is set/used for
* non video itags or if the sample rate value is less than or equal to 0. * non video itags or if the sample rate value is less than or equal to 0.
* </p> * </p>
* *
* @param fps the frame rate per second * @param fps the frame rate
*/ */
public void setFps(final int fps) { public void setFps(final int fps) {
if (fps > 0) { this.fps = fps > 0 ? fps : FPS_NOT_APPLICABLE_OR_UNKNOWN;
this.fps = fps;
} else {
this.fps = FPS_NOT_APPLICABLE_OR_UNKNOWN;
}
} }
public int getInitStart() { public int getInitStart() {
@ -314,13 +314,13 @@ public class ItagItem implements Serializable {
} }
/** /**
* Get the resolution string associated to this {@code ItagItem}. * Get the resolution string associated with this {@code ItagItem}.
* *
* <p> * <p>
* It is only known for video itags. * It is only known for video itags.
* </p> * </p>
* *
* @return the resolution string associated to this {@code ItagItem} or * @return the resolution string associated with this {@code ItagItem} or
* {@code null}. * {@code null}.
*/ */
@Nullable @Nullable
@ -361,7 +361,7 @@ public class ItagItem implements Serializable {
* *
* <p> * <p>
* It is only known for audio itags, so {@link #SAMPLE_RATE_UNKNOWN} is returned for non audio * It is only known for audio itags, so {@link #SAMPLE_RATE_UNKNOWN} is returned for non audio
* itags or if the sample rate is unknown. * itags, or if the sample rate is unknown.
* </p> * </p>
* *
* @return the sample rate or {@link #SAMPLE_RATE_UNKNOWN} * @return the sample rate or {@link #SAMPLE_RATE_UNKNOWN}
@ -374,8 +374,8 @@ public class ItagItem implements Serializable {
* Set the sample rate. * Set the sample rate.
* *
* <p> * <p>
* It is only known for audio itags, so {@link #SAMPLE_RATE_UNKNOWN} is set/used for non video * It is only known for audio itags, so {@link #SAMPLE_RATE_UNKNOWN} is set/used for non audio
* itags or if the sample rate value is less than or equal to 0. * itags, or if the sample rate value is less than or equal to 0.
* </p> * </p>
* *
* @param sampleRate the sample rate of an audio itag * @param sampleRate the sample rate of an audio itag
@ -392,8 +392,8 @@ public class ItagItem implements Serializable {
* Get the number of audio channels. * Get the number of audio channels.
* *
* <p> * <p>
* It is only known for audio streams, so {@link #AUDIO_CHANNELS_NOT_APPLICABLE_OR_UNKNOWN} is * It is only known for audio itags, so {@link #AUDIO_CHANNELS_NOT_APPLICABLE_OR_UNKNOWN} is
* returned for video streams or if it is unknown. * returned for non audio itags, or if it is unknown.
* </p> * </p>
* *
* @return the number of audio channels or {@link #AUDIO_CHANNELS_NOT_APPLICABLE_OR_UNKNOWN} * @return the number of audio channels or {@link #AUDIO_CHANNELS_NOT_APPLICABLE_OR_UNKNOWN}
@ -406,28 +406,26 @@ public class ItagItem implements Serializable {
* Set the number of audio channels. * Set the number of audio channels.
* *
* <p> * <p>
* It is only known for audio itag, so {@link #AUDIO_CHANNELS_NOT_APPLICABLE_OR_UNKNOWN} is * It is only known for audio itags, so {@link #AUDIO_CHANNELS_NOT_APPLICABLE_OR_UNKNOWN} is
* set/used for non audio itags or if the {@code audioChannels} value is less than or equal to * set/used for non audio itags, or if the {@code audioChannels} value is less than or equal to
* 0. * 0.
* </p> * </p>
* *
* @param audioChannels the number of audio channels of an audio itag * @param audioChannels the number of audio channels of an audio itag
*/ */
public void setAudioChannels(final int audioChannels) { public void setAudioChannels(final int audioChannels) {
if (audioChannels > 0) { this.audioChannels = audioChannels > 0
this.audioChannels = audioChannels; ? audioChannels
} else { : AUDIO_CHANNELS_NOT_APPLICABLE_OR_UNKNOWN;
this.audioChannels = AUDIO_CHANNELS_NOT_APPLICABLE_OR_UNKNOWN;
}
} }
/** /**
* Get the {@code targetDurationSec} value. * Get the {@code targetDurationSec} value.
* *
* <p> * <p>
* This value is an average time in seconds of sequences duration of livestreams and ended * This value is the average time in seconds of the duration of sequences of livestreams and
* livestreams. It is only returned for these stream types by YouTube and makes no sense for * ended livestreams. It is only returned by YouTube for these stream types, and makes no sense
* videos, so {@link #TARGET_DURATION_SEC_UNKNOWN} is returned for video streams. * for videos, so {@link #TARGET_DURATION_SEC_UNKNOWN} is returned for those.
* </p> * </p>
* *
* @return the {@code targetDurationSec} value or {@link #TARGET_DURATION_SEC_UNKNOWN} * @return the {@code targetDurationSec} value or {@link #TARGET_DURATION_SEC_UNKNOWN}
@ -440,25 +438,23 @@ public class ItagItem implements Serializable {
* Set the {@code targetDurationSec} value. * Set the {@code targetDurationSec} value.
* *
* <p> * <p>
* This value is an average time in seconds of sequences duration of livestreams and ended * This value is the average time in seconds of the duration of sequences of livestreams and
* livestreams. * ended livestreams.
* </p> * </p>
* *
* <p> * <p>
* It is only returned for these stream types by YouTube and makes no sense for * It is only returned for these stream types by YouTube and makes no sense for videos, so
* videos, so {@link #TARGET_DURATION_SEC_UNKNOWN} will be set/used for video streams or if * {@link #TARGET_DURATION_SEC_UNKNOWN} will be set/used for video streams or if this value is
* this value is less than or equal to 0. * less than or equal to 0.
* </p> * </p>
* *
* @param targetDurationSec the target duration of a segment of streams which are using the * @param targetDurationSec the target duration of a segment of streams which are using the
* live delivery method type * live delivery method type
*/ */
public void setTargetDurationSec(final int targetDurationSec) { public void setTargetDurationSec(final int targetDurationSec) {
if (targetDurationSec > 0) { this.targetDurationSec = targetDurationSec > 0
this.targetDurationSec = targetDurationSec; ? targetDurationSec
} else { : TARGET_DURATION_SEC_UNKNOWN;
this.targetDurationSec = TARGET_DURATION_SEC_UNKNOWN;
}
} }
/** /**
@ -487,11 +483,9 @@ public class ItagItem implements Serializable {
* milliseconds * milliseconds
*/ */
public void setApproxDurationMs(final long approxDurationMs) { public void setApproxDurationMs(final long approxDurationMs) {
if (approxDurationMs > 0) { this.approxDurationMs = approxDurationMs > 0
this.approxDurationMs = approxDurationMs; ? approxDurationMs
} else { : APPROX_DURATION_MS_UNKNOWN;
this.approxDurationMs = APPROX_DURATION_MS_UNKNOWN;
}
} }
/** /**
@ -519,10 +513,6 @@ public class ItagItem implements Serializable {
* @param contentLength the content length of a DASH progressive stream * @param contentLength the content length of a DASH progressive stream
*/ */
public void setContentLength(final long contentLength) { public void setContentLength(final long contentLength) {
if (contentLength > 0) { this.contentLength = contentLength > 0 ? contentLength : CONTENT_LENGTH_UNKNOWN;
this.contentLength = contentLength;
} else {
this.contentLength = CONTENT_LENGTH_UNKNOWN;
}
} }
} }

View file

@ -35,7 +35,6 @@ import static org.schabi.newpipe.extractor.utils.Utils.*;
* It relies on external classes from the {@link org.w3c.dom} and {@link javax.xml} packages. * It relies on external classes from the {@link org.w3c.dom} and {@link javax.xml} packages.
* </p> * </p>
*/ */
@SuppressWarnings({"ConstantConditions", "unused"})
public final class YoutubeDashManifestCreator { public final class YoutubeDashManifestCreator {
/** /**
@ -115,6 +114,7 @@ public final class YoutubeDashManifestCreator {
* </p> * </p>
*/ */
PROGRESSIVE, PROGRESSIVE,
/** /**
* YouTube's OTF delivery method which uses a sequence parameter to get segments of * YouTube's OTF delivery method which uses a sequence parameter to get segments of
* streams. * streams.
@ -124,12 +124,14 @@ public final class YoutubeDashManifestCreator {
* metadata needed to build the stream source (sidx boxes, segment length, segment count, * metadata needed to build the stream source (sidx boxes, segment length, segment count,
* duration, ...) * duration, ...)
* </p> * </p>
*
* <p> * <p>
* Only used for videos; mostly those with a small amount of views, or ended livestreams * Only used for videos; mostly those with a small amount of views, or ended livestreams
* which have just been re-encoded as normal videos. * which have just been re-encoded as normal videos.
* </p> * </p>
*/ */
OTF, OTF,
/** /**
* YouTube's delivery method for livestreams which uses a sequence parameter to get * YouTube's delivery method for livestreams which uses a sequence parameter to get
* segments of streams. * segments of streams.
@ -139,6 +141,7 @@ public final class YoutubeDashManifestCreator {
* metadata (sidx boxes, segment length, ...), which make no need of an initialization * metadata (sidx boxes, segment length, ...), which make no need of an initialization
* segment. * segment.
* </p> * </p>
*
* <p> * <p>
* Only used for livestreams (ended or running). * Only used for livestreams (ended or running).
* </p> * </p>
@ -225,27 +228,27 @@ public final class YoutubeDashManifestCreator {
*/ */
@Nonnull @Nonnull
public static String createDashManifestFromOtfStreamingUrl( public static String createDashManifestFromOtfStreamingUrl(
@Nonnull String otfBaseStreamingUrl, @Nonnull final String otfBaseStreamingUrl,
@Nonnull final ItagItem itagItem, @Nonnull final ItagItem itagItem,
final long durationSecondsFallback) final long durationSecondsFallback) throws YoutubeDashManifestCreationException {
throws YoutubeDashManifestCreationException {
if (GENERATED_OTF_MANIFESTS.containsKey(otfBaseStreamingUrl)) { if (GENERATED_OTF_MANIFESTS.containsKey(otfBaseStreamingUrl)) {
return GENERATED_OTF_MANIFESTS.get(otfBaseStreamingUrl).getSecond(); return Objects.requireNonNull(GENERATED_OTF_MANIFESTS.get(otfBaseStreamingUrl))
.getSecond();
} }
final String originalOtfBaseStreamingUrl = otfBaseStreamingUrl; String realOtfBaseStreamingUrl = otfBaseStreamingUrl;
// Try to avoid redirects when streaming the content by saving the last URL we get // Try to avoid redirects when streaming the content by saving the last URL we get
// from video servers. // from video servers.
final Response response = getInitializationResponse(otfBaseStreamingUrl, final Response response = getInitializationResponse(realOtfBaseStreamingUrl,
itagItem, DeliveryType.OTF); itagItem, DeliveryType.OTF);
otfBaseStreamingUrl = response.latestUrl().replace(SQ_0, EMPTY_STRING) realOtfBaseStreamingUrl = response.latestUrl().replace(SQ_0, EMPTY_STRING)
.replace(RN_0, EMPTY_STRING).replace(ALR_YES, EMPTY_STRING); .replace(RN_0, EMPTY_STRING).replace(ALR_YES, EMPTY_STRING);
final int responseCode = response.responseCode(); final int responseCode = response.responseCode();
if (responseCode != 200) { if (responseCode != 200) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Unable to create the DASH manifest: could not get the initialization URL of the OTF stream: response code " "Unable to create the DASH manifest: could not get the initialization URL of "
+ responseCode); + "the OTF stream: response code " + responseCode);
} }
final String[] segmentDuration; final String[] segmentDuration;
@ -266,7 +269,8 @@ public final class YoutubeDashManifestCreator {
} }
} catch (final Exception e) { } catch (final Exception e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Unable to generate the DASH manifest: could not get the duration of segments", e); "Unable to generate the DASH manifest: could not get the duration of segments",
e);
} }
final Document document = generateDocumentAndMpdElement(segmentDuration, DeliveryType.OTF, final Document document = generateDocumentAndMpdElement(segmentDuration, DeliveryType.OTF,
@ -278,7 +282,7 @@ public final class YoutubeDashManifestCreator {
if (itagItem.itagType == ItagItem.ItagType.AUDIO) { if (itagItem.itagType == ItagItem.ItagType.AUDIO) {
generateAudioChannelConfigurationElement(document, itagItem); generateAudioChannelConfigurationElement(document, itagItem);
} }
generateSegmentTemplateElement(document, otfBaseStreamingUrl, DeliveryType.OTF); generateSegmentTemplateElement(document, realOtfBaseStreamingUrl, DeliveryType.OTF);
generateSegmentTimelineElement(document); generateSegmentTimelineElement(document);
collectSegmentsData(segmentDuration); collectSegmentsData(segmentDuration);
generateSegmentElementsForOtfStreams(document); generateSegmentElementsForOtfStreams(document);
@ -286,7 +290,7 @@ public final class YoutubeDashManifestCreator {
SEGMENTS_DURATION.clear(); SEGMENTS_DURATION.clear();
DURATION_REPETITIONS.clear(); DURATION_REPETITIONS.clear();
return buildResult(originalOtfBaseStreamingUrl, document, GENERATED_OTF_MANIFESTS); return buildResult(otfBaseStreamingUrl, document, GENERATED_OTF_MANIFESTS);
} }
/** /**
@ -358,36 +362,37 @@ public final class YoutubeDashManifestCreator {
*/ */
@Nonnull @Nonnull
public static String createDashManifestFromPostLiveStreamDvrStreamingUrl( public static String createDashManifestFromPostLiveStreamDvrStreamingUrl(
@Nonnull String postLiveStreamDvrStreamingUrl, @Nonnull final String postLiveStreamDvrStreamingUrl,
@Nonnull final ItagItem itagItem, @Nonnull final ItagItem itagItem,
final int targetDurationSec, final int targetDurationSec,
final long durationSecondsFallback) final long durationSecondsFallback) throws YoutubeDashManifestCreationException {
throws YoutubeDashManifestCreationException {
if (GENERATED_POST_LIVE_DVR_STREAMS_MANIFESTS.containsKey(postLiveStreamDvrStreamingUrl)) { if (GENERATED_POST_LIVE_DVR_STREAMS_MANIFESTS.containsKey(postLiveStreamDvrStreamingUrl)) {
return GENERATED_POST_LIVE_DVR_STREAMS_MANIFESTS.get(postLiveStreamDvrStreamingUrl) return Objects.requireNonNull(GENERATED_POST_LIVE_DVR_STREAMS_MANIFESTS.get(
.getSecond(); postLiveStreamDvrStreamingUrl)).getSecond();
} }
final String originalPostLiveStreamDvrStreamingUrl = postLiveStreamDvrStreamingUrl; String realPostLiveStreamDvrStreamingUrl = postLiveStreamDvrStreamingUrl;
final String streamDuration; final String streamDuration;
final String segmentCount; final String segmentCount;
if (targetDurationSec <= 0) { if (targetDurationSec <= 0) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: the targetDurationSec value is less than or equal to 0 (" + targetDurationSec + ")"); "Could not generate the DASH manifest: the targetDurationSec value is less "
+ "than or equal to 0 (" + targetDurationSec + ")");
} }
try { try {
// Try to avoid redirects when streaming the content by saving the latest URL we get // Try to avoid redirects when streaming the content by saving the latest URL we get
// from video servers. // from video servers.
final Response response = getInitializationResponse(postLiveStreamDvrStreamingUrl, final Response response = getInitializationResponse(realPostLiveStreamDvrStreamingUrl,
itagItem, DeliveryType.LIVE); itagItem, DeliveryType.LIVE);
postLiveStreamDvrStreamingUrl = response.latestUrl().replace(SQ_0, EMPTY_STRING) realPostLiveStreamDvrStreamingUrl = response.latestUrl().replace(SQ_0, EMPTY_STRING)
.replace(RN_0, EMPTY_STRING).replace(ALR_YES, EMPTY_STRING); .replace(RN_0, EMPTY_STRING).replace(ALR_YES, EMPTY_STRING);
final int responseCode = response.responseCode(); final int responseCode = response.responseCode();
if (responseCode != 200) { if (responseCode != 200) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: could not get the initialization URL of the post-live-DVR stream: response code " "Could not generate the DASH manifest: could not get the initialization "
+ "segment of the post-live-DVR stream: response code "
+ responseCode); + responseCode);
} }
@ -396,15 +401,18 @@ public final class YoutubeDashManifestCreator {
segmentCount = responseHeaders.get("X-Head-Seqnum").get(0); segmentCount = responseHeaders.get("X-Head-Seqnum").get(0);
} catch (final IndexOutOfBoundsException e) { } catch (final IndexOutOfBoundsException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: could not get the value of the X-Head-Time-Millis or the X-Head-Seqnum header of the post-live-DVR streaming URL", e); "Could not generate the DASH manifest: could not get the value of the "
+ "X-Head-Time-Millis or the X-Head-Seqnum header of the post-live-DVR"
+ "streaming URL", e);
} }
if (isNullOrEmpty(segmentCount)) { if (isNullOrEmpty(segmentCount)) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: could not get the number of segments of the post-live-DVR stream"); "Could not generate the DASH manifest: could not get the number of segments of"
+ "the post-live-DVR stream");
} }
final Document document = generateDocumentAndMpdElement(new String[]{streamDuration}, final Document document = generateDocumentAndMpdElement(new String[] {streamDuration},
DeliveryType.LIVE, itagItem, durationSecondsFallback); DeliveryType.LIVE, itagItem, durationSecondsFallback);
generatePeriodElement(document); generatePeriodElement(document);
generateAdaptationSetElement(document, itagItem); generateAdaptationSetElement(document, itagItem);
@ -413,11 +421,12 @@ public final class YoutubeDashManifestCreator {
if (itagItem.itagType == ItagItem.ItagType.AUDIO) { if (itagItem.itagType == ItagItem.ItagType.AUDIO) {
generateAudioChannelConfigurationElement(document, itagItem); generateAudioChannelConfigurationElement(document, itagItem);
} }
generateSegmentTemplateElement(document, postLiveStreamDvrStreamingUrl, DeliveryType.LIVE); generateSegmentTemplateElement(document, realPostLiveStreamDvrStreamingUrl,
DeliveryType.LIVE);
generateSegmentTimelineElement(document); generateSegmentTimelineElement(document);
generateSegmentElementForPostLiveDvrStreams(document, targetDurationSec, segmentCount); generateSegmentElementForPostLiveDvrStreams(document, targetDurationSec, segmentCount);
return buildResult(originalPostLiveStreamDvrStreamingUrl, document, return buildResult(postLiveStreamDvrStreamingUrl, document,
GENERATED_POST_LIVE_DVR_STREAMS_MANIFESTS); GENERATED_POST_LIVE_DVR_STREAMS_MANIFESTS);
} }
@ -486,13 +495,14 @@ public final class YoutubeDashManifestCreator {
@Nonnull final ItagItem itagItem, @Nonnull final ItagItem itagItem,
final long durationSecondsFallback) throws YoutubeDashManifestCreationException { final long durationSecondsFallback) throws YoutubeDashManifestCreationException {
if (GENERATED_PROGRESSIVE_STREAMS_MANIFESTS.containsKey(progressiveStreamingBaseUrl)) { if (GENERATED_PROGRESSIVE_STREAMS_MANIFESTS.containsKey(progressiveStreamingBaseUrl)) {
return GENERATED_PROGRESSIVE_STREAMS_MANIFESTS.get(progressiveStreamingBaseUrl) return Objects.requireNonNull(GENERATED_PROGRESSIVE_STREAMS_MANIFESTS.get(
.getSecond(); progressiveStreamingBaseUrl)).getSecond();
} }
if (durationSecondsFallback <= 0) { if (durationSecondsFallback <= 0) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: the durationSecondsFallback value is less than or equal to 0 (" + durationSecondsFallback + ")"); "Could not generate the DASH manifest: the durationSecondsFallback value is"
+ "less than or equal to 0 (" + durationSecondsFallback + ")");
} }
final Document document = generateDocumentAndMpdElement(new String[]{}, final Document document = generateDocumentAndMpdElement(new String[]{},
@ -508,7 +518,8 @@ public final class YoutubeDashManifestCreator {
generateSegmentBaseElement(document, itagItem); generateSegmentBaseElement(document, itagItem);
generateInitializationElement(document, itagItem); generateInitializationElement(document, itagItem);
return buildResult(progressiveStreamingBaseUrl, document, GENERATED_PROGRESSIVE_STREAMS_MANIFESTS); return buildResult(progressiveStreamingBaseUrl, document,
GENERATED_PROGRESSIVE_STREAMS_MANIFESTS);
} }
/** /**
@ -564,7 +575,8 @@ public final class YoutubeDashManifestCreator {
return downloader.post(baseStreamingUrl, headers, emptyBody); return downloader.post(baseStreamingUrl, headers, emptyBody);
} catch (final IOException | ExtractionException e) { } catch (final IOException | ExtractionException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: error when trying to get the ANDROID streaming post-live-DVR URL response", e); "Could not generate the DASH manifest: error when trying to get the "
+ "ANDROID streaming post-live-DVR URL response", e);
} }
} }
@ -579,10 +591,12 @@ public final class YoutubeDashManifestCreator {
} catch (final IOException | ExtractionException e) { } catch (final IOException | ExtractionException e) {
if (isAnAndroidStreamingUrl) { if (isAnAndroidStreamingUrl) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: error when trying to get the ANDROID streaming URL response", e); "Could not generate the DASH manifest: error when trying to get the "
+ "ANDROID streaming URL response", e);
} else { } else {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: error when trying to get the streaming URL response", e); "Could not generate the DASH manifest: error when trying to get the "
+ "streaming URL response", e);
} }
} }
} }
@ -658,16 +672,18 @@ public final class YoutubeDashManifestCreator {
if (responseCode != 200) { if (responseCode != 200) {
if (deliveryType == DeliveryType.LIVE) { if (deliveryType == DeliveryType.LIVE) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: could not get the initialization URL of the post-live-DVR stream: response code " "Could not generate the DASH manifest: could not get the "
+ responseCode); + "initialization URL of the post-live-DVR stream: "
+ "response code " + responseCode);
} else if (deliveryType == DeliveryType.OTF) { } else if (deliveryType == DeliveryType.OTF) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: could not get the initialization URL of the OTF stream: response code " "Could not generate the DASH manifest: could not get the "
+ "initialization URL of the OTF stream: response code "
+ responseCode); + responseCode);
} else { } else {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: could not fetch the URL of the progressive stream: response code " "Could not generate the DASH manifest: could not fetch the URL of "
+ responseCode); + "the progressive stream: response code " + responseCode);
} }
} }
@ -678,7 +694,8 @@ public final class YoutubeDashManifestCreator {
"Content-Type")); "Content-Type"));
} catch (final NullPointerException e) { } catch (final NullPointerException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: could not get the Content-Type header from the streaming URL", e); "Could not generate the DASH manifest: could not get the Content-Type "
+ "header from the streaming URL", e);
} }
// The response body is the redirection URL // The response body is the redirection URL
@ -692,16 +709,19 @@ public final class YoutubeDashManifestCreator {
if (redirectsCount >= MAXIMUM_REDIRECT_COUNT) { if (redirectsCount >= MAXIMUM_REDIRECT_COUNT) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: too many redirects when trying to get the WEB streaming URL response"); "Could not generate the DASH manifest: too many redirects when trying to "
+ "get the WEB streaming URL response");
} }
// This should never be reached, but is required because we don't want to return null // This should never be reached, but is required because we don't want to return null
// here // here
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: error when trying to get the WEB streaming URL response"); "Could not generate the DASH manifest: error when trying to get the WEB "
+ "streaming URL response");
} catch (final IOException | ExtractionException e) { } catch (final IOException | ExtractionException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: error when trying to get the WEB streaming URL response", e); "Could not generate the DASH manifest: error when trying to get the WEB "
+ "streaming URL response", e);
} }
} }
@ -731,7 +751,8 @@ public final class YoutubeDashManifestCreator {
} }
} catch (final NumberFormatException e) { } catch (final NumberFormatException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: unable to get the segments of the stream", e); "Could not generate the DASH manifest: unable to get the segments of the "
+ "stream", e);
} }
} }
@ -767,7 +788,8 @@ public final class YoutubeDashManifestCreator {
return streamLengthMs; return streamLengthMs;
} catch (final NumberFormatException e) { } catch (final NumberFormatException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: unable to get the length of the stream", e); "Could not generate the DASH manifest: unable to get the length of the stream",
e);
} }
} }
@ -778,6 +800,7 @@ public final class YoutubeDashManifestCreator {
* The generated {@code <MPD>} element looks like the manifest returned into the player * The generated {@code <MPD>} element looks like the manifest returned into the player
* response of videos with OTF streams: * response of videos with OTF streams:
* </p> * </p>
*
* <p> * <p>
* {@code <MPD xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" * {@code <MPD xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
* xmlns="urn:mpeg:DASH:schema:MPD:2011" * xmlns="urn:mpeg:DASH:schema:MPD:2011"
@ -787,6 +810,7 @@ public final class YoutubeDashManifestCreator {
* (where {@code $duration$} represents the duration in seconds (a number with 3 digits after * (where {@code $duration$} represents the duration in seconds (a number with 3 digits after
* the decimal point) * the decimal point)
* </p> * </p>
*
* <p> * <p>
* If the duration is an integer or a double with less than 3 digits after the decimal point, * If the duration is an integer or a double with less than 3 digits after the decimal point,
* it will be converted into a double with 3 digits after the decimal point. * it will be converted into a double with 3 digits after the decimal point.
@ -859,8 +883,10 @@ public final class YoutubeDashManifestCreator {
streamDuration = durationSecondsFallback * 1000; streamDuration = durationSecondsFallback * 1000;
} else { } else {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the MPD element of the DASH manifest to the document: " "Could not generate or append the MPD element of the DASH "
+ "the duration of the stream could not be determined and the durationSecondsFallback is less than or equal to 0"); + "manifest to the document: the duration of the stream "
+ "could not be determined and the "
+ "durationSecondsFallback is less than or equal to 0");
} }
} }
} }
@ -870,7 +896,8 @@ public final class YoutubeDashManifestCreator {
mpdElement.setAttributeNode(mediaPresentationDurationAttribute); mpdElement.setAttributeNode(mediaPresentationDurationAttribute);
} catch (final Exception e) { } catch (final Exception e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the MPD element of the DASH manifest to the document", e); "Could not generate or append the MPD element of the DASH manifest to the "
+ "document", e);
} }
return document; return document;
@ -898,7 +925,8 @@ public final class YoutubeDashManifestCreator {
mpdElement.appendChild(periodElement); mpdElement.appendChild(periodElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the Period element of the DASH manifest to the document", e); "Could not generate or append the Period element of the DASH manifest to the "
+ "document", e);
} }
} }
@ -921,7 +949,8 @@ public final class YoutubeDashManifestCreator {
@Nonnull final ItagItem itagItem) @Nonnull final ItagItem itagItem)
throws YoutubeDashManifestCreationException { throws YoutubeDashManifestCreationException {
try { try {
final Element periodElement = (Element) document.getElementsByTagName("Period").item(0); final Element periodElement = (Element) document.getElementsByTagName("Period")
.item(0);
final Element adaptationSetElement = document.createElement("AdaptationSet"); final Element adaptationSetElement = document.createElement("AdaptationSet");
final Attr idAttribute = document.createAttribute("id"); final Attr idAttribute = document.createAttribute("id");
@ -931,21 +960,25 @@ public final class YoutubeDashManifestCreator {
final MediaFormat mediaFormat = itagItem.getMediaFormat(); final MediaFormat mediaFormat = itagItem.getMediaFormat();
if (mediaFormat == null || isNullOrEmpty(mediaFormat.mimeType)) { if (mediaFormat == null || isNullOrEmpty(mediaFormat.mimeType)) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the AdaptationSet element of the DASH manifest to the document: the MediaFormat or the mime type of the MediaFormat of the ItagItem is null or empty"); "Could not generate the AdaptationSet element of the DASH manifest to the "
+ "document: the MediaFormat or the mime type of the MediaFormat "
+ "of the ItagItem is null or empty");
} }
final Attr mimeTypeAttribute = document.createAttribute("mimeType"); final Attr mimeTypeAttribute = document.createAttribute("mimeType");
mimeTypeAttribute.setValue(mediaFormat.mimeType); mimeTypeAttribute.setValue(mediaFormat.mimeType);
adaptationSetElement.setAttributeNode(mimeTypeAttribute); adaptationSetElement.setAttributeNode(mimeTypeAttribute);
final Attr subsegmentAlignmentAttribute = document.createAttribute("subsegmentAlignment"); final Attr subsegmentAlignmentAttribute = document.createAttribute(
"subsegmentAlignment");
subsegmentAlignmentAttribute.setValue("true"); subsegmentAlignmentAttribute.setValue("true");
adaptationSetElement.setAttributeNode(subsegmentAlignmentAttribute); adaptationSetElement.setAttributeNode(subsegmentAlignmentAttribute);
periodElement.appendChild(adaptationSetElement); periodElement.appendChild(adaptationSetElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the AdaptationSet element of the DASH manifest to the document", e); "Could not generate or append the AdaptationSet element of the DASH manifest "
+ "to the document", e);
} }
} }
@ -956,9 +989,11 @@ public final class YoutubeDashManifestCreator {
* <p> * <p>
* This element, with its attributes and values, is: * This element, with its attributes and values, is:
* </p> * </p>
*
* <p> * <p>
* {@code <Role schemeIdUri="urn:mpeg:DASH:role:2011" value="main"/>} * {@code <Role schemeIdUri="urn:mpeg:DASH:role:2011" value="main"/>}
* </p> * </p>
*
* <p> * <p>
* The {@code <AdaptationSet>} element needs to be generated before this element with * The {@code <AdaptationSet>} element needs to be generated before this element with
* {@link #generateAdaptationSetElement(Document, ItagItem)}). * {@link #generateAdaptationSetElement(Document, ItagItem)}).
@ -967,7 +1002,8 @@ public final class YoutubeDashManifestCreator {
* @param document the {@link Document} on which the the {@code <Role>} element will be * @param document the {@link Document} on which the the {@code <Role>} element will be
* appended * appended
* @throws YoutubeDashManifestCreationException if something goes wrong when generating or * @throws YoutubeDashManifestCreationException if something goes wrong when generating or
* appending the {@code <Role>} element to the document * appending the {@code <Role>} element to the
* document
*/ */
private static void generateRoleElement(@Nonnull final Document document) private static void generateRoleElement(@Nonnull final Document document)
throws YoutubeDashManifestCreationException { throws YoutubeDashManifestCreationException {
@ -987,7 +1023,8 @@ public final class YoutubeDashManifestCreator {
adaptationSetElement.appendChild(roleElement); adaptationSetElement.appendChild(roleElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the Role element of the DASH manifest to the document", e); "Could not generate or append the Role element of the DASH manifest to the "
+ "document", e);
} }
} }
@ -1018,7 +1055,9 @@ public final class YoutubeDashManifestCreator {
final int id = itagItem.id; final int id = itagItem.id;
if (id <= 0) { if (id <= 0) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the Representation element of the DASH manifest to the document: the id of the ItagItem is less than or equal to 0"); "Could not generate the Representation element of the DASH manifest to "
+ "the document: the id of the ItagItem is less than or equal to "
+ "0");
} }
final Attr idAttribute = document.createAttribute("id"); final Attr idAttribute = document.createAttribute("id");
idAttribute.setValue(String.valueOf(id)); idAttribute.setValue(String.valueOf(id));
@ -1027,7 +1066,8 @@ public final class YoutubeDashManifestCreator {
final String codec = itagItem.getCodec(); final String codec = itagItem.getCodec();
if (isNullOrEmpty(codec)) { if (isNullOrEmpty(codec)) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the AdaptationSet element of the DASH manifest to the document: the codecs value is null or empty"); "Could not generate the AdaptationSet element of the DASH manifest to the "
+ "document: the codec value is null or empty");
} }
final Attr codecsAttribute = document.createAttribute("codecs"); final Attr codecsAttribute = document.createAttribute("codecs");
codecsAttribute.setValue(codec); codecsAttribute.setValue(codec);
@ -1044,7 +1084,9 @@ public final class YoutubeDashManifestCreator {
final int bitrate = itagItem.getBitrate(); final int bitrate = itagItem.getBitrate();
if (bitrate <= 0) { if (bitrate <= 0) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the Representation element of the DASH manifest to the document: the bitrate of the ItagItem is less than or equal to 0"); "Could not generate the Representation element of the DASH manifest to "
+ "the document: the bitrate of the ItagItem is less than or "
+ "equal to 0");
} }
final Attr bandwidthAttribute = document.createAttribute("bandwidth"); final Attr bandwidthAttribute = document.createAttribute("bandwidth");
bandwidthAttribute.setValue(String.valueOf(bitrate)); bandwidthAttribute.setValue(String.valueOf(bitrate));
@ -1057,7 +1099,9 @@ public final class YoutubeDashManifestCreator {
final int width = itagItem.getWidth(); final int width = itagItem.getWidth();
if (height <= 0 && width <= 0) { if (height <= 0 && width <= 0) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the Representation element of the DASH manifest to the document: the width and the height of the ItagItem are less than or equal to 0"); "Could not generate the Representation element of the DASH manifest "
+ "to the document: the width and the height of the ItagItem "
+ "are less than or equal to 0");
} }
if (width > 0) { if (width > 0) {
@ -1087,7 +1131,8 @@ public final class YoutubeDashManifestCreator {
adaptationSetElement.appendChild(representationElement); adaptationSetElement.appendChild(representationElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the Representation element of the DASH manifest to the document", e); "Could not generate or append the Representation element of the DASH manifest "
+ "to the document", e);
} }
} }
@ -1098,6 +1143,7 @@ public final class YoutubeDashManifestCreator {
* <p> * <p>
* This method is only used when generating DASH manifests of audio streams. * This method is only used when generating DASH manifests of audio streams.
* </p> * </p>
*
* <p> * <p>
* It will produce the following element: * It will produce the following element:
* <br> * <br>
@ -1108,6 +1154,7 @@ public final class YoutubeDashManifestCreator {
* (where {@code audioChannelsValue} is get from the {@link ItagItem} passed as the second * (where {@code audioChannelsValue} is get from the {@link ItagItem} passed as the second
* parameter of this method) * parameter of this method)
* </p> * </p>
*
* <p> * <p>
* The {@code <Representation>} element needs to be generated before this element with * The {@code <Representation>} element needs to be generated before this element with
* {@link #generateRepresentationElement(Document, ItagItem)}). * {@link #generateRepresentationElement(Document, ItagItem)}).
@ -1139,7 +1186,8 @@ public final class YoutubeDashManifestCreator {
final int audioChannels = itagItem.getAudioChannels(); final int audioChannels = itagItem.getAudioChannels();
if (audioChannels <= 0) { if (audioChannels <= 0) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: the audioChannels value is less than or equal to 0 (" + audioChannels + ")"); "Could not generate the DASH manifest: the audioChannels value is less "
+ "than or equal to 0 (" + audioChannels + ")");
} }
valueAttribute.setValue(String.valueOf(itagItem.getAudioChannels())); valueAttribute.setValue(String.valueOf(itagItem.getAudioChannels()));
audioChannelConfigurationElement.setAttributeNode(valueAttribute); audioChannelConfigurationElement.setAttributeNode(valueAttribute);
@ -1147,7 +1195,8 @@ public final class YoutubeDashManifestCreator {
representationElement.appendChild(audioChannelConfigurationElement); representationElement.appendChild(audioChannelConfigurationElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the AudioChannelConfiguration element of the DASH manifest to the document", e); "Could not generate or append the AudioChannelConfiguration element of the "
+ "DASH manifest to the document", e);
} }
} }
@ -1158,6 +1207,7 @@ public final class YoutubeDashManifestCreator {
* <p> * <p>
* This method is only used when generating DASH manifests from progressive streams. * This method is only used when generating DASH manifests from progressive streams.
* </p> * </p>
*
* <p> * <p>
* The {@code <Representation>} element needs to be generated before this element with * The {@code <Representation>} element needs to be generated before this element with
* {@link #generateRepresentationElement(Document, ItagItem)}). * {@link #generateRepresentationElement(Document, ItagItem)}).
@ -1182,7 +1232,8 @@ public final class YoutubeDashManifestCreator {
representationElement.appendChild(baseURLElement); representationElement.appendChild(baseURLElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the BaseURL element of the DASH manifest to the document", e); "Could not generate or append the BaseURL element of the DASH manifest to the "
+ "document", e);
} }
} }
@ -1193,6 +1244,7 @@ public final class YoutubeDashManifestCreator {
* <p> * <p>
* This method is only used when generating DASH manifests from progressive streams. * This method is only used when generating DASH manifests from progressive streams.
* </p> * </p>
*
* <p> * <p>
* It generates the following element: * It generates the following element:
* <br> * <br>
@ -1201,6 +1253,7 @@ public final class YoutubeDashManifestCreator {
* (where {@code indexStart} and {@code indexEnd} are gotten from the {@link ItagItem} passed * (where {@code indexStart} and {@code indexEnd} are gotten from the {@link ItagItem} passed
* as the second parameter) * as the second parameter)
* </p> * </p>
*
* <p> * <p>
* The {@code <Representation>} element needs to be generated before this element with * The {@code <Representation>} element needs to be generated before this element with
* {@link #generateRepresentationElement(Document, ItagItem)}). * {@link #generateRepresentationElement(Document, ItagItem)}).
@ -1227,12 +1280,14 @@ public final class YoutubeDashManifestCreator {
final int indexStart = itagItem.getIndexStart(); final int indexStart = itagItem.getIndexStart();
if (indexStart < 0) { if (indexStart < 0) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: the indexStart value of the ItagItem is less than to 0 (" + indexStart + ")"); "Could not generate the DASH manifest: the indexStart value of the "
+ "ItagItem is less than to 0 (" + indexStart + ")");
} }
final int indexEnd = itagItem.getIndexEnd(); final int indexEnd = itagItem.getIndexEnd();
if (indexEnd < 0) { if (indexEnd < 0) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: the indexEnd value of the ItagItem is less than to 0 (" + indexStart + ")"); "Could not generate the DASH manifest: the indexEnd value of the ItagItem "
+ "is less than to 0 (" + indexStart + ")");
} }
indexRangeAttribute.setValue(indexStart + "-" + indexEnd); indexRangeAttribute.setValue(indexStart + "-" + indexEnd);
@ -1241,7 +1296,8 @@ public final class YoutubeDashManifestCreator {
representationElement.appendChild(segmentBaseElement); representationElement.appendChild(segmentBaseElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the SegmentBase element of the DASH manifest to the document", e); "Could not generate or append the SegmentBase element of the DASH manifest to "
+ "the document", e);
} }
} }
@ -1252,6 +1308,7 @@ public final class YoutubeDashManifestCreator {
* <p> * <p>
* This method is only used when generating DASH manifests from progressive streams. * This method is only used when generating DASH manifests from progressive streams.
* </p> * </p>
*
* <p> * <p>
* It generates the following element: * It generates the following element:
* <br> * <br>
@ -1260,6 +1317,7 @@ public final class YoutubeDashManifestCreator {
* (where {@code indexStart} and {@code indexEnd} are gotten from the {@link ItagItem} passed * (where {@code indexStart} and {@code indexEnd} are gotten from the {@link ItagItem} passed
* as the second parameter) * as the second parameter)
* </p> * </p>
*
* <p> * <p>
* The {@code <SegmentBase>} element needs to be generated before this element with * The {@code <SegmentBase>} element needs to be generated before this element with
* {@link #generateSegmentBaseElement(Document, ItagItem)}). * {@link #generateSegmentBaseElement(Document, ItagItem)}).
@ -1286,12 +1344,14 @@ public final class YoutubeDashManifestCreator {
final int initStart = itagItem.getInitStart(); final int initStart = itagItem.getInitStart();
if (initStart < 0) { if (initStart < 0) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: the initStart value of the ItagItem is less than to 0 (" + initStart + ")"); "Could not generate the DASH manifest: the initStart value of the "
+ "ItagItem is less than to 0 (" + initStart + ")");
} }
final int initEnd = itagItem.getInitEnd(); final int initEnd = itagItem.getInitEnd();
if (initEnd < 0) { if (initEnd < 0) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate the DASH manifest: the initEnd value of the ItagItem is less than to 0 (" + initEnd + ")"); "Could not generate the DASH manifest: the initEnd value of the ItagItem "
+ "is less than to 0 (" + initEnd + ")");
} }
rangeAttribute.setValue(initStart + "-" + initEnd); rangeAttribute.setValue(initStart + "-" + initEnd);
@ -1300,7 +1360,8 @@ public final class YoutubeDashManifestCreator {
segmentBaseElement.appendChild(initializationElement); segmentBaseElement.appendChild(initializationElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the Initialization element of the DASH manifest to the document", e); "Could not generate or append the Initialization element of the DASH manifest "
+ "to the document", e);
} }
} }
@ -1311,6 +1372,7 @@ public final class YoutubeDashManifestCreator {
* <p> * <p>
* This method is only used when generating DASH manifests from OTF and post-live-DVR streams. * This method is only used when generating DASH manifests from OTF and post-live-DVR streams.
* </p> * </p>
*
* <p> * <p>
* It will produce a {@code <SegmentTemplate>} element with the following attributes: * It will produce a {@code <SegmentTemplate>} element with the following attributes:
* <ul> * <ul>
@ -1372,7 +1434,8 @@ public final class YoutubeDashManifestCreator {
representationElement.appendChild(segmentTemplateElement); representationElement.appendChild(segmentTemplateElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the SegmentTemplate element of the DASH manifest to the document", e); "Could not generate or append the SegmentTemplate element of the DASH "
+ "manifest to the document", e);
} }
} }
@ -1401,7 +1464,8 @@ public final class YoutubeDashManifestCreator {
segmentTemplateElement.appendChild(segmentTimelineElement); segmentTemplateElement.appendChild(segmentTimelineElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the SegmentTimeline element of the DASH manifest to the document", e); "Could not generate or append the SegmentTimeline element of the DASH "
+ "manifest to the document", e);
} }
} }
@ -1413,16 +1477,20 @@ public final class YoutubeDashManifestCreator {
* so we just have to loop into {@link #SEGMENTS_DURATION} and {@link #DURATION_REPETITIONS} * so we just have to loop into {@link #SEGMENTS_DURATION} and {@link #DURATION_REPETITIONS}
* to generate the following element for each duration: * to generate the following element for each duration:
* </p> * </p>
*
* <p> * <p>
* {@code <S d="segmentDuration" r="durationRepetition" />} * {@code <S d="segmentDuration" r="durationRepetition" />}
* </p> * </p>
*
* <p> * <p>
* If there is no repetition of the duration between two segments, the {@code r} attribute is * If there is no repetition of the duration between two segments, the {@code r} attribute is
* not added to the {@code S} element. * not added to the {@code S} element.
* </p> * </p>
*
* <p> * <p>
* These elements will be appended as children of the {@code <SegmentTimeline>} element. * These elements will be appended as children of the {@code <SegmentTimeline>} element.
* </p> * </p>
*
* <p> * <p>
* The {@code <SegmentTimeline>} element needs to be generated before this element with * The {@code <SegmentTimeline>} element needs to be generated before this element with
* {@link #generateSegmentTimelineElement(Document)}. * {@link #generateSegmentTimelineElement(Document)}.
@ -1462,7 +1530,8 @@ public final class YoutubeDashManifestCreator {
} catch (final DOMException | IllegalStateException | IndexOutOfBoundsException e) { } catch (final DOMException | IllegalStateException | IndexOutOfBoundsException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the segment (S) elements of the DASH manifest to the document", e); "Could not generate or append the segment (S) elements of the DASH manifest "
+ "to the document", e);
} }
} }
@ -1507,7 +1576,8 @@ public final class YoutubeDashManifestCreator {
segmentTimelineElement.appendChild(sElement); segmentTimelineElement.appendChild(sElement);
} catch (final DOMException e) { } catch (final DOMException e) {
throw new YoutubeDashManifestCreationException( throw new YoutubeDashManifestCreationException(
"Could not generate or append the segment (S) elements of the DASH manifest to the document", e); "Could not generate or append the segment (S) elements of the DASH manifest "
+ "to the document", e);
} }
} }

View file

@ -1596,9 +1596,9 @@ public final class YoutubeParsingHelper {
} }
/** /**
* Check if the streaming URL is a URL from the YouTube {@code WEB} client. * Check if the streaming URL is from the YouTube {@code WEB} client.
* *
* @param url the streaming URL on which check if it's a {@code WEB} streaming URL. * @param url the streaming URL to be checked.
* @return true if it's a {@code WEB} streaming URL, false otherwise * @return true if it's a {@code WEB} streaming URL, false otherwise
*/ */
public static boolean isWebStreamingUrl(@Nonnull final String url) { public static boolean isWebStreamingUrl(@Nonnull final String url) {
@ -1620,7 +1620,7 @@ public final class YoutubeParsingHelper {
/** /**
* Check if the streaming URL is a URL from the YouTube {@code ANDROID} client. * Check if the streaming URL is a URL from the YouTube {@code ANDROID} client.
* *
* @param url the streaming URL on which check if it's a {@code ANDROID} streaming URL. * @param url the streaming URL to be checked.
* @return true if it's a {@code ANDROID} streaming URL, false otherwise * @return true if it's a {@code ANDROID} streaming URL, false otherwise
*/ */
public static boolean isAndroidStreamingUrl(@Nonnull final String url) { public static boolean isAndroidStreamingUrl(@Nonnull final String url) {

View file

@ -0,0 +1,80 @@
package org.schabi.newpipe.extractor.services.youtube.extractors;
import org.schabi.newpipe.extractor.services.youtube.ItagItem;
import javax.annotation.Nonnull;
import java.io.Serializable;
/**
* Class to build easier {@link org.schabi.newpipe.extractor.stream.Stream}s for
* {@link YoutubeStreamExtractor}.
*
* <p>
* It stores, per stream:
* <ul>
* <li>its content (the URL/the base URL of streams);</li>
* <li>whether its content is the URL the content itself or the base URL;</li>
* <li>its associated {@link ItagItem}.</li>
* </ul>
* </p>
*/
final class ItagInfo implements Serializable {
@Nonnull
private final String content;
@Nonnull
private final ItagItem itagItem;
private boolean isUrl;
/**
* Creates a new {@code ItagInfo} instance.
*
* @param content the content of the stream, which must be not null
* @param itagItem the {@link ItagItem} associated with the stream, which must be not null
*/
ItagInfo(@Nonnull final String content,
@Nonnull final ItagItem itagItem) {
this.content = content;
this.itagItem = itagItem;
}
/**
* Sets whether the stream is a URL.
*
* @param isUrl whether the content is a URL
*/
void setIsUrl(final boolean isUrl) {
this.isUrl = isUrl;
}
/**
* Gets the content stored in this {@code ItagInfo} instance, which is either the URL to the
* content itself or the base URL.
*
* @return the content stored in this {@code ItagInfo} instance
*/
@Nonnull
String getContent() {
return content;
}
/**
* Gets the {@link ItagItem} associated with this {@code ItagInfo} instance.
*
* @return the {@link ItagItem} associated with this {@code ItagInfo} instance, which is not
* null
*/
@Nonnull
ItagItem getItagItem() {
return itagItem;
}
/**
* Gets whether the content stored is the URL to the content itself or the base URL of it.
*
* @return whether the content stored is the URL to the content itself or the base URL of it
* @see #getContent() for more details
*/
boolean getIsUrl() {
return isUrl;
}
}

View file

@ -25,6 +25,7 @@ import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.RACY_CHECK_OK; import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.RACY_CHECK_OK;
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.VIDEO_ID; import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.VIDEO_ID;
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.createDesktopPlayerBody; import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.createDesktopPlayerBody;
import static org.schabi.newpipe.extractor.services.youtube.ItagItem.CONTENT_LENGTH_UNKNOWN;
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.fixThumbnailUrl; import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.fixThumbnailUrl;
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.generateContentPlaybackNonce; import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.generateContentPlaybackNonce;
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.generateTParameter; import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.generateTParameter;
@ -66,7 +67,6 @@ import org.schabi.newpipe.extractor.localization.DateWrapper;
import org.schabi.newpipe.extractor.localization.Localization; import org.schabi.newpipe.extractor.localization.Localization;
import org.schabi.newpipe.extractor.localization.TimeAgoParser; import org.schabi.newpipe.extractor.localization.TimeAgoParser;
import org.schabi.newpipe.extractor.localization.TimeAgoPatternsManager; import org.schabi.newpipe.extractor.localization.TimeAgoPatternsManager;
import org.schabi.newpipe.extractor.services.youtube.ItagInfo;
import org.schabi.newpipe.extractor.services.youtube.ItagItem; import org.schabi.newpipe.extractor.services.youtube.ItagItem;
import org.schabi.newpipe.extractor.services.youtube.YoutubeJavaScriptExtractor; import org.schabi.newpipe.extractor.services.youtube.YoutubeJavaScriptExtractor;
import org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper; import org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper;
@ -666,9 +666,10 @@ public class YoutubeStreamExtractor extends StreamExtractor {
} }
private void setStreamType() { private void setStreamType() {
if (playerResponse.getObject("playabilityStatus").has("liveStreamability") if (playerResponse.getObject("playabilityStatus").has("liveStreamability")) {
|| playerResponse.getObject("videoDetails").getBoolean("isPostLiveDvr", false)) {
streamType = StreamType.LIVE_STREAM; streamType = StreamType.LIVE_STREAM;
} else if (playerResponse.getObject("videoDetails").getBoolean("isPostLiveDvr", false)) {
streamType = StreamType.POST_LIVE_STREAM;
} else { } else {
streamType = StreamType.VIDEO_STREAM; streamType = StreamType.VIDEO_STREAM;
} }
@ -1171,7 +1172,7 @@ public class YoutubeStreamExtractor extends StreamExtractor {
for (final Pair<JsonObject, String> pair : streamingDataAndCpnLoopList) { for (final Pair<JsonObject, String> pair : streamingDataAndCpnLoopList) {
itagInfos.addAll(getStreamsFromStreamingDataKey(pair.getFirst(), streamingDataKey, itagInfos.addAll(getStreamsFromStreamingDataKey(pair.getFirst(), streamingDataKey,
itagTypeWanted, streamType, pair.getSecond())); itagTypeWanted, pair.getSecond()));
} }
final List<T> streamList = new ArrayList<>(); final List<T> streamList = new ArrayList<>();
@ -1189,6 +1190,32 @@ public class YoutubeStreamExtractor extends StreamExtractor {
} }
} }
/**
* Get the {@link StreamBuilderHelper} which will be used to build {@link AudioStream}s in
* {@link #getItags(String, ItagItem.ItagType, StreamBuilderHelper, String)}
*
* <p>
* The {@code StreamBuilderHelper} will set the following attributes in the
* {@link AudioStream}s built:
* <ul>
* <li>the {@link ItagItem}'s id of the stream as its id;</li>
* <li>{@link ItagInfo#getContent()} and {@link ItagInfo#getIsUrl()} as its content and
* and as the value of {@code isUrl};</li>
* <li>the media format returned by the {@link ItagItem} as its media format;</li>
* <li>its average bitrate with the value returned by {@link
* ItagItem#getAverageBitrate()};</li>
* <li>the {@link ItagItem};</li>
* <li>the {@link DeliveryMethod#DASH DASH delivery method}, for OTF streams, live streams
* and ended streams.</li>
* </ul>
* </p>
*
* <p>
* Note that the {@link ItagItem} comes from an {@link ItagInfo} instance.
* </p>
*
* @return a {@link StreamBuilderHelper} to build {@link AudioStream}s
*/
@Nonnull @Nonnull
private StreamBuilderHelper<AudioStream> getAudioStreamBuilderHelper() { private StreamBuilderHelper<AudioStream> getAudioStreamBuilderHelper() {
return new StreamBuilderHelper<AudioStream>() { return new StreamBuilderHelper<AudioStream>() {
@ -1203,9 +1230,11 @@ public class YoutubeStreamExtractor extends StreamExtractor {
.setAverageBitrate(itagItem.getAverageBitrate()) .setAverageBitrate(itagItem.getAverageBitrate())
.setItagItem(itagItem); .setItagItem(itagItem);
if (streamType != StreamType.VIDEO_STREAM || !itagInfo.getIsUrl()) { if (streamType == StreamType.LIVE_STREAM
// YouTube uses the DASH delivery method for videos on OTF streams and || streamType == StreamType.POST_LIVE_STREAM
// for all streams of post-live streams and live streams || !itagInfo.getIsUrl()) {
// For YouTube videos on OTF streams and for all streams of post-live streams
// and live streams, only the DASH delivery method can be used.
builder.setDeliveryMethod(DeliveryMethod.DASH); builder.setDeliveryMethod(DeliveryMethod.DASH);
} }
@ -1214,6 +1243,40 @@ public class YoutubeStreamExtractor extends StreamExtractor {
}; };
} }
/**
* Get the {@link StreamBuilderHelper} which will be used to build {@link VideoStream}s in
* {@link #getItags(String, ItagItem.ItagType, StreamBuilderHelper, String)}
*
* <p>
* The {@code StreamBuilderHelper} will set the following attributes in the
* {@link VideoStream}s built:
* <ul>
* <li>the {@link ItagItem}'s id of the stream as its id;</li>
* <li>{@link ItagInfo#getContent()} and {@link ItagInfo#getIsUrl()} as its content and
* and as the value of {@code isUrl};</li>
* <li>the media format returned by the {@link ItagItem} as its media format;</li>
* <li>whether it is video-only with the {@code areStreamsVideoOnly} parameter</li>
* <li>the {@link ItagItem};</li>
* <li>the resolution, by trying to use, in this order:
* <ol>
* <li>the height returned by the {@link ItagItem} + {@code p} + the frame rate if
* it is more than 30;</li>
* <li>the default resolution string from the {@link ItagItem};</li>
* <li>an {@link Utils#EMPTY_STRING empty string}.</li>
* </ol>
* </li>
* <li>the {@link DeliveryMethod#DASH DASH delivery method}, for OTF streams, live streams
* and ended streams.</li>
* </ul>
*
* <p>
* Note that the {@link ItagItem} comes from an {@link ItagInfo} instance.
* </p>
*
* @param areStreamsVideoOnly whether the {@link StreamBuilderHelper} will set the video
* streams as video-only streams
* @return a {@link StreamBuilderHelper} to build {@link VideoStream}s
*/
@Nonnull @Nonnull
private StreamBuilderHelper<VideoStream> getVideoStreamBuilderHelper( private StreamBuilderHelper<VideoStream> getVideoStreamBuilderHelper(
final boolean areStreamsVideoOnly) { final boolean areStreamsVideoOnly) {
@ -1241,12 +1304,13 @@ public class YoutubeStreamExtractor extends StreamExtractor {
builder.setResolution(stringBuilder.toString()); builder.setResolution(stringBuilder.toString());
} else { } else {
final String resolutionString = itagItem.getResolutionString(); final String resolutionString = itagItem.getResolutionString();
builder.setResolution(resolutionString != null ? resolutionString : ""); builder.setResolution(resolutionString != null ? resolutionString
: EMPTY_STRING);
} }
if (streamType != StreamType.VIDEO_STREAM || !itagInfo.getIsUrl()) { if (streamType != StreamType.VIDEO_STREAM || !itagInfo.getIsUrl()) {
// YouTube uses the DASH delivery method for videos on OTF streams and // For YouTube videos on OTF streams and for all streams of post-live streams
// for all streams of post-live streams and live streams // and live streams, only the DASH delivery method can be used.
builder.setDeliveryMethod(DeliveryMethod.DASH); builder.setDeliveryMethod(DeliveryMethod.DASH);
} }
@ -1260,15 +1324,15 @@ public class YoutubeStreamExtractor extends StreamExtractor {
final JsonObject streamingData, final JsonObject streamingData,
final String streamingDataKey, final String streamingDataKey,
@Nonnull final ItagItem.ItagType itagTypeWanted, @Nonnull final ItagItem.ItagType itagTypeWanted,
@Nonnull final StreamType contentStreamType, @Nonnull final String contentPlaybackNonce) throws ParsingException {
@Nonnull final String contentPlaybackNonce) {
if (streamingData == null || !streamingData.has(streamingDataKey)) { if (streamingData == null || !streamingData.has(streamingDataKey)) {
return Collections.emptyList(); return Collections.emptyList();
} }
final String videoId = getId();
final List<ItagInfo> itagInfos = new ArrayList<>(); final List<ItagInfo> itagInfos = new ArrayList<>();
final JsonArray formats = streamingData.getArray(streamingDataKey); final JsonArray formats = streamingData.getArray(streamingDataKey);
for (int i = 0; i < formats.size(); i++) { for (int i = 0; i != formats.size(); ++i) {
final JsonObject formatData = formats.getObject(i); final JsonObject formatData = formats.getObject(i);
final int itag = formatData.getInt("itag"); final int itag = formatData.getInt("itag");
@ -1279,79 +1343,10 @@ public class YoutubeStreamExtractor extends StreamExtractor {
try { try {
final ItagItem itagItem = ItagItem.getItag(itag); final ItagItem itagItem = ItagItem.getItag(itag);
final ItagItem.ItagType itagType = itagItem.itagType; final ItagItem.ItagType itagType = itagItem.itagType;
if (itagItem.itagType != itagTypeWanted) { if (itagType == itagTypeWanted) {
continue; buildAndAddItagInfoToList(videoId, itagInfos, formatData, itagItem,
itagType, contentPlaybackNonce);
} }
String streamUrl;
if (formatData.has("url")) {
streamUrl = formatData.getString("url") + "&cpn="
+ contentPlaybackNonce;
} else {
// This url has an obfuscated signature
final String cipherString = formatData.has(CIPHER)
? formatData.getString(CIPHER)
: formatData.getString(SIGNATURE_CIPHER);
final Map<String, String> cipher = Parser.compatParseMap(
cipherString);
streamUrl = cipher.get("url") + "&" + cipher.get("sp") + "="
+ deobfuscateSignature(cipher.get("s"));
}
if (isWebStreamingUrl(streamUrl)) {
streamUrl = tryDecryptUrl(streamUrl, getId()) + "&cver="
+ getClientVersion();
}
final JsonObject initRange = formatData.getObject("initRange");
final JsonObject indexRange = formatData.getObject("indexRange");
final String mimeType = formatData.getString("mimeType", EMPTY_STRING);
final String codec = mimeType.contains("codecs")
? mimeType.split("\"")[1] : EMPTY_STRING;
itagItem.setBitrate(formatData.getInt("bitrate"));
itagItem.setWidth(formatData.getInt("width"));
itagItem.setHeight(formatData.getInt("height"));
itagItem.setInitStart(Integer.parseInt(initRange.getString("start",
"-1")));
itagItem.setInitEnd(Integer.parseInt(initRange.getString("end",
"-1")));
itagItem.setIndexStart(Integer.parseInt(indexRange.getString("start",
"-1")));
itagItem.setIndexEnd(Integer.parseInt(indexRange.getString("end",
"-1")));
itagItem.setQuality(formatData.getString("quality"));
itagItem.setCodec(codec);
if (contentStreamType != StreamType.VIDEO_STREAM) {
itagItem.setTargetDurationSec(formatData.getInt(
"targetDurationSec"));
}
if (itagType == ItagItem.ItagType.VIDEO
|| itagType == ItagItem.ItagType.VIDEO_ONLY) {
itagItem.setFps(formatData.getInt("fps"));
}
if (itagType == ItagItem.ItagType.AUDIO) {
itagItem.setSampleRate(Integer.parseInt(formatData.getString(
"audioSampleRate")));
itagItem.setAudioChannels(formatData.getInt("audioChannels"));
}
itagItem.setContentLength(Long.parseLong(formatData.getString(
"contentLength", "-1")));
final ItagInfo itagInfo = new ItagInfo(streamUrl, itagItem);
if (contentStreamType == StreamType.VIDEO_STREAM) {
itagInfo.setIsUrl(!formatData.getString("type", EMPTY_STRING)
.equalsIgnoreCase("FORMAT_STREAM_TYPE_OTF"));
} else {
// We are currently not able to generate DASH manifests for running
// livestreams, so because of the requirements of StreamInfo
// objects, return these streams as DASH URL streams (even if they
// are not playable).
// Ended livestreams are returned as non URL streams
itagInfo.setIsUrl(contentStreamType != StreamType.POST_LIVE_STREAM);
}
itagInfos.add(itagInfo);
} catch (final IOException | ExtractionException ignored) { } catch (final IOException | ExtractionException ignored) {
} }
} }
@ -1359,6 +1354,83 @@ public class YoutubeStreamExtractor extends StreamExtractor {
return itagInfos; return itagInfos;
} }
private void buildAndAddItagInfoToList(
@Nonnull final String videoId,
@Nonnull final List<ItagInfo> itagInfos,
@Nonnull final JsonObject formatData,
@Nonnull final ItagItem itagItem,
@Nonnull final ItagItem.ItagType itagType,
@Nonnull final String contentPlaybackNonce) throws IOException, ExtractionException {
String streamUrl;
if (formatData.has("url")) {
streamUrl = formatData.getString("url");
} else {
// This url has an obfuscated signature
final String cipherString = formatData.has(CIPHER)
? formatData.getString(CIPHER)
: formatData.getString(SIGNATURE_CIPHER);
final Map<String, String> cipher = Parser.compatParseMap(
cipherString);
streamUrl = cipher.get("url") + "&" + cipher.get("sp") + "="
+ deobfuscateSignature(cipher.get("s"));
}
// Add the content playback nonce to the stream URL
streamUrl += "&" + CPN + "=" + contentPlaybackNonce;
if (isWebStreamingUrl(streamUrl)) {
streamUrl = tryDecryptUrl(streamUrl, videoId) + "&cver=" + getClientVersion();
}
final JsonObject initRange = formatData.getObject("initRange");
final JsonObject indexRange = formatData.getObject("indexRange");
final String mimeType = formatData.getString("mimeType", EMPTY_STRING);
final String codec = mimeType.contains("codecs")
? mimeType.split("\"")[1] : EMPTY_STRING;
itagItem.setBitrate(formatData.getInt("bitrate"));
itagItem.setWidth(formatData.getInt("width"));
itagItem.setHeight(formatData.getInt("height"));
itagItem.setInitStart(Integer.parseInt(initRange.getString("start", "-1")));
itagItem.setInitEnd(Integer.parseInt(initRange.getString("end", "-1")));
itagItem.setIndexStart(Integer.parseInt(indexRange.getString("start", "-1")));
itagItem.setIndexEnd(Integer.parseInt(indexRange.getString("end", "-1")));
itagItem.setQuality(formatData.getString("quality"));
itagItem.setCodec(codec);
if (streamType == StreamType.LIVE_STREAM || streamType == StreamType.POST_LIVE_STREAM) {
itagItem.setTargetDurationSec(formatData.getInt("targetDurationSec"));
}
if (itagType == ItagItem.ItagType.VIDEO || itagType == ItagItem.ItagType.VIDEO_ONLY) {
itagItem.setFps(formatData.getInt("fps"));
}
if (itagType == ItagItem.ItagType.AUDIO) {
// YouTube return the audio sample rate as a string
itagItem.setSampleRate(Integer.parseInt(formatData.getString("audioSampleRate")));
itagItem.setAudioChannels(formatData.getInt("audioChannels"));
}
// YouTube return the content length as a string
itagItem.setContentLength(Long.parseLong(formatData.getString("contentLength",
String.valueOf(CONTENT_LENGTH_UNKNOWN))));
final ItagInfo itagInfo = new ItagInfo(streamUrl, itagItem);
if (streamType == StreamType.VIDEO_STREAM) {
itagInfo.setIsUrl(!formatData.getString("type", EMPTY_STRING)
.equalsIgnoreCase("FORMAT_STREAM_TYPE_OTF"));
} else {
// We are currently not able to generate DASH manifests for running
// livestreams, so because of the requirements of StreamInfo
// objects, return these streams as DASH URL streams (even if they
// are not playable).
// Ended livestreams are returned as non URL streams
itagInfo.setIsUrl(streamType != StreamType.POST_LIVE_STREAM);
}
itagInfos.add(itagInfo);
}
@Nonnull @Nonnull
@Override @Override

View file

@ -68,10 +68,10 @@ public final class AudioStream extends Stream {
} }
/** /**
* Set the identifier of the {@link SubtitlesStream}. * Set the identifier of the {@link AudioStream}.
* *
* <p> * <p>
* It <b>must be not null</b> and should be non empty. * It <b>must not be null</b> and should be non empty.
* </p> * </p>
* *
* <p> * <p>
@ -79,7 +79,7 @@ public final class AudioStream extends Stream {
* Stream#ID_UNKNOWN ID_UNKNOWN} of the {@link Stream} class. * Stream#ID_UNKNOWN ID_UNKNOWN} of the {@link Stream} class.
* </p> * </p>
* *
* @param id the identifier of the {@link SubtitlesStream}, which must be not null * @param id the identifier of the {@link AudioStream}, which must not be null
* @return this {@link Builder} instance * @return this {@link Builder} instance
*/ */
public Builder setId(@Nonnull final String id) { public Builder setId(@Nonnull final String id) {
@ -91,7 +91,7 @@ public final class AudioStream extends Stream {
* Set the content of the {@link AudioStream}. * Set the content of the {@link AudioStream}.
* *
* <p> * <p>
* It must be non null and should be non empty. * It must not be null, and should be non empty.
* </p> * </p>
* *
* @param content the content of the {@link AudioStream} * @param content the content of the {@link AudioStream}
@ -111,8 +111,8 @@ public final class AudioStream extends Stream {
* <p> * <p>
* It should be one of the audio {@link MediaFormat}s ({@link MediaFormat#M4A M4A}, * It should be one of the audio {@link MediaFormat}s ({@link MediaFormat#M4A M4A},
* {@link MediaFormat#WEBMA WEBMA}, {@link MediaFormat#MP3 MP3}, {@link MediaFormat#OPUS * {@link MediaFormat#WEBMA WEBMA}, {@link MediaFormat#MP3 MP3}, {@link MediaFormat#OPUS
* OPUS}, {@link MediaFormat#OGG OGG}, {@link MediaFormat#WEBMA_OPUS WEBMA_OPUS}) but can * OPUS}, {@link MediaFormat#OGG OGG}, or {@link MediaFormat#WEBMA_OPUS WEBMA_OPUS}) but
* be {@code null} if the media format could not be determined. * can be {@code null} if the media format could not be determined.
* </p> * </p>
* *
* <p> * <p>
@ -131,7 +131,7 @@ public final class AudioStream extends Stream {
* Set the {@link DeliveryMethod} of the {@link AudioStream}. * Set the {@link DeliveryMethod} of the {@link AudioStream}.
* *
* <p> * <p>
* It must be not null. * It must not be null.
* </p> * </p>
* *
* <p> * <p>
@ -139,7 +139,7 @@ public final class AudioStream extends Stream {
* </p> * </p>
* *
* @param deliveryMethod the {@link DeliveryMethod} of the {@link AudioStream}, which must * @param deliveryMethod the {@link DeliveryMethod} of the {@link AudioStream}, which must
* be not null * not be null
* @return this {@link Builder} instance * @return this {@link Builder} instance
*/ */
public Builder setDeliveryMethod(@Nonnull final DeliveryMethod deliveryMethod) { public Builder setDeliveryMethod(@Nonnull final DeliveryMethod deliveryMethod) {
@ -151,8 +151,8 @@ public final class AudioStream extends Stream {
* Set the base URL of the {@link AudioStream}. * Set the base URL of the {@link AudioStream}.
* *
* <p> * <p>
* Base URLs are for instance, for non-URLs content, the DASH or HLS manifest from which * For non-URL contents, the base URL is, for instance, a link to the DASH or HLS manifest
* they have been parsed. * from which the URLs have been parsed.
* </p> * </p>
* *
* <p> * <p>
@ -213,7 +213,7 @@ public final class AudioStream extends Stream {
* *
* @return a new {@link AudioStream} using the builder's current values * @return a new {@link AudioStream} using the builder's current values
* @throws IllegalStateException if {@code id}, {@code content} (and so {@code isUrl}) or * @throws IllegalStateException if {@code id}, {@code content} (and so {@code isUrl}) or
* {@code deliveryMethod} have been not set or set as {@code null} * {@code deliveryMethod} have been not set, or have been set as {@code null}
*/ */
@Nonnull @Nonnull
public AudioStream build() { public AudioStream build() {
@ -244,8 +244,8 @@ public final class AudioStream extends Stream {
/** /**
* Create a new audio stream. * Create a new audio stream.
* *
* @param id the ID which uniquely identifies the stream, e.g. for YouTube this * @param id the identifier which uniquely identifies the stream, e.g. for YouTube
* would be the itag * this would be the itag
* @param content the content or the URL of the stream, depending on whether isUrl is * @param content the content or the URL of the stream, depending on whether isUrl is
* true * true
* @param isUrl whether content is the URL or the actual content of e.g. a DASH * @param isUrl whether content is the URL or the actual content of e.g. a DASH
@ -258,6 +258,7 @@ public final class AudioStream extends Stream {
* @param baseUrl the base URL of the stream (see {@link Stream#getBaseUrl()} for more * @param baseUrl the base URL of the stream (see {@link Stream#getBaseUrl()} for more
* information) * information)
*/ */
@SuppressWarnings("checkstyle:ParameterNumber")
private AudioStream(@Nonnull final String id, private AudioStream(@Nonnull final String id,
@Nonnull final String content, @Nonnull final String content,
final boolean isUrl, final boolean isUrl,

View file

@ -13,25 +13,43 @@ public enum DeliveryMethod {
PROGRESSIVE_HTTP, PROGRESSIVE_HTTP,
/** /**
* Enum constant which represents the use of the DASH adaptive streaming method to fetch a * Enum constant which represents the use of the DASH (Dynamic Adaptive Streaming over HTTP)
* {@link Stream stream}. * adaptive streaming method to fetch a {@link Stream stream}.
*
* @see <a href="https://en.wikipedia.org/wiki/Dynamic_Adaptive_Streaming_over_HTTP">the
* Dynamic Adaptive Streaming over HTTP Wikipedia page</a> and <a href="https://dashif.org/">
* DASH Industry Forum's website</a> for more information about the DASH delivery method
*/ */
DASH, DASH,
/** /**
* Enum constant which represents the use of the HLS adaptive streaming method to fetch a * Enum constant which represents the use of the HLS (HTTP Live Streaming) adaptive streaming
* {@link Stream stream}. * method to fetch a {@link Stream stream}.
*
* @see <a href="https://en.wikipedia.org/wiki/HTTP_Live_Streaming">the HTTP Live Streaming
* page</a> and <a href="https://developer.apple.com/streaming">Apple's developers website page
* about HTTP Live Streaming</a> for more information about the HLS delivery method
*/ */
HLS, HLS,
/** /**
* Enum constant which represents the use of the SmoothStreaming adaptive streaming method to * Enum constant which represents the use of the SmoothStreaming adaptive streaming method to
* fetch a {@link Stream stream}. * fetch a {@link Stream stream}.
*
* @see <a href="https://en.wikipedia.org/wiki/Adaptive_bitrate_streaming
* #Microsoft_Smooth_Streaming_(MSS)">Wikipedia's page about adaptive bitrate streaming,
* section <i>Microsoft Smooth Streaming (MSS)</i></a> for more information about the
* SmoothStreaming delivery method
*/ */
SS, SS,
/** /**
* Enum constant which represents the use of a torrent to fetch a {@link Stream stream}. * Enum constant which represents the use of a torrent file to fetch a {@link Stream stream}.
*
* @see <a href="https://en.wikipedia.org/wiki/BitTorrent">Wikipedia's BitTorrent's page</a>,
* <a href="https://en.wikipedia.org/wiki/Torrent_file">Wikipedia's page about torrent files
* </a> and <a href=""https://www.bittorrent.org/></a> for more information about the
* BitTorrent protocol
*/ */
TORRENT TORRENT
} }

View file

@ -19,11 +19,11 @@ public abstract class Stream implements Serializable {
public static final String ID_UNKNOWN = " "; public static final String ID_UNKNOWN = " ";
/** /**
* An integer to represent that the itag id returned is not available (only for YouTube, this * An integer to represent that the itag ID returned is not available (only for YouTube; this
* should never happen) or not applicable (for other services than YouTube). * should never happen) or not applicable (for other services than YouTube).
* *
* <p> * <p>
* An itag should not have a negative value so {@code -1} is used for this constant. * An itag should not have a negative value, so {@code -1} is used for this constant.
* </p> * </p>
*/ */
public static final int ITAG_NOT_AVAILABLE_OR_NOT_APPLICABLE = -1; public static final int ITAG_NOT_AVAILABLE_OR_NOT_APPLICABLE = -1;
@ -38,8 +38,8 @@ public abstract class Stream implements Serializable {
/** /**
* Instantiates a new {@code Stream} object. * Instantiates a new {@code Stream} object.
* *
* @param id the ID which uniquely identifies the file, e.g. for YouTube this would * @param id the identifier which uniquely identifies the file, e.g. for YouTube
* be the itag * this would be the itag
* @param content the content or URL, depending on whether isUrl is true * @param content the content or URL, depending on whether isUrl is true
* @param isUrl whether content is the URL or the actual content of e.g. a DASH * @param isUrl whether content is the URL or the actual content of e.g. a DASH
* manifest * manifest
@ -63,10 +63,10 @@ public abstract class Stream implements Serializable {
} }
/** /**
* Checks if the list already contains one stream with equals stats. * Checks if the list already contains a stream with the same statistics.
* *
* @param stream the stream which will be compared to the streams in the stream list * @param stream the stream to be compared against the streams in the stream list
* @param streamList the list of {@link Stream Streams} which will be compared * @param streamList the list of {@link Stream}s which will be compared
* @return whether the list already contains one stream with equals stats * @return whether the list already contains one stream with equals stats
*/ */
public static boolean containSimilarStream(final Stream stream, public static boolean containSimilarStream(final Stream stream,
@ -83,16 +83,16 @@ public abstract class Stream implements Serializable {
} }
/** /**
* Reveals whether two streams have the same stats ({@link MediaFormat media format} and * Reveals whether two streams have the same statistics ({@link MediaFormat media format} and
* {@link DeliveryMethod delivery method}). * {@link DeliveryMethod delivery method}).
* *
* <p> * <p>
* If the {@link MediaFormat media format} of the stream is unknown, the streams are compared * If the {@link MediaFormat media format} of the stream is unknown, the streams are compared
* by only using the {@link DeliveryMethod delivery method} and their id. * by using only the {@link DeliveryMethod delivery method} and their ID.
* </p> * </p>
* *
* <p> * <p>
* Note: This method always returns always false if the stream passed is null. * Note: This method always returns false if the stream passed is null.
* </p> * </p>
* *
* @param cmp the stream object to be compared to this stream object * @param cmp the stream object to be compared to this stream object
@ -118,9 +118,12 @@ public abstract class Stream implements Serializable {
/** /**
* Reveals whether two streams are equal. * Reveals whether two streams are equal.
* *
* @param cmp the stream object to be compared to this stream object * @param cmp a {@link Stream} object to be compared to this {@link Stream} instance.
* @return whether streams are equal * @return whether the compared streams are equal
* @deprecated Use {@link #equalStats(Stream)} to compare statistics of two streams and
* {@link #equals(Object)} to compare the equality of two streams instead.
*/ */
@Deprecated
public boolean equals(final Stream cmp) { public boolean equals(final Stream cmp) {
return equalStats(cmp) && content.equals(cmp.content); return equalStats(cmp) && content.equals(cmp.content);
} }
@ -129,19 +132,19 @@ public abstract class Stream implements Serializable {
* Gets the identifier of this stream, e.g. the itag for YouTube. * Gets the identifier of this stream, e.g. the itag for YouTube.
* *
* <p> * <p>
* It should be normally unique but {@link #ID_UNKNOWN} may be returned as the identifier if * It should normally be unique, but {@link #ID_UNKNOWN} may be returned as the identifier if
* one used by the stream extractor cannot be extracted, if the extractor uses a value from a * the one used by the stream extractor cannot be extracted, which could happen if the
* streaming service. * extractor uses a value from a streaming service.
* </p> * </p>
* *
* @return the id (which may be {@link #ID_UNKNOWN}) * @return the identifier (which may be {@link #ID_UNKNOWN})
*/ */
public String getId() { public String getId() {
return id; return id;
} }
/** /**
* Gets the URL of this stream if the content is a URL, or {@code null} if that's the not case. * Gets the URL of this stream if the content is a URL, or {@code null} otherwise.
* *
* @return the URL if the content is a URL, {@code null} otherwise * @return the URL if the content is a URL, {@code null} otherwise
* @deprecated Use {@link #getContent()} instead. * @deprecated Use {@link #getContent()} instead.
@ -162,10 +165,10 @@ public abstract class Stream implements Serializable {
} }
/** /**
* Returns if the content is a URL or not. * Returns whether the content is a URL or not.
* *
* @return {@code true} if the content of this stream content is a URL, {@code false} * @return {@code true} if the content of this stream is a URL, {@code false} if it's the
* if it is the actual content * actual content
*/ */
public boolean isUrl() { public boolean isUrl() {
return isUrl; return isUrl;
@ -182,9 +185,9 @@ public abstract class Stream implements Serializable {
} }
/** /**
* Gets the format id, which can be unknown. * Gets the format ID, which can be unknown.
* *
* @return the format id or {@link #FORMAT_ID_UNKNOWN} * @return the format ID or {@link #FORMAT_ID_UNKNOWN}
*/ */
public int getFormatId() { public int getFormatId() {
if (mediaFormat != null) { if (mediaFormat != null) {
@ -208,7 +211,7 @@ public abstract class Stream implements Serializable {
* *
* <p> * <p>
* If the stream is not a DASH stream or an HLS stream, this value will always be null. * If the stream is not a DASH stream or an HLS stream, this value will always be null.
* It may be also null for these streams too. * It may also be null for these streams too.
* </p> * </p>
* *
* @return the base URL of the stream or {@code null} * @return the base URL of the stream or {@code null}
@ -222,7 +225,7 @@ public abstract class Stream implements Serializable {
* Gets the {@link ItagItem} of a stream. * Gets the {@link ItagItem} of a stream.
* *
* <p> * <p>
* If the stream is not a YouTube stream, this value will always be null. * If the stream is not from YouTube, this value will always be null.
* </p> * </p>
* *
* @return the {@link ItagItem} of the stream or {@code null} * @return the {@link ItagItem} of the stream or {@code null}
@ -242,11 +245,14 @@ public abstract class Stream implements Serializable {
final Stream stream = (Stream) obj; final Stream stream = (Stream) obj;
return id.equals(stream.id) && mediaFormat == stream.mediaFormat return id.equals(stream.id) && mediaFormat == stream.mediaFormat
&& deliveryMethod == stream.deliveryMethod; && deliveryMethod == stream.deliveryMethod
&& content.equals(stream.content)
&& isUrl == stream.isUrl
&& Objects.equals(baseUrl, stream.baseUrl);
} }
@Override @Override
public int hashCode() { public int hashCode() {
return Objects.hash(id, mediaFormat, deliveryMethod); return Objects.hash(id, mediaFormat, deliveryMethod, content, isUrl, baseUrl);
} }
} }

View file

@ -17,10 +17,10 @@ public enum StreamType {
NONE, NONE,
/** /**
* Enum constant to indicate that the stream type of stream content is a video. * Enum constant to indicate that the stream type of stream content is a live video.
* *
* <p> * <p>
* Note that contents <strong>can contain audio streams</strong> even if they also contain * Note that contents <strong>may contain audio streams</strong> even if they also contain
* video streams (video-only or video with audio, depending of the stream/the content/the * video streams (video-only or video with audio, depending of the stream/the content/the
* service). * service).
* </p> * </p>
@ -46,23 +46,22 @@ public enum StreamType {
* *
* <p> * <p>
* Note that contents <strong>can contain audio live streams</strong> even if they also contain * Note that contents <strong>can contain audio live streams</strong> even if they also contain
* live video streams (video-only or video with audio, depending of the stream/the content/the * live video streams (so video-only or video with audio, depending on the stream/the content/
* service). * the service).
* </p> * </p>
*/ */
LIVE_STREAM, LIVE_STREAM,
/** /**
* Enum constant to indicate that the stream type of stream content is a live audio content. * Enum constant to indicate that the stream type of stream content is a live audio.
* *
* <p> * <p>
* Note that contents returned as live audio streams should not return live video streams. * Note that contents returned as live audio streams should not return live video streams.
* </p> * </p>
* *
* <p> * <p>
* So, in order to prevent unexpected behaviors, stream extractors which are returning this * To prevent unexpected behavior, stream extractors which are returning this stream type for a
* stream type for a content should ensure that no live video stream is returned for this * content should ensure that no live video stream is returned along with it.
* content.
* </p> * </p>
*/ */
AUDIO_LIVE_STREAM, AUDIO_LIVE_STREAM,
@ -72,10 +71,10 @@ public enum StreamType {
* ended live video stream. * ended live video stream.
* *
* <p> * <p>
* Note that most of ended live video (or audio) contents may be extracted as * Note that most of the content of an ended live video (or audio) may be extracted as {@link
* {@link #VIDEO_STREAM regular video contents} (or * #VIDEO_STREAM regular video contents} (or {@link #AUDIO_STREAM regular audio contents})
* {@link #AUDIO_STREAM regular audio contents}) later, because the service may encode them * later, because the service may encode them again later as normal video/audio streams. That's
* again later as normal video/audio streams. That's the case for example on YouTube. * the case on YouTube, for example.
* </p> * </p>
* *
* <p> * <p>

View file

@ -35,7 +35,7 @@ public final class SubtitlesStream extends Stream {
private Boolean autoGenerated; private Boolean autoGenerated;
/** /**
* Create a new {@link Builder} instance with its default values. * Create a new {@link Builder} instance with default values.
*/ */
public Builder() { public Builder() {
} }
@ -43,7 +43,7 @@ public final class SubtitlesStream extends Stream {
/** /**
* Set the identifier of the {@link SubtitlesStream}. * Set the identifier of the {@link SubtitlesStream}.
* *
* @param id the identifier of the {@link SubtitlesStream}, which should be not null * @param id the identifier of the {@link SubtitlesStream}, which should not be null
* (otherwise the fallback to create the identifier will be used when building * (otherwise the fallback to create the identifier will be used when building
* the builder) * the builder)
* @return this {@link Builder} instance * @return this {@link Builder} instance
@ -57,10 +57,10 @@ public final class SubtitlesStream extends Stream {
* Set the content of the {@link SubtitlesStream}. * Set the content of the {@link SubtitlesStream}.
* *
* <p> * <p>
* It must be non null and should be non empty. * It must not be null, and should be non empty.
* </p> * </p>
* *
* @param content the content of the {@link SubtitlesStream} * @param content the content of the {@link SubtitlesStream}, which must not be null
* @param isUrl whether the content is a URL * @param isUrl whether the content is a URL
* @return this {@link Builder} instance * @return this {@link Builder} instance
*/ */
@ -78,7 +78,7 @@ public final class SubtitlesStream extends Stream {
* It should be one of the subtitles {@link MediaFormat}s ({@link MediaFormat#SRT SRT}, * It should be one of the subtitles {@link MediaFormat}s ({@link MediaFormat#SRT SRT},
* {@link MediaFormat#TRANSCRIPT1 TRANSCRIPT1}, {@link MediaFormat#TRANSCRIPT2 * {@link MediaFormat#TRANSCRIPT1 TRANSCRIPT1}, {@link MediaFormat#TRANSCRIPT2
* TRANSCRIPT2}, {@link MediaFormat#TRANSCRIPT3 TRANSCRIPT3}, {@link MediaFormat#TTML * TRANSCRIPT2}, {@link MediaFormat#TRANSCRIPT3 TRANSCRIPT3}, {@link MediaFormat#TTML
* TTML}, {@link MediaFormat#VTT VTT}) but can be {@code null} if the media format could * TTML}, or {@link MediaFormat#VTT VTT}) but can be {@code null} if the media format could
* not be determined. * not be determined.
* </p> * </p>
* *
@ -99,7 +99,7 @@ public final class SubtitlesStream extends Stream {
* Set the {@link DeliveryMethod} of the {@link SubtitlesStream}. * Set the {@link DeliveryMethod} of the {@link SubtitlesStream}.
* *
* <p> * <p>
* It must be not null. * It must not be null.
* </p> * </p>
* *
* <p> * <p>
@ -107,7 +107,7 @@ public final class SubtitlesStream extends Stream {
* </p> * </p>
* *
* @param deliveryMethod the {@link DeliveryMethod} of the {@link SubtitlesStream}, which * @param deliveryMethod the {@link DeliveryMethod} of the {@link SubtitlesStream}, which
* must be not null * must not be null
* @return this {@link Builder} instance * @return this {@link Builder} instance
*/ */
public Builder setDeliveryMethod(@Nonnull final DeliveryMethod deliveryMethod) { public Builder setDeliveryMethod(@Nonnull final DeliveryMethod deliveryMethod) {
@ -119,8 +119,8 @@ public final class SubtitlesStream extends Stream {
* Set the base URL of the {@link SubtitlesStream}. * Set the base URL of the {@link SubtitlesStream}.
* *
* <p> * <p>
* Base URLs are for instance, for non-URLs content, the DASH or HLS manifest from which * For non-URL contents, the base URL is, for instance, a link to the DASH or HLS manifest
* they have been parsed. * from which the URLs have been parsed.
* </p> * </p>
* *
* <p> * <p>
@ -139,7 +139,7 @@ public final class SubtitlesStream extends Stream {
* Set the language code of the {@link SubtitlesStream}. * Set the language code of the {@link SubtitlesStream}.
* *
* <p> * <p>
* It <b>must be not null</b> and should be not an empty string. * It <b>must not be null</b> and should not be an empty string.
* </p> * </p>
* *
* @param languageCode the language code of the {@link SubtitlesStream} * @param languageCode the language code of the {@link SubtitlesStream}
@ -151,10 +151,10 @@ public final class SubtitlesStream extends Stream {
} }
/** /**
* Set whether the subtitles have been generated by the streaming service. * Set whether the subtitles have been auto-generated by the streaming service.
* *
* @param autoGenerated whether the subtitles have been generated by the streaming * @param autoGenerated whether the subtitles have been generated by the streaming
* service * service
* @return this {@link Builder} instance * @return this {@link Builder} instance
*/ */
public Builder setAutoGenerated(final boolean autoGenerated) { public Builder setAutoGenerated(final boolean autoGenerated) {
@ -172,13 +172,13 @@ public final class SubtitlesStream extends Stream {
* *
* <p> * <p>
* If no identifier has been set, an identifier will be generated using the language code * If no identifier has been set, an identifier will be generated using the language code
* and the media format suffix if the media format is known * and the media format suffix, if the media format is known.
* </p> * </p>
* *
* @return a new {@link SubtitlesStream} using the builder's current values * @return a new {@link SubtitlesStream} using the builder's current values
* @throws IllegalStateException if {@code id}, {@code content} (and so {@code isUrl}), * @throws IllegalStateException if {@code id}, {@code content} (and so {@code isUrl}),
* {@code deliveryMethod}, {@code languageCode} or the {@code isAutogenerated} have been * {@code deliveryMethod}, {@code languageCode} or the {@code isAutogenerated} have been
* not set or set as {@code null} * not set, or have been set as {@code null}
*/ */
@Nonnull @Nonnull
public SubtitlesStream build() { public SubtitlesStream build() {
@ -219,28 +219,29 @@ public final class SubtitlesStream extends Stream {
/** /**
* Create a new subtitles stream. * Create a new subtitles stream.
* *
* @param id the ID which uniquely identifies the stream, e.g. for YouTube this * @param id the identifier which uniquely identifies the stream, e.g. for YouTube
* would be the itag * this would be the itag
* @param content the content or the URL of the stream, depending on whether isUrl is * @param content the content or the URL of the stream, depending on whether isUrl is
* true * true
* @param isUrl whether content is the URL or the actual content of e.g. a DASH * @param isUrl whether content is the URL or the actual content of e.g. a DASH
* manifest * manifest
* @param format the {@link MediaFormat} used by the stream * @param mediaFormat the {@link MediaFormat} used by the stream
* @param deliveryMethod the {@link DeliveryMethod} of the stream * @param deliveryMethod the {@link DeliveryMethod} of the stream
* @param languageCode the language code of the stream * @param languageCode the language code of the stream
* @param autoGenerated whether the subtitles are auto-generated by the streaming service * @param autoGenerated whether the subtitles are auto-generated by the streaming service
* @param baseUrl the base URL of the stream (see {@link Stream#getBaseUrl()} for more * @param baseUrl the base URL of the stream (see {@link Stream#getBaseUrl()} for more
* information) * information)
*/ */
@SuppressWarnings("checkstyle:ParameterNumber")
private SubtitlesStream(@Nonnull final String id, private SubtitlesStream(@Nonnull final String id,
@Nonnull final String content, @Nonnull final String content,
final boolean isUrl, final boolean isUrl,
@Nullable final MediaFormat format, @Nullable final MediaFormat mediaFormat,
@Nonnull final DeliveryMethod deliveryMethod, @Nonnull final DeliveryMethod deliveryMethod,
@Nonnull final String languageCode, @Nonnull final String languageCode,
final boolean autoGenerated, final boolean autoGenerated,
@Nullable final String baseUrl) { @Nullable final String baseUrl) {
super(id, content, isUrl, format, deliveryMethod, baseUrl); super(id, content, isUrl, mediaFormat, deliveryMethod, baseUrl);
/* /*
* Locale.forLanguageTag only for Android API >= 21 * Locale.forLanguageTag only for Android API >= 21
@ -262,7 +263,7 @@ public final class SubtitlesStream extends Stream {
} }
this.code = languageCode; this.code = languageCode;
this.format = format; this.format = mediaFormat;
this.autoGenerated = autoGenerated; this.autoGenerated = autoGenerated;
} }

View file

@ -81,7 +81,7 @@ public final class VideoStream extends Stream {
* Set the identifier of the {@link VideoStream}. * Set the identifier of the {@link VideoStream}.
* *
* <p> * <p>
* It <b>must be not null</b> and should be non empty. * It must not be null, and should be non empty.
* </p> * </p>
* *
* <p> * <p>
@ -89,7 +89,7 @@ public final class VideoStream extends Stream {
* Stream#ID_UNKNOWN ID_UNKNOWN} of the {@link Stream} class. * Stream#ID_UNKNOWN ID_UNKNOWN} of the {@link Stream} class.
* </p> * </p>
* *
* @param id the identifier of the {@link VideoStream}, which must be not null * @param id the identifier of the {@link VideoStream}, which must not be null
* @return this {@link Builder} instance * @return this {@link Builder} instance
*/ */
public Builder setId(@Nonnull final String id) { public Builder setId(@Nonnull final String id) {
@ -101,7 +101,7 @@ public final class VideoStream extends Stream {
* Set the content of the {@link VideoStream}. * Set the content of the {@link VideoStream}.
* *
* <p> * <p>
* It must be non null and should be non empty. * It must not be null, and should be non empty.
* </p> * </p>
* *
* @param content the content of the {@link VideoStream} * @param content the content of the {@link VideoStream}
@ -120,8 +120,8 @@ public final class VideoStream extends Stream {
* *
* <p> * <p>
* It should be one of the video {@link MediaFormat}s ({@link MediaFormat#MPEG_4 MPEG_4}, * It should be one of the video {@link MediaFormat}s ({@link MediaFormat#MPEG_4 MPEG_4},
* {@link MediaFormat#v3GPP v3GPP}, {@link MediaFormat#WEBM WEBM}) but can be {@code null} * {@link MediaFormat#v3GPP v3GPP}, or {@link MediaFormat#WEBM WEBM}) but can be {@code
* if the media format could not be determined. * null} if the media format could not be determined.
* </p> * </p>
* *
* <p> * <p>
@ -140,7 +140,7 @@ public final class VideoStream extends Stream {
* Set the {@link DeliveryMethod} of the {@link VideoStream}. * Set the {@link DeliveryMethod} of the {@link VideoStream}.
* *
* <p> * <p>
* It must be not null. * It must not be null.
* </p> * </p>
* *
* <p> * <p>
@ -148,7 +148,7 @@ public final class VideoStream extends Stream {
* </p> * </p>
* *
* @param deliveryMethod the {@link DeliveryMethod} of the {@link VideoStream}, which must * @param deliveryMethod the {@link DeliveryMethod} of the {@link VideoStream}, which must
* be not null * not be null
* @return this {@link Builder} instance * @return this {@link Builder} instance
*/ */
public Builder setDeliveryMethod(@Nonnull final DeliveryMethod deliveryMethod) { public Builder setDeliveryMethod(@Nonnull final DeliveryMethod deliveryMethod) {
@ -160,8 +160,8 @@ public final class VideoStream extends Stream {
* Set the base URL of the {@link VideoStream}. * Set the base URL of the {@link VideoStream}.
* *
* <p> * <p>
* Base URLs are for instance, for non-URLs content, the DASH or HLS manifest from which * For non-URL contents, the base URL is, for instance, a link to the DASH or HLS manifest
* they have been parsed. * from which the URLs have been parsed.
* </p> * </p>
* *
* <p> * <p>
@ -245,8 +245,8 @@ public final class VideoStream extends Stream {
* *
* @return a new {@link VideoStream} using the builder's current values * @return a new {@link VideoStream} using the builder's current values
* @throws IllegalStateException if {@code id}, {@code content} (and so {@code isUrl}), * @throws IllegalStateException if {@code id}, {@code content} (and so {@code isUrl}),
* {@code deliveryMethod}, {@code isVideoOnly} or {@code resolution} have been not set or * {@code deliveryMethod}, {@code isVideoOnly} or {@code resolution} have been not set, or
* set as {@code null} * have been set as {@code null}
*/ */
@Nonnull @Nonnull
public VideoStream build() { public VideoStream build() {
@ -289,8 +289,8 @@ public final class VideoStream extends Stream {
/** /**
* Create a new video stream. * Create a new video stream.
* *
* @param id the ID which uniquely identifies the stream, e.g. for YouTube this * @param id the identifier which uniquely identifies the stream, e.g. for YouTube
* would be the itag * this would be the itag
* @param content the content or the URL of the stream, depending on whether isUrl is * @param content the content or the URL of the stream, depending on whether isUrl is
* true * true
* @param isUrl whether content is the URL or the actual content of e.g. a DASH * @param isUrl whether content is the URL or the actual content of e.g. a DASH
@ -303,6 +303,7 @@ public final class VideoStream extends Stream {
* @param baseUrl the base URL of the stream (see {@link Stream#getBaseUrl()} for more * @param baseUrl the base URL of the stream (see {@link Stream#getBaseUrl()} for more
* information) * information)
*/ */
@SuppressWarnings("checkstyle:ParameterNumber")
private VideoStream(@Nonnull final String id, private VideoStream(@Nonnull final String id,
@Nonnull final String content, @Nonnull final String content,
final boolean isUrl, final boolean isUrl,

View file

@ -53,7 +53,7 @@ class YoutubeDashManifestCreatorTest {
// Setting a higher number may let Google video servers return a lot of 403s // Setting a higher number may let Google video servers return a lot of 403s
private static final int MAXIMUM_NUMBER_OF_STREAMS_TO_TEST = 3; private static final int MAXIMUM_NUMBER_OF_STREAMS_TO_TEST = 3;
public static class testGenerationOfOtfAndProgressiveManifests { public static class TestGenerationOfOtfAndProgressiveManifests {
private static final String url = "https://www.youtube.com/watch?v=DJ8GQUNUXGM"; private static final String url = "https://www.youtube.com/watch?v=DJ8GQUNUXGM";
private static YoutubeStreamExtractor extractor; private static YoutubeStreamExtractor extractor;