Crates.io | vet |
lib.rs | vet |
version | 0.1.0 |
source | src |
created_at | 2022-07-21 19:31:41.841338 |
updated_at | 2022-07-21 19:31:41.841338 |
description | Arbitrary type validation |
homepage | |
repository | https://github.com/metanomial/vet |
max_upload_size | |
id | 629829 |
size | 23,971 |
Arbitrary type validation.
Add a dependency entry in Cargo.toml
.
[dependencies]
vet = "0.1"
Implement the Vet
trait on your type.
use vet::{Valid, Vet};
// A valid username consists of between 3 and 32 alphanumeric characters
#[derive(Debug)]
struct Username(String);
#[derive(Debug, PartialEq)]
enum InvalidUsername {
TooShort, // Under 3 characters
TooLong, // Over 3 characters
InvalidChar, // Contains non-alphanumeric character
}
impl Vet for Username {
type Error = InvalidUsername;
// Arbitrary logic to validate the Username type
fn is_valid(&self) -> Result<(), Self::Error> {
if self.0.len() < 3 {
return Err(Self::Error::TooShort);
}
if self.0.len() > 32 {
return Err(Self::Error::TooLong);
}
if self.0.chars().any(|c| !c.is_alphanumeric()) {
return Err(Self::Error::InvalidChar);
}
Ok(())
}
}
Valid
-wrapped types provide safety guarantees about their contents.
fn main() {
let args: Vec<String> = env::args().collect();
let username: Username = Username(args[1].clone());
let username: Result<Valid<Username>, InvalidUsername> = username.vet();
match username {
Ok(n) => create_account(n),
Err(e) => eprintln!("Invalid username: {:?}", e),
}
}
// Any `Valid<Username>` passed is guaranteed to be 3-32 alphanumeric characters.
fn create_account(username: Valid<Username>) {
let username = username.into_inner(); // Unwrap
println!("Account {:?} created", username);
}
Implementations are provided for generic arrays [T: Vet; N]
and the standard
library types Vec<T: Vet>
and Option<T: Vet>
.
Arrays and Vec
s are only valid if all of their individual elements are valid:
let usernames = vec![
Username("日向".to_string()),
Username("seán462".to_string()),
Username("lone wolf".to_string())
].vet();
// Invalid, whitespace in the third element
let contact_numbers = [
PhoneNumber("427-313-0255"),
PhoneNumber("+1 (708) 484-0523")
].vet();
// Valid, all elements passed vetting
Options containing None
are always valid:
let mut email: Option<EmailAddress> = None;
email.vet(); // Valid
let mut email: Option<EmailAddress> = Some("benjamin@@metanomial.com");
email.vet(); // Invalid, regex test failed
The default std
feature flag can be disabled to use this library in no_std
contexts. Modify your dependency entry in Cargo.toml
like so:
[dependencies]
vet = { version = "0.1", default-features = false }
In no_std contexts with a memory allocator, implementations for Vec
can be
reenabled with the alloc
feature flag:
[dependencies]
vet = { version = "0.1", default-features = false, features = ["alloc"] }
Licensed under either of
at your option.
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.