diff options
Diffstat (limited to '')
-rw-r--r-- | third_party/rust/inherent/src/lib.rs | 123 |
1 files changed, 123 insertions, 0 deletions
diff --git a/third_party/rust/inherent/src/lib.rs b/third_party/rust/inherent/src/lib.rs new file mode 100644 index 0000000000..7ac6390831 --- /dev/null +++ b/third_party/rust/inherent/src/lib.rs @@ -0,0 +1,123 @@ +//! [![github]](https://github.com/dtolnay/inherent) [![crates-io]](https://crates.io/crates/inherent) [![docs-rs]](https://docs.rs/inherent) +//! +//! [github]: https://img.shields.io/badge/github-8da0cb?style=for-the-badge&labelColor=555555&logo=github +//! [crates-io]: https://img.shields.io/badge/crates.io-fc8d62?style=for-the-badge&labelColor=555555&logo=rust +//! [docs-rs]: https://img.shields.io/badge/docs.rs-66c2a5?style=for-the-badge&labelColor=555555&logoColor=white&logo= +//! +//! <br> +//! +//! ##### An attribute macro to make trait methods callable without the trait in scope. +//! +//! # Example +//! +//! ```rust +//! mod types { +//! use inherent::inherent; +//! +//! trait Trait { +//! fn f(self); +//! } +//! +//! pub struct Struct; +//! +//! #[inherent(pub)] +//! impl Trait for Struct { +//! fn f(self) {} +//! } +//! } +//! +//! fn main() { +//! // types::Trait is not in scope, but method can be called. +//! types::Struct.f(); +//! } +//! ``` +//! +//! Without the `inherent` macro on the trait impl, this would have failed with the +//! following error: +//! +//! ```console +//! error[E0599]: no method named `f` found for type `types::Struct` in the current scope +//! --> src/main.rs:18:19 +//! | +//! 8 | pub struct Struct; +//! | ------------------ method `f` not found for this +//! ... +//! 18 | types::Struct.f(); +//! | ^ +//! | +//! = help: items from traits can only be used if the trait is implemented and in scope +//! = note: the following trait defines an item `f`, perhaps you need to implement it: +//! candidate #1: `types::Trait` +//! ``` +//! +//! The `inherent` macro expands to inherent methods on the `Self` type of the trait +//! impl that forward to the trait methods. In the case above, the generated code +//! would be: +//! +//! ```rust +//! # trait Trait { +//! # fn f(self); +//! # } +//! # +//! # pub struct Struct; +//! # +//! # impl Trait for Struct { +//! # fn f(self) {} +//! # } +//! # +//! impl Struct { +//! pub fn f(self) { +//! <Self as Trait>::f(self) +//! } +//! } +//! ``` +//! +//! # Visibility +//! +//! Ordinary trait methods have the same visibility as the trait or the `Self` type, +//! whichever's is smaller. Neither of these visibilities is knowable to the +//! `inherent` macro, so we simply emit all inherent methods as private by default. +//! A different visibility may be specified explicitly in the `inherent` macro +//! invocation. +//! +//! ```rust +//! # use inherent::inherent; +//! # +//! # trait Trait {} +//! # +//! # struct A; +//! #[inherent] // private inherent methods are the default +//! # impl Trait for A {} +//! +//! # struct B; +//! #[inherent(pub)] // all methods pub +//! # impl Trait for B {} +//! +//! # struct C; +//! #[inherent(crate)] // all methods pub(crate) +//! # impl Trait for C {} +//! +//! # struct D; +//! #[inherent(in path::to)] // all methods pub(in path::to) +//! # impl Trait for D {} +//! ``` + +extern crate proc_macro; + +mod default_methods; +mod expand; +mod parse; +mod visibility; + +use proc_macro::TokenStream; +use syn::parse_macro_input; + +use crate::parse::TraitImpl; +use crate::visibility::Visibility; + +#[proc_macro_attribute] +pub fn inherent(args: TokenStream, input: TokenStream) -> TokenStream { + let vis = parse_macro_input!(args as Visibility); + let input = parse_macro_input!(input as TraitImpl); + expand::inherent(vis, input).into() +} |