Fix generic type substitution in impl trait with assoc type

This commit is contained in:
Petr Nevyhoštěný 2021-12-22 19:48:14 +01:00
parent 0add6e95e5
commit d9b3242bcd
3 changed files with 411 additions and 11 deletions

View file

@ -942,4 +942,310 @@ impl FooB for Foo {
"#, "#,
) )
} }
#[test]
fn test_assoc_type_when_trait_with_same_name_in_scope() {
check_assist(
add_missing_impl_members,
r#"
pub trait Foo {}
pub trait Types {
type Foo;
}
pub trait Behavior<T: Types> {
fn reproduce(&self, foo: T::Foo);
}
pub struct Impl;
impl<T: Types> Behavior<T> for Impl { $0 }"#,
r#"
pub trait Foo {}
pub trait Types {
type Foo;
}
pub trait Behavior<T: Types> {
fn reproduce(&self, foo: T::Foo);
}
pub struct Impl;
impl<T: Types> Behavior<T> for Impl {
fn reproduce(&self, foo: <T as Types>::Foo) {
${0:todo!()}
}
}"#,
);
}
#[test]
fn test_assoc_type_on_concrete_type() {
check_assist(
add_missing_impl_members,
r#"
pub trait Types {
type Foo;
}
impl Types for u32 {
type Foo = bool;
}
pub trait Behavior<T: Types> {
fn reproduce(&self, foo: T::Foo);
}
pub struct Impl;
impl Behavior<u32> for Impl { $0 }"#,
r#"
pub trait Types {
type Foo;
}
impl Types for u32 {
type Foo = bool;
}
pub trait Behavior<T: Types> {
fn reproduce(&self, foo: T::Foo);
}
pub struct Impl;
impl Behavior<u32> for Impl {
fn reproduce(&self, foo: <u32 as Types>::Foo) {
${0:todo!()}
}
}"#,
);
}
#[test]
fn test_assoc_type_on_concrete_type_qualified() {
check_assist(
add_missing_impl_members,
r#"
pub trait Types {
type Foo;
}
impl Types for std::string::String {
type Foo = bool;
}
pub trait Behavior<T: Types> {
fn reproduce(&self, foo: T::Foo);
}
pub struct Impl;
impl Behavior<std::string::String> for Impl { $0 }"#,
r#"
pub trait Types {
type Foo;
}
impl Types for std::string::String {
type Foo = bool;
}
pub trait Behavior<T: Types> {
fn reproduce(&self, foo: T::Foo);
}
pub struct Impl;
impl Behavior<std::string::String> for Impl {
fn reproduce(&self, foo: <std::string::String as Types>::Foo) {
${0:todo!()}
}
}"#,
);
}
#[test]
fn test_assoc_type_on_concrete_type_multi_option_ambiguous() {
check_assist(
add_missing_impl_members,
r#"
pub trait Types {
type Foo;
}
pub trait Types2 {
type Foo;
}
impl Types for u32 {
type Foo = bool;
}
impl Types2 for u32 {
type Foo = String;
}
pub trait Behavior<T: Types + Types2> {
fn reproduce(&self, foo: <T as Types2>::Foo);
}
pub struct Impl;
impl Behavior<u32> for Impl { $0 }"#,
r#"
pub trait Types {
type Foo;
}
pub trait Types2 {
type Foo;
}
impl Types for u32 {
type Foo = bool;
}
impl Types2 for u32 {
type Foo = String;
}
pub trait Behavior<T: Types + Types2> {
fn reproduce(&self, foo: <T as Types2>::Foo);
}
pub struct Impl;
impl Behavior<u32> for Impl {
fn reproduce(&self, foo: <u32 as Types2>::Foo) {
${0:todo!()}
}
}"#,
);
}
#[test]
fn test_assoc_type_on_concrete_type_multi_option() {
check_assist(
add_missing_impl_members,
r#"
pub trait Types {
type Foo;
}
pub trait Types2 {
type Bar;
}
impl Types for u32 {
type Foo = bool;
}
impl Types2 for u32 {
type Bar = String;
}
pub trait Behavior<T: Types + Types2> {
fn reproduce(&self, foo: T::Bar);
}
pub struct Impl;
impl Behavior<u32> for Impl { $0 }"#,
r#"
pub trait Types {
type Foo;
}
pub trait Types2 {
type Bar;
}
impl Types for u32 {
type Foo = bool;
}
impl Types2 for u32 {
type Bar = String;
}
pub trait Behavior<T: Types + Types2> {
fn reproduce(&self, foo: T::Bar);
}
pub struct Impl;
impl Behavior<u32> for Impl {
fn reproduce(&self, foo: <u32 as Types2>::Bar) {
${0:todo!()}
}
}"#,
);
}
#[test]
fn test_assoc_type_on_concrete_type_multi_option_foreign() {
check_assist(
add_missing_impl_members,
r#"
mod bar {
pub trait Types2 {
type Bar;
}
}
pub trait Types {
type Foo;
}
impl Types for u32 {
type Foo = bool;
}
impl bar::Types2 for u32 {
type Bar = String;
}
pub trait Behavior<T: Types + bar::Types2> {
fn reproduce(&self, foo: T::Bar);
}
pub struct Impl;
impl Behavior<u32> for Impl { $0 }"#,
r#"
mod bar {
pub trait Types2 {
type Bar;
}
}
pub trait Types {
type Foo;
}
impl Types for u32 {
type Foo = bool;
}
impl bar::Types2 for u32 {
type Bar = String;
}
pub trait Behavior<T: Types + bar::Types2> {
fn reproduce(&self, foo: T::Bar);
}
pub struct Impl;
impl Behavior<u32> for Impl {
fn reproduce(&self, foo: <u32 as bar::Types2>::Bar) {
${0:todo!()}
}
}"#,
);
}
} }

View file

@ -118,14 +118,20 @@ struct Ctx<'a> {
impl<'a> Ctx<'a> { impl<'a> Ctx<'a> {
fn apply(&self, item: &SyntaxNode) { fn apply(&self, item: &SyntaxNode) {
for event in item.preorder() { // `transform_path` may update a node's parent and that would break the
let node = match event { // tree traversal. Thus all paths in the tree are collected into a vec
syntax::WalkEvent::Enter(_) => continue, // so that such operation is safe.
syntax::WalkEvent::Leave(it) => it, let paths = item
}; .preorder()
if let Some(path) = ast::Path::cast(node.clone()) { .filter_map(|event| match event {
self.transform_path(path); syntax::WalkEvent::Enter(_) => None,
} syntax::WalkEvent::Leave(node) => Some(node),
})
.filter_map(ast::Path::cast)
.collect::<Vec<_>>();
for path in paths {
self.transform_path(path);
} }
} }
fn transform_path(&self, path: ast::Path) -> Option<()> { fn transform_path(&self, path: ast::Path) -> Option<()> {
@ -145,10 +151,60 @@ impl<'a> Ctx<'a> {
match resolution { match resolution {
hir::PathResolution::TypeParam(tp) => { hir::PathResolution::TypeParam(tp) => {
if let Some(subst) = self.substs.get(&tp) { if let Some(subst) = self.substs.get(&tp) {
ted::replace(path.syntax(), subst.clone_subtree().clone_for_update().syntax()) let parent = path.syntax().parent()?;
if let Some(parent) = ast::Path::cast(parent.clone()) {
// Path inside path means that there is an associated
// type on the type parameter. It is necessary to fully
// qualify the type with `as Trait`. Even though it
// might be unnecessary if `subst` is generic type,
// always fully qualifying the path is safer because of
// potential clash of associated types from multiple
// traits
let trait_ref = find_trait_for_assoc_type(
self.source_scope,
tp,
parent.segment()?.name_ref()?,
)
.and_then(|trait_ref| {
let found_path = self.target_module.find_use_path(
self.source_scope.db.upcast(),
hir::ModuleDef::Trait(trait_ref),
)?;
match ast::make::ty_path(mod_path_to_ast(&found_path)) {
ast::Type::PathType(path_ty) => Some(path_ty),
_ => None,
}
});
let segment = ast::make::path_segment_ty(subst.clone(), trait_ref);
let qualified =
ast::make::path_from_segments(std::iter::once(segment), false);
ted::replace(path.syntax(), qualified.clone_for_update().syntax());
} else if let Some(path_ty) = ast::PathType::cast(parent) {
ted::replace(
path_ty.syntax(),
subst.clone_subtree().clone_for_update().syntax(),
);
} else {
ted::replace(
path.syntax(),
subst.clone_subtree().clone_for_update().syntax(),
);
}
} }
} }
hir::PathResolution::Def(def) => { hir::PathResolution::Def(def) => {
if let hir::ModuleDef::Trait(_) = def {
if matches!(path.segment()?.kind()?, ast::PathSegmentKind::Type { .. }) {
// `speculative_resolve` resolves segments like `<T as
// Trait>` into `Trait`, but just the trait name should
// not be used as the replacement of the original
// segment.
return None;
}
}
let found_path = let found_path =
self.target_module.find_use_path(self.source_scope.db.upcast(), def)?; self.target_module.find_use_path(self.source_scope.db.upcast(), def)?;
let res = mod_path_to_ast(&found_path).clone_for_update(); let res = mod_path_to_ast(&found_path).clone_for_update();
@ -195,3 +251,33 @@ fn get_type_args_from_arg_list(generic_arg_list: ast::GenericArgList) -> Option<
Some(result) Some(result)
} }
fn find_trait_for_assoc_type(
scope: &SemanticsScope,
type_param: hir::TypeParam,
assoc_type: ast::NameRef,
) -> Option<hir::Trait> {
let db = scope.db;
let trait_bounds = type_param.trait_bounds(db);
let assoc_type_name = assoc_type.text();
for trait_ in trait_bounds {
let type_aliases = trait_.items(db).into_iter().filter_map(|item| match item {
hir::AssocItem::TypeAlias(ta) => Some(ta),
_ => None,
});
for type_alias in type_aliases {
if assoc_type_name.as_str() == type_alias.name(db).as_text()?.as_str() {
// It is fine to return the first match because in case of
// multiple possibilities, the exact trait must be disambiguated
// in the definition of trait being implemented, so this search
// should not be needed.
return Some(trait_);
}
}
}
None
}

View file

@ -166,6 +166,14 @@ pub fn path_segment(name_ref: ast::NameRef) -> ast::PathSegment {
ast_from_text(&format!("use {};", name_ref)) ast_from_text(&format!("use {};", name_ref))
} }
pub fn path_segment_ty(type_ref: ast::Type, trait_ref: Option<ast::PathType>) -> ast::PathSegment {
let text = match trait_ref {
Some(trait_ref) => format!("fn f(x: <{} as {}>) {{}}", type_ref, trait_ref),
None => format!("fn f(x: <{}>) {{}}", type_ref),
};
ast_from_text(&text)
}
pub fn path_segment_self() -> ast::PathSegment { pub fn path_segment_self() -> ast::PathSegment {
ast_from_text("use self;") ast_from_text("use self;")
} }
@ -196,9 +204,9 @@ pub fn path_from_segments(
) -> ast::Path { ) -> ast::Path {
let segments = segments.into_iter().map(|it| it.syntax().clone()).join("::"); let segments = segments.into_iter().map(|it| it.syntax().clone()).join("::");
ast_from_text(&if is_abs { ast_from_text(&if is_abs {
format!("use ::{};", segments) format!("fn f(x: ::{}) {{}}", segments)
} else { } else {
format!("use {};", segments) format!("fn f(x: {}) {{}}", segments)
}) })
} }