use std::borrow::Cow; use crate::{ error::{Error, ExtractionError}, model::{ paginator::{ContinuationEndpoint, Paginator}, VideoItem, }, param::Language, serializer::MapResult, }; use super::{response, ClientType, MapResponse, QBrowse, QBrowseParams, RustyPipeQuery}; impl RustyPipeQuery { /// Get the videos from the YouTube startpage #[tracing::instrument(skip(self))] pub async fn startpage(&self) -> Result, Error> { let context = self.get_context(ClientType::Desktop, true, None).await; let request_body = QBrowse { context, browse_id: "FEwhat_to_watch", }; self.execute_request::( ClientType::Desktop, "startpage", "", "browse", &request_body, ) .await } /// Get the videos from the YouTube trending page #[tracing::instrument(skip(self))] pub async fn trending(&self) -> Result, Error> { let context = self.get_context(ClientType::Desktop, true, None).await; let request_body = QBrowseParams { context, browse_id: "FEtrending", params: "4gIOGgxtb3N0X3BvcHVsYXI%3D", }; self.execute_request::( ClientType::Desktop, "trends", "", "browse", &request_body, ) .await } } impl MapResponse> for response::Startpage { fn map_response( self, _id: &str, lang: crate::param::Language, _deobf: Option<&crate::deobfuscate::DeobfData>, vdata: Option<&str>, ) -> Result>, ExtractionError> { let grid = self .contents .two_column_browse_results_renderer .contents .into_iter() .next() .ok_or(ExtractionError::InvalidData(Cow::Borrowed("no contents")))? .tab_renderer .content .section_list_renderer .contents; Ok(map_startpage_videos( grid, lang, self.response_context .visitor_data .or_else(|| vdata.map(str::to_owned)), )) } } impl MapResponse> for response::Trending { fn map_response( self, _id: &str, lang: crate::param::Language, _deobf: Option<&crate::deobfuscate::DeobfData>, _vdata: Option<&str>, ) -> Result>, ExtractionError> { let items = self .contents .two_column_browse_results_renderer .contents .into_iter() .next() .ok_or(ExtractionError::InvalidData(Cow::Borrowed("no contents")))? .tab_renderer .content .section_list_renderer .contents; let mut mapper = response::YouTubeListMapper::::new(lang); mapper.map_response(items); Ok(MapResult { c: mapper.items, warnings: mapper.warnings, }) } } fn map_startpage_videos( videos: MapResult>, lang: Language, visitor_data: Option, ) -> MapResult> { let mut mapper = response::YouTubeListMapper::::new(lang); mapper.map_response(videos); MapResult { c: Paginator::new_ext( None, mapper.items, mapper.ctoken, visitor_data, ContinuationEndpoint::Browse, ), warnings: mapper.warnings, } } #[cfg(test)] mod tests { use std::{fs::File, io::BufReader}; use path_macro::path; use rstest::rstest; use crate::{ client::{response, MapResponse}, model::{paginator::Paginator, VideoItem}, param::Language, serializer::MapResult, util::tests::TESTFILES, }; #[test] fn map_startpage() { let json_path = path!(*TESTFILES / "trends" / "startpage.json"); let json_file = File::open(json_path).unwrap(); let startpage: response::Startpage = serde_json::from_reader(BufReader::new(json_file)).unwrap(); let map_res: MapResult> = startpage .map_response("", Language::En, None, None) .unwrap(); assert!( map_res.warnings.is_empty(), "deserialization/mapping warnings: {:?}", map_res.warnings ); insta::assert_ron_snapshot!("map_startpage", map_res.c, { ".items[].publish_date" => "[date]", }); } #[rstest] #[case::base("videos")] #[case::page_header_renderer("20230501_page_header_renderer")] fn map_trending(#[case] name: &str) { let json_path = path!(*TESTFILES / "trends" / format!("trending_{name}.json")); let json_file = File::open(json_path).unwrap(); let startpage: response::Trending = serde_json::from_reader(BufReader::new(json_file)).unwrap(); let map_res: MapResult> = startpage .map_response("", Language::En, None, None) .unwrap(); assert!( map_res.warnings.is_empty(), "deserialization/mapping warnings: {:?}", map_res.warnings ); insta::assert_ron_snapshot!(format!("map_trending_{name}"), map_res.c, { "[].publish_date" => "[date]", }); } }