mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-25 19:35:06 +00:00
Simplify exclusion logic
This commit is contained in:
parent
fd6717799c
commit
46ac9ff5e3
7 changed files with 120 additions and 141 deletions
30
Cargo.lock
generated
30
Cargo.lock
generated
|
@ -103,15 +103,6 @@ version = "1.2.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "cf1de2fe8c75bc145a2f577add951f8134889b4795d47466a54a5c846d691693"
|
checksum = "cf1de2fe8c75bc145a2f577add951f8134889b4795d47466a54a5c846d691693"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "bstr"
|
|
||||||
version = "0.2.13"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "31accafdb70df7871592c058eca3985b71104e15ac32f64706022c58867da931"
|
|
||||||
dependencies = [
|
|
||||||
"memchr",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "byteorder"
|
name = "byteorder"
|
||||||
version = "1.3.4"
|
version = "1.3.4"
|
||||||
|
@ -403,12 +394,6 @@ dependencies = [
|
||||||
"serde_json",
|
"serde_json",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "fnv"
|
|
||||||
version = "1.0.7"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "fs_extra"
|
name = "fs_extra"
|
||||||
version = "1.1.0"
|
version = "1.1.0"
|
||||||
|
@ -473,19 +458,6 @@ version = "0.22.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "aaf91faf136cb47367fa430cd46e37a788775e7fa104f8b4bcb3861dc389b724"
|
checksum = "aaf91faf136cb47367fa430cd46e37a788775e7fa104f8b4bcb3861dc389b724"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "globset"
|
|
||||||
version = "0.4.5"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "7ad1da430bd7281dde2576f44c84cc3f0f7b475e7202cd503042dff01a8c8120"
|
|
||||||
dependencies = [
|
|
||||||
"aho-corasick",
|
|
||||||
"bstr",
|
|
||||||
"fnv",
|
|
||||||
"log",
|
|
||||||
"regex",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "goblin"
|
name = "goblin"
|
||||||
version = "0.2.3"
|
version = "0.2.3"
|
||||||
|
@ -1494,7 +1466,6 @@ dependencies = [
|
||||||
"env_logger",
|
"env_logger",
|
||||||
"expect",
|
"expect",
|
||||||
"flycheck",
|
"flycheck",
|
||||||
"globset",
|
|
||||||
"itertools",
|
"itertools",
|
||||||
"jod-thread",
|
"jod-thread",
|
||||||
"log",
|
"log",
|
||||||
|
@ -2003,7 +1974,6 @@ name = "vfs-notify"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"crossbeam-channel",
|
"crossbeam-channel",
|
||||||
"globset",
|
|
||||||
"jod-thread",
|
"jod-thread",
|
||||||
"log",
|
"log",
|
||||||
"notify",
|
"notify",
|
||||||
|
|
|
@ -17,7 +17,6 @@ path = "src/bin/main.rs"
|
||||||
anyhow = "1.0.26"
|
anyhow = "1.0.26"
|
||||||
crossbeam-channel = "0.4.0"
|
crossbeam-channel = "0.4.0"
|
||||||
env_logger = { version = "0.7.1", default-features = false }
|
env_logger = { version = "0.7.1", default-features = false }
|
||||||
globset = "0.4.4"
|
|
||||||
itertools = "0.9.0"
|
itertools = "0.9.0"
|
||||||
jod-thread = "0.1.0"
|
jod-thread = "0.1.0"
|
||||||
log = "0.4.8"
|
log = "0.4.8"
|
||||||
|
|
|
@ -13,7 +13,6 @@ log = "0.4.8"
|
||||||
rustc-hash = "1.0"
|
rustc-hash = "1.0"
|
||||||
jod-thread = "0.1.0"
|
jod-thread = "0.1.0"
|
||||||
walkdir = "2.3.1"
|
walkdir = "2.3.1"
|
||||||
globset = "0.4.5"
|
|
||||||
crossbeam-channel = "0.4.0"
|
crossbeam-channel = "0.4.0"
|
||||||
notify = "5.0.0-pre.3"
|
notify = "5.0.0-pre.3"
|
||||||
|
|
||||||
|
|
|
@ -1,42 +0,0 @@
|
||||||
//! See `Include`.
|
|
||||||
use std::convert::TryFrom;
|
|
||||||
|
|
||||||
use globset::{Glob, GlobSet, GlobSetBuilder};
|
|
||||||
use paths::{RelPath, RelPathBuf};
|
|
||||||
|
|
||||||
/// `Include` is the opposite of .gitignore.
|
|
||||||
///
|
|
||||||
/// It describes the set of files inside some directory.
|
|
||||||
///
|
|
||||||
/// The current implementation is very limited, it allows including file globs
|
|
||||||
/// and recursively excluding directories.
|
|
||||||
#[derive(Debug, Clone)]
|
|
||||||
pub(crate) struct Include {
|
|
||||||
include_files: GlobSet,
|
|
||||||
exclude_dirs: Vec<RelPathBuf>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Include {
|
|
||||||
pub(crate) fn new(include: Vec<String>) -> Include {
|
|
||||||
let mut include_files = GlobSetBuilder::new();
|
|
||||||
let mut exclude_dirs = Vec::new();
|
|
||||||
|
|
||||||
for glob in include {
|
|
||||||
if glob.starts_with("!/") {
|
|
||||||
if let Ok(path) = RelPathBuf::try_from(&glob["!/".len()..]) {
|
|
||||||
exclude_dirs.push(path)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
include_files.add(Glob::new(&glob).unwrap());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
let include_files = include_files.build().unwrap();
|
|
||||||
Include { include_files, exclude_dirs }
|
|
||||||
}
|
|
||||||
pub(crate) fn include_file(&self, path: &RelPath) -> bool {
|
|
||||||
self.include_files.is_match(path)
|
|
||||||
}
|
|
||||||
pub(crate) fn exclude_dir(&self, path: &RelPath) -> bool {
|
|
||||||
self.exclude_dirs.iter().any(|excluded| path.starts_with(excluded))
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -6,9 +6,7 @@
|
||||||
//!
|
//!
|
||||||
//! Hopefully, one day a reliable file watching/walking crate appears on
|
//! Hopefully, one day a reliable file watching/walking crate appears on
|
||||||
//! crates.io, and we can reduce this to trivial glue code.
|
//! crates.io, and we can reduce this to trivial glue code.
|
||||||
mod include;
|
use std::convert::TryFrom;
|
||||||
|
|
||||||
use std::convert::{TryFrom, TryInto};
|
|
||||||
|
|
||||||
use crossbeam_channel::{never, select, unbounded, Receiver, Sender};
|
use crossbeam_channel::{never, select, unbounded, Receiver, Sender};
|
||||||
use notify::{RecommendedWatcher, RecursiveMode, Watcher};
|
use notify::{RecommendedWatcher, RecursiveMode, Watcher};
|
||||||
|
@ -16,8 +14,6 @@ use paths::{AbsPath, AbsPathBuf};
|
||||||
use vfs::loader;
|
use vfs::loader;
|
||||||
use walkdir::WalkDir;
|
use walkdir::WalkDir;
|
||||||
|
|
||||||
use crate::include::Include;
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct NotifyHandle {
|
pub struct NotifyHandle {
|
||||||
// Relative order of fields below is significant.
|
// Relative order of fields below is significant.
|
||||||
|
@ -53,7 +49,7 @@ type NotifyEvent = notify::Result<notify::Event>;
|
||||||
|
|
||||||
struct NotifyActor {
|
struct NotifyActor {
|
||||||
sender: loader::Sender,
|
sender: loader::Sender,
|
||||||
config: Vec<(AbsPathBuf, Include, bool)>,
|
watched_entries: Vec<loader::Entry>,
|
||||||
// Drop order is significant.
|
// Drop order is significant.
|
||||||
watcher: Option<(RecommendedWatcher, Receiver<NotifyEvent>)>,
|
watcher: Option<(RecommendedWatcher, Receiver<NotifyEvent>)>,
|
||||||
}
|
}
|
||||||
|
@ -66,7 +62,7 @@ enum Event {
|
||||||
|
|
||||||
impl NotifyActor {
|
impl NotifyActor {
|
||||||
fn new(sender: loader::Sender) -> NotifyActor {
|
fn new(sender: loader::Sender) -> NotifyActor {
|
||||||
NotifyActor { sender, config: Vec::new(), watcher: None }
|
NotifyActor { sender, watched_entries: Vec::new(), watcher: None }
|
||||||
}
|
}
|
||||||
fn next_event(&self, receiver: &Receiver<Message>) -> Option<Event> {
|
fn next_event(&self, receiver: &Receiver<Message>) -> Option<Event> {
|
||||||
let watcher_receiver = self.watcher.as_ref().map(|(_, receiver)| receiver);
|
let watcher_receiver = self.watcher.as_ref().map(|(_, receiver)| receiver);
|
||||||
|
@ -93,15 +89,17 @@ impl NotifyActor {
|
||||||
let n_total = config.load.len();
|
let n_total = config.load.len();
|
||||||
self.send(loader::Message::Progress { n_total, n_done: 0 });
|
self.send(loader::Message::Progress { n_total, n_done: 0 });
|
||||||
|
|
||||||
self.config.clear();
|
self.watched_entries.clear();
|
||||||
|
|
||||||
for (i, entry) in config.load.into_iter().enumerate() {
|
for (i, entry) in config.load.into_iter().enumerate() {
|
||||||
let watch = config.watch.contains(&i);
|
let watch = config.watch.contains(&i);
|
||||||
|
if watch {
|
||||||
|
self.watched_entries.push(entry.clone())
|
||||||
|
}
|
||||||
let files = self.load_entry(entry, watch);
|
let files = self.load_entry(entry, watch);
|
||||||
self.send(loader::Message::Loaded { files });
|
self.send(loader::Message::Loaded { files });
|
||||||
self.send(loader::Message::Progress { n_total, n_done: i + 1 });
|
self.send(loader::Message::Progress { n_total, n_done: i + 1 });
|
||||||
}
|
}
|
||||||
self.config.sort_by(|x, y| x.0.cmp(&y.0));
|
|
||||||
}
|
}
|
||||||
Message::Invalidate(path) => {
|
Message::Invalidate(path) => {
|
||||||
let contents = read(path.as_path());
|
let contents = read(path.as_path());
|
||||||
|
@ -116,34 +114,27 @@ impl NotifyActor {
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|path| AbsPathBuf::try_from(path).unwrap())
|
.map(|path| AbsPathBuf::try_from(path).unwrap())
|
||||||
.filter_map(|path| {
|
.filter_map(|path| {
|
||||||
let is_dir = path.is_dir();
|
if path.is_dir()
|
||||||
let is_file = path.is_file();
|
&& self
|
||||||
|
.watched_entries
|
||||||
let config_idx =
|
.iter()
|
||||||
match self.config.binary_search_by(|it| it.0.cmp(&path)) {
|
.any(|entry| entry.contains_dir(&path))
|
||||||
Ok(it) => it,
|
{
|
||||||
Err(it) => it.saturating_sub(1),
|
|
||||||
};
|
|
||||||
let include = self.config.get(config_idx).and_then(|it| {
|
|
||||||
let rel_path = path.strip_prefix(&it.0)?;
|
|
||||||
Some((rel_path, &it.1))
|
|
||||||
});
|
|
||||||
|
|
||||||
if let Some((rel_path, include)) = include {
|
|
||||||
if is_dir && include.exclude_dir(&rel_path)
|
|
||||||
|| is_file && !include.include_file(&rel_path)
|
|
||||||
{
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if is_dir {
|
|
||||||
self.watch(path);
|
self.watch(path);
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
if !is_file {
|
|
||||||
|
if !path.is_file() {
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
if !self
|
||||||
|
.watched_entries
|
||||||
|
.iter()
|
||||||
|
.any(|entry| entry.contains_file(&path))
|
||||||
|
{
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
let contents = read(&path);
|
let contents = read(&path);
|
||||||
Some((path, contents))
|
Some((path, contents))
|
||||||
})
|
})
|
||||||
|
@ -170,43 +161,42 @@ impl NotifyActor {
|
||||||
(file, contents)
|
(file, contents)
|
||||||
})
|
})
|
||||||
.collect::<Vec<_>>(),
|
.collect::<Vec<_>>(),
|
||||||
loader::Entry::Directory { path, include } => {
|
loader::Entry::Directories(dirs) => {
|
||||||
let include = Include::new(include);
|
let mut res = Vec::new();
|
||||||
self.config.push((path.clone(), include.clone(), watch));
|
|
||||||
|
|
||||||
let files = WalkDir::new(&path)
|
for root in dirs.include.iter() {
|
||||||
.into_iter()
|
let walkdir = WalkDir::new(root).into_iter().filter_entry(|entry| {
|
||||||
.filter_entry(|entry| {
|
if !entry.file_type().is_dir() {
|
||||||
let abs_path: &AbsPath = entry.path().try_into().unwrap();
|
return true;
|
||||||
match abs_path.strip_prefix(&path) {
|
|
||||||
Some(rel_path) => {
|
|
||||||
!(entry.file_type().is_dir() && include.exclude_dir(rel_path))
|
|
||||||
}
|
|
||||||
None => false,
|
|
||||||
}
|
}
|
||||||
})
|
let path = AbsPath::assert(entry.path());
|
||||||
.filter_map(|entry| entry.ok())
|
root == path
|
||||||
.filter_map(|entry| {
|
|| dirs.exclude.iter().chain(&dirs.include).all(|it| it != path)
|
||||||
|
});
|
||||||
|
|
||||||
|
let files = walkdir.filter_map(|it| it.ok()).filter_map(|entry| {
|
||||||
let is_dir = entry.file_type().is_dir();
|
let is_dir = entry.file_type().is_dir();
|
||||||
let is_file = entry.file_type().is_file();
|
let is_file = entry.file_type().is_file();
|
||||||
let abs_path = AbsPathBuf::try_from(entry.into_path()).unwrap();
|
let abs_path = AbsPathBuf::assert(entry.into_path());
|
||||||
if is_dir && watch {
|
if is_dir && watch {
|
||||||
self.watch(abs_path.clone());
|
self.watch(abs_path.clone());
|
||||||
}
|
}
|
||||||
let rel_path = abs_path.strip_prefix(&path)?;
|
if !is_file {
|
||||||
if is_file && include.include_file(&rel_path) {
|
return None;
|
||||||
Some(abs_path)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
}
|
||||||
|
let ext = abs_path.extension().unwrap_or_default();
|
||||||
|
if dirs.extensions.iter().all(|it| it.as_str() != ext) {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
Some(abs_path)
|
||||||
});
|
});
|
||||||
|
|
||||||
files
|
res.extend(files.map(|file| {
|
||||||
.map(|file| {
|
|
||||||
let contents = read(file.as_path());
|
let contents = read(file.as_path());
|
||||||
(file, contents)
|
(file, contents)
|
||||||
})
|
}));
|
||||||
.collect()
|
}
|
||||||
|
res
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,10 +3,25 @@ use std::fmt;
|
||||||
|
|
||||||
use paths::{AbsPath, AbsPathBuf};
|
use paths::{AbsPath, AbsPathBuf};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug, Clone)]
|
||||||
pub enum Entry {
|
pub enum Entry {
|
||||||
Files(Vec<AbsPathBuf>),
|
Files(Vec<AbsPathBuf>),
|
||||||
Directory { path: AbsPathBuf, include: Vec<String> },
|
Directories(Directories),
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Specifies a set of files on the file system.
|
||||||
|
///
|
||||||
|
/// A file is included if:
|
||||||
|
/// * it has included extension
|
||||||
|
/// * it is under an `include` path
|
||||||
|
/// * it is not under `exclude` path
|
||||||
|
///
|
||||||
|
/// If many include/exclude paths match, the longest one wins.
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct Directories {
|
||||||
|
pub extensions: Vec<String>,
|
||||||
|
pub include: Vec<AbsPathBuf>,
|
||||||
|
pub exclude: Vec<AbsPathBuf>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
|
@ -33,21 +48,70 @@ pub trait Handle: fmt::Debug {
|
||||||
|
|
||||||
impl Entry {
|
impl Entry {
|
||||||
pub fn rs_files_recursively(base: AbsPathBuf) -> Entry {
|
pub fn rs_files_recursively(base: AbsPathBuf) -> Entry {
|
||||||
Entry::Directory { path: base, include: globs(&["*.rs", "!/.git/"]) }
|
Entry::Directories(dirs(base, &[".git"]))
|
||||||
}
|
}
|
||||||
pub fn local_cargo_package(base: AbsPathBuf) -> Entry {
|
pub fn local_cargo_package(base: AbsPathBuf) -> Entry {
|
||||||
Entry::Directory { path: base, include: globs(&["*.rs", "!/target/", "!/.git/"]) }
|
Entry::Directories(dirs(base, &[".git", "target"]))
|
||||||
}
|
}
|
||||||
pub fn cargo_package_dependency(base: AbsPathBuf) -> Entry {
|
pub fn cargo_package_dependency(base: AbsPathBuf) -> Entry {
|
||||||
Entry::Directory {
|
Entry::Directories(dirs(base, &[".git", "/tests", "/examples", "/benches"]))
|
||||||
path: base,
|
}
|
||||||
include: globs(&["*.rs", "!/tests/", "!/examples/", "!/benches/", "!/.git/"]),
|
|
||||||
|
pub fn contains_file(&self, path: &AbsPath) -> bool {
|
||||||
|
match self {
|
||||||
|
Entry::Files(files) => files.iter().any(|it| it == path),
|
||||||
|
Entry::Directories(dirs) => dirs.contains_file(path),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn contains_dir(&self, path: &AbsPath) -> bool {
|
||||||
|
match self {
|
||||||
|
Entry::Files(_) => false,
|
||||||
|
Entry::Directories(dirs) => dirs.contains_dir(path),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn globs(globs: &[&str]) -> Vec<String> {
|
impl Directories {
|
||||||
globs.iter().map(|it| it.to_string()).collect()
|
pub fn contains_file(&self, path: &AbsPath) -> bool {
|
||||||
|
let ext = path.extension().unwrap_or_default();
|
||||||
|
if self.extensions.iter().all(|it| it.as_str() != ext) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
self.includes_path(path)
|
||||||
|
}
|
||||||
|
pub fn contains_dir(&self, path: &AbsPath) -> bool {
|
||||||
|
self.includes_path(path)
|
||||||
|
}
|
||||||
|
fn includes_path(&self, path: &AbsPath) -> bool {
|
||||||
|
let mut include = None;
|
||||||
|
for incl in &self.include {
|
||||||
|
if is_prefix(incl, path) {
|
||||||
|
include = Some(match include {
|
||||||
|
Some(prev) if is_prefix(incl, prev) => prev,
|
||||||
|
_ => incl,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let include = match include {
|
||||||
|
Some(it) => it,
|
||||||
|
None => return false,
|
||||||
|
};
|
||||||
|
for excl in &self.exclude {
|
||||||
|
if is_prefix(excl, path) && is_prefix(include, excl) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
|
||||||
|
fn is_prefix(short: &AbsPath, long: &AbsPath) -> bool {
|
||||||
|
long.strip_prefix(short).is_some()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn dirs(base: AbsPathBuf, exclude: &[&str]) -> Directories {
|
||||||
|
let exclude = exclude.iter().map(|it| base.join(it)).collect::<Vec<_>>();
|
||||||
|
Directories { extensions: vec!["rs".to_string()], include: vec![base], exclude }
|
||||||
}
|
}
|
||||||
|
|
||||||
impl fmt::Debug for Message {
|
impl fmt::Debug for Message {
|
||||||
|
|
|
@ -54,7 +54,6 @@ fn check_licenses() {
|
||||||
let expected = "
|
let expected = "
|
||||||
0BSD OR MIT OR Apache-2.0
|
0BSD OR MIT OR Apache-2.0
|
||||||
Apache-2.0
|
Apache-2.0
|
||||||
Apache-2.0 / MIT
|
|
||||||
Apache-2.0 OR BSL-1.0
|
Apache-2.0 OR BSL-1.0
|
||||||
Apache-2.0 OR MIT
|
Apache-2.0 OR MIT
|
||||||
Apache-2.0 WITH LLVM-exception OR Apache-2.0 OR MIT
|
Apache-2.0 WITH LLVM-exception OR Apache-2.0 OR MIT
|
||||||
|
|
Loading…
Reference in a new issue