const-str 1.1.0

compile-time string operations
Documentation
use core::cmp::Ordering;

pub struct Compare<T1, T2>(pub T1, pub T2);

impl Compare<&[u8], &[u8]> {
    pub const fn const_eval(&self) -> Ordering {
        crate::bytes::compare(self.0, self.1)
    }
}

impl<const L1: usize, const L2: usize> Compare<&[u8; L1], &[u8; L2]> {
    pub const fn const_eval(&self) -> Ordering {
        crate::bytes::compare(self.0, self.1)
    }
}

impl Compare<&str, &str> {
    pub const fn const_eval(&self) -> Ordering {
        crate::str::compare(self.0, self.1)
    }
}

/// Compares two strings lexicographically.
///
/// This macro is [const-fn compatible](./index.html#const-fn-compatible).
///
/// See also [`equal!`](crate::equal).
///
/// # Examples
///
/// ```
/// use core::cmp::Ordering;
///
/// const A: &str = "1";
/// const B: &str = "10";
/// const C: &str = "2";
///
/// const ORD: Ordering = const_str::compare!(A, B);
/// assert_eq!(ORD, Ordering::Less);
///
/// assert!(const_str::compare!(<, A, B));
/// assert!(const_str::compare!(<=, A, B));
///
/// assert!(const_str::compare!(>, C, A));
/// assert!(const_str::compare!(>=, C, A));
///
/// assert!(const_str::compare!(==, A, A));
/// ```
///
#[macro_export]
macro_rules! compare {
    (<, $lhs: expr, $rhs: expr) => {{
        use ::core::cmp::Ordering;
        let ordering = $crate::__ctfe::Compare($lhs, $rhs).const_eval();
        matches!(ordering, Ordering::Less)
    }};
    (>, $lhs: expr, $rhs: expr) => {{
        use ::core::cmp::Ordering;
        let ordering = $crate::__ctfe::Compare($lhs, $rhs).const_eval();
        matches!(ordering, Ordering::Greater)
    }};
    (==, $lhs: expr, $rhs: expr) => {{
        use ::core::cmp::Ordering;
        let ordering = $crate::__ctfe::Compare($lhs, $rhs).const_eval();
        matches!(ordering, Ordering::Equal)
    }};
    (<=, $lhs: expr, $rhs: expr) => {{
        use ::core::cmp::Ordering;
        let ordering = $crate::__ctfe::Compare($lhs, $rhs).const_eval();
        matches!(ordering, Ordering::Less | Ordering::Equal)
    }};
    (>=, $lhs: expr, $rhs: expr) => {{
        use ::core::cmp::Ordering;
        let ordering = $crate::__ctfe::Compare($lhs, $rhs).const_eval();
        matches!(ordering, Ordering::Greater | Ordering::Equal)
    }};
    ($lhs: expr, $rhs: expr) => {
        $crate::__ctfe::Compare($lhs, $rhs).const_eval()
    };
}

#[cfg(test)]
mod tests {
    use core::cmp::Ordering;

    #[test]
    fn test_compare_str() {
        const A: &str = "apple";
        const B: &str = "banana";
        const C: &str = "apple";

        const ORD1: Ordering = compare!(A, B);
        const ORD2: Ordering = compare!(B, A);
        const ORD3: Ordering = compare!(A, C);

        assert_eq!(ORD1, Ordering::Less);
        assert_eq!(ORD2, Ordering::Greater);
        assert_eq!(ORD3, Ordering::Equal);

        let lt = compare!(<, A, B);
        let gt = compare!(>, B, A);
        let eq = compare!(==, A, C);
        let le1 = compare!(<=, A, B);
        let le2 = compare!(<=, A, C);
        let ge1 = compare!(>=, B, A);
        let ge2 = compare!(>=, A, C);

        assert!(lt);
        assert!(gt);
        assert!(eq);
        assert!(le1);
        assert!(le2);
        assert!(ge1);
        assert!(ge2);
    }

    #[test]
    fn test_compare_bytes() {
        const A: &[u8] = b"apple";
        const B: &[u8] = b"banana";
        const C: &[u8] = b"apple";

        const ORD1: Ordering = compare!(A, B);
        const ORD2: Ordering = compare!(B, A);
        const ORD3: Ordering = compare!(A, C);

        assert_eq!(ORD1, Ordering::Less);
        assert_eq!(ORD2, Ordering::Greater);
        assert_eq!(ORD3, Ordering::Equal);
    }

    #[test]
    fn test_compare_byte_arrays() {
        const A: &[u8; 5] = b"apple";
        const B: &[u8; 6] = b"banana";
        const C: &[u8; 5] = b"apple";

        const ORD1: Ordering = compare!(A, B);
        const ORD2: Ordering = compare!(B, A);
        const ORD3: Ordering = compare!(A, C);

        assert_eq!(ORD1, Ordering::Less);
        assert_eq!(ORD2, Ordering::Greater);
        assert_eq!(ORD3, Ordering::Equal);
    }

    #[test]
    fn test_compare_runtime() {
        use super::*;

        // Runtime tests for Compare with &str
        let cmp_str = Compare("apple", "banana");
        assert_eq!(cmp_str.const_eval(), Ordering::Less);

        let cmp_str2 = Compare("zebra", "apple");
        assert_eq!(cmp_str2.const_eval(), Ordering::Greater);

        let cmp_str3 = Compare("test", "test");
        assert_eq!(cmp_str3.const_eval(), Ordering::Equal);

        // Runtime tests for Compare with bytes
        let cmp_bytes = Compare(b"hello".as_slice(), b"world".as_slice());
        assert_eq!(cmp_bytes.const_eval(), Ordering::Less);

        // Runtime tests for Compare with byte arrays
        let arr1: &[u8; 3] = b"abc";
        let arr2: &[u8; 3] = b"xyz";
        let cmp_arr = Compare(arr1, arr2);
        assert_eq!(cmp_arr.const_eval(), Ordering::Less);
    }
}