180 lines
5.7 KiB
Rust
180 lines
5.7 KiB
Rust
use crate::{
|
|
client::{
|
|
response::{self, music_item::MusicListMapper},
|
|
ClientType, MapResponse, QBrowseParams, RustyPipeQuery,
|
|
},
|
|
error::{Error, ExtractionError},
|
|
model::{
|
|
paginator::{ContinuationEndpoint, Paginator},
|
|
AlbumItem, ArtistItem, MusicPlaylistItem, TrackItem,
|
|
},
|
|
serializer::MapResult,
|
|
};
|
|
|
|
use super::MapRespCtx;
|
|
|
|
impl RustyPipeQuery {
|
|
/// Get a list of tracks from YouTube Music which the current user recently played
|
|
///
|
|
/// Requires authentication cookies.
|
|
#[tracing::instrument(skip(self), level = "error")]
|
|
pub async fn music_history(&self) -> Result<Paginator<TrackItem>, Error> {
|
|
let request_body = QBrowseParams {
|
|
browse_id: "FEmusic_history",
|
|
params: "oggECgIIAQ%3D%3D",
|
|
};
|
|
|
|
self.clone()
|
|
.authenticated()
|
|
.execute_request::<response::MusicHistory, _, _>(
|
|
ClientType::DesktopMusic,
|
|
"music_history",
|
|
"",
|
|
"browse",
|
|
&request_body,
|
|
)
|
|
.await
|
|
}
|
|
|
|
/// Get a list of YouTube Music artists which the current user subscribed to
|
|
///
|
|
/// Requires authentication cookies.
|
|
#[tracing::instrument(skip(self), level = "error")]
|
|
pub async fn music_saved_artists(&self) -> Result<Paginator<ArtistItem>, Error> {
|
|
self.clone()
|
|
.authenticated()
|
|
.continuation(
|
|
"4qmFsgIyEh5GRW11c2ljX2xpYnJhcnlfY29ycHVzX2FydGlzdHMaEGdnTUdLZ1FJQUJBQm9BWUI%3D",
|
|
ContinuationEndpoint::MusicBrowse,
|
|
None,
|
|
)
|
|
.await
|
|
}
|
|
|
|
/// Get a list of YouTube Music albums which the current user has added to their collection
|
|
///
|
|
/// Requires authentication cookies.
|
|
#[tracing::instrument(skip(self), level = "error")]
|
|
pub async fn music_saved_albums(&self) -> Result<Paginator<AlbumItem>, Error> {
|
|
self.clone()
|
|
.authenticated()
|
|
.continuation(
|
|
"4qmFsgIoEhRGRW11c2ljX2xpa2VkX2FsYnVtcxoQZ2dNR0tnUUlBQkFCb0FZQg%3D%3D",
|
|
ContinuationEndpoint::MusicBrowse,
|
|
None,
|
|
)
|
|
.await
|
|
}
|
|
|
|
/// Get a list of YouTube Music tracks which the current user has added to their collection
|
|
///
|
|
/// Requires authentication cookies.
|
|
#[tracing::instrument(skip(self), level = "error")]
|
|
pub async fn music_saved_tracks(&self) -> Result<Paginator<TrackItem>, Error> {
|
|
self.clone()
|
|
.authenticated()
|
|
.continuation(
|
|
"4qmFsgIoEhRGRW11c2ljX2xpa2VkX3ZpZGVvcxoQZ2dNR0tnUUlBQkFCb0FZQg%3D%3D",
|
|
ContinuationEndpoint::MusicBrowse,
|
|
None,
|
|
)
|
|
.await
|
|
}
|
|
|
|
/// Get a list of YouTube Music playlists which the current user has added to their collection
|
|
///
|
|
/// Requires authentication cookies.
|
|
#[tracing::instrument(skip(self), level = "error")]
|
|
pub async fn music_saved_playlists(&self) -> Result<Paginator<MusicPlaylistItem>, Error> {
|
|
self.clone()
|
|
.authenticated()
|
|
.continuation(
|
|
"4qmFsgIrEhdGRW11c2ljX2xpa2VkX3BsYXlsaXN0cxoQZ2dNR0tnUUlBQkFCb0FZQg%3D%3D",
|
|
ContinuationEndpoint::MusicBrowse,
|
|
None,
|
|
)
|
|
.await
|
|
}
|
|
}
|
|
|
|
impl MapResponse<Paginator<TrackItem>> for response::MusicHistory {
|
|
fn map_response(
|
|
self,
|
|
ctx: &MapRespCtx<'_>,
|
|
) -> Result<MapResult<Paginator<TrackItem>>, ExtractionError> {
|
|
let contents = match self.contents {
|
|
response::music_playlist::Contents::SingleColumnBrowseResultsRenderer(c) => {
|
|
c.contents
|
|
.into_iter()
|
|
.next()
|
|
.ok_or(ExtractionError::InvalidData("no content".into()))?
|
|
.tab_renderer
|
|
.content
|
|
.section_list_renderer
|
|
}
|
|
response::music_playlist::Contents::TwoColumnBrowseResultsRenderer {
|
|
secondary_contents,
|
|
..
|
|
} => secondary_contents.section_list_renderer,
|
|
};
|
|
|
|
let mut mapper = MusicListMapper::new(ctx.lang);
|
|
|
|
for shelf in contents.contents {
|
|
let shelf = if let response::music_item::ItemSection::MusicShelfRenderer(s) = shelf {
|
|
s
|
|
} else {
|
|
continue;
|
|
};
|
|
mapper.map_response(shelf.contents);
|
|
}
|
|
|
|
let map_res = mapper.conv_items();
|
|
|
|
let ctoken = contents
|
|
.continuations
|
|
.into_iter()
|
|
.next()
|
|
.map(|c| c.next_continuation_data.continuation);
|
|
|
|
Ok(MapResult {
|
|
c: Paginator::new_ext(
|
|
None,
|
|
map_res.c,
|
|
ctoken,
|
|
None,
|
|
ContinuationEndpoint::MusicBrowse,
|
|
true,
|
|
),
|
|
warnings: map_res.warnings,
|
|
})
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use std::{fs::File, io::BufReader};
|
|
|
|
use path_macro::path;
|
|
|
|
use crate::util::tests::TESTFILES;
|
|
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn map_history() {
|
|
let json_path = path!(*TESTFILES / "music_history" / "music_history.json");
|
|
let json_file = File::open(json_path).unwrap();
|
|
|
|
let history: response::MusicHistory =
|
|
serde_json::from_reader(BufReader::new(json_file)).unwrap();
|
|
let map_res = history.map_response(&MapRespCtx::test("")).unwrap();
|
|
|
|
assert!(
|
|
map_res.warnings.is_empty(),
|
|
"deserialization/mapping warnings: {:?}",
|
|
map_res.warnings
|
|
);
|
|
insta::assert_ron_snapshot!(map_res.c);
|
|
}
|
|
}
|