use std::fmt::Debug;
use std::panic;
use rand;
use tester::Status::{Discard, Fail, Pass};
use {Arbitrary, Gen, StdGen};
pub struct QuickCheck<G> {
tests: usize,
max_tests: usize,
gen: G,
}
impl QuickCheck<StdGen<rand::ThreadRng>> {
pub fn new() -> QuickCheck<StdGen<rand::ThreadRng>> {
QuickCheck {
tests: 100,
max_tests: 10000,
gen: StdGen::new(rand::thread_rng(), 100),
}
}
}
impl<G: Gen> QuickCheck<G> {
pub fn tests(mut self, tests: usize) -> QuickCheck<G> {
self.tests = tests;
self
}
pub fn max_tests(mut self, max_tests: usize) -> QuickCheck<G> {
self.max_tests = max_tests;
self
}
pub fn gen(mut self, gen: G) -> QuickCheck<G> {
self.gen = gen;
self
}
pub fn quicktest<A>(&mut self, f: A) -> Result<usize, TestResult>
where A: Testable {
let mut ntests: usize = 0;
for _ in 0..self.max_tests {
if ntests >= self.tests {
break
}
match f.result(&mut self.gen) {
TestResult { status: Pass, .. } => ntests += 1,
TestResult { status: Discard, .. } => continue,
r @ TestResult { status: Fail, .. } => return Err(r),
}
}
Ok(ntests)
}
pub fn quickcheck<A>(&mut self, f: A) where A: Testable {
let _ = ::env_logger::init();
match self.quicktest(f) {
Ok(ntests) => info!("(Passed {} QuickCheck tests.)", ntests),
Err(result) => panic!(result.failed_msg()),
}
}
}
pub fn quickcheck<A: Testable>(f: A) { QuickCheck::new().quickcheck(f) }
#[derive(Clone, Debug)]
pub struct TestResult {
status: Status,
arguments: Vec<String>,
err: Option<String>,
}
#[derive(Clone, Debug)]
enum Status { Pass, Fail, Discard }
impl TestResult {
pub fn passed() -> TestResult { TestResult::from_bool(true) }
pub fn failed() -> TestResult { TestResult::from_bool(false) }
pub fn error<S: Into<String>>(msg: S) -> TestResult {
let mut r = TestResult::from_bool(false);
r.err = Some(msg.into());
r
}
pub fn discard() -> TestResult {
TestResult {
status: Discard,
arguments: vec![],
err: None,
}
}
pub fn from_bool(b: bool) -> TestResult {
TestResult {
status: if b { Pass } else { Fail },
arguments: vec![],
err: None,
}
}
pub fn must_fail<T, F>(f: F) -> TestResult
where F: FnOnce() -> T, F: Send + 'static, T: Send + 'static {
let f = panic::AssertUnwindSafe(f);
TestResult::from_bool(panic::catch_unwind(f).is_err())
}
pub fn is_failure(&self) -> bool {
match self.status {
Fail => true,
Pass|Discard => false,
}
}
pub fn is_error(&self) -> bool {
self.is_failure() && self.err.is_some()
}
fn failed_msg(&self) -> String {
match self.err {
None => {
format!("[quickcheck] TEST FAILED. Arguments: ({})",
self.arguments.connect(", "))
}
Some(ref err) => {
format!("[quickcheck] TEST FAILED (runtime error). \
Arguments: ({})\nError: {}",
self.arguments.connect(", "), err)
}
}
}
}
pub trait Testable : Send + 'static {
fn result<G: Gen>(&self, &mut G) -> TestResult;
}
impl Testable for bool {
fn result<G: Gen>(&self, _: &mut G) -> TestResult {
TestResult::from_bool(*self)
}
}
impl Testable for () {
fn result<G: Gen>(&self, _: &mut G) -> TestResult {
TestResult::passed()
}
}
impl Testable for TestResult {
fn result<G: Gen>(&self, _: &mut G) -> TestResult { self.clone() }
}
impl<A, E> Testable for Result<A, E>
where A: Testable, E: Debug + Send + 'static {
fn result<G: Gen>(&self, g: &mut G) -> TestResult {
match *self {
Ok(ref r) => r.result(g),
Err(ref err) => TestResult::error(format!("{:?}", err)),
}
}
}
macro_rules! testable_fn {
($($name: ident),*) => {
impl<T: Testable,
$($name: Arbitrary + Debug),*> Testable for fn($($name),*) -> T {
#[allow(non_snake_case)]
fn result<G_: Gen>(&self, g: &mut G_) -> TestResult {
fn shrink_failure<T: Testable, G_: Gen, $($name: Arbitrary + Debug),*>(
g: &mut G_,
self_: fn($($name),*) -> T,
a: ($($name,)*),
) -> Option<TestResult> {
for t in a.shrink() {
let ($($name,)*) = t.clone();
let mut r_new = safe(move || {self_($($name),*)}).result(g);
if r_new.is_failure() {
let ($($name,)*) : ($($name,)*) = t.clone();
r_new.arguments = vec![$(format!("{:?}", $name),)*];
let shrunk = shrink_failure(g, self_, t);
return Some(shrunk.unwrap_or(r_new))
}
}
None
}
let self_ = *self;
let a: ($($name,)*) = Arbitrary::arbitrary(g);
let ( $($name,)* ) = a.clone();
let mut r = safe(move || {self_($($name),*)}).result(g);
let ( $($name,)* ) = a.clone();
r.arguments = vec![$(format!("{:?}", $name),)*];
match r.status {
Pass|Discard => r,
Fail => {
shrink_failure(g, self_, a).unwrap_or(r)
}
}
}
}}}
testable_fn!();
testable_fn!(A);
testable_fn!(A, B);
testable_fn!(A, B, C);
testable_fn!(A, B, C, D);
testable_fn!(A, B, C, D, E);
testable_fn!(A, B, C, D, E, F);
testable_fn!(A, B, C, D, E, F, G);
testable_fn!(A, B, C, D, E, F, G, H);
testable_fn!(A, B, C, D, E, F, G, H, I);
testable_fn!(A, B, C, D, E, F, G, H, I, J);
testable_fn!(A, B, C, D, E, F, G, H, I, J, K);
testable_fn!(A, B, C, D, E, F, G, H, I, J, K, L);
fn safe<T, F>(fun: F) -> Result<T, String>
where F: FnOnce() -> T, F: Send + 'static, T: Send + 'static {
panic::catch_unwind(panic::AssertUnwindSafe(fun)).map_err(|any_err| {
if let Some(&s) = any_err.downcast_ref::<&str>() {
s.to_owned()
} else if let Some(s) = any_err.downcast_ref::<String>() {
s.to_owned()
} else {
"UNABLE TO SHOW RESULT OF PANIC.".to_owned()
}
})
}
trait AShow : Arbitrary + Debug {}
impl<A: Arbitrary + Debug> AShow for A {}
#[cfg(test)]
mod test {
use QuickCheck;
#[test]
fn shrinking_regression_issue_126() {
fn thetest(vals: Vec<bool>) -> bool {
vals.iter().filter(|&v| *v).count() < 2
}
let failing_case =
QuickCheck::new()
.quicktest(thetest as fn(vals: Vec<bool>) -> bool)
.unwrap_err();
let expected_argument = format!("{:?}", [true, true]);
assert_eq!(failing_case.arguments, vec![expected_argument]);
}
}