use portaudio; use vorbis; use std::sync::{mpsc, Mutex, Arc, Condvar}; use std::thread; use metadata::TrackRef; use session::Session; use audio_file::AudioFile; use audio_decrypt::AudioDecrypt; use util::{self, SpotifyId, Subfile}; use librespot_protocol::spirc::PlayStatus; pub enum PlayerCommand { Load(SpotifyId, bool, u32), Play, Pause, Stop, Seek(u32) } pub struct PlayerState { pub status: PlayStatus, pub position_ms: u32, pub position_measured_at: i64, pub update_time: i64 } struct PlayerInternal<'s> { state: Arc<(Mutex, Condvar)>, session: &'s Session, commands: mpsc::Receiver, } pub struct Player<'s> { pub state: Arc<(Mutex, Condvar)>, commands: mpsc::Sender, #[allow(dead_code)] thread: thread::JoinGuard<'s, ()>, } impl <'s> Player<'s> { pub fn new(session: &Session) -> Player { let (cmd_tx, cmd_rx) = mpsc::channel(); let state = Arc::new((Mutex::new(PlayerState { status: PlayStatus::kPlayStatusStop, position_ms: 0, position_measured_at: 0, update_time: util::now_ms(), }), Condvar::new())); let internal = PlayerInternal { session: session, commands: cmd_rx, state: state.clone() }; Player { commands: cmd_tx, state: state, thread: thread::scoped(move || { internal.run() }) } } pub fn command(&self, cmd: PlayerCommand) { self.commands.send(cmd).unwrap(); } } impl <'s> PlayerInternal<'s> { fn run(self) { portaudio::initialize().unwrap(); let stream = portaudio::stream::Stream::::open_default( 0, 2, 44100.0, portaudio::stream::FRAMES_PER_BUFFER_UNSPECIFIED, None ).unwrap(); let mut decoder = None; loop { match self.commands.try_recv() { Ok(PlayerCommand::Load(id, play, position)) => { println!("Load"); let mut h = self.state.0.lock().unwrap(); if h.status == PlayStatus::kPlayStatusPlay { stream.stop().unwrap(); } h.status = PlayStatus::kPlayStatusLoading; h.position_ms = position; h.position_measured_at = util::now_ms(); h.update_time = util::now_ms(); drop(h); let track : TrackRef = self.session.metadata(id); let file_id = *track.wait().unwrap().files.first().unwrap(); let key = self.session.audio_key(track.id(), file_id).into_inner(); decoder = Some( vorbis::Decoder::new( Subfile::new( AudioDecrypt::new(key, AudioFile::new(self.session, file_id)), 0xa7)).unwrap()); decoder.as_mut().unwrap().time_seek(position as f64 / 1000f64).unwrap(); let mut h = self.state.0.lock().unwrap(); h.status = if play { stream.start().unwrap(); PlayStatus::kPlayStatusPlay } else { PlayStatus::kPlayStatusPause }; h.position_ms = position; h.position_measured_at = util::now_ms(); h.update_time = util::now_ms(); println!("Load Done"); } Ok(PlayerCommand::Seek(ms)) => { let mut h = self.state.0.lock().unwrap(); decoder.as_mut().unwrap().time_seek(ms as f64 / 1000f64).unwrap(); h.position_ms = (decoder.as_mut().unwrap().time_tell().unwrap() * 1000f64) as u32; h.position_measured_at = util::now_ms(); h.update_time = util::now_ms(); }, Ok(PlayerCommand::Play) => { println!("Play"); let mut h = self.state.0.lock().unwrap(); h.status = PlayStatus::kPlayStatusPlay; h.update_time = util::now_ms(); stream.start().unwrap(); }, Ok(PlayerCommand::Pause) => { let mut h = self.state.0.lock().unwrap(); h.status = PlayStatus::kPlayStatusPause; h.update_time = util::now_ms(); stream.stop().unwrap(); }, Ok(PlayerCommand::Stop) => { let mut h = self.state.0.lock().unwrap(); if h.status == PlayStatus::kPlayStatusPlay { stream.stop().unwrap(); } h.status = PlayStatus::kPlayStatusPause; h.update_time = util::now_ms(); decoder = None; }, Err(..) => (), } if self.state.0.lock().unwrap().status == PlayStatus::kPlayStatusPlay { match decoder.as_mut().unwrap().packets().next().unwrap() { Ok(packet) => { match stream.write(&packet.data) { Ok(_) => (), Err(portaudio::PaError::OutputUnderflowed) => eprintln!("Underflow"), Err(e) => panic!("PA Error {}", e) }; }, Err(vorbis::VorbisError::Hole) => (), Err(e) => panic!("Vorbis error {:?}", e) } let mut h = self.state.0.lock().unwrap(); h.position_ms = (decoder.as_mut().unwrap().time_tell().unwrap() * 1000f64) as u32; h.position_measured_at = util::now_ms(); } } drop(stream); portaudio::terminate().unwrap(); } }