use num::{BigUint, Integer, Zero, One}; use rand::{Rng, Rand}; use std::io; use std::ops::{Mul, Rem, Shr}; use std::fs; use std::path::Path; use time; mod int128; mod spotify_id; mod arcvec; mod subfile; pub use util::int128::u128; pub use util::spotify_id::{SpotifyId, FileId}; pub use util::arcvec::ArcVec; pub use util::subfile::Subfile; #[macro_export] macro_rules! eprintln( ($($arg:tt)*) => ( { use std::io::Write; writeln!(&mut ::std::io::stderr(), $($arg)* ).unwrap() } ) ); #[macro_export] macro_rules! eprint( ($($arg:tt)*) => ( { use std::io::Write; write!(&mut ::std::io::stderr(), $($arg)* ).unwrap() } ) ); pub fn rand_vec(rng: &mut G, size: usize) -> Vec { rng.gen_iter().take(size).collect() } pub mod version { // FIXME: Unfortunately, this doesn't work when using syntex // And for some reason, cfg-gating it doesn't work //include!(concat!(env!("OUT_DIR"), "/version.rs")); pub fn short_sha() -> String { "unknown".to_owned() } pub fn version_string() -> String { format!("librespot-{}", short_sha()) } } pub fn hexdump(data: &[u8]) { for b in data.iter() { eprint!("{:02X} ", b); } eprintln!(""); } pub trait IgnoreExt { fn ignore(self); } impl IgnoreExt for Result { fn ignore(self) { match self { Ok(_) => (), Err(_) => (), } } } pub fn now_ms() -> i64 { let ts = time::now_utc().to_timespec(); ts.sec * 1000 + ts.nsec as i64 / 1000000 } pub fn mkdir_existing(path: &Path) -> io::Result<()> { fs::create_dir(path).or_else(|err| { if err.kind() == io::ErrorKind::AlreadyExists { Ok(()) } else { Err(err) } }) } pub fn powm(base: &BigUint, exp: &BigUint, modulus: &BigUint) -> BigUint { let mut base = base.clone(); let mut exp = exp.clone(); let mut result: BigUint = One::one(); while !exp.is_zero() { if exp.is_odd() { result = result.mul(&base).rem(modulus); } exp = exp.shr(1); base = (&base).mul(&base).rem(modulus); } result } pub struct StrChunks<'s>(&'s str, usize); pub trait StrChunksExt { fn chunks(&self, size: usize) -> StrChunks; } impl StrChunksExt for str { fn chunks(&self, size: usize) -> StrChunks { StrChunks(self, size) } } impl<'s> Iterator for StrChunks<'s> { type Item = &'s str; fn next(&mut self) -> Option<&'s str> { let &mut StrChunks(data, size) = self; if data.is_empty() { None } else { let ret = Some(&data[..size]); self.0 = &data[size..]; ret } } }