Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit e99984c

Browse files
committedDec 27, 2023
Suggest quoting unquoted idents in attrs
1 parent f8fe517 commit e99984c

8 files changed

+90
-7
lines changed
 

‎compiler/rustc_parse/messages.ftl

+2-1
Original file line numberDiff line numberDiff line change
@@ -406,7 +406,8 @@ parse_invalid_logical_operator = `{$incorrect}` is not a logical operator
406406
.use_amp_amp_for_conjunction = use `&&` to perform logical conjunction
407407
.use_pipe_pipe_for_disjunction = use `||` to perform logical disjunction
408408
409-
parse_invalid_meta_item = expected unsuffixed literal or identifier, found `{$token}`
409+
parse_invalid_meta_item = expected unsuffixed literal{$or_ident}, found `{$token}`
410+
.suggestion = surround the identifier with quotation marks to parse it as a string
410411
411412
parse_invalid_unicode_escape = invalid unicode character escape
412413
.label = invalid escape

‎compiler/rustc_parse/src/errors.rs

+12
Original file line numberDiff line numberDiff line change
@@ -971,6 +971,18 @@ pub(crate) struct InvalidMetaItem {
971971
#[primary_span]
972972
pub span: Span,
973973
pub token: Token,
974+
pub or_ident: &'static str,
975+
#[subdiagnostic]
976+
pub sugg: Option<InvalidMetaItemSuggQuoteIdent>,
977+
}
978+
979+
#[derive(Subdiagnostic)]
980+
#[multipart_suggestion(parse_suggestion, applicability = "machine-applicable")]
981+
pub(crate) struct InvalidMetaItemSuggQuoteIdent {
982+
#[suggestion_part(code = "\"")]
983+
pub before: Span,
984+
#[suggestion_part(code = "\"")]
985+
pub after: Span,
974986
}
975987

976988
#[derive(Subdiagnostic)]

‎compiler/rustc_parse/src/parser/attr.rs

+15-4
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
use crate::errors::{InvalidMetaItem, SuffixedLiteralInAttribute};
1+
use crate::errors::{InvalidMetaItem, InvalidMetaItemSuggQuoteIdent, SuffixedLiteralInAttribute};
22
use crate::fluent_generated as fluent;
33

44
use super::{AttrWrapper, Capturing, FnParseMode, ForceCollect, Parser, PathStyle};
@@ -416,9 +416,20 @@ impl<'a> Parser<'a> {
416416
Err(err) => err.cancel(),
417417
}
418418

419-
Err(self
420-
.dcx()
421-
.create_err(InvalidMetaItem { span: self.token.span, token: self.token.clone() }))
419+
let token = self.token.clone();
420+
let (sugg, or_ident) = if self.prev_token == token::Eq && !self.token.span.from_expansion()
421+
{
422+
let before = self.token.span.shrink_to_lo();
423+
while matches!(self.token.kind, token::Ident(..)) {
424+
self.bump();
425+
}
426+
let after = self.prev_token.span.shrink_to_hi();
427+
(Some(InvalidMetaItemSuggQuoteIdent { before, after }), "")
428+
} else {
429+
(None, " or identifier")
430+
};
431+
432+
Err(self.dcx().create_err(InvalidMetaItem { span: token.span, token, or_ident, sugg }))
422433
}
423434
}
424435

‎tests/ui/deprecation/issue-66340-deprecated-attr-non-meta-grammar.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,5 +7,5 @@ fn main() {
77
}
88

99
#[deprecated(note = test)]
10-
//~^ ERROR expected unsuffixed literal or identifier, found `test`
10+
//~^ ERROR expected unsuffixed literal, found `test`
1111
fn foo() {}
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,13 @@
1-
error: expected unsuffixed literal or identifier, found `test`
1+
error: expected unsuffixed literal, found `test`
22
--> $DIR/issue-66340-deprecated-attr-non-meta-grammar.rs:9:21
33
|
44
LL | #[deprecated(note = test)]
55
| ^^^^
6+
|
7+
help: surround the identifier with quotation marks to parse it as a string
8+
|
9+
LL | #[deprecated(note = "test")]
10+
| + +
611

712
error: aborting due to 1 previous error
813

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
// compile-flags: -Zdeduplicate-diagnostics=yes
2+
// run-rustfix
3+
4+
fn main() {
5+
#[cfg(key="foo")]
6+
//~^ ERROR expected unsuffixed literal, found `foo`
7+
//~| HELP surround the identifier with quotation marks to parse it as a string
8+
println!();
9+
#[cfg(key="bar")]
10+
println!();
11+
#[cfg(key="foo bar baz")]
12+
//~^ ERROR expected unsuffixed literal, found `foo`
13+
//~| HELP surround the identifier with quotation marks to parse it as a string
14+
println!();
15+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
// compile-flags: -Zdeduplicate-diagnostics=yes
2+
// run-rustfix
3+
4+
fn main() {
5+
#[cfg(key=foo)]
6+
//~^ ERROR expected unsuffixed literal, found `foo`
7+
//~| HELP surround the identifier with quotation marks to parse it as a string
8+
println!();
9+
#[cfg(key="bar")]
10+
println!();
11+
#[cfg(key=foo bar baz)]
12+
//~^ ERROR expected unsuffixed literal, found `foo`
13+
//~| HELP surround the identifier with quotation marks to parse it as a string
14+
println!();
15+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
error: expected unsuffixed literal, found `foo`
2+
--> $DIR/attr-unquoted-ident.rs:5:15
3+
|
4+
LL | #[cfg(key=foo)]
5+
| ^^^
6+
|
7+
help: surround the identifier with quotation marks to parse it as a string
8+
|
9+
LL | #[cfg(key="foo")]
10+
| + +
11+
12+
error: expected unsuffixed literal, found `foo`
13+
--> $DIR/attr-unquoted-ident.rs:11:15
14+
|
15+
LL | #[cfg(key=foo bar baz)]
16+
| ^^^
17+
|
18+
help: surround the identifier with quotation marks to parse it as a string
19+
|
20+
LL | #[cfg(key="foo bar baz")]
21+
| + +
22+
23+
error: aborting due to 2 previous errors
24+

0 commit comments

Comments
 (0)
Please sign in to comment.