|
@@ -1,15 +1,17 @@
|
|
use futures::sync::oneshot;
|
|
use futures::sync::oneshot;
|
|
use futures::{future, Future};
|
|
use futures::{future, Future};
|
|
|
|
+use std;
|
|
use std::borrow::Cow;
|
|
use std::borrow::Cow;
|
|
|
|
+use std::io::{Read, Seek, SeekFrom, Result};
|
|
use std::mem;
|
|
use std::mem;
|
|
|
|
+use std::process::Command;
|
|
use std::sync::mpsc::{RecvError, TryRecvError, RecvTimeoutError};
|
|
use std::sync::mpsc::{RecvError, TryRecvError, RecvTimeoutError};
|
|
use std::thread;
|
|
use std::thread;
|
|
use std::time::Duration;
|
|
use std::time::Duration;
|
|
-use std;
|
|
|
|
|
|
|
|
use core::config::{Bitrate, PlayerConfig};
|
|
use core::config::{Bitrate, PlayerConfig};
|
|
use core::session::Session;
|
|
use core::session::Session;
|
|
-use core::util::{self, SpotifyId, Subfile};
|
|
|
|
|
|
+use core::util::SpotifyId;
|
|
|
|
|
|
use audio_backend::Sink;
|
|
use audio_backend::Sink;
|
|
use audio::{AudioFile, AudioDecrypt};
|
|
use audio::{AudioFile, AudioDecrypt};
|
|
@@ -375,13 +377,13 @@ impl PlayerInternal {
|
|
|
|
|
|
fn run_onstart(&self) {
|
|
fn run_onstart(&self) {
|
|
if let Some(ref program) = self.config.onstart {
|
|
if let Some(ref program) = self.config.onstart {
|
|
- util::run_program(program)
|
|
|
|
|
|
+ run_program(program)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
fn run_onstop(&self) {
|
|
fn run_onstop(&self) {
|
|
if let Some(ref program) = self.config.onstop {
|
|
if let Some(ref program) = self.config.onstop {
|
|
- util::run_program(program)
|
|
|
|
|
|
+ run_program(program)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
@@ -478,3 +480,50 @@ impl ::std::fmt::Debug for PlayerCommand {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+struct Subfile<T: Read + Seek> {
|
|
|
|
+ stream: T,
|
|
|
|
+ offset: u64,
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+impl<T: Read + Seek> Subfile<T> {
|
|
|
|
+ pub fn new(mut stream: T, offset: u64) -> Subfile<T> {
|
|
|
|
+ stream.seek(SeekFrom::Start(offset)).unwrap();
|
|
|
|
+ Subfile {
|
|
|
|
+ stream: stream,
|
|
|
|
+ offset: offset,
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+impl<T: Read + Seek> Read for Subfile<T> {
|
|
|
|
+ fn read(&mut self, buf: &mut [u8]) -> Result<usize> {
|
|
|
|
+ self.stream.read(buf)
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+impl<T: Read + Seek> Seek for Subfile<T> {
|
|
|
|
+ fn seek(&mut self, mut pos: SeekFrom) -> Result<u64> {
|
|
|
|
+ pos = match pos {
|
|
|
|
+ SeekFrom::Start(offset) => SeekFrom::Start(offset + self.offset),
|
|
|
|
+ x => x,
|
|
|
|
+ };
|
|
|
|
+
|
|
|
|
+ let newpos = try!(self.stream.seek(pos));
|
|
|
|
+ if newpos > self.offset {
|
|
|
|
+ Ok(newpos - self.offset)
|
|
|
|
+ } else {
|
|
|
|
+ Ok(0)
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+fn run_program(program: &str) {
|
|
|
|
+ info!("Running {}", program);
|
|
|
|
+ let mut v: Vec<&str> = program.split_whitespace().collect();
|
|
|
|
+ let status = Command::new(&v.remove(0))
|
|
|
|
+ .args(&v)
|
|
|
|
+ .status()
|
|
|
|
+ .expect("program failed to start");
|
|
|
|
+ info!("Exit status: {}", status);
|
|
|
|
+}
|