#constructor-macro #proc-macro #macro-derive #derive #macro #procedural #traits

macro no-std derive-ctor

Adds #[derive(ctor)] which allows for the auto-generation of struct, enum, and union constructors

13 releases (7 stable)

1.0.6 Jun 11, 2024
1.0.5 Jun 9, 2024
1.0.0 May 31, 2024
0.2.3 May 30, 2024
0.1.1 May 28, 2024

#456 in Rust patterns

MIT license

52KB
988 lines

derive-ctor

derive-ctor is a Rust procedural macro crate that allows you to easily generate constructor methods for your structs. With the #[derive(ctor)] attribute, you can automatically create a constructor(s) for structs, enums, and unions. The crate also provides various options to customize the generated constructor methods.

Features

  • Automatically generate a constructor method for structs, enums, and unions with #[derive(ctor)].
  • Customize the name and visibility of the auto-generated constructor using #[ctor(visibility method_name)].
    • Supports const constructors by adding the "const" keyword.
    • Provide a list of names to generate multiple constructors.
  • Customize field behavior in the constructor with the following properties (used in #[ctor(PROPETY)]):
    • cloned - Changes the parameter type to accept a reference type which is then cloned into the created struct.
    • default - Exclude the field from the generated method and use its default value.
    • expr(EXPRESSION) - Exclude the field from the generated method and use the defined expression as its default value.
      • expr!(EXPRESSION) to add the annotated field as a required parameter, allowing the expression to reference itself.
      • Use expr(TYPE -> EXPRESSION) to add a parameter with the specified type, which will be used to generate the final field value.
    • into - Change the parameter type for the generated method to impl Into<Type>.
    • iter(FROM_TYPE) - Change the parameter type for the generated method to impl IntoIterator<Item=FROM_TYPE>.
  • No reliance on the standard library (no-std out of the box).
  • Usability with structs, enums, and unions are toggleable as features (all are enabled by default)

Basic Usage

Add derive-ctor to your Cargo.toml:

[dependencies]
derive-ctor = "1.0.6"

Annotate your struct with #[derive(ctor)] to automatically generate a new constructor:

use derive_ctor::ctor;

#[derive(ctor)]
struct MyStruct {
    field1: i32,
    field2: String
}

let my_struct = MyStruct::new(1, String::from("Foo"));

Struct Configurations

Visibility and Constructor Name

You can modify the name and visibility of the generated method, and define additional constructors by using the #[ctor] attribute on the target struct after ctor is derived.

In the following example, three constructor methods are created: new, with_defaults, and internal. These methods all inherit their respective visibilities defined within the #[ctor] attribute.

use derive_ctor::ctor;

#[derive(ctor)]
#[ctor(pub new, pub(crate) other, const internal)]
struct MyStruct {
    field1: i32,
    field2: String
}

let my_struct1 = MyStruct::new(100, "A".to_string());
let my_struct2 = MyStruct::other(200, "B".to_string());
let my_struct3 = MyStruct::internal(300, "C".to_string());

Auto-implement "Default" Trait

The Default trait can be auto implemented by specifying a ctor with the name default in the ctor attribute. Note: all fields must have a generated value in order for the implementation to be valid. Additionally, declaring default(all) will automatically mark all non-annotated fields with #[ctor(default)]

use derive_ctor::ctor;

#[derive(ctor)]
#[ctor(default)]
struct MyStruct {
    #[ctor(default)]
    field1: i32,
    #[ctor(expr(true))]
    field2: bool
}

let default: MyStruct = Default::default();


#[derive(ctor)]
#[ctor(default(all))]
struct OtherStruct {
    field1: i32,
    #[ctor(expr(true))]
    field2: bool
}

let default2: OtherStruct = Default::default();

Constructor Properties

Custom constructor definitions can also take one of the following properties to implement on all non-configured fields

  • default - Marks all non-annotated fields as #[ctor(default)]
  • into - Marks all non-annotated fields as #[ctor(into)]
use derive_ctor::ctor;

#[derive(ctor)]
#[ctor(new(into), with_defaults(default))]
struct MyStruct {
    name: String,
    #[ctor(expr(13))]
    value: i32
}

let example1 = MyStruct::new("FooBar");
let example2 = MyStruct::with_defaults();

Enum and Union Configurations

By default, a constructor will be generated for each variant. This constructor by default will match the name of its respective variant and will be public. This default behaviour can be changed by annotating the enum with #[ctor(prefix = PREFIX, visibility = VISIBILITY)]. Note that both parameters are optional within the attribute. Specifying this attribute will change the default generated method for each variant, however, each variant can additionally define its own configuration which overrides the one defined by the enum.

Standard variant constructor example

use derive_ctor::ctor;

#[derive(ctor)]
enum MyEnum {
    Variant1,
    Variant2(i32),
    Variant3 { value: bool }
}

let v1 = MyEnum::variant1();
let v2 = MyEnum::variant2(100);
let v3 = MyEnum::variant3(true);

Configured variant constructor example

Variant constructor configuration is identical to struct constructor configuration. Refer to the below for sample syntax or go-back to the struct constructor configuration for more information.

use derive_ctor::ctor;

#[derive(ctor)]
#[ctor(prefix = new, vis = pub(crate))]
enum MyEnum {
    #[ctor(const pub v1, other)]
    Variant1,
    Variant2,
    Variant3
}

const v1_1: MyEnum = MyEnum::v1();
let v1_2 = MyEnum::other();
let v2 = MyEnum::new_variant2();
let v3 = MyEnum::new_variant3();

If a variant is derived with #[ctor(none)] it will not have a constructor generated for it. Similarly, if a variant is derived with #[ctor(default)] the Default trait will be automatically implemented to generate that variant.

Unions express the same behaviours as enums except applicable to the fields of the union rather than the variants of an enum.

use derive_ctor::ctor;

#[derive(ctor)]
#[ctor(prefix = new, vis = pub(crate))]
union MyUnion {
    #[ctor(const new)]
    v1: i32,
    v2: f32,
    v3: u32
}

const VAL: MyUnion = MyUnion::new(100);
let v2 = MyUnion::new_v2(123.231);
let v3 = MyUnion::new_v3(414224); 

Field Configurations

Fields can also be annotated with #[ctor(PROPERTY)] to change their behaviour in the generated methods. These configurations work for ALL enum-types and structs! The following are the available properties that can be used with the field-attributes

#[ctor(cloned)] - This property creates a parameter that accepts a type reference of the annotated field and then clones it to generate the final value.

use derive_ctor::ctor;

#[derive(ctor)]
struct MyStruct {
    field1: i32,
    #[ctor(cloned)]
    field2: String
}

let string = String::from("Foo");
let my_struct = MyStruct::new(100, &string);

#[ctor(default)] - This property excludes the annotated field from the constructor and uses its default value.

use derive_ctor::ctor;

#[derive(ctor)]
struct MyStruct {
    field1: i32,
    #[ctor(default)]
    field2: String
}

let my_struct = MyStruct::new(100);

#[ctor(into)] - This property modifies the parameter type of the annotated field for the generated method converting it from Type -> impl Into<Type>.

use derive_ctor::ctor;

#[derive(ctor)]
struct MyStruct {
    field1: i32,
    #[ctor(into)] // the parameter type will now be impl Into<String> instead of String
    field2: String
}

let my_struct = MyStruct::new(100, "Foo");

#[ctor(expr(EXPRESSION))] - This property excludes the annotated field from the constructor and utilizes the defined expression to generate its value.

Alternatives:

  • #[ctor(expr!(EXPRESSION))] - Unlike the above attribute, this attribute will add the annotated field as a required parameter for the given constructor, this allows for the provided EXPRESSION to reference the parameter and modify the passed value.
  • #[ctor(expr(TYPE -> EXPRESSION))] - This attribute behaves similar to the variation above, however, the required parameter type will be of the type provided in the attribute, thus allowing for a constructor to accept and map a parameter from one type to the type used by the struct field.
use derive_ctor::ctor;

#[derive(ctor)]
struct MyStruct {
    field1: i32,
    #[ctor(expr(String::from("Foo")))]
    field2: String,
    #[ctor(expr!(field3 + 100))]
    field3: u32,
    #[ctor(expr(i32 -> field4 < 0))]
    field4: bool
}

let my_struct = MyStruct::new(100, 5, -20); // generates MyStruct { field1: 100, field2: "foo", field3: 105, field4: true }

#[ctor(iter(TYPE))] - This property adds a parameter with the type: impl IntoIterator<Item=TYPE> and then generates the annotated struct value by calling .into_iter().collect() on the parameter value.

use std::collections::HashSet;
use derive_ctor::ctor;

#[derive(ctor)]
struct MyStruct {
    field1: i32,
    #[ctor(iter(usize))]
    field2: HashSet<usize>
}

let my_struct = MyStruct::new(0, vec![1, 1, 2, 3, 4]);

Advanced Configuration

Field attributes can additionally be configured with a list of indices corresponding to the methods to use the generated value for. This allows for the creation of multiple functions with different parameter requirements.

use derive_ctor::ctor;

#[derive(ctor)]
#[ctor(new, with_defaults)]
struct MyStruct {
    field1: i32,
    #[ctor(default = [1])]
    field2: String,
    #[ctor(default = 1)] // brackets can be removed if specifying only 1 index
    field3: bool,
    #[ctor(default = [0, 1])]
    field4: u64,
    #[ctor(default)] // this is the same as specifying all indices
    field5: u64
}

let my_struct1 = MyStruct::new(100, "Foo".to_string(), true);
let my_struct2 = MyStruct::with_defaults(100);

Non-Default Features

shorthand - Allows the usage of "shorthand" attributes on fields. For example, instead of #[ctor(expr(EXPRESSION)] you can use #[expr(EXPRESSION)] instead.

use derive_ctor::ctor;

#[derive(ctor)]
struct MyStruct {
    #[expr(100)]
    value: u32
}

let my_struct = MyStruct::new();

Dependencies

~250–700KB
~17K SLoC