Crate nutype

source ·
Expand description

Rust Nutype Logo

The newtype with guarantees.

Philosphy

Nutype embraces the simple idea: the type system can be leveraged to track the fact that something was done, so there is no need to do it again.

If a piece of data was once sanitized and validated we can rely on the types instead of sanitizing and validating again and again.

Quick start

use nutype::nutype;

#[nutype(
    sanitize(trim, lowercase)
    validate(present, max_len = 20)
)]
pub struct Username(String);

Now we can create usernames:

assert_eq!(
    Username::new("   FooBar  ").unwrap().into_inner(),
    "foobar"
);

But we cannot create invalid ones:

assert_eq!(
    Username::new("   "),
    Err(UsernameError::Missing),
);

assert_eq!(
    Username::new("TheUserNameIsVeryVeryLong"),
    Err(UsernameError::TooLong),
);

Note, that we also explicitly got UsernameError enum generated.

Ok, but let’s try to obtain an instance of Username that violates the validation rules:

let username = Username("".to_string())

// error[E0423]: cannot initialize a tuple struct which contains private fields
let mut username = Username::new("foo").unwrap();
username.0 = "".to_string();

// error[E0616]: field `0` of struct `Username` is private

Haha. It’s does not seem to be easy!

A few more examples

Here are some other examples of what you can do with nutype.

You can skip sanitize and use a custom validator with:

#[nutype(validate(with = |n| n % 2 == 1))]
struct OddNumber(i64);

You can skip validation, if you need sanitization only:

#[nutype(sanitize(trim, lowercase))]
struct Username(String);

In that case Username::new(String) simply returns Username, not Result.

You can derive traits. A lot of traits! For example:

#[nutype]
#[derive(*)]
struct Username(String);

The code above derives the following traits for Username: Debug, Clone, PartialEq, Eq, PartialOrd, Ord, FromStr, AsRef, Hash. * is just a syntax sugar for “derive whatever makes sense to derive by default”, which is very subjective and opinionated. It’s rather an experimental feature that was born from the fact that #[nutype] has to mess with #[derive] anyway, because users are not supposed to be able to derive traits like DerefMut or BorrowMut. That would allow to mutate the inner (protected) value which undermines the entire idea of nutype.

Inner types

Available sanitizers, validators and derivable traits are determined by the inner type, which falls into the following categories:

  • String
  • Integer (u8, u16,u32, u64, u128, i8, i16, i32, i64, i128, usize, isize)
  • Float (f32, f64)

String

At the moment the string inner type supports only String (owned) type.

String sanitizers

SanitizerDescriptionExample
trimRemoves leading and trailing whitespacestrim
lowercaseConverts the string to lowercaselowercase
uppercaseConverts the string to uppercaseuppercase
withCustom sanitizer. A function or closure that receives String and returns Stringwith = \|mut s: String\| { s.truncate(5); s }

String validators

ValidatorDescriptionError variantExample
max_lenMax length of the stringTooLongmax_len = 255
min_lenMin length of the stringTooShortmin_len = 5
presentRejects an empty stringMissingpresent
withCustom validator. A function or closure that receives &str and returns boolInvalidwith = \|s: &str\| s.contains('@')

String derivable traits

The following traits can be derived for a string-based type: Debug, Clone, PartialEq, Eq, PartialOrd, Ord, FromStr, AsRef, From, TryFrom, Into, Hash, Borrow, Display, Serialize, Deserialize.

Integer

The integer inner types are: u8, u16,u32, u64, u128, i8, i16, i32, i64, i128, usize, isize.

Integer sanitizers

SanitizerDescriptionExample
withCustom sanitizer.with = \|raw\| raw.clamp(0, 100)

Integer validators

ValidatorDescriptionError variantExample
maxMaximum valid valueTooBigmax = 99
minMinimum valid valueTooSmallmin = 18
withCustom validatorInvalidwith = \|num\| num % 2 == 0

Integer derivable traits

The following traits can be derived for an integer-based type: Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, FromStr, AsRef, Into, From, TryFrom, Hash, Borrow, Display, Serialize, Deserialize.

Float

The float inner types are: f32, f64.

Float sanitizers

SanitizerDescriptionExample
withCustom sanitizer.with = \|val\| val.clamp(0.0, 100.0)

Float validators

ValidatorDescriptionError variantExample
maxMaximum valid valueTooBigmax = 100.0
minMinimum valid valueTooSmallmin = 0.0
withCustom validatorInvalidwith = \|val\| val != 50.0

Float derivable traits

The following traits can be derived for a float-based type: Debug, Clone, Copy, PartialEq, PartialOrd, FromStr, AsRef, Into, From, TryFrom, Hash, Borrow, Display, Serialize, Deserialize.

Custom sanitizers

You can set custom sanitizers using option with. A custom sanitizer is a function or closure that receives a value of an inner type with ownership and returns a sanitized value back.

For example, this one

#[nutype(sanitize(with = new_to_old))]
pub struct CityName(String);

fn new_to_old(s: String) -> String {
    s.replace("New", "Old")
}

is equal to the following one:

#[nutype(sanitize(with = |s| s.replace("New", "Old") ))]
pub struct CityName(String);

And works the same way:

let city = CityName::new("New York");
assert_eq!(city.into_inner(), "Old York");

Custom validators

In similar fashion it’s possible to define custom validators, but a validation function receives a reference and returns bool. Think of it as a predicate.

#[nutype(validate(with = is_valid_name))]
pub struct Name(String);

fn is_valid_name(name: &str) -> bool {
    // A fancy way to verify if the first character is uppercase
    name.chars().next().map(char::is_uppercase).unwrap_or(false)
}

Feature flags

  • serde1 - integrations with serde crate. Allows to derive Serialize and Deserialize traits.

Support Ukrainian military forces 🇺🇦

Today I live in Berlin, I have a luxury to live a physically safe life. But I am Ukrainian. The first 25 years of my life I spent in Kharkiv, the second-largest city in Ukraine, 60km away from the border with russia. Today about a third of my home city is destroyed by russians. My parents, my relatives and my friends had to survive the artillery and air attack, living for over a month in basements.

Some of them have managed to evacuate to EU. Some others are trying to live “normal lifes” in Kharkiv, doing there daily duties. And there are some who are at the front line right now, risking their lives every second to protect the rest.

I encourage you to donate to Charity foundation of Serhiy Prytula. Just pick the project you like and donate. This is one of the best known foundations, you can watch a little documentary about it. Your contribution to the Ukrainian military force is a contribution to my calmness, so I can spend more time developing the project.

Thank you.

Attribute Macros