Skip to content

Commit 34d109f

Browse files
authored
Add new lint unneeded_struct_pattern (#13465)
Closes #13400. changelog: [`unneeded_struct_pattern`]: Add new lint
2 parents a895beb + 52abb8f commit 34d109f

9 files changed

+660
-2
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -6179,6 +6179,7 @@ Released 2018-09-13
61796179
[`unnecessary_unwrap`]: https://rust-lang.github.io/rust-clippy/master/index.html#unnecessary_unwrap
61806180
[`unnecessary_wraps`]: https://rust-lang.github.io/rust-clippy/master/index.html#unnecessary_wraps
61816181
[`unneeded_field_pattern`]: https://rust-lang.github.io/rust-clippy/master/index.html#unneeded_field_pattern
6182+
[`unneeded_struct_pattern`]: https://rust-lang.github.io/rust-clippy/master/index.html#unneeded_struct_pattern
61826183
[`unneeded_wildcard_pattern`]: https://rust-lang.github.io/rust-clippy/master/index.html#unneeded_wildcard_pattern
61836184
[`unnested_or_patterns`]: https://rust-lang.github.io/rust-clippy/master/index.html#unnested_or_patterns
61846185
[`unreachable`]: https://rust-lang.github.io/rust-clippy/master/index.html#unreachable

clippy_lints/src/arbitrary_source_item_ordering.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -445,8 +445,8 @@ fn convert_assoc_item_kind(value: AssocItemKind) -> SourceItemOrderingTraitAssoc
445445
#[allow(clippy::enum_glob_use)] // Very local glob use for legibility.
446446
use SourceItemOrderingTraitAssocItemKind::*;
447447
match value {
448-
AssocItemKind::Const { .. } => Const,
449-
AssocItemKind::Type { .. } => Type,
448+
AssocItemKind::Const => Const,
449+
AssocItemKind::Type => Type,
450450
AssocItemKind::Fn { .. } => Fn,
451451
}
452452
}

clippy_lints/src/declared_lints.rs

+1
Original file line numberDiff line numberDiff line change
@@ -757,6 +757,7 @@ pub static LINTS: &[&crate::LintInfo] = &[
757757
crate::unnecessary_self_imports::UNNECESSARY_SELF_IMPORTS_INFO,
758758
crate::unnecessary_struct_initialization::UNNECESSARY_STRUCT_INITIALIZATION_INFO,
759759
crate::unnecessary_wraps::UNNECESSARY_WRAPS_INFO,
760+
crate::unneeded_struct_pattern::UNNEEDED_STRUCT_PATTERN_INFO,
760761
crate::unnested_or_patterns::UNNESTED_OR_PATTERNS_INFO,
761762
crate::unsafe_removed_from_name::UNSAFE_REMOVED_FROM_NAME_INFO,
762763
crate::unused_async::UNUSED_ASYNC_INFO,

clippy_lints/src/lib.rs

+2
Original file line numberDiff line numberDiff line change
@@ -374,6 +374,7 @@ mod unnecessary_owned_empty_strings;
374374
mod unnecessary_self_imports;
375375
mod unnecessary_struct_initialization;
376376
mod unnecessary_wraps;
377+
mod unneeded_struct_pattern;
377378
mod unnested_or_patterns;
378379
mod unsafe_removed_from_name;
379380
mod unused_async;
@@ -970,5 +971,6 @@ pub fn register_lints(store: &mut rustc_lint::LintStore, conf: &'static Conf) {
970971
store.register_late_pass(|_| Box::new(manual_ignore_case_cmp::ManualIgnoreCaseCmp));
971972
store.register_late_pass(|_| Box::new(unnecessary_literal_bound::UnnecessaryLiteralBound));
972973
store.register_late_pass(move |_| Box::new(arbitrary_source_item_ordering::ArbitrarySourceItemOrdering::new(conf)));
974+
store.register_late_pass(|_| Box::new(unneeded_struct_pattern::UnneededStructPattern));
973975
// add lints here, do not remove this comment, it's used in `new_lint`
974976
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
use clippy_utils::diagnostics::span_lint_and_sugg;
2+
use clippy_utils::is_from_proc_macro;
3+
use rustc_errors::Applicability;
4+
use rustc_hir::def::{DefKind, Res};
5+
use rustc_hir::{Pat, PatKind, QPath};
6+
use rustc_lint::{LateContext, LateLintPass};
7+
use rustc_session::declare_lint_pass;
8+
9+
declare_clippy_lint! {
10+
/// ### What it does
11+
/// Checks for struct patterns that match against unit variant.
12+
///
13+
/// ### Why is this bad?
14+
/// Struct pattern `{ }` or `{ .. }` is not needed for unit variant.
15+
///
16+
/// ### Example
17+
/// ```no_run
18+
/// match Some(42) {
19+
/// Some(v) => v,
20+
/// None { .. } => 0,
21+
/// };
22+
/// // Or
23+
/// match Some(42) {
24+
/// Some(v) => v,
25+
/// None { } => 0,
26+
/// };
27+
/// ```
28+
/// Use instead:
29+
/// ```no_run
30+
/// match Some(42) {
31+
/// Some(v) => v,
32+
/// None => 0,
33+
/// };
34+
/// ```
35+
#[clippy::version = "1.83.0"]
36+
pub UNNEEDED_STRUCT_PATTERN,
37+
style,
38+
"using struct pattern to match against unit variant"
39+
}
40+
41+
declare_lint_pass!(UnneededStructPattern => [UNNEEDED_STRUCT_PATTERN]);
42+
43+
impl LateLintPass<'_> for UnneededStructPattern {
44+
fn check_pat(&mut self, cx: &LateContext<'_>, pat: &Pat<'_>) {
45+
if !pat.span.from_expansion()
46+
&& let PatKind::Struct(path, [], _) = &pat.kind
47+
&& let QPath::Resolved(_, path) = path
48+
&& let Res::Def(DefKind::Variant, did) = path.res
49+
{
50+
let enum_did = cx.tcx.parent(did);
51+
let variant = cx.tcx.adt_def(enum_did).variant_with_id(did);
52+
53+
let has_only_fields_brackets = variant.ctor.is_some() && variant.fields.is_empty();
54+
let non_exhaustive_activated = !variant.def_id.is_local() && variant.is_field_list_non_exhaustive();
55+
if !has_only_fields_brackets || non_exhaustive_activated {
56+
return;
57+
}
58+
59+
if is_from_proc_macro(cx, *path) {
60+
return;
61+
}
62+
63+
if let Some(brackets_span) = pat.span.trim_start(path.span) {
64+
span_lint_and_sugg(
65+
cx,
66+
UNNEEDED_STRUCT_PATTERN,
67+
brackets_span,
68+
"struct pattern is not needed for a unit variant",
69+
"remove the struct pattern",
70+
String::new(),
71+
Applicability::MachineApplicable,
72+
);
73+
}
74+
}
75+
}
76+
}

tests/ui/auxiliary/non-exhaustive-enum.rs

+21
Original file line numberDiff line numberDiff line change
@@ -6,3 +6,24 @@ pub enum ErrorKind {
66
#[doc(hidden)]
77
Uncategorized,
88
}
9+
10+
#[non_exhaustive]
11+
pub enum ExtNonExhaustiveEnum {
12+
Unit,
13+
Tuple(i32),
14+
Struct { field: i32 },
15+
}
16+
17+
pub enum ExtNonExhaustiveVariant {
18+
ExhaustiveUnit,
19+
#[non_exhaustive]
20+
Unit,
21+
#[non_exhaustive]
22+
Tuple(i32),
23+
#[non_exhaustive]
24+
StructNoField {},
25+
#[non_exhaustive]
26+
Struct {
27+
field: i32,
28+
},
29+
}
+177
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,177 @@
1+
//@aux-build:non-exhaustive-enum.rs
2+
#![allow(
3+
clippy::manual_unwrap_or_default,
4+
clippy::manual_unwrap_or,
5+
clippy::redundant_pattern_matching
6+
)]
7+
#![warn(clippy::unneeded_struct_pattern)]
8+
9+
extern crate non_exhaustive_enum;
10+
use non_exhaustive_enum::*;
11+
12+
fn noop() {}
13+
14+
fn main() {
15+
match Some(114514) {
16+
Some(v) => v,
17+
None => 0,
18+
};
19+
20+
match Some(1919810) {
21+
Some(v) => v,
22+
None => 0,
23+
};
24+
25+
match Some(123456) {
26+
Some(v) => v,
27+
None => 0,
28+
};
29+
30+
match Some(Some(123456)) {
31+
Some(Some(v)) => v,
32+
Some(None) => 0,
33+
None => 0,
34+
};
35+
36+
if let None = Some(0) {}
37+
if let None = Some(0) {}
38+
if let Some(None) = Some(Some(0)) {}
39+
let None = Some(0) else { panic!() };
40+
let None = Some(0) else { panic!() };
41+
let Some(None) = Some(Some(0)) else { panic!() };
42+
43+
enum Custom {
44+
HasFields {
45+
field: i32,
46+
},
47+
HasBracketsNoFields {},
48+
NoBrackets,
49+
#[non_exhaustive]
50+
NoBracketsNonExhaustive,
51+
Init,
52+
};
53+
54+
match Custom::Init {
55+
Custom::HasFields { field: value } => value,
56+
Custom::HasBracketsNoFields {} => 0,
57+
Custom::NoBrackets => 0, //~ ERROR: struct pattern is not needed for a unit variant
58+
Custom::NoBracketsNonExhaustive => 0, //~ ERROR: struct pattern is not needed for a unit variant
59+
_ => 0,
60+
};
61+
62+
match Custom::Init {
63+
Custom::HasFields { field: value } => value,
64+
Custom::HasBracketsNoFields { .. } => 0,
65+
Custom::NoBrackets => 0, //~ ERROR: struct pattern is not needed for a unit variant
66+
Custom::NoBracketsNonExhaustive => 0, //~ ERROR: struct pattern is not needed for a unit variant
67+
_ => 0,
68+
};
69+
70+
match Custom::Init {
71+
Custom::NoBrackets if true => 0, //~ ERROR: struct pattern is not needed for a unit variant
72+
_ => 0,
73+
};
74+
75+
match Custom::Init {
76+
Custom::NoBrackets | Custom::NoBracketsNonExhaustive => 0, //~ ERROR: struct pattern is not needed for a unit variant
77+
_ => 0,
78+
};
79+
80+
if let Custom::HasFields { field: value } = Custom::Init {
81+
noop();
82+
}
83+
if let Custom::HasBracketsNoFields {} = Custom::Init {
84+
noop();
85+
}
86+
if let Custom::HasBracketsNoFields { .. } = Custom::Init {
87+
noop();
88+
}
89+
if let Custom::NoBrackets = Custom::Init {
90+
//~^ ERROR: struct pattern is not needed for a unit variant
91+
noop();
92+
}
93+
if let Custom::NoBrackets = Custom::Init {
94+
//~^ ERROR: struct pattern is not needed for a unit variant
95+
noop();
96+
}
97+
if let Custom::NoBrackets | Custom::NoBracketsNonExhaustive = Custom::Init {
98+
//~^ ERROR: struct pattern is not needed for a unit variant
99+
noop();
100+
}
101+
if let Custom::NoBracketsNonExhaustive = Custom::Init {
102+
//~^ ERROR: struct pattern is not needed for a unit variant
103+
noop();
104+
}
105+
if let Custom::NoBracketsNonExhaustive = Custom::Init {
106+
//~^ ERROR: struct pattern is not needed for a unit variant
107+
noop();
108+
}
109+
110+
let Custom::HasFields { field: value } = Custom::Init else {
111+
panic!()
112+
};
113+
114+
let Custom::HasBracketsNoFields {} = Custom::Init else {
115+
panic!()
116+
};
117+
118+
let Custom::HasBracketsNoFields { .. } = Custom::Init else {
119+
panic!()
120+
};
121+
let Custom::NoBrackets = Custom::Init else { panic!() }; //~ ERROR: struct pattern is not needed for a unit variant
122+
123+
let Custom::NoBrackets = Custom::Init else {
124+
//~^ ERROR: struct pattern is not needed for a unit variant
125+
panic!()
126+
};
127+
let Custom::NoBracketsNonExhaustive = Custom::Init else {
128+
//~^ ERROR: struct pattern is not needed for a unit variant
129+
panic!()
130+
};
131+
let Custom::NoBracketsNonExhaustive = Custom::Init else {
132+
//~^ ERROR: struct pattern is not needed for a unit variant
133+
panic!()
134+
};
135+
136+
enum Refutable {
137+
Variant,
138+
}
139+
140+
fn pat_in_fn_param_1(Refutable::Variant: Refutable) {} //~ ERROR: struct pattern is not needed for a unit variant
141+
fn pat_in_fn_param_2(Refutable::Variant: Refutable) {} //~ ERROR: struct pattern is not needed for a unit variant
142+
143+
for Refutable::Variant in [] {} //~ ERROR: struct pattern is not needed for a unit variant
144+
for Refutable::Variant in [] {} //~ ERROR: struct pattern is not needed for a unit variant
145+
}
146+
147+
fn external_crate() {
148+
use ExtNonExhaustiveVariant::*;
149+
150+
match ExhaustiveUnit {
151+
// Expected
152+
ExhaustiveUnit => 0,
153+
_ => 0,
154+
};
155+
156+
match ExhaustiveUnit {
157+
// Exhaustive variant
158+
ExhaustiveUnit => 0, //~ ERROR: struct pattern is not needed for a unit variant
159+
_ => 0,
160+
};
161+
162+
match ExhaustiveUnit {
163+
// Exhaustive variant
164+
ExhaustiveUnit => 0, //~ ERROR: struct pattern is not needed for a unit variant
165+
_ => 0,
166+
};
167+
168+
match ExhaustiveUnit {
169+
ExhaustiveUnit => 0,
170+
// vvvvv Non-exhaustive variants, should all be ignored
171+
Unit { .. } => 0,
172+
Tuple { 0: field, .. } => field,
173+
StructNoField { .. } => 0,
174+
Struct { field, .. } => field,
175+
_ => 0,
176+
};
177+
}

0 commit comments

Comments
 (0)