Skip to content

Commit 133d001

Browse files
committed
world symbols
1 parent ed2ac17 commit 133d001

File tree

4 files changed

+199
-7
lines changed

4 files changed

+199
-7
lines changed

crates/libanalysis/Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -8,5 +8,6 @@ log = "0.4.2"
88
failure = "0.1.2"
99
parking_lot = "0.6.3"
1010
once_cell = "0.1.4"
11+
fst = { git = "https://github.com/matklad/fst", branch = "subsequence"}
1112
libsyntax2 = { path = "../libsyntax2" }
1213
libeditor = { path = "../libeditor" }

crates/libanalysis/idx.rs

+84
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
1+
use std::path::PathBuf;
2+
3+
use fst;
4+
use fst::IntoStreamer;
5+
use file;
6+
7+
use fall_tree::{TextRange, NodeType};
8+
use indxr::{FileIndex, IndexableFileSet};
9+
10+
use editor::line_index::{LineCol, LineIndex};
11+
use editor::fst_subseq::FstSubSeq;
12+
use editor::file_symbols::process_symbols;
13+
14+
use syntax::{STRUCT_DEF, ENUM_DEF, TRAIT_DEF, TYPE_DEF};
15+
16+
17+
pub struct SymbolIndex {
18+
index: FileIndex<FileSymbols>,
19+
}
20+
21+
impl SymbolIndex {
22+
pub fn new(roots: Vec<PathBuf>) -> SymbolIndex {
23+
let file_set = IndexableFileSet::new(roots, "rs");
24+
let index = FileIndex::new(file_set, Box::new(|path| {
25+
let text = file::get_text(path).ok()?;
26+
Some(FileSymbols::new(text))
27+
}));
28+
SymbolIndex { index }
29+
}
30+
31+
pub fn query(&self, query: &str) -> Vec<(PathBuf, Symbol)> {
32+
let mut query = Query::new(query);
33+
let mut result = Vec::new();
34+
self.process_query(&query, &mut result);
35+
if result.is_empty() && !query.all_symbols {
36+
query.all_symbols = true;
37+
self.process_query(&query, &mut result);
38+
}
39+
result
40+
}
41+
42+
fn process_query(&self, query: &Query, acc: &mut Vec<(PathBuf, Symbol)>) {
43+
self.index.process_files(&mut |file| {
44+
query.process(&file.value, &mut |symbol| {
45+
acc.push((file.path.clone(), symbol))
46+
});
47+
acc.len() > 512
48+
});
49+
}
50+
}
51+
52+
struct Query {
53+
query: String,
54+
all_symbols: bool,
55+
}
56+
57+
impl Query {
58+
fn new(query: &str) -> Query {
59+
let all_symbols = query.contains("#");
60+
let query: String = query.chars()
61+
.filter(|&c| c != '#')
62+
.flat_map(char::to_lowercase)
63+
.collect();
64+
Query { query, all_symbols }
65+
}
66+
67+
fn process(&self, file: &FileSymbols, acc: &mut FnMut(Symbol)) {
68+
fn is_type(ty: NodeType) -> bool {
69+
match ty {
70+
STRUCT_DEF | ENUM_DEF | TRAIT_DEF| TYPE_DEF => true,
71+
_ => false,
72+
}
73+
}
74+
75+
let a = FstSubSeq::new(&self.query);
76+
for idx in file.map.search(a).into_stream().into_values() {
77+
let idx = idx as usize;
78+
let symbol = file.symbols[idx].clone();
79+
if self.all_symbols || is_type(symbol.ty) {
80+
acc(symbol)
81+
}
82+
}
83+
}
84+
}

crates/libanalysis/src/lib.rs

+40-7
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,9 @@ extern crate log;
66
extern crate once_cell;
77
extern crate libsyntax2;
88
extern crate libeditor;
9+
extern crate fst;
10+
11+
mod symbol_index;
912

1013
use once_cell::sync::OnceCell;
1114

@@ -14,8 +17,11 @@ use std::{
1417
collections::hash_map::HashMap,
1518
path::{PathBuf, Path},
1619
};
20+
1721
use libsyntax2::ast;
18-
use libeditor::LineIndex;
22+
use libeditor::{LineIndex, FileSymbol};
23+
24+
use self::symbol_index::{FileSymbols, Query};
1925

2026
pub type Result<T> = ::std::result::Result<T, ::failure::Error>;
2127

@@ -70,12 +76,7 @@ impl WorldState {
7076
impl World {
7177
pub fn file_syntax(&self, path: &Path) -> Result<ast::File> {
7278
let data = self.file_data(path)?;
73-
let syntax = data.syntax
74-
.get_or_init(|| {
75-
trace!("parsing: {}", path.display());
76-
ast::File::parse(&data.text)
77-
}).clone();
78-
Ok(syntax)
79+
Ok(data.syntax(path).clone())
7980
}
8081

8182
pub fn file_line_index(&self, path: &Path) -> Result<LineIndex> {
@@ -88,6 +89,16 @@ impl World {
8889
Ok(index.clone())
8990
}
9091

92+
pub fn world_symbols(&self, query: &str, f: &mut FnMut(&Path, &FileSymbol) -> Search) {
93+
let q = Query::new(query);
94+
for (path, data) in self.data.file_map.iter() {
95+
let symbols = data.symbols(path.as_path());
96+
if q.process(symbols, &mut |symbol| f(path, symbol)) == Search::Break {
97+
break;
98+
}
99+
}
100+
}
101+
91102
fn file_data(&self, path: &Path) -> Result<Arc<FileData>> {
92103
match self.data.file_map.get(path) {
93104
Some(data) => Ok(data.clone()),
@@ -96,6 +107,12 @@ impl World {
96107
}
97108
}
98109

110+
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
111+
pub enum Search {
112+
Continue,
113+
Break,
114+
}
115+
99116

100117
#[derive(Default, Debug)]
101118
struct WorldData {
@@ -105,6 +122,7 @@ struct WorldData {
105122
#[derive(Debug)]
106123
struct FileData {
107124
text: String,
125+
symbols: OnceCell<FileSymbols>,
108126
syntax: OnceCell<ast::File>,
109127
lines: OnceCell<LineIndex>,
110128
}
@@ -113,8 +131,23 @@ impl FileData {
113131
fn new(text: String) -> FileData {
114132
FileData {
115133
text,
134+
symbols: OnceCell::new(),
116135
syntax: OnceCell::new(),
117136
lines: OnceCell::new(),
118137
}
119138
}
139+
140+
fn syntax(&self, path: &Path) -> &ast::File {
141+
self.syntax
142+
.get_or_init(|| {
143+
trace!("parsing: {}", path.display());
144+
ast::File::parse(&self.text)
145+
})
146+
}
147+
148+
fn symbols(&self, path: &Path) -> &FileSymbols {
149+
let syntax = self.syntax(path);
150+
self.symbols
151+
.get_or_init(|| FileSymbols::new(syntax))
152+
}
120153
}
+74
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
use libeditor::{FileSymbol, file_symbols};
2+
use libsyntax2::{
3+
ast,
4+
SyntaxKind::{self, *},
5+
};
6+
use fst::{self, IntoStreamer};
7+
8+
use Search;
9+
10+
#[derive(Debug)]
11+
pub(crate) struct FileSymbols {
12+
symbols: Vec<FileSymbol>,
13+
map: fst::Map,
14+
}
15+
16+
impl FileSymbols {
17+
pub(crate) fn new(file: &ast::File) -> FileSymbols {
18+
let mut symbols = file_symbols(file)
19+
.into_iter()
20+
.map(|s| (s.name.as_str().to_lowercase(), s))
21+
.collect::<Vec<_>>();
22+
23+
symbols.sort_by(|s1, s2| s1.0.cmp(&s2.0));
24+
symbols.dedup_by(|s1, s2| s1.0 == s2.0);
25+
let (names, symbols): (Vec<String>, Vec<FileSymbol>) =
26+
symbols.into_iter().unzip();
27+
28+
let map = fst::Map::from_iter(
29+
names.into_iter().zip(0u64..)
30+
).unwrap();
31+
FileSymbols { symbols, map }
32+
}
33+
}
34+
35+
pub(crate) struct Query {
36+
query: String,
37+
all_symbols: bool,
38+
}
39+
40+
impl Query {
41+
pub(crate) fn new(query: &str) -> Query {
42+
let all_symbols = query.contains("#");
43+
let query: String = query.chars()
44+
.filter(|&c| c != '#')
45+
.flat_map(char::to_lowercase)
46+
.collect();
47+
Query { query, all_symbols }
48+
}
49+
50+
pub(crate) fn process(
51+
&self,
52+
file: &FileSymbols,
53+
acc: &mut FnMut(&FileSymbol) -> Search,
54+
) -> Search {
55+
fn is_type(kind: SyntaxKind) -> bool {
56+
match kind {
57+
STRUCT | ENUM | TRAIT | TYPE_ITEM => true,
58+
_ => false,
59+
}
60+
}
61+
let automaton = fst::automaton::Subsequence::new(&self.query);
62+
for idx in file.map.search(automaton).into_stream().into_values() {
63+
let idx = idx as usize;
64+
let symbol = &file.symbols[idx];
65+
if self.all_symbols || is_type(symbol.kind) {
66+
if acc(&symbol) == Search::Break {
67+
return Search::Break;
68+
}
69+
}
70+
}
71+
Search::Continue
72+
}
73+
}
74+

0 commit comments

Comments
 (0)