2017-01-10 16:31:12 +00:00
|
|
|
#[macro_use] extern crate log;
|
2015-07-19 20:36:14 +00:00
|
|
|
extern crate getopts;
|
2015-07-09 19:08:56 +00:00
|
|
|
extern crate librespot;
|
2016-12-31 11:51:44 +00:00
|
|
|
extern crate ctrlc;
|
2017-01-10 16:31:12 +00:00
|
|
|
extern crate env_logger;
|
2015-04-25 20:32:07 +00:00
|
|
|
|
2017-01-10 16:31:12 +00:00
|
|
|
use env_logger::LogBuilder;
|
2017-01-05 13:25:14 +00:00
|
|
|
use std::io::{stderr, Write};
|
2016-04-24 12:29:26 +00:00
|
|
|
use std::process::exit;
|
2015-07-02 17:24:25 +00:00
|
|
|
use std::thread;
|
2017-01-10 16:31:12 +00:00
|
|
|
use std::env;
|
|
|
|
use std::path::PathBuf;
|
|
|
|
use std::str::FromStr;
|
2015-07-19 20:36:14 +00:00
|
|
|
|
2015-07-08 19:50:44 +00:00
|
|
|
use librespot::spirc::SpircManager;
|
2017-01-06 16:18:41 +00:00
|
|
|
use librespot::authentication::get_credentials;
|
2017-01-10 16:31:12 +00:00
|
|
|
use librespot::audio_backend::{self, BACKENDS};
|
|
|
|
use librespot::cache::{Cache, DefaultCache, NoCache};
|
|
|
|
use librespot::player::Player;
|
|
|
|
use librespot::session::{Bitrate, Config, Session};
|
2017-01-20 19:39:05 +00:00
|
|
|
use librespot::mixer::softmixer::SoftMixer;
|
2017-01-10 16:31:12 +00:00
|
|
|
use librespot::version;
|
2016-03-16 00:05:05 +00:00
|
|
|
|
2016-03-14 23:41:51 +00:00
|
|
|
fn usage(program: &str, opts: &getopts::Options) -> String {
|
2015-07-19 20:36:14 +00:00
|
|
|
let brief = format!("Usage: {} [options]", program);
|
|
|
|
format!("{}", opts.usage(&brief))
|
|
|
|
}
|
|
|
|
|
2017-01-10 16:31:12 +00:00
|
|
|
fn setup_logging(verbose: bool) {
|
|
|
|
let mut builder = LogBuilder::new();
|
2016-01-02 02:30:24 +00:00
|
|
|
|
2017-01-10 16:31:12 +00:00
|
|
|
match env::var("RUST_LOG") {
|
|
|
|
Ok(config) => {
|
|
|
|
builder.parse(&config);
|
|
|
|
builder.init().unwrap();
|
|
|
|
|
|
|
|
if verbose {
|
|
|
|
warn!("`--verbose` flag overidden by `RUST_LOG` environment variable");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(_) => {
|
|
|
|
if verbose {
|
|
|
|
builder.parse("mdns=info,librespot=trace");
|
|
|
|
} else {
|
|
|
|
builder.parse("mdns=info,librespot=info");
|
|
|
|
}
|
|
|
|
builder.init().unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn list_backends() {
|
|
|
|
println!("Available Backends : ");
|
|
|
|
for (&(name, _), idx) in BACKENDS.iter().zip(0..) {
|
|
|
|
if idx == 0 {
|
|
|
|
println!("- {} (default)", name);
|
|
|
|
} else {
|
|
|
|
println!("- {}", name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn setup(args: &[String]) -> (Session, Player) {
|
|
|
|
let mut opts = getopts::Options::new();
|
|
|
|
opts.optopt("c", "cache", "Path to a directory where files will be cached.", "CACHE")
|
|
|
|
.reqopt("n", "name", "Device name", "NAME")
|
|
|
|
.optopt("b", "bitrate", "Bitrate (96, 160 or 320). Defaults to 160", "BITRATE")
|
|
|
|
.optopt("", "onstart", "Run PROGRAM when playback is about to begin.", "PROGRAM")
|
|
|
|
.optopt("", "onstop", "Run PROGRAM when playback has ended.", "PROGRAM")
|
|
|
|
.optflag("v", "verbose", "Enable verbose output")
|
|
|
|
.optopt("u", "username", "Username to sign in with", "USERNAME")
|
|
|
|
.optopt("p", "password", "Password", "PASSWORD")
|
|
|
|
.optopt("", "backend", "Audio backend to use. Use '?' to list options", "BACKEND")
|
|
|
|
.optopt("", "device", "Audio device to use. Use '?' to list options", "DEVICE");
|
2016-03-14 23:41:51 +00:00
|
|
|
|
2015-07-19 20:36:14 +00:00
|
|
|
let matches = match opts.parse(&args[1..]) {
|
2016-01-02 15:19:39 +00:00
|
|
|
Ok(m) => m,
|
|
|
|
Err(f) => {
|
2017-01-05 13:25:14 +00:00
|
|
|
writeln!(stderr(), "error: {}\n{}", f.to_string(), usage(&args[0], &opts)).unwrap();
|
|
|
|
exit(1);
|
2016-04-24 11:15:53 +00:00
|
|
|
}
|
|
|
|
};
|
2016-03-13 20:45:31 +00:00
|
|
|
|
2017-01-10 16:31:12 +00:00
|
|
|
let verbose = matches.opt_present("verbose");
|
|
|
|
setup_logging(verbose);
|
|
|
|
|
|
|
|
info!("librespot {} ({}). Built on {}.",
|
|
|
|
version::short_sha(),
|
|
|
|
version::commit_date(),
|
|
|
|
version::short_now());
|
|
|
|
|
|
|
|
let backend_name = matches.opt_str("backend");
|
|
|
|
if backend_name == Some("?".into()) {
|
|
|
|
list_backends();
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
let backend = audio_backend::find(backend_name.as_ref())
|
|
|
|
.expect("Invalid backend");
|
|
|
|
|
|
|
|
let bitrate = matches.opt_str("b").as_ref()
|
|
|
|
.map(|bitrate| Bitrate::from_str(bitrate).expect("Invalid bitrate"))
|
|
|
|
.unwrap_or(Bitrate::Bitrate160);
|
|
|
|
|
|
|
|
let config = Config {
|
|
|
|
user_agent: version::version_string(),
|
|
|
|
device_name: matches.opt_str("name").unwrap(),
|
|
|
|
bitrate: bitrate,
|
|
|
|
onstart: matches.opt_str("onstart"),
|
|
|
|
onstop: matches.opt_str("onstop"),
|
|
|
|
};
|
|
|
|
|
|
|
|
let cache = matches.opt_str("c").map(|cache_location| {
|
|
|
|
Box::new(DefaultCache::new(PathBuf::from(cache_location)).unwrap())
|
|
|
|
as Box<Cache + Send + Sync>
|
|
|
|
}).unwrap_or_else(|| Box::new(NoCache));
|
|
|
|
|
|
|
|
let session = Session::new(config, cache);
|
2017-01-05 13:25:14 +00:00
|
|
|
|
2017-01-06 16:18:41 +00:00
|
|
|
let credentials = get_credentials(&session, matches.opt_str("username"),
|
2017-01-10 16:31:12 +00:00
|
|
|
matches.opt_str("password"));
|
2016-04-24 12:29:26 +00:00
|
|
|
session.login(credentials).unwrap();
|
2015-05-09 10:07:24 +00:00
|
|
|
|
2017-01-20 19:39:05 +00:00
|
|
|
let mixer = SoftMixer::new();
|
|
|
|
|
2017-01-10 16:31:12 +00:00
|
|
|
let device_name = matches.opt_str("device");
|
2017-01-20 19:39:05 +00:00
|
|
|
let player = Player::new(session.clone(), Box::new(mixer), move || {
|
2017-01-10 16:31:12 +00:00
|
|
|
(backend)(device_name.as_ref().map(AsRef::as_ref))
|
|
|
|
});
|
|
|
|
|
|
|
|
(session, player)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let args: Vec<String> = std::env::args().collect();
|
|
|
|
let (session, player) = setup(&args);
|
2016-03-20 16:16:11 +00:00
|
|
|
|
2016-03-13 22:35:09 +00:00
|
|
|
let spirc = SpircManager::new(session.clone(), player);
|
2016-09-20 18:59:41 +00:00
|
|
|
let spirc_signal = spirc.clone();
|
2016-01-02 15:19:39 +00:00
|
|
|
thread::spawn(move || spirc.run());
|
2016-12-31 11:51:44 +00:00
|
|
|
|
|
|
|
ctrlc::set_handler(move || {
|
|
|
|
spirc_signal.send_goodbye();
|
|
|
|
exit(0);
|
|
|
|
});
|
2015-07-02 17:24:25 +00:00
|
|
|
|
2016-01-01 23:16:12 +00:00
|
|
|
loop {
|
|
|
|
session.poll();
|
|
|
|
}
|
2015-07-01 17:49:03 +00:00
|
|
|
}
|