Skip to content

Commit 111adde

Browse files
committed
Auto merge of #115324 - francorbacho:master, r=davidtwco
Suggest removing redundant arguments in format!() Closes #105225. This is also a follow-up to #105635, which seems to have become stale. r? `@estebank`
2 parents 913ceae + cbc6b65 commit 111adde

File tree

8 files changed

+291
-4
lines changed

8 files changed

+291
-4
lines changed

compiler/rustc_builtin_macros/messages.ftl

+14
Original file line numberDiff line numberDiff line change
@@ -137,6 +137,20 @@ builtin_macros_format_positional_after_named = positional arguments cannot follo
137137
.label = positional arguments must be before named arguments
138138
.named_args = named argument
139139
140+
builtin_macros_format_redundant_args = redundant {$n ->
141+
[one] argument
142+
*[more] arguments
143+
}
144+
.help = {$n ->
145+
[one] the formatting string already captures the binding directly, it doesn't need to be included in the argument list
146+
*[more] the formatting strings already captures the bindings directly, they don't need to be included in the argument list
147+
}
148+
.note = {$n ->
149+
[one] the formatting specifier is referencing the binding already
150+
*[more] the formatting specifiers are referencing the bindings already
151+
}
152+
.suggestion = this can be removed
153+
140154
builtin_macros_format_remove_raw_ident = remove the `r#`
141155
142156
builtin_macros_format_requires_string = requires at least a format string argument

compiler/rustc_builtin_macros/src/errors.rs

+21
Original file line numberDiff line numberDiff line change
@@ -646,6 +646,27 @@ pub(crate) struct FormatPositionalMismatch {
646646
pub(crate) highlight: SingleLabelManySpans,
647647
}
648648

649+
#[derive(Diagnostic)]
650+
#[diag(builtin_macros_format_redundant_args)]
651+
pub(crate) struct FormatRedundantArgs {
652+
#[primary_span]
653+
pub(crate) span: MultiSpan,
654+
pub(crate) n: usize,
655+
656+
#[note]
657+
pub(crate) note: MultiSpan,
658+
659+
#[subdiagnostic]
660+
pub(crate) sugg: Option<FormatRedundantArgsSugg>,
661+
}
662+
663+
#[derive(Subdiagnostic)]
664+
#[multipart_suggestion(builtin_macros_suggestion, applicability = "machine-applicable")]
665+
pub(crate) struct FormatRedundantArgsSugg {
666+
#[suggestion_part(code = "")]
667+
pub(crate) spans: Vec<Span>,
668+
}
669+
649670
#[derive(Diagnostic)]
650671
#[diag(builtin_macros_test_case_non_item)]
651672
pub(crate) struct TestCaseNonItem {

compiler/rustc_builtin_macros/src/format.rs

+110-4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
use parse::Position::ArgumentNamed;
12
use rustc_ast::ptr::P;
23
use rustc_ast::tokenstream::TokenStream;
34
use rustc_ast::{token, StmtKind};
@@ -7,7 +8,9 @@ use rustc_ast::{
78
FormatDebugHex, FormatOptions, FormatPlaceholder, FormatSign, FormatTrait,
89
};
910
use rustc_data_structures::fx::FxHashSet;
10-
use rustc_errors::{Applicability, MultiSpan, PResult, SingleLabelManySpans};
11+
use rustc_errors::{
12+
Applicability, DiagnosticBuilder, ErrorGuaranteed, MultiSpan, PResult, SingleLabelManySpans,
13+
};
1114
use rustc_expand::base::{self, *};
1215
use rustc_parse_format as parse;
1316
use rustc_span::symbol::{Ident, Symbol};
@@ -364,8 +367,8 @@ fn make_format_args(
364367
let mut unfinished_literal = String::new();
365368
let mut placeholder_index = 0;
366369

367-
for piece in pieces {
368-
match piece {
370+
for piece in &pieces {
371+
match *piece {
369372
parse::Piece::String(s) => {
370373
unfinished_literal.push_str(s);
371374
}
@@ -513,7 +516,17 @@ fn make_format_args(
513516
// If there's a lot of unused arguments,
514517
// let's check if this format arguments looks like another syntax (printf / shell).
515518
let detect_foreign_fmt = unused.len() > args.explicit_args().len() / 2;
516-
report_missing_placeholders(ecx, unused, detect_foreign_fmt, str_style, fmt_str, fmt_span);
519+
report_missing_placeholders(
520+
ecx,
521+
unused,
522+
&used,
523+
&args,
524+
&pieces,
525+
detect_foreign_fmt,
526+
str_style,
527+
fmt_str,
528+
fmt_span,
529+
);
517530
}
518531

519532
// Only check for unused named argument names if there are no other errors to avoid causing
@@ -580,6 +593,9 @@ fn invalid_placeholder_type_error(
580593
fn report_missing_placeholders(
581594
ecx: &mut ExtCtxt<'_>,
582595
unused: Vec<(Span, bool)>,
596+
used: &[bool],
597+
args: &FormatArguments,
598+
pieces: &[parse::Piece<'_>],
583599
detect_foreign_fmt: bool,
584600
str_style: Option<usize>,
585601
fmt_str: &str,
@@ -598,6 +614,26 @@ fn report_missing_placeholders(
598614
})
599615
};
600616

617+
let placeholders = pieces
618+
.iter()
619+
.filter_map(|piece| {
620+
if let parse::Piece::NextArgument(argument) = piece && let ArgumentNamed(binding) = argument.position {
621+
let span = fmt_span.from_inner(InnerSpan::new(argument.position_span.start, argument.position_span.end));
622+
Some((span, binding))
623+
} else { None }
624+
})
625+
.collect::<Vec<_>>();
626+
627+
if !placeholders.is_empty() {
628+
if let Some(mut new_diag) =
629+
report_redundant_format_arguments(ecx, &args, used, placeholders)
630+
{
631+
diag.cancel();
632+
new_diag.emit();
633+
return;
634+
}
635+
}
636+
601637
// Used to ensure we only report translations for *one* kind of foreign format.
602638
let mut found_foreign = false;
603639

@@ -685,6 +721,76 @@ fn report_missing_placeholders(
685721
diag.emit();
686722
}
687723

724+
/// This function detects and reports unused format!() arguments that are
725+
/// redundant due to implicit captures (e.g. `format!("{x}", x)`).
726+
fn report_redundant_format_arguments<'a>(
727+
ecx: &mut ExtCtxt<'a>,
728+
args: &FormatArguments,
729+
used: &[bool],
730+
placeholders: Vec<(Span, &str)>,
731+
) -> Option<DiagnosticBuilder<'a, ErrorGuaranteed>> {
732+
let mut fmt_arg_indices = vec![];
733+
let mut args_spans = vec![];
734+
let mut fmt_spans = vec![];
735+
736+
for (i, unnamed_arg) in args.unnamed_args().iter().enumerate().rev() {
737+
let Some(ty) = unnamed_arg.expr.to_ty() else { continue };
738+
let Some(argument_binding) = ty.kind.is_simple_path() else { continue };
739+
let argument_binding = argument_binding.as_str();
740+
741+
if used[i] {
742+
continue;
743+
}
744+
745+
let matching_placeholders = placeholders
746+
.iter()
747+
.filter(|(_, inline_binding)| argument_binding == *inline_binding)
748+
.map(|(span, _)| span)
749+
.collect::<Vec<_>>();
750+
751+
if !matching_placeholders.is_empty() {
752+
fmt_arg_indices.push(i);
753+
args_spans.push(unnamed_arg.expr.span);
754+
for span in &matching_placeholders {
755+
if fmt_spans.contains(*span) {
756+
continue;
757+
}
758+
fmt_spans.push(**span);
759+
}
760+
}
761+
}
762+
763+
if !args_spans.is_empty() {
764+
let multispan = MultiSpan::from(fmt_spans);
765+
let mut suggestion_spans = vec![];
766+
767+
for (arg_span, fmt_arg_idx) in args_spans.iter().zip(fmt_arg_indices.iter()) {
768+
let span = if fmt_arg_idx + 1 == args.explicit_args().len() {
769+
*arg_span
770+
} else {
771+
arg_span.until(args.explicit_args()[*fmt_arg_idx + 1].expr.span)
772+
};
773+
774+
suggestion_spans.push(span);
775+
}
776+
777+
let sugg = if args.named_args().len() == 0 {
778+
Some(errors::FormatRedundantArgsSugg { spans: suggestion_spans })
779+
} else {
780+
None
781+
};
782+
783+
return Some(ecx.create_err(errors::FormatRedundantArgs {
784+
n: args_spans.len(),
785+
span: MultiSpan::from(args_spans),
786+
note: multispan,
787+
sugg,
788+
}));
789+
}
790+
791+
None
792+
}
793+
688794
/// Handle invalid references to positional arguments. Output different
689795
/// errors for the case where all arguments are positional and for when
690796
/// there are named arguments or numbered positional arguments in the
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
fn main() {
2+
let x = "x";
3+
let y = "y";
4+
5+
println!("{x}", x, x = y);
6+
//~^ ERROR: redundant argument
7+
8+
println!("{x}", x = y, x = y);
9+
//~^ ERROR: duplicate argument named `x`
10+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
error: redundant argument
2+
--> $DIR/issue-105225-named-args.rs:5:21
3+
|
4+
LL | println!("{x}", x, x = y);
5+
| ^
6+
|
7+
note: the formatting specifier is referencing the binding already
8+
--> $DIR/issue-105225-named-args.rs:5:16
9+
|
10+
LL | println!("{x}", x, x = y);
11+
| ^
12+
13+
error: duplicate argument named `x`
14+
--> $DIR/issue-105225-named-args.rs:8:28
15+
|
16+
LL | println!("{x}", x = y, x = y);
17+
| - ^ duplicate argument
18+
| |
19+
| previously here
20+
21+
error: aborting due to 2 previous errors
22+
+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
// run-rustfix
2+
3+
fn main() {
4+
let x = "x";
5+
let y = "y";
6+
7+
println!("{x}", );
8+
//~^ ERROR: redundant argument
9+
10+
println!("{x} {}", x, );
11+
//~^ ERROR: redundant argument
12+
13+
println!("{} {x}", x, );
14+
//~^ ERROR: redundant argument
15+
16+
println!("{x} {y}", );
17+
//~^ ERROR: redundant arguments
18+
19+
println!("{} {} {x} {y} {}", x, x, x, );
20+
//~^ ERROR: redundant arguments
21+
}

tests/ui/did_you_mean/issue-105225.rs

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
// run-rustfix
2+
3+
fn main() {
4+
let x = "x";
5+
let y = "y";
6+
7+
println!("{x}", x);
8+
//~^ ERROR: redundant argument
9+
10+
println!("{x} {}", x, x);
11+
//~^ ERROR: redundant argument
12+
13+
println!("{} {x}", x, x);
14+
//~^ ERROR: redundant argument
15+
16+
println!("{x} {y}", x, y);
17+
//~^ ERROR: redundant arguments
18+
19+
println!("{} {} {x} {y} {}", x, x, x, y, y);
20+
//~^ ERROR: redundant arguments
21+
}
+72
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
error: redundant argument
2+
--> $DIR/issue-105225.rs:7:21
3+
|
4+
LL | println!("{x}", x);
5+
| ^ help: this can be removed
6+
|
7+
note: the formatting specifier is referencing the binding already
8+
--> $DIR/issue-105225.rs:7:16
9+
|
10+
LL | println!("{x}", x);
11+
| ^
12+
13+
error: redundant argument
14+
--> $DIR/issue-105225.rs:10:27
15+
|
16+
LL | println!("{x} {}", x, x);
17+
| ^ help: this can be removed
18+
|
19+
note: the formatting specifier is referencing the binding already
20+
--> $DIR/issue-105225.rs:10:16
21+
|
22+
LL | println!("{x} {}", x, x);
23+
| ^
24+
25+
error: redundant argument
26+
--> $DIR/issue-105225.rs:13:27
27+
|
28+
LL | println!("{} {x}", x, x);
29+
| ^ help: this can be removed
30+
|
31+
note: the formatting specifier is referencing the binding already
32+
--> $DIR/issue-105225.rs:13:19
33+
|
34+
LL | println!("{} {x}", x, x);
35+
| ^
36+
37+
error: redundant arguments
38+
--> $DIR/issue-105225.rs:16:25
39+
|
40+
LL | println!("{x} {y}", x, y);
41+
| ^ ^
42+
|
43+
note: the formatting specifiers are referencing the bindings already
44+
--> $DIR/issue-105225.rs:16:16
45+
|
46+
LL | println!("{x} {y}", x, y);
47+
| ^ ^
48+
help: this can be removed
49+
|
50+
LL - println!("{x} {y}", x, y);
51+
LL + println!("{x} {y}", );
52+
|
53+
54+
error: redundant arguments
55+
--> $DIR/issue-105225.rs:19:43
56+
|
57+
LL | println!("{} {} {x} {y} {}", x, x, x, y, y);
58+
| ^ ^
59+
|
60+
note: the formatting specifiers are referencing the bindings already
61+
--> $DIR/issue-105225.rs:19:26
62+
|
63+
LL | println!("{} {} {x} {y} {}", x, x, x, y, y);
64+
| ^
65+
help: this can be removed
66+
|
67+
LL - println!("{} {} {x} {y} {}", x, x, x, y, y);
68+
LL + println!("{} {} {x} {y} {}", x, x, x, );
69+
|
70+
71+
error: aborting due to 5 previous errors
72+

0 commit comments

Comments
 (0)