Compare commits

...

3 commits

48 changed files with 27834 additions and 100 deletions

View file

@ -22,6 +22,8 @@ pub async fn download_testfiles(project_root: &Path) {
comments_latest(&testfiles).await; comments_latest(&testfiles).await;
recommendations(&testfiles).await; recommendations(&testfiles).await;
channel_videos(&testfiles).await; channel_videos(&testfiles).await;
channel_shorts(&testfiles).await;
channel_livestreams(&testfiles).await;
channel_playlists(&testfiles).await; channel_playlists(&testfiles).await;
channel_info(&testfiles).await; channel_info(&testfiles).await;
channel_videos_cont(&testfiles).await; channel_videos_cont(&testfiles).await;
@ -258,6 +260,36 @@ async fn channel_videos(testfiles: &Path) {
} }
} }
async fn channel_shorts(testfiles: &Path) {
let mut json_path = testfiles.to_path_buf();
json_path.push("channel");
json_path.push("channel_shorts.json");
if json_path.exists() {
return;
}
let rp = rp_testfile(&json_path);
rp.query()
.channel_shorts("UCh8gHdtzO2tXd593_bjErWg")
.await
.unwrap();
}
async fn channel_livestreams(testfiles: &Path) {
let mut json_path = testfiles.to_path_buf();
json_path.push("channel");
json_path.push("channel_livestreams.json");
if json_path.exists() {
return;
}
let rp = rp_testfile(&json_path);
rp.query()
.channel_livestreams("UC2DjFE7Xf11URZqWBigcVOQ")
.await
.unwrap();
}
async fn channel_playlists(testfiles: &Path) { async fn channel_playlists(testfiles: &Path) {
let mut json_path = testfiles.to_path_buf(); let mut json_path = testfiles.to_path_buf();
json_path.push("channel"); json_path.push("channel");

View file

@ -13,7 +13,10 @@ use crate::{
util::{self, TryRemove}, util::{self, TryRemove},
}; };
use super::{response, ClientType, MapResponse, RustyPipeQuery, YTContext}; use super::{
response::{self, channel::ChannelContent},
ClientType, MapResponse, RustyPipeQuery, YTContext,
};
#[derive(Debug, Serialize)] #[derive(Debug, Serialize)]
#[serde(rename_all = "camelCase")] #[serde(rename_all = "camelCase")]
@ -27,6 +30,10 @@ struct QChannel<'a> {
enum Params { enum Params {
#[serde(rename = "EgZ2aWRlb3PyBgQKAjoA")] #[serde(rename = "EgZ2aWRlb3PyBgQKAjoA")]
Videos, Videos,
#[serde(rename = "EgZzaG9ydHPyBgUKA5oBAA%3D%3D")]
Shorts,
#[serde(rename = "EgdzdHJlYW1z8gYECgJ6AA%3D%3D")]
Live,
#[serde(rename = "EglwbGF5bGlzdHMgAQ%3D%3D")] #[serde(rename = "EglwbGF5bGlzdHMgAQ%3D%3D")]
Playlists, Playlists,
#[serde(rename = "EgVhYm91dPIGBAoCEgA%3D")] #[serde(rename = "EgVhYm91dPIGBAoCEgA%3D")]
@ -34,20 +41,22 @@ enum Params {
} }
impl RustyPipeQuery { impl RustyPipeQuery {
pub async fn channel_videos( async fn _channel_videos(
&self, &self,
channel_id: &str, channel_id: &str,
params: Params,
operation: &str,
) -> Result<Channel<Paginator<VideoItem>>, Error> { ) -> Result<Channel<Paginator<VideoItem>>, Error> {
let context = self.get_context(ClientType::Desktop, true, None).await; let context = self.get_context(ClientType::Desktop, true, None).await;
let request_body = QChannel { let request_body = QChannel {
context, context,
browse_id: channel_id, browse_id: channel_id,
params: Params::Videos, params,
}; };
self.execute_request::<response::Channel, _, _>( self.execute_request::<response::Channel, _, _>(
ClientType::Desktop, ClientType::Desktop,
"channel_videos", operation,
channel_id, channel_id,
"browse", "browse",
&request_body, &request_body,
@ -55,6 +64,30 @@ impl RustyPipeQuery {
.await .await
} }
pub async fn channel_videos(
&self,
channel_id: &str,
) -> Result<Channel<Paginator<VideoItem>>, Error> {
self._channel_videos(channel_id, Params::Videos, "channel_videos")
.await
}
pub async fn channel_shorts(
&self,
channel_id: &str,
) -> Result<Channel<Paginator<VideoItem>>, Error> {
self._channel_videos(channel_id, Params::Shorts, "channel_shorts")
.await
}
pub async fn channel_livestreams(
&self,
channel_id: &str,
) -> Result<Channel<Paginator<VideoItem>>, Error> {
self._channel_videos(channel_id, Params::Live, "channel_livestreams")
.await
}
pub async fn channel_playlists( pub async fn channel_playlists(
&self, &self,
channel_id: &str, channel_id: &str,
@ -102,8 +135,8 @@ impl MapResponse<Channel<Paginator<VideoItem>>> for response::Channel {
lang: Language, lang: Language,
_deobf: Option<&crate::deobfuscate::Deobfuscator>, _deobf: Option<&crate::deobfuscate::Deobfuscator>,
) -> Result<MapResult<Channel<Paginator<VideoItem>>>, ExtractionError> { ) -> Result<MapResult<Channel<Paginator<VideoItem>>>, ExtractionError> {
let content = map_channel_content(self.contents, id, self.alerts)?; let content = map_channel_content(self.contents, self.alerts)?;
let grid = match content { let grid = match content.content {
response::channel::ChannelContent::GridRenderer { items } => Some(items), response::channel::ChannelContent::GridRenderer { items } => Some(items),
_ => None, _ => None,
}; };
@ -112,10 +145,15 @@ impl MapResponse<Channel<Paginator<VideoItem>>> for response::Channel {
Ok(MapResult { Ok(MapResult {
c: map_channel( c: map_channel(
self.header, MapChannelData {
self.metadata, header: self.header,
self.microformat, metadata: self.metadata,
v_res.c, microformat: self.microformat,
visitor_data: self.response_context.visitor_data,
has_shorts: content.has_shorts,
has_live: content.has_live,
content: v_res.c,
},
id, id,
lang, lang,
)?, )?,
@ -131,8 +169,8 @@ impl MapResponse<Channel<Paginator<PlaylistItem>>> for response::Channel {
lang: Language, lang: Language,
_deobf: Option<&crate::deobfuscate::Deobfuscator>, _deobf: Option<&crate::deobfuscate::Deobfuscator>,
) -> Result<MapResult<Channel<Paginator<PlaylistItem>>>, ExtractionError> { ) -> Result<MapResult<Channel<Paginator<PlaylistItem>>>, ExtractionError> {
let content = map_channel_content(self.contents, id, self.alerts)?; let content = map_channel_content(self.contents, self.alerts)?;
let grid = match content { let grid = match content.content {
response::channel::ChannelContent::GridRenderer { items } => Some(items), response::channel::ChannelContent::GridRenderer { items } => Some(items),
_ => None, _ => None,
}; };
@ -143,10 +181,15 @@ impl MapResponse<Channel<Paginator<PlaylistItem>>> for response::Channel {
Ok(MapResult { Ok(MapResult {
c: map_channel( c: map_channel(
self.header, MapChannelData {
self.metadata, header: self.header,
self.microformat, metadata: self.metadata,
p_res.c, microformat: self.microformat,
visitor_data: self.response_context.visitor_data,
has_shorts: content.has_shorts,
has_live: content.has_live,
content: p_res.c,
},
id, id,
lang, lang,
)?, )?,
@ -162,9 +205,9 @@ impl MapResponse<Channel<ChannelInfo>> for response::Channel {
lang: Language, lang: Language,
_deobf: Option<&crate::deobfuscate::Deobfuscator>, _deobf: Option<&crate::deobfuscate::Deobfuscator>,
) -> Result<MapResult<Channel<ChannelInfo>>, ExtractionError> { ) -> Result<MapResult<Channel<ChannelInfo>>, ExtractionError> {
let content = map_channel_content(self.contents, id, self.alerts)?; let content = map_channel_content(self.contents, self.alerts)?;
let mut warnings = Vec::new(); let mut warnings = Vec::new();
let meta = match content { let meta = match content.content {
response::channel::ChannelContent::ChannelAboutFullMetadataRenderer(meta) => Some(meta), response::channel::ChannelContent::ChannelAboutFullMetadataRenderer(meta) => Some(meta),
_ => None, _ => None,
}; };
@ -201,10 +244,15 @@ impl MapResponse<Channel<ChannelInfo>> for response::Channel {
Ok(MapResult { Ok(MapResult {
c: map_channel( c: map_channel(
self.header, MapChannelData {
self.metadata, header: self.header,
self.microformat, metadata: self.metadata,
cinfo, microformat: self.microformat,
visitor_data: self.response_context.visitor_data,
has_shorts: content.has_shorts,
has_live: content.has_live,
content: cinfo,
},
id, id,
lang, lang,
)?, )?,
@ -251,25 +299,37 @@ fn map_vanity_url(url: &str, id: &str) -> Option<String> {
}) })
} }
fn map_channel<T>( struct MapChannelData<T> {
header: Option<response::channel::Header>, header: Option<response::channel::Header>,
metadata: Option<response::channel::Metadata>, metadata: Option<response::channel::Metadata>,
microformat: Option<response::channel::Microformat>, microformat: Option<response::channel::Microformat>,
visitor_data: Option<String>,
has_shorts: bool,
has_live: bool,
content: T, content: T,
}
fn map_channel<T>(
d: MapChannelData<T>,
id: &str, id: &str,
lang: Language, lang: Language,
) -> Result<Channel<T>, ExtractionError> { ) -> Result<Channel<T>, ExtractionError> {
let header = header.ok_or(ExtractionError::ContentUnavailable(Cow::Borrowed( let header = d
"channel not found", .header
)))?; .ok_or(ExtractionError::ContentUnavailable(Cow::Borrowed(
let metadata = metadata "channel not found",
)))?;
let metadata = d
.metadata
.ok_or(ExtractionError::ContentUnavailable(Cow::Borrowed( .ok_or(ExtractionError::ContentUnavailable(Cow::Borrowed(
"channel not found", "channel not found",
)))? )))?
.channel_metadata_renderer; .channel_metadata_renderer;
let microformat = microformat.ok_or(ExtractionError::ContentUnavailable(Cow::Borrowed( let microformat = d
"channel not found", .microformat
)))?; .ok_or(ExtractionError::ContentUnavailable(Cow::Borrowed(
"channel not found",
)))?;
if metadata.external_id != id { if metadata.external_id != id {
return Err(ExtractionError::WrongResult(format!( return Err(ExtractionError::WrongResult(format!(
@ -298,7 +358,10 @@ fn map_channel<T>(
banner: header.banner.into(), banner: header.banner.into(),
mobile_banner: header.mobile_banner.into(), mobile_banner: header.mobile_banner.into(),
tv_banner: header.tv_banner.into(), tv_banner: header.tv_banner.into(),
content, has_shorts: d.has_shorts,
has_live: d.has_live,
visitor_data: d.visitor_data,
content: d.content,
}, },
response::channel::Header::CarouselHeaderRenderer(carousel) => { response::channel::Header::CarouselHeaderRenderer(carousel) => {
let hdata = carousel let hdata = carousel
@ -332,17 +395,25 @@ fn map_channel<T>(
banner: Vec::new(), banner: Vec::new(),
mobile_banner: Vec::new(), mobile_banner: Vec::new(),
tv_banner: Vec::new(), tv_banner: Vec::new(),
content, has_shorts: d.has_shorts,
has_live: d.has_live,
visitor_data: d.visitor_data,
content: d.content,
} }
} }
}) })
} }
struct MappedChannelContent {
content: response::channel::ChannelContent,
has_shorts: bool,
has_live: bool,
}
fn map_channel_content( fn map_channel_content(
contents: Option<response::channel::Contents>, contents: Option<response::channel::Contents>,
id: &str,
alerts: Option<Vec<response::Alert>>, alerts: Option<Vec<response::Alert>>,
) -> Result<response::channel::ChannelContent, ExtractionError> { ) -> Result<MappedChannelContent, ExtractionError> {
match contents { match contents {
Some(contents) => { Some(contents) => {
let tabs = contents.two_column_browse_results_renderer.tabs; let tabs = contents.two_column_browse_results_renderer.tabs;
@ -352,42 +423,70 @@ fn map_channel_content(
)); ));
} }
let (channel_content, target_id) = tabs let cmp_url_suffix = |endpoint: &response::channel::ChannelTabEndpoint,
.into_iter() expect: &str| {
.filter_map(|tab| { endpoint
let content = tab.tab_renderer.content; .command_metadata
match (content.section_list_renderer, content.rich_grid_renderer) { .web_command_metadata
(Some(mut section_list_renderer), _) => { .url
let content = .ends_with(expect)
section_list_renderer.contents.try_swap_remove(0).and_then( };
|mut i| i.item_section_renderer.contents.try_swap_remove(0),
);
content.map(|c| (c, section_list_renderer.target_id)) let mut has_shorts = false;
} let mut has_live = false;
(None, Some(rich_grid_renderer)) => Some(( let mut featured_tab = false;
response::channel::ChannelContent::GridRenderer {
items: rich_grid_renderer.contents,
},
rich_grid_renderer.target_id,
)),
(None, None) => None,
}
})
.next()
.ok_or(ExtractionError::InvalidData(Cow::Borrowed(
"could not extract content",
)))?;
if let Some(target_id) = target_id { for tab in &tabs {
// YouTube falls back to the featured page if the channel does not have a "videos" tab. if cmp_url_suffix(&tab.tab_renderer.endpoint, "/featured")
// This is the case for YouTube Music channels. && (tab.tab_renderer.content.section_list_renderer.is_some()
if target_id.starts_with(&format!("browse-feed{}featured", id)) { || tab.tab_renderer.content.rich_grid_renderer.is_some())
return Ok(response::channel::ChannelContent::None); {
featured_tab = true;
} else if cmp_url_suffix(&tab.tab_renderer.endpoint, "/shorts") {
has_shorts = true;
} else if cmp_url_suffix(&tab.tab_renderer.endpoint, "/streams") {
has_live = true;
} }
} }
Ok(channel_content) let channel_content = tabs
.into_iter()
.filter_map(|tab| {
let content = tab.tab_renderer.content;
match (content.rich_grid_renderer, content.section_list_renderer) {
(Some(rich_grid), _) => Some(ChannelContent::GridRenderer {
items: rich_grid.contents,
}),
(None, Some(section_list)) => {
let mut contents = section_list.contents;
contents.try_swap_remove(0).and_then(|mut i| {
i.item_section_renderer.contents.try_swap_remove(0)
})
}
(None, None) => None,
}
})
.next();
let content = match channel_content {
Some(content) => content,
None => {
// YouTube may show the "Featured" tab if the requested tab is empty/does not exist
if featured_tab {
response::channel::ChannelContent::None
} else {
return Err(ExtractionError::InvalidData(Cow::Borrowed(
"could not extract content",
)));
}
}
};
Ok(MappedChannelContent {
content,
has_shorts,
has_live,
})
} }
None => Err(response::alerts_to_err(alerts)), None => Err(response::alerts_to_err(alerts)),
} }
@ -407,16 +506,18 @@ mod tests {
}; };
#[rstest] #[rstest]
#[case::base("base", "UC2DjFE7Xf11URZqWBigcVOQ")] #[case::base("videos_base", "UC2DjFE7Xf11URZqWBigcVOQ")]
#[case::music("music", "UC_vmjW5e1xEHhYjY2a0kK1A")] #[case::music("videos_music", "UC_vmjW5e1xEHhYjY2a0kK1A")]
#[case::withshorts("videos_shorts", "UCh8gHdtzO2tXd593_bjErWg")]
#[case::live("videos_live", "UChs0pSaEoNLV4mevBFGaoKA")]
#[case::empty("videos_empty", "UCxBa895m48H5idw5li7h-0g")]
#[case::upcoming("videos_upcoming", "UCcvfHa-GHSOHFAjU0-Ie57A")]
#[case::richgrid("videos_20221011_richgrid", "UCh8gHdtzO2tXd593_bjErWg")]
#[case::richgrid2("videos_20221011_richgrid2", "UC2DjFE7Xf11URZqWBigcVOQ")]
#[case::shorts("shorts", "UCh8gHdtzO2tXd593_bjErWg")] #[case::shorts("shorts", "UCh8gHdtzO2tXd593_bjErWg")]
#[case::live("live", "UChs0pSaEoNLV4mevBFGaoKA")] #[case::livestreams("livestreams", "UC2DjFE7Xf11URZqWBigcVOQ")]
#[case::empty("empty", "UCxBa895m48H5idw5li7h-0g")]
#[case::upcoming("upcoming", "UCcvfHa-GHSOHFAjU0-Ie57A")]
#[case::richgrid("20221011_richgrid", "UCh8gHdtzO2tXd593_bjErWg")]
#[case::richgrid2("20221011_richgrid2", "UC2DjFE7Xf11URZqWBigcVOQ")]
fn map_channel_videos(#[case] name: &str, #[case] id: &str) { fn map_channel_videos(#[case] name: &str, #[case] id: &str) {
let filename = format!("testfiles/channel/channel_videos_{}.json", name); let filename = format!("testfiles/channel/channel_{}.json", name);
let json_path = Path::new(&filename); let json_path = Path::new(&filename);
let json_file = File::open(json_path).unwrap(); let json_file = File::open(json_path).unwrap();
@ -431,12 +532,12 @@ mod tests {
map_res.warnings map_res.warnings
); );
if name == "upcoming" { if name == "videos_upcoming" {
insta::assert_ron_snapshot!(format!("map_channel_videos_{}", name), map_res.c, { insta::assert_ron_snapshot!(format!("map_channel_{}", name), map_res.c, {
".content.items[1:].publish_date" => "[date]", ".content.items[1:].publish_date" => "[date]",
}); });
} else { } else {
insta::assert_ron_snapshot!(format!("map_channel_videos_{}", name), map_res.c, { insta::assert_ron_snapshot!(format!("map_channel_{}", name), map_res.c, {
".content.items[].publish_date" => "[date]", ".content.items[].publish_date" => "[date]",
}); });
} }

View file

@ -275,6 +275,7 @@ impl MapResponse<VideoPlayer> for response::Player {
expires_in_seconds: streaming_data.expires_in_seconds, expires_in_seconds: streaming_data.expires_in_seconds,
hls_manifest_url: streaming_data.hls_manifest_url, hls_manifest_url: streaming_data.hls_manifest_url,
dash_manifest_url: streaming_data.dash_manifest_url, dash_manifest_url: streaming_data.dash_manifest_url,
visitor_data: self.response_context.visitor_data,
}, },
warnings, warnings,
}) })

View file

@ -169,6 +169,7 @@ impl MapResponse<Playlist> for response::Playlist {
channel, channel,
last_update, last_update,
last_update_txt, last_update_txt,
visitor_data: self.response_context.visitor_data,
}, },
warnings, warnings,
}) })

View file

@ -3,9 +3,7 @@ use serde_with::serde_as;
use serde_with::{DefaultOnError, VecSkipError}; use serde_with::{DefaultOnError, VecSkipError};
use super::url_endpoint::NavigationEndpoint; use super::url_endpoint::NavigationEndpoint;
use super::{Alert, ChannelBadge}; use super::{Alert, ChannelBadge, ContentsRenderer, ResponseContext, Thumbnails, YouTubeListItem};
use super::{ContentRenderer, ContentsRenderer};
use super::{Thumbnails, YouTubeListItem};
use crate::serializer::ignore_any; use crate::serializer::ignore_any;
use crate::serializer::{text::Text, MapResult, VecLogError}; use crate::serializer::{text::Text, MapResult, VecLogError};
@ -21,6 +19,7 @@ pub(crate) struct Channel {
pub microformat: Option<Microformat>, pub microformat: Option<Microformat>,
#[serde_as(as = "Option<DefaultOnError>")] #[serde_as(as = "Option<DefaultOnError>")]
pub alerts: Option<Vec<Alert>>, pub alerts: Option<Vec<Alert>>,
pub response_context: ResponseContext,
} }
#[derive(Debug, Deserialize)] #[derive(Debug, Deserialize)]
@ -42,11 +41,19 @@ pub(crate) struct TabsRenderer {
#[derive(Debug, Deserialize)] #[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")] #[serde(rename_all = "camelCase")]
pub(crate) struct TabRendererWrap { pub(crate) struct TabRendererWrap {
pub tab_renderer: ContentRenderer<TabContent>, pub tab_renderer: TabRenderer,
}
#[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")]
pub(crate) struct TabRenderer {
#[serde(default)]
pub content: TabContent,
pub endpoint: ChannelTabEndpoint,
} }
#[serde_as] #[serde_as]
#[derive(Debug, Deserialize)] #[derive(Default, Debug, Deserialize)]
#[serde(rename_all = "camelCase")] #[serde(rename_all = "camelCase")]
pub(crate) struct TabContent { pub(crate) struct TabContent {
#[serde(default)] #[serde(default)]
@ -58,14 +65,28 @@ pub(crate) struct TabContent {
pub rich_grid_renderer: Option<RichGridRenderer>, pub rich_grid_renderer: Option<RichGridRenderer>,
} }
#[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")]
pub(crate) struct ChannelTabEndpoint {
pub command_metadata: ChannelTabCommandMetadata,
}
#[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")]
pub(crate) struct ChannelTabCommandMetadata {
pub web_command_metadata: ChannelTabWebCommandMetadata,
}
#[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")]
pub(crate) struct ChannelTabWebCommandMetadata {
pub url: String,
}
#[derive(Debug, Deserialize)] #[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")] #[serde(rename_all = "camelCase")]
pub(crate) struct SectionListRenderer { pub(crate) struct SectionListRenderer {
pub contents: Vec<ItemSectionRendererWrap>, pub contents: Vec<ItemSectionRendererWrap>,
/// - **Videos**: browse-feedUC2DjFE7Xf11URZqWBigcVOQvideos (...)
/// - **Playlists**: browse-feedUC2DjFE7Xf11URZqWBigcVOQplaylists104 (...)
/// - **Info**: None
pub target_id: Option<String>,
} }
/// Seems to be currently A/B tested, as of 11.10.2022 /// Seems to be currently A/B tested, as of 11.10.2022
@ -75,10 +96,6 @@ pub(crate) struct SectionListRenderer {
pub(crate) struct RichGridRenderer { pub(crate) struct RichGridRenderer {
#[serde_as(as = "VecLogError<_>")] #[serde_as(as = "VecLogError<_>")]
pub contents: MapResult<Vec<YouTubeListItem>>, pub contents: MapResult<Vec<YouTubeListItem>>,
/// - **Videos**: browse-feedUC2DjFE7Xf11URZqWBigcVOQvideos (...)
/// - **Playlists**: browse-feedUC2DjFE7Xf11URZqWBigcVOQplaylists104 (...)
/// - **Info**: None
pub target_id: Option<String>,
} }
#[derive(Debug, Deserialize)] #[derive(Debug, Deserialize)]

View file

@ -4,7 +4,7 @@ use serde::Deserialize;
use serde_with::serde_as; use serde_with::serde_as;
use serde_with::{json::JsonString, DefaultOnError}; use serde_with::{json::JsonString, DefaultOnError};
use super::Thumbnails; use super::{ResponseContext, Thumbnails};
use crate::serializer::{text::Text, MapResult, VecLogError}; use crate::serializer::{text::Text, MapResult, VecLogError};
#[derive(Debug, Deserialize)] #[derive(Debug, Deserialize)]
@ -14,6 +14,7 @@ pub(crate) struct Player {
pub streaming_data: Option<StreamingData>, pub streaming_data: Option<StreamingData>,
pub captions: Option<Captions>, pub captions: Option<Captions>,
pub video_details: Option<VideoDetails>, pub video_details: Option<VideoDetails>,
pub response_context: ResponseContext,
} }
#[derive(Debug, Deserialize)] #[derive(Debug, Deserialize)]

View file

@ -6,7 +6,8 @@ use crate::serializer::{ignore_any, MapResult, VecLogError};
use crate::util::MappingError; use crate::util::MappingError;
use super::{ use super::{
Alert, ContentRenderer, ContentsRenderer, ContinuationEndpoint, Thumbnails, ThumbnailsWrap, Alert, ContentRenderer, ContentsRenderer, ContinuationEndpoint, ResponseContext, Thumbnails,
ThumbnailsWrap,
}; };
#[serde_as] #[serde_as]
@ -18,6 +19,7 @@ pub(crate) struct Playlist {
pub sidebar: Option<Sidebar>, pub sidebar: Option<Sidebar>,
#[serde_as(as = "Option<DefaultOnError>")] #[serde_as(as = "Option<DefaultOnError>")]
pub alerts: Option<Vec<Alert>>, pub alerts: Option<Vec<Alert>>,
pub response_context: ResponseContext,
} }
#[serde_as] #[serde_as]

View file

@ -1,7 +1,7 @@
use serde::Deserialize; use serde::Deserialize;
use serde_with::{json::JsonString, serde_as}; use serde_with::{json::JsonString, serde_as};
use super::video_item::YouTubeListRendererWrap; use super::{video_item::YouTubeListRendererWrap, ResponseContext};
#[serde_as] #[serde_as]
#[derive(Debug, Deserialize)] #[derive(Debug, Deserialize)]
@ -10,6 +10,7 @@ pub(crate) struct Search {
#[serde_as(as = "Option<JsonString>")] #[serde_as(as = "Option<JsonString>")]
pub estimated_results: Option<u64>, pub estimated_results: Option<u64>,
pub contents: Contents, pub contents: Contents,
pub response_context: ResponseContext,
} }
#[derive(Debug, Deserialize)] #[derive(Debug, Deserialize)]

View file

@ -1,6 +1,8 @@
use fancy_regex::Regex;
use once_cell::sync::Lazy;
use serde::Deserialize; use serde::Deserialize;
use serde_with::{json::JsonString, serde_as, DefaultOnError, VecSkipError}; use serde_with::{json::JsonString, serde_as, DefaultOnError, VecSkipError};
use time::OffsetDateTime; use time::{Duration, OffsetDateTime};
use super::{ChannelBadge, ContinuationEndpoint, Thumbnails}; use super::{ChannelBadge, ContinuationEndpoint, Thumbnails};
use crate::{ use crate::{
@ -8,7 +10,7 @@ use crate::{
param::Language, param::Language,
serializer::{ serializer::{
ignore_any, ignore_any,
text::{Text, TextComponent}, text::{AccessibilityText, Text, TextComponent},
MapResult, VecLogError, MapResult, VecLogError,
}, },
timeago, timeago,
@ -21,6 +23,7 @@ use crate::{
pub(crate) enum YouTubeListItem { pub(crate) enum YouTubeListItem {
#[serde(alias = "gridVideoRenderer", alias = "compactVideoRenderer")] #[serde(alias = "gridVideoRenderer", alias = "compactVideoRenderer")]
VideoRenderer(VideoRenderer), VideoRenderer(VideoRenderer),
ReelItemRenderer(ReelItemRenderer),
#[serde(alias = "gridPlaylistRenderer")] #[serde(alias = "gridPlaylistRenderer")]
PlaylistRenderer(PlaylistRenderer), PlaylistRenderer(PlaylistRenderer),
@ -98,6 +101,7 @@ pub(crate) struct VideoRenderer {
#[serde_as(as = "VecSkipError<_>")] #[serde_as(as = "VecSkipError<_>")]
pub badges: Vec<VideoBadge>, pub badges: Vec<VideoBadge>,
/// Contains Short/Live tag /// Contains Short/Live tag
#[serde(default)]
#[serde_as(as = "VecSkipError<_>")] #[serde_as(as = "VecSkipError<_>")]
pub thumbnail_overlays: Vec<TimeOverlay>, pub thumbnail_overlays: Vec<TimeOverlay>,
/// Abbreviated video description (on startpage) /// Abbreviated video description (on startpage)
@ -110,6 +114,27 @@ pub(crate) struct VideoRenderer {
pub upcoming_event_data: Option<UpcomingEventData>, pub upcoming_event_data: Option<UpcomingEventData>,
} }
/// Short video item
#[serde_as]
#[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")]
pub(crate) struct ReelItemRenderer {
pub video_id: String,
pub thumbnail: Thumbnails,
#[serde_as(as = "Text")]
pub headline: String,
/// Contains `No views` if the view count is zero
#[serde_as(as = "Option<Text>")]
pub view_count_text: Option<String>,
/// video duration
///
/// Example: `the horror maze - 44 seconds - play video`
///
/// Dashes may be `\u2013` (emdash)
#[serde_as(as = "Option<AccessibilityText>")]
pub accessibility: Option<String>,
}
/// Playlist displayed in search results /// Playlist displayed in search results
#[serde_as] #[serde_as]
#[derive(Debug, Deserialize)] #[derive(Debug, Deserialize)]
@ -363,6 +388,39 @@ impl<T> YouTubeListMapper<T> {
} }
} }
fn map_short_video(&self, video: ReelItemRenderer) -> VideoItem {
static ACCESSIBILITY_SEP_REGEX: Lazy<Regex> =
Lazy::new(|| Regex::new(" [-\u{2013}] (.+) [-\u{2013}] ").unwrap());
VideoItem {
id: video.video_id,
title: video.headline,
length: video.accessibility.and_then(|acc| {
ACCESSIBILITY_SEP_REGEX
.captures(&acc)
.ok()
.flatten()
.and_then(|cap| {
cap.get(1).and_then(|c| {
timeago::parse_timeago(self.lang, c.as_str())
.map(|ta| Duration::from(ta).whole_seconds() as u32)
})
})
}),
thumbnail: video.thumbnail.into(),
channel: None,
publish_date: None,
publish_date_txt: None,
view_count: video
.view_count_text
.map(|txt| util::parse_numeric(&txt).unwrap_or_default()),
is_live: false,
is_short: true,
is_upcoming: false,
short_description: None,
}
}
fn map_playlist(playlist: PlaylistRenderer) -> PlaylistItem { fn map_playlist(playlist: PlaylistRenderer) -> PlaylistItem {
PlaylistItem { PlaylistItem {
id: playlist.playlist_id, id: playlist.playlist_id,
@ -413,6 +471,10 @@ impl YouTubeListMapper<YouTubeItem> {
YouTubeListItem::VideoRenderer(video) => { YouTubeListItem::VideoRenderer(video) => {
self.items.push(YouTubeItem::Video(self.map_video(video))); self.items.push(YouTubeItem::Video(self.map_video(video)));
} }
YouTubeListItem::ReelItemRenderer(video) => {
self.items
.push(YouTubeItem::Video(self.map_short_video(video)));
}
YouTubeListItem::PlaylistRenderer(playlist) => self YouTubeListItem::PlaylistRenderer(playlist) => self
.items .items
.push(YouTubeItem::Playlist(Self::map_playlist(playlist))), .push(YouTubeItem::Playlist(Self::map_playlist(playlist))),
@ -449,6 +511,9 @@ impl YouTubeListMapper<VideoItem> {
YouTubeListItem::VideoRenderer(video) => { YouTubeListItem::VideoRenderer(video) => {
self.items.push(self.map_video(video)); self.items.push(self.map_video(video));
} }
YouTubeListItem::ReelItemRenderer(video) => {
self.items.push(self.map_short_video(video));
}
YouTubeListItem::ContinuationItemRenderer { YouTubeListItem::ContinuationItemRenderer {
continuation_endpoint, continuation_endpoint,
} => self.ctoken = Some(continuation_endpoint.continuation_command.token), } => self.ctoken = Some(continuation_endpoint.continuation_command.token),

View file

@ -109,6 +109,7 @@ impl MapResponse<SearchResult> for response::Search {
c: SearchResult { c: SearchResult {
items: Paginator::new(self.estimated_results, mapper.items, mapper.ctoken), items: Paginator::new(self.estimated_results, mapper.items, mapper.ctoken),
corrected_query: mapper.corrected_query, corrected_query: mapper.corrected_query,
visitor_data: self.response_context.visitor_data,
}, },
warnings: mapper.warnings, warnings: mapper.warnings,
}) })

View file

@ -142,6 +142,9 @@ Channel(
height: 1192, height: 1192,
), ),
], ],
has_shorts: false,
has_live: false,
visitor_data: Some("CgszMUUzZDlGLWxiRSipqr2ZBg%3D%3D"),
content: ChannelInfo( content: ChannelInfo(
create_date: Some("2009-04-04"), create_date: Some("2009-04-04"),
view_count: Some(186854342), view_count: Some(186854342),

View file

@ -142,6 +142,9 @@ Channel(
height: 1192, height: 1192,
), ),
], ],
has_shorts: false,
has_live: false,
visitor_data: Some("CgttaWpyTVpUN1AyZyioqr2ZBg%3D%3D"),
content: Paginator( content: Paginator(
count: None, count: None,
items: [ items: [

File diff suppressed because it is too large Load diff

View file

@ -113,6 +113,9 @@ Channel(
height: 1192, height: 1192,
), ),
], ],
has_shorts: true,
has_live: false,
visitor_data: Some("CgtQdE9zVVR3NVBDbyjz0ZKaBg%3D%3D"),
content: Paginator( content: Paginator(
count: None, count: None,
items: [ items: [

View file

@ -142,6 +142,9 @@ Channel(
height: 1192, height: 1192,
), ),
], ],
has_shorts: false,
has_live: true,
visitor_data: Some("Cgs4ZFVmMzVlU1dxbyiBqpeaBg%3D%3D"),
content: Paginator( content: Paginator(
count: None, count: None,
items: [ items: [

View file

@ -142,6 +142,9 @@ Channel(
height: 1192, height: 1192,
), ),
], ],
has_shorts: false,
has_live: false,
visitor_data: Some("CgszNU5rbDVZS2hMcyim4K2ZBg%3D%3D"),
content: Paginator( content: Paginator(
count: None, count: None,
items: [ items: [

View file

@ -30,6 +30,9 @@ Channel(
banner: [], banner: [],
mobile_banner: [], mobile_banner: [],
tv_banner: [], tv_banner: [],
has_shorts: false,
has_live: false,
visitor_data: Some("Cgtvc2s4UllvTGl6byigxseZBg%3D%3D"),
content: Paginator( content: Paginator(
count: Some(0), count: Some(0),
items: [], items: [],

View file

@ -126,6 +126,9 @@ Channel(
height: 1192, height: 1192,
), ),
], ],
has_shorts: false,
has_live: false,
visitor_data: Some("CgtkYXJITElwYmd4OCj85a2ZBg%3D%3D"),
content: Paginator( content: Paginator(
count: Some(21), count: Some(21),
items: [ items: [

View file

@ -113,6 +113,9 @@ Channel(
height: 1192, height: 1192,
), ),
], ],
has_shorts: false,
has_live: false,
visitor_data: Some("CgtCV1l2R2Rzb2ZSZyiu4a2ZBg%3D%3D"),
content: Paginator( content: Paginator(
count: Some(0), count: Some(0),
items: [], items: [],

View file

@ -113,6 +113,9 @@ Channel(
height: 1192, height: 1192,
), ),
], ],
has_shorts: false,
has_live: false,
visitor_data: Some("CgtneXVRbGtSMWtlYyj75a2ZBg%3D%3D"),
content: Paginator( content: Paginator(
count: None, count: None,
items: [ items: [

View file

@ -130,6 +130,9 @@ Channel(
height: 1192, height: 1192,
), ),
], ],
has_shorts: false,
has_live: false,
visitor_data: Some("Cgs4Ri1tLW1KNWozNCjGk8yZBg%3D%3D"),
content: Paginator( content: Paginator(
count: None, count: None,
items: [ items: [

View file

@ -424,4 +424,5 @@ VideoPlayer(
expires_in_seconds: 21540, expires_in_seconds: 21540,
hls_manifest_url: None, hls_manifest_url: None,
dash_manifest_url: Some("https://manifest.googlevideo.com/api/manifest/dash/expire/1659481355/ei/q1jpYtOPEYSBgQeHmqbwAQ/ip/2003%3Ade%3Aaf0e%3A2f00%3Ade47%3A297%3Aa6db%3A774e/id/a4fbddf14c6649b4/source/youtube/requiressl/yes/playback_host/rr5---sn-h0jeenek.googlevideo.com/mh/mQ/mm/31%2C29/mn/sn-h0jeenek%2Csn-h0jelnez/ms/au%2Crdu/mv/m/mvi/5/pl/37/hfr/1/as/fmp4_audio_clear%2Cfmp4_sd_hd_clear/initcwndbps/1527500/vprv/1/mt/1659459429/fvip/4/itag_bl/376%2C377%2C384%2C385%2C612%2C613%2C617%2C619%2C623%2C628%2C655%2C656%2C660%2C662%2C666%2C671/keepalive/yes/fexp/24001373%2C24007246/itag/0/sparams/expire%2Cei%2Cip%2Cid%2Csource%2Crequiressl%2Chfr%2Cas%2Cvprv%2Citag/sig/AOq0QJ8wRAIgMm4a_MIHA3YUszKeruSy3exs5JwNjJAyLAwxL0yPdNMCIANb9GDMSTp_NT-PPhbvYMwRULJ5a9BO6MYD9FuWprC1/lsparams/playback_host%2Cmh%2Cmm%2Cmn%2Cms%2Cmv%2Cmvi%2Cpl%2Cinitcwndbps/lsig/AG3C_xAwRQIgETSOwhwWVMy7gmrFXZlJu655ToLzSwOEsT16oRyrWhACIQDkvOEw1fImz5omu4iVIRNFe-z-JC9v8WUyx281dW2NOw%3D%3D"), dash_manifest_url: Some("https://manifest.googlevideo.com/api/manifest/dash/expire/1659481355/ei/q1jpYtOPEYSBgQeHmqbwAQ/ip/2003%3Ade%3Aaf0e%3A2f00%3Ade47%3A297%3Aa6db%3A774e/id/a4fbddf14c6649b4/source/youtube/requiressl/yes/playback_host/rr5---sn-h0jeenek.googlevideo.com/mh/mQ/mm/31%2C29/mn/sn-h0jeenek%2Csn-h0jelnez/ms/au%2Crdu/mv/m/mvi/5/pl/37/hfr/1/as/fmp4_audio_clear%2Cfmp4_sd_hd_clear/initcwndbps/1527500/vprv/1/mt/1659459429/fvip/4/itag_bl/376%2C377%2C384%2C385%2C612%2C613%2C617%2C619%2C623%2C628%2C655%2C656%2C660%2C662%2C666%2C671/keepalive/yes/fexp/24001373%2C24007246/itag/0/sparams/expire%2Cei%2Cip%2Cid%2Csource%2Crequiressl%2Chfr%2Cas%2Cvprv%2Citag/sig/AOq0QJ8wRAIgMm4a_MIHA3YUszKeruSy3exs5JwNjJAyLAwxL0yPdNMCIANb9GDMSTp_NT-PPhbvYMwRULJ5a9BO6MYD9FuWprC1/lsparams/playback_host%2Cmh%2Cmm%2Cmn%2Cms%2Cmv%2Cmvi%2Cpl%2Cinitcwndbps/lsig/AG3C_xAwRQIgETSOwhwWVMy7gmrFXZlJu655ToLzSwOEsT16oRyrWhACIQDkvOEw1fImz5omu4iVIRNFe-z-JC9v8WUyx281dW2NOw%3D%3D"),
visitor_data: Some("Cgt2aHFtQU5YZFBvYyirsaWXBg%3D%3D"),
) )

View file

@ -541,4 +541,5 @@ VideoPlayer(
expires_in_seconds: 21540, expires_in_seconds: 21540,
hls_manifest_url: None, hls_manifest_url: None,
dash_manifest_url: Some("https://manifest.googlevideo.com/api/manifest/dash/expire/1659481355/ei/q1jpYtq3BJCX1gKVyJGQDg/ip/2003%3Ade%3Aaf0e%3A2f00%3Ade47%3A297%3Aa6db%3A774e/id/a4fbddf14c6649b4/source/youtube/requiressl/yes/playback_host/rr4---sn-h0jelnez.googlevideo.com/mh/mQ/mm/31%2C26/mn/sn-h0jelnez%2Csn-4g5edn6k/ms/au%2Conr/mv/m/mvi/4/pl/37/hfr/all/as/fmp4_audio_clear%2Cwebm_audio_clear%2Cwebm2_audio_clear%2Cfmp4_sd_hd_clear%2Cwebm2_sd_hd_clear/initcwndbps/1513750/spc/lT-KhrZGE2opztWyVdAtyUNlb8dXPDs/vprv/1/mt/1659459429/fvip/4/keepalive/yes/fexp/24001373%2C24007246/itag/0/sparams/expire%2Cei%2Cip%2Cid%2Csource%2Crequiressl%2Chfr%2Cas%2Cspc%2Cvprv%2Citag/sig/AOq0QJ8wRgIhAPEjHK19PKVHqQeia6WF4qubuMYk74LGi8F8lk5ZMPkFAiEAsaB2pKQWBvuPnNUnbdQXHc-izgsHJUP793woC2xNJlg%3D/lsparams/playback_host%2Cmh%2Cmm%2Cmn%2Cms%2Cmv%2Cmvi%2Cpl%2Cinitcwndbps/lsig/AG3C_xAwRQIgOY4xu4H9wqPVZ7vF2i0hFcOnqrur1XGoA43a7ZEuuSUCIQCyPxBKXUQrKFmknNEGpX5GSWySKgMw_xHBikWpKpKwvg%3D%3D"), dash_manifest_url: Some("https://manifest.googlevideo.com/api/manifest/dash/expire/1659481355/ei/q1jpYtq3BJCX1gKVyJGQDg/ip/2003%3Ade%3Aaf0e%3A2f00%3Ade47%3A297%3Aa6db%3A774e/id/a4fbddf14c6649b4/source/youtube/requiressl/yes/playback_host/rr4---sn-h0jelnez.googlevideo.com/mh/mQ/mm/31%2C26/mn/sn-h0jelnez%2Csn-4g5edn6k/ms/au%2Conr/mv/m/mvi/4/pl/37/hfr/all/as/fmp4_audio_clear%2Cwebm_audio_clear%2Cwebm2_audio_clear%2Cfmp4_sd_hd_clear%2Cwebm2_sd_hd_clear/initcwndbps/1513750/spc/lT-KhrZGE2opztWyVdAtyUNlb8dXPDs/vprv/1/mt/1659459429/fvip/4/keepalive/yes/fexp/24001373%2C24007246/itag/0/sparams/expire%2Cei%2Cip%2Cid%2Csource%2Crequiressl%2Chfr%2Cas%2Cspc%2Cvprv%2Citag/sig/AOq0QJ8wRgIhAPEjHK19PKVHqQeia6WF4qubuMYk74LGi8F8lk5ZMPkFAiEAsaB2pKQWBvuPnNUnbdQXHc-izgsHJUP793woC2xNJlg%3D/lsparams/playback_host%2Cmh%2Cmm%2Cmn%2Cms%2Cmv%2Cmvi%2Cpl%2Cinitcwndbps/lsig/AG3C_xAwRQIgOY4xu4H9wqPVZ7vF2i0hFcOnqrur1XGoA43a7ZEuuSUCIQCyPxBKXUQrKFmknNEGpX5GSWySKgMw_xHBikWpKpKwvg%3D%3D"),
visitor_data: Some("CgtoS1pCMVJTNUJISSirsaWXBg%3D%3D"),
) )

View file

@ -365,4 +365,5 @@ VideoPlayer(
expires_in_seconds: 21540, expires_in_seconds: 21540,
hls_manifest_url: None, hls_manifest_url: None,
dash_manifest_url: Some("https://manifest.googlevideo.com/api/manifest/dash/expire/1659487474/ei/knDpYub6BojEgAf6jbLgDw/ip/2003%3Ade%3Aaf0e%3A2f00%3Ade47%3A297%3Aa6db%3A774e/id/a4fbddf14c6649b4/source/youtube/requiressl/yes/playback_host/rr5---sn-h0jeenek.googlevideo.com/mh/mQ/mm/31%2C29/mn/sn-h0jeenek%2Csn-h0jelnez/ms/au%2Crdu/mv/m/mvi/5/pl/37/hfr/all/as/fmp4_audio_clear%2Cwebm_audio_clear%2Cwebm2_audio_clear%2Cfmp4_sd_hd_clear%2Cwebm2_sd_hd_clear/initcwndbps/1418750/spc/lT-Khox4YuJQ2wmH79zYALRvsWTPCUc/vprv/1/mt/1659465669/fvip/4/keepalive/yes/fexp/24001373%2C24007246/itag/0/sparams/expire%2Cei%2Cip%2Cid%2Csource%2Crequiressl%2Chfr%2Cas%2Cspc%2Cvprv%2Citag/sig/AOq0QJ8wRAIgErABhAEaoKHUDu9dDbpxE_8gR4b8WWAi61fnu8UKnuICIEYrEKcHvqHdO4V3R7cvSGwi_HGH34IlQsKbziOfMBov/lsparams/playback_host%2Cmh%2Cmm%2Cmn%2Cms%2Cmv%2Cmvi%2Cpl%2Cinitcwndbps/lsig/AG3C_xAwRQIgJxHmH0Sxo3cY_pW_ZzQ3hW9-7oz6K_pZWcUdrDDQ2sQCIQDJYNINQwLgKelgbO3CZYx7sMxdUAFpWdokmRBQ77vwvw%3D%3D"), dash_manifest_url: Some("https://manifest.googlevideo.com/api/manifest/dash/expire/1659487474/ei/knDpYub6BojEgAf6jbLgDw/ip/2003%3Ade%3Aaf0e%3A2f00%3Ade47%3A297%3Aa6db%3A774e/id/a4fbddf14c6649b4/source/youtube/requiressl/yes/playback_host/rr5---sn-h0jeenek.googlevideo.com/mh/mQ/mm/31%2C29/mn/sn-h0jeenek%2Csn-h0jelnez/ms/au%2Crdu/mv/m/mvi/5/pl/37/hfr/all/as/fmp4_audio_clear%2Cwebm_audio_clear%2Cwebm2_audio_clear%2Cfmp4_sd_hd_clear%2Cwebm2_sd_hd_clear/initcwndbps/1418750/spc/lT-Khox4YuJQ2wmH79zYALRvsWTPCUc/vprv/1/mt/1659465669/fvip/4/keepalive/yes/fexp/24001373%2C24007246/itag/0/sparams/expire%2Cei%2Cip%2Cid%2Csource%2Crequiressl%2Chfr%2Cas%2Cspc%2Cvprv%2Citag/sig/AOq0QJ8wRAIgErABhAEaoKHUDu9dDbpxE_8gR4b8WWAi61fnu8UKnuICIEYrEKcHvqHdO4V3R7cvSGwi_HGH34IlQsKbziOfMBov/lsparams/playback_host%2Cmh%2Cmm%2Cmn%2Cms%2Cmv%2Cmvi%2Cpl%2Cinitcwndbps/lsig/AG3C_xAwRQIgJxHmH0Sxo3cY_pW_ZzQ3hW9-7oz6K_pZWcUdrDDQ2sQCIQDJYNINQwLgKelgbO3CZYx7sMxdUAFpWdokmRBQ77vwvw%3D%3D"),
visitor_data: Some("CgszSHZWNWs0SDhpTSiS4aWXBg%3D%3D"),
) )

View file

@ -160,4 +160,5 @@ VideoPlayer(
expires_in_seconds: 21540, expires_in_seconds: 21540,
hls_manifest_url: Some("https://manifest.googlevideo.com/api/manifest/hls_variant/expire/1659481355/ei/q1jpYq-xHs7NgQev0bfwAQ/ip/2003%3Ade%3Aaf0e%3A2f00%3Ade47%3A297%3Aa6db%3A774e/id/a4fbddf14c6649b4/source/youtube/requiressl/yes/playback_host/rr4---sn-h0jelnez.googlevideo.com/mh/mQ/mm/31%2C29/mn/sn-h0jelnez%2Csn-h0jeenek/ms/au%2Crdu/mv/m/mvi/4/pl/37/hfr/1/demuxed/1/tts_caps/1/maudio/1/initcwndbps/1513750/vprv/1/go/1/mt/1659459429/fvip/5/nvgoi/1/short_key/1/ncsapi/1/keepalive/yes/fexp/24001373%2C24007246/dover/13/itag/0/playlist_type/DVR/sparams/expire%2Cei%2Cip%2Cid%2Csource%2Crequiressl%2Chfr%2Cdemuxed%2Ctts_caps%2Cmaudio%2Cvprv%2Cgo%2Citag%2Cplaylist_type/sig/AOq0QJ8wRQIhAIYnEHvIgJtJ8hehAXNtVY3qsgsq_GdOhWf2hkJZe6lCAiBxaRY_nubYp6hBizcAg_KFkKnkG-t2XYLRQ5wGdM3AjA%3D%3D/lsparams/playback_host%2Cmh%2Cmm%2Cmn%2Cms%2Cmv%2Cmvi%2Cpl%2Cinitcwndbps/lsig/AG3C_xAwRgIhAM_91Kk_0VLuSsR6nLCY7LdtWojyRAzXSScd_X9ShRROAiEA1AF4VY04F71NsAI8_j3iqjuXnWL9s6NoXHq7P8-bHx8%3D/file/index.m3u8"), hls_manifest_url: Some("https://manifest.googlevideo.com/api/manifest/hls_variant/expire/1659481355/ei/q1jpYq-xHs7NgQev0bfwAQ/ip/2003%3Ade%3Aaf0e%3A2f00%3Ade47%3A297%3Aa6db%3A774e/id/a4fbddf14c6649b4/source/youtube/requiressl/yes/playback_host/rr4---sn-h0jelnez.googlevideo.com/mh/mQ/mm/31%2C29/mn/sn-h0jelnez%2Csn-h0jeenek/ms/au%2Crdu/mv/m/mvi/4/pl/37/hfr/1/demuxed/1/tts_caps/1/maudio/1/initcwndbps/1513750/vprv/1/go/1/mt/1659459429/fvip/5/nvgoi/1/short_key/1/ncsapi/1/keepalive/yes/fexp/24001373%2C24007246/dover/13/itag/0/playlist_type/DVR/sparams/expire%2Cei%2Cip%2Cid%2Csource%2Crequiressl%2Chfr%2Cdemuxed%2Ctts_caps%2Cmaudio%2Cvprv%2Cgo%2Citag%2Cplaylist_type/sig/AOq0QJ8wRQIhAIYnEHvIgJtJ8hehAXNtVY3qsgsq_GdOhWf2hkJZe6lCAiBxaRY_nubYp6hBizcAg_KFkKnkG-t2XYLRQ5wGdM3AjA%3D%3D/lsparams/playback_host%2Cmh%2Cmm%2Cmn%2Cms%2Cmv%2Cmvi%2Cpl%2Cinitcwndbps/lsig/AG3C_xAwRgIhAM_91Kk_0VLuSsR6nLCY7LdtWojyRAzXSScd_X9ShRROAiEA1AF4VY04F71NsAI8_j3iqjuXnWL9s6NoXHq7P8-bHx8%3D/file/index.m3u8"),
dash_manifest_url: None, dash_manifest_url: None,
visitor_data: Some("Cgs4TXV4dk13WVEyWSirsaWXBg%3D%3D"),
) )

View file

@ -541,4 +541,5 @@ VideoPlayer(
expires_in_seconds: 21540, expires_in_seconds: 21540,
hls_manifest_url: None, hls_manifest_url: None,
dash_manifest_url: Some("https://manifest.googlevideo.com/api/manifest/dash/expire/1659481355/ei/q1jpYv-eJ9uF6dsPhvyH8As/ip/2003%3Ade%3Aaf0e%3A2f00%3Ade47%3A297%3Aa6db%3A774e/id/a4fbddf14c6649b4/source/youtube/requiressl/yes/playback_host/rr4---sn-h0jelnez.googlevideo.com/mh/mQ/mm/31%2C29/mn/sn-h0jelnez%2Csn-h0jeenek/ms/au%2Crdu/mv/m/mvi/4/pl/37/hfr/all/as/fmp4_audio_clear%2Cfmp4_sd_hd_clear/initcwndbps/1527500/vprv/1/mt/1659459429/fvip/5/keepalive/yes/fexp/24001373%2C24007246/itag/0/sparams/expire%2Cei%2Cip%2Cid%2Csource%2Crequiressl%2Chfr%2Cas%2Cvprv%2Citag/sig/AOq0QJ8wRQIhANKWS7GCN4pSoHIQ6BMZdOaHAD0I25nHwRj7ds4qrxdEAiBsd9l8WIceqF7-2xyR82DGecCiS9hgUIPJhdNhkwVpHg%3D%3D/lsparams/playback_host%2Cmh%2Cmm%2Cmn%2Cms%2Cmv%2Cmvi%2Cpl%2Cinitcwndbps/lsig/AG3C_xAwRQIgMbu-wTOcXGCwGh27y0YZHktumKM1sopgxfQf8LCcCnECIQDnhFbgddOxwiQbnMOIcCn6ncpN54UyALRNigUSCp9Deg%3D%3D"), dash_manifest_url: Some("https://manifest.googlevideo.com/api/manifest/dash/expire/1659481355/ei/q1jpYv-eJ9uF6dsPhvyH8As/ip/2003%3Ade%3Aaf0e%3A2f00%3Ade47%3A297%3Aa6db%3A774e/id/a4fbddf14c6649b4/source/youtube/requiressl/yes/playback_host/rr4---sn-h0jelnez.googlevideo.com/mh/mQ/mm/31%2C29/mn/sn-h0jelnez%2Csn-h0jeenek/ms/au%2Crdu/mv/m/mvi/4/pl/37/hfr/all/as/fmp4_audio_clear%2Cfmp4_sd_hd_clear/initcwndbps/1527500/vprv/1/mt/1659459429/fvip/5/keepalive/yes/fexp/24001373%2C24007246/itag/0/sparams/expire%2Cei%2Cip%2Cid%2Csource%2Crequiressl%2Chfr%2Cas%2Cvprv%2Citag/sig/AOq0QJ8wRQIhANKWS7GCN4pSoHIQ6BMZdOaHAD0I25nHwRj7ds4qrxdEAiBsd9l8WIceqF7-2xyR82DGecCiS9hgUIPJhdNhkwVpHg%3D%3D/lsparams/playback_host%2Cmh%2Cmm%2Cmn%2Cms%2Cmv%2Cmvi%2Cpl%2Cinitcwndbps/lsig/AG3C_xAwRQIgMbu-wTOcXGCwGh27y0YZHktumKM1sopgxfQf8LCcCnECIQDnhFbgddOxwiQbnMOIcCn6ncpN54UyALRNigUSCp9Deg%3D%3D"),
visitor_data: Some("CgtacUJOMG81dTI3cyirsaWXBg%3D%3D"),
) )

View file

@ -3142,4 +3142,5 @@ Playlist(
)), )),
last_update: "[date]", last_update: "[date]",
last_update_txt: Some("Last updated on Aug 7, 2022"), last_update_txt: Some("Last updated on Aug 7, 2022"),
visitor_data: Some("CgtZdi1GV3N3TnBuQSi46K-YBg%3D%3D"),
) )

View file

@ -2088,4 +2088,5 @@ Playlist(
)), )),
last_update: "[date]", last_update: "[date]",
last_update_txt: Some("Last updated on Jul 2, 2014"), last_update_txt: Some("Last updated on Jul 2, 2014"),
visitor_data: Some("CgtXY0lqNWN5VlNmTSik8q-YBg%3D%3D"),
) )

View file

@ -3041,4 +3041,5 @@ Playlist(
channel: None, channel: None,
last_update: "[date]", last_update: "[date]",
last_update_txt: Some("Updated today"), last_update_txt: Some("Updated today"),
visitor_data: Some("CgtLZVdRQ1dkM2VDVSi46K-YBg%3D%3D"),
) )

View file

@ -734,4 +734,5 @@ SearchResult(
endpoint: browse, endpoint: browse,
), ),
corrected_query: Some("doobydobap"), corrected_query: Some("doobydobap"),
visitor_data: Some("Cgs4MEJMc3FmVzVadyiNy4-aBg%3D%3D"),
) )

View file

@ -10,4 +10,5 @@ SearchResult(
endpoint: browse, endpoint: browse,
), ),
corrected_query: None, corrected_query: None,
visitor_data: Some("Cgs1Q0NxX3llelBxWSi85ZGaBg%3D%3D"),
) )

View file

@ -691,4 +691,5 @@ SearchResult(
endpoint: browse, endpoint: browse,
), ),
corrected_query: None, corrected_query: None,
visitor_data: Some("CgstZjhyS1IyR1R6dyiX4JGaBg%3D%3D"),
) )

View file

@ -750,4 +750,5 @@ VideoDetails(
ctoken: Some("Eg0SC1plZXJybnVMaTVFGAYyOCIRIgtaZWVycm51TGk1RTABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"), ctoken: Some("Eg0SC1plZXJybnVMaTVFGAYyOCIRIgtaZWVycm51TGk1RTABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"),
endpoint: next, endpoint: next,
), ),
visitor_data: Some("CgtCeURHR09uNlJ5TSjOiLqZBg%3D%3D"),
) )

View file

@ -824,4 +824,5 @@ VideoDetails(
ctoken: Some("Eg0SC1plZXJybnVMaTVFGAYyOCIRIgtaZWVycm51TGk1RTABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"), ctoken: Some("Eg0SC1plZXJybnVMaTVFGAYyOCIRIgtaZWVycm51TGk1RTABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"),
endpoint: next, endpoint: next,
), ),
visitor_data: Some("Cgs2V0p6ZW5ab1ozTSjkrpaaBg%3D%3D"),
) )

View file

@ -1282,4 +1282,5 @@ VideoDetails(
ctoken: Some("Eg0SC25GREJ4QlVmRTc0GAYyOCIRIgtuRkRCeEJVZkU3NDABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"), ctoken: Some("Eg0SC25GREJ4QlVmRTc0GAYyOCIRIgtuRkRCeEJVZkU3NDABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"),
endpoint: next, endpoint: next,
), ),
visitor_data: Some("Cgtidzg4MlRTb3FKSSiqipeaBg%3D%3D"),
) )

View file

@ -54,4 +54,5 @@ VideoDetails(
ctoken: Some("Eg0SC0hSS3UwY3Zycl9vGAYyOCIRIgtIUkt1MGN2cnJfbzABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"), ctoken: Some("Eg0SC0hSS3UwY3Zycl9vGAYyOCIRIgtIUkt1MGN2cnJfbzABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"),
endpoint: next, endpoint: next,
), ),
visitor_data: Some("CgtxUUdrc1VSVE54Zyja1KiZBg%3D%3D"),
) )

View file

@ -771,4 +771,5 @@ VideoDetails(
ctoken: Some("Eg0SCzByYjlDZk92b2prGAYyOCIRIgswcmI5Q2ZPdm9qazABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"), ctoken: Some("Eg0SCzByYjlDZk92b2prGAYyOCIRIgswcmI5Q2ZPdm9qazABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"),
endpoint: next, endpoint: next,
), ),
visitor_data: Some("CgtoY1pQUF8wNW1qayjSjpSZBg%3D%3D"),
) )

View file

@ -1239,4 +1239,5 @@ VideoDetails(
ctoken: Some("Eg0SC25GREJ4QlVmRTc0GAYyOCIRIgtuRkRCeEJVZkU3NDABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"), ctoken: Some("Eg0SC25GREJ4QlVmRTc0GAYyOCIRIgtuRkRCeEJVZkU3NDABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"),
endpoint: next, endpoint: next,
), ),
visitor_data: Some("CgtIV0JjSUtDQm9LQSjUjpSZBg%3D%3D"),
) )

View file

@ -830,4 +830,5 @@ VideoDetails(
ctoken: None, ctoken: None,
endpoint: next, endpoint: next,
), ),
visitor_data: Some("CgtnQS1WdzlNNkNCSSiSmKiZBg%3D%3D"),
) )

View file

@ -576,4 +576,5 @@ VideoDetails(
ctoken: None, ctoken: None,
endpoint: next, endpoint: next,
), ),
visitor_data: Some("CgtzclhqZVpoajVhVSi76qeZBg%3D%3D"),
) )

View file

@ -787,4 +787,5 @@ VideoDetails(
ctoken: Some("Eg0SC1plZXJybnVMaTVFGAYyOCIRIgtaZWVycm51TGk1RTABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"), ctoken: Some("Eg0SC1plZXJybnVMaTVFGAYyOCIRIgtaZWVycm51TGk1RTABeAIwAUIhZW5nYWdlbWVudC1wYW5lbC1jb21tZW50cy1zZWN0aW9u"),
endpoint: next, endpoint: next,
), ),
visitor_data: Some("Cgtjemd0bDVxU1N1QSjRjpSZBg%3D%3D"),
) )

View file

@ -252,7 +252,7 @@ impl MapResponse<VideoDetails> for response::VideoDetails {
let mut res = map_recommendations( let mut res = map_recommendations(
r, r,
sr.secondary_results.continuations, sr.secondary_results.continuations,
self.response_context.visitor_data, self.response_context.visitor_data.clone(),
lang, lang,
); );
warnings.append(&mut res.warnings); warnings.append(&mut res.warnings);
@ -343,6 +343,7 @@ impl MapResponse<VideoDetails> for response::VideoDetails {
None, None,
crate::param::ContinuationEndpoint::Next, crate::param::ContinuationEndpoint::Next,
), ),
visitor_data: self.response_context.visitor_data,
}, },
warnings, warnings,
}) })

View file

@ -112,8 +112,12 @@ pub struct VideoPlayer {
pub subtitles: Vec<Subtitle>, pub subtitles: Vec<Subtitle>,
/// Lifetime of the stream URLs in seconds /// Lifetime of the stream URLs in seconds
pub expires_in_seconds: u32, pub expires_in_seconds: u32,
/// HLS manifest URL (for livestreams)
pub hls_manifest_url: Option<String>, pub hls_manifest_url: Option<String>,
/// Dash manifest URL (for livestreams)
pub dash_manifest_url: Option<String>, pub dash_manifest_url: Option<String>,
/// YouTube visitor data cookie
pub visitor_data: Option<String>,
} }
/// Video metadata from the player /// Video metadata from the player
@ -461,6 +465,8 @@ pub struct Playlist {
pub last_update: Option<Date>, pub last_update: Option<Date>,
/// Textual last update date /// Textual last update date
pub last_update_txt: Option<String>, pub last_update_txt: Option<String>,
/// YouTube visitor data cookie
pub visitor_data: Option<String>,
} }
/// YouTube video extracted from a playlist /// YouTube video extracted from a playlist
@ -543,6 +549,8 @@ pub struct VideoDetails {
/// ///
/// Is initially empty. /// Is initially empty.
pub latest_comments: Paginator<Comment>, pub latest_comments: Paginator<Comment>,
/// YouTube visitor data cookie
pub visitor_data: Option<String>,
} }
/// Chapter of a video /// Chapter of a video
@ -675,6 +683,12 @@ pub struct Channel<T> {
pub mobile_banner: Vec<Thumbnail>, pub mobile_banner: Vec<Thumbnail>,
/// Banner image shown above the channel (16:9 fullscreen format for TV) /// Banner image shown above the channel (16:9 fullscreen format for TV)
pub tv_banner: Vec<Thumbnail>, pub tv_banner: Vec<Thumbnail>,
/// Does the channel have a *Shorts* tab?
pub has_shorts: bool,
/// Does the channel have a *Live* tab?
pub has_live: bool,
/// YouTube visitor data cookie
pub visitor_data: Option<String>,
/// Content fetched from the channel /// Content fetched from the channel
pub content: T, pub content: T,
} }
@ -745,6 +759,8 @@ pub struct SearchResult {
/// for the corrected search term and displays it on top of the /// for the corrected search term and displays it on top of the
/// search results page. /// search results page.
pub corrected_query: Option<String>, pub corrected_query: Option<String>,
/// YouTube visitor data cookie
pub visitor_data: Option<String>,
} }
/// YouTube item (Video/Channel/Playlist) /// YouTube item (Video/Channel/Playlist)

View file

@ -82,17 +82,27 @@ impl Mul<u8> for TimeAgo {
} }
} }
impl From<TimeAgo> for Duration {
fn from(ta: TimeAgo) -> Self {
match ta.unit {
TimeUnit::Second => Duration::seconds(ta.n as i64),
TimeUnit::Minute => Duration::minutes(ta.n as i64),
TimeUnit::Hour => Duration::hours(ta.n as i64),
TimeUnit::Day => Duration::days(ta.n as i64),
TimeUnit::Week => Duration::weeks(ta.n as i64),
TimeUnit::Month => Duration::days(ta.n as i64 * 30),
TimeUnit::Year => Duration::days(ta.n as i64 * 365),
}
}
}
impl From<TimeAgo> for OffsetDateTime { impl From<TimeAgo> for OffsetDateTime {
fn from(ta: TimeAgo) -> Self { fn from(ta: TimeAgo) -> Self {
let ts = util::now_sec(); let ts = util::now_sec();
match ta.unit { match ta.unit {
TimeUnit::Second => ts - Duration::seconds(ta.n as i64),
TimeUnit::Minute => ts - Duration::minutes(ta.n as i64),
TimeUnit::Hour => ts - Duration::hours(ta.n as i64),
TimeUnit::Day => ts - Duration::days(ta.n as i64),
TimeUnit::Week => ts - Duration::weeks(ta.n as i64),
TimeUnit::Month => ts.replace_date(util::shift_months(ts.date(), -(ta.n as i32))), TimeUnit::Month => ts.replace_date(util::shift_months(ts.date(), -(ta.n as i32))),
TimeUnit::Year => ts.replace_date(util::shift_years(ts.date(), -(ta.n as i32))), TimeUnit::Year => ts.replace_date(util::shift_years(ts.date(), -(ta.n as i32))),
_ => ts - Duration::from(ta),
} }
} }
} }

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

View file

@ -12,6 +12,8 @@ use rustypipe::model::{
}; };
use rustypipe::param::search_filter::{self, SearchFilter}; use rustypipe::param::search_filter::{self, SearchFilter};
const VISITOR_DATA_3TAB_CHANNEL_LAYOUT: &str = "CgtOa256ckVkcG5YVSiirbyaBg%3D%3D";
//#PLAYER //#PLAYER
#[rstest] #[rstest]
@ -908,6 +910,75 @@ async fn channel_videos() {
); );
} }
#[tokio::test]
async fn channel_shorts() {
let rp = RustyPipe::builder()
.strict()
.visitor_data(VISITOR_DATA_3TAB_CHANNEL_LAYOUT)
.build();
let channel = rp
.query()
.channel_shorts("UCh8gHdtzO2tXd593_bjErWg")
.await
.unwrap();
// dbg!(&channel);
assert_eq!(channel.id, "UCh8gHdtzO2tXd593_bjErWg");
assert_eq!(channel.name, "Doobydobap");
assert!(
channel.subscriber_count.unwrap() > 2800000,
"expected >2.8M subscribers, got {}",
channel.subscriber_count.unwrap()
);
assert!(!channel.avatar.is_empty(), "got no thumbnails");
assert_eq!(channel.verification, Verification::Verified);
assert!(channel
.description
.contains("Hi, I\u{2019}m Tina, aka Doobydobap"));
assert_eq!(
channel.vanity_url.as_ref().unwrap(),
"https://www.youtube.com/c/Doobydobap"
);
assert!(!channel.banner.is_empty(), "got no banners");
assert!(!channel.mobile_banner.is_empty(), "got no mobile banners");
assert!(!channel.tv_banner.is_empty(), "got no tv banners");
assert!(
!channel.content.items.is_empty() && !channel.content.is_exhausted(),
"got no shorts"
);
let next = channel.content.next(&rp.query()).await.unwrap().unwrap();
assert!(
!next.is_exhausted() && !next.items.is_empty(),
"no more shorts"
);
}
#[tokio::test]
async fn channel_livestreams() {
let rp = RustyPipe::builder()
.visitor_data(VISITOR_DATA_3TAB_CHANNEL_LAYOUT)
.strict()
.build();
let channel = rp
.query()
.channel_livestreams("UC2DjFE7Xf11URZqWBigcVOQ")
.await
.unwrap();
// dbg!(&channel);
assert_channel_eevblog(&channel);
assert!(
!channel.content.items.is_empty() && !channel.content.is_exhausted(),
"got no streams"
);
let next = channel.content.next(&rp.query()).await.unwrap().unwrap();
assert!(!next.items.is_empty(), "no more streams");
}
#[tokio::test] #[tokio::test]
async fn channel_playlists() { async fn channel_playlists() {
let rp = RustyPipe::builder().strict().build(); let rp = RustyPipe::builder().strict().build();