|
@@ -8,12 +8,13 @@ extern crate librespot_protocol as protocol;
|
|
|
|
|
|
pub mod cover;
|
|
|
|
|
|
+use futures::future;
|
|
|
use futures::Future;
|
|
|
use linear_map::LinearMap;
|
|
|
|
|
|
use librespot_core::mercury::MercuryError;
|
|
|
use librespot_core::session::Session;
|
|
|
-use librespot_core::spotify_id::{FileId, SpotifyId};
|
|
|
+use librespot_core::spotify_id::{FileId, SpotifyAudioType, SpotifyId};
|
|
|
|
|
|
pub use protocol::metadata::AudioFile_Format as FileFormat;
|
|
|
|
|
@@ -52,13 +53,81 @@ where
|
|
|
&& (!has_allowed || countrylist_contains(allowed.as_str(), country))
|
|
|
}
|
|
|
|
|
|
+// A wrapper with fields the player needs
|
|
|
+#[derive(Debug, Clone)]
|
|
|
+pub struct AudioItem {
|
|
|
+ pub id: SpotifyId,
|
|
|
+ pub uri: String,
|
|
|
+ pub files: LinearMap<FileFormat, FileId>,
|
|
|
+ pub name: String,
|
|
|
+ pub available: bool,
|
|
|
+ pub alternatives: Option<Vec<SpotifyId>>,
|
|
|
+}
|
|
|
+
|
|
|
+impl AudioItem {
|
|
|
+ pub fn get_audio_item(
|
|
|
+ session: &Session,
|
|
|
+ id: SpotifyId,
|
|
|
+ ) -> Box<dyn Future<Item = AudioItem, Error = MercuryError>> {
|
|
|
+ match id.audio_type {
|
|
|
+ SpotifyAudioType::Track => Track::get_audio_item(session, id),
|
|
|
+ SpotifyAudioType::Podcast => Episode::get_audio_item(session, id),
|
|
|
+ SpotifyAudioType::NonPlayable => {
|
|
|
+ Box::new(future::err::<AudioItem, MercuryError>(MercuryError))
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+trait AudioFiles {
|
|
|
+ fn get_audio_item(
|
|
|
+ session: &Session,
|
|
|
+ id: SpotifyId,
|
|
|
+ ) -> Box<dyn Future<Item = AudioItem, Error = MercuryError>>;
|
|
|
+}
|
|
|
+
|
|
|
+impl AudioFiles for Track {
|
|
|
+ fn get_audio_item(
|
|
|
+ session: &Session,
|
|
|
+ id: SpotifyId,
|
|
|
+ ) -> Box<dyn Future<Item = AudioItem, Error = MercuryError>> {
|
|
|
+ Box::new(Self::get(session, id).and_then(move |item| {
|
|
|
+ Ok(AudioItem {
|
|
|
+ id: id,
|
|
|
+ uri: format!("spotify:track:{}", id.to_base62()),
|
|
|
+ files: item.files,
|
|
|
+ name: item.name,
|
|
|
+ available: item.available,
|
|
|
+ alternatives: Some(item.alternatives),
|
|
|
+ })
|
|
|
+ }))
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl AudioFiles for Episode {
|
|
|
+ fn get_audio_item(
|
|
|
+ session: &Session,
|
|
|
+ id: SpotifyId,
|
|
|
+ ) -> Box<dyn Future<Item = AudioItem, Error = MercuryError>> {
|
|
|
+ Box::new(Self::get(session, id).and_then(move |item| {
|
|
|
+ Ok(AudioItem {
|
|
|
+ id: id,
|
|
|
+ uri: format!("spotify:episode:{}", id.to_base62()),
|
|
|
+ files: item.files,
|
|
|
+ name: item.name,
|
|
|
+ available: item.available,
|
|
|
+ alternatives: None,
|
|
|
+ })
|
|
|
+ }))
|
|
|
+ }
|
|
|
+}
|
|
|
pub trait Metadata: Send + Sized + 'static {
|
|
|
type Message: protobuf::Message;
|
|
|
|
|
|
fn base_url() -> &'static str;
|
|
|
fn parse(msg: &Self::Message, session: &Session) -> Self;
|
|
|
|
|
|
- fn get(session: &Session, id: SpotifyId) -> Box<Future<Item = Self, Error = MercuryError>> {
|
|
|
+ fn get(session: &Session, id: SpotifyId) -> Box<dyn Future<Item = Self, Error = MercuryError>> {
|
|
|
let uri = format!("{}/{}", Self::base_url(), id.to_base16());
|
|
|
let request = session.mercury().get(uri);
|
|
|
|
|
@@ -93,6 +162,29 @@ pub struct Album {
|
|
|
pub covers: Vec<FileId>,
|
|
|
}
|
|
|
|
|
|
+#[derive(Debug, Clone)]
|
|
|
+pub struct Episode {
|
|
|
+ pub id: SpotifyId,
|
|
|
+ pub name: String,
|
|
|
+ pub external_url: String,
|
|
|
+ pub duration: i32,
|
|
|
+ pub language: String,
|
|
|
+ pub show: SpotifyId,
|
|
|
+ pub files: LinearMap<FileFormat, FileId>,
|
|
|
+ pub covers: Vec<FileId>,
|
|
|
+ pub available: bool,
|
|
|
+ pub explicit: bool,
|
|
|
+}
|
|
|
+
|
|
|
+#[derive(Debug, Clone)]
|
|
|
+pub struct Show {
|
|
|
+ pub id: SpotifyId,
|
|
|
+ pub name: String,
|
|
|
+ pub publisher: String,
|
|
|
+ pub episodes: Vec<SpotifyId>,
|
|
|
+ pub covers: Vec<FileId>,
|
|
|
+}
|
|
|
+
|
|
|
#[derive(Debug, Clone)]
|
|
|
pub struct Artist {
|
|
|
pub id: SpotifyId,
|
|
@@ -222,6 +314,92 @@ impl Metadata for Artist {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+// Podcast
|
|
|
+impl Metadata for Episode {
|
|
|
+ type Message = protocol::metadata::Episode;
|
|
|
+
|
|
|
+ fn base_url() -> &'static str {
|
|
|
+ "hm://metadata/3/episode"
|
|
|
+ }
|
|
|
+
|
|
|
+ fn parse(msg: &Self::Message, session: &Session) -> Self {
|
|
|
+ let country = session.country();
|
|
|
+
|
|
|
+ let files = msg
|
|
|
+ .get_file()
|
|
|
+ .iter()
|
|
|
+ .filter(|file| file.has_file_id())
|
|
|
+ .map(|file| {
|
|
|
+ let mut dst = [0u8; 20];
|
|
|
+ dst.clone_from_slice(file.get_file_id());
|
|
|
+ (file.get_format(), FileId(dst))
|
|
|
+ })
|
|
|
+ .collect();
|
|
|
+
|
|
|
+ let covers = msg
|
|
|
+ .get_covers()
|
|
|
+ .get_image()
|
|
|
+ .iter()
|
|
|
+ .filter(|image| image.has_file_id())
|
|
|
+ .map(|image| {
|
|
|
+ let mut dst = [0u8; 20];
|
|
|
+ dst.clone_from_slice(image.get_file_id());
|
|
|
+ FileId(dst)
|
|
|
+ })
|
|
|
+ .collect::<Vec<_>>();
|
|
|
+
|
|
|
+ Episode {
|
|
|
+ id: SpotifyId::from_raw(msg.get_gid()).unwrap(),
|
|
|
+ name: msg.get_name().to_owned(),
|
|
|
+ external_url: msg.get_external_url().to_owned(),
|
|
|
+ duration: msg.get_duration().to_owned(),
|
|
|
+ language: msg.get_language().to_owned(),
|
|
|
+ show: SpotifyId::from_raw(msg.get_show().get_gid()).unwrap(),
|
|
|
+ covers: covers,
|
|
|
+ files: files,
|
|
|
+ available: parse_restrictions(msg.get_restriction(), &country, "premium"),
|
|
|
+ explicit: msg.get_explicit().to_owned(),
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl Metadata for Show {
|
|
|
+ type Message = protocol::metadata::Show;
|
|
|
+
|
|
|
+ fn base_url() -> &'static str {
|
|
|
+ "hm://metadata/3/show"
|
|
|
+ }
|
|
|
+
|
|
|
+ fn parse(msg: &Self::Message, _: &Session) -> Self {
|
|
|
+ let episodes = msg
|
|
|
+ .get_episode()
|
|
|
+ .iter()
|
|
|
+ .filter(|episode| episode.has_gid())
|
|
|
+ .map(|episode| SpotifyId::from_raw(episode.get_gid()).unwrap())
|
|
|
+ .collect::<Vec<_>>();
|
|
|
+
|
|
|
+ let covers = msg
|
|
|
+ .get_covers()
|
|
|
+ .get_image()
|
|
|
+ .iter()
|
|
|
+ .filter(|image| image.has_file_id())
|
|
|
+ .map(|image| {
|
|
|
+ let mut dst = [0u8; 20];
|
|
|
+ dst.clone_from_slice(image.get_file_id());
|
|
|
+ FileId(dst)
|
|
|
+ })
|
|
|
+ .collect::<Vec<_>>();
|
|
|
+
|
|
|
+ Show {
|
|
|
+ id: SpotifyId::from_raw(msg.get_gid()).unwrap(),
|
|
|
+ name: msg.get_name().to_owned(),
|
|
|
+ publisher: msg.get_publisher().to_owned(),
|
|
|
+ episodes: episodes,
|
|
|
+ covers: covers,
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
struct StrChunks<'s>(&'s str, usize);
|
|
|
|
|
|
trait StrChunksExt {
|