29 releases

0.8.0 Dec 16, 2022
0.7.11 Apr 2, 2021
0.7.10 Jan 20, 2021
0.7.9 Dec 22, 2020
0.1.1 Sep 5, 2017

#127 in Testing

Download history 1558/week @ 2024-08-22 1339/week @ 2024-08-29 1471/week @ 2024-09-05 1422/week @ 2024-09-12 1286/week @ 2024-09-19 1341/week @ 2024-09-26 967/week @ 2024-10-03 1303/week @ 2024-10-10 1007/week @ 2024-10-17 1104/week @ 2024-10-24 1482/week @ 2024-10-31 833/week @ 2024-11-07 705/week @ 2024-11-14 425/week @ 2024-11-21 723/week @ 2024-11-28 817/week @ 2024-12-05

2,844 downloads per month
Used in 10 crates

MIT license

29KB
302 lines

logo

Mocking framework for Rust (currently only nightly). See documentation for more.

#[mockable]
mod hello_world {
    pub fn world() -> &'static str {
        "world"
    }

    pub fn hello_world() -> String {
        format!("Hello {}!", world())
    }
}

#[test]
fn mock_test() {
    hello_world::world.mock_safe(|| MockResult::Return("mocking"));

    assert_eq!("Hello mocking!", hello_world::hello_world());
}

lib.rs:

Mocking framework for Rust (currently only nightly)

#[mockable]
mod hello_world {
    pub fn world() -> &'static str {
        "world"
    }

    pub fn hello_world() -> String {
        format!("Hello {}!", world())
    }
}

#[test]
fn mock_test() {
    hello_world::world.mock_safe(|| MockResult::Return("mocking"));

    assert_eq!("Hello mocking!", hello_world::hello_world());
}

Introduction

This is a user guide showing Rust project set up for testing with mocks.

It is highly recommended to use mocks ONLY for test runs and NEVER in release builds! Mocktopus is not designed for high performance and will slow down code execution.

Note: this guide shows set up of mocking for test builds only.

Prerequisites

Add Mocktopus dev-dependency to project's Cargo.toml:

[dev-dependencies]
mocktopus = "0.7.0"

Enable procedural macros in crate root:

#![cfg_attr(test, feature(proc_macro_hygiene))]

Import Mocktopus (skip for Rust 2018):

#[cfg(test)]
extern crate mocktopus;

Making functions mockable

To make functions mockable they must be annotated with provided procedural macros. See documentation for all their possibilities and rules.

To use these macros import them into namespace:

#[cfg(test)]
use mocktopus::macros::*;

Annotate mockable code like standalone functions or impl blocks:

#[mockable]
fn my_fn() {}

#[mockable]
impl Struct {
    fn my_method() {}
}

It's NOT legal to annotate single funciton in impl block:

impl Struct {
    #[mockable] // WRONG, will break Mocktopus
    fn my_method() {}
}

It is possible to annotate modules, which makes all their potentially mockable content mockable:

#[cfg_attr(test, mockable)]
mod my_module {
    fn my_fn() {}
}

This does NOT work for modules in separate file:

#[cfg_attr(test, mockable)] // WRONG, has no effect
mod my_module;

Mocking

Import tools for mocking in test module:

#[cfg(test)]
mod tests {
    use mocktopus::mocking::*;

Among others this imports trait Mockable. It is implemented for all functions and provides an interface for setting up mocks:

#[test]
fn my_test() {
    my_function.mock_safe(|| MockResult::Return(1));

    assert_eq!(1, my_function());
}

It is also possible to mock struct methods, either from own impls, traits or trait defaults:

// Mocking method
MyStruct::my_method.mock_safe(|| MockResult::Return(1));
// Mocking trait method
MyStruct::my_trait_method.mock_safe(|| MockResult::Return(2));
// Mocking default trait method
MyStruct::my_trait_default_method.mock_safe(|| MockResult::Return(3));

Mocking with mock_safe is simplest, but the Mockable trait has more, see documantation.

Mocking range

Every mock works only in thread, in which it was set. All Rust test runs are executed in independent threads, so mocks do not leak between them:

#[cfg_attr(test, mockable)]
fn common_fn() -> u32 {
    0
}

#[test]
fn common_fn_test_1() {
    assert_eq!(0, common_fn());

    common_fn.mock_safe(|| MockResult::Return(1));

    assert_eq!(1, common_fn());
}

#[test]
fn common_fn_test_2() {
    assert_eq!(0, common_fn());

    common_fn.mock_safe(|| MockResult::Return(2));

    assert_eq!(2, common_fn());
}

Mock closure

mock_safe has single argument: a closure, which takes same input as mocked function and returns a MockResult. Whenever the mocked function is called, its inputs are passed to the closure:

#[cfg_attr(test, mockable)]
fn my_function_1(_: u32) {
    return
}

#[test]
fn my_function_1_test() {
    my_function_1.mock_safe(|x| {
        assert_eq!(2, x);
        MockResult::Return(())
    });

    my_function_1(2); // Passes
    my_function_1(3); // Panics
}

If the closure returns MockResult::Return, the mocked function does not run. It immediately returns with a value, which is passed inside MockResult::Return:

#[cfg_attr(test, mockable)]
fn my_function_2() -> u32 {
    unreachable!()
}

#[test]
fn my_function_2_test() {
    my_function_2.mock_safe(|| MockResult::Return(3));

    assert_eq!(3, my_function_2());
}

If the closure returns MockResult::Continue, the mocked function runs normally, but with changed arguments. The new arguments are returned from closure in tuple inside MockResult::Continue:

#[cfg_attr(test, mockable)]
fn my_function_3(x: u32, y: u32) -> u32 {
    x + y
}

#[test]
fn my_function_3_test() {
    my_function_3.mock_safe(|x, y| MockResult::Continue((x, y + 1)));

    assert_eq!(3, my_function_3(1, 1));
}

Mocking generics

When mocking generic functions, all its generics must be defined and only this variant will be affected:

#[cfg_attr(test, mockable)]
fn generic_fn<T: Display>(t: T) -> String {
    t.to_string()
}

#[test]
fn generic_fn_test() {
    generic_fn::<u32>.mock_safe(|_| MockResult::Return("mocked".to_string()));

    assert_eq!("1", generic_fn(1i32));
    assert_eq!("mocked", generic_fn(1u32));
}

The only exception are lifetimes, they are ignored:

#[cfg_attr(test, mockable)]
fn lifetime_generic_fn<'a>(string: &'a String) -> &'a str {
    string.as_ref()
}

#[test]
fn lifetime_generic_fn_test() {
    lifetime_generic_fn.mock_safe(|_| MockResult::Return("mocked"));

    assert_eq!("mocked", lifetime_generic_fn(&"not mocked".to_string()));
}

Same rules apply to methods and structures:

struct GenericStruct<'a, T: Display + 'a>(&'a T);

#[cfg_attr(test, mockable)]
impl<'a, T: Display + 'a> GenericStruct<'a, T> {
    fn to_string(&self) -> String {
        self.0.to_string()
    }
}

static VALUE: u32 = 1;

#[test]
fn lifetime_generic_fn_test() {
    GenericStruct::<u32>::to_string.mock_safe(|_| MockResult::Return("mocked".to_string()));

    assert_eq!("mocked", GenericStruct(&VALUE).to_string());
    assert_eq!("mocked", GenericStruct(&2u32).to_string());
    assert_eq!("2", GenericStruct(&2i32).to_string());
}

Mocking async

Mocking async functions is almost exactly the same as non-async:

#[cfg_attr(test, mockable)]
async fn sleep(ms: u64) {
    tokio::time::delay_for(std::time::Duration::from_millis(ms)).await;
}

#[tokio::test]
async fn sleep_test() {
    sleep.mock_safe(|_| MockResult::Return(Box::pin(async move { () })));

    sleep(10000).await;
}

Mocking tricks

Returning reference to value created inside mock

#[mockable]
fn my_fn(my_string: &String) -> &String {
    my_string
}

#[test]
fn my_fn_test() {
    my_fn.mock_safe(|_| MockResult::Return(Box::leak(Box::new("mocked".to_string()))));

    assert_eq!("mocked", my_fn(&"not mocked 1"));
    assert_eq!("mocked", my_fn(&"not mocked 2"));
}

The trick is to store referenced value in a Box::new and then prevent its deallocation with Box::leak. This makes structure live forever and returns a &'static mut reference to it. The value is not freed until process termination, so it's viable solution only for use in tests and only if structure doesn't block a lot of resources like huge amounts of memory, open file handlers, sockets, etc.

Returning value created outside of mock

#[mockable]
fn my_fn() -> String {
    "not mocked".to_string()
}

#[test]
fn my_fn_test() {
    mock = Some("mocked".to_string());
    my_fn.mock_safe(move || MockResult::Return(mock.unwrap()));

    assert_eq!("mocked", my_fn());
    // assert_eq!("mocked", my_fn()); // WILL PANIC!
}

This makes function return predefined value on first call and panic on second one. It could return MockResult::Continue instead of panicking to mock only first call.

Returned values can be stored in a vector if mock should return different value on different calls:

#[test]
fn my_fn_test() {
    mut mock = vec!["mocked 1".to_string(), "mocked 2".to_string()];
    my_fn.mock_safe(move || MockResult::Return(mock.remove(0)));

    assert_eq!("mocked 1", my_fn());
    assert_eq!("mocked 2", my_fn());
    // assert_eq!("mocked 3", my_fn()); // WILL PANIC!
}

The vector can store MockResults for more complex mocking.

Dependencies

~1.5MB
~37K SLoC