2023-04-10 16:30:28 +00:00
|
|
|
# Trait Checking
|
|
|
|
|
|
|
|
Besides [type checking](type_checking.md), we might want to examine if
|
|
|
|
a specific type `Ty` implements certain trait when implementing a lint.
|
|
|
|
There are three approaches to achieve this, depending on if the target trait
|
|
|
|
that we want to examine has a [diagnostic item][diagnostic_items],
|
|
|
|
[lang item][lang_items], or neither.
|
|
|
|
|
|
|
|
## Using Diagnostic Items
|
|
|
|
|
|
|
|
As explained in the [Rust Compiler Development Guide][rustc_dev_guide], diagnostic items
|
|
|
|
are introduced for identifying types via [Symbols][symbol].
|
|
|
|
|
|
|
|
For instance, if we want to examine whether an expression implements
|
|
|
|
the `Iterator` trait, we could simply write the following code,
|
|
|
|
providing the `LateContext` (`cx`), our expression at hand, and
|
|
|
|
the symbol of the trait in question:
|
|
|
|
|
|
|
|
```rust
|
|
|
|
use clippy_utils::is_trait_method;
|
|
|
|
use rustc_hir::Expr;
|
|
|
|
use rustc_lint::{LateContext, LateLintPass};
|
|
|
|
use rustc_span::symbol::sym;
|
|
|
|
|
|
|
|
impl LateLintPass<'_> for CheckIteratorTraitLint {
|
|
|
|
fn check_expr(&mut self, cx: &LateContext<'_>, expr: &Expr<'_>) {
|
2023-04-18 18:50:41 +00:00
|
|
|
let implements_iterator = cx.tcx.get_diagnostic_item(sym::Iterator).map_or(false, |id| {
|
|
|
|
implements_trait(cx, cx.typeck_results().expr_ty(arg), id, &[])
|
|
|
|
});
|
|
|
|
if implements_iterator {
|
|
|
|
// [...]
|
|
|
|
}
|
|
|
|
|
2023-04-10 16:30:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
> **Note**: Refer to [this index][symbol_index] for all the defined `Symbol`s.
|
|
|
|
|
|
|
|
## Using Lang Items
|
|
|
|
|
|
|
|
Besides diagnostic items, we can also use [`lang_items`][lang_items].
|
2023-04-18 18:50:41 +00:00
|
|
|
Take a look at the documentation to find that `LanguageItems` contains
|
|
|
|
all language items defined in the compiler.
|
2023-04-10 16:30:28 +00:00
|
|
|
|
|
|
|
Using one of its `*_trait` method, we could obtain the [DefId] of any
|
|
|
|
specific item, such as `Clone`, `Copy`, `Drop`, `Eq`, which are familiar
|
|
|
|
to many Rustaceans.
|
|
|
|
|
|
|
|
For instance, if we want to examine whether an expression `expr` implements
|
|
|
|
`Drop` trait, we could access `LanguageItems` via our `LateContext`'s
|
|
|
|
[TyCtxt], which provides a `lang_items` method that will return the id of
|
|
|
|
`Drop` trait to us. Then, by calling Clippy utils function `implements_trait`
|
|
|
|
we can check that the `Ty` of the `expr` implements the trait:
|
|
|
|
|
|
|
|
```rust
|
|
|
|
use clippy_utils::implements_trait;
|
|
|
|
use rustc_hir::Expr;
|
|
|
|
use rustc_lint::{LateContext, LateLintPass};
|
|
|
|
|
|
|
|
impl LateLintPass<'_> for CheckDropTraitLint {
|
|
|
|
fn check_expr(&mut self, cx: &LateContext<'_>, expr: &Expr<'_>) {
|
|
|
|
let ty = cx.typeck_results().expr_ty(expr);
|
|
|
|
if cx.tcx.lang_items()
|
|
|
|
.drop_trait()
|
|
|
|
.map_or(false, |id| implements_trait(cx, ty, id, &[])) {
|
|
|
|
println!("`expr` implements `Drop` trait!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
## Using Type Path
|
|
|
|
|
2023-04-18 18:50:41 +00:00
|
|
|
If neither diagnostic item nor a language item is available, we can use
|
2023-04-10 16:30:28 +00:00
|
|
|
[`clippy_utils::paths`][paths] with the `match_trait_method` to determine trait
|
|
|
|
implementation.
|
|
|
|
|
2023-04-18 18:50:41 +00:00
|
|
|
> **Note**: This approach should be avoided if possible, the best thing to do would be to make a PR to [`rust-lang/rust`][rust].
|
2023-04-10 16:30:28 +00:00
|
|
|
|
|
|
|
Below, we check if the given `expr` implements `tokio`'s
|
|
|
|
[`AsyncReadExt`][AsyncReadExt] trait:
|
|
|
|
|
|
|
|
```rust
|
|
|
|
use clippy_utils::{match_trait_method, paths};
|
|
|
|
use rustc_hir::Expr;
|
|
|
|
use rustc_lint::{LateContext, LateLintPass};
|
|
|
|
|
|
|
|
impl LateLintPass<'_> for CheckTokioAsyncReadExtTrait {
|
|
|
|
fn check_expr(&mut self, cx: &LateContext<'_>, expr: &Expr<'_>) {
|
2023-04-18 18:50:41 +00:00
|
|
|
if match_trait_method(cx, expr, &paths::CORE_ITER_CLONED) {
|
|
|
|
println!("`expr` implements `CORE_ITER_CLONED` trait!");
|
2023-04-10 16:30:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
[AsyncReadExt]: https://docs.rs/tokio/latest/tokio/io/trait.AsyncReadExt.html
|
|
|
|
[DefId]: https://doc.rust-lang.org/nightly/nightly-rustc/rustc_hir/def_id/struct.DefId.html
|
|
|
|
[diagnostic_items]: https://rustc-dev-guide.rust-lang.org/diagnostics/diagnostic-items.html
|
|
|
|
[lang_items]: https://doc.rust-lang.org/nightly/nightly-rustc/rustc_hir/lang_items/struct.LanguageItems.html
|
|
|
|
[paths]: https://github.com/rust-lang/rust-clippy/blob/master/clippy_utils/src/paths.rs
|
|
|
|
[rustc_dev_guide]: https://rustc-dev-guide.rust-lang.org/
|
|
|
|
[symbol]: https://doc.rust-lang.org/nightly/nightly-rustc/rustc_span/symbol/struct.Symbol.html
|
|
|
|
[symbol_index]: https://doc.rust-lang.org/beta/nightly-rustc/rustc_span/symbol/sym/index.html
|
|
|
|
[TyCtxt]: https://doc.rust-lang.org/nightly/nightly-rustc/rustc_middle/ty/context/struct.TyCtxt.html
|
|
|
|
[using_diagnostic_items]: https://rustc-dev-guide.rust-lang.org/diagnostics/diagnostic-items.html#using-diagnostic-items
|
2023-04-18 18:50:41 +00:00
|
|
|
[rust]: https://github.com/rust-lang/rust
|