#[macro_use]
extern crate generator;
use generator::*;
#[test]
fn test_return() {
let mut g = Gn::new_scoped(|_s| {
return 42;
});
assert_eq!(g.next(), Some(42));
assert!(g.is_done());
}
#[test]
fn generator_is_done() {
let mut g = Gn::<()>::new(|| {
yield_with(());
});
g.next();
assert!(!g.is_done());
g.next();
assert!(g.is_done());
}
#[test]
fn generator_is_done1() {
let mut g = Gn::new_scoped(|mut s| {
s.yield_(2);
done!();
});
assert_eq!(g.next(), Some(2));
assert!(!g.is_done());
assert_eq!(g.next(), None);
assert!(g.is_done());
}
#[test]
fn test_yield_a() {
let mut g = Gn::<i32>::new(|| {
let r: i32 = yield_(10).unwrap();
r * 2
});
let i = g.raw_send(None).unwrap();
assert_eq!(i, 10);
let i = g.send(3);
assert_eq!(i, 6);
assert!(g.is_done());
}
#[test]
fn test_yield_with() {
let mut g = Gn::new(|| {
yield_with(10);
20
});
let i = g.send(());
assert!(i == 10);
let j = g.next();
assert!(j.unwrap() == 20);
}
#[test]
#[should_panic]
fn test_yield_with_type_error() {
let mut g = Gn::<()>::new(|| {
yield_with(10u32);
20i32
});
g.next();
}
#[test]
#[should_panic]
fn test_get_yield_type_error() {
let mut g = Gn::<u32>::new(|| {
get_yield::<i32>();
});
g.send(10);
}
#[test]
#[should_panic]
fn test_deep_yield_with_type_error() {
let mut g = Gn::<()>::new(|| {
let mut g = Gn::<()>::new(|| {
yield_with(0);
});
g.next();
});
g.next();
}
#[test]
fn test_scoped() {
use std::rc::Rc;
use std::cell::RefCell;
let x = Rc::new(RefCell::new(10));
let x1 = x.clone();
let mut g = Gn::<()>::new(move || {
*x1.borrow_mut() = 20;
yield_with(());
*x1.borrow_mut() = 5;
});
g.next();
assert!(*x.borrow() == 20);
g.next();
assert!(*x.borrow() == 5);
assert!(g.is_done());
}
#[test]
fn test_scoped_1() {
let mut x = 10;
{
let mut g = Gn::<()>::new(|| {
x = 5;
});
g.next();
}
assert!(x == 5);
}
#[test]
fn test_inner_ref() {
let mut g = Gn::<()>::new_scoped(|mut s| {
use std::mem;
use std::ptr;
let mut x: u32 = 10;
s.yield_(unsafe { mem::transmute(&mut x) });
assert!(x == 5);
unsafe { &mut *ptr::null_mut() }
});
let a = g.next().unwrap();
assert!(*a == 10);
*a = 5;
}
#[test]
fn test_drop() {
let mut x = 10;
{
let mut g = Gn::<()>::new(|| {
x = 1;
yield_with(());
x = 5;
});
g.send(());
}
assert!(x == 1);
}
#[test]
fn test_ill_drop() {
let mut x = 10u32;
{
Gn::<u32>::new(|| {
x = 5;
x = get_yield().unwrap_or(0);
});
}
assert!(x == 10);
}
#[test]
fn test_loop_drop() {
let mut x = 10u32;
{
let mut g: Generator<_, ()> = Gn::<()>::new(|| {
x = 5;
loop {
yield_with(());
}
});
g.send(());
}
assert!(x == 5);
}
#[test]
fn test_panic_inside() {
use std::panic::{catch_unwind, AssertUnwindSafe};
let mut x = 10;
{
let mut wrapper = AssertUnwindSafe(&mut x);
if let Err(panic) = catch_unwind(move || {
let mut g = Gn::<()>::new(|| {
**wrapper = 5;
panic!("panic inside!");
});
g.resume();
}) {
match panic.downcast_ref::<&str>() {
Some(msg) => println!("get panic: {:?}", msg),
None => println!("can't get panic message"),
}
}
}
assert!(x == 5);
}
#[test]
#[allow(unreachable_code)]
fn test_cancel() {
let mut g = Gn::<()>::new(|| {
let mut i = 0;
loop {
yield_with(i);
i += 1;
}
i
});
loop {
let i: i32 = g.next().unwrap();
if i > 10 {
unsafe {
g.cancel();
}
break;
}
}
assert!(g.is_done());
}
#[test]
#[should_panic]
fn test_yield_from_functor_context() {
yield_::<(), _>(0);
}
#[test]
#[should_panic]
fn test_yield_with_from_functor_context() {
yield_with(0);
}
#[test]
fn test_yield_from_generator_context() {
let mut g = Gn::<()>::new(|| {
let mut g1 = Gn::<()>::new(|| {
yield_with(5);
10
});
let i = g1.send(());
yield_with(i);
0
});
let n = g.send(());
assert!(n == 5);
let n = g.send(());
assert!(n == 0);
}
#[test]
fn test_yield_from() {
let mut g = Gn::<()>::new(|| {
let g1 = Gn::<()>::new(|| {
yield_with(5);
10
});
yield_from(g1);
0
});
let n = g.send(());
assert!(n == 5);
let n = g.send(());
assert!(n == 10);
let n = g.send(());
assert!(n == 0);
assert!(g.is_done());
}
#[test]
fn test_yield_from_send() {
let mut g = Gn::<u32>::new(|| {
let g1 = Gn::<u32>::new(|| {
let mut i: u32 = yield_(1u32).unwrap();
i = yield_(i * 2).unwrap();
i * 2
});
let i = yield_from(g1).unwrap();
assert_eq!(i, 10);
0u32
});
let n = g.raw_send(None).unwrap();
assert!(n == 1);
let n = g.send(3);
assert!(n == 6);
let n = g.send(4);
assert!(n == 8);
let n = g.send(10);
assert!(n == 0);
assert!(g.is_done());
}
#[test]
#[should_panic]
fn test_yield_from_send_type_miss_match() {
let mut g = Gn::<u32>::new(|| {
let g1 = Gn::<u32>::new(|| {
let mut i: u32 = yield_(1u32).unwrap();
i = yield_(i * 2).unwrap();
i * 2
});
yield_from(g1);
0
});
let n = g.send(3);
assert!(n == 1);
let n = g.send(4);
assert!(n == 6);
let n = g.send(10);
assert!(n == 8);
let n = g.send(0);
assert!(n == 0);
assert!(g.is_done());
}
#[test]
fn test_scope_gen() {
let mut g = Gn::new_scoped(|mut s| {
let i = s.yield_(0).unwrap();
i * 2
});
assert_eq!(g.raw_send(None), Some(0));
assert_eq!(g.raw_send(Some(3)), Some(6));
assert_eq!(g.raw_send(None), None);
}
#[test]
fn test_scope_yield_from_send() {
let mut g = Gn::new_scoped(|mut s| {
let g1 = Gn::new_scoped(|mut s| {
let mut i: u32 = s.yield_(1u32).unwrap();
i = s.yield_(i * 2).unwrap();
i * 2
});
let i = s.yield_from(g1).unwrap();
i * 2
});
let n = g.send(3);
assert_eq!(n, 1);
let n = g.send(4);
assert_eq!(n, 8);
let n = g.send(10);
assert_eq!(n, 20);
let n = g.send(7);
assert!(n == 14);
assert!(g.is_done());
}
#[test]
fn test_re_init() {
let clo = || {
|mut s: Scope<(), _>| {
s.yield_(0);
s.yield_(3);
5
}
};
let mut g = GeneratorImpl::new(0x800);
let s = g.get_scope();
g.init(|| clo()(s));
assert_eq!(g.next(), Some(0));
assert_eq!(g.next(), Some(3));
assert_eq!(g.next(), Some(5));
assert_eq!(g.is_done(), true);
let s = g.get_scope();
g.init(|| clo()(s));
assert_eq!(g.next(), Some(0));
assert_eq!(g.next(), Some(3));
assert_eq!(g.next(), Some(5));
assert_eq!(g.is_done(), true);
}