1
Fork 0

Rollup merge of #56248 - estebank:suggest-bare-pub, r=petrochenkov

Suggest an appropriate token when encountering `pub Ident<'a>`

Fix #55403. Follow up to #45997.
This commit is contained in:
Mazdak Farrokhzad 2018-12-08 08:43:43 +01:00 committed by GitHub
commit ac15b4f4bd
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
18 changed files with 124 additions and 31 deletions

View file

@ -5811,20 +5811,14 @@ impl<'a> Parser<'a> {
} }
fn complain_if_pub_macro(&mut self, vis: &VisibilityKind, sp: Span) { fn complain_if_pub_macro(&mut self, vis: &VisibilityKind, sp: Span) {
if let Err(mut err) = self.complain_if_pub_macro_diag(vis, sp) {
err.emit();
}
}
fn complain_if_pub_macro_diag(&mut self, vis: &VisibilityKind, sp: Span) -> PResult<'a, ()> {
match *vis { match *vis {
VisibilityKind::Inherited => Ok(()), VisibilityKind::Inherited => {}
_ => { _ => {
let is_macro_rules: bool = match self.token { let is_macro_rules: bool = match self.token {
token::Ident(sid, _) => sid.name == Symbol::intern("macro_rules"), token::Ident(sid, _) => sid.name == Symbol::intern("macro_rules"),
_ => false, _ => false,
}; };
if is_macro_rules { let mut err = if is_macro_rules {
let mut err = self.diagnostic() let mut err = self.diagnostic()
.struct_span_err(sp, "can't qualify macro_rules invocation with `pub`"); .struct_span_err(sp, "can't qualify macro_rules invocation with `pub`");
err.span_suggestion_with_applicability( err.span_suggestion_with_applicability(
@ -5833,13 +5827,14 @@ impl<'a> Parser<'a> {
"#[macro_export]".to_owned(), "#[macro_export]".to_owned(),
Applicability::MaybeIncorrect // speculative Applicability::MaybeIncorrect // speculative
); );
Err(err) err
} else { } else {
let mut err = self.diagnostic() let mut err = self.diagnostic()
.struct_span_err(sp, "can't qualify macro invocation with `pub`"); .struct_span_err(sp, "can't qualify macro invocation with `pub`");
err.help("try adjusting the macro to put `pub` inside the invocation"); err.help("try adjusting the macro to put `pub` inside the invocation");
Err(err) err
} };
err.emit();
} }
} }
} }
@ -6148,9 +6143,6 @@ impl<'a> Parser<'a> {
fn consume_block(&mut self, delim: token::DelimToken) { fn consume_block(&mut self, delim: token::DelimToken) {
let mut brace_depth = 0; let mut brace_depth = 0;
if !self.eat(&token::OpenDelim(delim)) {
return;
}
loop { loop {
if self.eat(&token::OpenDelim(delim)) { if self.eat(&token::OpenDelim(delim)) {
brace_depth += 1; brace_depth += 1;
@ -6161,7 +6153,7 @@ impl<'a> Parser<'a> {
brace_depth -= 1; brace_depth -= 1;
continue; continue;
} }
} else if self.eat(&token::Eof) || self.eat(&token::CloseDelim(token::NoDelim)) { } else if self.token == token::Eof || self.eat(&token::CloseDelim(token::NoDelim)) {
return; return;
} else { } else {
self.bump(); self.bump();
@ -7410,17 +7402,27 @@ impl<'a> Parser<'a> {
return Err(err); return Err(err);
} else if self.look_ahead(1, |t| *t == token::OpenDelim(token::Paren)) { } else if self.look_ahead(1, |t| *t == token::OpenDelim(token::Paren)) {
let ident = self.parse_ident().unwrap(); let ident = self.parse_ident().unwrap();
self.bump(); // `(`
let kw_name = if let Ok(Some(_)) = self.parse_self_arg() {
"method"
} else {
"function"
};
self.consume_block(token::Paren); self.consume_block(token::Paren);
let (kw, kw_name, ambiguous) = if self.check(&token::RArrow) || let (kw, kw_name, ambiguous) = if self.check(&token::RArrow) {
self.check(&token::OpenDelim(token::Brace)) self.eat_to_tokens(&[&token::OpenDelim(token::Brace)]);
{ self.bump(); // `{`
("fn", "method", false) ("fn", kw_name, false)
} else if self.check(&token::OpenDelim(token::Brace)) {
self.bump(); // `{`
("fn", kw_name, false)
} else if self.check(&token::Colon) { } else if self.check(&token::Colon) {
let kw = "struct"; let kw = "struct";
(kw, kw, false) (kw, kw, false)
} else { } else {
("fn` or `struct", "method or struct", true) ("fn` or `struct", "function or struct", true)
}; };
self.consume_block(token::Brace);
let msg = format!("missing `{}` for {} definition", kw, kw_name); let msg = format!("missing `{}` for {} definition", kw, kw_name);
let mut err = self.diagnostic().struct_span_err(sp, &msg); let mut err = self.diagnostic().struct_span_err(sp, &msg);
@ -7447,6 +7449,32 @@ impl<'a> Parser<'a> {
} }
} }
return Err(err); return Err(err);
} else if self.look_ahead(1, |t| *t == token::Lt) {
let ident = self.parse_ident().unwrap();
self.eat_to_tokens(&[&token::Gt]);
self.bump(); // `>`
let (kw, kw_name, ambiguous) = if self.eat(&token::OpenDelim(token::Paren)) {
if let Ok(Some(_)) = self.parse_self_arg() {
("fn", "method", false)
} else {
("fn", "function", false)
}
} else if self.check(&token::OpenDelim(token::Brace)) {
("struct", "struct", false)
} else {
("fn` or `struct", "function or struct", true)
};
let msg = format!("missing `{}` for {} definition", kw, kw_name);
let mut err = self.diagnostic().struct_span_err(sp, &msg);
if !ambiguous {
err.span_suggestion_short_with_applicability(
sp,
&format!("add `{}` here to parse `{}` as a public {}", kw, ident, kw_name),
format!(" {} ", kw),
Applicability::MachineApplicable,
);
}
return Err(err);
} }
} }
self.parse_macro_use_or_failure(attrs, macros_allowed, attributes_allowed, lo, visibility) self.parse_macro_use_or_failure(attrs, macros_allowed, attributes_allowed, lo, visibility)

View file

@ -9,7 +9,7 @@
// except according to those terms. // except according to those terms.
pub foo(s: usize) { bar() } pub foo(s: usize) { bar() }
//~^ ERROR missing `fn` for method definition //~^ ERROR missing `fn` for function definition
fn main() { fn main() {
foo(2); foo(2);

View file

@ -1,9 +1,9 @@
error: missing `fn` for method definition error: missing `fn` for function definition
--> $DIR/pub-ident-fn-2.rs:11:4 --> $DIR/pub-ident-fn-2.rs:11:4
| |
LL | pub foo(s: usize) { bar() } LL | pub foo(s: usize) { bar() }
| ^ | ^
help: add `fn` here to parse `foo` as a public method help: add `fn` here to parse `foo` as a public function
| |
LL | pub fn foo(s: usize) { bar() } LL | pub fn foo(s: usize) { bar() }
| ^^ | ^^

View file

@ -9,6 +9,6 @@
// except according to those terms. // except according to those terms.
pub S(); pub S();
//~^ ERROR missing `fn` or `struct` for method or struct definition //~^ ERROR missing `fn` or `struct` for function or struct definition
fn main() {} fn main() {}

View file

@ -1,4 +1,4 @@
error: missing `fn` or `struct` for method or struct definition error: missing `fn` or `struct` for function or struct definition
--> $DIR/pub-ident-fn-or-struct-2.rs:11:4 --> $DIR/pub-ident-fn-or-struct-2.rs:11:4
| |
LL | pub S(); LL | pub S();

View file

@ -9,6 +9,6 @@
// except according to those terms. // except according to those terms.
pub S (foo) bar pub S (foo) bar
//~^ ERROR missing `fn` or `struct` for method or struct definition //~^ ERROR missing `fn` or `struct` for function or struct definition
fn main() {} fn main() {}

View file

@ -1,4 +1,4 @@
error: missing `fn` or `struct` for method or struct definition error: missing `fn` or `struct` for function or struct definition
--> $DIR/pub-ident-fn-or-struct.rs:11:4 --> $DIR/pub-ident-fn-or-struct.rs:11:4
| |
LL | pub S (foo) bar LL | pub S (foo) bar

View file

@ -0,0 +1,6 @@
pub bar<'a>(&self, _s: &'a usize) -> bool { true }
//~^ ERROR missing `fn` for method definition
fn main() {
bar(2);
}

View file

@ -0,0 +1,12 @@
error: missing `fn` for method definition
--> $DIR/pub-ident-fn-with-lifetime-2.rs:1:4
|
LL | pub bar<'a>(&self, _s: &'a usize) -> bool { true }
| ^^^
help: add `fn` here to parse `bar` as a public method
|
LL | pub fn bar<'a>(&self, _s: &'a usize) -> bool { true }
| ^^
error: aborting due to previous error

View file

@ -0,0 +1,6 @@
pub foo<'a>(_s: &'a usize) -> bool { true }
//~^ ERROR missing `fn` for function definition
fn main() {
foo(2);
}

View file

@ -0,0 +1,12 @@
error: missing `fn` for function definition
--> $DIR/pub-ident-fn-with-lifetime.rs:1:4
|
LL | pub foo<'a>(_s: &'a usize) -> bool { true }
| ^^^
help: add `fn` here to parse `foo` as a public function
|
LL | pub fn foo<'a>(_s: &'a usize) -> bool { true }
| ^^
error: aborting due to previous error

View file

@ -11,7 +11,7 @@
// run-rustfix // run-rustfix
pub fn foo(_s: usize) -> bool { true } pub fn foo(_s: usize) -> bool { true }
//~^ ERROR missing `fn` for method definition //~^ ERROR missing `fn` for function definition
fn main() { fn main() {
foo(2); foo(2);

View file

@ -11,7 +11,7 @@
// run-rustfix // run-rustfix
pub foo(_s: usize) -> bool { true } pub foo(_s: usize) -> bool { true }
//~^ ERROR missing `fn` for method definition //~^ ERROR missing `fn` for function definition
fn main() { fn main() {
foo(2); foo(2);

View file

@ -1,9 +1,9 @@
error: missing `fn` for method definition error: missing `fn` for function definition
--> $DIR/pub-ident-fn.rs:13:4 --> $DIR/pub-ident-fn.rs:13:4
| |
LL | pub foo(_s: usize) -> bool { true } LL | pub foo(_s: usize) -> bool { true }
| ^^^ | ^^^
help: add `fn` here to parse `foo` as a public method help: add `fn` here to parse `foo` as a public function
| |
LL | pub fn foo(_s: usize) -> bool { true } LL | pub fn foo(_s: usize) -> bool { true }
| ^^ | ^^

View file

@ -0,0 +1,4 @@
pub S<'a> {
//~^ ERROR missing `struct` for struct definition
}
fn main() {}

View file

@ -0,0 +1,12 @@
error: missing `struct` for struct definition
--> $DIR/pub-ident-struct-with-lifetime.rs:1:4
|
LL | pub S<'a> {
| ^
help: add `struct` here to parse `S` as a public struct
|
LL | pub struct S<'a> {
| ^^^^^^
error: aborting due to previous error

View file

@ -0,0 +1,5 @@
fn main() {
}
pub foo<'a>
//~^ ERROR missing `fn` or `struct` for function or struct definition

View file

@ -0,0 +1,8 @@
error: missing `fn` or `struct` for function or struct definition
--> $DIR/pub-ident-with-lifetime-incomplete.rs:4:4
|
LL | pub foo<'a>
| ^^^
error: aborting due to previous error