Skip to content

Commit

Permalink
Add ability to supply own PO tokens (Mithronn#46)
Browse files Browse the repository at this point in the history
  • Loading branch information
nick42d committed Oct 8, 2024
1 parent 102bb71 commit b3c1c7c
Show file tree
Hide file tree
Showing 4 changed files with 22 additions and 31 deletions.
2 changes: 0 additions & 2 deletions src/constants.rs
Original file line number Diff line number Diff line change
Expand Up @@ -52,8 +52,6 @@ pub(crate) const DEFAULT_DL_CHUNK_SIZE: u64 = 10485760;
/// Default max number of retries for a web reqwest.
pub(crate) const DEFAULT_MAX_RETRIES: u32 = 3;

pub(crate) const POTOKEN_EXPERIMENTS: &[&str] = &["51217476", "51217102"];

pub static INNERTUBE_CLIENT: Lazy<HashMap<&str, (&str, &str, &str)>> =
// (clientVersion, clientName, json value)
Lazy::new(|| {
Expand Down
23 changes: 18 additions & 5 deletions src/info.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ use reqwest::{
use reqwest_middleware::{ClientBuilder, ClientWithMiddleware};
use reqwest_retry::{policies::ExponentialBackoff, RetryTransientMiddleware};
use scraper::{Html, Selector};
use serde_json::json;
use std::{
borrow::{Borrow, Cow},
path::Path,
Expand All @@ -27,9 +28,9 @@ use crate::{
CustomRetryableStrategy, PlayerResponse, VideoError, VideoInfo, VideoOptions, YTConfig,
},
utils::{
between, check_experiments, choose_format, clean_video_details, get_functions, get_html,
get_html5player, get_random_v6_ip, get_video_id, get_ytconfig, is_age_restricted_from_html,
is_live, is_not_yet_broadcasted, is_play_error, is_player_response_error, is_private_video,
between, choose_format, clean_video_details, get_functions, get_html, get_html5player,
get_random_v6_ip, get_video_id, get_ytconfig, is_age_restricted_from_html, is_live,
is_not_yet_broadcasted, is_play_error, is_player_response_error, is_private_video,
is_rental, parse_live_video_formats, parse_video_formats, sort_formats,
},
};
Expand Down Expand Up @@ -197,11 +198,12 @@ impl<'opts> Video<'opts> {

// POToken experiment detected fallback to ios client (Webpage contains broken
// formats)
if check_experiments(&response) && !is_live(&player_response) {
if !is_live(&player_response) {
let ios_ytconfig = self
.get_player_ytconfig(
&response,
INNERTUBE_CLIENT.get("ios").cloned().unwrap_or_default(),
self.options.request_options.po_token.as_ref(),
)
.await?;

Expand All @@ -219,6 +221,7 @@ impl<'opts> Video<'opts> {
.get("tv_embedded")
.cloned()
.unwrap_or_default(),
self.options.request_options.po_token.as_ref(),
)
.await?;

Expand Down Expand Up @@ -542,6 +545,7 @@ impl<'opts> Video<'opts> {
&self,
html: &str,
configs: (&str, &str, &str),
po_token: Option<&String>,
) -> Result<String, VideoError> {
use std::str::FromStr;

Expand All @@ -551,7 +555,7 @@ impl<'opts> Video<'opts> {
let sts = ytcfg.sts.unwrap_or(0);
let video_id = self.get_video_id();

let query = serde_json::from_str::<serde_json::Value>(&format!(
let mut query = serde_json::from_str::<serde_json::Value>(&format!(
r#"{{
{client}
"playbackContext": {{
Expand All @@ -564,6 +568,15 @@ impl<'opts> Video<'opts> {
}}"#
))
.unwrap_or_default();
if let Some(po_token) = po_token {
query
.as_object_mut()
.expect("Declared as object above")
.insert(
"serviceIntegrityDimensions".to_string(),
json!({"poToken": po_token}),
);
}

static CONFIGS: Lazy<(HeaderMap, &str)> = Lazy::new(|| {
(HeaderMap::from_iter([
Expand Down
3 changes: 3 additions & 0 deletions src/structs.rs
Original file line number Diff line number Diff line change
Expand Up @@ -248,6 +248,9 @@ pub struct RequestOptions {
/// };
/// ```
pub max_retries: Option<u32>,
/// Supply a YouTube Proof of Origin token. Use at your own risk.
/// See https://github.com/yt-dlp/yt-dlp/wiki/Extractors#po-token-guide for more information.
pub po_token: Option<String>,
}

#[derive(thiserror::Error, Debug)]
Expand Down
25 changes: 1 addition & 24 deletions src/utils.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ use urlencoding::decode;
use crate::{
constants::{
AGE_RESTRICTED_URLS, AUDIO_ENCODING_RANKS, BASE_URL, FORMATS, IPV6_REGEX, PARSE_INT_REGEX,
POTOKEN_EXPERIMENTS, VALID_QUERY_DOMAINS, VIDEO_ENCODING_RANKS,
VALID_QUERY_DOMAINS, VIDEO_ENCODING_RANKS,
},
info_extras::{get_author, get_chapters, get_dislikes, get_likes, get_storyboards},
structs::{
Expand Down Expand Up @@ -996,29 +996,6 @@ pub fn get_ytconfig(html: &str) -> Result<YTConfig, VideoError> {
}
}

pub fn check_experiments(html: &str) -> bool {
if let Some(configs) = get_ytconfig(html)
.ok()
.and_then(|x| x.web_player_context_configs.clone())
.and_then(|v| v.as_object().cloned())
{
return configs.iter().any(|(_, config)| {
config
.get("serializedExperimentIds")
.and_then(|v| v.as_str())
.map(|ids| {
let ids_set = ids.split(',').collect::<Vec<&str>>();
POTOKEN_EXPERIMENTS
.iter()
.any(|token| ids_set.contains(token))
})
.unwrap_or(false)
});
}

false
}

type CacheFunctions = Lazy<RwLock<Option<(String, Vec<(String, String)>)>>>;
static FUNCTIONS: CacheFunctions = Lazy::new(|| RwLock::new(None));

Expand Down

0 comments on commit b3c1c7c

Please sign in to comment.