Unnamed repository; edit this file 'description' to name the repository.
Diffstat (limited to 'crates/ide-db/src/symbol_index.rs')
-rw-r--r--crates/ide-db/src/symbol_index.rs682
1 files changed, 663 insertions, 19 deletions
diff --git a/crates/ide-db/src/symbol_index.rs b/crates/ide-db/src/symbol_index.rs
index eb0529d6b5..183f6b6495 100644
--- a/crates/ide-db/src/symbol_index.rs
+++ b/crates/ide-db/src/symbol_index.rs
@@ -27,7 +27,7 @@ use std::{
ops::ControlFlow,
};
-use base_db::{LibraryRoots, LocalRoots, RootQueryDb, SourceRootId};
+use base_db::{CrateOrigin, LangCrateOrigin, LibraryRoots, LocalRoots, RootQueryDb, SourceRootId};
use fst::{Automaton, Streamer, raw::IndexedValue};
use hir::{
Crate, Module,
@@ -35,29 +35,84 @@ use hir::{
import_map::{AssocSearchMode, SearchMode},
symbols::{FileSymbol, SymbolCollector},
};
+use itertools::Itertools;
use rayon::prelude::*;
use salsa::Update;
use crate::RootDatabase;
+/// A query for searching symbols in the workspace or dependencies.
+///
+/// This struct configures how symbol search is performed, including the search text,
+/// matching strategy, and filtering options. It is used by [`world_symbols`] to find
+/// symbols across the codebase.
+///
+/// # Example
+/// ```ignore
+/// let mut query = Query::new("MyStruct".to_string());
+/// query.only_types(); // Only search for type definitions
+/// query.libs(); // Include library dependencies
+/// query.exact(); // Use exact matching instead of fuzzy
+/// ```
#[derive(Debug, Clone)]
pub struct Query {
+ /// The item name to search for (last segment of the path, or full query if no path).
+ /// When empty with a non-empty `path_filter`, returns all items in that module.
query: String,
+ /// Lowercase version of [`Self::query`], pre-computed for efficiency.
+ /// Used to build FST automata for case-insensitive index lookups.
lowercased: String,
+ /// Path segments to filter by (all segments except the last).
+ /// Empty if no `::` in the original query.
+ path_filter: Vec<String>,
+ /// If true, the first path segment must be a crate name (query started with `::`).
+ anchor_to_crate: bool,
+ /// The search strategy to use when matching symbols.
+ /// - [`SearchMode::Exact`]: Symbol name must exactly match the query.
+ /// - [`SearchMode::Fuzzy`]: Symbol name must contain all query characters in order (subsequence match).
+ /// - [`SearchMode::Prefix`]: Symbol name must start with the query string.
+ ///
+ /// Defaults to [`SearchMode::Fuzzy`].
mode: SearchMode,
+ /// Controls filtering of trait-associated items (methods, constants, types).
+ /// - [`AssocSearchMode::Include`]: Include both associated and non-associated items.
+ /// - [`AssocSearchMode::Exclude`]: Exclude trait-associated items from results.
+ /// - [`AssocSearchMode::AssocItemsOnly`]: Only return trait-associated items.
+ ///
+ /// Defaults to [`AssocSearchMode::Include`].
assoc_mode: AssocSearchMode,
+ /// Whether the final symbol name comparison should be case-sensitive.
+ /// When `false`, matching is case-insensitive (e.g., "foo" matches "Foo").
+ ///
+ /// Defaults to `false`.
case_sensitive: bool,
+ /// When `true`, only return type definitions: structs, enums, unions,
+ /// type aliases, built-in types, and traits. Functions, constants, statics,
+ /// and modules are excluded.
+ ///
+ /// Defaults to `false`.
only_types: bool,
+ /// When `true`, search library dependency roots instead of local workspace crates.
+ /// This enables finding symbols in external dependencies including the standard library.
+ ///
+ /// Defaults to `false` (search local workspace only).
libs: bool,
+ /// When `true`, exclude re-exported/imported symbols from results,
+ /// showing only the original definitions.
+ ///
+ /// Defaults to `false`.
exclude_imports: bool,
}
impl Query {
pub fn new(query: String) -> Query {
- let lowercased = query.to_lowercase();
+ let (path_filter, item_query, anchor_to_crate) = Self::parse_path_query(&query);
+ let lowercased = item_query.to_lowercase();
Query {
- query,
+ query: item_query,
lowercased,
+ path_filter,
+ anchor_to_crate,
only_types: false,
libs: false,
mode: SearchMode::Fuzzy,
@@ -67,6 +122,35 @@ impl Query {
}
}
+ /// Parse a query string that may contain path segments.
+ ///
+ /// Returns (path_filter, item_query, anchor_to_crate) where:
+ /// - `path_filter`: Path segments to match (all but the last segment)
+ /// - `item_query`: The item name to search for (last segment)
+ /// - `anchor_to_crate`: Whether the first segment must be a crate name
+ fn parse_path_query(query: &str) -> (Vec<String>, String, bool) {
+ // Check for leading :: (absolute path / crate search)
+ let (query, anchor_to_crate) = match query.strip_prefix("::") {
+ Some(q) => (q, true),
+ None => (query, false),
+ };
+
+ let Some((prefix, query)) = query.rsplit_once("::") else {
+ return (vec![], query.to_owned(), anchor_to_crate);
+ };
+
+ let prefix: Vec<_> =
+ prefix.split("::").filter(|s| !s.is_empty()).map(ToOwned::to_owned).collect();
+
+ (prefix, query.to_owned(), anchor_to_crate)
+ }
+
+ /// Returns true if this query is searching for crates
+ /// (i.e., the query was "::" alone or "::foo" for fuzzy crate search)
+ fn is_crate_search(&self) -> bool {
+ self.anchor_to_crate && self.path_filter.is_empty()
+ }
+
pub fn only_types(&mut self) {
self.only_types = true;
}
@@ -123,19 +207,41 @@ pub fn crate_symbols(db: &dyn HirDatabase, krate: Crate) -> Box<[&SymbolIndex<'_
// That is, `#` switches from "types" to all symbols, `*` switches from the current
// workspace to dependencies.
//
-// Note that filtering does not currently work in VSCode due to the editor never
-// sending the special symbols to the language server. Instead, you can configure
-// the filtering via the `rust-analyzer.workspace.symbol.search.scope` and
-// `rust-analyzer.workspace.symbol.search.kind` settings. Symbols prefixed
-// with `__` are hidden from the search results unless configured otherwise.
+// This also supports general Rust path syntax with the usual rules.
+//
+// Note that paths do not currently work in VSCode due to the editor never
+// sending the special symbols to the language server. Some other editors might not support the # or
+// * search either, instead, you can configure the filtering via the
+// `rust-analyzer.workspace.symbol.search.scope` and `rust-analyzer.workspace.symbol.search.kind`
+// settings. Symbols prefixed with `__` are hidden from the search results unless configured
+// otherwise.
//
// | Editor | Shortcut |
// |---------|-----------|
// | VS Code | <kbd>Ctrl+T</kbd>
-pub fn world_symbols(db: &RootDatabase, query: Query) -> Vec<FileSymbol<'_>> {
+pub fn world_symbols(db: &RootDatabase, mut query: Query) -> Vec<FileSymbol<'_>> {
let _p = tracing::info_span!("world_symbols", query = ?query.query).entered();
- let indices: Vec<_> = if query.libs {
+ // Search for crates by name (handles "::" and "::foo" queries)
+ let indices: Vec<_> = if query.is_crate_search() {
+ query.only_types = false;
+ vec![SymbolIndex::extern_prelude_symbols(db)]
+ // If we have a path filter, resolve it to target modules
+ } else if !query.path_filter.is_empty() {
+ query.only_types = false;
+ let target_modules = resolve_path_to_modules(
+ db,
+ &query.path_filter,
+ query.anchor_to_crate,
+ query.case_sensitive,
+ );
+
+ if target_modules.is_empty() {
+ return vec![];
+ }
+
+ target_modules.iter().map(|&module| SymbolIndex::module_symbols(db, module)).collect()
+ } else if query.libs {
LibraryRoots::get(db)
.roots(db)
.par_iter()
@@ -154,17 +260,103 @@ pub fn world_symbols(db: &RootDatabase, query: Query) -> Vec<FileSymbol<'_>> {
crates
.par_iter()
.for_each_with(db.clone(), |snap, &krate| _ = crate_symbols(snap, krate.into()));
- crates.into_iter().flat_map(|krate| Vec::from(crate_symbols(db, krate.into()))).collect()
+ crates
+ .into_iter()
+ .flat_map(|krate| Vec::from(crate_symbols(db, krate.into())))
+ .chain(std::iter::once(SymbolIndex::extern_prelude_symbols(db)))
+ .collect()
};
let mut res = vec![];
- query.search::<()>(&indices, |f| {
+
+ // Normal search: use FST to match item name
+ query.search::<()>(db, &indices, |f| {
res.push(f.clone());
ControlFlow::Continue(())
});
+
res
}
+/// Resolve a path filter to the target module(s) it points to.
+/// Returns the modules whose symbol indices should be searched.
+///
+/// The path_filter contains segments like ["std", "vec"] for a query like "std::vec::Vec".
+/// We resolve this by:
+/// 1. Finding crates matching the first segment
+/// 2. Walking down the module tree following subsequent segments
+fn resolve_path_to_modules(
+ db: &dyn HirDatabase,
+ path_filter: &[String],
+ anchor_to_crate: bool,
+ case_sensitive: bool,
+) -> Vec<Module> {
+ let [first_segment, rest_segments @ ..] = path_filter else {
+ return vec![];
+ };
+
+ // Helper for name comparison
+ let names_match = |actual: &str, expected: &str| -> bool {
+ if case_sensitive { actual == expected } else { actual.eq_ignore_ascii_case(expected) }
+ };
+
+ // Find crates matching the first segment
+ let matching_crates: Vec<Crate> = Crate::all(db)
+ .into_iter()
+ .filter(|krate| {
+ krate
+ .display_name(db)
+ .is_some_and(|name| names_match(name.crate_name().as_str(), first_segment))
+ })
+ .collect();
+
+ // If anchor_to_crate is true, first segment MUST be a crate name
+ // If anchor_to_crate is false, first segment could be a crate OR a module in local crates
+ let mut candidate_modules: Vec<(Module, bool)> = vec![];
+
+ // Add crate root modules for matching crates
+ for krate in matching_crates {
+ candidate_modules.push((krate.root_module(db), krate.origin(db).is_local()));
+ }
+
+ // If not anchored to crate, also search for modules matching first segment in local crates
+ if !anchor_to_crate {
+ for &root in LocalRoots::get(db).roots(db).iter() {
+ for &krate in db.source_root_crates(root).iter() {
+ let root_module = Crate::from(krate).root_module(db);
+ for child in root_module.children(db) {
+ if let Some(name) = child.name(db)
+ && names_match(name.as_str(), first_segment)
+ {
+ candidate_modules.push((child, true));
+ }
+ }
+ }
+ }
+ }
+
+ // Walk down the module tree for remaining path segments
+ for segment in rest_segments {
+ candidate_modules = candidate_modules
+ .into_iter()
+ .flat_map(|(module, local)| {
+ module
+ .modules_in_scope(db, !local)
+ .into_iter()
+ .filter(|(name, _)| names_match(name.as_str(), segment))
+ .map(move |(_, module)| (module, local))
+ })
+ .unique()
+ .collect();
+
+ if candidate_modules.is_empty() {
+ break;
+ }
+ }
+
+ candidate_modules.into_iter().map(|(module, _)| module).collect()
+}
+
#[derive(Default)]
pub struct SymbolIndex<'db> {
symbols: Box<[FileSymbol<'db>]>,
@@ -236,6 +428,39 @@ impl<'db> SymbolIndex<'db> {
module_symbols(db, InternedModuleId::new(db, hir::ModuleId::from(module)))
}
+
+ /// The symbol index for all extern prelude crates.
+ pub fn extern_prelude_symbols(db: &dyn HirDatabase) -> &SymbolIndex<'_> {
+ #[salsa::tracked(returns(ref))]
+ fn extern_prelude_symbols<'db>(db: &'db dyn HirDatabase) -> SymbolIndex<'db> {
+ let _p = tracing::info_span!("extern_prelude_symbols").entered();
+
+ // We call this without attaching because this runs in parallel, so we need to attach here.
+ hir::attach_db(db, || {
+ let mut collector = SymbolCollector::new(db, false);
+
+ for krate in Crate::all(db) {
+ if krate
+ .display_name(db)
+ .is_none_or(|name| name.canonical_name().as_str() == "build-script-build")
+ {
+ continue;
+ }
+ if let CrateOrigin::Lang(LangCrateOrigin::Dependency | LangCrateOrigin::Other) =
+ krate.origin(db)
+ {
+ // don't show dependencies of the sysroot
+ continue;
+ }
+ collector.push_crate_root(krate);
+ }
+
+ SymbolIndex::new(collector.finish())
+ })
+ }
+
+ extern_prelude_symbols(db)
+ }
}
impl fmt::Debug for SymbolIndex<'_> {
@@ -336,12 +561,15 @@ impl<'db> SymbolIndex<'db> {
}
impl Query {
+ /// Search symbols in the given indices.
pub(crate) fn search<'db, T>(
- self,
+ &self,
+ db: &'db RootDatabase,
indices: &[&'db SymbolIndex<'db>],
cb: impl FnMut(&'db FileSymbol<'db>) -> ControlFlow<T>,
) -> Option<T> {
let _p = tracing::info_span!("symbol_index::Query::search").entered();
+
let mut op = fst::map::OpBuilder::new();
match self.mode {
SearchMode::Exact => {
@@ -350,7 +578,7 @@ impl Query {
for index in indices.iter() {
op = op.add(index.map.search(&automaton));
}
- self.search_maps(indices, op.union(), cb)
+ self.search_maps(db, indices, op.union(), cb)
}
SearchMode::Fuzzy => {
let automaton = fst::automaton::Subsequence::new(&self.lowercased);
@@ -358,7 +586,7 @@ impl Query {
for index in indices.iter() {
op = op.add(index.map.search(&automaton));
}
- self.search_maps(indices, op.union(), cb)
+ self.search_maps(db, indices, op.union(), cb)
}
SearchMode::Prefix => {
let automaton = fst::automaton::Str::new(&self.lowercased).starts_with();
@@ -366,13 +594,14 @@ impl Query {
for index in indices.iter() {
op = op.add(index.map.search(&automaton));
}
- self.search_maps(indices, op.union(), cb)
+ self.search_maps(db, indices, op.union(), cb)
}
}
}
fn search_maps<'db, T>(
&self,
+ db: &'db RootDatabase,
indices: &[&'db SymbolIndex<'db>],
mut stream: fst::map::Union<'_>,
mut cb: impl FnMut(&'db FileSymbol<'db>) -> ControlFlow<T>,
@@ -380,18 +609,21 @@ impl Query {
let ignore_underscore_prefixed = !self.query.starts_with("__");
while let Some((_, indexed_values)) = stream.next() {
for &IndexedValue { index, value } in indexed_values {
- let symbol_index = &indices[index];
+ let symbol_index = indices[index];
let (start, end) = SymbolIndex::map_value_to_range(value);
for symbol in &symbol_index.symbols[start..end] {
let non_type_for_type_only_query = self.only_types
- && !matches!(
+ && !(matches!(
symbol.def,
hir::ModuleDef::Adt(..)
| hir::ModuleDef::TypeAlias(..)
| hir::ModuleDef::BuiltinType(..)
| hir::ModuleDef::Trait(..)
- );
+ ) || matches!(
+ symbol.def,
+ hir::ModuleDef::Module(module) if module.is_crate_root(db)
+ ));
if non_type_for_type_only_query || !self.matches_assoc_mode(symbol.is_assoc) {
continue;
}
@@ -576,4 +808,416 @@ pub struct Foo;
let symbols = world_symbols(&db, query);
expect_file!["./test_data/test_symbols_exclude_imports.txt"].assert_debug_eq(&symbols);
}
+
+ #[test]
+ fn test_parse_path_query() {
+ // Plain query - no path
+ let (path, item, anchor) = Query::parse_path_query("Item");
+ assert_eq!(path, Vec::<String>::new());
+ assert_eq!(item, "Item");
+ assert!(!anchor);
+
+ // Path with item
+ let (path, item, anchor) = Query::parse_path_query("foo::Item");
+ assert_eq!(path, vec!["foo"]);
+ assert_eq!(item, "Item");
+ assert!(!anchor);
+
+ // Multi-segment path
+ let (path, item, anchor) = Query::parse_path_query("foo::bar::Item");
+ assert_eq!(path, vec!["foo", "bar"]);
+ assert_eq!(item, "Item");
+ assert!(!anchor);
+
+ // Leading :: (anchor to crate)
+ let (path, item, anchor) = Query::parse_path_query("::std::vec::Vec");
+ assert_eq!(path, vec!["std", "vec"]);
+ assert_eq!(item, "Vec");
+ assert!(anchor);
+
+ // Just "::" - return all crates
+ let (path, item, anchor) = Query::parse_path_query("::");
+ assert_eq!(path, Vec::<String>::new());
+ assert_eq!(item, "");
+ assert!(anchor);
+
+ // "::foo" - fuzzy search crate names
+ let (path, item, anchor) = Query::parse_path_query("::foo");
+ assert_eq!(path, Vec::<String>::new());
+ assert_eq!(item, "foo");
+ assert!(anchor);
+
+ // Trailing ::
+ let (path, item, anchor) = Query::parse_path_query("foo::");
+ assert_eq!(path, vec!["foo"]);
+ assert_eq!(item, "");
+ assert!(!anchor);
+
+ // Full path with trailing ::
+ let (path, item, anchor) = Query::parse_path_query("foo::bar::");
+ assert_eq!(path, vec!["foo", "bar"]);
+ assert_eq!(item, "");
+ assert!(!anchor);
+
+ // Absolute path with trailing ::
+ let (path, item, anchor) = Query::parse_path_query("::std::vec::");
+ assert_eq!(path, vec!["std", "vec"]);
+ assert_eq!(item, "");
+ assert!(anchor);
+
+ // Empty segments should be filtered
+ let (path, item, anchor) = Query::parse_path_query("foo::::bar");
+ assert_eq!(path, vec!["foo"]);
+ assert_eq!(item, "bar");
+ assert!(!anchor);
+ }
+
+ #[test]
+ fn test_path_search() {
+ let (mut db, _) = RootDatabase::with_many_files(
+ r#"
+//- /lib.rs crate:main
+mod inner;
+pub struct RootStruct;
+
+//- /inner.rs
+pub struct InnerStruct;
+pub mod nested {
+ pub struct NestedStruct;
+}
+"#,
+ );
+
+ let mut local_roots = FxHashSet::default();
+ local_roots.insert(WORKSPACE);
+ LocalRoots::get(&db).set_roots(&mut db).to(local_roots);
+
+ // Search for item in specific module
+ let query = Query::new("inner::InnerStruct".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(names.contains(&"InnerStruct"), "Expected InnerStruct in {:?}", names);
+
+ // Search for item in nested module
+ let query = Query::new("inner::nested::NestedStruct".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(names.contains(&"NestedStruct"), "Expected NestedStruct in {:?}", names);
+
+ // Search with crate prefix
+ let query = Query::new("main::inner::InnerStruct".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(names.contains(&"InnerStruct"), "Expected InnerStruct in {:?}", names);
+
+ // Wrong path should return empty
+ let query = Query::new("wrong::InnerStruct".to_owned());
+ let symbols = world_symbols(&db, query);
+ assert!(symbols.is_empty(), "Expected empty results for wrong path");
+ }
+
+ #[test]
+ fn test_path_search_module() {
+ let (mut db, _) = RootDatabase::with_many_files(
+ r#"
+//- /lib.rs crate:main
+mod mymod;
+
+//- /mymod.rs
+pub struct MyStruct;
+pub fn my_func() {}
+pub const MY_CONST: u32 = 1;
+"#,
+ );
+
+ let mut local_roots = FxHashSet::default();
+ local_roots.insert(WORKSPACE);
+ LocalRoots::get(&db).set_roots(&mut db).to(local_roots);
+
+ // Browse all items in module
+ let query = Query::new("main::mymod::".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+
+ assert!(names.contains(&"MyStruct"), "Expected MyStruct in {:?}", names);
+ assert!(names.contains(&"my_func"), "Expected my_func in {:?}", names);
+ assert!(names.contains(&"MY_CONST"), "Expected MY_CONST in {:?}", names);
+ }
+
+ #[test]
+ fn test_fuzzy_item_with_path() {
+ let (mut db, _) = RootDatabase::with_many_files(
+ r#"
+//- /lib.rs crate:main
+mod mymod;
+
+//- /mymod.rs
+pub struct MyLongStructName;
+"#,
+ );
+
+ let mut local_roots = FxHashSet::default();
+ local_roots.insert(WORKSPACE);
+ LocalRoots::get(&db).set_roots(&mut db).to(local_roots);
+
+ // Fuzzy match on item name with exact path
+ let query = Query::new("main::mymod::MyLong".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(
+ names.contains(&"MyLongStructName"),
+ "Expected fuzzy match for MyLongStructName in {:?}",
+ names
+ );
+ }
+
+ #[test]
+ fn test_case_insensitive_path() {
+ let (mut db, _) = RootDatabase::with_many_files(
+ r#"
+//- /lib.rs crate:main
+mod MyMod;
+
+//- /MyMod.rs
+pub struct MyStruct;
+"#,
+ );
+
+ let mut local_roots = FxHashSet::default();
+ local_roots.insert(WORKSPACE);
+ LocalRoots::get(&db).set_roots(&mut db).to(local_roots);
+
+ // Case insensitive path matching (default)
+ let query = Query::new("main::mymod::MyStruct".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(names.contains(&"MyStruct"), "Expected case-insensitive match in {:?}", names);
+ }
+
+ #[test]
+ fn test_absolute_path_search() {
+ let (mut db, _) = RootDatabase::with_many_files(
+ r#"
+//- /lib.rs crate:mycrate
+mod inner;
+pub struct CrateRoot;
+
+//- /inner.rs
+pub struct InnerItem;
+"#,
+ );
+
+ let mut local_roots = FxHashSet::default();
+ local_roots.insert(WORKSPACE);
+ LocalRoots::get(&db).set_roots(&mut db).to(local_roots);
+
+ // Absolute path with leading ::
+ let query = Query::new("::mycrate::inner::InnerItem".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(
+ names.contains(&"InnerItem"),
+ "Expected InnerItem with absolute path in {:?}",
+ names
+ );
+
+ // Absolute path should NOT match if crate name is wrong
+ let query = Query::new("::wrongcrate::inner::InnerItem".to_owned());
+ let symbols = world_symbols(&db, query);
+ assert!(symbols.is_empty(), "Expected empty results for wrong crate name");
+ }
+
+ #[test]
+ fn test_wrong_path_returns_empty() {
+ let (mut db, _) = RootDatabase::with_many_files(
+ r#"
+//- /lib.rs crate:main
+mod existing;
+
+//- /existing.rs
+pub struct MyStruct;
+"#,
+ );
+
+ let mut local_roots = FxHashSet::default();
+ local_roots.insert(WORKSPACE);
+ LocalRoots::get(&db).set_roots(&mut db).to(local_roots);
+
+ // Non-existent module path
+ let query = Query::new("nonexistent::MyStruct".to_owned());
+ let symbols = world_symbols(&db, query);
+ assert!(symbols.is_empty(), "Expected empty results for non-existent path");
+
+ // Correct item, wrong module
+ let query = Query::new("wrongmod::MyStruct".to_owned());
+ let symbols = world_symbols(&db, query);
+ assert!(symbols.is_empty(), "Expected empty results for wrong module");
+ }
+
+ #[test]
+ fn test_root_module_items() {
+ let (mut db, _) = RootDatabase::with_many_files(
+ r#"
+//- /lib.rs crate:mylib
+pub struct RootItem;
+pub fn root_fn() {}
+"#,
+ );
+
+ let mut local_roots = FxHashSet::default();
+ local_roots.insert(WORKSPACE);
+ LocalRoots::get(&db).set_roots(&mut db).to(local_roots);
+
+ // Items at crate root - path is just the crate name
+ let query = Query::new("mylib::RootItem".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(names.contains(&"RootItem"), "Expected RootItem at crate root in {:?}", names);
+
+ let query = Query::new("mylib::".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(names.contains(&"RootItem"), "Expected RootItem {:?}", names);
+ assert!(names.contains(&"root_fn"), "Expected root_fn {:?}", names);
+ }
+
+ #[test]
+ fn test_crate_search_all() {
+ // Test that sole "::" returns all crates
+ let (mut db, _) = RootDatabase::with_many_files(
+ r#"
+//- /lib.rs crate:alpha
+pub struct AlphaStruct;
+
+//- /beta.rs crate:beta
+pub struct BetaStruct;
+
+//- /gamma.rs crate:gamma
+pub struct GammaStruct;
+"#,
+ );
+
+ let mut local_roots = FxHashSet::default();
+ local_roots.insert(WORKSPACE);
+ LocalRoots::get(&db).set_roots(&mut db).to(local_roots);
+
+ // Sole "::" should return all crates (as module symbols)
+ let query = Query::new("::".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+
+ assert!(names.contains(&"alpha"), "Expected alpha crate in {:?}", names);
+ assert!(names.contains(&"beta"), "Expected beta crate in {:?}", names);
+ assert!(names.contains(&"gamma"), "Expected gamma crate in {:?}", names);
+ assert_eq!(symbols.len(), 3, "Expected exactly 3 crates, got {:?}", names);
+ }
+
+ #[test]
+ fn test_crate_search_fuzzy() {
+ // Test that "::foo" fuzzy-matches crate names
+ let (mut db, _) = RootDatabase::with_many_files(
+ r#"
+//- /lib.rs crate:my_awesome_lib
+pub struct AwesomeStruct;
+
+//- /other.rs crate:another_lib
+pub struct OtherStruct;
+
+//- /foo.rs crate:foobar
+pub struct FooStruct;
+"#,
+ );
+
+ let mut local_roots = FxHashSet::default();
+ local_roots.insert(WORKSPACE);
+ LocalRoots::get(&db).set_roots(&mut db).to(local_roots);
+
+ // "::foo" should fuzzy-match crate names containing "foo"
+ let query = Query::new("::foo".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+
+ assert!(names.contains(&"foobar"), "Expected foobar crate in {:?}", names);
+ assert_eq!(symbols.len(), 1, "Expected only foobar crate, got {:?}", names);
+
+ // "::awesome" should match my_awesome_lib
+ let query = Query::new("::awesome".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+
+ assert!(names.contains(&"my_awesome_lib"), "Expected my_awesome_lib crate in {:?}", names);
+ assert_eq!(symbols.len(), 1, "Expected only my_awesome_lib crate, got {:?}", names);
+
+ // "::lib" should match multiple crates
+ let query = Query::new("::lib".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+
+ assert!(names.contains(&"my_awesome_lib"), "Expected my_awesome_lib in {:?}", names);
+ assert!(names.contains(&"another_lib"), "Expected another_lib in {:?}", names);
+ assert_eq!(symbols.len(), 2, "Expected 2 crates matching 'lib', got {:?}", names);
+
+ // "::nonexistent" should return empty
+ let query = Query::new("::nonexistent".to_owned());
+ let symbols = world_symbols(&db, query);
+ assert!(symbols.is_empty(), "Expected empty results for non-matching crate pattern");
+ }
+
+ #[test]
+ fn test_path_search_with_use_reexport() {
+ // Test that module resolution works for `use` items (re-exports), not just `mod` items
+ let (mut db, _) = RootDatabase::with_many_files(
+ r#"
+//- /lib.rs crate:main
+mod inner;
+pub use inner::nested;
+
+//- /inner.rs
+pub mod nested {
+ pub struct NestedStruct;
+ pub fn nested_fn() {}
+}
+"#,
+ );
+
+ let mut local_roots = FxHashSet::default();
+ local_roots.insert(WORKSPACE);
+ LocalRoots::get(&db).set_roots(&mut db).to(local_roots);
+
+ // Search via the re-exported path (main::nested::NestedStruct)
+ // This should work because `nested` is in scope via `pub use inner::nested`
+ let query = Query::new("main::nested::NestedStruct".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(
+ names.contains(&"NestedStruct"),
+ "Expected NestedStruct via re-exported path in {:?}",
+ names
+ );
+
+ // Also verify the original path still works
+ let query = Query::new("main::inner::nested::NestedStruct".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(
+ names.contains(&"NestedStruct"),
+ "Expected NestedStruct via original path in {:?}",
+ names
+ );
+
+ // Browse the re-exported module
+ let query = Query::new("main::nested::".to_owned());
+ let symbols = world_symbols(&db, query);
+ let names: Vec<_> = symbols.iter().map(|s| s.name.as_str()).collect();
+ assert!(
+ names.contains(&"NestedStruct"),
+ "Expected NestedStruct when browsing re-exported module in {:?}",
+ names
+ );
+ assert!(
+ names.contains(&"nested_fn"),
+ "Expected nested_fn when browsing re-exported module in {:?}",
+ names
+ );
+ }
}