#![cfg(feature = "std")]
pub fn has_entry<KeyT, MatcherT>(
key: KeyT,
inner: MatcherT,
) -> __internal::HasEntryMatcher<KeyT, MatcherT> {
__internal::HasEntryMatcher { key, inner }
}
pub mod __internal {
use crate::description::Description;
use crate::matcher::{Describable, Matcher, MatcherResult};
use core::fmt::Debug;
use core::hash::Hash;
use std::collections::HashMap;
#[doc(hidden)]
pub struct HasEntryMatcher<KeyT, MatcherT> {
pub(super) key: KeyT,
pub(super) inner: MatcherT,
}
impl<KeyT: Debug + Eq + Hash, ValueT: Debug, MatcherT: Matcher<ValueT>>
Matcher<HashMap<KeyT, ValueT>> for HasEntryMatcher<KeyT, MatcherT>
{
fn matches(&self, actual: &HashMap<KeyT, ValueT>) -> MatcherResult {
if let Some(value) = actual.get(&self.key) {
self.inner.matches(value)
} else {
MatcherResult::NoMatch
}
}
fn explain_match(&self, actual: &HashMap<KeyT, ValueT>) -> Description {
if let Some(value) = actual.get(&self.key) {
format!(
"which contains key {:?}, but is mapped to value {:#?}, {}",
self.key,
value,
self.inner.explain_match(value)
)
.into()
} else {
format!("which doesn't contain key {:?}", self.key).into()
}
}
}
impl<KeyT: Debug, MatcherT: Describable> Describable for HasEntryMatcher<KeyT, MatcherT> {
fn describe(&self, matcher_result: MatcherResult) -> Description {
match matcher_result {
MatcherResult::Match => format!(
"contains key {:?}, which value {}",
self.key,
self.inner.describe(MatcherResult::Match)
)
.into(),
MatcherResult::NoMatch => format!(
"doesn't contain key {:?} or contains key {:?}, which value {}",
self.key,
self.key,
self.inner.describe(MatcherResult::NoMatch)
)
.into(),
}
}
}
}
#[cfg(test)]
mod tests {
use super::has_entry;
use crate::prelude::*;
use indoc::indoc;
use std::collections::HashMap;
#[test]
fn has_entry_does_not_match_empty_hash_map() -> TestResult<()> {
let value: HashMap<i32, i32> = HashMap::new();
verify_that!(value, not(has_entry(0, eq(0))))
}
#[test]
fn has_entry_matches_hash_map_with_value() -> TestResult<()> {
let value: HashMap<i32, i32> = HashMap::from([(0, 0)]);
verify_that!(value, has_entry(0, eq(0)))
}
#[test]
fn has_entry_does_not_match_hash_map_with_wrong_value() -> TestResult<()> {
let value: HashMap<i32, i32> = HashMap::from([(0, 1)]);
verify_that!(value, not(has_entry(0, eq(0))))
}
#[test]
fn has_entry_does_not_match_hash_map_with_wrong_key() -> TestResult<()> {
let value: HashMap<i32, i32> = HashMap::from([(1, 0)]);
verify_that!(value, not(has_entry(0, eq(0))))
}
#[test]
fn has_entry_shows_correct_message_when_key_is_not_present() -> TestResult<()> {
let result = verify_that!(HashMap::from([(0, 0)]), has_entry(1, eq(0)));
verify_that!(
result,
err(displays_as(contains_substring(indoc!(
"
Value of: HashMap::from([(0, 0)])
Expected: contains key 1, which value is equal to 0
Actual: {0: 0},
which doesn't contain key 1
"
))))
)
}
#[test]
fn has_entry_shows_correct_message_when_key_has_non_matching_value() -> TestResult<()> {
let result = verify_that!(HashMap::from([(0, 0)]), has_entry(0, eq(1)));
verify_that!(
result,
err(displays_as(contains_substring(indoc!(
"
Value of: HashMap::from([(0, 0)])
Expected: contains key 0, which value is equal to 1
Actual: {0: 0},
which contains key 0, but is mapped to value 0, which isn't equal to 1
"
))))
)
}
}