use std::{ convert::{TryFrom, TryInto}, fmt::Debug, ops::{Deref, DerefMut}, }; use uuid::Uuid; use crate::{ artist::{Artists, ArtistsWithRole}, audio::{ file::AudioFiles, item::{AudioItem, AudioItemResult, InnerAudioItem}, }, availability::{Availabilities, UnavailabilityReason}, content_rating::ContentRatings, external_id::ExternalIds, restriction::Restrictions, sale_period::SalePeriods, util::{impl_deref_wrapped, impl_try_from_repeated}, Album, Metadata, RequestResult, }; use librespot_core::{date::Date, Error, Session, SpotifyId}; use librespot_protocol as protocol; #[derive(Debug, Clone)] pub struct Track { pub id: SpotifyId, pub name: String, pub album: Album, pub artists: Artists, pub number: i32, pub disc_number: i32, pub duration: i32, pub popularity: i32, pub is_explicit: bool, pub external_ids: ExternalIds, pub restrictions: Restrictions, pub files: AudioFiles, pub alternatives: Tracks, pub sale_periods: SalePeriods, pub previews: AudioFiles, pub tags: Vec, pub earliest_live_timestamp: Date, pub has_lyrics: bool, pub availability: Availabilities, pub licensor: Uuid, pub language_of_performance: Vec, pub content_ratings: ContentRatings, pub original_title: String, pub version_title: String, pub artists_with_role: ArtistsWithRole, } #[derive(Debug, Clone, Default)] pub struct Tracks(pub Vec); impl_deref_wrapped!(Tracks, Vec); #[async_trait] impl InnerAudioItem for Track { async fn get_audio_item(session: &Session, id: SpotifyId) -> AudioItemResult { let track = Self::get(session, &id).await?; let alternatives = { if track.alternatives.is_empty() { None } else { Some(track.alternatives.clone()) } }; // TODO: check meaning of earliest_live_timestamp in let availability = if Date::now_utc() < track.earliest_live_timestamp { Err(UnavailabilityReason::Embargo) } else { Self::available_for_user( &session.user_data(), &track.availability, &track.restrictions, ) }; Ok(AudioItem { id, spotify_uri: id.to_uri()?, files: track.files, name: track.name, duration: track.duration, availability, alternatives, is_explicit: track.is_explicit, }) } } #[async_trait] impl Metadata for Track { type Message = protocol::metadata::Track; async fn request(session: &Session, track_id: &SpotifyId) -> RequestResult { session.spclient().get_track_metadata(track_id).await } fn parse(msg: &Self::Message, _: &SpotifyId) -> Result { Self::try_from(msg) } } impl TryFrom<&::Message> for Track { type Error = librespot_core::Error; fn try_from(track: &::Message) -> Result { Ok(Self { id: track.try_into()?, name: track.get_name().to_owned(), album: track.get_album().try_into()?, artists: track.get_artist().try_into()?, number: track.get_number(), disc_number: track.get_disc_number(), duration: track.get_duration(), popularity: track.get_popularity(), is_explicit: track.get_explicit(), external_ids: track.get_external_id().into(), restrictions: track.get_restriction().into(), files: track.get_file().into(), alternatives: track.get_alternative().try_into()?, sale_periods: track.get_sale_period().try_into()?, previews: track.get_preview().into(), tags: track.get_tags().to_vec(), earliest_live_timestamp: Date::from_timestamp_ms(track.get_earliest_live_timestamp())?, has_lyrics: track.get_has_lyrics(), availability: track.get_availability().try_into()?, licensor: Uuid::from_slice(track.get_licensor().get_uuid()) .unwrap_or_else(|_| Uuid::nil()), language_of_performance: track.get_language_of_performance().to_vec(), content_ratings: track.get_content_rating().into(), original_title: track.get_original_title().to_owned(), version_title: track.get_version_title().to_owned(), artists_with_role: track.get_artist_with_role().try_into()?, }) } } impl_try_from_repeated!(::Message, Tracks);