mirror of
https://github.com/watchexec/watchexec.git
synced 2024-11-16 17:18:30 +01:00
254 lines
5.6 KiB
Rust
254 lines
5.6 KiB
Rust
|
#![allow(dead_code)]
|
||
|
|
||
|
use std::{
|
||
|
ffi::OsString,
|
||
|
path::{Path, PathBuf},
|
||
|
sync::Arc,
|
||
|
};
|
||
|
|
||
|
use watchexec::{
|
||
|
error::RuntimeError,
|
||
|
event::{Event, FileType, Tag},
|
||
|
filter::{
|
||
|
globset::GlobsetFilterer,
|
||
|
tagged::{Filter, Matcher, Op, Pattern, TaggedFilterer},
|
||
|
Filterer,
|
||
|
},
|
||
|
ignore_files::IgnoreFile,
|
||
|
project::ProjectType,
|
||
|
};
|
||
|
|
||
|
pub mod globset {
|
||
|
pub use super::file;
|
||
|
pub use super::globset_filt as filt;
|
||
|
pub use super::Applies;
|
||
|
pub use super::Harness;
|
||
|
}
|
||
|
|
||
|
pub mod globset_ig {
|
||
|
pub use super::globset::*;
|
||
|
pub use super::globset_igfilt as filt;
|
||
|
}
|
||
|
|
||
|
pub mod tagged {
|
||
|
pub use super::file;
|
||
|
pub use super::tagged_filt as filt;
|
||
|
pub use super::Applies;
|
||
|
pub use super::FilterExt;
|
||
|
pub use super::Harness;
|
||
|
pub use super::{filter, not_filter};
|
||
|
}
|
||
|
|
||
|
pub mod tagged_ig {
|
||
|
pub use super::tagged::*;
|
||
|
pub use super::tagged_igfilt as filt;
|
||
|
}
|
||
|
|
||
|
pub trait Harness {
|
||
|
fn check_path(
|
||
|
&self,
|
||
|
path: PathBuf,
|
||
|
file_type: Option<FileType>,
|
||
|
) -> std::result::Result<bool, RuntimeError>;
|
||
|
|
||
|
fn path_pass(&self, path: &str, file_type: Option<FileType>, pass: bool) {
|
||
|
let origin = dunce::canonicalize(".").unwrap();
|
||
|
let full_path = if let Some(suf) = path.strip_prefix("/test/") {
|
||
|
origin.join(suf)
|
||
|
} else if Path::new(path).has_root() {
|
||
|
path.into()
|
||
|
} else {
|
||
|
origin.join(path)
|
||
|
};
|
||
|
|
||
|
tracing::info!(?path, ?file_type, ?pass, "check");
|
||
|
|
||
|
assert_eq!(
|
||
|
self.check_path(full_path, file_type).unwrap(),
|
||
|
pass,
|
||
|
"{} {:?} (expected {})",
|
||
|
match file_type {
|
||
|
Some(FileType::File) => "file",
|
||
|
Some(FileType::Dir) => "dir",
|
||
|
Some(FileType::Symlink) => "symlink",
|
||
|
Some(FileType::Other) => "other",
|
||
|
None => "path",
|
||
|
},
|
||
|
path,
|
||
|
if pass { "pass" } else { "fail" }
|
||
|
);
|
||
|
}
|
||
|
|
||
|
fn file_does_pass(&self, path: &str) {
|
||
|
self.path_pass(path, Some(FileType::File), true);
|
||
|
}
|
||
|
|
||
|
fn file_doesnt_pass(&self, path: &str) {
|
||
|
self.path_pass(path, Some(FileType::File), false);
|
||
|
}
|
||
|
|
||
|
fn dir_does_pass(&self, path: &str) {
|
||
|
self.path_pass(path, Some(FileType::Dir), true);
|
||
|
}
|
||
|
|
||
|
fn dir_doesnt_pass(&self, path: &str) {
|
||
|
self.path_pass(path, Some(FileType::Dir), false);
|
||
|
}
|
||
|
|
||
|
fn unk_does_pass(&self, path: &str) {
|
||
|
self.path_pass(path, None, true);
|
||
|
}
|
||
|
|
||
|
fn unk_doesnt_pass(&self, path: &str) {
|
||
|
self.path_pass(path, None, false);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl Harness for GlobsetFilterer {
|
||
|
fn check_path(
|
||
|
&self,
|
||
|
path: PathBuf,
|
||
|
file_type: Option<FileType>,
|
||
|
) -> std::result::Result<bool, RuntimeError> {
|
||
|
let event = Event {
|
||
|
tags: vec![Tag::Path { path, file_type }],
|
||
|
metadata: Default::default(),
|
||
|
};
|
||
|
|
||
|
self.check_event(&event)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl Harness for TaggedFilterer {
|
||
|
fn check_path(
|
||
|
&self,
|
||
|
path: PathBuf,
|
||
|
file_type: Option<FileType>,
|
||
|
) -> std::result::Result<bool, RuntimeError> {
|
||
|
let event = Event {
|
||
|
tags: vec![Tag::Path { path, file_type }],
|
||
|
metadata: Default::default(),
|
||
|
};
|
||
|
|
||
|
self.check_event(&event)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub fn globset_filt(filters: &[&str], ignores: &[&str], extensions: &[&str]) -> GlobsetFilterer {
|
||
|
let origin = dunce::canonicalize(".").unwrap();
|
||
|
GlobsetFilterer::new(
|
||
|
origin,
|
||
|
filters.iter().map(|s| (s.to_string(), None)),
|
||
|
ignores.iter().map(|s| (s.to_string(), None)),
|
||
|
extensions.iter().map(OsString::from),
|
||
|
)
|
||
|
.expect("making filterer")
|
||
|
}
|
||
|
|
||
|
pub async fn globset_igfilt(origin: &str, ignore_files: &[IgnoreFile]) -> GlobsetFilterer {
|
||
|
let mut ignores = Vec::new();
|
||
|
for file in ignore_files {
|
||
|
tracing::info!(?file, "loading ignore file");
|
||
|
ignores.extend(
|
||
|
GlobsetFilterer::list_from_ignore_file(file)
|
||
|
.await
|
||
|
.expect("adding ignorefile"),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
let origin = dunce::canonicalize(".").unwrap().join(origin);
|
||
|
GlobsetFilterer::new(origin, vec![], ignores, vec![]).expect("making filterer")
|
||
|
}
|
||
|
|
||
|
pub async fn tagged_filt(filters: &[Filter]) -> Arc<TaggedFilterer> {
|
||
|
let origin = dunce::canonicalize(".").unwrap();
|
||
|
let filterer = TaggedFilterer::new(origin.clone(), origin).expect("creating filterer");
|
||
|
filterer.add_filters(filters).await.expect("adding filters");
|
||
|
tracing_subscriber::fmt::try_init().ok();
|
||
|
filterer
|
||
|
}
|
||
|
|
||
|
pub async fn tagged_igfilt(origin: &str, ignore_files: &[IgnoreFile]) -> Arc<TaggedFilterer> {
|
||
|
let origin = dunce::canonicalize(".").unwrap().join(origin);
|
||
|
tracing_subscriber::fmt::try_init().ok();
|
||
|
let filterer = TaggedFilterer::new(origin.clone(), origin).expect("creating filterer");
|
||
|
for file in ignore_files {
|
||
|
tracing::info!(?file, "loading ignore file");
|
||
|
filterer
|
||
|
.add_ignore_file(file)
|
||
|
.await
|
||
|
.expect("adding ignore file");
|
||
|
}
|
||
|
filterer
|
||
|
}
|
||
|
|
||
|
pub fn file(name: &str) -> IgnoreFile {
|
||
|
let path = dunce::canonicalize(".")
|
||
|
.unwrap()
|
||
|
.join("tests")
|
||
|
.join("ignores")
|
||
|
.join(name);
|
||
|
IgnoreFile {
|
||
|
path,
|
||
|
applies_in: None,
|
||
|
applies_to: None,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub trait Applies {
|
||
|
fn applies_in(self, origin: &str) -> Self;
|
||
|
fn applies_to(self, project_type: ProjectType) -> Self;
|
||
|
}
|
||
|
|
||
|
impl Applies for IgnoreFile {
|
||
|
fn applies_in(mut self, origin: &str) -> Self {
|
||
|
let origin = dunce::canonicalize(".").unwrap().join(origin);
|
||
|
self.applies_in = Some(origin);
|
||
|
self
|
||
|
}
|
||
|
|
||
|
fn applies_to(mut self, project_type: ProjectType) -> Self {
|
||
|
self.applies_to = Some(project_type);
|
||
|
self
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub fn filter(pat: &str) -> Filter {
|
||
|
Filter {
|
||
|
in_path: None,
|
||
|
on: Matcher::Path,
|
||
|
op: Op::Glob,
|
||
|
pat: Pattern::Glob(pat.into()),
|
||
|
negate: false,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub fn not_filter(pat: &str) -> Filter {
|
||
|
Filter {
|
||
|
in_path: None,
|
||
|
on: Matcher::Path,
|
||
|
op: Op::NotGlob,
|
||
|
pat: Pattern::Glob(pat.into()),
|
||
|
negate: false,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub trait FilterExt {
|
||
|
fn in_path(self) -> Self
|
||
|
where
|
||
|
Self: Sized,
|
||
|
{
|
||
|
self.in_subpath("")
|
||
|
}
|
||
|
|
||
|
fn in_subpath(self, sub: impl AsRef<Path>) -> Self;
|
||
|
}
|
||
|
|
||
|
impl FilterExt for Filter {
|
||
|
fn in_subpath(mut self, sub: impl AsRef<Path>) -> Self {
|
||
|
let origin = dunce::canonicalize(".").unwrap();
|
||
|
self.in_path = Some(origin.join(sub));
|
||
|
self
|
||
|
}
|
||
|
}
|