A derivable macro for declaring a builder pattern.
```rust use builder_pattern::Builder;
struct Person { #[into] name: String, age: i32, #[default(Gender::Nonbinary)] gender: Gender, }
let p1 = Person::new() .name(String::from("Joe")) .age(27) .build();
// Orders does not matter.
let p2 = Person::new()
.age(32)
// &str
is implicitly converted into String
// because of into
attribute!
.name("Jack")
.gender(Gender::Male).build();
// name
field required - Compilation error.
let p3 = Person::new()
.age(15)
.build();
```
Add builder-pattern
to Cargo.toml
.
```toml
[dependencies] builder-pattern = "0.3" ```
build
function is allowed only the all of required fields are provided. No Result, No Unwrap. Just use it.#[default(expr)]
A field having this attribute will be considered as optional and the expr
will be evaluated as a default value of the field. build
function can be called without providing this field.
#[into]
A setter function for a field having this attribute will accept an Into
trait as a parameter. You can use this setter with implicit conversion.
Example:
```rust
struct Test { #[into] pub name: String, }
let test = Test::new()
// &str
is implicitly converted into String
.
.name("Hello")
.build();
```
#[validator(expr)]
Implement a validator for a field. expr
could be a validating function that takes the field's type and returns Result
.
```rust
struct Test { #[validator(isnotempty)] #[into] pub name: String, }
fn isnotempty(name: String) -> Result let test1 = Test::new().name(""); // Err(())
let test2 = Test::new().name("Hello").unwrap().build();
``` This crate generates documentations for the builder functions. If you documentate the fields,
the builder functions for them also copy the documentations. Example code: ```rust struct Test {
/// A positive integer.
pub positive: i32, }
``` Generated code: `` /// A builder for impl TestBuilder impl impl The following code ```rust struct Person {
#[into]
#[validator(isnotempty)]
name: String,
age: i32,
#[default(Gender::Nonbinary)]
gender: Gender,
} ``` will generates: ```rust
struct PersonBuilder impl Person {
// Create an empty builder
fn new() -> PersonBuilder<(), (), ()> {
PersonBuilder {
name: None,
age: None,
// Default value
gender: Some(Gender::Nonbinary),
_phantom: PhantomData
}
}
} // Builder for // Builder for // Builder for // Auto-Generated Documentions
Example
[derive(Builder)]
/// A integer having zero as a default value.
#[default(0)]
pub zero: i32,
rust
impl Test {
/// Creating a builder.
/// ## Required fields
/// ###
positive
/// - Type:
i32
///
/// A positive integer.
///
/// ## Optional fields
/// ###
zero
/// - Type:
i32
/// - Default:
0`
///
/// A integer having zero as a default value.
fn new() -> TestBuilder<(), ()> {
TestBuilder {
_phatom: PhantomData,
positive: None,
zero: Some(0),
}
}
}Test
.
struct TestBuilderi32
///
/// A positive integer.
pub fn positive(self, value: i32) -> TestBuilder0
///
/// A integer having zero as a default value.
pub fn zero(self, value: i32) -> TestBuilderHow it works
[derive(Builder)]
name
.
implInto
traits.
fn nameIntoType
into String
.
name: Some(value.into()),
age: self.age,
gender: self.gender,
phantom: PhantomData,
}),
Err() => Err(())
}
}
}age
.
implgender
.
implbuild
function
// It can be called regardless of whether T3
is ()
or Gender
.
impl