#[cfg(unix)]
use libc;
use mock_command::*;
use protocol::CacheStats;
use std::collections::HashMap;
use std::env;
use std::ffi::OsString;
use std::fs::{self,File};
use std::io;
use std::path::{Path,PathBuf};
use std::sync::{Arc,Mutex};
use tempdir::TempDir;
macro_rules! stringvec {
( $( $x:expr ),* ) => {
vec!($( $x.to_owned(), )*)
};
}
macro_rules! assert_neq {
($left:expr , $right:expr) => ({
match (&($left), &($right)) {
(left_val, right_val) => {
if !(*left_val != *right_val) {
panic!("assertion failed: `(left != right)` \
(left: `{:?}`, right: `{:?}`)", left_val, right_val)
}
}
}
})
}
macro_rules! assert_map_contains {
( $map:ident , $( ($key:expr, $val:expr) ),* ) => {
$(
match $map.get(&$key) {
Some(&ref v) =>
assert!($val == *v, format!("{} key `{:?}` doesn't match expected! (expected `{:?}` != actual `{:?}`)", stringify!($map), $key, $val, v)),
None => panic!("{} missing key `{:?}`", stringify!($map), $key),
}
)*
}
}
pub fn new_creator() -> Arc<Mutex<MockCommandCreator>> {
Arc::new(Mutex::new(MockCommandCreator::new()))
}
pub fn next_command(creator : &Arc<Mutex<MockCommandCreator>>,
child: io::Result<MockChild>) {
creator.lock().unwrap().next_command_spawns(child);
}
pub fn next_command_calls<C: Fn() -> io::Result<MockChild> + Send + 'static>(creator : &Arc<Mutex<MockCommandCreator>>, call: C) {
creator.lock().unwrap().next_command_calls(call);
}
pub struct TestFixture {
pub tempdir: TempDir,
pub paths: OsString,
pub bins: Vec<PathBuf>,
}
pub const SUBDIRS: &'static [&'static str] = &["a", "b", "c"];
pub const BIN_NAME: &'static str = "bin";
pub fn create_file<F : FnOnce(File) -> io::Result<()>>(dir: &Path, path: &str, fill_contents: F) -> io::Result<PathBuf> {
let b = dir.join(path);
let f = try!(fs::File::create(&b));
try!(fill_contents(f));
b.canonicalize()
}
pub fn touch(dir: &Path, path: &str) -> io::Result<PathBuf> {
create_file(dir, path, |_f| Ok(()))
}
#[cfg(unix)]
pub fn mk_bin_contents<F : FnOnce(File) -> io::Result<()>>(dir: &Path, path: &str, fill_contents: F) -> io::Result<PathBuf> {
use std::os::unix::fs::OpenOptionsExt;
let bin = dir.join(path);
let f = try!(fs::OpenOptions::new()
.write(true)
.create(true)
.mode(0o666 | (libc::S_IXUSR as u32))
.open(&bin));
try!(fill_contents(f));
bin.canonicalize()
}
#[cfg(unix)]
pub fn mk_bin(dir: &Path, path: &str) -> io::Result<PathBuf> {
mk_bin_contents(dir, path, |_f| Ok(()))
}
#[cfg(not(unix))]
pub fn mk_bin_contents<F : FnOnce(File) -> io::Result<()>>(dir: &Path, path: &str, contents: F) -> io::Result<PathBuf> {
create_file(dir, Path::new(path).with_extension(env::consts::EXE_EXTENSION).to_str().unwrap(), contents)
}
#[cfg(not(unix))]
pub fn mk_bin(dir: &Path, path: &str) -> io::Result<PathBuf> {
touch(dir, Path::new(path).with_extension(env::consts::EXE_EXTENSION).to_str().unwrap())
}
impl TestFixture {
pub fn new() -> TestFixture {
let tempdir = TempDir::new("sccache_test").unwrap();
let mut builder = fs::DirBuilder::new();
builder.recursive(true);
let mut paths = vec!();
let mut bins = vec!();
for d in SUBDIRS.iter() {
let p = tempdir.path().join(d);
builder.create(&p).unwrap();
bins.push(mk_bin(&p, &BIN_NAME).unwrap());
paths.push(p);
}
TestFixture {
tempdir: tempdir,
paths: env::join_paths(paths).unwrap(),
bins: bins,
}
}
#[allow(dead_code)]
pub fn touch(&self, path: &str) -> io::Result<PathBuf> {
touch(self.tempdir.path(), &path)
}
}
#[derive(Debug, PartialEq)]
pub enum CacheStat {
Count(u64),
String(String),
Size(u64),
}
pub fn cache_stats_map(mut stats: CacheStats) -> HashMap<String, CacheStat> {
stats.take_stats().into_iter().map(|mut s| (s.take_name(), if s.has_count() { CacheStat::Count(s.get_count()) } else if s.has_size() { CacheStat::Size(s.get_size()) } else { CacheStat::String(s.take_str()) })).collect()
}
#[test]
fn test_map_contains_ok() {
use std::collections::HashMap;
let mut m = HashMap::new();
m.insert("a", 1);
m.insert("b", 2);
assert_map_contains!(m, ("a", 1), ("b", 2));
}
#[test]
#[should_panic]
fn test_map_contains_missing_key() {
use std::collections::HashMap;
let mut m = HashMap::new();
m.insert("a", 1);
assert_map_contains!(m, ("a", 1), ("b", 2));
}
#[test]
#[should_panic]
fn test_map_contains_wrong_value() {
use std::collections::HashMap;
let mut m = HashMap::new();
m.insert("a", 1);
m.insert("b", 3);
assert_map_contains!(m, ("a", 1), ("b", 2));
}