bevy_lint/lints/suspicious/
iter_current_update_events.rs

1//! Checks for calls to `Events::<T>::iter_current_update_events()`.
2//!
3//! # Motivation
4//!
5//! `Events::<T>::iter_current_update_events()` lets you read all of the current events since
6//! `Events::<T>::update()` was last called, similar to `EventReader<T>`. Unlike `EventReader<T>`,
7//! `iter_current_update_events()` does not track which events have already been read. As such,
8//! `iter_current_update_events()` is highly discouraged because it may skip events or yield them
9//! multiple times.
10//!
11//! # Example
12//!
13//! ```
14//! # use bevy::prelude::*;
15//! #
16//! #[derive(Event)]
17//! struct MyEvent;
18//!
19//! fn my_system(events: Res<Events<MyEvent>>) {
20//!     for event in events.iter_current_update_events() {
21//!         // ...
22//!     }
23//! }
24//! ```
25//!
26//! Use instead:
27//!
28//! ```
29//! # use bevy::prelude::*;
30//! #
31//! #[derive(Event)]
32//! struct MyEvent;
33//!
34//! fn my_system(mut events: EventReader<MyEvent>) {
35//!     for event in events.read() {
36//!         // ...
37//!     }
38//! }
39//! ```
40
41use clippy_utils::{diagnostics::span_lint_and_help, ty::match_type};
42use rustc_hir::Expr;
43use rustc_lint::{LateContext, LateLintPass};
44use rustc_span::Symbol;
45
46use crate::{declare_bevy_lint, declare_bevy_lint_pass, utils::hir_parse::MethodCall};
47
48declare_bevy_lint! {
49    pub ITER_CURRENT_UPDATE_EVENTS,
50    super::SUSPICIOUS,
51    "called `Events::<T>::iter_current_update_events()`",
52}
53
54declare_bevy_lint_pass! {
55    pub IterCurrentUpdateEvents => [ITER_CURRENT_UPDATE_EVENTS.lint],
56
57    @default = {
58        iter_current_update_events: Symbol = Symbol::intern("iter_current_update_events"),
59    },
60}
61
62impl<'tcx> LateLintPass<'tcx> for IterCurrentUpdateEvents {
63    fn check_expr(&mut self, cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>) {
64        // If this expression was generated by an external macro, don't lint it.
65        if expr.span.in_external_macro(cx.tcx.sess.source_map()) {
66            return;
67        }
68
69        if let Some(method_call) = MethodCall::try_from(cx, expr) {
70            // Find the adjusted type of the receiver. Type adjustment does things like
71            // auto-dereference and type coercion. In this example, we use the adjusted type so
72            // that we can also handle `Res<Events<T>>`.
73            //
74            // ```
75            // fn plain(events: Events<T>) {
76            //     // Original type is `Events<T>`, adjusted type is `Events<T>`.
77            //     let _ = events.iter_current_update_events();
78            // }
79            //
80            // fn res(events: Res<Events<T>>) {
81            //     // Original type is `Res<Events<T>>`, adjusted type is `Events<T>`.
82            //     let _ = events.iter_current_update_events();
83            // }
84            // ```
85            let src_ty = cx
86                .typeck_results()
87                .expr_ty_adjusted(method_call.receiver)
88                .peel_refs();
89
90            if !match_type(cx, src_ty, &crate::paths::EVENTS) {
91                return;
92            }
93
94            if method_call.method_path.ident.name == self.iter_current_update_events {
95                span_lint_and_help(
96                    cx,
97                    ITER_CURRENT_UPDATE_EVENTS.lint,
98                    method_call.span,
99                    ITER_CURRENT_UPDATE_EVENTS.lint.desc,
100                    None,
101                    "`iter_current_update_events()` does not track which events have already been seen, consider using `EventReader<T>` instead",
102                );
103            }
104        }
105    }
106}