use std::{convert::Infallible, ops::ControlFlow};
use hir::{
AsAssocItem, AssocItem, AssocItemContainer, Complete, Crate, FindPathConfig, HasCrate,
ItemInNs, ModPath, Module, ModuleDef, Name, PathResolution, PrefixKind, ScopeDef, Semantics,
SemanticsScope, Trait, Type,
};
use itertools::Itertools;
use parser::SyntaxKind;
use rustc_hash::{FxHashMap, FxHashSet};
use smallvec::{SmallVec, smallvec};
use stdx::never;
use syntax::{
AstNode, SyntaxNode,
ast::{self, HasName, make},
};
use crate::{
FxIndexSet, RootDatabase,
items_locator::{self, AssocSearchMode, DEFAULT_QUERY_SEARCH_LIMIT},
};
#[derive(Debug, Clone, PartialEq, Eq, Hash, Copy)]
pub struct ImportPathConfig {
pub prefer_no_std: bool,
pub prefer_prelude: bool,
pub prefer_absolute: bool,
}
#[derive(Debug)]
pub enum ImportCandidate<'db> {
Path(PathImportCandidate),
TraitAssocItem(TraitImportCandidate<'db>),
TraitMethod(TraitImportCandidate<'db>),
}
#[derive(Debug)]
pub struct TraitImportCandidate<'db> {
pub receiver_ty: Type<'db>,
pub assoc_item_name: NameToImport,
}
#[derive(Debug)]
struct PathDefinitionKinds {
modules: bool,
bang_macros: bool,
attr_macros: bool,
value_namespace: bool,
type_namespace: bool,
records: bool,
tuple_structs: bool,
structs_and_consts: bool,
}
impl PathDefinitionKinds {
const ALL_DISABLED: Self = Self {
modules: false,
bang_macros: false,
attr_macros: false,
value_namespace: false,
type_namespace: false,
records: false,
tuple_structs: false,
structs_and_consts: false,
};
const ALL_ENABLED: Self = Self {
modules: true,
bang_macros: true,
attr_macros: true,
value_namespace: true,
type_namespace: true,
records: true,
tuple_structs: true,
structs_and_consts: true,
};
const PATH_PAT_KINDS: PathDefinitionKinds =
Self { structs_and_consts: true, bang_macros: true, ..Self::ALL_DISABLED };
fn deduce_from_path(path: &ast::Path, exact: bool) -> Self {
let Some(parent) = path.syntax().parent() else {
return Self::ALL_ENABLED;
};
let mut result = match parent.kind() {
SyntaxKind::PATH => Self { modules: true, type_namespace: true, ..Self::ALL_DISABLED },
SyntaxKind::MACRO_CALL => Self { bang_macros: true, ..Self::ALL_DISABLED },
SyntaxKind::PATH_META | SyntaxKind::KEY_VALUE_META | SyntaxKind::TOKEN_TREE_META => {
Self { attr_macros: true, ..Self::ALL_DISABLED }
}
SyntaxKind::USE_TREE => {
if ast::UseTree::cast(parent).unwrap().use_tree_list().is_some() {
Self { modules: true, ..Self::ALL_DISABLED }
} else {
Self::ALL_ENABLED
}
}
SyntaxKind::VISIBILITY => Self { modules: true, ..Self::ALL_DISABLED },
SyntaxKind::ASM_SYM => Self { value_namespace: true, ..Self::ALL_DISABLED },
SyntaxKind::PATH_EXPR => Self {
value_namespace: true,
bang_macros: true,
type_namespace: true,
..Self::ALL_DISABLED
},
SyntaxKind::PATH_PAT => Self::PATH_PAT_KINDS,
SyntaxKind::TUPLE_STRUCT_PAT => {
Self { tuple_structs: true, bang_macros: true, ..Self::ALL_DISABLED }
}
SyntaxKind::RECORD_EXPR | SyntaxKind::RECORD_PAT => {
Self { records: true, bang_macros: true, ..Self::ALL_DISABLED }
}
SyntaxKind::PATH_TYPE => {
Self { type_namespace: true, bang_macros: true, ..Self::ALL_DISABLED }
}
SyntaxKind::ERROR => Self::ALL_ENABLED,
_ => {
never!("this match should cover all possible parents of paths\nparent={parent:#?}");
Self::ALL_ENABLED
}
};
if !exact {
result.modules = true;
result.type_namespace = true;
}
result
}
}
#[derive(Debug)]
pub struct PathImportCandidate {
pub qualifier: Vec<Name>,
pub name: NameToImport,
pub after: Vec<Name>,
definition_kinds: PathDefinitionKinds,
}
#[derive(Debug, Clone)]
pub enum NameToImport {
Exact(String, bool),
Prefix(String, bool),
Fuzzy(String, bool),
}
impl NameToImport {
pub fn exact_case_sensitive(s: String) -> NameToImport {
let s = match s.strip_prefix("r#") {
Some(s) => s.to_owned(),
None => s,
};
NameToImport::Exact(s, true)
}
pub fn fuzzy(s: String) -> NameToImport {
let s = match s.strip_prefix("r#") {
Some(s) => s.to_owned(),
None => s,
};
let case_sensitive = s.chars().any(|c| c.is_uppercase());
NameToImport::Fuzzy(s, case_sensitive)
}
pub fn text(&self) -> &str {
match self {
NameToImport::Prefix(text, _)
| NameToImport::Exact(text, _)
| NameToImport::Fuzzy(text, _) => text.as_str(),
}
}
}
#[derive(Debug)]
pub struct ImportAssets<'db> {
import_candidate: ImportCandidate<'db>,
candidate_node: SyntaxNode,
module_with_candidate: Module,
}
impl<'db> ImportAssets<'db> {
pub fn for_method_call(
method_call: &ast::MethodCallExpr,
sema: &Semantics<'db, RootDatabase>,
) -> Option<Self> {
let candidate_node = method_call.syntax().clone();
Some(Self {
import_candidate: ImportCandidate::for_method_call(sema, method_call)?,
module_with_candidate: sema.scope(&candidate_node)?.module(),
candidate_node,
})
}
pub fn for_exact_path(
fully_qualified_path: &ast::Path,
sema: &Semantics<'db, RootDatabase>,
) -> Option<Self> {
let candidate_node = fully_qualified_path.syntax().clone();
if let Some(use_tree) = candidate_node.ancestors().find_map(ast::UseTree::cast) {
if use_tree.syntax().parent().and_then(ast::Use::cast).is_none()
|| fully_qualified_path.qualifier().is_some()
{
return None;
}
}
Some(Self {
import_candidate: ImportCandidate::for_regular_path(sema, fully_qualified_path)?,
module_with_candidate: sema.scope(&candidate_node)?.module(),
candidate_node,
})
}
pub fn for_ident_pat(sema: &Semantics<'db, RootDatabase>, pat: &ast::IdentPat) -> Option<Self> {
if !pat.is_simple_ident() {
return None;
}
let name = pat.name()?;
let candidate_node = pat.syntax().clone();
Some(Self {
import_candidate: ImportCandidate::for_name(sema, &name)?,
module_with_candidate: sema.scope(&candidate_node)?.module(),
candidate_node,
})
}
pub fn for_fuzzy_path(
module_with_candidate: Module,
path: Option<&ast::Path>,
qualifier: Option<ast::Path>,
fuzzy_name: String,
sema: &Semantics<'db, RootDatabase>,
candidate_node: SyntaxNode,
) -> Option<Self> {
Some(Self {
import_candidate: ImportCandidate::for_fuzzy_path(path, qualifier, fuzzy_name, sema)?,
module_with_candidate,
candidate_node,
})
}
pub fn for_fuzzy_method_call(
module_with_method_call: Module,
receiver_ty: Type<'db>,
fuzzy_method_name: String,
candidate_node: SyntaxNode,
) -> Option<Self> {
Some(Self {
import_candidate: ImportCandidate::TraitMethod(TraitImportCandidate {
receiver_ty,
assoc_item_name: NameToImport::fuzzy(fuzzy_method_name),
}),
module_with_candidate: module_with_method_call,
candidate_node,
})
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub struct CompleteInFlyimport(pub bool);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct LocatedImport {
pub import_path: ModPath,
pub item_to_import: ItemInNs,
pub original_item: ItemInNs,
pub complete_in_flyimport: CompleteInFlyimport,
}
impl LocatedImport {
pub fn new(
import_path: ModPath,
item_to_import: ItemInNs,
original_item: ItemInNs,
complete_in_flyimport: CompleteInFlyimport,
) -> Self {
Self { import_path, item_to_import, original_item, complete_in_flyimport }
}
pub fn new_no_completion(
import_path: ModPath,
item_to_import: ItemInNs,
original_item: ItemInNs,
) -> Self {
Self {
import_path,
item_to_import,
original_item,
complete_in_flyimport: CompleteInFlyimport(true),
}
}
}
impl<'db> ImportAssets<'db> {
pub fn import_candidate(&self) -> &ImportCandidate<'db> {
&self.import_candidate
}
pub fn search_for_imports(
&self,
sema: &Semantics<'db, RootDatabase>,
cfg: ImportPathConfig,
prefix_kind: PrefixKind,
) -> impl Iterator<Item = LocatedImport> {
let _p = tracing::info_span!("ImportAssets::search_for_imports").entered();
self.search_for(sema, Some(prefix_kind), cfg)
}
pub fn search_for_relative_paths(
&self,
sema: &Semantics<'db, RootDatabase>,
cfg: ImportPathConfig,
) -> impl Iterator<Item = LocatedImport> {
let _p = tracing::info_span!("ImportAssets::search_for_relative_paths").entered();
self.search_for(sema, None, cfg)
}
pub fn path_fuzzy_name_to_prefix(&mut self) {
if let ImportCandidate::Path(PathImportCandidate { name: to_import, .. }) =
&mut self.import_candidate
{
let (name, case_sensitive) = match to_import {
NameToImport::Fuzzy(name, case_sensitive) => {
(std::mem::take(name), *case_sensitive)
}
_ => return,
};
*to_import = NameToImport::Prefix(name, case_sensitive);
}
}
pub fn path_fuzzy_name_to_exact(&mut self) {
if let ImportCandidate::Path(PathImportCandidate { name: to_import, .. }) =
&mut self.import_candidate
{
let (name, case_sensitive) = match to_import {
NameToImport::Fuzzy(name, case_sensitive) => {
(std::mem::take(name), *case_sensitive)
}
_ => return,
};
*to_import = NameToImport::Exact(name, case_sensitive);
}
}
fn search_for(
&self,
sema: &Semantics<'db, RootDatabase>,
prefixed: Option<PrefixKind>,
cfg: ImportPathConfig,
) -> impl Iterator<Item = LocatedImport> {
let _p = tracing::info_span!("ImportAssets::search_for").entered();
let scope = match sema.scope(&self.candidate_node) {
Some(it) => it,
None => return <FxIndexSet<_>>::default().into_iter(),
};
let cfg = FindPathConfig {
prefer_no_std: cfg.prefer_no_std,
prefer_prelude: cfg.prefer_prelude,
prefer_absolute: cfg.prefer_absolute,
allow_unstable: sema.is_nightly(scope.krate()),
};
let db = sema.db;
let krate = self.module_with_candidate.krate(sema.db);
let scope_definitions = self.scope_definitions(sema);
let mod_path = |item| {
get_mod_path(
db,
item_for_path_search(db, item)?,
&self.module_with_candidate,
prefixed,
cfg,
)
.filter(|path| path.len() > 1)
};
match &self.import_candidate {
ImportCandidate::Path(path_candidate) => path_applicable_imports(
db,
&scope,
krate,
path_candidate,
mod_path,
|item_to_import| !scope_definitions.contains(&ScopeDef::from(item_to_import)),
),
ImportCandidate::TraitAssocItem(trait_candidate)
| ImportCandidate::TraitMethod(trait_candidate) => trait_applicable_items(
db,
krate,
&scope,
trait_candidate,
matches!(self.import_candidate, ImportCandidate::TraitAssocItem(_)),
mod_path,
|trait_to_import| {
!scope_definitions
.contains(&ScopeDef::ModuleDef(ModuleDef::Trait(trait_to_import)))
},
),
}
.into_iter()
}
fn scope_definitions(&self, sema: &Semantics<'_, RootDatabase>) -> FxHashSet<ScopeDef> {
let _p = tracing::info_span!("ImportAssets::scope_definitions").entered();
let mut scope_definitions = FxHashSet::default();
if let Some(scope) = sema.scope(&self.candidate_node) {
scope.process_all_names(&mut |_, scope_def| {
scope_definitions.insert(scope_def);
});
}
scope_definitions
}
}
fn path_applicable_imports(
db: &RootDatabase,
scope: &SemanticsScope<'_>,
current_crate: Crate,
path_candidate: &PathImportCandidate,
mod_path: impl Fn(ItemInNs) -> Option<ModPath> + Copy,
scope_filter: impl Fn(ItemInNs) -> bool + Copy,
) -> FxIndexSet<LocatedImport> {
let _p = tracing::info_span!("ImportAssets::path_applicable_imports").entered();
let mut result = match &*path_candidate.qualifier {
[] => {
items_locator::items_with_name(
db,
current_crate,
path_candidate.name.clone(),
AssocSearchMode::Exclude,
)
.filter(|(item, _)| {
filter_by_definition_kind(db, *item, &path_candidate.definition_kinds)
})
.filter_map(|(item, do_not_complete)| {
if !scope_filter(item) {
return None;
}
let mod_path = mod_path(item)?;
Some(LocatedImport::new(
mod_path,
item,
item,
CompleteInFlyimport(do_not_complete != Complete::IgnoreFlyimport),
))
})
.take(DEFAULT_QUERY_SEARCH_LIMIT)
.collect()
}
[first_qsegment, qualifier_rest @ ..] => items_locator::items_with_name(
db,
current_crate,
NameToImport::Exact(first_qsegment.as_str().to_owned(), true),
AssocSearchMode::Exclude,
)
.flat_map(|(item, do_not_complete)| {
validate_resolvable(
db,
scope,
mod_path,
scope_filter,
&path_candidate.name,
item,
qualifier_rest,
CompleteInFlyimport(do_not_complete != Complete::IgnoreFlyimport),
)
})
.take(DEFAULT_QUERY_SEARCH_LIMIT)
.collect(),
};
filter_candidates_by_after_path(db, scope, path_candidate, &mut result);
result
}
fn filter_by_definition_kind(
db: &RootDatabase,
item: ItemInNs,
allowed: &PathDefinitionKinds,
) -> bool {
let item = item.into_module_def();
let struct_per_kind = |struct_kind| {
allowed.structs_and_consts
|| match struct_kind {
hir::StructKind::Record => allowed.records,
hir::StructKind::Tuple => allowed.value_namespace || allowed.tuple_structs,
hir::StructKind::Unit => allowed.value_namespace,
}
};
match item {
ModuleDef::Module(_) => allowed.modules,
ModuleDef::Function(_) => allowed.value_namespace,
ModuleDef::Adt(hir::Adt::Struct(item)) => {
allowed.type_namespace || struct_per_kind(item.kind(db))
}
ModuleDef::Adt(hir::Adt::Enum(_)) => allowed.type_namespace,
ModuleDef::Adt(hir::Adt::Union(_)) => {
allowed.type_namespace || allowed.records || allowed.structs_and_consts
}
ModuleDef::EnumVariant(item) => struct_per_kind(item.kind(db)),
ModuleDef::Const(_) => allowed.value_namespace || allowed.structs_and_consts,
ModuleDef::Static(_) => allowed.value_namespace,
ModuleDef::Trait(_) => allowed.type_namespace,
ModuleDef::TypeAlias(_) => allowed.type_namespace,
ModuleDef::BuiltinType(_) => allowed.type_namespace,
ModuleDef::Macro(item) => {
if item.is_fn_like(db) {
allowed.bang_macros
} else {
allowed.attr_macros
}
}
}
}
fn filter_candidates_by_after_path(
db: &RootDatabase,
scope: &SemanticsScope<'_>,
path_candidate: &PathImportCandidate,
imports: &mut FxIndexSet<LocatedImport>,
) {
if imports.len() <= 1 {
return;
}
let Some((last_after, after_except_last)) = path_candidate.after.split_last() else {
return;
};
let original_imports = imports.clone();
let traits_in_scope = scope.visible_traits();
imports.retain(|import| {
let items = if after_except_last.is_empty() {
smallvec![import.original_item]
} else {
let ItemInNs::Types(ModuleDef::Module(item)) = import.original_item else {
return false;
};
item.resolve_mod_path(db, after_except_last.iter().cloned())
.into_iter()
.flatten()
.collect::<SmallVec<[_; 3]>>()
};
items.into_iter().any(|item| {
let has_last_method = |ty: hir::Type<'_>| {
ty.iterate_path_candidates(db, scope, &traits_in_scope, Some(last_after), |_| {
Some(())
})
.is_some()
};
match item {
ItemInNs::Types(ModuleDef::Module(module)) => module
.resolve_mod_path(db, [last_after.clone()])
.is_some_and(|mut it| it.any(|_| true)),
ItemInNs::Types(ModuleDef::Adt(it)) => has_last_method(it.ty(db)),
ItemInNs::Types(ModuleDef::BuiltinType(it)) => has_last_method(it.ty(db)),
ItemInNs::Types(ModuleDef::TypeAlias(it)) => has_last_method(it.ty(db)),
ItemInNs::Types(ModuleDef::Trait(it)) => it
.items(db)
.into_iter()
.any(|assoc_item| assoc_item.name(db) == Some(last_after.clone())),
_ => false,
}
})
});
if imports.is_empty() {
*imports = original_imports;
}
}
fn validate_resolvable(
db: &RootDatabase,
scope: &SemanticsScope<'_>,
mod_path: impl Fn(ItemInNs) -> Option<ModPath>,
scope_filter: impl Fn(ItemInNs) -> bool,
candidate: &NameToImport,
resolved_qualifier: ItemInNs,
unresolved_qualifier: &[Name],
complete_in_flyimport: CompleteInFlyimport,
) -> SmallVec<[LocatedImport; 1]> {
let _p = tracing::info_span!("ImportAssets::import_for_item").entered();
let qualifier = (|| {
let mut adjusted_resolved_qualifier = resolved_qualifier;
if !unresolved_qualifier.is_empty() {
match resolved_qualifier {
ItemInNs::Types(ModuleDef::Module(module)) => {
adjusted_resolved_qualifier = module
.resolve_mod_path(db, unresolved_qualifier.iter().cloned())?
.next()?;
}
_ => return None,
}
}
match adjusted_resolved_qualifier {
ItemInNs::Types(def) => Some(def),
_ => None,
}
})();
let Some(qualifier) = qualifier else { return SmallVec::new() };
let Some(import_path_candidate) = mod_path(resolved_qualifier) else { return SmallVec::new() };
let mut result = SmallVec::new();
let ty = match qualifier {
ModuleDef::Module(module) => {
items_locator::items_with_name_in_module::<Infallible>(
db,
module,
candidate.clone(),
AssocSearchMode::Exclude,
|item| {
if scope_filter(item) {
result.push(LocatedImport::new(
import_path_candidate.clone(),
resolved_qualifier,
item,
complete_in_flyimport,
));
}
ControlFlow::Continue(())
},
);
return result;
}
ModuleDef::Trait(_) => return SmallVec::new(),
ModuleDef::TypeAlias(alias) => alias.ty(db),
ModuleDef::BuiltinType(builtin) => builtin.ty(db),
ModuleDef::Adt(adt) => adt.ty(db),
_ => return SmallVec::new(),
};
ty.iterate_path_candidates::<Infallible>(db, scope, &FxHashSet::default(), None, |assoc| {
if assoc.container_or_implemented_trait(db).is_some() {
return None;
}
let name = assoc.name(db)?;
let is_match = match candidate {
NameToImport::Prefix(text, true) => name.as_str().starts_with(text),
NameToImport::Prefix(text, false) => {
name.as_str().chars().zip(text.chars()).all(|(name_char, candidate_char)| {
name_char.eq_ignore_ascii_case(&candidate_char)
})
}
NameToImport::Exact(text, true) => name.as_str() == text,
NameToImport::Exact(text, false) => name.as_str().eq_ignore_ascii_case(text),
NameToImport::Fuzzy(text, true) => text.chars().all(|c| name.as_str().contains(c)),
NameToImport::Fuzzy(text, false) => text
.chars()
.all(|c| name.as_str().chars().any(|name_char| name_char.eq_ignore_ascii_case(&c))),
};
if !is_match {
return None;
}
result.push(LocatedImport::new(
import_path_candidate.clone(),
resolved_qualifier,
assoc_to_item(assoc),
complete_in_flyimport,
));
None
});
result
}
pub fn item_for_path_search(db: &RootDatabase, item: ItemInNs) -> Option<ItemInNs> {
Some(match item {
ItemInNs::Types(_) | ItemInNs::Values(_) => match item_as_assoc(db, item) {
Some(assoc_item) => item_for_path_search_assoc(db, assoc_item)?,
None => item,
},
ItemInNs::Macros(_) => item,
})
}
fn item_for_path_search_assoc(db: &RootDatabase, assoc_item: AssocItem) -> Option<ItemInNs> {
Some(match assoc_item.container(db) {
AssocItemContainer::Trait(trait_) => ItemInNs::from(ModuleDef::from(trait_)),
AssocItemContainer::Impl(impl_) => {
ItemInNs::from(ModuleDef::from(impl_.self_ty(db).as_adt()?))
}
})
}
fn trait_applicable_items<'db>(
db: &'db RootDatabase,
current_crate: Crate,
scope: &SemanticsScope<'db>,
trait_candidate: &TraitImportCandidate<'db>,
trait_assoc_item: bool,
mod_path: impl Fn(ItemInNs) -> Option<ModPath>,
scope_filter: impl Fn(hir::Trait) -> bool,
) -> FxIndexSet<LocatedImport> {
let _p = tracing::info_span!("ImportAssets::trait_applicable_items").entered();
let inherent_traits = trait_candidate.receiver_ty.applicable_inherent_traits(db);
let env_traits = trait_candidate.receiver_ty.env_traits(db);
let related_traits = inherent_traits.chain(env_traits).collect::<FxHashSet<_>>();
let mut required_assoc_items = FxHashMap::default();
let mut trait_candidates: FxHashSet<_> = items_locator::items_with_name(
db,
current_crate,
trait_candidate.assoc_item_name.clone(),
AssocSearchMode::AssocItemsOnly,
)
.filter_map(|(input, do_not_complete)| Some((item_as_assoc(db, input)?, do_not_complete)))
.filter_map(|(assoc, do_not_complete)| {
if !trait_assoc_item && matches!(assoc, AssocItem::Const(_) | AssocItem::TypeAlias(_)) {
return None;
}
let assoc_item_trait = assoc.container_trait(db)?;
if related_traits.contains(&assoc_item_trait) {
return None;
}
required_assoc_items
.insert(assoc, CompleteInFlyimport(do_not_complete != Complete::IgnoreFlyimport));
Some(assoc_item_trait.into())
})
.collect();
let autoderef_method_receiver = {
let mut deref_chain = trait_candidate.receiver_ty.autoderef(db).collect::<Vec<_>>();
if let Some((ty, _len)) = deref_chain.last().and_then(|ty| ty.as_array(db)) {
let slice = Type::new_slice(ty);
deref_chain.push(slice);
}
deref_chain
.into_iter()
.flat_map(|ty| {
let fingerprint = ty.fingerprint_for_trait_impl()?;
let mut crates = vec![];
if let Some(adt) = ty.as_adt() {
crates.push((adt.krate(db).into(), fingerprint));
}
crates.push((ty.krate(db).into(), fingerprint));
Some(crates)
})
.flatten()
.unique()
.collect::<Vec<_>>()
};
if autoderef_method_receiver.is_empty() {
return Default::default();
}
if !autoderef_method_receiver
.iter()
.any(|(_, fingerprint)| matches!(fingerprint, hir::SimplifiedType::Placeholder))
{
trait_candidates.retain(|&candidate_trait_id| {
let defining_crate_for_trait = Trait::from(candidate_trait_id).krate(db);
let trait_impls_in_crate =
hir::TraitImpls::for_crate(db, defining_crate_for_trait.into());
let definitions_exist_in_trait_crate =
autoderef_method_receiver.iter().any(|(_, fingerprint)| {
trait_impls_in_crate
.has_impls_for_trait_and_self_ty(candidate_trait_id, fingerprint)
});
let definitions_exist_in_receiver_crate = || {
autoderef_method_receiver.iter().any(|(krate, fingerprint)| {
hir::TraitImpls::for_crate(db, *krate)
.has_impls_for_trait_and_self_ty(candidate_trait_id, fingerprint)
})
};
definitions_exist_in_trait_crate || definitions_exist_in_receiver_crate()
});
}
let mut located_imports = FxIndexSet::default();
let mut trait_import_paths = FxHashMap::default();
if trait_assoc_item {
trait_candidate.receiver_ty.iterate_path_candidates(
db,
scope,
&trait_candidates,
None,
|assoc| {
if let Some(&complete_in_flyimport) = required_assoc_items.get(&assoc) {
let located_trait = assoc.container_trait(db).filter(|&it| scope_filter(it))?;
let trait_item = ItemInNs::from(ModuleDef::from(located_trait));
let import_path = trait_import_paths
.entry(trait_item)
.or_insert_with(|| mod_path(trait_item))
.clone()?;
located_imports.insert(LocatedImport::new(
import_path,
trait_item,
assoc_to_item(assoc),
complete_in_flyimport,
));
}
None::<()>
},
)
} else {
trait_candidate.receiver_ty.iterate_method_candidates_with_traits(
db,
scope,
&trait_candidates,
None,
|function| {
let assoc = function.as_assoc_item(db)?;
if let Some(&complete_in_flyimport) = required_assoc_items.get(&assoc) {
let located_trait = assoc.container_trait(db).filter(|&it| scope_filter(it))?;
let trait_item = ItemInNs::from(ModuleDef::from(located_trait));
let import_path = trait_import_paths
.entry(trait_item)
.or_insert_with(|| mod_path(trait_item))
.clone()?;
located_imports.insert(LocatedImport::new(
import_path,
trait_item,
assoc_to_item(assoc),
complete_in_flyimport,
));
}
None::<()>
},
)
};
located_imports
}
fn assoc_to_item(assoc: AssocItem) -> ItemInNs {
match assoc {
AssocItem::Function(f) => ItemInNs::from(ModuleDef::from(f)),
AssocItem::Const(c) => ItemInNs::from(ModuleDef::from(c)),
AssocItem::TypeAlias(t) => ItemInNs::from(ModuleDef::from(t)),
}
}
#[tracing::instrument(skip_all)]
fn get_mod_path(
db: &RootDatabase,
item_to_search: ItemInNs,
module_with_candidate: &Module,
prefixed: Option<PrefixKind>,
cfg: FindPathConfig,
) -> Option<ModPath> {
if let Some(prefix_kind) = prefixed {
module_with_candidate.find_use_path(db, item_to_search, prefix_kind, cfg)
} else {
module_with_candidate.find_path(db, item_to_search, cfg)
}
}
impl<'db> ImportCandidate<'db> {
fn for_method_call(
sema: &Semantics<'db, RootDatabase>,
method_call: &ast::MethodCallExpr,
) -> Option<Self> {
match sema.resolve_method_call(method_call) {
Some(_) => None,
None => Some(Self::TraitMethod(TraitImportCandidate {
receiver_ty: sema.type_of_expr(&method_call.receiver()?)?.adjusted(),
assoc_item_name: NameToImport::exact_case_sensitive(
method_call.name_ref()?.to_string(),
),
})),
}
}
fn for_regular_path(sema: &Semantics<'db, RootDatabase>, path: &ast::Path) -> Option<Self> {
if sema.resolve_path(path).is_some() {
return None;
}
let after = std::iter::successors(path.parent_path(), |it| it.parent_path())
.map(|seg| seg.segment()?.name_ref().map(|name| Name::new_root(&name.text())))
.collect::<Option<_>>()?;
path_import_candidate(
sema,
Some(path),
path.qualifier(),
NameToImport::exact_case_sensitive(path.segment()?.name_ref()?.to_string()),
after,
)
}
fn for_name(sema: &Semantics<'db, RootDatabase>, name: &ast::Name) -> Option<Self> {
if sema
.scope(name.syntax())?
.speculative_resolve(&make::ext::ident_path(&name.text()))
.is_some()
{
return None;
}
Some(ImportCandidate::Path(PathImportCandidate {
qualifier: vec![],
name: NameToImport::exact_case_sensitive(name.to_string()),
after: vec![],
definition_kinds: PathDefinitionKinds::PATH_PAT_KINDS,
}))
}
fn for_fuzzy_path(
path: Option<&ast::Path>,
qualifier: Option<ast::Path>,
fuzzy_name: String,
sema: &Semantics<'db, RootDatabase>,
) -> Option<Self> {
path_import_candidate(sema, path, qualifier, NameToImport::fuzzy(fuzzy_name), Vec::new())
}
}
fn path_import_candidate<'db>(
sema: &Semantics<'db, RootDatabase>,
path: Option<&ast::Path>,
qualifier: Option<ast::Path>,
name: NameToImport,
after: Vec<Name>,
) -> Option<ImportCandidate<'db>> {
let definition_kinds = path.map_or(PathDefinitionKinds::ALL_ENABLED, |path| {
PathDefinitionKinds::deduce_from_path(path, matches!(name, NameToImport::Exact(..)))
});
Some(match qualifier {
Some(qualifier) => match sema.resolve_path(&qualifier) {
Some(PathResolution::Def(ModuleDef::BuiltinType(_))) | None => {
if qualifier.first_qualifier().is_none_or(|it| sema.resolve_path(&it).is_none()) {
let qualifier = qualifier
.segments()
.map(|seg| seg.name_ref().map(|name| Name::new_root(&name.text())))
.collect::<Option<Vec<_>>>()?;
ImportCandidate::Path(PathImportCandidate {
qualifier,
name,
after,
definition_kinds,
})
} else {
return None;
}
}
Some(PathResolution::Def(ModuleDef::Adt(assoc_item_path))) => {
ImportCandidate::TraitAssocItem(TraitImportCandidate {
receiver_ty: assoc_item_path.ty(sema.db),
assoc_item_name: name,
})
}
Some(PathResolution::Def(ModuleDef::TypeAlias(alias))) => {
let ty = alias.ty(sema.db);
if ty.as_adt().is_some() {
ImportCandidate::TraitAssocItem(TraitImportCandidate {
receiver_ty: ty,
assoc_item_name: name,
})
} else {
return None;
}
}
Some(_) => return None,
},
None => ImportCandidate::Path(PathImportCandidate {
qualifier: vec![],
name,
after,
definition_kinds,
}),
})
}
fn item_as_assoc(db: &RootDatabase, item: ItemInNs) -> Option<AssocItem> {
item.into_module_def().as_assoc_item(db)
}