Skip to content

Commit 2d3a9a5

Browse files
committed
remove braces when fixing a nested use tree into a single use
1 parent 13f7623 commit 2d3a9a5

File tree

4 files changed

+135
-1
lines changed

4 files changed

+135
-1
lines changed

compiler/rustc_resolve/src/check_unused.rs

+26-1
Original file line numberDiff line numberDiff line change
@@ -292,14 +292,15 @@ fn calc_unused_spans(
292292
UnusedSpanResult::Used
293293
}
294294
}
295-
ast::UseTreeKind::Nested { items: ref nested, .. } => {
295+
ast::UseTreeKind::Nested { items: ref nested, span: tree_span } => {
296296
if nested.is_empty() {
297297
return UnusedSpanResult::Unused { spans: vec![use_tree.span], remove: full_span };
298298
}
299299

300300
let mut unused_spans = Vec::new();
301301
let mut to_remove = Vec::new();
302302
let mut used_childs = 0;
303+
let mut contains_self = false;
303304
let mut previous_unused = false;
304305
for (pos, (use_tree, use_tree_id)) in nested.iter().enumerate() {
305306
let remove = match calc_unused_spans(unused_import, use_tree, *use_tree_id) {
@@ -339,13 +340,37 @@ fn calc_unused_spans(
339340
to_remove.push(remove_span);
340341
}
341342
}
343+
contains_self |= use_tree.prefix == kw::SelfLower
344+
&& matches!(use_tree.kind, ast::UseTreeKind::Simple(None));
342345
previous_unused = remove.is_some();
343346
}
344347
if unused_spans.is_empty() {
345348
UnusedSpanResult::Used
346349
} else if used_childs == 0 {
347350
UnusedSpanResult::Unused { spans: unused_spans, remove: full_span }
348351
} else {
352+
// If there is only one remaining child that is used, the braces around the use
353+
// tree are not needed anymore. In that case, we determine the span of the left
354+
// brace and the right brace, and tell rustfix to remove them as well.
355+
//
356+
// This means that `use a::{B, C};` will be turned into `use a::B;` rather than
357+
// `use a::{B};`, removing a rustfmt roundtrip.
358+
//
359+
// Note that we cannot remove the braces if the only item inside the use tree is
360+
// `self`: `use foo::{self};` is valid Rust syntax, while `use foo::self;` errors
361+
// out. We also cannot turn `use foo::{self}` into `use foo`, as the former doesn't
362+
// import types with the same name as the module.
363+
if used_childs == 1 && !contains_self {
364+
// Left brace, from the start of the nested group to the first item.
365+
to_remove.push(
366+
tree_span.shrink_to_lo().to(nested.first().unwrap().0.span.shrink_to_lo()),
367+
);
368+
// Right brace, from the end of the last item to the end of the nested group.
369+
to_remove.push(
370+
nested.last().unwrap().0.span.shrink_to_hi().to(tree_span.shrink_to_hi()),
371+
);
372+
}
373+
349374
UnusedSpanResult::PartialUnused { spans: unused_spans, remove: to_remove }
350375
}
351376
}
+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
//@ run-rustfix
2+
//@ check-pass
3+
4+
#![warn(unused_imports)]
5+
6+
pub mod nested {
7+
pub struct A;
8+
pub struct B;
9+
pub struct C;
10+
pub struct D;
11+
pub mod even_more {
12+
pub struct E;
13+
pub struct F;
14+
pub struct G;
15+
}
16+
pub mod another {
17+
pub struct H;
18+
pub struct I;
19+
}
20+
}
21+
22+
use nested::B;
23+
//~^ WARN unused import
24+
25+
use nested::even_more::F;
26+
//~^^^^^^^ WARN unused import
27+
28+
// Note that the following fix should result in `::{self}`, not `::self`. The latter is invalid
29+
// Rust syntax, so the braces should not be removed.
30+
use nested::another::{self};
31+
//~^ WARN unused import
32+
33+
fn main() {
34+
let _ = (B, F, another::I);
35+
}
+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
//@ run-rustfix
2+
//@ check-pass
3+
4+
#![warn(unused_imports)]
5+
6+
pub mod nested {
7+
pub struct A;
8+
pub struct B;
9+
pub struct C;
10+
pub struct D;
11+
pub mod even_more {
12+
pub struct E;
13+
pub struct F;
14+
pub struct G;
15+
}
16+
pub mod another {
17+
pub struct H;
18+
pub struct I;
19+
}
20+
}
21+
22+
use nested::{A, B, C};
23+
//~^ WARN unused import
24+
25+
use nested::{
26+
D,
27+
even_more::{
28+
E,
29+
F,
30+
G,
31+
},
32+
};
33+
//~^^^^^^^ WARN unused import
34+
35+
// Note that the following fix should result in `::{self}`, not `::self`. The latter is invalid
36+
// Rust syntax, so the braces should not be removed.
37+
use nested::another::{self, I};
38+
//~^ WARN unused import
39+
40+
fn main() {
41+
let _ = (B, F, another::I);
42+
}
+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
warning: unused imports: `A`, `C`
2+
--> $DIR/unused-imports.rs:22:14
3+
|
4+
LL | use nested::{A, B, C};
5+
| ^ ^
6+
|
7+
note: the lint level is defined here
8+
--> $DIR/unused-imports.rs:4:9
9+
|
10+
LL | #![warn(unused_imports)]
11+
| ^^^^^^^^^^^^^^
12+
13+
warning: unused imports: `D`, `E`, `G`
14+
--> $DIR/unused-imports.rs:26:5
15+
|
16+
LL | D,
17+
| ^
18+
LL | even_more::{
19+
LL | E,
20+
| ^
21+
LL | F,
22+
LL | G,
23+
| ^
24+
25+
warning: unused import: `I`
26+
--> $DIR/unused-imports.rs:37:29
27+
|
28+
LL | use nested::another::{self, I};
29+
| ^
30+
31+
warning: 3 warnings emitted
32+

0 commit comments

Comments
 (0)