241 lines
7.3 KiB
Rust
241 lines
7.3 KiB
Rust
use serde::Deserialize;
|
|
use serde_with::{serde_as, DefaultOnError};
|
|
|
|
use crate::{model::UrlTarget, util};
|
|
|
|
/// navigation/resolve_url response model
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct ResolvedUrl {
|
|
pub endpoint: NavigationEndpoint,
|
|
}
|
|
|
|
#[serde_as]
|
|
#[derive(Debug, Deserialize, Default)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct NavigationEndpoint {
|
|
#[serde(default)]
|
|
#[serde_as(deserialize_as = "DefaultOnError")]
|
|
pub watch_endpoint: Option<WatchEndpoint>,
|
|
#[serde(default)]
|
|
#[serde_as(deserialize_as = "DefaultOnError")]
|
|
pub browse_endpoint: Option<BrowseEndpoint>,
|
|
#[serde(default)]
|
|
#[serde_as(deserialize_as = "DefaultOnError")]
|
|
pub url_endpoint: Option<UrlEndpoint>,
|
|
#[serde(default)]
|
|
#[serde_as(deserialize_as = "DefaultOnError")]
|
|
pub command_metadata: Option<CommandMetadata>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct WatchEndpoint {
|
|
pub video_id: String,
|
|
pub playlist_id: Option<String>,
|
|
#[serde(default)]
|
|
pub start_time_seconds: u32,
|
|
#[serde(default)]
|
|
pub watch_endpoint_music_supported_configs: WatchEndpointConfigWrap,
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
pub(crate) struct BrowseEndpoint {
|
|
pub browse_id: String,
|
|
pub params: String,
|
|
pub browse_endpoint_context_supported_configs: Option<BrowseEndpointConfig>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct BrowseEndpointWrap {
|
|
pub browse_endpoint: BrowseEndpoint,
|
|
}
|
|
|
|
impl<'de> Deserialize<'de> for BrowseEndpoint {
|
|
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
|
where
|
|
D: serde::Deserializer<'de>,
|
|
{
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
struct BEp {
|
|
pub browse_id: String,
|
|
#[serde(default)]
|
|
pub params: String,
|
|
pub browse_endpoint_context_supported_configs: Option<BrowseEndpointConfig>,
|
|
}
|
|
|
|
let bep = BEp::deserialize(deserializer)?;
|
|
|
|
// Remove the VL prefix from the playlist id
|
|
#[allow(clippy::map_unwrap_or)]
|
|
let browse_id = bep
|
|
.browse_endpoint_context_supported_configs
|
|
.as_ref()
|
|
.and_then(
|
|
|cfg| match cfg.browse_endpoint_context_music_config.page_type {
|
|
PageType::Playlist => bep.browse_id.strip_prefix("VL"),
|
|
_ => None,
|
|
},
|
|
)
|
|
.map(str::to_owned)
|
|
.unwrap_or(bep.browse_id);
|
|
|
|
Ok(Self {
|
|
browse_id,
|
|
params: bep.params,
|
|
browse_endpoint_context_supported_configs: bep
|
|
.browse_endpoint_context_supported_configs,
|
|
})
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct UrlEndpoint {
|
|
pub url: String,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct BrowseEndpointConfig {
|
|
pub browse_endpoint_context_music_config: BrowseEndpointMusicConfig,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct BrowseEndpointMusicConfig {
|
|
pub page_type: PageType,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct CommandMetadata {
|
|
pub web_command_metadata: WebCommandMetadata,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct WebCommandMetadata {
|
|
pub web_page_type: PageType,
|
|
}
|
|
|
|
#[derive(Default, Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct WatchEndpointConfigWrap {
|
|
pub watch_endpoint_music_config: WatchEndpointConfig,
|
|
}
|
|
|
|
#[serde_as]
|
|
#[derive(Default, Debug, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub(crate) struct WatchEndpointConfig {
|
|
#[serde(default)]
|
|
#[serde_as(deserialize_as = "DefaultOnError")]
|
|
pub music_video_type: MusicVideoType,
|
|
}
|
|
|
|
#[derive(Default, Debug, Clone, Copy, Deserialize, PartialEq, Eq)]
|
|
pub(crate) enum MusicVideoType {
|
|
#[default]
|
|
#[serde(rename = "MUSIC_VIDEO_TYPE_OMV")]
|
|
Video,
|
|
#[serde(rename = "MUSIC_VIDEO_TYPE_ATV")]
|
|
Track,
|
|
}
|
|
|
|
#[derive(Debug, Clone, Copy, Deserialize, PartialEq, Eq)]
|
|
pub(crate) enum PageType {
|
|
#[serde(
|
|
rename = "MUSIC_PAGE_TYPE_ARTIST",
|
|
alias = "MUSIC_PAGE_TYPE_AUDIOBOOK_ARTIST"
|
|
)]
|
|
Artist,
|
|
#[serde(rename = "MUSIC_PAGE_TYPE_ARTIST_DISCOGRAPHY")]
|
|
ArtistDiscography,
|
|
#[serde(rename = "MUSIC_PAGE_TYPE_ALBUM", alias = "MUSIC_PAGE_TYPE_AUDIOBOOK")]
|
|
Album,
|
|
#[serde(
|
|
rename = "WEB_PAGE_TYPE_CHANNEL",
|
|
alias = "MUSIC_PAGE_TYPE_USER_CHANNEL"
|
|
)]
|
|
Channel,
|
|
#[serde(rename = "MUSIC_PAGE_TYPE_PLAYLIST", alias = "WEB_PAGE_TYPE_PLAYLIST")]
|
|
Playlist,
|
|
#[serde(rename = "MUSIC_PAGE_TYPE_UNKNOWN")]
|
|
Unknown,
|
|
}
|
|
|
|
impl PageType {
|
|
pub(crate) fn to_url_target(self, id: String) -> Option<UrlTarget> {
|
|
match self {
|
|
PageType::Artist | PageType::Channel => Some(UrlTarget::Channel { id }),
|
|
PageType::ArtistDiscography => id
|
|
.strip_prefix(util::ARTIST_DISCOGRAPHY_PREFIX)
|
|
.map(|id| UrlTarget::Channel { id: id.to_owned() }),
|
|
PageType::Album => Some(UrlTarget::Album { id }),
|
|
PageType::Playlist => Some(UrlTarget::Playlist { id }),
|
|
PageType::Unknown => None,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
|
|
pub(crate) enum MusicPageType {
|
|
Artist,
|
|
Album,
|
|
Playlist,
|
|
Track { is_video: bool },
|
|
Unknown,
|
|
None,
|
|
}
|
|
|
|
impl From<PageType> for MusicPageType {
|
|
fn from(t: PageType) -> Self {
|
|
match t {
|
|
PageType::Artist => MusicPageType::Artist,
|
|
PageType::Album => MusicPageType::Album,
|
|
PageType::Playlist => MusicPageType::Playlist,
|
|
PageType::Channel | PageType::ArtistDiscography => MusicPageType::None,
|
|
PageType::Unknown => MusicPageType::Unknown,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl NavigationEndpoint {
|
|
pub(crate) fn music_page(self) -> Option<(MusicPageType, String)> {
|
|
self.browse_endpoint
|
|
.and_then(|be| {
|
|
be.browse_endpoint_context_supported_configs.map(|config| {
|
|
(
|
|
config.browse_endpoint_context_music_config.page_type.into(),
|
|
be.browse_id,
|
|
)
|
|
})
|
|
})
|
|
.or_else(|| {
|
|
self.watch_endpoint.map(|watch| {
|
|
if watch
|
|
.playlist_id
|
|
.map(|plid| plid.starts_with("RDQM"))
|
|
.unwrap_or_default()
|
|
{
|
|
// Genre radios (e.g. "pop radio") will be skipped
|
|
(MusicPageType::None, watch.video_id)
|
|
} else {
|
|
(
|
|
MusicPageType::Track {
|
|
is_video: watch
|
|
.watch_endpoint_music_supported_configs
|
|
.watch_endpoint_music_config
|
|
.music_video_type
|
|
== MusicVideoType::Video,
|
|
},
|
|
watch.video_id,
|
|
)
|
|
}
|
|
})
|
|
})
|
|
}
|
|
}
|