This crate contains a builder!
macro to declare a struct and a corresponding builder.
The macro is inspired from jadpole/builder-macro, and is designed to remove duplication of field declaration, as well as generating appropriate setter methods.
Include the macro inside your crate's lib.rs
or main.rs
.
```rust
extern crate builder_macro; ```
The simplest usage of the builder macro to generate a non-consuming builder is:
rust,ignore
builder!(BuilderName -> StructName {
fieldname: Type = Some(default_value), // or None if there is no sensible default
});
The above will generate a module private struct and a non-consuming builder with a single private field.
For example, given the following declaration:
rust
builder!(BuilderName -> StructName {
value: i32 = Some(1),
});
The generated builder and struct will be:
```rust struct StructName { value: i32, }
struct BuilderName {
value: Option
impl BuilderName { /// Construct the builder pub fn new() -> BuilderName { BuilderName { value: Some(1), } }
/// Build the struct
pub fn build(&self) -> StructName {
let value = self.value.clone().unwrap();
StructName{value: value,}
}
#[allow(dead_code)]
/// Specify a value for the $F_NAME field
pub fn value(&mut self, value: i32) -> &mut Self {
self.value = Some(value);
self
}
} ```
The full macro usage format is:
```rust // We declare the builder insider a module simply to demonstrate scope mod inner { builder! { /// StructName is an example struct. /// These docs are copied over to the generated struct. pub BuilderName -> StructName { /// afield is an i32 which must be between 0 and 100 inclusive // the trailing comma is mandatory due to how the macro is parsed pub afield: i32 = Some(50),
// None means no default value, a value must be specified when building
// meta attributes are copied over to the struct's fields
#[allow(dead_code)]
a_private_field: &'static str = None,
}, assertions: {
assert!(a_field >= 0);
assert!(a_field <= 100);
// Yes you can assert on private fields
assert!(!a_private_field.is_empty());
}
}
}
let mystruct = inner::BuilderName::new() .aprivatefield("I must set this to a non-empty string") .build(); asserteq!(50, mystruct.afield); ```
To generate a consuming builder, instead of using ->
, use =>
between the builder name and the target struct
name.
```rust trait Magic { fn abracadabra(&mut self) -> i32; } struct Dust { value: i32, }; impl Magic for Dust { fn abracadabra(&mut self) -> i32 { self.value } }
// Note: we use => instead of -> for the consuming variant of the builder
builder!(MyStructBuilder => MyStruct {
fieldtrait: Box
let mut my_struct = MyStructBuilder::new().build();
asserteq!(mystruct.fieldtrait.abracadabra(), 1); asserteq!(mystruct.fieldvec[0].abracadabra(), 2); ```
Generate a builder and struct with module private visibility:
```rust builder!(MyStructBuilder -> MyStruct { fieldi32: i32 = Some(123), fieldstr: &'static str = Some("abc"), });
let mystruct = MyStructBuilder::new() .fieldi32(456) .build(); asserteq!(mystruct.fieldi32, 456); asserteq!(mystruct.fieldstr, "abc"); // uses default ```
Generate a builder and struct with public visibility:
```rust mod inner { builder!(pub MyStructBuilder -> MyStruct { pub fieldi32: i32 = Some(123), fieldstr: &'static str = Some("abc"), }); }
let mystruct = inner::MyStructBuilder::new() .fieldi32(456) .build(); asserteq!(mystruct.field_i32, 456);
// The next line will fail compilation if uncommented as fieldstr is private // asserteq!(mystruct.fieldstr, "abc"); ```
The MIT License