1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359
|
# derive-where
[](https://crates.io/crates/derive-where)
[](https://github.com/ModProg/derive-where/actions/workflows/test.yml)
[](https://docs.rs/crate/derive-where)
## Description
Attribute proc-macro to simplify deriving standard and other traits with
custom generic type bounds.
## Usage
The [`derive_where`] attribute can be used just like
std's `#[derive(...)]` statements:
```rust
#[derive_where(Clone, Debug)]
struct Example<T>(PhantomData<T>);
```
This will generate trait implementations for `Example` for any `T`,
as opposed to std's derives, which would only implement these traits with
`T: Trait` bound to the corresponding trait.
Multiple [`derive_where`] attributes can be added to an
item, but only the first one must use any path qualifications.
```rust
#[derive_where::derive_where(Clone, Debug)]
#[derive_where(Eq, PartialEq)]
struct Example1<T>(PhantomData<T>);
```
If using a different package name, you must specify this:
```rust
#[derive_where(crate = derive_where_)]
#[derive_where(Clone, Debug)]
struct Example<T>(PhantomData<T>);
```
In addition, the following convenience options are available:
### Generic type bounds
Separated from the list of traits with a semi-colon, types to bind to can be
specified. This example will restrict the implementation for `Example` to
`T: Clone`:
```rust
#[derive_where(Clone, Debug; T)]
struct Example<T, U>(T, PhantomData<U>);
```
It is also possible to specify the bounds to be applied. This will
bind implementation for `Example` to `T: Super`:
```rust
trait Super: Clone + Debug {}
#[derive_where(Clone, Debug; T: Super)]
struct Example<T>(PhantomData<T>);
```
But more complex trait bounds are possible as well.
The example below will restrict the [`Clone`] implementation for `Example`
to `T::Type: Clone`:
```rust
trait Trait {
type Type;
}
struct Impl;
impl Trait for Impl {
type Type = i32;
}
#[derive_where(Clone, Debug; T::Type)]
struct Example<T: Trait>(T::Type);
```
Any combination of options listed here can be used to satisfy a
specific constrain. It is also possible to use multiple separate
constrain specifications when required:
```rust
#[derive_where(Clone, Debug; T)]
#[derive_where(Eq, PartialEq; U)]
struct Example<T, U>(PhantomData<T>, PhantomData<U>);
```
### Enum default
Since Rust 1.62 deriving [`Default`] on an enum is possible with the
`#[default]` attribute. Derive-where allows this with a
`#[derive_where(default)]` attribute:
```rust
#[derive_where(Clone, Default)]
enum Example<T> {
#[derive_where(default)]
A(PhantomData<T>),
}
```
### Skipping fields
With a `skip` or `skip_inner` attribute fields can be skipped for traits
that allow it, which are: [`Debug`], [`Hash`], [`Ord`], [`PartialOrd`],
[`PartialEq`], [`Zeroize`] and [`ZeroizeOnDrop`].
```rust
#[derive_where(Debug, PartialEq; T)]
struct Example<T>(#[derive_where(skip)] T);
assert_eq!(format!("{:?}", Example(42)), "Example");
assert_eq!(Example(42), Example(0));
```
It is also possible to skip all fields in an item or variant if desired:
```rust
#[derive_where(Debug, PartialEq)]
#[derive_where(skip_inner)]
struct StructExample<T>(T);
assert_eq!(format!("{:?}", StructExample(42)), "StructExample");
assert_eq!(StructExample(42), StructExample(0));
#[derive_where(Debug, PartialEq)]
enum EnumExample<T> {
#[derive_where(skip_inner)]
A(T),
}
assert_eq!(format!("{:?}", EnumExample::A(42)), "A");
assert_eq!(EnumExample::A(42), EnumExample::A(0));
```
Selective skipping of fields for certain traits is also an option, both in
`skip` and `skip_inner`. To prevent breaking invariants defined for these
traits, some of them can only be skipped in groups. The following groups are
available:
- [`Debug`]
- `EqHashOrd`: Skips [`Eq`], [`Hash`], [`Ord`], [`PartialOrd`] and
[`PartialEq`].
- [`Hash`]
- `Zeroize`: Skips [`Zeroize`] and [`ZeroizeOnDrop`].
```rust
#[derive_where(Debug, PartialEq)]
#[derive_where(skip_inner(Debug))]
struct Example<T>(i32, PhantomData<T>);
assert_eq!(format!("{:?}", Example(42, PhantomData::<()>)), "Example");
assert_ne!(
Example(42, PhantomData::<()>),
Example(0, PhantomData::<()>)
);
```
### Incomparable variants/items
Similar to the `skip` attribute, `incomparable` can be used to skip variants
or items in [`PartialEq`] and [`PartialOrd`] trait implementations, meaning
they will always yield `false` for `eq` and `None` for `partial_cmp`. This
results in all comparisons but `!=`, i.e. `==`, `<`, `<=`, `>=` and `>`,
with the marked variant or struct evaluating to `false`.
```rust
# use derive_where::derive_where;
#[derive(Debug)]
#[derive_where(PartialEq, PartialOrd)]
enum EnumExample {
#[derive_where(incomparable)]
Incomparable,
Comparable,
}
assert_eq!(EnumExample::Comparable, EnumExample::Comparable);
assert_ne!(EnumExample::Incomparable, EnumExample::Incomparable);
assert!(!(EnumExample::Comparable >= EnumExample::Incomparable));
assert!(!(EnumExample::Comparable <= EnumExample::Incomparable));
assert!(!(EnumExample::Incomparable >= EnumExample::Incomparable));
assert!(!(EnumExample::Incomparable <= EnumExample::Incomparable));
#[derive(Debug)]
#[derive_where(PartialEq, PartialOrd)]
#[derive_where(incomparable)]
struct StructExample;
assert_ne!(StructExample, StructExample);
assert!(!(StructExample >= StructExample));
assert!(!(StructExample <= StructExample));
```
Note that it is not possible to use `incomparable` with [`Eq`] or [`Ord`] as
that would break their invariants.
### `Zeroize` options
`Zeroize` has two options:
- `crate`: an item-level option which specifies a path to the [`zeroize`]
crate in case of a re-export or rename.
- `fqs`: a field-level option which will use fully-qualified-syntax instead
of calling the [`zeroize`][method@zeroize] method on `self` directly. This
is to avoid ambiguity between another method also called `zeroize`.
```rust
#[derive_where(Zeroize(crate = zeroize_))]
struct Example(#[derive_where(Zeroize(fqs))] i32);
impl Example {
// If we didn't specify the `fqs` option, this would lead to a compile
// error because of method ambiguity.
fn zeroize(&mut self) {
self.0 = 1;
}
}
let mut test = Example(42);
// Will call the struct method.
test.zeroize();
assert_eq!(test.0, 1);
// WIll call the `Zeroize::zeroize` method.
Zeroize::zeroize(&mut test);
assert_eq!(test.0, 0);
```
### `ZeroizeOnDrop` options
If the `zeroize-on-drop` feature is enabled, it implements [`ZeroizeOnDrop`]
and can be implemented without [`Zeroize`], otherwise it only implements
[`Drop`] and requires [`Zeroize`] to be implemented.
[`ZeroizeOnDrop`] has one option:
- `crate`: an item-level option which specifies a path to the [`zeroize`]
crate in case of a re-export or rename.
```rust
#[derive_where(ZeroizeOnDrop(crate = zeroize_))]
struct Example(i32);
assert!(core::mem::needs_drop::<Example>());
```
### Supported traits
The following traits can be derived with derive-where:
- [`Clone`]
- [`Copy`]
- [`Debug`]
- [`Default`]
- [`Eq`]
- [`Hash`]
- [`Ord`]
- [`PartialEq`]
- [`PartialOrd`]
- [`Zeroize`]: Only available with the `zeroize` crate feature.
- [`ZeroizeOnDrop`]: Only available with the `zeroize` crate feature. If the
`zeroize-on-drop` feature is enabled, it implements [`ZeroizeOnDrop`],
otherwise it only implements [`Drop`].
### Supported items
Structs, tuple structs, unions and enums are supported. Derive-where tries
it's best to discourage usage that could be covered by std's `derive`. For
example unit structs and enums only containing unit variants aren't
supported.
Unions only support [`Clone`] and [`Copy`].
[`PartialOrd`] and [`Ord`] need to determine the discriminant type to
function correctly. To protect against a potential future change to the
default discriminant type, some compile-time validation is inserted to
ascertain that the type remains `isize`.
### `no_std` support
`no_std` support is provided by default.
## Crate features
- `nightly`: Implements [`Ord`] and [`PartialOrd`] with the help of
[`core::intrinsics::discriminant_value`], which is what Rust does by
default too. This requires a nightly version of the Rust compiler.
- `safe`: `safe`: Uses only safe ways to access the discriminant of the enum
for [`Ord`] and [`PartialOrd`]. It also replaces all cases of
[`core::hint::unreachable_unchecked`] in [`Ord`], [`PartialEq`] and
[`PartialOrd`], which is what std uses, with [`unreachable`].
- `zeroize`: Allows deriving [`Zeroize`] and [`zeroize`][method@zeroize] on
[`Drop`].
- `zeroize-on-drop`: Allows deriving [`Zeroize`] and [`ZeroizeOnDrop`] and
requires [`zeroize`] v1.5.
## MSRV
The current MSRV is 1.57 and is being checked by the CI. A change will be
accompanied by a minor version bump. If MSRV is important to you, use
`derive-where = "~1.x"` to pin a specific minor version to your crate.
## Alternatives
- [derivative](https://crates.io/crates/derivative) [](https://crates.io/crates/derivative)
is a great alternative with many options. Notably it doesn't support
`no_std` and requires an extra `#[derive(Derivative)]` to use.
- [derive_bounded](https://crates.io/crates/derive_bounded) [](https://crates.io/crates/derive_bounded)
is a new alternative still in development.
## Changelog
See the [CHANGELOG] file for details.
## License
Licensed under either of
- Apache License, Version 2.0 ([LICENSE-APACHE] or <http://www.apache.org/licenses/LICENSE-2.0>)
- MIT license ([LICENSE-MIT] or <http://opensource.org/licenses/MIT>)
at your option.
### Contribution
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.
[CHANGELOG]: https://github.com/ModProg/derive-where/blob/main/CHANGELOG.md
[LICENSE-MIT]: https://github.com/ModProg/derive-where/blob/main/LICENSE-MIT
[LICENSE-APACHE]: https://github.com/ModProg/derive-where/blob/main/LICENSE-APACHE
[`Debug`]: https://doc.rust-lang.org/core/fmt/trait.Debug.html
[`Default`]: https://doc.rust-lang.org/core/default/trait.Default.html
[`Hash`]: https://doc.rust-lang.org/core/hash/trait.Hash.html
[`zeroize`]: https://docs.rs/zeroize
[`Zeroize`]: https://docs.rs/zeroize/latest/zeroize/trait.Zeroize.html
[`ZeroizeOnDrop`]: https://docs.rs/zeroize/latest/zeroize/trait.ZeroizeOnDrop.html
[method@zeroize]: https://docs.rs/zeroize/latest/zeroize/trait.Zeroize.html#tymethod.zeroize
[`Clone`]: https://doc.rust-lang.org/core/clone/trait.Clone.html
[`Copy`]: https://doc.rust-lang.org/core/marker/trait.Copy.html
[`core::hint::unreachable_unchecked`]: https://doc.rust-lang.org/core/hint/fn.unreachable_unchecked.html
[`core::intrinsics::discriminant_value`]: https://doc.rust-lang.org/core/intrinsics/fn.discriminant_value.html
[`derive_where`]: https://docs.rs/derive-where/latest/derive_where/attr.derive_where.html
[`Discriminant`]: https://doc.rust-lang.org/core/mem/struct.Discriminant.html
[`Drop`]: https://doc.rust-lang.org/core/ops/trait.Drop.html
[`Eq`]: https://doc.rust-lang.org/core/cmp/trait.Eq.html
[`i32`]: https://doc.rust-lang.org/core/primitive.i32.html
[`isize`]: https://doc.rust-lang.org/core/primitive.isize.html
[`Ord`]: https://doc.rust-lang.org/core/cmp/trait.Ord.html
[`PartialEq`]: https://doc.rust-lang.org/core/cmp/trait.PartialEq.html
[`PartialOrd`]: https://doc.rust-lang.org/core/cmp/trait.PartialOrd.html
[`transmute`]: https://doc.rust-lang.org/core/mem/fn.transmute.html
[`unreachable`]: https://doc.rust-lang.org/core/macro.unreachable.html
|