Skip to content

Commit b946ecd

Browse files
committed
Suggest using enum when a variant is used as a type
Given a file: ```rust enum Fruit { Apple(i64), Orange(i64), } fn should_return_fruit() -> Apple { Apple(5) } ``` Provide the following output: ```rust error[E0412]: cannot find type `Apple` in this scope --> file.rs:16:29 | 16 | fn should_return_fruit() -> Apple { | ^^^^^ not found in this scope | help: there is an enum variant `Fruit::Apple`, did you mean to use `Fruit`? --> file.rs:12:5 | 12 | Apple(i64), | ^^^^^^^^^^ error[E0425]: cannot find function `Apple` in this scope --> file.rs:17:5 | 17 | Apple(5) | ^^^^^ not found in this scope | = help: possible candidate is found in another module, you can import it into scope: `use Fruit::Apple;` ```
1 parent 5e122f5 commit b946ecd

File tree

3 files changed

+148
-0
lines changed

3 files changed

+148
-0
lines changed

src/librustc_resolve/lib.rs

+30
Original file line numberDiff line numberDiff line change
@@ -2220,6 +2220,7 @@ impl<'a> Resolver<'a> {
22202220
-> PathResolution {
22212221
let ns = source.namespace();
22222222
let is_expected = &|def| source.is_expected(def);
2223+
let is_enum_variant = &|def| if let Def::Variant(..) = def { true } else { false };
22232224

22242225
// Base error is amended with one short label and possibly some longer helps/notes.
22252226
let report_errors = |this: &mut Self, def: Option<Def>| {
@@ -2270,6 +2271,19 @@ impl<'a> Resolver<'a> {
22702271
if !candidates.is_empty() {
22712272
// Report import candidates as help and proceed searching for labels.
22722273
show_candidates(&mut err, &candidates, def.is_some());
2274+
} else if is_expected(Def::Enum(DefId::local(CRATE_DEF_INDEX))) {
2275+
let enum_candidates = this.lookup_import_candidates(name, ns, is_enum_variant);
2276+
for suggestion in enum_candidates {
2277+
let (variant_path, enum_path) = import_candidate_to_paths(&suggestion);
2278+
let msg = format!("there is an enum variant `{}`, did you mean to use `{}`?",
2279+
variant_path,
2280+
enum_path);
2281+
if suggestion.path.span == DUMMY_SP {
2282+
err.help(&msg);
2283+
} else {
2284+
err.span_help(suggestion.path.span, &msg);
2285+
}
2286+
}
22732287
}
22742288
if path.len() == 1 && this.self_type_is_available() {
22752289
if let Some(candidate) = this.lookup_assoc_candidate(name, ns, is_expected) {
@@ -3422,6 +3436,22 @@ fn path_names_to_string(path: &Path) -> String {
34223436
names_to_string(&path.segments.iter().map(|seg| seg.identifier).collect::<Vec<_>>())
34233437
}
34243438

3439+
/// Get the path for an enum and the variant from an `ImportSuggestion` for an enum variant.
3440+
fn import_candidate_to_paths(suggestion: &ImportSuggestion) -> (String, String) {
3441+
let variant_path = &suggestion.path;
3442+
let variant_path_string = path_names_to_string(variant_path);
3443+
3444+
let path_len = suggestion.path.segments.len();
3445+
let enum_path = ast::Path {
3446+
span: suggestion.path.span,
3447+
segments: suggestion.path.segments[0..path_len - 1].to_vec(),
3448+
};
3449+
let enum_path_string = path_names_to_string(&enum_path);
3450+
3451+
(variant_path_string, enum_path_string)
3452+
}
3453+
3454+
34253455
/// When an entity with a given name is not available in scope, we search for
34263456
/// entities with that name in all crates. This method allows outputting the
34273457
/// results of this search in a programmer-friendly way
+44
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
// Copyright 2017 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
enum Fruit {
12+
Apple(i64),
13+
Orange(i64),
14+
}
15+
16+
fn should_return_fruit() -> Apple {
17+
Apple(5)
18+
}
19+
20+
fn should_return_fruit_too() -> Fruit::Apple {
21+
Apple(5)
22+
}
23+
24+
fn foo() -> Ok {
25+
Ok(())
26+
}
27+
28+
fn bar() -> Variant3 {
29+
}
30+
31+
fn qux() -> Some {
32+
Some(1)
33+
}
34+
35+
fn main() {}
36+
37+
mod x {
38+
enum Enum {
39+
Variant1,
40+
Variant2(),
41+
Variant3(usize),
42+
Variant4 {},
43+
}
44+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
error[E0412]: cannot find type `Apple` in this scope
2+
--> $DIR/issue-35675.rs:16:29
3+
|
4+
16 | fn should_return_fruit() -> Apple {
5+
| ^^^^^ not found in this scope
6+
|
7+
help: there is an enum variant `Fruit::Apple`, did you mean to use `Fruit`?
8+
--> $DIR/issue-35675.rs:12:5
9+
|
10+
12 | Apple(i64),
11+
| ^^^^^^^^^^
12+
13+
error[E0425]: cannot find function `Apple` in this scope
14+
--> $DIR/issue-35675.rs:17:5
15+
|
16+
17 | Apple(5)
17+
| ^^^^^ not found in this scope
18+
|
19+
= help: possible candidate is found in another module, you can import it into scope:
20+
`use Fruit::Apple;`
21+
22+
error[E0573]: expected type, found variant `Fruit::Apple`
23+
--> $DIR/issue-35675.rs:20:33
24+
|
25+
20 | fn should_return_fruit_too() -> Fruit::Apple {
26+
| ^^^^^^^^^^^^ not a type
27+
|
28+
help: there is an enum variant `Fruit::Apple`, did you mean to use `Fruit`?
29+
--> $DIR/issue-35675.rs:12:5
30+
|
31+
12 | Apple(i64),
32+
| ^^^^^^^^^^
33+
34+
error[E0425]: cannot find function `Apple` in this scope
35+
--> $DIR/issue-35675.rs:21:5
36+
|
37+
21 | Apple(5)
38+
| ^^^^^ not found in this scope
39+
|
40+
= help: possible candidate is found in another module, you can import it into scope:
41+
`use Fruit::Apple;`
42+
43+
error[E0573]: expected type, found variant `Ok`
44+
--> $DIR/issue-35675.rs:24:13
45+
|
46+
24 | fn foo() -> Ok {
47+
| ^^ not a type
48+
|
49+
= help: there is an enum variant `std::result::Result::Ok`, did you mean to use `std::result::Result`?
50+
= help: there is an enum variant `std::prelude::v1::Ok`, did you mean to use `std::prelude::v1`?
51+
52+
error[E0412]: cannot find type `Variant3` in this scope
53+
--> $DIR/issue-35675.rs:28:13
54+
|
55+
28 | fn bar() -> Variant3 {
56+
| ^^^^^^^^ not found in this scope
57+
|
58+
help: there is an enum variant `x::Enum::Variant3`, did you mean to use `x::Enum`?
59+
--> $DIR/issue-35675.rs:41:9
60+
|
61+
41 | Variant3(usize),
62+
| ^^^^^^^^^^^^^^^
63+
64+
error[E0573]: expected type, found variant `Some`
65+
--> $DIR/issue-35675.rs:31:13
66+
|
67+
31 | fn qux() -> Some {
68+
| ^^^^ not a type
69+
|
70+
= help: there is an enum variant `std::prelude::v1::Some`, did you mean to use `std::prelude::v1`?
71+
= help: there is an enum variant `std::prelude::v1::Option::Some`, did you mean to use `std::prelude::v1::Option`?
72+
73+
error: aborting due to 7 previous errors
74+

0 commit comments

Comments
 (0)