上次为大家介绍了mockall的部分核心功能,这次将继续介绍这个库提供的其他一些在单元测试时常用的功能。
mockall (第二部分)
一个强大的Rust对象模拟库
Mockall 可以模拟几乎所有的结构体和特征。模拟出的对象可在单元测试中作为替代实际的依赖对象使用。
规定调用次数
默认情况下,每个“期望”允许被调用无限次。但是Mockall允许开发者自己定义某个“期望”会被调用的次数(固定次数或某个范围)来测试代码的行为是否正确。
#[automock]
trait Foo {
fn foo(&self, x: u32);
}
let mut mock = MockFoo::new();
mock.expect_foo()
.times(1)
.return_const(());
mock.foo(0); // Ok
mock.foo(1); // Panics!
运行次序序列
默认情况下,“期待”的调用与运行不会要求按规定次序执行。但是在Mockall中开发者可以通过Sequence
规定次序。任何“期待”都可以被添加进同一次序序列中,并且没有对象限制。
#[automock]
trait Foo {
fn foo(&self);
}
let mut seq = Sequence::new();
let mut mock1 = MockFoo::new();
mock1.expect_foo()
.times(1)
.in_sequence(&mut seq)
.returning(|| ());
let mut mock2 = MockFoo::new();
mock2.expect_foo()
.times(1)
.in_sequence(&mut seq)
.returning(|| ());
mock2.foo(); // Panics! mock1.foo 应该先被调用
检查点
某些情况下,在测试运行中,有必要验证全部“期待”是否被满足,丢弃已有的、或添加新的“期待,检查点可用来达成此目的。每一个模拟对象都会有一个checkpoint
方法。当其被调用,Mockall会立即验证此方法的所有“期待”。任何没有被满足的“期待”都会被当做panic
处理。之后,这些“期待”会被清除以便加入新的“期待”以继续进行测试。
#[automock]
trait Foo {
fn foo(&self);
}
let mut mock = MockFoo::new();
mock.expect_foo()
.times(2)
.returning(|| ());
mock.foo();
mock.checkpoint(); // Panics! foo 还未被调用2次
#[automock]
trait Foo {
fn foo(&self);
}
let mut mock = MockFoo::new();
mock.expect_foo()
.times(1)
.returning(|| ());
mock.foo();
mock.checkpoint();
mock.foo(); // Panics! 此期待被清除
通过引用传递的参数
Mockall也可以模拟使用通过引用传递参数的方法。但是需要注意的是:匹配器Predicate
将通过值处理参数,不通过引用
#[automock]
trait Foo {
fn foo(&self, x: &u32) -> u32;
}
let mut mock = MockFoo::new();
let e = mock.expect_foo()
// Note that x is a &u32, not a &&u32
.withf(|x: &u32| *x == 5)
.returning(|x: &u32| *x + 1);
assert_eq!(6, mock.foo(&5));
引用返回值
Mockall可以使用引用返回值,但是有一个限制:返回引用的证明周期必须与模拟对象的生命周期一致,或者使用'static
。
Mockall会为返回引用的方法创建不同的“期待”类型。它们的API除了设置返回值的方式不同外与普通的“期待”一样。
返回'static
引用的方法与任何其他返回'static
值的方法并无差异。
struct Thing(u32);
#[automock]
trait Container {
fn get(&self, i: u32) -> &'static Thing;
}
const THING: Thing = Thing(42);
let mut mock = MockContainer::new();
mock.expect_get()
.return_const(&THING);
assert_eq!(42, mock.get(0).0);
参数中有&self
的方法示例:
struct Thing(u32);
#[automock]
trait Container {
fn get(&self, i: u32) -> &Thing;
}
let thing = Thing(42);
let mut mock = MockContainer::new();
mock.expect_get()
.return_const(thing);
assert_eq!(42, mock.get(0).0);
参数中用&mut self
的方法示例:
struct Thing(u32);
#[automock]
trait Container {
fn get_mut(&mut self, i: u32) -> &mut Thing;
}
let thing = Thing(42);
let mut mock = MockContainer::new();
mock.expect_get_mut()
.return_var(thing);
mock.get_mut(0).0 = 43;
assert_eq!(43, mock.get_mut(0).0);
作为Deref
常见目标的超大类型比较特殊。Mockall会自动使用类型所属的形式。目前支持CStr
, OsStr
, Path
, str
。使用这一特性是完全自动的。
#[automock]
trait Foo {
fn name(&self) -> &str;
}
let mut mock = MockFoo::new();
mock.expect_name().return_const("abcd".to_owned());
assert_eq!("abcd", mock.name());
评论区
写评论还没有评论