educe/
lib.rs

1/*!
2# Educe
3
4This crate offers procedural macros designed to facilitate the swift implementation of Rust's built-in traits.
5
6## Features
7
8By default, every trait this crate supports will be enabled. You can disable all of them by turning off the default features and enable only the traits that you want to use by adding them to the `features` explicitly.
9
10For example,
11
12```toml
13[dependencies.educe]
14version = "*"
15features = ["Debug", "Clone", "Copy", "Hash", "Default"]
16default-features = false
17```
18
19## Traits
20
21#### Debug
22
23Use `#[derive(Educe)]` and `#[educe(Debug)]` to implement the `Debug` trait for a struct, enum, or union. This allows you to modify the names of your types, variants, and fields. You can also choose to ignore specific fields or set a method to replace the `Debug` trait. Additionally, you have the option to format a struct as a tuple and vice versa.
24
25###### Basic Usage
26
27```rust
28# #[cfg(feature = "Debug")]
29# {
30use educe::Educe;
31
32#[derive(Educe)]
33#[educe(Debug)]
34struct Struct {
35    f1: u8
36}
37
38#[derive(Educe)]
39#[educe(Debug)]
40enum Enum {
41    V1,
42    V2 {
43        f1: u8,
44    },
45    V3(u8),
46}
47# }
48```
49
50###### Change the Name of a Type, a Variant or a Field
51
52The `name` parameter can rename a type, a variant or a field. If you set it to `false`, the name can be ignored or forced to show otherwise.
53
54```rust
55# #[cfg(feature = "Debug")]
56# {
57use educe::Educe;
58
59#[derive(Educe)]
60#[educe(Debug(name(Struct2)))]
61struct Struct {
62    #[educe(Debug(name(f)))]
63    f1: u8
64}
65
66#[derive(Educe)]
67#[educe(Debug(name = true))]
68enum Enum {
69    #[educe(Debug(name = false))]
70    V1,
71    #[educe(Debug(name(V)))]
72    V2 {
73        #[educe(Debug(name(f)))]
74        f1: u8,
75    },
76    #[educe(Debug(name = false))]
77    V3(u8),
78}
79# }
80```
81
82###### Ignore Fields
83
84The `ignore` parameter can ignore a specific field.
85
86```rust
87# #[cfg(feature = "Debug")]
88# {
89use educe::Educe;
90
91#[derive(Educe)]
92#[educe(Debug)]
93struct Struct {
94    #[educe(Debug(ignore))]
95    f1: u8
96}
97
98#[derive(Educe)]
99#[educe(Debug)]
100enum Enum {
101    V1,
102    V2 {
103        #[educe(Debug(ignore))]
104        f1: u8,
105    },
106    V3(
107        #[educe(Debug(ignore))]
108        u8
109    ),
110}
111# }
112```
113
114###### Fake Structs and Tuples
115
116With the `named_field` parameter, structs can be formatted as tuples and tuples can be formatted as structs.
117
118```rust
119# #[cfg(feature = "Debug")]
120# {
121use educe::Educe;
122
123#[derive(Educe)]
124#[educe(Debug(named_field = false))]
125struct Struct {
126    f1: u8
127}
128
129#[derive(Educe)]
130#[educe(Debug)]
131enum Enum {
132    V1,
133    #[educe(Debug(named_field = false))]
134    V2 {
135        f1: u8,
136    },
137    #[educe(Debug(named_field = true))]
138    V3(
139        u8,
140        #[educe(Debug(name(value)))]
141        i32
142    ),
143}
144# }
145```
146
147###### Use Another Method to Handle the Formatting
148
149The `method` parameter can be utilized to replace the implementation of the `Debug` trait for a field, eliminating the need to implement the `Debug` trait for the type of that field.
150
151```rust
152# #[cfg(feature = "Debug")]
153# {
154use educe::Educe;
155
156use std::fmt::{self, Formatter};
157
158fn fmt<T>(_s: &T, f: &mut Formatter<'_>) -> fmt::Result {
159    f.write_str("Hi")
160}
161
162#[derive(Educe)]
163#[educe(Debug)]
164enum Enum<T> {
165    V1,
166    V2 {
167        #[educe(Debug(method(fmt)))]
168        f1: u8,
169    },
170    V3(
171        #[educe(Debug(method(std::fmt::UpperHex::fmt)))]
172        u8,
173        #[educe(Debug(method(fmt)))]
174        T
175    ),
176}
177# }
178```
179
180###### Generic Parameters Bound to the `Debug` Trait or Others
181
182Generic parameters will be automatically bound to the `Debug` trait if necessary.
183
184```rust
185# #[cfg(feature = "Debug")]
186# {
187use educe::Educe;
188
189#[derive(Educe)]
190#[educe(Debug)]
191enum Enum<T, K> {
192    V1,
193    V2 {
194        f1: K,
195    },
196    V3(
197        T
198    ),
199}
200# }
201```
202
203Or you can set the where predicates by yourself.
204
205```rust
206# #[cfg(feature = "Debug")]
207# {
208use educe::Educe;
209
210use std::fmt::{self, Formatter};
211
212fn fmt<D>(_s: &D, f: &mut Formatter<'_>) -> fmt::Result {
213    f.write_str("Hi")
214}
215
216#[derive(Educe)]
217#[educe(Debug(bound(T: std::fmt::Debug)))]
218enum Enum<T, K> {
219    V1,
220    V2 {
221        #[educe(Debug(method(fmt)))]
222        f1: K,
223    },
224    V3(
225        T
226    ),
227}
228# }
229```
230
231In the above case, `T` is bound to the `Debug` trait, but `K` is not.
232
233Or, you can have `educe` replicate the behaviour of `std`'s `derive`'s, where a bound is produced for *every* generic parameter, without regard to how it's used in the structure:
234
235```rust
236# #[cfg(feature = "Debug")]
237# {
238use educe::Educe;
239
240#[derive(Educe)]
241#[educe(Debug(bound(*)))]
242struct Struct<T> {
243    #[educe(Debug(ignore))]
244    f: T,
245}
246# }
247```
248
249This can be useful if you don't want to make the trait implementation part of your permanent public API. In this example, `Struct<T>` doesn't implement `Debug` unless `T` does. I.e., it has a `T: Debug` bound even though that's not needed right now. Later we might want to display `f`; we wouldn't then need to make a breaking API change by adding the bound.
250
251This was the behaviour of `Trait(bound)` in educe 0.4.x and earlier.
252
253###### Union
254
255A union will be formatted as a `u8` slice because we don't know its fields at runtime. The fields of a union cannot be ignored, renamed, or formatted with other methods. The implementation is **unsafe** because it may expose uninitialized memory.
256
257```rust
258# #[cfg(feature = "Debug")]
259# {
260use educe::Educe;
261
262#[derive(Educe)]
263#[educe(Debug(unsafe))]
264union Union {
265    f1: u8,
266    f2: i32,
267}
268# }
269```
270
271#### Clone
272
273Use `#[derive(Educe)]` and `#[educe(Clone)]` to implement the `Clone` trait for a struct, an enum, or a union. You can set a method to replace the `Clone` trait.
274
275###### Basic Usage
276
277```rust
278# #[cfg(feature = "Clone")]
279# {
280use educe::Educe;
281
282#[derive(Educe)]
283#[educe(Clone)]
284struct Struct {
285    f1: u8
286}
287
288#[derive(Educe)]
289#[educe(Clone)]
290enum Enum {
291    V1,
292    V2 {
293        f1: u8,
294    },
295    V3(u8),
296}
297# }
298```
299
300###### Use Another Method to Perform Cloning
301
302The `method` parameter can be utilized to replace the implementation of the `Clone` trait for a field, eliminating the need to implement the `Clone` trait for the type of that field.
303
304```rust
305# #[cfg(feature = "Clone")]
306# {
307use educe::Educe;
308
309fn clone(v: &u8) -> u8 {
310    v + 100
311}
312
313trait A {
314    fn add(&self, rhs: u8) -> Self;
315}
316
317fn clone2<T: A>(v: &T) -> T {
318    v.add(100)
319}
320
321#[derive(Educe)]
322#[educe(Clone)]
323enum Enum<T: A> {
324    V1,
325    V2 {
326        #[educe(Clone(method(clone)))]
327        f1: u8,
328    },
329    V3(
330        #[educe(Clone(method(clone2)))]
331        T
332    ),
333}
334# }
335```
336
337###### Generic Parameters Bound to the `Clone` Trait or Others
338
339Generic parameters will be automatically bound to the `Clone` trait if necessary. If the `#[educe(Copy)]` attribute exists, they will be bound to the `Copy` trait.
340
341```rust
342# #[cfg(feature = "Clone")]
343# {
344use educe::Educe;
345
346#[derive(Educe)]
347#[educe(Clone)]
348enum Enum<T, K> {
349    V1,
350    V2 {
351        f1: K,
352    },
353    V3(
354        T
355    ),
356}
357# }
358```
359
360Or you can set the where predicates by yourself.
361
362```rust
363# #[cfg(feature = "Clone")]
364# {
365use educe::Educe;
366
367trait A {
368    fn add(&self, rhs: u8) -> Self;
369}
370
371fn clone<T: A>(v: &T) -> T {
372    v.add(100)
373}
374
375#[derive(Educe)]
376#[educe(Clone(bound(T: std::clone::Clone)))]
377enum Enum<T, K: A> {
378    V1,
379    V2 {
380        #[educe(Clone(method(clone)))]
381        f1: K,
382    },
383    V3(
384        T
385    ),
386}
387# }
388```
389
390In the above case, `T` is bound to the `Clone` trait, but `K` is not.
391
392Or, you can have `educe` replicate the behaviour of `std`'s `derive`'s by using `bound(*)`. See the [`Debug`](#debug) section for more information.
393
394```rust
395# #[cfg(feature = "Clone")]
396# {
397use educe::Educe;
398
399trait A {
400    fn add(&self, rhs: u8) -> Self;
401}
402
403fn clone<T: A>(v: &T) -> T {
404    v.add(100)
405}
406
407#[derive(Educe)]
408#[educe(Clone(bound(*)))]
409struct Struct<T: A> {
410    #[educe(Clone(method(clone)))]
411    f: T,
412}
413# }
414```
415
416###### Union
417
418Refer to the introduction of the `#[educe(Copy)]` attribute.
419
420#### Copy
421
422Use `#[derive(Educe)]` and `#[educe(Copy)]` to implement the `Copy` trait for a struct, an enum, or a union.
423
424###### Basic Usage
425
426```rust
427# #[cfg(all(feature = "Clone", feature = "Copy"))]
428# {
429use educe::Educe;
430
431#[derive(Educe)]
432#[educe(Copy, Clone)]
433struct Struct {
434    f1: u8
435}
436
437#[derive(Educe)]
438#[educe(Copy, Clone)]
439enum Enum {
440    V1,
441    V2 {
442        f1: u8,
443    },
444    V3(u8),
445}
446# }
447```
448
449###### Generic Parameters Bound to the `Copy` Trait or Others
450
451All generic parameters will be automatically bound to the `Copy` trait.
452
453```rust
454# #[cfg(all(feature = "Clone", feature = "Copy"))]
455# {
456use educe::Educe;
457
458#[derive(Educe)]
459#[educe(Copy, Clone)]
460enum Enum<T, K> {
461    V1,
462    V2 {
463        f1: K,
464    },
465    V3(
466        T
467    ),
468}
469# }
470```
471
472Or you can set the where predicates by yourself.
473
474```rust
475# #[cfg(all(feature = "Clone", feature = "Copy"))]
476# {
477use educe::Educe;
478
479trait A {
480    fn add(&self, rhs: u8) -> Self;
481}
482
483fn clone<T: A>(v: &T) -> T {
484    v.add(100)
485}
486
487#[derive(Educe)]
488#[educe(Copy, Clone(bound(T: Copy, K: A + Copy)))]
489enum Enum<T, K> {
490    V1,
491    V2 {
492        #[educe(Clone(method(clone)))]
493        f1: K,
494    },
495    V3(
496        T
497    ),
498}
499# }
500```
501
502Note that utilizing custom cloning methods for a type that implements the `Copy` and `Clone` traits may not be entirely appropriate.
503
504###### Union
505
506The `#[educe(Copy, Clone)]` attribute can be used for a union. The fields of a union cannot be cloned with other methods.
507
508```rust
509# #[cfg(all(feature = "Clone", feature = "Copy"))]
510# {
511use educe::Educe;
512
513#[derive(Educe)]
514#[educe(Copy, Clone)]
515union Union {
516    f1: u8,
517}
518# }
519```
520
521#### PartialEq
522
523Use `#[derive(Educe)]` and `#[educe(PartialEq)]` to implement the `PartialEq` trait for a struct, enum, or union. You can also choose to ignore specific fields or set a method to replace the `PartialEq` trait.
524
525###### Basic Usage
526
527```rust
528# #[cfg(feature = "PartialEq")]
529# {
530use educe::Educe;
531
532#[derive(Educe)]
533#[educe(PartialEq)]
534struct Struct {
535    f1: u8
536}
537
538#[derive(Educe)]
539#[educe(PartialEq)]
540enum Enum {
541    V1,
542    V2 {
543        f1: u8,
544    },
545    V3(u8),
546}
547# }
548```
549
550###### Ignore Fields
551
552The `ignore` parameter can ignore a specific field.
553
554```rust
555# #[cfg(feature = "PartialEq")]
556# {
557use educe::Educe;
558
559#[derive(Educe)]
560#[educe(PartialEq)]
561struct Struct {
562    #[educe(PartialEq(ignore))]
563    f1: u8
564}
565
566#[derive(Educe)]
567#[educe(PartialEq)]
568enum Enum {
569    V1,
570    V2 {
571        #[educe(PartialEq(ignore))]
572        f1: u8,
573    },
574    V3(
575        #[educe(PartialEq(ignore))]
576        u8
577    ),
578}
579# }
580```
581
582###### Use Another Method to Perform Comparison
583
584The `method` parameter can be utilized to replace the implementation of the `PartialEq` trait for a field, eliminating the need to implement the `PartialEq` trait for the type of that field.
585
586```rust
587# #[cfg(feature = "PartialEq")]
588# {
589use educe::Educe;
590
591fn eq(a: &u8, b: &u8) -> bool {
592    a + 1 == *b
593}
594
595trait A {
596    fn is_same(&self, other: &Self) -> bool;
597}
598
599fn eq2<T: A>(a: &T, b: &T) -> bool {
600    a.is_same(b)
601}
602
603#[derive(Educe)]
604#[educe(PartialEq)]
605enum Enum<T: A> {
606    V1,
607    V2 {
608        #[educe(PartialEq(method(eq)))]
609        f1: u8,
610    },
611    V3(
612        #[educe(PartialEq(method(eq2)))]
613        T
614    ),
615}
616# }
617```
618
619###### Generic Parameters Bound to the `PartialEq` Trait or Others
620
621Generic parameters will be automatically bound to the `PartialEq` trait if necessary.
622
623```rust
624# #[cfg(feature = "PartialEq")]
625# {
626use educe::Educe;
627
628#[derive(Educe)]
629#[educe(PartialEq)]
630enum Enum<T, K> {
631    V1,
632    V2 {
633        f1: K,
634    },
635    V3(
636        T
637    ),
638}
639# }
640```
641
642Or you can set the where predicates by yourself.
643
644```rust
645# #[cfg(feature = "PartialEq")]
646# {
647use educe::Educe;
648
649trait A {
650    fn is_same(&self, other: &Self) -> bool;
651}
652
653fn eq<T: A>(a: &T, b: &T) -> bool {
654    a.is_same(b)
655}
656
657#[derive(Educe)]
658#[educe(PartialEq(bound(T: std::cmp::PartialEq, K: A)))]
659enum Enum<T, K> {
660    V1,
661    V2 {
662        #[educe(PartialEq(method(eq)))]
663        f1: K,
664    },
665    V3(
666        T
667    ),
668}
669# }
670```
671
672In the above case, `T` is bound to the `PartialEq` trait, but `K` is not.
673
674You can have `educe` replicate the behaviour of `std`'s `derive`'s by using `bound(*)`. See the [`Debug`](#debug) section for more information.
675
676```rust
677# #[cfg(feature = "PartialEq")]
678# {
679use educe::Educe;
680
681#[derive(Educe)]
682#[educe(PartialEq(bound(*)))]
683struct Struct<T> {
684    #[educe(PartialEq(ignore))]
685    f: T,
686}
687# }
688```
689
690###### Union
691
692The `#[educe(PartialEq(unsafe))]` attribute can be used for a union. The fields of a union cannot be compared with other methods. The implementation is **unsafe** because it disregards the specific fields it utilizes.
693
694```rust
695# #[cfg(feature = "PartialEq")]
696# {
697use educe::Educe;
698
699#[derive(Educe)]
700#[educe(PartialEq(unsafe))]
701union Union {
702    f1: u8,
703    f2: i32
704}
705# }
706```
707
708#### Eq
709
710Use `#[derive(Educe)]` and `#[educe(Eq)]` to implement the `Eq` trait for a struct, enum, or union. You can also choose to ignore specific fields or set a method to replace the `PartialEq` trait.
711
712###### Basic Usage
713
714```rust
715# #[cfg(all(feature = "PartialEq", feature = "Eq"))]
716# {
717use educe::Educe;
718
719#[derive(Educe)]
720#[educe(PartialEq, Eq)]
721struct Struct {
722    f1: u8
723}
724
725#[derive(Educe)]
726#[educe(PartialEq, Eq)]
727enum Enum {
728    V1,
729    V2 {
730        f1: u8,
731    },
732    V3(u8),
733}
734# }
735```
736
737###### Ignore Fields
738
739The `ignore` parameter can ignore a specific field.
740
741```rust
742# #[cfg(all(feature = "PartialEq", feature = "Eq"))]
743# {
744use educe::Educe;
745
746#[derive(Educe)]
747#[educe(PartialEq, Eq)]
748struct Struct {
749    #[educe(Eq(ignore))]
750    f1: u8
751}
752
753#[derive(Educe)]
754#[educe(PartialEq, Eq)]
755enum Enum {
756    V1,
757    V2 {
758        #[educe(Eq(ignore))]
759        f1: u8,
760    },
761    V3(
762        #[educe(Eq(ignore))]
763        u8
764    ),
765}
766# }
767```
768
769###### Use Another Method to Perform Comparison
770
771The `method` parameter can be utilized to replace the implementation of the `Eq` trait for a field, eliminating the need to implement the `PartialEq` trait for the type of that field.
772
773```rust
774# #[cfg(all(feature = "PartialEq", feature = "Eq"))]
775# {
776use educe::Educe;
777
778fn eq(a: &u8, b: &u8) -> bool {
779    a + 1 == *b
780}
781
782trait A {
783    fn is_same(&self, other: &Self) -> bool;
784}
785
786fn eq2<T: A>(a: &T, b: &T) -> bool {
787    a.is_same(b)
788}
789
790#[derive(Educe)]
791#[educe(PartialEq, Eq)]
792enum Enum<T: A> {
793    V1,
794    V2 {
795        #[educe(Eq(method(eq)))]
796        f1: u8,
797    },
798    V3(
799        #[educe(Eq(method(eq2)))]
800        T
801    ),
802}
803# }
804```
805
806###### Generic Parameters Bound to the `PartialEq` Trait or Others
807
808Generic parameters will be automatically bound to the `PartialEq` trait if necessary.
809
810```rust
811# #[cfg(all(feature = "PartialEq", feature = "Eq"))]
812# {
813use educe::Educe;
814
815#[derive(Educe)]
816#[educe(PartialEq, Eq)]
817enum Enum<T, K> {
818    V1,
819    V2 {
820        f1: K,
821    },
822    V3(
823        T
824    ),
825}
826# }
827```
828
829Or you can set the where predicates by yourself.
830
831```rust
832# #[cfg(all(feature = "PartialEq", feature = "Eq"))]
833# {
834use educe::Educe;
835
836trait A {
837    fn is_same(&self, other: &Self) -> bool;
838}
839
840fn eq<T: A>(a: &T, b: &T) -> bool {
841    a.is_same(b)
842}
843
844#[derive(Educe)]
845#[educe(PartialEq(bound(T: std::cmp::PartialEq, K: A)), Eq)]
846enum Enum<T, K> {
847    V1,
848    V2 {
849        #[educe(Eq(method(eq)))]
850        f1: K,
851    },
852    V3(
853        T
854    ),
855}
856# }
857```
858
859###### Union
860
861The `#[educe(PartialEq(unsafe), Eq)]` attribute can be used for a union. The fields of a union cannot be compared with other methods. The implementation is **unsafe** because it disregards the specific fields it utilizes.
862
863```rust
864# #[cfg(all(feature = "PartialEq", feature = "Eq"))]
865# {
866use educe::Educe;
867
868#[derive(Educe)]
869#[educe(PartialEq(unsafe), Eq)]
870union Union {
871    f1: u8,
872    f2: i32
873}
874# }
875```
876
877#### PartialOrd
878
879Use `#[derive(Educe)]` and `#[educe(PartialOrd)]` to implement the `PartialOrd` trait for a struct or enum. You can also choose to ignore specific fields or set a method to replace the `PartialOrd` trait.
880
881###### Basic Usage
882
883```rust
884# #[cfg(feature = "PartialOrd")]
885# {
886use educe::Educe;
887
888#[derive(PartialEq, Educe)]
889#[educe(PartialOrd)]
890struct Struct {
891    f1: u8
892}
893
894#[derive(PartialEq, Educe)]
895#[educe(PartialOrd)]
896enum Enum {
897    V1,
898    V2 {
899        f1: u8,
900    },
901    V3(u8),
902}
903# }
904```
905
906###### Ignore Fields
907
908The `ignore` parameter can ignore a specific field.
909
910```rust
911# #[cfg(feature = "PartialOrd")]
912# {
913use educe::Educe;
914
915#[derive(PartialEq, Educe)]
916#[educe(PartialOrd)]
917struct Struct {
918    #[educe(PartialOrd(ignore))]
919    f1: u8
920}
921
922#[derive(PartialEq, Educe)]
923#[educe(PartialOrd)]
924enum Enum {
925    V1,
926    V2 {
927        #[educe(PartialOrd(ignore))]
928        f1: u8,
929    },
930    V3(
931        #[educe(PartialOrd(ignore))]
932        u8
933    ),
934}
935# }
936```
937
938###### Use Another Method to Perform Comparison
939
940The `method` parameter can be utilized to replace the implementation of the `PartialOrd` trait for a field, eliminating the need to implement the `PartialOrd` trait for the type of that field.
941
942```rust
943# #[cfg(all(feature = "PartialEq", feature = "PartialOrd"))]
944# {
945use educe::Educe;
946
947use std::cmp::Ordering;
948
949fn partial_cmp(a: &u8, b: &u8) -> Option<Ordering> {
950    if a > b {
951        Some(Ordering::Less)
952    } else if a < b {
953        Some(Ordering::Greater)
954    } else {
955        Some(Ordering::Equal)
956    }
957}
958
959trait A {
960    fn value(&self) -> u8;
961}
962
963fn partial_cmp2<T: A>(a: &T, b: &T) -> Option<Ordering> {
964    partial_cmp(&a.value(), &b.value())
965}
966
967#[derive(Educe)]
968#[educe(PartialEq, PartialOrd)]
969enum Enum<T: A> {
970    V1,
971    V2 {
972        #[educe(PartialOrd(method(partial_cmp)))]
973        f1: u8,
974    },
975    V3(
976        #[educe(PartialOrd(method(partial_cmp2)))]
977        T
978    ),
979}
980# }
981```
982
983###### Ranking
984
985Each field can add a `#[educe(PartialOrd(rank = priority_value))]` attribute, where `priority_value` is an integer value indicating its comparison precedence (lower values indicate higher priority). The default `priority_value` for a field depends on its ordinal position (lower towards the front) and starts with `isize::MIN`.
986
987```rust
988# #[cfg(feature = "PartialOrd")]
989# {
990use educe::Educe;
991
992#[derive(PartialEq, Educe)]
993#[educe(PartialOrd)]
994struct Struct {
995    #[educe(PartialOrd(rank = 1))]
996    f1: u8,
997    #[educe(PartialOrd(rank = 0))]
998    f2: u8,
999}
1000# }
1001```
1002
1003For variants, the discriminant can be explicitly set for comparison.
1004
1005```rust,ignore
1006# #[cfg(feature = "PartialOrd")]
1007# {
1008use educe::Educe;
1009
1010#[derive(PartialEq, Educe)]
1011#[educe(PartialOrd)]
1012#[repr(u8)]
1013enum Enum {
1014    Three { f1: u8 } = 3,
1015    Two(u8) = 2,
1016    One = 1,
1017}
1018# }
1019```
1020
1021###### Generic Parameters Bound to the `PartialOrd` Trait or Others
1022
1023Generic parameters will be automatically bound to the `PartialOrd` trait if necessary.
1024
1025```rust
1026# #[cfg(feature = "PartialOrd")]
1027# {
1028use educe::Educe;
1029
1030#[derive(PartialEq, Educe)]
1031#[educe(PartialOrd)]
1032enum Enum<T, K> {
1033    V1,
1034    V2 {
1035        f1: K,
1036    },
1037    V3(
1038        T
1039    ),
1040}
1041# }
1042```
1043
1044Or you can set the where predicates by yourself.
1045
1046```rust
1047# #[cfg(feature = "PartialOrd")]
1048# {
1049use educe::Educe;
1050
1051use std::cmp::Ordering;
1052
1053trait A {
1054    fn value(&self) -> u8;
1055}
1056
1057fn partial_cmp<T: A>(a: &T, b: &T) -> Option<Ordering> {
1058    a.value().partial_cmp(&b.value())
1059}
1060
1061#[derive(PartialEq, Educe)]
1062#[educe(PartialOrd(bound(T: std::cmp::PartialOrd, K: PartialEq + A)))]
1063enum Enum<T, K> {
1064    V1,
1065    V2 {
1066        #[educe(PartialOrd(method(partial_cmp)))]
1067        f1: K,
1068    },
1069    V3(
1070        T
1071    ),
1072}
1073# }
1074```
1075
1076In the above case, `T` is bound to the `PartialOrd` trait, but `K` is not.
1077
1078You can have `educe` replicate the behaviour of `std`'s `derive`'s by using `bound(*)`. See the [`Debug`](#debug) section for more information.
1079
1080```rust
1081# #[cfg(feature = "PartialOrd")]
1082# {
1083use educe::Educe;
1084
1085#[derive(PartialEq, Educe)]
1086#[educe(PartialOrd(bound(*)))]
1087struct Struct<T> {
1088    #[educe(PartialOrd(ignore))]
1089    f: T,
1090}
1091# }
1092```
1093
1094###### Union
1095
1096The `#[educe(PartialEq(unsafe))]` attribute can be used for a union. The fields of a union cannot be compared with other methods. The implementation is **unsafe** because it disregards the specific fields it utilizes.
1097
1098```rust
1099# #[cfg(feature = "PartialEq")]
1100# {
1101use educe::Educe;
1102
1103#[derive(Educe)]
1104#[educe(PartialEq(unsafe))]
1105union Union {
1106    f1: u8,
1107    f2: i32
1108}
1109# }
1110```
1111
1112#### Ord
1113
1114Use `#[derive(Educe)]` and `#[educe(Ord)]` to implement the `Ord` trait for a struct or enum. You can also choose to ignore specific fields or set a method to replace the `Ord` trait.
1115
1116###### Basic Usage
1117
1118```rust
1119# #[cfg(all(feature = "PartialOrd", feature = "Ord"))]
1120# {
1121use educe::Educe;
1122
1123#[derive(PartialEq, Eq, Educe)]
1124#[educe(PartialOrd, Ord)]
1125struct Struct {
1126    f1: u8
1127}
1128
1129#[derive(PartialEq, Eq, Educe)]
1130#[educe(PartialOrd, Ord)]
1131enum Enum {
1132    V1,
1133    V2 {
1134        f1: u8,
1135    },
1136    V3(u8),
1137}
1138# }
1139```
1140
1141###### Ignore Fields
1142
1143The `ignore` parameter can ignore a specific field.
1144
1145```rust
1146# #[cfg(all(feature = "PartialOrd", feature = "Ord"))]
1147# {
1148use educe::Educe;
1149
1150#[derive(PartialEq, Eq, Educe)]
1151#[educe(PartialOrd, Ord)]
1152struct Struct {
1153    #[educe(Ord(ignore))]
1154    f1: u8
1155}
1156
1157#[derive(PartialEq, Eq, Educe)]
1158#[educe(PartialOrd, Ord)]
1159enum Enum {
1160    V1,
1161    V2 {
1162        #[educe(Ord(ignore))]
1163        f1: u8,
1164    },
1165    V3(
1166        #[educe(Ord(ignore))]
1167        u8
1168    ),
1169}
1170# }
1171```
1172
1173###### Use Another Method to Perform Comparison
1174
1175The `method` parameter can be utilized to replace the implementation of the `Ord` trait for a field, eliminating the need to implement the `Ord` trait for the type of that field.
1176
1177```rust
1178# #[cfg(all(feature = "PartialEq", feature = "Eq", feature = "PartialOrd", feature = "Ord"))]
1179# {
1180use educe::Educe;
1181
1182use std::cmp::Ordering;
1183
1184fn cmp(a: &u8, b: &u8) -> Ordering {
1185    if a > b {
1186        Ordering::Less
1187    } else if a < b {
1188        Ordering::Greater
1189    } else {
1190        Ordering::Equal
1191    }
1192}
1193
1194trait A {
1195    fn value(&self) -> u8;
1196}
1197
1198fn cmp2<T: A>(a: &T, b: &T) -> Ordering {
1199    cmp(&a.value(), &b.value())
1200}
1201
1202#[derive(Educe)]
1203#[educe(PartialEq, Eq, PartialOrd, Ord)]
1204enum Enum<T: A> {
1205    V1,
1206    V2 {
1207        #[educe(Ord(method(cmp)))]
1208        f1: u8,
1209    },
1210    V3(
1211        #[educe(Ord(method(cmp2)))]
1212        T
1213    ),
1214}
1215# }
1216```
1217
1218###### Ranking
1219
1220Each field can add a `#[educe(Ord(rank = priority_value))]` attribute, where `priority_value` is an integer value indicating its comparison precedence (lower values indicate higher priority). The default `priority_value` for a field depends on its ordinal position (lower towards the front) and starts with `isize::MIN`.
1221
1222```rust
1223# #[cfg(all(feature = "PartialOrd", feature = "Ord"))]
1224# {
1225use educe::Educe;
1226
1227#[derive(PartialEq, Eq, Educe)]
1228#[educe(PartialOrd, Ord)]
1229struct Struct {
1230    #[educe(Ord(rank = 1))]
1231    f1: u8,
1232    #[educe(Ord(rank = 0))]
1233    f2: u8,
1234}
1235# }
1236```
1237
1238For variants, the discriminant can be explicitly set for comparison.
1239
1240```rust,ignore
1241# #[cfg(all(feature = "PartialOrd", feature = "Ord"))]
1242# {
1243use educe::Educe;
1244
1245#[derive(PartialEq, Eq, Educe)]
1246#[educe(PartialOrd, Ord)]
1247#[repr(u8)]
1248enum Enum {
1249    Three { f1: u8 } = 3,
1250    Two(u8) = 2,
1251    One = 1,
1252}
1253# }
1254```
1255
1256###### Generic Parameters Bound to the `Ord` Trait or Others
1257
1258Generic parameters will be automatically bound to the `Ord` trait if necessary.
1259
1260```rust
1261# #[cfg(all(feature = "PartialOrd", feature = "Ord"))]
1262# {
1263use educe::Educe;
1264
1265#[derive(PartialEq, Eq, Educe)]
1266#[educe(PartialOrd, Ord)]
1267enum Enum<T, K> {
1268    V1,
1269    V2 {
1270        f1: K,
1271    },
1272    V3(
1273        T
1274    ),
1275}
1276# }
1277```
1278
1279Or you can set the where predicates by yourself.
1280
1281```rust
1282# #[cfg(all(feature = "PartialOrd", feature = "Ord"))]
1283# {
1284use educe::Educe;
1285
1286use std::cmp::Ordering;
1287
1288trait A {
1289    fn value(&self) -> u8;
1290}
1291
1292fn cmp<T: A>(a: &T, b: &T) -> Ordering {
1293    a.value().cmp(&b.value())
1294}
1295
1296#[derive(PartialEq, Eq, Educe)]
1297#[educe(PartialOrd, Ord(bound(T: std::cmp::Ord, K: std::cmp::Ord + A)))]
1298enum Enum<T, K> {
1299    V1,
1300    V2 {
1301        #[educe(PartialOrd(method(cmp)))]
1302        f1: K,
1303    },
1304    V3(
1305        T
1306    ),
1307}
1308# }
1309```
1310
1311#### Hash
1312
1313Use `#[derive(Educe)]` and `#[educe(Hash)]` to implement the `Hash` trait for a struct, enum, or union. You can also choose to ignore specific fields or set a method to replace the `Hash` trait.
1314
1315###### Basic Usage
1316
1317```rust
1318# #[cfg(feature = "Hash")]
1319# {
1320use educe::Educe;
1321
1322#[derive(Educe)]
1323#[educe(Hash)]
1324struct Struct {
1325    f1: u8
1326}
1327
1328#[derive(Educe)]
1329#[educe(Hash)]
1330enum Enum {
1331    V1,
1332    V2 {
1333        f1: u8,
1334    },
1335    V3(u8),
1336}
1337# }
1338```
1339
1340###### Ignore Fields
1341
1342The `ignore` parameter can ignore a specific field.
1343
1344```rust
1345# #[cfg(feature = "Hash")]
1346# {
1347use educe::Educe;
1348
1349#[derive(Educe)]
1350#[educe(Hash)]
1351struct Struct {
1352    #[educe(Hash(ignore))]
1353    f1: u8
1354}
1355
1356#[derive(Educe)]
1357#[educe(Hash)]
1358enum Enum {
1359    V1,
1360    V2 {
1361        #[educe(Hash(ignore))]
1362        f1: u8,
1363    },
1364    V3(
1365        #[educe(Hash(ignore))]
1366        u8
1367    ),
1368}
1369# }
1370```
1371
1372###### Use Another Method for Hashing
1373
1374The `method` parameter can be utilized to replace the implementation of the `Hash` trait for a field, eliminating the need to implement the `Hash` trait for the type of that field.
1375
1376```rust
1377# #[cfg(feature = "Hash")]
1378# {
1379use educe::Educe;
1380
1381use std::hash::{Hash, Hasher};
1382
1383fn hash<H: Hasher>(_s: &u8, state: &mut H) {
1384    Hash::hash(&100, state)
1385}
1386
1387fn hash2<H: Hasher, T>(_s: &T, state: &mut H) {
1388    Hash::hash(&100, state)
1389}
1390
1391#[derive(Educe)]
1392#[educe(Hash)]
1393enum Enum<T> {
1394    V1,
1395    V2 {
1396        #[educe(Hash(method(hash)))]
1397        f1: u8,
1398    },
1399    V3(
1400        #[educe(Hash(method(hash2)))]
1401        T
1402    ),
1403}
1404# }
1405```
1406
1407###### Generic Parameters Bound to the `Hash` Trait or Others
1408
1409Generic parameters will be automatically bound to the `Hash` trait if necessary.
1410
1411```rust
1412# #[cfg(feature = "Hash")]
1413# {
1414use educe::Educe;
1415
1416#[derive(Educe)]
1417#[educe(Hash)]
1418enum Enum<T, K> {
1419    V1,
1420    V2 {
1421        f1: K,
1422    },
1423    V3(
1424        T
1425    ),
1426}
1427# }
1428```
1429
1430Or you can set the where predicates by yourself.
1431
1432```rust
1433# #[cfg(feature = "Hash")]
1434# {
1435use educe::Educe;
1436
1437use std::hash::{Hash, Hasher};
1438
1439trait A {
1440    fn value(&self) -> u8;
1441}
1442
1443fn hash<H: Hasher, T: A>(s: &T, state: &mut H) {
1444    Hash::hash(&s.value(), state)
1445}
1446
1447#[derive(Educe)]
1448#[educe(Hash(bound(T: std::hash::Hash, K: A)))]
1449enum Enum<T, K> {
1450    V1,
1451    V2 {
1452        #[educe(Hash(method(hash)))]
1453        f1: K,
1454    },
1455    V3(
1456        T
1457    ),
1458}
1459# }
1460```
1461
1462In the above case, `T` is bound to the `Hash` trait, but `K` is not.
1463
1464You can have `educe` replicate the behaviour of `std`'s `derive`'s by using `bound(*)`. See the [`Debug`](#debug) section for more information.
1465
1466```rust
1467# #[cfg(feature = "Hash")]
1468# {
1469use educe::Educe;
1470
1471#[derive(Educe)]
1472#[educe(Hash(bound(*)))]
1473struct Struct<T> {
1474    #[educe(Hash(ignore))]
1475    f: T,
1476}
1477# }
1478```
1479
1480###### Union
1481
1482The `#[educe(PartialEq(unsafe), Eq, Hash(unsafe))]` attribute can be used for a union. The fields of a union cannot be hashed with other methods. The implementation is **unsafe** because it disregards the specific fields it utilizes.
1483
1484```rust
1485# #[cfg(all(feature = "PartialEq", feature = "Eq", feature = "Hash"))]
1486# {
1487use educe::Educe;
1488
1489#[derive(Educe)]
1490#[educe(PartialEq(unsafe), Eq, Hash(unsafe))]
1491union Union {
1492    f1: u8,
1493    f2: i32
1494}
1495# }
1496```
1497
1498#### Default
1499
1500Use `#[derive(Educe)]` and `#[educe(Default)]` to implement the `Default` trait for a struct, enum, or union. You can also choose to ignore specific fields or set a method to replace the `Hash` trait.
1501
1502###### Basic Usage
1503
1504For enums and unions, it is necessary to designate a default variant (for enums) and a default field (for unions) unless the enum has only one variant or the union has only one field.
1505
1506```rust
1507# #[cfg(feature = "Default")]
1508# {
1509use educe::Educe;
1510
1511#[derive(Educe)]
1512#[educe(Default)]
1513struct Struct {
1514    f1: u8
1515}
1516
1517#[derive(Educe)]
1518#[educe(Default)]
1519enum Enum {
1520    V1,
1521    #[educe(Default)]
1522    V2 {
1523        f1: u8,
1524    },
1525    V3(u8),
1526}
1527
1528#[derive(Educe)]
1529#[educe(Default)]
1530union Union {
1531    f1: u8,
1532    #[educe(Default)]
1533    f2: f64,
1534}
1535# }
1536```
1537
1538###### The Default Value for the Entire Type
1539
1540```rust
1541# #[cfg(feature = "Default")]
1542# {
1543use educe::Educe;
1544
1545#[derive(Educe)]
1546#[educe(Default(expression = Struct { f1: 1 }))]
1547struct Struct {
1548    f1: u8
1549}
1550
1551#[derive(Educe)]
1552#[educe(Default(expression = Enum::Struct { f1: 1 }))]
1553enum Enum {
1554    Unit,
1555    Struct {
1556        f1: u8
1557    },
1558    Tuple(u8),
1559}
1560
1561#[derive(Educe)]
1562#[educe(Default(expression = Union { f1: 1 }))]
1563union Union {
1564    f1: u8,
1565    f2: f64,
1566}
1567# }
1568```
1569
1570You may need to activate the `full` feature to enable support for advanced expressions.
1571
1572###### The Default Values for Specific Fields
1573
1574```rust
1575# #[cfg(feature = "Default")]
1576# {
1577use educe::Educe;
1578
1579#[derive(Educe)]
1580#[educe(Default)]
1581struct Struct {
1582    #[educe(Default = 1)]
1583    f1: u8,
1584    #[educe(Default = 11111111111111111111111111111)]
1585    f2: i128,
1586    #[educe(Default = 1.1)]
1587    f3: f64,
1588    #[educe(Default = true)]
1589    f4: bool,
1590    #[educe(Default = "Hi")]
1591    f5: &'static str,
1592    #[educe(Default = "Hello")]
1593    f6: String,
1594    #[educe(Default = 'M')]
1595    f7: char,
1596}
1597
1598#[derive(Educe)]
1599#[educe(Default)]
1600enum Enum {
1601    Unit,
1602    #[educe(Default)]
1603    Tuple(
1604        #[educe(Default(expression = 0 + 1))]
1605        u8,
1606        #[educe(Default(expression = -11111111111111111111111111111 * -1))]
1607        i128,
1608        #[educe(Default(expression = 1.0 + 0.1))]
1609        f64,
1610        #[educe(Default(expression = !false))]
1611        bool,
1612        #[educe(Default(expression = "Hi"))]
1613        &'static str,
1614        #[educe(Default(expression = String::from("Hello")))]
1615        String,
1616        #[educe(Default(expression = 'M'))]
1617        char,
1618    ),
1619}
1620
1621#[derive(Educe)]
1622#[educe(Default)]
1623union Union {
1624    f1: u8,
1625    f2: i128,
1626    f3: f64,
1627    f4: bool,
1628    #[educe(Default = "Hi")]
1629    f5: &'static str,
1630    f6: char,
1631}
1632# }
1633```
1634
1635###### Generic Parameters Bound to the `Default` Trait or Others
1636
1637Generic parameters will be automatically bound to the `Default` trait if necessary.
1638
1639```rust
1640# #[cfg(feature = "Default")]
1641# {
1642use educe::Educe;
1643
1644#[derive(Educe)]
1645#[educe(Default)]
1646enum Enum<T> {
1647    Unit,
1648    #[educe(Default)]
1649    Struct {
1650        f1: T
1651    },
1652    Tuple(T),
1653}
1654# }
1655```
1656
1657Or you can set the where predicates by yourself.
1658
1659```rust
1660# #[cfg(feature = "Default")]
1661# {
1662use educe::Educe;
1663
1664#[derive(Educe)]
1665#[educe(Default(bound(T: std::default::Default)))]
1666enum Enum<T> {
1667    Unit,
1668    #[educe(Default)]
1669    Struct {
1670        f1: T
1671    },
1672    Tuple(T),
1673}
1674# }
1675```
1676
1677###### The `new` Associated Function
1678
1679With the `#[educe(Default(new))]` attribute, your type will include an additional associated function called `new`. This function can be utilized to invoke the `default` method of the `Default` trait.
1680
1681```rust
1682# #[cfg(feature = "Default")]
1683# {
1684use educe::Educe;
1685
1686#[derive(Educe)]
1687#[educe(Default(new))]
1688struct Struct {
1689    f1: u8
1690}
1691# }
1692```
1693
1694#### Deref
1695
1696Use `#[derive(Educe)]` and `#[educe(Deref)]` to implement the `Deref` trait for a struct or enum.
1697
1698###### Basic Usage
1699
1700You must designate a field as the default for obtaining an immutable reference unless the number of fields is exactly one.
1701
1702```rust
1703# #[cfg(feature = "Deref")]
1704# {
1705use educe::Educe;
1706
1707#[derive(Educe)]
1708#[educe(Deref)]
1709struct Struct {
1710    f1: u8,
1711    #[educe(Deref)]
1712    f2: u8,
1713}
1714
1715#[derive(Educe)]
1716#[educe(Deref)]
1717enum Enum {
1718    Struct {
1719        f1: u8
1720    },
1721    Struct2 {
1722        f1: u8,
1723        #[educe(Deref)]
1724        f2: u8,
1725    },
1726    Tuple(u8),
1727    Tuple2(
1728        u8,
1729        #[educe(Deref)]
1730        u8
1731    ),
1732}
1733# }
1734```
1735
1736#### DerefMut
1737
1738Use `#[derive(Educe)]` and `#[educe(DerefMut)]` to implement the `DerefMut` trait for a struct or enum.
1739
1740###### Basic Usage
1741
1742You must designate a field as the default for obtaining an mutable reference unless the number of fields is exactly one.
1743
1744```rust
1745# #[cfg(all(feature = "Deref", feature = "DerefMut"))]
1746# {
1747use educe::Educe;
1748
1749#[derive(Educe)]
1750#[educe(Deref, DerefMut)]
1751struct Struct {
1752    f1: u8,
1753    #[educe(Deref, DerefMut)]
1754    f2: u8,
1755}
1756
1757#[derive(Educe)]
1758#[educe(Deref, DerefMut)]
1759enum Enum {
1760    Struct {
1761        f1: u8
1762    },
1763    Struct2 {
1764        f1: u8,
1765        #[educe(Deref, DerefMut)]
1766        f2: u8,
1767    },
1768    Tuple(u8),
1769    Tuple2(
1770        #[educe(DerefMut)]
1771        u8,
1772        #[educe(Deref)]
1773        u8
1774    ),
1775}
1776# }
1777```
1778
1779The mutable dereferencing fields do not need to be the same as the immutable dereferencing fields, but their types must be consistent.
1780
1781#### Into
1782
1783Use `#[derive(Educe)]` and `#[educe(Into(type))]` to implement the `Into<type>` trait for a struct or enum.
1784
1785###### Basic Usage
1786
1787You need to designate a field as the default for `Into<type>` conversion unless the number of fields is exactly one. If you don't, educe will automatically try to find a proper one.
1788
1789```rust
1790# #[cfg(feature = "Into")]
1791# {
1792use educe::Educe;
1793
1794#[derive(Educe)]
1795#[educe(Into(u8), Into(u16))]
1796struct Struct {
1797    f1: u8,
1798    f2: u16,
1799}
1800
1801#[derive(Educe)]
1802#[educe(Into(u8))]
1803enum Enum {
1804    V1 {
1805        f1: u8,
1806        #[educe(Into(u8))]
1807        f2: u8,
1808    },
1809    V2 (
1810        u8
1811    ),
1812}
1813# }
1814```
1815
1816###### Use Another Method to Perform Into Conversion
1817
1818The `method` parameter can be utilized to replace the implementation of the `Into` trait for a field, eliminating the need to implement the `Into` trait for the type of that field.
1819
1820```rust
1821# #[cfg(feature = "Into")]
1822# {
1823use educe::Educe;
1824
1825fn into(v: u16) -> u8 {
1826    v as u8
1827}
1828
1829#[derive(Educe)]
1830#[educe(Into(u8))]
1831enum Enum {
1832    V1 {
1833        #[educe(Into(u8, method(into)))]
1834        f1: u16,
1835    },
1836    V2 (
1837        u8
1838    ),
1839}
1840# }
1841```
1842
1843###### Generic Parameters Bound to the `Into` Trait or Others
1844
1845Generic parameters will be automatically bound to the `Into<type>` trait if necessary.
1846
1847```rust
1848# #[cfg(feature = "Into")]
1849# {
1850use educe::Educe;
1851
1852#[derive(Educe)]
1853#[educe(Into(u8))]
1854enum Enum<T, K> {
1855    V1 {
1856        f1: K,
1857    },
1858    V2 (
1859        T
1860    ),
1861}
1862# }
1863```
1864
1865Or you can set the where predicates by yourself.
1866
1867```rust
1868# #[cfg(feature = "Into")]
1869# {
1870use educe::Educe;
1871
1872fn into<T>(_v: T) -> u8 {
1873    0
1874}
1875
1876#[derive(Educe)]
1877#[educe(Into(u8, bound(K: Into<u8>)))]
1878enum Enum<T, K> {
1879    V1 {
1880        f1: K,
1881    },
1882    V2 (
1883        #[educe(Into(u8, method(into)))]
1884        T
1885    ),
1886}
1887# }
1888```
1889
1890*/
1891
1892#![cfg_attr(docsrs, feature(doc_auto_cfg))]
1893
1894mod common;
1895#[allow(dead_code)]
1896mod panic;
1897mod supported_traits;
1898mod trait_handlers;
1899
1900use std::collections::HashMap;
1901
1902use proc_macro::TokenStream;
1903use supported_traits::Trait;
1904use syn::{
1905    parse::{Parse, ParseStream},
1906    parse_macro_input,
1907    punctuated::Punctuated,
1908    DeriveInput, Meta, Token,
1909};
1910#[allow(unused)]
1911use trait_handlers::{TraitHandler, TraitHandlerMultiple};
1912
1913fn derive_input_handler(ast: DeriveInput) -> syn::Result<proc_macro2::TokenStream> {
1914    let mut token_stream = proc_macro2::TokenStream::new();
1915    let mut trait_meta_map: HashMap<Trait, Vec<Meta>> = HashMap::new();
1916
1917    for attr in ast.attrs.iter() {
1918        let path = attr.path();
1919
1920        if path.is_ident("educe") {
1921            if let Meta::List(list) = &attr.meta {
1922                let result =
1923                    list.parse_args_with(Punctuated::<Meta, Token![,]>::parse_terminated)?;
1924
1925                for meta in result {
1926                    let path = meta.path();
1927
1928                    let t = match Trait::from_path(path) {
1929                        Some(t) => t,
1930                        None => return Err(panic::unsupported_trait(meta.path())),
1931                    };
1932
1933                    if let Some(v_meta) = trait_meta_map.get_mut(&t) {
1934                        // except for those traits containing generics types
1935
1936                        #[cfg(feature = "Into")]
1937                        if t == Trait::Into {
1938                            v_meta.push(meta);
1939
1940                            continue;
1941                        }
1942
1943                        // avoid unused warnings
1944                        let _ = v_meta;
1945
1946                        return Err(panic::reuse_a_trait(path.get_ident().unwrap()));
1947                    }
1948
1949                    trait_meta_map.insert(t, vec![meta]);
1950                }
1951            } else {
1952                return Err(panic::educe_format_incorrect(path.get_ident().unwrap()));
1953            }
1954        }
1955    }
1956
1957    let traits: Vec<Trait> = trait_meta_map.keys().copied().collect();
1958
1959    #[cfg(feature = "Debug")]
1960    {
1961        if let Some(meta) = trait_meta_map.get(&Trait::Debug) {
1962            trait_handlers::debug::DebugHandler::trait_meta_handler(
1963                &ast,
1964                &mut token_stream,
1965                &traits,
1966                &meta[0],
1967            )?;
1968        }
1969    }
1970
1971    #[cfg(feature = "Clone")]
1972    {
1973        if let Some(meta) = trait_meta_map.get(&Trait::Clone) {
1974            trait_handlers::clone::CloneHandler::trait_meta_handler(
1975                &ast,
1976                &mut token_stream,
1977                &traits,
1978                &meta[0],
1979            )?;
1980        }
1981    }
1982
1983    #[cfg(feature = "Copy")]
1984    {
1985        if let Some(meta) = trait_meta_map.get(&Trait::Copy) {
1986            trait_handlers::copy::CopyHandler::trait_meta_handler(
1987                &ast,
1988                &mut token_stream,
1989                &traits,
1990                &meta[0],
1991            )?;
1992        }
1993    }
1994
1995    #[cfg(feature = "PartialEq")]
1996    {
1997        if let Some(meta) = trait_meta_map.get(&Trait::PartialEq) {
1998            trait_handlers::partial_eq::PartialEqHandler::trait_meta_handler(
1999                &ast,
2000                &mut token_stream,
2001                &traits,
2002                &meta[0],
2003            )?;
2004        }
2005    }
2006
2007    #[cfg(feature = "Eq")]
2008    {
2009        if let Some(meta) = trait_meta_map.get(&Trait::Eq) {
2010            trait_handlers::eq::EqHandler::trait_meta_handler(
2011                &ast,
2012                &mut token_stream,
2013                &traits,
2014                &meta[0],
2015            )?;
2016        }
2017    }
2018
2019    #[cfg(feature = "PartialOrd")]
2020    {
2021        if let Some(meta) = trait_meta_map.get(&Trait::PartialOrd) {
2022            trait_handlers::partial_ord::PartialOrdHandler::trait_meta_handler(
2023                &ast,
2024                &mut token_stream,
2025                &traits,
2026                &meta[0],
2027            )?;
2028        }
2029    }
2030
2031    #[cfg(feature = "Ord")]
2032    {
2033        if let Some(meta) = trait_meta_map.get(&Trait::Ord) {
2034            trait_handlers::ord::OrdHandler::trait_meta_handler(
2035                &ast,
2036                &mut token_stream,
2037                &traits,
2038                &meta[0],
2039            )?;
2040        }
2041    }
2042
2043    #[cfg(feature = "Hash")]
2044    {
2045        if let Some(meta) = trait_meta_map.get(&Trait::Hash) {
2046            trait_handlers::hash::HashHandler::trait_meta_handler(
2047                &ast,
2048                &mut token_stream,
2049                &traits,
2050                &meta[0],
2051            )?;
2052        }
2053    }
2054
2055    #[cfg(feature = "Default")]
2056    {
2057        if let Some(meta) = trait_meta_map.get(&Trait::Default) {
2058            trait_handlers::default::DefaultHandler::trait_meta_handler(
2059                &ast,
2060                &mut token_stream,
2061                &traits,
2062                &meta[0],
2063            )?;
2064        }
2065    }
2066
2067    #[cfg(feature = "Deref")]
2068    {
2069        if let Some(meta) = trait_meta_map.get(&Trait::Deref) {
2070            trait_handlers::deref::DerefHandler::trait_meta_handler(
2071                &ast,
2072                &mut token_stream,
2073                &traits,
2074                &meta[0],
2075            )?;
2076        }
2077    }
2078
2079    #[cfg(feature = "DerefMut")]
2080    {
2081        if let Some(meta) = trait_meta_map.get(&Trait::DerefMut) {
2082            trait_handlers::deref_mut::DerefMutHandler::trait_meta_handler(
2083                &ast,
2084                &mut token_stream,
2085                &traits,
2086                &meta[0],
2087            )?;
2088        }
2089    }
2090
2091    #[cfg(feature = "Into")]
2092    {
2093        if let Some(meta) = trait_meta_map.get(&Trait::Into) {
2094            trait_handlers::into::IntoHandler::trait_meta_handler(
2095                &ast,
2096                &mut token_stream,
2097                &traits,
2098                meta,
2099            )?;
2100        }
2101    }
2102
2103    if trait_meta_map.contains_key(&Trait::_Nothing) {
2104        // avoid unused warnings
2105        let _ = &ast;
2106        let _ = &mut token_stream;
2107        let _ = traits;
2108        unreachable!();
2109    }
2110
2111    if token_stream.is_empty() {
2112        return Err(panic::derive_attribute_not_set_up_yet());
2113    }
2114
2115    Ok(token_stream)
2116}
2117
2118#[proc_macro_derive(Educe, attributes(educe))]
2119pub fn educe_derive(input: TokenStream) -> TokenStream {
2120    struct MyDeriveInput(proc_macro2::TokenStream);
2121
2122    impl Parse for MyDeriveInput {
2123        #[inline]
2124        fn parse(input: ParseStream) -> syn::Result<Self> {
2125            let token_stream = derive_input_handler(input.parse::<DeriveInput>()?)?;
2126
2127            Ok(Self(token_stream))
2128        }
2129    }
2130
2131    // Parse the token stream
2132    let derive_input = parse_macro_input!(input as MyDeriveInput);
2133
2134    derive_input.0.into()
2135}