Macros 1.1 custom derive to simplify struct validation inspired by marshmallow and Django validators.
It requires Rust 1.36.
Installation:
[dependencies]
validator = "0.10"
validator_derive = "0.10"
A short example:
#[macro_use]
extern crate validator_derive;
extern crate validator;
#[macro_use]
extern crate serde_derive;
extern crate serde_json;
// A trait that the Validate derive will impl
use validator::{Validate, ValidationError};
#[derive(Debug, Validate, Deserialize)]
struct SignupData {
#[validate(email)]
mail: String,
#[validate(phone)]
phone: String,
#[validate(url)]
site: String,
#[validate(length(min = 1), custom = "validate_unique_username")]
#[serde(rename = "firstName")]
first_name: String,
#[validate(range(min = 18, max = 20))]
age: u32,
}
fn validate_unique_username(username: &str) -> Result<(), ValidationError> {
if username == "xXxShad0wxXx" {
// the value of the username will automatically be added later
return Err(ValidationError::new("terrible_username"));
}
Ok(())
}
match signup_data.validate() {
Ok(_) => (),
Err(e) => return e;
};
The validate()
method returns a Result<(), ValidationErrors>
. In the case of an invalid result, the
ValidationErrors
instance includes a map of errors keyed against the struct's field names. Errors may be represented
in three ways, as described by the ValidationErrorsKind
enum:
#[derive(Debug, Serialize, Clone, PartialEq)]
#[serde(untagged)]
pub enum ValidationErrorsKind {
Struct(Box<ValidationErrors>),
List(BTreeMap<usize, Box<ValidationErrors>>),
Field(Vec<ValidationError>),
}
In the simple example above, any errors would be of the Field(Vec<ValidationError>)
type, where a single
ValidationError
has the following structure:
#[derive(Debug, PartialEq, Clone, Serialize, Deserialize)]
pub struct ValidationError {
pub code: Cow<'static, str>,
pub message: Option<Cow<'static, str>>,
pub params: HashMap<Cow<'static, str>, Value>,
}
The value of the field will automatically be added to the params with a key of value
.
Note that validator
works in conjunction with serde: in the example we can see that the first_name
field is renamed from/to firstName
. Any error on that field will be in the firstName
key of the hashmap,
not first_name
.
If you are adding a validation on a Option<..>
field, it will only be ran if there is a value. The exception
being must_match
that doesn't currently work with Option
due to me not finding a use case for it. If you have one,
please comment on Keats#7.
The other two ValidationErrorsKind
types represent errors discovered in nested (vectors of) structs, as described in
this example:
#[macro_use]
extern crate validator_derive;
extern crate validator;
#[macro_use]
extern crate serde_derive;
extern crate serde_json;
#[derive(Debug, Validate, Deserialize)]
struct SignupData {
#[validate]
contact_details: ContactDetails,
#[validate]
preferences: Vec<Preference>,
#[validate(required)]
allow_cookies: Option<bool>,
}
#[derive(Debug, Validate, Deserialize)]
struct ContactDetails {
#[validate(email)]
mail: String,
#[validate(phone)]
phone: String
}
#[derive(Debug, Validate, Deserialize)]
struct Preference {
#[validate(length(min = 4))]
name: String,
value: bool,
}
match signup_data.validate() {
Ok(_) => (),
Err(e) => return e;
};
Here, the ContactDetails
and Preference
structs are nested within the parent SignupData
struct. Because
these child types also derive Validate
, the fields where they appear can be tagged for inclusion in the parent
struct's validation method.
Any errors found in a single nested struct (the contact_details
field in this example) would be returned as a
Struct(Box<ValidationErrors>)
type in the parent's ValidationErrors
result.
Any errors found in a vector of nested structs (the preferences
field in this example) would be returned as a
List(BTreeMap<usize, Box<ValidationErrors>>)
type in the parent's ValidationErrors
result, where the map is keyed on
the index of invalid vector entries.
You will need to import the Validate
trait, and optionally use the attr_literals
feature.
The validator
crate can also be used without the custom derive as it exposes all the
validation functions and types.
The crate comes with some built-in validators and you can have several validators for a given field.
Tests whether the String is a valid email according to the HTML5 regex, which means it will mark
some esoteric emails as invalid that won't be valid in a email
input as well.
This validator doesn't take any arguments: #[validate(email)]
.
Tests whether the String is a valid URL.
This validator doesn't take any arguments: #[validate(url)]
;
Tests whether a String or a Vec match the length requirement given. length
has 3 integer arguments:
- min
- max
- equal
Using equal
excludes the min
or max
and will result in a compilation error if they are found.
At least one argument is required with a maximum of 2 (having min
and max
at the same time).
Examples:
#[validate(length(min = 1, max = 10))]
#[validate(length(min = 1))]
#[validate(length(max = 10))]
#[validate(length(equal = 10))]
Tests whether a number is in the given range. range
takes between 1 and 2 number arguments: min
and max
.
Examples:
#[validate(range(min = 1, max = 10))]
#[validate(range(min = 1))]
#[validate(range(min = 1, max = 10.8))]
#[validate(range(min = 1.1, max = 10.8))]
#[validate(range(max = 10.8))]
Tests whether the 2 fields are equal. must_match
takes 1 string argument. It will error if the field
mentioned is missing or has a different type than the field the attribute is on.
Examples:
#[validate(must_match = "password2")]
#[validate(must_match(other = "password2"))]
Tests whether the string contains the substring given or if a key is present in a hashmap. contains
takes
1 string argument.
Examples:
#[validate(contains = "gmail")]
#[validate(contains(pattern = "gmail"))]
Tests whether the string matches the regex given. regex
takes
1 string argument: the path to a static Regex instance.
Examples:
lazy_static! {
static ref RE_TWO_CHARS: Regex = Regex::new(r"[a-z]{2}$").unwrap();
}
#[validate(regex = "RE_TWO_CHARS")]
#[validate(regex(path = "RE_TWO_CHARS"))]
Test whether the string is a valid credit card number.
Examples:
#[validate(credit_card)]
Tests whether the String is a valid phone number (in international format, ie.
containing the country indicator like +14152370800
for an US number — where 4152370800
is the national number equivalent, which is seen as invalid).
To use this validator, you must enable the phone
feature for the validator_derive
crate.
This validator doesn't take any arguments: #[validate(phone)]
;
Calls one of your functions to perform a custom validation.
The field will be given as a parameter to the function, which should return a Result<(), ValidationError>
.
Examples:
#[validate(custom = "validate_something")]
#[validate(custom = "::utils::validate_something")]
#[validate(custom(function = "validate_something"))]
Performs validation on a field with a type that also implements the Validate trait (or a vector of such types).
Examples:
#[validate]
Tests whether the String has any utf-8 control caracters, fails validation if it does.
To use this validator, you must enable the unic
feature for the validator_derive
crate.
This validator doesn't take any arguments: #[validate(non_control_character)]
;
Tests whether the Option<T>
field is Some
;
Tests whether the Option<T>
field is Some
and performs validation as nested
do;
Often, some error validation can only be applied when looking at the full struct, here's how it works here:
#[derive(Debug, Validate, Deserialize)]
#[validate(schema(function = "validate_category", skip_on_field_errors = false))]
struct CategoryData {
category: String,
name: String,
}
The function mentioned should return a Result<(), ValidationError>
and will be called after validation is done for all fields.
The skip_on_field_errors
defaults to true
if not present and will ensure that the function is not called
if an error happened while validating the struct fields.
Any error on the struct level validation will appear in the key __all__
of the hashmap of errors.
Each validator can take 2 optional arguments in addition to their own arguments:
message
: a message to go with the error, for example if you want to do i18ncode
: each validator has a default error code (for example theregex
validator code isregex
) but it can be overriden if necessary, mainly needed for thecustom
validator
Note that these arguments can't be applied to nested validation calls with #[validate]
.
For example, the following attributes all work:
// code attribute
#[validate(email(code = "code_str"))]
#[validate(credit_card(code = "code_str"))]
#[validate(length(min = 5, max = 10, code = "code_str"))]
#[validate(regex(path = "static_regex", code = "code_str"))]
#[validate(custom(function = "custom_fn", code = "code_str"))]
#[validate(contains(pattern = "pattern_str", code = "code_str"))]
#[validate(must_match(other = "match_value", code = "code_str"))]
// message attribute
#[validate(url(message = "message_str"))]
#[validate(length(min = 5, max = 10, message = "message_str"))]
#[validate(regex(path = "static_regex", message = "message_str"))]
#[validate(custom(function = "custom_fn", message = "message_str"))]
#[validate(contains(pattern = "pattern_str", message = "message_str"))]
#[validate(must_match(other = "match_value", message = "message_str"))]
// both attributes
#[validate(url(message = "message", code = "code_str"))]
#[validate(email(code = "code_str", message = "message"))]
#[validate(custom(function = "custom_fn", code = "code_str", message = "message_str"))]
- Add a blanket Validate implementation for references
- Add
Required
andRequiredNested
validators
- Add
non_control_characters
validation
ValidationErrors::errors
andValidationErrors::field_errors
now use&self
instead ofself
- Move to edition 2018
- Change error type to allow use with nested validation
- Make validators work on
Cow
- Feature gate the card validator
- Fix credit card validation being incorrect in enum
- Add international phone number and credit card validation
- Re-design
ValidationError
andValidate
trait
- Handle
Required
andRequiredNested
validators
- Update syn & quote
- Move to edition 2018
- Use literals in macros now that it's stable -> bumping minimum Rust version to 1.30
- Allow nested validation
- Make validators work on
Cow
- Update dependencies
- Feature gate the card validator
- Fix path for regex starting with
::
- Update syn and quote
- Support
Option<Option<T>>
types
- Fix path for custom validators starting with
::
- Update syn and quote
- Add international phone number and credit card derive
- Change generated code to make the new design of errors work
- Fix range validator not working on Option
- Update to serde 1.0
- Fix potential conflicts with other attributes
- Validators now work on
Option
field and struct/fields with lifetimes
- Add
contains
andregex
validator - BREAKING: change
Errors
type to be a newtype in order to extend it
- Remove need for
attr_literals
feature - Fix error when not having validation on each field
- Add struct level validation
- Add
must_match
validator