Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Recover from pub let #107047

Closed
wants to merge 4 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 7 additions & 0 deletions compiler/rustc_parse/locales/en-US.ftl
Original file line number Diff line number Diff line change
Expand Up @@ -439,6 +439,13 @@ parse_visibility_not_followed_by_item = visibility `{$vis}` is not followed by a
.label = the visibility
.help = you likely meant to define an item, e.g., `{$vis} fn foo() {"{}"}`

parse_visibility_followed_by_let = visibility does not apply to `let` statement
.suggestion = remove pub

parse_invalid_let_outside_function = `let` is invalid outside of a function
.suggestion_static = consider using `static` instead of `let`
.suggestion_const = consider using `const` instead of `let`

parse_default_not_followed_by_item = `default` is not followed by an item
.label = the `default` qualifier
.note = only `fn`, `const`, `type`, or `impl` items may be prefixed by `default`
Expand Down
27 changes: 27 additions & 0 deletions compiler/rustc_parse/src/errors.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1403,6 +1403,33 @@ pub(crate) struct VisibilityNotFollowedByItem {
pub vis: Visibility,
}

#[derive(Diagnostic)]
#[diag(parse_visibility_followed_by_let)]
pub(crate) struct VisibilityFollowedByLet {
#[primary_span]
#[suggestion(code = "let", applicability = "machine-applicable")]
pub span: Span,
}

#[derive(Diagnostic)]
#[diag(parse_invalid_let_outside_function)]
pub(crate) struct InvalidLetOutsideFunction {
#[primary_span]
#[suggestion(
parse_suggestion_static,
style = "verbose",
code = "static",
applicability = "machine-applicable"
)]
#[suggestion(
parse_suggestion_const,
style = "verbose",
code = "const",
applicability = "machine-applicable"
)]
pub span: Span,
}

#[derive(Diagnostic)]
#[diag(parse_default_not_followed_by_item)]
#[note]
Expand Down
19 changes: 10 additions & 9 deletions compiler/rustc_parse/src/parser/item.rs
Original file line number Diff line number Diff line change
Expand Up @@ -68,16 +68,17 @@ impl<'a> Parser<'a> {
}

if !self.eat(term) {
let token_str = super::token_descr(&self.token);
if !self.maybe_consume_incorrect_semicolon(&items) {
let msg = &format!("expected item, found {token_str}");
let mut err = self.struct_span_err(self.token.span, msg);
let label = if self.is_kw_followed_by_ident(kw::Let) {
"consider using `const` or `static` instead of `let` for global variables"
} else {
"expected item"
};
err.span_label(self.token.span, label);
if self.is_kw_followed_by_ident(kw::Let) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This looks good.

let err = self
.sess
.create_err(errors::InvalidLetOutsideFunction { span: self.token.span });
return Err(err);
}
let token_str = super::token_descr(&self.token);
let mut err = self
.struct_span_err(self.token.span, &format!("expected item, found {token_str}"));
err.span_label(self.token.span, "expected item");
return Err(err);
}
}
Expand Down
5 changes: 5 additions & 0 deletions compiler/rustc_parse/src/parser/stmt.rs
Original file line number Diff line number Diff line change
Expand Up @@ -111,6 +111,11 @@ impl<'a> Parser<'a> {
// Do not attempt to parse an expression if we're done here.
self.error_outer_attrs(attrs);
self.mk_stmt(lo, StmtKind::Empty)
} else if self.prev_token.is_keyword(kw::Pub) && self.token.is_keyword(kw::Let) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The name of the function is parse_stmt_without_recovery, so I don't think it's a good idea to do recovery here. I think there was some kind of other function that gets called on invalid statements (I forgot the name but -Ztreat-err-as-bug may guide you there). Maybe you could check there?

let err = self.sess.create_err(errors::VisibilityFollowedByLet {
span: self.prev_token.span.to(self.token.span),
});
return Err(err);
} else if self.token != token::CloseDelim(Delimiter::Brace) {
// Remainder are line-expr stmts.
let e = match force_collect {
Expand Down
5 changes: 5 additions & 0 deletions tests/ui/parser/let-outside-function.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
let a = 1;
//~^ ERROR `let` is invalid outside of a function

fn main() {
}
17 changes: 17 additions & 0 deletions tests/ui/parser/let-outside-function.stderr
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
error: `let` is invalid outside of a function
--> $DIR/let-outside-function.rs:1:1
|
LL | let a = 1;
| ^^^
|
help: consider using `static` instead of `let`
|
LL | static a = 1;
| ~~~~~~
help: consider using `const` instead of `let`
|
LL | const a = 1;
| ~~~~~

error: aborting due to previous error

9 changes: 9 additions & 0 deletions tests/ui/parser/pub-let.fixed
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
// run-rustfix

#![allow(unused_variables)]

fn main() {
let x = 1;
//~^ ERROR visibility `pub` is not followed by an item
//~^^ ERROR visibility does not apply to `let` statement
}
9 changes: 9 additions & 0 deletions tests/ui/parser/pub-let.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
// run-rustfix

#![allow(unused_variables)]

fn main() {
pub let x = 1;
//~^ ERROR visibility `pub` is not followed by an item
//~^^ ERROR visibility does not apply to `let` statement
}
16 changes: 16 additions & 0 deletions tests/ui/parser/pub-let.stderr
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
error: visibility `pub` is not followed by an item
--> $DIR/pub-let.rs:6:5
|
LL | pub let x = 1;
| ^^^ the visibility
|
= help: you likely meant to define an item, e.g., `pub fn foo() {}`

error: visibility does not apply to `let` statement
--> $DIR/pub-let.rs:6:5
|
LL | pub let x = 1;
| ^^^^^^^ help: remove pub: `let`

error: aborting due to 2 previous errors

2 changes: 1 addition & 1 deletion tests/ui/parser/suggest-const-for-global-var.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
let X: i32 = 12;
//~^ ERROR expected item, found keyword `let`
//~^ ERROR `let` is invalid outside of a function

fn main() {
println!("{}", X);
Expand Down
13 changes: 11 additions & 2 deletions tests/ui/parser/suggest-const-for-global-var.stderr
Original file line number Diff line number Diff line change
@@ -1,8 +1,17 @@
error: expected item, found keyword `let`
error: `let` is invalid outside of a function
--> $DIR/suggest-const-for-global-var.rs:1:1
|
LL | let X: i32 = 12;
| ^^^ consider using `const` or `static` instead of `let` for global variables
| ^^^
|
help: consider using `static` instead of `let`
|
LL | static X: i32 = 12;
| ~~~~~~
help: consider using `const` instead of `let`
|
LL | const X: i32 = 12;
| ~~~~~

error: aborting due to previous error