#![allow(
clippy::approx_constant,
clippy::useless_vec,
clippy::len_zero,
clippy::unnecessary_cast,
clippy::redundant_closure,
clippy::too_many_arguments,
clippy::type_complexity,
clippy::needless_borrow,
clippy::enum_variant_names,
clippy::upper_case_acronyms,
clippy::inconsistent_digit_grouping,
clippy::unit_cmp,
clippy::assertions_on_constants,
clippy::iter_on_single_items,
clippy::expect_fun_call,
clippy::redundant_pattern_matching,
variant_size_differences,
clippy::absurd_extreme_comparisons,
clippy::nonminimal_bool,
clippy::for_kv_map,
clippy::needless_range_loop,
clippy::single_match,
clippy::collapsible_if,
clippy::needless_return,
clippy::redundant_clone,
clippy::map_entry,
clippy::match_single_binding,
clippy::bool_comparison,
clippy::derivable_impls,
clippy::manual_range_contains,
clippy::needless_borrows_for_generic_args,
clippy::manual_map,
clippy::vec_init_then_push,
clippy::identity_op,
clippy::manual_flatten,
clippy::single_char_pattern,
clippy::search_is_some,
clippy::option_map_unit_fn,
clippy::while_let_on_iterator,
clippy::clone_on_copy,
clippy::box_collection,
clippy::redundant_field_names,
clippy::ptr_arg,
clippy::large_enum_variant,
clippy::match_ref_pats,
clippy::needless_pass_by_value,
clippy::unused_unit,
clippy::let_and_return,
clippy::suspicious_else_formatting,
clippy::manual_strip,
clippy::match_like_matches_macro,
clippy::from_over_into,
clippy::wrong_self_convention,
clippy::inherent_to_string,
clippy::new_without_default,
clippy::unnecessary_wraps,
clippy::field_reassign_with_default,
clippy::manual_find,
clippy::unnecessary_lazy_evaluations,
clippy::should_implement_trait,
clippy::missing_safety_doc,
clippy::unusual_byte_groupings,
clippy::bool_assert_comparison,
clippy::zero_prefixed_literal,
clippy::await_holding_lock,
clippy::manual_saturating_arithmetic,
clippy::explicit_counter_loop,
clippy::needless_lifetimes,
clippy::single_component_path_imports,
clippy::uninlined_format_args,
clippy::iter_cloned_collect,
clippy::manual_str_repeat,
clippy::excessive_precision,
clippy::precedence,
clippy::unnecessary_literal_unwrap
)]
use oxicode::validation::{CollectionValidator, NumericValidator, ValidationError};
#[cfg(feature = "alloc")]
use oxicode::validation::{Constraint, Constraints, ValidationConfig, ValidationResult, Validator};
#[cfg(test)]
mod validation_comprehensive_tests {
use super::*;
#[cfg(feature = "alloc")]
#[test]
fn test_01_no_constraints_always_passes_u64() {
let v: Validator<u64> = Validator::new();
assert_eq!(v.constraint_count(), 0);
assert!(
v.validate(&0u64).is_ok(),
"0 must pass unconstrained validator"
);
assert!(
v.validate(&u64::MAX).is_ok(),
"u64::MAX must pass unconstrained validator"
);
assert!(
v.validate(&u64::MIN).is_ok(),
"u64::MIN must pass unconstrained validator"
);
}
#[cfg(feature = "alloc")]
#[test]
fn test_02_string_validator_max_len_10_passes() {
use oxicode::validation::StringValidator;
let sv = StringValidator::new().max_len(10);
assert!(
sv.validate("short").is_ok(),
"5-char string must pass max_len(10)"
);
assert!(
sv.validate("0123456789").is_ok(),
"10-char string must pass max_len(10)"
);
}
#[cfg(feature = "alloc")]
#[test]
fn test_03_string_validator_max_len_5_fails_for_long() {
use oxicode::validation::StringValidator;
let sv = StringValidator::new().max_len(5);
assert!(
sv.validate("toolong").is_err(),
"7-char string must fail max_len(5)"
);
}
#[cfg(feature = "alloc")]
#[test]
fn test_04_string_validator_min_len_3_passes() {
use oxicode::validation::StringValidator;
let sv = StringValidator::new().min_len(3);
assert!(
sv.validate("abc").is_ok(),
"exactly 3 chars must pass min_len(3)"
);
assert!(
sv.validate("longer string").is_ok(),
"longer string must pass min_len(3)"
);
}
#[cfg(feature = "alloc")]
#[test]
fn test_05_string_validator_min_len_10_fails_for_short() {
use oxicode::validation::StringValidator;
let sv = StringValidator::new().min_len(10);
assert!(
sv.validate("hi").is_err(),
"2-char string must fail min_len(10)"
);
}
#[test]
fn test_06_numeric_validator_min_only_passes() {
let v = NumericValidator::<i16>::new().min(-100i16);
assert!(v.validate(&-100i16).is_ok(), "exactly at min must pass");
assert!(v.validate(&0i16).is_ok(), "above min must pass");
assert!(v.validate(&i16::MAX).is_ok(), "i16::MAX must pass");
}
#[test]
fn test_07_numeric_validator_max_only_passes() {
let v = NumericValidator::<u8>::new().max(200u8);
assert!(v.validate(&0u8).is_ok(), "0 must pass max(200)");
assert!(v.validate(&200u8).is_ok(), "exactly at max must pass");
}
#[test]
fn test_08_numeric_validator_out_of_range_fails() {
let v = NumericValidator::<u8>::new().max(200u8);
assert!(v.validate(&201u8).is_err(), "201 must fail max(200)");
}
#[test]
fn test_09_numeric_validator_range_f32_passes() {
let v = NumericValidator::<f32>::new().range(-1.0f32, 1.0f32);
assert!(
v.validate(&0.0f32).is_ok(),
"0.0 must pass range [-1.0, 1.0]"
);
assert!(v.validate(&-1.0f32).is_ok(), "exactly -1.0 must pass");
assert!(v.validate(&1.0f32).is_ok(), "exactly 1.0 must pass");
}
#[test]
fn test_10_numeric_validator_range_f32_fails() {
let v = NumericValidator::<f32>::new().range(-1.0f32, 1.0f32);
assert!(
v.validate(&1.1f32).is_err(),
"1.1 must fail range [-1.0, 1.0]"
);
assert!(
v.validate(&-1.1f32).is_err(),
"-1.1 must fail range [-1.0, 1.0]"
);
}
#[test]
fn test_11_collection_validator_max_items_passes_via_len() {
let cv = CollectionValidator::new().max_len(8);
assert!(cv.validate_len(0).is_ok(), "length 0 must pass max_len(8)");
assert!(cv.validate_len(8).is_ok(), "length 8 must pass max_len(8)");
}
#[test]
fn test_12_collection_validator_max_items_fails_via_len() {
let cv = CollectionValidator::new().max_len(8);
assert!(cv.validate_len(9).is_err(), "length 9 must fail max_len(8)");
}
#[test]
fn test_13_collection_validator_min_items_passes() {
let cv = CollectionValidator::new().min_len(3);
let arr: [u32; 3] = [1, 2, 3];
assert!(
cv.validate(&arr).is_ok(),
"3-element slice must pass min_len(3)"
);
let arr2: [u32; 10] = [0; 10];
assert!(
cv.validate(&arr2).is_ok(),
"10-element slice must pass min_len(3)"
);
}
#[test]
fn test_14_collection_validator_min_items_fails() {
let cv = CollectionValidator::new().min_len(5);
let arr: [u8; 4] = [0; 4];
assert!(
cv.validate(&arr).is_err(),
"4-element slice must fail min_len(5)"
);
assert!(cv.validate_len(0).is_err(), "length 0 must fail min_len(5)");
}
#[cfg(feature = "alloc")]
#[test]
fn test_15_ascii_only_constraint_passes_for_ascii() {
let c = Constraints::ascii_only();
let ascii_samples = ["hello", "RUST_2024", "!@#$", "0123456789"];
for s in &ascii_samples {
assert!(
matches!(c.validate(*s), ValidationResult::Valid),
"pure ASCII '{}' must be Valid",
s
);
}
}
#[cfg(feature = "alloc")]
#[test]
fn test_16_ascii_only_constraint_fails_for_unicode() {
let c = Constraints::ascii_only();
let unicode_samples = ["こんにちは", "🦀", "naïve", "Ünïcode"];
for s in &unicode_samples {
assert!(
matches!(c.validate(*s), ValidationResult::Invalid(_)),
"non-ASCII '{}' must be Invalid",
s
);
}
}
#[cfg(feature = "alloc")]
#[test]
fn test_17_validate_or_default_returns_valid_string() {
let v: Validator<String> = Validator::new()
.constraint("tag", Constraints::min_len(1))
.constraint("tag", Constraints::max_len(20));
let result = v.validate_or_default("rust".to_string(), "default".to_string());
assert_eq!(result, "rust", "valid string must be returned unchanged");
}
#[cfg(feature = "alloc")]
#[test]
fn test_18_validate_or_default_returns_default_string_on_fail() {
let v: Validator<String> = Validator::new().constraint("tag", Constraints::max_len(3));
let too_long = "this_is_too_long".to_string();
let result = v.validate_or_default(too_long, "fallback".to_string());
assert_eq!(
result, "fallback",
"invalid string must return the fallback default"
);
}
#[cfg(feature = "alloc")]
#[test]
fn test_19_multiple_validators_chained_mixed_api() {
let mut v: Validator<i32> =
Validator::new().constraint("n", Constraints::range(Some(-50i32), Some(50i32)));
v.add_constraint(
"n",
Constraints::custom(|x: &i32| x % 2 == 0, "must be even", "even-check"),
);
assert_eq!(v.constraint_count(), 2);
assert!(v.validate(&0).is_ok(), "0 satisfies both constraints");
assert!(v.validate(&-50).is_ok(), "-50 is even and at boundary");
assert!(v.validate(&50).is_ok(), "50 is even and at boundary");
assert!(v.validate(&52).is_err(), "52 is even but outside range");
assert!(v.validate(&3).is_err(), "3 is in range but odd");
}
#[cfg(feature = "alloc")]
#[test]
fn test_20_validation_config_fail_fast_with_i32() {
let config = ValidationConfig::new().with_fail_fast(true);
let mut v: Validator<i32> = Validator::with_config(config);
v.add_constraint("val", Constraints::range(Some(0i32), Some(100i32)));
v.add_constraint("val", Constraints::range(Some(-10i32), Some(10i32)));
v.add_constraint(
"val",
Constraints::custom(|x: &i32| *x == 0, "must be zero", "zero-check"),
);
let errors = v
.validate(&200i32)
.expect_err("200 must fail all constraints");
assert_eq!(
errors.len(),
1,
"fail_fast must stop at first error, got {} errors",
errors.len()
);
assert_eq!(errors[0].field, "val");
}
#[cfg(feature = "alloc")]
#[test]
fn test_21_encode_validate_decode_roundtrip() {
let byte_validator: Validator<Vec<u8>> = Validator::new()
.constraint("bytes", Constraints::min_len(1))
.constraint("bytes", Constraints::max_len(16));
let original: u32 = 12345u32;
let encoded = oxicode::encode_to_vec(&original).expect("encode_to_vec must not fail");
assert!(
byte_validator.validate(&encoded).is_ok(),
"encoded bytes length must be in [1, 16]; got {}",
encoded.len()
);
let (decoded, consumed): (u32, _) =
oxicode::decode_from_slice(&encoded).expect("decode_from_slice must not fail");
assert_eq!(decoded, original, "decoded value must equal original");
assert_eq!(consumed, encoded.len(), "all bytes must be consumed");
}
#[cfg(feature = "alloc")]
#[test]
fn test_22_custom_error_message_propagated() {
const ERR_MSG: &str = "value must be a multiple of seven";
let v: Validator<i32> = Validator::new().constraint(
"lucky",
Constraints::custom(|x: &i32| x % 7 == 0, ERR_MSG, "multiple-of-seven"),
);
assert!(v.validate(&0).is_ok(), "0 is divisible by 7");
assert!(v.validate(&49).is_ok(), "49 is divisible by 7");
assert!(v.validate(&-7).is_ok(), "-7 is divisible by 7");
let errors = v.validate(&3i32).expect_err("3 is not divisible by 7");
assert_eq!(errors.len(), 1);
assert_eq!(
errors[0].message, ERR_MSG,
"error message must be '{}'",
ERR_MSG
);
assert_eq!(errors[0].field, "lucky");
let display = format!("{}", errors[0]);
assert!(
display.contains(ERR_MSG),
"Display output must contain the custom message: {}",
display
);
let manual = ValidationError::new("lucky", ERR_MSG);
assert_eq!(manual.field, "lucky");
assert_eq!(manual.message, ERR_MSG);
}
}