[YouTube] Add documentation about parameters added and clients versions and key
Also move the iPhone device machine id to a constant, explain how it is used and move the licence in the header of the file, and fix missing imports in YoutubeStreamExtractor (due to a rebase issue).
This commit is contained in:
parent
1dad3bfe8b
commit
2e3da445e6
2 changed files with 69 additions and 4 deletions
|
@ -80,6 +80,9 @@ public final class YoutubeParsingHelper {
|
||||||
private YoutubeParsingHelper() {
|
private YoutubeParsingHelper() {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The base URL of requests of the {@code WEB} client to the InnerTube internal API
|
||||||
|
*/
|
||||||
public static final String YOUTUBEI_V1_URL = "https://www.youtube.com/youtubei/v1/";
|
public static final String YOUTUBEI_V1_URL = "https://www.youtube.com/youtubei/v1/";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -91,14 +94,60 @@ public final class YoutubeParsingHelper {
|
||||||
* </p>
|
* </p>
|
||||||
**/
|
**/
|
||||||
public static final String DISABLE_PRETTY_PRINT_PARAMETER = "&prettyPrint=false";
|
public static final String DISABLE_PRETTY_PRINT_PARAMETER = "&prettyPrint=false";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A parameter sent by official clients named {@code contentPlaybackNonce}.
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* It is sent by official clients on videoplayback requests, and by all clients (except the
|
||||||
|
* {@code WEB} one to the player requests.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* It is composed of 16 characters which are generated from
|
||||||
|
* {@link #CONTENT_PLAYBACK_NONCE_ALPHABET this alphabet}, with the use of strong random
|
||||||
|
* values.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @see #generateContentPlaybackNonce()
|
||||||
|
*/
|
||||||
public static final String CPN = "cpn";
|
public static final String CPN = "cpn";
|
||||||
public static final String VIDEO_ID = "videoId";
|
public static final String VIDEO_ID = "videoId";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The client version for InnerTube requests with the {@code WEB} client, used as the last
|
||||||
|
* fallback if the extraction of the real one failed.
|
||||||
|
*
|
||||||
|
* You can get it directly either into YouTube pages or the service worker JavaScript file
|
||||||
|
* ({@code https://www.youtube.com/sw.js}) (also applies for YouTube Music).
|
||||||
|
*/
|
||||||
private static final String HARDCODED_CLIENT_VERSION = "2.20220315.01.00";
|
private static final String HARDCODED_CLIENT_VERSION = "2.20220315.01.00";
|
||||||
private static final String HARDCODED_KEY = "AIzaSyAO_FJ2SlqU8Q4STEHLGCilw_Y9_11qcW8";
|
private static final String HARDCODED_KEY = "AIzaSyAO_FJ2SlqU8Q4STEHLGCilw_Y9_11qcW8";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The InnerTube API key used by the {@code ANDROID} client. Found with the help of
|
||||||
|
* reverse-engineering app network requests.
|
||||||
|
*/
|
||||||
private static final String ANDROID_YOUTUBE_KEY = "AIzaSyA8eiZmM1FaDVjRy-df2KTyQ_vz_yYM39w";
|
private static final String ANDROID_YOUTUBE_KEY = "AIzaSyA8eiZmM1FaDVjRy-df2KTyQ_vz_yYM39w";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The InnerTube API key used by the {@code iOS} client. Found with the help of
|
||||||
|
* reverse-engineering app network requests.
|
||||||
|
*/
|
||||||
private static final String IOS_YOUTUBE_KEY = "AIzaSyB-63vPrdThhKuerbB2N_l7Kwwcxj6yUAc";
|
private static final String IOS_YOUTUBE_KEY = "AIzaSyB-63vPrdThhKuerbB2N_l7Kwwcxj6yUAc";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The hardcoded client version of the Android app used for InnerTube requests with this
|
||||||
|
* client.
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* It can be extracted by getting the latest release version of the app in an APK repository
|
||||||
|
* such as APKMirror.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @implNote This version is also used for the {@code iOS} client, as getting the app version
|
||||||
|
* without an iPhone device is not so easily.
|
||||||
|
*/
|
||||||
private static final String MOBILE_YOUTUBE_CLIENT_VERSION = "17.10.35";
|
private static final String MOBILE_YOUTUBE_CLIENT_VERSION = "17.10.35";
|
||||||
|
|
||||||
private static String clientVersion;
|
private static String clientVersion;
|
||||||
|
@ -128,6 +177,16 @@ public final class YoutubeParsingHelper {
|
||||||
private static final String CONTENT_PLAYBACK_NONCE_ALPHABET =
|
private static final String CONTENT_PLAYBACK_NONCE_ALPHABET =
|
||||||
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_";
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The device machine id for the iPhone 13, used to get 60fps with the {@code iOS} client.
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* See <a href="https://gist.github.com/adamawolf/3048717">this GitHub Gist</a> for more
|
||||||
|
* information.
|
||||||
|
* </p>
|
||||||
|
*/
|
||||||
|
private static final String IOS_DEVICE_MODEL = "iPhone14,5";
|
||||||
|
|
||||||
private static Random numberGenerator = new SecureRandom();
|
private static Random numberGenerator = new SecureRandom();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1071,7 +1130,7 @@ public final class YoutubeParsingHelper {
|
||||||
.value("clientName", "IOS")
|
.value("clientName", "IOS")
|
||||||
.value("clientVersion", MOBILE_YOUTUBE_CLIENT_VERSION)
|
.value("clientVersion", MOBILE_YOUTUBE_CLIENT_VERSION)
|
||||||
// Device model is required to get 60fps streams
|
// Device model is required to get 60fps streams
|
||||||
.value("deviceModel", "iPhone14,5")
|
.value("deviceModel", IOS_DEVICE_MODEL)
|
||||||
.value("platform", "MOBILE")
|
.value("platform", "MOBILE")
|
||||||
.value("hl", localization.getLocalizationCode())
|
.value("hl", localization.getLocalizationCode())
|
||||||
.value("gl", contentCountry.getCountryCode())
|
.value("gl", contentCountry.getCountryCode())
|
||||||
|
@ -1169,7 +1228,7 @@ public final class YoutubeParsingHelper {
|
||||||
.value("clientVersion", MOBILE_YOUTUBE_CLIENT_VERSION)
|
.value("clientVersion", MOBILE_YOUTUBE_CLIENT_VERSION)
|
||||||
.value("clientScreen", "EMBED")
|
.value("clientScreen", "EMBED")
|
||||||
// Device model is required to get 60fps streams
|
// Device model is required to get 60fps streams
|
||||||
.value("deviceModel", "iPhone14,5")
|
.value("deviceModel", IOS_DEVICE_MODEL)
|
||||||
.value("platform", "MOBILE")
|
.value("platform", "MOBILE")
|
||||||
.value("hl", localization.getLocalizationCode())
|
.value("hl", localization.getLocalizationCode())
|
||||||
.value("gl", contentCountry.getCountryCode())
|
.value("gl", contentCountry.getCountryCode())
|
||||||
|
@ -1208,6 +1267,8 @@ public final class YoutubeParsingHelper {
|
||||||
: prepareDesktopJsonBuilder(localization, contentCountry))
|
: prepareDesktopJsonBuilder(localization, contentCountry))
|
||||||
.object("playbackContext")
|
.object("playbackContext")
|
||||||
.object("contentPlaybackContext")
|
.object("contentPlaybackContext")
|
||||||
|
// Some parameters which are sent by the official WEB client (probably some
|
||||||
|
// of them are not useful)
|
||||||
.value("currentUrl", "/watch?v=" + videoId)
|
.value("currentUrl", "/watch?v=" + videoId)
|
||||||
.value("vis", 0)
|
.value("vis", 0)
|
||||||
.value("splay", false)
|
.value("splay", false)
|
||||||
|
@ -1260,9 +1321,10 @@ public final class YoutubeParsingHelper {
|
||||||
*/
|
*/
|
||||||
@Nonnull
|
@Nonnull
|
||||||
public static String getIosUserAgent(@Nullable final Localization localization) {
|
public static String getIosUserAgent(@Nullable final Localization localization) {
|
||||||
// Spoofing an iPhone 13 running iOS 15.4 with the hardcoded mobile client version
|
// Spoofing an iPhone running iOS 15.4 with the hardcoded mobile client version
|
||||||
return "com.google.ios.youtube/" + MOBILE_YOUTUBE_CLIENT_VERSION
|
return "com.google.ios.youtube/" + MOBILE_YOUTUBE_CLIENT_VERSION
|
||||||
+ "(iPhone14,5; U; CPU iOS 15_4 like Mac OS X; "
|
+ "(" + IOS_DEVICE_MODEL
|
||||||
|
+ "; U; CPU iOS 15_4 like Mac OS X; "
|
||||||
+ (localization != null ? localization.getCountryCode()
|
+ (localization != null ? localization.getCountryCode()
|
||||||
: Localization.DEFAULT.getCountryCode())
|
: Localization.DEFAULT.getCountryCode())
|
||||||
+ ")";
|
+ ")";
|
||||||
|
|
|
@ -7,12 +7,15 @@ import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper
|
||||||
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;
|
||||||
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.getJsonAndroidPostResponse;
|
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.getJsonAndroidPostResponse;
|
||||||
|
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.getJsonIosPostResponse;
|
||||||
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.getJsonPostResponse;
|
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.getJsonPostResponse;
|
||||||
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.getTextFromObject;
|
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.getTextFromObject;
|
||||||
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.prepareAndroidMobileEmbedVideoJsonBuilder;
|
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.prepareAndroidMobileEmbedVideoJsonBuilder;
|
||||||
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.prepareAndroidMobileJsonBuilder;
|
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.prepareAndroidMobileJsonBuilder;
|
||||||
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.prepareDesktopEmbedVideoJsonBuilder;
|
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.prepareDesktopEmbedVideoJsonBuilder;
|
||||||
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.prepareDesktopJsonBuilder;
|
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.prepareDesktopJsonBuilder;
|
||||||
|
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.prepareIosMobileEmbedVideoJsonBuilder;
|
||||||
|
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.prepareIosMobileJsonBuilder;
|
||||||
import static org.schabi.newpipe.extractor.utils.Utils.EMPTY_STRING;
|
import static org.schabi.newpipe.extractor.utils.Utils.EMPTY_STRING;
|
||||||
import static org.schabi.newpipe.extractor.utils.Utils.UTF_8;
|
import static org.schabi.newpipe.extractor.utils.Utils.UTF_8;
|
||||||
import static org.schabi.newpipe.extractor.utils.Utils.isNullOrEmpty;
|
import static org.schabi.newpipe.extractor.utils.Utils.isNullOrEmpty;
|
||||||
|
|
Loading…
Reference in a new issue