5 releases (breaking)
new 0.5.0 | Jan 16, 2025 |
---|---|
0.4.0 | Jan 14, 2025 |
0.3.1 | Jan 12, 2025 |
0.2.0 | Jan 11, 2025 |
0.1.0 | Jan 10, 2025 |
#67 in #validator
584 downloads per month
Used in valust
59KB
1.5K
SLoC
Valust-Derive
Output
The derive macro Valust
will emit both an extra structure containing raw
data and an implementation of valust::Validator
.
The raw data struct
is identical to the structure definition given to the
Valust
macro (i.e., struct A { a: A }
becomes struct RawA { a: A }
, and
struct B(B)
becomes struct RawB(B)
). The fields are automatically derived by
Valust
. Specifically, fields with only valid
retain their type, fields with
one or more trans
have the type of the first transformer
in the first
trans
, and fields with forward
are determined by the
forward
field.
The default naming pattern of the raw data struct is RawXXX
. To override it,
use the rename
struct attribute.
Syntax
Field Attributes
Name | Syntax | Description | Example |
---|---|---|---|
valid |
valid(<validator>) |
Validator, uses raw data as input. | valid(a > 10) |
trans |
trans(<transformer>) |
Transformer, converts raw data to transformed data. | trans(try(String => s.parse::<u32>())) |
forward |
forward |
Forward the field and uses it's validator to validate the field. itself. | forward |
display |
display = bool , display(bool) |
Whether to display the value when an error occurs. Enabled by default. Performance issues. |
display = false |
Structure Attributes
Name | Syntax | Description | Example |
---|---|---|---|
forward_derive |
forward_derive(Ident) |
Add derive macros for the raw data struct. | forward_derive(Debug) |
pre |
pre(<validator>) |
Pre-validator, which uses raw data as input. | pre(a > 10, b + c != 0.0) |
post |
post(<validator>) |
Post-validator, which uses validated data as input. | post(a > 10) |
rename |
rename = "Ident" , rename(Ident) |
Rename the output raw data struct. Why we need rename ? |
rename(Original) |
Special Expressions
Validator Expression
- Plain validator:
- Syntax:
valid(<expr>)
- Example:
valid(a > 10)
- Syntax:
- Validator with custom message:
- Syntax:
valid((<expr>, <msg>))
- Example:
valid((b != 0, "`b` must be non-zero"))
- Syntax:
- Fallible validator:
- Syntax:
valid(try(<expr>))
- Example:
valid(try(s.parse::<u32>() > 10))
- Syntax:
- Fallible validator with message:
- Syntax:
valid(try(<expr>, <msg>))
- Example:
valid(try(s.parse::<u32>() > 10, "`s` must be non-zero-string))
- Syntax:
Transformer Expression
- Plain transformer:
- Syntax:
trans(Type => <expr>)
- Example:
trans(String => s.trim())
- Syntax:
- Fallible transformer:
- Syntax:
trans(try(Type => <expr>))
- Example:
trans(try(String => s.parse::<u32>()))
- Syntax:
- Fallible transformer with message:
- Syntax:
trans(try(Type => <expr>, <msg>))
- Example:
trans(try(String => s.parse::<u32>(), "fail to parse number"))
- Syntax:
Example
use valust::Validate;
use valust_derive::Valust;
use valust_utils::convert::parse_to;
#[derive(Debug, Valust)]
#[forward_derive(Debug)]
pub struct Inner {
#[display = true]
#[valid(code > 10.0)]
pub code: f64,
}
#[derive(Debug, Valust)]
#[forward_derive(Debug)]
pub struct Outer {
#[forward]
pub inner: Inner,
#[trans(String => extra.trim())]
#[trans(try(String => fn(parse_to::<u32>)))]
pub extra: u32,
}
Macro Expansion
#[automatically_derived]
#[derive(Debug)]
pub struct RawInner {
pub code: f64,
}
#[automatically_derived]
impl ::valust::Validate<Inner> for RawInner {
fn validate(
self,
) -> ::std::result::Result<Inner, ::valust::error::ValidationError> {
#![allow(non_snake_case)]
let RawInner { code } = self;
let mut __valust_error_Inner = ::valust::error::ValidationError::new();
fn _valust_process_code(
code: f64,
_valust_error: &mut ::valust::error::ValidationError,
) -> ::std::option::Option<f64> {
if !(code > 10.0) {
_valust_error.push_validate_error(
::valust::error::validate::ValidateError {
field: "code",
path: format!("{}", "code"),
value: format!("(f64) {:?}", code),
cause: ::std::option::Option::None,
message: ::std::option::Option::Some(
"code must be greater than 10.0",
),
expression: "code > 10.0",
type_name: "f64",
},
);
}
::std::option::Option::Some(code)
}
let code: ::std::option::Option<f64> =
_valust_process_code(code, &mut __valust_error_Inner);
__valust_error_Inner.check()?;
let mut __valust_error_Inner = ::valust::error::ValidationError::new();
let code =
code.expect("Unexpected error occurred while processing field `code`");
__valust_error_Inner.check()?;
::std::result::Result::Ok(Inner { code })
}
}
#[automatically_derived]
#[derive(Debug)]
pub struct RawOuter {
pub inner: RawInner,
pub extra: String,
}
#[automatically_derived]
impl ::valust::Validate<Outer> for RawOuter {
fn validate(
self,
) -> ::std::result::Result<Outer, ::valust::error::ValidationError> {
#![allow(non_snake_case)]
let RawOuter { inner, extra } = self;
let mut __valust_error_Outer = ::valust::error::ValidationError::new();
fn _valust_process_inner(
inner: RawInner,
_valust_error: &mut ::valust::error::ValidationError,
) -> ::std::option::Option<Inner> {
let inner = match (<RawInner as ::valust::Validate<Inner>>::validate(inner))
{
Ok(value) => value,
Err(__valust_err_inner) => {
_valust_error.extend_error("inner", __valust_err_inner);
return None;
}
};
::std::option::Option::Some(inner)
}
let inner: ::std::option::Option<Inner> =
_valust_process_inner(inner, &mut __valust_error_Outer);
fn _valust_process_extra(
extra: String,
_valust_error: &mut ::valust::error::ValidationError,
) -> ::std::option::Option<u32> {
let extra = extra.trim();
let __format_err_clone_extra = extra.clone();
let extra = match (parse_to::<u32>(extra)) {
Ok(value) => value,
Err(__valust_err_extra) => {
_valust_error.push_transform_error(
::valust::error::transform::TransformError {
field: "extra",
path: format!("{}", "extra"),
value: format!("(String) {:?}", __format_err_clone_extra),
cause: ::std::boxed::Box::new(__valust_err_extra),
message: ::std::option::Option::None,
expression: "parse_to :: < u32 > (extra)",
source_type_name: "String",
target_type_name: "u32",
},
);
return None;
}
};
::std::option::Option::Some(extra)
}
let extra: ::std::option::Option<u32> =
_valust_process_extra(extra, &mut __valust_error_Outer);
__valust_error_Outer.check()?;
let mut __valust_error_Outer = ::valust::error::ValidationError::new();
let inner =
inner.expect("Unexpected error occurred while processing field `inner`");
let extra =
extra.expect("Unexpected error occurred while processing field `extra`");
__valust_error_Outer.check()?;
::std::result::Result::Ok(Outer { inner, extra })
}
}
Appendix
Forward
ing a field
Fields that implement valust::Validator
are not automatically recognized by
Valust, but Valust can leverage pre-defined Validator implementations.
Specifically, by using the forward
field attribute, Valust
will execute the
valust::Validator::validate
method of the field's type, automatically
extending the error path
field. Additionally, it will automatically change the
corresponding field of the raw data struct to the original data type of that
field.
The raw data type could be inferred by the compiler, so you don't need to
specify it even if you've rename
d it.
Why we need rename
Though we don't need to specify the raw data type when executing the
validator, we might need to construct them directly. Sadly, due to rust's
language syntax limitations, we cannot construct an unnamed struct using type
alias (i.e. valust::Raw::<Foo>
). That's why we may need rename
a raw
type.
Performance issues when displaying error messages
Displaying huge data may lead to performance issues, as the internal
formatter will clone
the data for fear that user-defined expressions might
take the field by-value instead of by-ref.
Dependencies
~260–720KB
~17K SLoC