|
@@ -12,12 +12,12 @@ use std::time::Duration;
|
|
|
|
|
|
use config::{Bitrate, PlayerConfig};
|
|
|
use librespot_core::session::Session;
|
|
|
-use librespot_core::spotify_id::{FileId, SpotifyId, SpotifyTrackType};
|
|
|
+use librespot_core::spotify_id::SpotifyId;
|
|
|
|
|
|
use audio::{AudioDecrypt, AudioFile};
|
|
|
use audio::{VorbisDecoder, VorbisPacket};
|
|
|
use audio_backend::Sink;
|
|
|
-use metadata::{Episode, FileFormat, Metadata, Track};
|
|
|
+use metadata::{AudioItem, FileFormat};
|
|
|
use mixer::AudioFilter;
|
|
|
|
|
|
pub struct Player {
|
|
@@ -512,87 +512,65 @@ impl PlayerInternal {
|
|
|
let _ = self.event_sender.unbounded_send(event.clone());
|
|
|
}
|
|
|
|
|
|
- fn find_available_alternative<'a>(&self, track: &'a Track) -> Option<Cow<'a, Track>> {
|
|
|
- if track.available {
|
|
|
- Some(Cow::Borrowed(track))
|
|
|
+ fn find_available_alternative<'a>(&self, audio: &'a AudioItem) -> Option<Cow<'a, AudioItem>> {
|
|
|
+ if audio.available {
|
|
|
+ Some(Cow::Borrowed(audio))
|
|
|
} else {
|
|
|
- let alternatives = track
|
|
|
- .alternatives
|
|
|
- .iter()
|
|
|
- .map(|alt_id| Track::get(&self.session, *alt_id));
|
|
|
- let alternatives = future::join_all(alternatives).wait().unwrap();
|
|
|
-
|
|
|
- alternatives.into_iter().find(|alt| alt.available).map(Cow::Owned)
|
|
|
+ if let Some(alternatives) = &audio.alternatives {
|
|
|
+ let alternatives = alternatives
|
|
|
+ .iter()
|
|
|
+ .map(|alt_id| AudioItem::get_audio_item(&self.session, *alt_id));
|
|
|
+ let alternatives = future::join_all(alternatives).wait().unwrap();
|
|
|
+ alternatives.into_iter().find(|alt| alt.available).map(Cow::Owned)
|
|
|
+ } else {
|
|
|
+ None
|
|
|
+ }
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- fn get_file_id(&self, spotify_id: SpotifyId) -> Option<FileId> {
|
|
|
- let file_id = match spotify_id.track_type {
|
|
|
- SpotifyTrackType::Track => {
|
|
|
- let track = Track::get(&self.session, spotify_id).wait().unwrap();
|
|
|
-
|
|
|
- info!(
|
|
|
- "Loading track \"{}\" with Spotify URI \"spotify:track:{}\"",
|
|
|
- track.name,
|
|
|
- spotify_id.to_base62()
|
|
|
- );
|
|
|
-
|
|
|
- let track = match self.find_available_alternative(&track) {
|
|
|
- Some(track) => track,
|
|
|
- None => {
|
|
|
- warn!("Track \"{}\" is not available", track.name);
|
|
|
- return None;
|
|
|
- }
|
|
|
- };
|
|
|
+ fn load_track(&self, spotify_id: SpotifyId, position: i64) -> Option<(Decoder, f32)> {
|
|
|
+ let audio = AudioItem::get_audio_item(&self.session, spotify_id)
|
|
|
+ .wait()
|
|
|
+ .unwrap();
|
|
|
+ info!("Loading <{}> with Spotify URI <{}>", audio.name, audio.uri);
|
|
|
|
|
|
- let format = match self.config.bitrate {
|
|
|
- Bitrate::Bitrate96 => FileFormat::OGG_VORBIS_96,
|
|
|
- Bitrate::Bitrate160 => FileFormat::OGG_VORBIS_160,
|
|
|
- Bitrate::Bitrate320 => FileFormat::OGG_VORBIS_320,
|
|
|
- };
|
|
|
- match track.files.get(&format) {
|
|
|
- Some(&file_id) => file_id,
|
|
|
- None => {
|
|
|
- warn!("Track \"{}\" is not available in format {:?}", track.name, format);
|
|
|
- return None;
|
|
|
- }
|
|
|
- }
|
|
|
+ let audio = match self.find_available_alternative(&audio) {
|
|
|
+ Some(audio) => audio,
|
|
|
+ None => {
|
|
|
+ warn!("<{}> is not available", audio.uri);
|
|
|
+ return None;
|
|
|
}
|
|
|
- // This should be refactored!
|
|
|
- SpotifyTrackType::Podcast => {
|
|
|
- let episode = Episode::get(&self.session, spotify_id).wait().unwrap();
|
|
|
- info!("Episode {:?}", episode);
|
|
|
-
|
|
|
- info!(
|
|
|
- "Loading episode \"{}\" with Spotify URI \"spotify:episode:{}\"",
|
|
|
- episode.name,
|
|
|
- spotify_id.to_base62()
|
|
|
- );
|
|
|
-
|
|
|
- // Podcasts seem to have only 96 OGG_VORBIS support, other filetypes indicate
|
|
|
- // AAC_24, MP4_128, MP4_128_DUAL, MP3_96 among others
|
|
|
- let format = match self.config.bitrate {
|
|
|
- _ => FileFormat::OGG_VORBIS_96,
|
|
|
- };
|
|
|
+ };
|
|
|
+ // (Most) podcasts seem to support only 96 bit Vorbis, so fall back to it
|
|
|
+ let formats = match self.config.bitrate {
|
|
|
+ Bitrate::Bitrate96 => [
|
|
|
+ FileFormat::OGG_VORBIS_96,
|
|
|
+ FileFormat::OGG_VORBIS_160,
|
|
|
+ FileFormat::OGG_VORBIS_320,
|
|
|
+ ],
|
|
|
+ Bitrate::Bitrate160 => [
|
|
|
+ FileFormat::OGG_VORBIS_160,
|
|
|
+ FileFormat::OGG_VORBIS_96,
|
|
|
+ FileFormat::OGG_VORBIS_320,
|
|
|
+ ],
|
|
|
+ Bitrate::Bitrate320 => [
|
|
|
+ FileFormat::OGG_VORBIS_320,
|
|
|
+ FileFormat::OGG_VORBIS_160,
|
|
|
+ FileFormat::OGG_VORBIS_96,
|
|
|
+ ],
|
|
|
+ };
|
|
|
+ let format = formats
|
|
|
+ .iter()
|
|
|
+ .find(|format| audio.files.contains_key(format))
|
|
|
+ .unwrap();
|
|
|
|
|
|
- match episode.files.get(&format) {
|
|
|
- Some(&file_id) => file_id,
|
|
|
- None => {
|
|
|
- warn!(
|
|
|
- "Episode \"{}\" is not available in format {:?}",
|
|
|
- episode.name, format
|
|
|
- );
|
|
|
- return None;
|
|
|
- }
|
|
|
- }
|
|
|
+ let file_id = match audio.files.get(&format) {
|
|
|
+ Some(&file_id) => file_id,
|
|
|
+ None => {
|
|
|
+ warn!("<{}> in not available in format {:?}", audio.name, format);
|
|
|
+ return None;
|
|
|
}
|
|
|
};
|
|
|
- return Some(file_id);
|
|
|
- }
|
|
|
-
|
|
|
- fn load_track(&self, spotify_id: SpotifyId, position: i64) -> Option<(Decoder, f32)> {
|
|
|
- let file_id = self.get_file_id(spotify_id).unwrap();
|
|
|
- info!("{:?} -> {:?}", spotify_id, file_id);
|
|
|
|
|
|
let key = self.session.audio_key().request(spotify_id, file_id);
|
|
|
let encrypted_file = AudioFile::open(&self.session, file_id);
|
|
@@ -619,9 +597,7 @@ impl PlayerInternal {
|
|
|
Err(err) => error!("Vorbis error: {:?}", err),
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
- // info!("Track \"{}\" loaded", track.name);
|
|
|
-
|
|
|
+ info!("<{}> loaded", audio.name);
|
|
|
Some((decoder, normalisation_factor))
|
|
|
}
|
|
|
}
|