2 releases

0.0.2 Jan 18, 2024
0.0.1 Jan 15, 2024

#82 in Testing

MIT/Apache

485KB
9K SLoC

clearcheck

clearcheck

Table of content

Introducing clearcheck

Write expressive and elegant assertions with ease!

clearcheck is designed to make assertion statements as clear and concise as possible. It allows chaining multiple assertions together for a fluent and intuitive syntax, leading to more self-documenting test cases.

let pass_phrase = "P@@sw0rd1 zebra alpha";
pass_phrase.should_not_be_empty()
    .should_have_at_least_length(10)
    .should_contain_all_characters(vec!['@', ' '])
    .should_contain_a_digit()
    .should_not_contain_ignoring_case("pass")
    .should_not_contain_ignoring_case("word");

Key features:

🔹 Fluent API: Chain assertions for a natural and readable experience.

🔹 Extensive assertions: Variety of assertions covering common validation needs.

🔹 Customizable: Extend with your own assertions for specific domain requirements.

🔹 Type-safe: Built with Rust's type system for reliable and expressive assertions.

🔹 Custom assertions: Craft assertions tailored to your exact needs, ensuring comprehensive validations for various data structures.

Usage

Add this to your Cargo.toml (no features):

[dev-dependencies]
clearcheck = { version = "0.0.2" }

Add this to your Cargo.toml (all features):

[dev-dependencies]
clearcheck = { version = "0.0.2", features = ["num", "date", "regex", "file"] }
chrono = { version = "0.4.31" }
num = { version = "0.4.1" }
regex = { version = "1.10.2" }
walkdir = { version = "2.4.0", features = [] }

Assertions vs Matchers

Assertions serve as the cornerstone of the test cases, defining the exact expectations the code must fulfill. They act as a contract, ensuring that each data type (/data structure) adheres to its intended behavior.

Matchers, on the other hand, provide the granular tools for carrying out these assertions. They examine data and verify that the data conforms to specific criteria.

In essence, assertions orchestrate the high-level validation logic, while matchers act as the code-level inspectors, ensuring every detail aligns with the expectations.

Supported assertions

Bool

Assertions
Assertion Description
should_be_true Asserts that the boolean evaluates to true.
should_be_false Asserts that the boolean evaluates to false.
Usage
let value = true;
value.should_be_true();

Char

Assertions
Assertion Description
should_be_in_inclusive_range Asserts that the character falls within the given inclusive range.
should_not_be_in_inclusive_range Asserts that the character does not fall within the given inclusive range.
should_be_in_exclusive_range Asserts that the character falls within the given exclusive range.
should_not_be_in_exclusive_range Asserts that the character does not fall within the given exclusive range.
should_be_equal_ignoring_case Asserts that the character equals other character, with case ignored.
should_not_be_equal_ignoring_case Asserts that the character does not equal other character, with case ignored.
Usage
let letter = 'd';
letter.should_be_in_inclusive_range('a'..='d');

let letter = 'D';
letter.should_be_equal_ignoring_case('d');

Collections (Vector, Arrays, Slices)

Assertions
Assertion Description
should_have_upper_bound Asserts that all elements in the collection are less than or equal to the given element.
should_have_lower_bound Asserts that all elements in the collection are greater than or equal to the given element.
should_contain_duplicates Asserts that the collection contains atleast one duplicate element.
should_not_contain_duplicates Asserts that the collection does not contain any duplicate element.
should_be_equal_ignoring_case Asserts that the elements in the collection are equal to those in other, ignoring case differences. (Only applicable where elements can be represented as strings).
should_not_be_equal_ignoring_case Asserts that the elements in the collection are not equal to those in other, ignoring case differences. (Only applicable where elements can be represented as strings).
should_be_monotonically_increasing Asserts that the elements in the collection are in non-decreasing order (allowing consecutive equal elements).
should_be_monotonically_decreasing Asserts that the elements in the collection are in non-increasing order (allowing consecutive equal elements).
should_be_strictly_increasing Asserts that the elements in the collection are in strictly increasing order (no consecutive elements can be equal).
should_be_strictly_decreasing Asserts that the elements in the collection are in strictly decreasing order (no consecutive elements can be equal).
should_contain Asserts that the collection contains the given element.
should_not_contain Asserts that the collection does not contain the given element.
should_contain_all Asserts that the collection contains all the given elements.
should_not_contain_all Asserts that the collection does not contain all the given elements.
should_contain_any Asserts that the collection contains any of the given elements.
should_not_contain_any Asserts that the collection does not contain any of the given elements.
should_be_empty Asserts that the collection is empty.
should_not_be_empty Asserts that the collection is not empty.
Size based assertions
Assertion Description
should_have_size Asserts that the size of the underlying collection is exactly the given size.
should_not_have_size Asserts that the size of the underlying collection is not the given size.
should_have_at_least_size Asserts that the size of the underlying collection is greater than or equal to the given size.
should_have_at_most_size Asserts that the size of the underlying collection is less than or equal to the given size.
should_be_same_size_as Asserts that the size of the underlying collection is same as that of the given collection.
should_have_size_in_inclusive_range Asserts that the size of the underlying collection falls within the given inclusive range.
should_not_have_size_in_inclusive_range Asserts that the size of the underlying collection does not fall within the given inclusive range.
should_have_size_in_exclusive_range Asserts that the size of the underlying collection falls within the given exclusive range.
should_not_have_size_in_exclusive_range Asserts that the size of the underlying collection does not fall within the given exclusive range.
Usage
let keywords = ["testing", "automation", "clearcheck", "junit"];
keywords.should_not_be_empty()
    .should_have_size_in_inclusive_range(4..=10)
    .should_not_contain_duplicates()
    .should_contain_any(vec!["junit", "clearcheck", "testing"])
    .should_not_contain_any(vec!["scalatest", "gotest"]);

Date (enabled by 'date' feature, depends on chrono)

Assertions
Assertion Description
should_have_same_year_as Asserts that the date has the same year as the other date.
should_not_have_same_year_as Asserts that the date does not have the same year as the other date.
should_have_year Asserts that the date has the same year as the given year.
should_not_have_year Asserts that the date does not have the same year as the given year.
should_have_same_month_as Asserts that the date has the same month as the other date.
should_not_have_same_month_as Asserts that the date does not have the same month as the other date.
should_have_month Asserts that the date has the same month as the given month.
should_not_have_month Asserts that the date does not have the same month as the given month.
should_have_same_day_as Asserts that the date has the same day as the other date.
should_not_have_same_day_as Asserts that the date does not have the same day as the other date.
should_have_day Asserts that the date has the same day as the given day.
should_not_have_day Asserts that the date does not have the same day as the given day.
should_be_a_leap_year Asserts that the date falls in a leap year.
should_not_be_a_leap_year Asserts that the date does not fall in a leap year.
Usage
use chrono::NaiveDate;

let date = NaiveDate::from_ymd_opt(2024, 1, 10).unwrap();
date
     .should_be_a_leap_year()
     .should_have_month(1)
     .should_be_greater_than(&NaiveDate::from_ymd_opt(2023, 1, 10).unwrap());

Filepath (enabled by 'file' feature, depends on walkdir)

Assertions
Assertion Description
should_be_a_directory Asserts that the path is a directory.
should_be_a_file Asserts that the path is a file.
should_be_a_symbolic_link Asserts that the path is a symbolic link.
should_be_zero_sized Asserts that the path corresponds to a zero sized file.
should_not_be_zero_sized Asserts that the path corresponds to a non-zero sized file.
should_be_readonly Asserts that the path corresponds to a readonly file.
should_be_writable Asserts that the path corresponds to a writable file.
should_be_absolute Asserts that the path is absolute.
should_be_relative Asserts that the path is relative.
should_have_extension Asserts that the path corresponds to a file with the given extension.
should_not_have_extension Asserts that the path corresponds to a file that does not have the given extension.
should_contain_file_name Asserts that the path corresponds to a directory that contains the given file name.
should_not_contain_file_name Asserts that the path corresponds to a directory that does not contain the given file name.
should_contain_all_file_names Asserts that the path corresponds to a directory that contains all the given file names.
should_not_contain_all_file_names Asserts that the path corresponds to a directory that does not contain all the given file names.
should_contain_any_of_file_names Asserts that the path corresponds to a directory that contains any of the given file names.
should_not_contain_any_of_file_names Asserts that the path corresponds to a directory that does not contain any of the given file names.
Usage
use tempdir::TempDir;

let temporary_directory = TempDir::new(".").unwrap();
let file_path_junit = temporary_directory.path().join("junit.txt");
let file_path_clearcheck = temporary_directory.path().join("clearcheck.txt");

let _ = File::create(file_path_junit).unwrap();
let _ = File::create(file_path_clearcheck).unwrap();

let directory_path = temporary_directory.path();
directory_path
    .should_be_a_directory()
    .should_contain_any_of_file_names(vec!["junit.txt", "clearcheck.txt"]);

Float (enabled by 'num' feature, depends on num)

Assertions
Assertion Description
should_be_nan Asserts that the floating-point value is NaN (Not a Number).
should_not_be_nan Asserts that the floating-point value is not NaN (Not a Number).
should_be_zero Asserts that the floating-point value is zero.
should_not_be_zero Asserts that the floating-point value is not zero.
should_be_positive Asserts that the floating-point value is positive.
should_be_negative Asserts that the floating-point value is negative.
should_be_in_inclusive_range_with_tolerance Asserts that the floating-point value falls within the given inclusive range with tolerance.
should_not_be_in_inclusive_range_with_tolerance Asserts that the floating-point value does not fall within the given inclusive range with tolerance.
should_be_in_exclusive_range_with_tolerance Asserts that the floating-point value falls within the given exclusive range with tolerance.
should_not_be_in_exclusive_range_with_tolerance Asserts that the floating-point value does not fall within the given exclusive range with tolerance.
Usage
let value: f64 = 1.34589;
value
   .should_not_be_nan()
   .should_be_positive()
   .should_be_in_inclusive_range_with_tolerance(1.11..=1.3458, 0.23);

Integer (enabled by 'num' feature, depends on num)

Assertions
Assertion Description
should_be_positive Asserts that the integer value is positive.
should_be_negative Asserts that the integer value is negative.
should_be_even Asserts that the integer value is even.
should_be_odd Asserts that the integer value is odd.
should_be_zero Asserts that the integer value is zero.
should_not_be_zero Asserts that the integer value is not zero.
Usage
let value = 24;
value
    .should_be_positive()
    .should_be_even()
    .should_be_in_inclusive_range(10..=40);

HashMap

Assertions
Assertion Description
should_contain_key Asserts that the HashMap contains the given key.
should_not_contain_key Asserts that the HashMap does not contain the given key.
should_contain_all_keys Asserts that the HashMap contains all the given keys.
should_not_contain_all_keys Asserts that the HashMap does not contain all the given keys.
should_contain_any_of_keys Asserts that the HashMap contains any of the given keys.
should_not_contain_any_of_keys Asserts that the HashMap does not contain any of the given keys.
should_contain_value Asserts that the HashMap contains the given value.
should_not_contain_value Asserts that the HashMap does not contain the given value.
should_contain_all_values Asserts that the HashMap contains all the given values.
should_not_contain_all_values Asserts that the HashMap does not contain all the given values.
should_contain_any_of_values Asserts that the HashMap contains any of the given values.
should_not_contain_any_of_values Asserts that the HashMap does not contain any of the given values.
should_contain Asserts that the HashMap contains the given key and the value.
should_not_contain Asserts that the HashMap does not contain the given key and the value.
should_contain_all Asserts that the HashMap contains all the entries from the given HashMap.
should_not_contain_all Asserts that the HashMap does not contain all the entries from the given HashMap.
should_contain_any Asserts that the HashMap contains any of the entries from the given HashMap.
should_not_contain_any Asserts that the HashMap does not contain any of the entries from the given HashMap.
should_be_empty Asserts that the HashMap is empty.
should_not_be_empty Asserts that the HashMap is not empty.
+ Size based assertions.
Usage
#[derive(Eq, Debug, PartialEq, Hash)]
struct Book {
    id: usize,
    title: &'static str,
}

impl Book {
    fn new(id: usize, title: &'static str) -> Self {
        Book { id, title }
    }
}

let mut book_id_by_name = HashMap::new();
book_id_by_name.insert("Database internals", 1);
book_id_by_name.insert("Designing data intensive applications", 2);

book_id_by_name
    .should_not_be_empty()
    .should_contain_key("Database internals")
    .should_contain_value(&1)
    .should_have_at_least_size(2)
    .should_contain("Database internals", &1);

Option

Assertions
Assertion Description
should_be_some Asserts that the Option evaluates to Some.
should_be_none Asserts that the Option evaluates to None.
Usage
let option = Some("clearcheck");
option.should_be_some();

Result

Assertions
Assertion Description
should_be_ok Asserts that the Result evaluates to Ok.
should_be_err Result evaluates to Err.
Usage
let value: Result<i32, &str> = Ok(32);
value.should_be_ok();

T: PartialOrd

Assertions
Assertion Description
should_be_greater_than Asserts that the self value is greater than the given value (other) according to the PartialOrd implementation.
should_be_greater_than_equal_to Asserts that the self value is greater than or equal to the given value (other) according to the PartialOrd implementation.
should_be_less_than Asserts that the self value is less than the given value (other) according to the PartialOrd implementation.
should_be_less_than_equal_to Asserts that the self value is less than or equal to the given value (other) according to the PartialOrd implementation.
should_not_be_greater_than Asserts that the self value is not greater than the given value (other) according to the PartialOrd implementation.
should_not_be_greater_than_equal_to Asserts that the self value is not greater than or equal to the given value (other) according to the PartialOrd implementation.
should_not_be_less_than Asserts that the self value is not less than the given value (other) according to the PartialOrd implementation.
should_not_be_less_than_equal_to Asserts that the self value is not less than or equal to the given value (other) according to the PartialOrd implementation.
should_be_in_inclusive_range Asserts that the self value falls within the given inclusive range.
should_not_be_in_inclusive_range Asserts that the self value does not fall within the given inclusive range.
should_be_in_exclusive_range Asserts that the self value falls within the given exclusive range.
should_not_be_in_exclusive_range Asserts that the self value does not fall within the given exclusive range.
Usage
let value = 12.56;
value
    .should_be_greater_than(&10.90)
    .should_be_less_than(&15.98)
    .should_be_in_inclusive_range(10.90..=13.10);

T: Eq

Assertions
Assertion Description
should_equal Asserts that the value held by self is equal to other.
should_not_equal Asserts that the value held by self is not equal to other.
Usage
#[derive(Debug, Eq, PartialEq)]
struct Book { name: &'static str }

let books = vec![
  Book {name: "Database internals"},
  Book {name: "Rust in action"}
];

let other = vec![
  Book {name: "Database internals"},
  Book {name: "Rust in action"}
];

books.should_equal(&other);

String

Assertions
Assertion Description
should_begin_with Asserts that the string begins with the given prefix.
should_not_begin_with Asserts that the string does not begin with the given prefix
should_end_with Asserts that the string ends with the given suffix.
should_not_end_with Asserts that the string does not end with the given suffix.
should_be_lower_case Asserts that the string is lowercase.
should_be_upper_case Asserts that the string is uppercase.
should_be_equal_ignoring_case Asserts that the string equals other string, with case ignored.
should_not_be_equal_ignoring_case Asserts that the string does not equal other string, with case ignored.
should_only_contain_digits Asserts that the string contains only digits.
should_contain_a_digit Asserts that the string contains a digit.
should_not_contain_digits Asserts that the string does not contain any digits.
should_contain_character Asserts that the string contains the given character.
should_not_contain_character Asserts that the string does not contain the given character.
should_contain_all_characters Asserts that the string contains all the given characters.
should_not_contain_all_characters Asserts that the string does not contain all the given characters.
should_contain_any_characters Asserts that the string contains any of the given characters.
should_not_contain_any_characters Asserts that the string does not contain any of the given characters.
should_contain Asserts that the string contains the given substring.
should_not_contain Asserts that the string does not contain the given substring.
should_contain_ignoring_case Asserts that the string contains the substring, ignoring case differences.
should_not_contain_ignoring_case Asserts that the string does not contain the substring, ignoring case differences.
should_be_empty Asserts that the string is empty (has zero characters).
should_not_be_empty Asserts that the string is not empty.
should_be_numeric Asserts that the string is numeric.
should_not_be_numeric Asserts that the string is not numeric.
should_match Asserts that the string matches the given regular expression. (enabled by 'regex' feature, depends on regex)
should_not_match Asserts that the string does not match the given regular expression. (enabled by 'regex' feature, depends on regex)
Length based assertions
Assertion Description
should_have_length Asserts that the length of the string is exactly the given length.
should_not_have_length Asserts that the length of the string is not the given length.
should_have_at_least_length Asserts that the length of the string is greater than or equal to the given length.
should_have_at_most_length Asserts that the length of the string is less than or equal to the given length.
should_have_length_in_inclusive_range Asserts that the length of the string falls within the given inclusive range.
should_not_have_length_in_inclusive_range Asserts that the length of the string does not fall within the given inclusive range.
should_have_length_in_exclusive_range Asserts that the length of the string falls within the given exclusive range.
should_not_have_length_in_exclusive_range Asserts that the length of the string does not fall within the given exclusive range.
Usage
let pass_phrase = "P@@sw0rd1 zebra alpha";
pass_phrase.should_not_be_empty()
    .should_have_at_least_length(10)
    .should_contain_all_characters(vec!['@', ' '])
    .should_contain_a_digit()
    .should_not_contain_ignoring_case("pass")
    .should_not_contain_ignoring_case("word");

Changelog

Version 0.0.2

Version 0.0.2 refactored the String assertions to remove the duplication and introduced the following assertions:

Collection (Predicate based assertions)
Assertion Description
should_satisfy_for_all Asserts that all the elements in the collection satisfy the given predicate.
should_not_satisfy_for_all Asserts that not all the elements in the collection satisfy the given predicate.
should_satisfy_for_any Asserts that any of the elements in the collection satisfy the given predicate.
should_not_satisfy_for_any Asserts that none of the elements in the collection satisfy the given predicate.
Collection (Min-max based assertions)
Assertion Description
should_have_min Asserts that the minimum value in the underlying collection equals the given minimum value.
should_not_have_min Asserts that the minimum value in the underlying collection does not equal the given minimum value.
should_have_max Asserts that the maximum value in the underlying collection equals the given maximum value.
should_not_have_max Asserts that the maximum value in the underlying collection does not equal the given maximum value.
should_have_min_in_inclusive_range Asserts that the minimum value in the underlying collection falls within the given inclusive range.
should_not_have_min_in_inclusive_range Asserts that the minimum value in the underlying collection does not fall within the given inclusive range.
should_have_min_in_exclusive_range Asserts that the minimum value in the underlying collection falls within the given exclusive range.
should_not_have_min_in_exclusive_range Asserts that the minimum value in the underlying collection does not fall within the given exclusive range.
should_have_max_in_inclusive_range Asserts that the maximum value in the underlying collection falls within the given inclusive range.
should_not_have_max_in_inclusive_range Asserts that the maximum value in the underlying collection does not fall within the given inclusive range.
should_have_max_in_exclusive_range Asserts that the maximum value in the underlying collection falls within the given exclusive range.
should_not_have_max_in_exclusive_range Asserts that the maximum value in the underlying collection does not fall within the given exclusive range.
Option (Predicate based assertions)
Assertion Description
should_be_some_and_satisfy Asserts that the Option value is Some and satisfies the given predicate.
should_be_some_and_not_satisfy Asserts that the Option value is Some and does not satisfy the given predicate.
Result (Predicate based assertions)
Assertion Description
should_be_ok_and_satisfy Asserts that the Result value is Ok and satisfies the given predicate.
should_be_ok_and_not_satisfy Asserts that the Result value is Ok and does not satisfy the given predicate.

Unleashing the power of custom matchers and assertions

While this crate comes loaded with a plethora of ready-made assertions, sometimes your testing needs demand a bespoke touch. clearcheck allows crafting your own custom matchers and assertions!

The possibilities are endless:

  • Domain-specific validation: Craft assertions that understand the nuances of your business logic.
  • Enhanced readability: Write clear and concise matchers that mirror your domain vocabulary, making your tests self-documenting and understandable.
  • Reduced redundancy: Eliminate repetitive code by encapsulating complex validation logic within reusable matchers.

Let's craft a custom password matcher with specific criteria like length, digits, and banned patterns.

fn be_a_valid_password<T: AsRef<str> + Debug>() -> Matchers<T> {
    MatchersBuilder::start_building_with_inverted(be_empty().boxed())
      .push(have_atleast_same_length(10).boxed())
      .push(contain_a_digit().boxed())
      .push(contain_any_of_characters(vec!['@', '#']).boxed())
      .push_inverted(begin_with("pass").boxed())
      .push_inverted(contain_ignoring_case("pass").boxed())
      .push_inverted(contain_ignoring_case("word").boxed())
      .combine_as_and()
}

The matcher enforces the following:

  • Input must not be empty.
  • Input must have a minimum length of 10 characters.
  • Input must contain at least one digit.
  • Input must contain any of the following characters: '@', '#'.
  • Input must not begin with the string "pass" (case-insensitive).
  • Input must not contain the strings "pass" or "word" (case-insensitive).

Let's combine it into a powerful assertion for valid passwords.

trait PasswordAssertion {
    fn should_be_a_valid_password(&self) -> &Self;
}

impl PasswordAssertion for &str {
    fn should_be_a_valid_password(&self) -> &Self {
        self.should(&be_a_valid_password());
        self
    }
}

Time to put our password assertion to use.

#[test]
fn should_be_a_valid_password() {
    let password = "P@@sw0rd9082";
    password.should_be_a_valid_password();
}

Let's add one more condition for a password to be valid. It must be greater than the string "pass". That means, a password is valid if:

  • it satisfies the previous matcher conditions
  • it is greater than the string "pass"

It might appear that we can combine an OrderedMatcher in the existing be_a_valid_password method.

A key idea behind matcher compatibility is that the matchers can only be combined if they work with the same data type(s).

In this case, OrderedMatcher handles any partially ordered types (T: PartialOrd), while other matchers in be_a_valid_password work with string slices (&str).

To get around this, a separate matcher, be_greater_than_pass, can be created to handle the "greater than 'pass'" condition. This matcher works with String values, ensuring compatibility.

fn be_greater_than_pass() -> Matchers<String> {
    MatchersBuilder::start_building(be_greater_than(String::from("pass")).boxed())
        .push_inverted(be_empty().boxed())
        .combine_as_and()
}

It returns an instance of Matchers<String>, so PasswordAssertion is now implemented for String. The method should_be_a_valid_password now uses both the composed matchers to assert that the password is valid.

impl PasswordAssertion for String {
    fn should_be_a_valid_password(&self) -> &Self {
        self.should(&be_a_valid_password());
        self.should(&be_greater_than_pass());
        self
    }
}

#[test]
fn should_be_a_valid_password() {
    let password = "P@@sw0rd9082".to_string();
    password.should_be_a_valid_password();
}

Rust features

clearcheck crate supports the following features:

Example project

Head over to the examples project to understand the usage of this crate.

Reference

Mention

  • Google bard helped with the documentation and README :).

Dependencies

~0–1MB
~14K SLoC