test-that 0.4.1

A rich assertion and matcher library based on GoogleTest
Documentation
// Copyright 2022 Google LLC
// Copyright 2026 Bradford Hovinen <bradford@hovinen.me>
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//      http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

/// Matches a value less (in the sense of `<`) than `expected`.
///
/// The types of the actual and expected values must be comparable via the
/// `PartialOrd` trait. Namely, type of the actual value must implement
/// `PartialOrd<Expected>`, where `Expected` is the type of the expected
/// value passed as an argument to `lt`.
///
/// ```
/// # use test_that::prelude::*;
/// # fn should_pass() -> TestResult<()> {
/// verify_that!(1, lt(2))?; // Passes
/// #     Ok(())
/// # }
/// # fn should_fail() -> TestResult<()> {
/// verify_that!(2, lt(2))?; // Fails
/// #     Ok(())
/// # }
/// # should_pass().unwrap();
/// # should_fail().unwrap_err();
/// ```
///
/// In most cases the params need to be the same type or they need to be cast
/// explicitly. This can be surprising when comparing integer types or
/// references:
///
/// ```compile_fail
/// # use test_that::prelude::*;
/// # fn should_not_compile() -> TestResult<()> {
/// verify_that!(123u32, lt(0u64))?; // Does not compile
/// verify_that!(123u32 as u64, lt(100000000u64))?; // Passes
/// #     Ok(())
/// # }
/// ```
///
/// ```compile_fail
/// # use test_that::prelude::*;
/// # fn should_not_compile() -> TestResult<()> {
/// let actual: &u32 = &2;
/// let expected: u32 = 70;
/// verify_that!(actual, lt(expected))?; // Does not compile
/// #     Ok(())
/// # }
/// ```
///
/// ```
/// # use test_that::prelude::*;
/// # fn should_pass() -> TestResult<()> {
/// let actual: &u32 = &2;
/// let expected: u32 = 70;
/// verify_that!(actual, lt(&expected))?; // Compiles and passes
/// #     Ok(())
/// # }
/// # should_pass().unwrap();
/// ```
///
/// You can find the standard library `PartialOrd` implementation in
/// <https://doc.rust-lang.org/core/cmp/trait.PartialOrd.html#implementors>
pub fn lt<ExpectedT>(expected: ExpectedT) -> __internal::LtMatcher<ExpectedT> {
    __internal::LtMatcher { expected }
}

pub mod __internal {
    use crate::{
        description::Description,
        matcher::{Describable, Matcher, MatcherResult},
    };
    use core::fmt::Debug;

    #[doc(hidden)]
    pub struct LtMatcher<ExpectedT> {
        pub(super) expected: ExpectedT,
    }

    impl<ActualT: Debug + PartialOrd<ExpectedT>, ExpectedT: Debug> Matcher<ActualT>
        for LtMatcher<ExpectedT>
    {
        fn matches(&self, actual: &ActualT) -> MatcherResult {
            (*actual < self.expected).into()
        }
    }

    impl<ExpectedT: Debug> Describable for LtMatcher<ExpectedT> {
        fn describe(&self, matcher_result: MatcherResult) -> Description {
            match matcher_result {
                MatcherResult::Match => format!("is less than {:?}", self.expected).into(),
                MatcherResult::NoMatch => {
                    format!("is greater than or equal to {:?}", self.expected).into()
                }
            }
        }
    }
}

#[cfg(test)]
mod tests {
    use super::lt;
    use crate::matcher::{Matcher, MatcherResult};
    use crate::prelude::*;
    use alloc::string::ToString;
    use indoc::indoc;

    #[test]
    fn lt_matches_i32_with_i32() -> TestResult<()> {
        let actual: i32 = 10000;
        let expected: i32 = 20000;
        verify_that!(actual, lt(expected))
    }

    #[test]
    fn lt_does_not_match_equal_i32() -> TestResult<()> {
        let matcher = lt(10);
        let result = matcher.matches(&10);
        verify_that!(result, eq(MatcherResult::NoMatch))
    }

    #[test]
    fn lt_does_not_match_lower_i32() -> TestResult<()> {
        let matcher = lt(-50);
        let result = matcher.matches(&50);
        verify_that!(result, eq(MatcherResult::NoMatch))
    }

    #[test]
    fn lt_matches_lesser_str() -> TestResult<()> {
        verify_that!("A", lt("B"))
    }

    #[test]
    fn lt_does_not_match_bigger_str() -> TestResult<()> {
        let matcher = lt("ab");
        let result = matcher.matches(&"az");
        verify_that!(result, eq(MatcherResult::NoMatch))
    }

    #[test]
    fn lt_mismatch_contains_actual_and_expected() -> TestResult<()> {
        let result = verify_that!(481, lt(45));
        let formatted_message = format!("{}", result.unwrap_err());

        verify_that!(
            formatted_message.as_str(),
            contains_substring(indoc!(
                "
                Value of: 481
                Expected: is less than 45
                Actual: 481,
                  which is greater than or equal to 45
                "
            ))
        )
    }

    // Test `lt` matcher where actual is `&OsString` and expected is `&str`.
    // Note that stdlib is a little bit inconsistent: `PartialOrd` exists for
    // `OsString` and `str`, but only in one direction: it's only possible to
    // compare `OsString` with `str` if `OsString` is on the left side of the
    // "<" operator (`impl PartialOrd<str> for OsString`).
    //
    // The comparison in the other direction is not defined.
    //
    // This means that the test case bellow effectively ensures that
    // `verify_that(actual, lt(expected))` works if `actual < expected` works
    // (regardless whether the `expected < actual` works`).
    #[cfg(feature = "std")]
    #[test]
    fn lt_matches_owned_osstring_reference_with_string_reference() -> TestResult<()> {
        use std::ffi::OsString;
        let expected = "C";
        let actual: OsString = "B".to_string().into();
        verify_that!(&actual, lt(expected))
    }

    #[cfg(feature = "std")]
    #[test]
    fn lt_matches_ipv6addr_with_ipaddr() -> TestResult<()> {
        use std::net::IpAddr;
        use std::net::Ipv6Addr;
        let actual: IpAddr = "127.0.0.1".parse().unwrap();
        let expected: Ipv6Addr = "2001:4860:4860::8844".parse().unwrap();
        verify_that!(actual, lt(expected))
    }

    #[test]
    fn lt_matches_with_custom_partial_ord() -> TestResult<()> {
        /// A custom "number" that is smaller than all other numbers. The only
        /// things we define about this "special" number is `PartialOrd` and
        /// `PartialEq` against `u32`.
        #[derive(Debug)]
        struct VeryLowNumber {}

        impl core::cmp::PartialEq<u32> for VeryLowNumber {
            fn eq(&self, _other: &u32) -> bool {
                false
            }
        }

        // PartialOrd (required for >) requires PartialEq.
        impl core::cmp::PartialOrd<u32> for VeryLowNumber {
            fn partial_cmp(&self, _other: &u32) -> Option<core::cmp::Ordering> {
                Some(core::cmp::Ordering::Less)
            }
        }

        impl core::cmp::PartialEq<VeryLowNumber> for u32 {
            fn eq(&self, _other: &VeryLowNumber) -> bool {
                false
            }
        }

        impl core::cmp::PartialOrd<VeryLowNumber> for u32 {
            fn partial_cmp(&self, _other: &VeryLowNumber) -> Option<core::cmp::Ordering> {
                Some(core::cmp::Ordering::Greater)
            }
        }

        let actual = VeryLowNumber {};
        let expected: u32 = 42;

        verify_that!(actual, lt(expected))
    }
}