use crate::description::Description;
use crate::matcher::{Describable, Matcher, MatcherResult};
use regex::Regex;
use std::fmt::Debug;
use std::ops::Deref;
pub fn matches_regex<PatternT: Deref<Target = str>>(
pattern: PatternT,
) -> __internal::MatchesRegexMatcher<PatternT> {
let adjusted_pattern = format!("^{}$", pattern.deref());
let regex = Regex::new(adjusted_pattern.as_str()).unwrap();
__internal::MatchesRegexMatcher { regex, pattern, _adjusted_pattern: adjusted_pattern }
}
pub mod __internal {
use super::*;
#[doc(hidden)]
pub struct MatchesRegexMatcher<PatternT: Deref<Target = str>> {
pub(super) regex: Regex,
pub(super) pattern: PatternT,
pub(super) _adjusted_pattern: String,
}
impl<PatternT, ActualT> Matcher<ActualT> for MatchesRegexMatcher<PatternT>
where
PatternT: Deref<Target = str>,
ActualT: AsRef<str> + Debug + ?Sized,
{
fn matches(&self, actual: &ActualT) -> MatcherResult {
self.regex.is_match(actual.as_ref()).into()
}
}
impl<PatternT: Deref<Target = str>> Describable for MatchesRegexMatcher<PatternT> {
fn describe(&self, matcher_result: MatcherResult) -> Description {
match matcher_result {
MatcherResult::Match => {
format!("matches the regular expression {:#?}", self.pattern.deref()).into()
}
MatcherResult::NoMatch => {
format!("doesn't match the regular expression {:#?}", self.pattern.deref())
.into()
}
}
}
}
}
#[cfg(test)]
mod tests {
use super::matches_regex;
use crate::matcher::{Describable as _, Matcher, MatcherResult};
use crate::prelude::*;
#[test]
fn matches_regex_matches_string_reference_with_pattern() -> TestResult<()> {
let matcher = matches_regex("S.*e");
let result = matcher.matches("Some value");
verify_that!(result, eq(MatcherResult::Match))
}
#[test]
fn matches_regex_does_not_match_string_without_pattern() -> TestResult<()> {
let matcher = matches_regex("Another");
let result = matcher.matches("Some value");
verify_that!(result, eq(MatcherResult::NoMatch))
}
#[test]
fn matches_regex_does_not_match_string_only_beginning_of_which_matches() -> TestResult<()> {
let matcher = matches_regex("Some");
let result = matcher.matches("Some value");
verify_that!(result, eq(MatcherResult::NoMatch))
}
#[test]
fn matches_regex_does_not_match_string_only_end_of_which_matches() -> TestResult<()> {
let matcher = matches_regex("value");
let result = matcher.matches("Some value");
verify_that!(result, eq(MatcherResult::NoMatch))
}
#[test]
fn matches_regex_matches_owned_string_with_pattern() -> TestResult<()> {
let matcher = matches_regex(".*value");
let result = matcher.matches(&"Some value".to_string());
verify_that!(result, eq(MatcherResult::Match))
}
#[test]
fn matches_regex_matches_string_when_regex_has_beginning_of_string_marker() -> TestResult<()> {
let matcher = matches_regex("^Some value");
let result = matcher.matches("Some value");
verify_that!(result, eq(MatcherResult::Match))
}
#[test]
fn matches_regex_matches_string_when_regex_has_end_of_string_marker() -> TestResult<()> {
let matcher = matches_regex("Some value$");
let result = matcher.matches("Some value");
verify_that!(result, eq(MatcherResult::Match))
}
#[test]
fn matches_regex_matches_string_when_regex_has_both_end_markers() -> TestResult<()> {
let matcher = matches_regex("^Some value$");
let result = matcher.matches("Some value");
verify_that!(result, eq(MatcherResult::Match))
}
#[test]
fn matches_regex_matches_string_reference_with_owned_string() -> TestResult<()> {
let matcher = matches_regex(".*value".to_string());
let result = matcher.matches("Some value");
verify_that!(result, eq(MatcherResult::Match))
}
#[test]
fn verify_that_works_with_owned_string() -> TestResult<()> {
verify_that!("Some value".to_string(), matches_regex(".*value"))
}
#[test]
fn matches_regex_displays_quoted_debug_of_pattern() -> TestResult<()> {
let matcher = matches_regex("\n");
verify_that!(
matcher.describe(MatcherResult::Match),
displays_as(eq("matches the regular expression \"\\n\""))
)
}
}