use core::marker::PhantomData;
pub fn each<MatcherT, Mode>(inner: MatcherT) -> __internal::EachMatcher<MatcherT, Mode> {
__internal::EachMatcher { inner, phantom: PhantomData }
}
pub mod __internal {
use crate::description::Description;
use crate::matcher::{Describable, Matcher, MatcherResult};
use crate::matchers::containers::{OwnedItems, RefItems};
use alloc::vec::Vec;
use core::fmt::Debug;
use core::marker::PhantomData;
#[doc(hidden)]
pub struct EachMatcher<MatcherT, Mode> {
pub(super) inner: MatcherT,
pub(super) phantom: PhantomData<Mode>,
}
impl<ElementT: Debug, ActualT: Debug + ?Sized, MatcherT: Matcher<ElementT>> Matcher<ActualT>
for EachMatcher<MatcherT, RefItems>
where
for<'a> &'a ActualT: IntoIterator<Item = &'a ElementT>,
{
fn matches(&self, actual: &ActualT) -> MatcherResult {
for element in actual {
if self.inner.matches(element).is_no_match() {
return MatcherResult::NoMatch;
}
}
MatcherResult::Match
}
fn explain_match(&self, actual: &ActualT) -> Description {
let mut non_matching_elements = Vec::new();
for (index, element) in actual.into_iter().enumerate() {
if self.inner.matches(element).is_no_match() {
non_matching_elements.push((index, element, self.inner.explain_match(element)));
}
}
if non_matching_elements.is_empty() {
return format!("whose each element {}", self.inner.describe(MatcherResult::Match))
.into();
}
if non_matching_elements.len() == 1 {
let (idx, element, explanation) = non_matching_elements.remove(0);
return format!("whose element #{idx} is {element:?}, {explanation}").into();
}
let failed_indexes = non_matching_elements
.iter()
.map(|&(idx, _, _)| format!("#{idx}"))
.collect::<Vec<_>>()
.join(", ");
let element_explanations = non_matching_elements
.iter()
.map(|&(_, element, ref explanation)| format!("{element:?}, {explanation}"))
.collect::<Description>()
.indent();
format!("whose elements {failed_indexes} don't match\n{element_explanations}").into()
}
}
impl<ElementT: Debug, ActualT: Debug + ?Sized, MatcherT: Matcher<ElementT>> Matcher<ActualT>
for EachMatcher<MatcherT, OwnedItems>
where
for<'a> &'a ActualT: IntoIterator<Item = ElementT>,
{
fn matches(&self, actual: &ActualT) -> MatcherResult {
for element in actual {
if self.inner.matches(&element).is_no_match() {
return MatcherResult::NoMatch;
}
}
MatcherResult::Match
}
fn explain_match(&self, actual: &ActualT) -> Description {
let mut non_matching_elements = Vec::new();
for (index, element) in actual.into_iter().enumerate() {
if self.inner.matches(&element).is_no_match() {
let explanation = self.inner.explain_match(&element);
non_matching_elements.push((index, element, explanation));
}
}
if non_matching_elements.is_empty() {
return format!("whose each element {}", self.inner.describe(MatcherResult::Match))
.into();
}
if non_matching_elements.len() == 1 {
let (idx, element, explanation) = non_matching_elements.remove(0);
return format!("whose element #{idx} is {element:?}, {explanation}").into();
}
let failed_indexes = non_matching_elements
.iter()
.map(|&(idx, _, _)| format!("#{idx}"))
.collect::<Vec<_>>()
.join(", ");
let element_explanations = non_matching_elements
.iter()
.map(|(_, element, explanation)| format!("{element:?}, {explanation}"))
.collect::<Description>()
.indent();
format!("whose elements {failed_indexes} don't match\n{element_explanations}").into()
}
}
impl<MatcherT: Describable, Mode> Describable for EachMatcher<MatcherT, Mode> {
fn describe(&self, matcher_result: MatcherResult) -> Description {
match matcher_result {
MatcherResult::Match => format!(
"only contains elements that {}",
self.inner.describe(MatcherResult::Match)
)
.into(),
MatcherResult::NoMatch => format!(
"contains no element that {}",
self.inner.describe(MatcherResult::Match)
)
.into(),
}
}
}
}
#[cfg(test)]
mod tests {
use super::each;
use crate::prelude::*;
use alloc::vec::Vec;
use indoc::indoc;
#[test]
fn each_matches_empty_vec() -> TestResult<()> {
let value: Vec<i32> = vec![];
verify_that!(value, each(gt(0)))
}
#[test]
fn each_matches_vec_with_one_element() -> TestResult<()> {
verify_that!(vec![1], each(gt(0)))
}
#[test]
fn each_matches_array_with_one_element() -> TestResult<()> {
verify_that!([1], each(gt(0)))
}
#[test]
fn each_matches_ref_to_array_using_deref_notation() -> TestResult<()> {
let value = [1];
let reference = &value;
verify_that!(*reference, each(gt(0)))
}
#[test]
fn each_matches_ref_to_array_using_points_to() -> TestResult<()> {
let value = [1];
let reference = &value;
verify_that!(reference, points_to(each(gt(0))))
}
#[test]
fn each_matches_slice_using_deref_notation() -> TestResult<()> {
let value = vec![1];
let slice = value.as_slice();
verify_that!(*slice, each(gt(0)))
}
#[test]
fn each_matches_slice_using_points_to() -> TestResult<()> {
let value = vec![1];
let slice = value.as_slice();
verify_that!(slice, points_to(each(gt(0))))
}
#[test]
fn each_matches_vec_with_two_elements() -> TestResult<()> {
let value = vec![1, 2];
verify_that!(value, each(gt(0)))
}
#[derive(Debug)]
struct OwnedItemContainer(Vec<i32>);
impl<'a> IntoIterator for &'a OwnedItemContainer {
type Item = i32;
type IntoIter = core::iter::Copied<core::slice::Iter<'a, i32>>;
fn into_iter(self) -> Self::IntoIter {
self.0.iter().copied()
}
}
#[test]
fn each_matches_on_container_when_ref_to_container_has_into_iterator_producing_owned_values()
-> TestResult<()> {
verify_that!(OwnedItemContainer(vec![1]), each(eq(1)))
}
#[cfg(feature = "std")]
#[test]
fn each_matches_hash_set_with_one_element() -> TestResult<()> {
use std::collections::HashSet;
let value: HashSet<i32> = [1].into();
verify_that!(value, each(gt(0)))
}
#[test]
fn each_does_not_match_when_first_element_does_not_match() -> TestResult<()> {
let value = vec![0];
verify_that!(value, not(each(gt(1))))
}
#[test]
fn each_does_not_match_when_second_element_does_not_match() -> TestResult<()> {
let value = vec![2, 0];
verify_that!(value, not(each(gt(1))))
}
#[test]
fn each_shows_correct_message_when_first_item_does_not_match() -> TestResult<()> {
let result = verify_that!(vec![0, 2, 3], each(gt(0)));
verify_that!(
result,
err(displays_as(contains_substring(indoc!(
"
Value of: vec![0, 2, 3]
Expected: only contains elements that is greater than 0
Actual: [0, 2, 3],
whose element #0 is 0, which is less than or equal to 0"
))))
)
}
#[test]
fn each_shows_correct_message_when_second_item_does_not_match() -> TestResult<()> {
let result = verify_that!(vec![1, 0, 3], each(gt(0)));
verify_that!(
result,
err(displays_as(contains_substring(indoc!(
"
Value of: vec![1, 0, 3]
Expected: only contains elements that is greater than 0
Actual: [1, 0, 3],
whose element #1 is 0, which is less than or equal to 0"
))))
)
}
#[test]
fn each_shows_correct_message_when_first_two_items_do_not_match() -> TestResult<()> {
let result = verify_that!(vec![0, 1, 3], each(gt(1)));
verify_that!(
result,
err(displays_as(contains_substring(indoc!(
"
Value of: vec![0, 1, 3]
Expected: only contains elements that is greater than 1
Actual: [0, 1, 3],
whose elements #0, #1 don't match
0, which is less than or equal to 1
1, which is less than or equal to 1"
))))
)
}
#[test]
fn each_shows_inner_explanation() -> TestResult<()> {
let result = verify_that!(vec![vec![1, 2], vec![1]], each(each(eq(1))));
verify_that!(
result,
err(displays_as(contains_substring(indoc!(
"
Expected: only contains elements that only contains elements that is equal to 1
Actual: [[1, 2], [1]],
whose element #0 is [1, 2], whose element #1 is 2, which isn't equal to 1"
))))
)
}
}