bevy_lint/lints/suspicious/
insert_event_resource.rs1use crate::{
43 declare_bevy_lint, declare_bevy_lint_pass,
44 utils::hir_parse::{MethodCall, generic_args_snippet, span_args},
45};
46use clippy_utils::{
47 diagnostics::span_lint_and_sugg,
48 source::{snippet, snippet_with_applicability},
49 sym,
50 ty::{match_type, ty_from_hir_ty},
51};
52use rustc_errors::Applicability;
53use rustc_hir::{Expr, GenericArg, GenericArgs, Path, PathSegment, QPath};
54use rustc_lint::{LateContext, LateLintPass};
55use rustc_middle::ty::{Ty, TyKind};
56use rustc_span::Symbol;
57use std::borrow::Cow;
58
59declare_bevy_lint! {
60 pub INSERT_EVENT_RESOURCE,
61 super::SUSPICIOUS,
62 "called `App::insert_resource(Events<T>)` or `App::init_resource::<Events<T>>()` instead of `App::add_event::<T>()`",
63}
64
65declare_bevy_lint_pass! {
66 pub InsertEventResource => [INSERT_EVENT_RESOURCE.lint],
67 @default = {
68 insert_resource: Symbol = sym!(insert_resource),
69 init_resource: Symbol = sym!(init_resource),
70 },
71}
72
73const HELP_MESSAGE: &str = "inserting an `Events` resource does not fully setup that event";
74
75impl<'tcx> LateLintPass<'tcx> for InsertEventResource {
76 fn check_expr(&mut self, cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>) {
77 if expr.span.in_external_macro(cx.tcx.sess.source_map()) {
79 return;
80 }
81
82 if let Some(method_call) = MethodCall::try_from(cx, expr) {
84 let src_ty = cx
87 .typeck_results()
88 .expr_ty(method_call.receiver)
89 .peel_refs();
90
91 if !match_type(cx, src_ty, &crate::paths::APP) {
93 return;
94 }
95
96 match method_call.method_path.ident.name {
99 symbol if symbol == self.insert_resource => {
100 check_insert_resource(cx, &method_call);
101 }
102 symbol if symbol == self.init_resource => {
103 check_init_resource(cx, &method_call);
104 }
105 _ => {}
106 }
107 }
108 }
109}
110
111fn check_insert_resource(cx: &LateContext<'_>, method_call: &MethodCall) {
113 let [arg] = method_call.args else {
115 return;
116 };
117
118 let ty = cx.typeck_results().expr_ty(arg);
120
121 if match_type(cx, ty, &crate::paths::EVENTS) {
123 let mut applicability = Applicability::MachineApplicable;
124
125 let event_ty_snippet = extract_ty_event_snippet(ty, &mut applicability);
126 let args_snippet = snippet(cx, span_args(method_call.args), "");
127 let generics_snippet = generic_args_snippet(cx, method_call.method_path);
128
129 if method_call.is_fully_qulified {
130 let receiver_snippet = snippet(cx, method_call.receiver.span, "");
131 span_lint_and_sugg(
132 cx,
133 INSERT_EVENT_RESOURCE.lint,
134 method_call.span,
135 format!(
136 "called `App::insert_resource{generics_snippet}({receiver_snippet}, {args_snippet})` instead of `App::add_event::<{event_ty_snippet}>({receiver_snippet})`"
137 ),
138 HELP_MESSAGE,
139 format!("App::add_event::<{event_ty_snippet}>({receiver_snippet})"),
140 applicability,
141 );
142 } else {
143 span_lint_and_sugg(
144 cx,
145 INSERT_EVENT_RESOURCE.lint,
146 method_call.span,
147 format!(
148 "called `App::insert_resource{generics_snippet}({args_snippet})` instead of `App::add_event::<{event_ty_snippet}>()`"
149 ),
150 HELP_MESSAGE,
151 format!("add_event::<{event_ty_snippet}>()"),
152 applicability,
153 );
154 }
155 }
156}
157
158fn extract_ty_event_snippet<'tcx>(
163 events_ty: Ty<'tcx>,
164 applicability: &mut Applicability,
165) -> Cow<'tcx, str> {
166 const DEFAULT: Cow<str> = Cow::Borrowed("T");
167
168 let TyKind::Adt(_, events_arguments) = events_ty.kind() else {
169 if let Applicability::MachineApplicable = applicability {
170 *applicability = Applicability::HasPlaceholders;
171 }
172
173 return DEFAULT;
174 };
175
176 let Some(event_snippet) = events_arguments.iter().next() else {
177 if let Applicability::MachineApplicable = applicability {
178 *applicability = Applicability::HasPlaceholders;
179 }
180
181 return DEFAULT;
182 };
183
184 format!("{event_snippet:?}").into()
185}
186
187fn check_init_resource<'tcx>(cx: &LateContext<'tcx>, method_call: &MethodCall<'tcx>) {
189 if let Some(&GenericArgs {
190 args: &[GenericArg::Type(resource_hir_ty)],
192 ..
193 }) = method_call.method_path.args
194 {
195 let resource_ty = ty_from_hir_ty(cx, resource_hir_ty.as_unambig_ty());
198
199 if match_type(cx, resource_ty, &crate::paths::EVENTS) {
201 let mut applicability = Applicability::MachineApplicable;
202
203 let event_ty_snippet =
204 extract_hir_event_snippet(cx, resource_hir_ty.as_unambig_ty(), &mut applicability);
205
206 let args_snippet = snippet(cx, span_args(method_call.args), "");
207 let generics_snippet = generic_args_snippet(cx, method_call.method_path);
208
209 if method_call.is_fully_qulified {
210 let receiver_snippet = snippet(cx, method_call.receiver.span, "");
211 span_lint_and_sugg(
212 cx,
213 INSERT_EVENT_RESOURCE.lint,
214 method_call.span,
215 format!(
216 "called `App::init_resource{generics_snippet}({receiver_snippet})` instead of `App::add_event::<{event_ty_snippet}>({receiver_snippet})`"
217 ),
218 HELP_MESSAGE,
219 format!("App::add_event::<{event_ty_snippet}>({receiver_snippet})"),
220 applicability,
221 );
222 } else {
223 span_lint_and_sugg(
224 cx,
225 INSERT_EVENT_RESOURCE.lint,
226 method_call.span,
227 format!(
228 "called `App::init_resource{generics_snippet}({args_snippet})` instead of `App::add_event::<{event_ty_snippet}>()`"
229 ),
230 HELP_MESSAGE,
231 format!("add_event::<{event_ty_snippet}>()"),
232 applicability,
233 );
234 }
235 }
236 }
237}
238
239fn extract_hir_event_snippet<'tcx>(
246 cx: &LateContext<'tcx>,
247 events_hir_ty: &rustc_hir::Ty<'tcx>,
248 applicability: &mut Applicability,
249) -> Cow<'static, str> {
250 const DEFAULT: Cow<str> = Cow::Borrowed("T");
251
252 let event_span = match events_hir_ty.kind {
254 rustc_hir::TyKind::Path(QPath::Resolved(
258 _,
259 &Path {
260 segments:
263 &[
264 ..,
265 PathSegment {
266 args:
268 Some(&GenericArgs {
269 args: &[GenericArg::Type(ty)],
270 ..
271 }),
272 ..
273 },
274 ],
275 ..
276 },
277 )) => {
278 ty.span
280 }
281 _ => {
284 if let Applicability::MachineApplicable = applicability {
285 *applicability = Applicability::HasPlaceholders;
286 }
287
288 return DEFAULT;
289 }
290 };
291
292 snippet_with_applicability(cx, event_span, &DEFAULT, applicability)
294}