From 0a55aacc077f26f5703035f7c5395d083db3d355 Mon Sep 17 00:00:00 2001 From: Huon Wilson Date: Fri, 16 Jan 2015 22:27:40 +1100 Subject: [PATCH] Prefer implemented traits in suggestions. If `a.method();` can't be resolved, we first look for implemented traits globally and suggest those. If there are no such traits found, we only then fall back to suggesting from the unfiltered list of traits. --- src/librustc_typeck/check/method/mod.rs | 7 +- src/librustc_typeck/check/method/probe.rs | 64 ++++++++++++++++++- src/librustc_typeck/check/method/suggest.rs | 57 +++++++++++------ .../auxiliary/no_method_suggested_traits.rs | 7 +- .../no-method-suggested-traits.rs | 46 +++++++++++-- 5 files changed, 149 insertions(+), 32 deletions(-) diff --git a/src/librustc_typeck/check/method/mod.rs b/src/librustc_typeck/check/method/mod.rs index a28b4ac3475..345bc5fd2aa 100644 --- a/src/librustc_typeck/check/method/mod.rs +++ b/src/librustc_typeck/check/method/mod.rs @@ -38,8 +38,9 @@ mod suggest; pub enum MethodError { // Did not find an applicable method, but we did find various - // static methods that may apply. - NoMatch(Vec), + // static methods that may apply, as well as a list of + // not-in-scope traits which may work. + NoMatch(Vec, Vec), // Multiple methods might apply. Ambiguity(Vec), @@ -65,7 +66,7 @@ pub fn exists<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>, { match probe::probe(fcx, span, method_name, self_ty, call_expr_id) { Ok(_) => true, - Err(NoMatch(_)) => false, + Err(NoMatch(_, _)) => false, Err(Ambiguity(_)) => true, } } diff --git a/src/librustc_typeck/check/method/probe.rs b/src/librustc_typeck/check/method/probe.rs index dc4d6c9a826..9df8875152e 100644 --- a/src/librustc_typeck/check/method/probe.rs +++ b/src/librustc_typeck/check/method/probe.rs @@ -11,6 +11,7 @@ use super::{MethodError,Ambiguity,NoMatch}; use super::MethodIndex; use super::{CandidateSource,ImplSource,TraitSource}; +use super::suggest; use check; use check::{FnCtxt, NoPreference}; @@ -25,6 +26,7 @@ use middle::infer::InferCtxt; use syntax::ast; use syntax::codemap::{Span, DUMMY_SP}; use std::collections::HashSet; +use std::mem; use std::rc::Rc; use util::ppaux::Repr; @@ -42,6 +44,7 @@ struct ProbeContext<'a, 'tcx:'a> { extension_candidates: Vec>, impl_dups: HashSet, static_candidates: Vec, + all_traits_search: bool, } struct CandidateStep<'tcx> { @@ -127,7 +130,7 @@ pub fn probe<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>, // take place in the `fcx.infcx().probe` below. let steps = match create_steps(fcx, span, self_ty) { Some(steps) => steps, - None => return Err(NoMatch(Vec::new())), + None => return Err(NoMatch(Vec::new(), Vec::new())), }; // Create a list of simplified self types, if we can. @@ -208,9 +211,17 @@ impl<'a,'tcx> ProbeContext<'a,'tcx> { steps: Rc::new(steps), opt_simplified_steps: opt_simplified_steps, static_candidates: Vec::new(), + all_traits_search: false, } } + fn reset(&mut self) { + self.inherent_candidates.clear(); + self.extension_candidates.clear(); + self.impl_dups.clear(); + self.static_candidates.clear(); + } + fn tcx(&self) -> &'a ty::ctxt<'tcx> { self.fcx.tcx() } @@ -446,6 +457,15 @@ impl<'a,'tcx> ProbeContext<'a,'tcx> { } } + fn assemble_extension_candidates_for_all_traits(&mut self) { + let mut duplicates = HashSet::new(); + for trait_info in suggest::all_traits(self.fcx.ccx) { + if duplicates.insert(trait_info.def_id) { + self.assemble_extension_candidates_for_trait(trait_info.def_id) + } + } + } + fn assemble_extension_candidates_for_trait(&mut self, trait_def_id: ast::DefId) { debug!("assemble_extension_candidates_for_trait(trait_def_id={})", @@ -715,7 +735,47 @@ impl<'a,'tcx> ProbeContext<'a,'tcx> { } } - Err(NoMatch(self.static_candidates)) + let static_candidates = mem::replace(&mut self.static_candidates, vec![]); + + let out_of_scope_traits = if !self.all_traits_search { + // things failed, and we haven't yet looked through all + // traits, so lets do that now: + self.reset(); + self.all_traits_search = true; + + let span = self.span; + let tcx = self.tcx(); + + self.assemble_extension_candidates_for_all_traits(); + + match self.pick() { + Ok(p) => vec![p.method_ty.container.id()], + Err(Ambiguity(v)) => v.into_iter().map(|source| { + match source { + TraitSource(id) => id, + ImplSource(impl_id) => { + match ty::trait_id_of_impl(tcx, impl_id) { + Some(id) => id, + None => tcx.sess.span_bug(span, + "found inherent method when looking \ + at traits") + } + } + } + }).collect(), + // it'd be really weird for this assertion to trigger, + // given the `vec![]` in the else branch below + Err(NoMatch(_, others)) => { + assert!(others.is_empty()); + vec![] + } + } + } else { + // we've just looked through all traits and didn't find + // anything at all. + vec![] + }; + Err(NoMatch(static_candidates, out_of_scope_traits)) } fn pick_step(&mut self, step: &CandidateStep<'tcx>) -> Option> { diff --git a/src/librustc_typeck/check/method/suggest.rs b/src/librustc_typeck/check/method/suggest.rs index aab1fa2a958..013c6e2f953 100644 --- a/src/librustc_typeck/check/method/suggest.rs +++ b/src/librustc_typeck/check/method/suggest.rs @@ -35,7 +35,7 @@ pub fn report_error<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>, error: MethodError) { match error { - MethodError::NoMatch(static_sources) => { + MethodError::NoMatch(static_sources, out_of_scope_traits) => { let cx = fcx.tcx(); let method_ustring = method_name.user_string(cx); @@ -75,7 +75,7 @@ pub fn report_error<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>, report_candidates(fcx, span, method_name, static_sources); } - suggest_traits_to_import(fcx, span, rcvr_ty, method_name) + suggest_traits_to_import(fcx, span, rcvr_ty, method_name, out_of_scope_traits) } MethodError::Ambiguity(sources) => { @@ -136,10 +136,35 @@ pub type AllTraitsVec = Vec; fn suggest_traits_to_import<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>, span: Span, _rcvr_ty: Ty<'tcx>, - method_name: ast::Name) + method_name: ast::Name, + valid_out_of_scope_traits: Vec) { let tcx = fcx.tcx(); + let method_ustring = method_name.user_string(tcx); + if !valid_out_of_scope_traits.is_empty() { + let mut candidates = valid_out_of_scope_traits; + candidates.sort(); + let msg = format!( + "methods from traits can only be called if the trait is in scope; \ + the following {traits_are} implemented and {define} a method `{name}`:", + traits_are = if candidates.len() == 1 {"trait is"} else {"traits are"}, + define = if candidates.len() == 1 {"defines"} else {"define"}, + name = method_ustring); + + fcx.sess().fileline_help(span, &msg[]); + + for (i, trait_did) in candidates.iter().enumerate() { + fcx.sess().fileline_help(span, + &*format!("candidate #{}: `{}`", + i + 1, + ty::item_path_str(fcx.tcx(), *trait_did))) + + } + return + } + + // there's no implemented traits, so lets suggest some traits to implement let mut candidates = all_traits(fcx.ccx) .filter(|info| trait_method(tcx, info.def_id, method_name).is_some()) .collect::>(); @@ -148,22 +173,16 @@ fn suggest_traits_to_import<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>, // sort from most relevant to least relevant candidates.sort_by(|a, b| a.cmp(b).reverse()); - let method_ustring = method_name.user_string(tcx); + let msg = format!( + "methods from traits can only be called if the trait is implemented and \ + in scope; no such traits are but the following {traits_define} a method `{name}`:", + traits_define = if candidates.len() == 1 {"trait defines"} else {"traits define"}, + name = method_ustring); - span_help!(fcx.sess(), span, - "methods from traits can only be called if the trait is implemented \ - and in scope; the following trait{s} define{inv_s} a method `{name}`:", - s = if candidates.len() == 1 {""} else {"s"}, - inv_s = if candidates.len() == 1 {"s"} else {""}, - name = method_ustring); + fcx.sess().fileline_help(span, &msg[]); for (i, trait_info) in candidates.iter().enumerate() { - // provide a good-as-possible span; the span of - // the trait if it is local, or the span of the - // method call itself if not - let trait_span = fcx.tcx().map.def_id_span(trait_info.def_id, span); - - fcx.sess().fileline_help(trait_span, + fcx.sess().fileline_help(span, &*format!("candidate #{}: `{}`", i + 1, ty::item_path_str(fcx.tcx(), trait_info.def_id))) @@ -173,7 +192,7 @@ fn suggest_traits_to_import<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>, #[derive(Copy)] pub struct TraitInfo { - def_id: ast::DefId, + pub def_id: ast::DefId, } impl TraitInfo { @@ -206,7 +225,7 @@ impl Ord for TraitInfo { } /// Retrieve all traits in this crate and any dependent crates. -fn all_traits<'a>(ccx: &'a CrateCtxt) -> AllTraits<'a> { +pub fn all_traits<'a>(ccx: &'a CrateCtxt) -> AllTraits<'a> { if ccx.all_traits.borrow().is_none() { use syntax::visit; @@ -268,7 +287,7 @@ fn all_traits<'a>(ccx: &'a CrateCtxt) -> AllTraits<'a> { } } -struct AllTraits<'a> { +pub struct AllTraits<'a> { borrow: cell::Ref<'a Option>, idx: usize } diff --git a/src/test/auxiliary/no_method_suggested_traits.rs b/src/test/auxiliary/no_method_suggested_traits.rs index fdeace00d4c..328561495ee 100644 --- a/src/test/auxiliary/no_method_suggested_traits.rs +++ b/src/test/auxiliary/no_method_suggested_traits.rs @@ -12,8 +12,13 @@ pub use reexport::Reexported; pub mod foo { pub trait PubPub { - fn method(&self); + fn method(&self) {} + + fn method3(&self) {} } + + impl PubPub for u32 {} + impl PubPub for i32 {} } pub mod bar { trait PubPriv { diff --git a/src/test/compile-fail/no-method-suggested-traits.rs b/src/test/compile-fail/no-method-suggested-traits.rs index 2565d2b7302..277800778a8 100644 --- a/src/test/compile-fail/no-method-suggested-traits.rs +++ b/src/test/compile-fail/no-method-suggested-traits.rs @@ -13,18 +13,50 @@ extern crate no_method_suggested_traits; mod foo { - trait Bar { //~ HELP `foo::Bar` - fn method(&self); + trait Bar { + fn method(&self) {} + + fn method2(&self) {} } + + impl Bar for u32 {} + + impl Bar for char {} } fn main() { 1u32.method(); //~^ ERROR does not implement - //~^^ HELP the following traits define a method `method` + //~^^ HELP the following traits are implemented and define a method `method` + //~^^^ HELP `foo::Bar` + //~^^^^ HELP `no_method_suggested_traits::foo::PubPub` + + 'a'.method(); + //~^ ERROR does not implement + //~^^ HELP the following trait is implemented and defines a method `method` + //~^^^ HELP `foo::Bar` + + 1i32.method(); + //~^ ERROR does not implement + //~^^ HELP the following trait is implemented and defines a method `method` + //~^^^ HELP `no_method_suggested_traits::foo::PubPub` + + 1u64.method(); + //~^ ERROR does not implement + //~^^ HELP the following traits define a method `method` + //~^^^ HELP `foo::Bar` + //~^^^^ HELP `no_method_suggested_traits::foo::PubPub` + //~^^^^^ HELP `no_method_suggested_traits::reexport::Reexported` + //~^^^^^^ HELP `no_method_suggested_traits::bar::PubPriv` + //~^^^^^^^ HELP `no_method_suggested_traits::qux::PrivPub` + //~^^^^^^^^ HELP `no_method_suggested_traits::quz::PrivPriv` + + 1u64.method2(); + //~^ ERROR does not implement + //~^^ HELP the following trait defines a method `method2` + //~^^^ HELP `foo::Bar` + 1u64.method3(); + //~^ ERROR does not implement + //~^^ HELP the following trait defines a method `method3` //~^^^ HELP `no_method_suggested_traits::foo::PubPub` - //~^^^^ HELP `no_method_suggested_traits::reexport::Reexported` - //~^^^^^ HELP `no_method_suggested_traits::bar::PubPriv` - //~^^^^^^ HELP `no_method_suggested_traits::qux::PrivPub` - //~^^^^^^^ HELP `no_method_suggested_traits::quz::PrivPriv` }