1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
use crate::validation::{impl_generic_composited_validation_1args, ValidateCompositedMinimum};
use crate::MinimumError;

/// Minimum validation of the number.
///
/// See <https://json-schema.org/understanding-json-schema/reference/numeric.html#range>
///
/// ```rust
/// use serde_json::json;
/// use serde_valid::{Validate, ValidateMinimum};
/// struct MyType(i32);
///
/// impl ValidateMinimum<i32> for MyType {
///     fn validate_minimum(&self, minimum: i32) -> Result<(), serde_valid::MinimumError> {
///         self.0.validate_minimum(minimum)
///     }
/// }
///
/// #[derive(Validate)]
/// struct TestStruct {
///     #[validate(minimum = 5)]
///     val: MyType,
/// }
///
/// let s = TestStruct { val: MyType(3) };
///
/// assert_eq!(
///     s.validate().unwrap_err().to_string(),
///     json!({
///         "errors": [],
///         "properties": {
///             "val": {
///                 "errors": ["The number must be `>= 5`."]
///             }
///         }
///     })
///     .to_string()
/// );
/// ```
pub trait ValidateMinimum<T>
where
    T: PartialOrd + PartialEq,
{
    fn validate_minimum(&self, minimum: T) -> Result<(), MinimumError>;
}

macro_rules! impl_validate_numeric_minimum {
    ($type:ty) => {
        impl ValidateMinimum<$type> for $type {
            fn validate_minimum(&self, minimum: $type) -> Result<(), MinimumError> {
                if *self >= minimum {
                    Ok(())
                } else {
                    Err(MinimumError::new(minimum))
                }
            }
        }

        impl_generic_composited_validation_1args!(Minimum, $type);
    };
}

impl_validate_numeric_minimum!(i8);
impl_validate_numeric_minimum!(i16);
impl_validate_numeric_minimum!(i32);
impl_validate_numeric_minimum!(i64);
#[cfg(feature = "i128")]
impl_validate_numeric_minimum!(i128);
impl_validate_numeric_minimum!(isize);
impl_validate_numeric_minimum!(u8);
impl_validate_numeric_minimum!(u16);
impl_validate_numeric_minimum!(u32);
impl_validate_numeric_minimum!(u64);
#[cfg(feature = "i128")]
impl_validate_numeric_minimum!(u128);
impl_validate_numeric_minimum!(usize);
impl_validate_numeric_minimum!(std::num::NonZeroI8);
impl_validate_numeric_minimum!(std::num::NonZeroI16);
impl_validate_numeric_minimum!(std::num::NonZeroI32);
impl_validate_numeric_minimum!(std::num::NonZeroI64);
#[cfg(feature = "i128")]
impl_validate_numeric_minimum!(std::num::NonZeroI128);
impl_validate_numeric_minimum!(std::num::NonZeroIsize);
impl_validate_numeric_minimum!(std::num::NonZeroU8);
impl_validate_numeric_minimum!(std::num::NonZeroU16);
impl_validate_numeric_minimum!(std::num::NonZeroU32);
impl_validate_numeric_minimum!(std::num::NonZeroU64);
#[cfg(feature = "i128")]
impl_validate_numeric_minimum!(std::num::NonZeroU128);
impl_validate_numeric_minimum!(std::num::NonZeroUsize);
impl_validate_numeric_minimum!(f32);
impl_validate_numeric_minimum!(f64);

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_validate_numeric_minimum_is_true() {
        assert!(ValidateMinimum::validate_minimum(&10, 9).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10, 10).is_ok());
    }

    #[test]
    fn test_validate_numeric_minimum_is_false() {
        assert!(ValidateMinimum::validate_minimum(&10, 11).is_err());
    }

    #[test]
    fn test_validate_numeric_minimum_specified_type() {
        assert!(ValidateMinimum::validate_minimum(&10, 10i8).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10, 10i16).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10, 10i32).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10, 10i64).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10, 10isize).is_ok());

        assert!(ValidateMinimum::validate_minimum(&10, 10u8).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10, 10u16).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10, 10u32).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10, 10u64).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10, 10usize).is_ok());

        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroI8::new(10).unwrap(),
            std::num::NonZeroI8::new(10).unwrap()
        )
        .is_ok());
        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroI16::new(10).unwrap(),
            std::num::NonZeroI16::new(10).unwrap()
        )
        .is_ok());
        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroI32::new(10).unwrap(),
            std::num::NonZeroI32::new(10).unwrap()
        )
        .is_ok());
        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroI64::new(10).unwrap(),
            std::num::NonZeroI64::new(10).unwrap()
        )
        .is_ok());
        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroIsize::new(10).unwrap(),
            std::num::NonZeroIsize::new(10).unwrap()
        )
        .is_ok());

        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroU8::new(10).unwrap(),
            std::num::NonZeroU8::new(10).unwrap()
        )
        .is_ok());
        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroU16::new(10).unwrap(),
            std::num::NonZeroU16::new(10).unwrap()
        )
        .is_ok());
        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroU32::new(10).unwrap(),
            std::num::NonZeroU32::new(10).unwrap()
        )
        .is_ok());
        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroU64::new(10).unwrap(),
            std::num::NonZeroU64::new(10).unwrap()
        )
        .is_ok());
        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroUsize::new(10).unwrap(),
            std::num::NonZeroUsize::new(10).unwrap()
        )
        .is_ok());

        assert!(ValidateMinimum::validate_minimum(&10.0, 10.0f32).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10.0, 10.0f64).is_ok());
    }

    #[test]
    #[cfg(feature = "i128")]
    fn test_validate_numeric_minimum_128() {
        assert!(ValidateMinimum::validate_minimum(&10, 10i128).is_ok());
        assert!(ValidateMinimum::validate_minimum(&10, 10u128).is_ok());
        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroI128::new(10).unwrap(),
            std::num::NonZeroI128::new(10).unwrap()
        )
        .is_ok());
        assert!(ValidateMinimum::validate_minimum(
            &std::num::NonZeroU128::new(10).unwrap(),
            std::num::NonZeroU128::new(10).unwrap()
        )
        .is_ok());
    }
}