Skip to content

Commit b7930d9

Browse files
committed
Auto merge of #21187 - oli-obk:feature/hint_struct_field_access, r=alexcrichton
rebase and fix of #19267
2 parents 5d2056a + 09d9924 commit b7930d9

File tree

4 files changed

+112
-1
lines changed

4 files changed

+112
-1
lines changed

src/librustc_typeck/check/mod.rs

+41-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT
1+
// Copyright 2012-2015 The Rust Project Developers. See the COPYRIGHT
22
// file at the top-level directory of this distribution and at
33
// http://rust-lang.org/COPYRIGHT.
44
//
@@ -108,6 +108,7 @@ use lint;
108108
use util::common::{block_query, indenter, loop_query};
109109
use util::ppaux::{self, Repr};
110110
use util::nodemap::{DefIdMap, FnvHashMap, NodeMap};
111+
use util::lev_distance::lev_distance;
111112

112113
use std::cell::{Cell, Ref, RefCell};
113114
use std::mem::replace;
@@ -3071,11 +3072,43 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
30713072
actual)
30723073
},
30733074
expr_t, None);
3075+
if let Some(t) = ty::ty_to_def_id(expr_t) {
3076+
suggest_field_names(t, field, tcx, vec![]);
3077+
}
30743078
}
30753079

30763080
fcx.write_error(expr.id);
30773081
}
30783082

3083+
// displays hints about the closest matches in field names
3084+
fn suggest_field_names<'tcx>(id : DefId,
3085+
field : &ast::SpannedIdent,
3086+
tcx : &ty::ctxt<'tcx>,
3087+
skip : Vec<&str>) {
3088+
let ident = token::get_ident(field.node);
3089+
let name = ident.get();
3090+
// only find fits with at least one matching letter
3091+
let mut best_dist = name.len();
3092+
let mut best = None;
3093+
let fields = ty::lookup_struct_fields(tcx, id);
3094+
for elem in fields.iter() {
3095+
let n = elem.name.as_str();
3096+
// ignore already set fields
3097+
if skip.iter().any(|&x| x == n) {
3098+
continue;
3099+
}
3100+
let dist = lev_distance(n, name);
3101+
if dist < best_dist {
3102+
best = Some(n);
3103+
best_dist = dist;
3104+
}
3105+
}
3106+
if let Some(n) = best {
3107+
tcx.sess.span_help(field.span,
3108+
format!("did you mean `{}`?", n).as_slice());
3109+
}
3110+
}
3111+
30793112
// Check tuple index expressions
30803113
fn check_tup_field(fcx: &FnCtxt,
30813114
expr: &ast::Expr,
@@ -3183,6 +3216,13 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
31833216
},
31843217
struct_ty,
31853218
None);
3219+
// prevent all specified fields from being suggested
3220+
let skip_fields = ast_fields.iter().map(|ref x| x.ident.node.name.as_str());
3221+
let actual_id = match enum_id_opt {
3222+
Some(_) => class_id,
3223+
None => ty::ty_to_def_id(struct_ty).unwrap()
3224+
};
3225+
suggest_field_names(actual_id, &field.ident, tcx, skip_fields.collect());
31863226
error_happened = true;
31873227
}
31883228
Some((_, true)) => {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
// Copyright 2015 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+
struct A {
12+
foo : i32,
13+
car : i32,
14+
barr : i32
15+
}
16+
17+
fn main() {
18+
let a = A {
19+
foo : 5,
20+
bar : 42,//~ ERROR structure `A` has no field named `bar`
21+
//~^ HELP did you mean `barr`?
22+
car : 9,
23+
};
24+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
// Copyright 2015 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+
struct A {
12+
foo : i32,
13+
car : i32,
14+
barr : i32
15+
}
16+
17+
fn main() {
18+
let a = A {
19+
foo : 5,
20+
bar : 42,//~ ERROR structure `A` has no field named `bar`
21+
//~^ HELP did you mean `car`?
22+
};
23+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
// Copyright 2012 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+
struct BuildData {
12+
foo: isize,
13+
bar: f32
14+
}
15+
16+
fn main() {
17+
let foo = BuildData {
18+
foo: 0,
19+
bar: 0.5,
20+
};
21+
let x = foo.baa;//~ ERROR attempted access of field `baa` on type `BuildData`
22+
//~^ HELP did you mean `bar`?
23+
println!("{}", x);
24+
}

0 commit comments

Comments
 (0)