summaryrefslogtreecommitdiffstats
path: root/vendor/globset/src
diff options
context:
space:
mode:
Diffstat (limited to 'vendor/globset/src')
-rw-r--r--vendor/globset/src/glob.rs55
-rw-r--r--vendor/globset/src/lib.rs51
-rw-r--r--vendor/globset/src/pathutil.rs2
-rw-r--r--vendor/globset/src/serde_impl.rs102
4 files changed, 193 insertions, 17 deletions
diff --git a/vendor/globset/src/glob.rs b/vendor/globset/src/glob.rs
index 6e35aeec3..d19c70ed2 100644
--- a/vendor/globset/src/glob.rs
+++ b/vendor/globset/src/glob.rs
@@ -208,6 +208,9 @@ struct GlobOptions {
/// Whether or not to use `\` to escape special characters.
/// e.g., when enabled, `\*` will match a literal `*`.
backslash_escape: bool,
+ /// Whether or not an empty case in an alternate will be removed.
+ /// e.g., when enabled, `{,a}` will match "" and "a".
+ empty_alternates: bool,
}
impl GlobOptions {
@@ -216,6 +219,7 @@ impl GlobOptions {
case_insensitive: false,
literal_separator: false,
backslash_escape: !is_separator('\\'),
+ empty_alternates: false,
}
}
}
@@ -633,6 +637,16 @@ impl<'a> GlobBuilder<'a> {
self.opts.backslash_escape = yes;
self
}
+
+ /// Toggle whether an empty pattern in a list of alternates is accepted.
+ ///
+ /// For example, if this is set then the glob `foo{,.txt}` will match both `foo` and `foo.txt`.
+ ///
+ /// By default this is false.
+ pub fn empty_alternates(&mut self, yes: bool) -> &mut GlobBuilder<'a> {
+ self.opts.empty_alternates = yes;
+ self
+ }
}
impl Tokens {
@@ -714,7 +728,7 @@ impl Tokens {
for pat in patterns {
let mut altre = String::new();
self.tokens_to_regex(options, &pat, &mut altre);
- if !altre.is_empty() {
+ if !altre.is_empty() || options.empty_alternates {
parts.push(altre);
}
}
@@ -722,7 +736,7 @@ impl Tokens {
// It is possible to have an empty set in which case the
// resulting alternation '()' would be an error.
if !parts.is_empty() {
- re.push('(');
+ re.push_str("(?:");
re.push_str(&parts.join("|"));
re.push(')');
}
@@ -1020,6 +1034,7 @@ mod tests {
casei: Option<bool>,
litsep: Option<bool>,
bsesc: Option<bool>,
+ ealtre: Option<bool>,
}
macro_rules! syntax {
@@ -1059,6 +1074,9 @@ mod tests {
if let Some(bsesc) = $options.bsesc {
builder.backslash_escape(bsesc);
}
+ if let Some(ealtre) = $options.ealtre {
+ builder.empty_alternates(ealtre);
+ }
let pat = builder.build().unwrap();
assert_eq!(format!("(?-u){}", $re), pat.regex());
}
@@ -1082,6 +1100,9 @@ mod tests {
if let Some(bsesc) = $options.bsesc {
builder.backslash_escape(bsesc);
}
+ if let Some(ealtre) = $options.ealtre {
+ builder.empty_alternates(ealtre);
+ }
let pat = builder.build().unwrap();
let matcher = pat.compile_matcher();
let strategic = pat.compile_strategic_matcher();
@@ -1110,6 +1131,9 @@ mod tests {
if let Some(bsesc) = $options.bsesc {
builder.backslash_escape(bsesc);
}
+ if let Some(ealtre) = $options.ealtre {
+ builder.empty_alternates(ealtre);
+ }
let pat = builder.build().unwrap();
let matcher = pat.compile_matcher();
let strategic = pat.compile_strategic_matcher();
@@ -1195,13 +1219,23 @@ mod tests {
syntaxerr!(err_range2, "[z--]", ErrorKind::InvalidRange('z', '-'));
const CASEI: Options =
- Options { casei: Some(true), litsep: None, bsesc: None };
+ Options { casei: Some(true), litsep: None, bsesc: None, ealtre: None };
const SLASHLIT: Options =
- Options { casei: None, litsep: Some(true), bsesc: None };
- const NOBSESC: Options =
- Options { casei: None, litsep: None, bsesc: Some(false) };
+ Options { casei: None, litsep: Some(true), bsesc: None, ealtre: None };
+ const NOBSESC: Options = Options {
+ casei: None,
+ litsep: None,
+ bsesc: Some(false),
+ ealtre: None,
+ };
const BSESC: Options =
- Options { casei: None, litsep: None, bsesc: Some(true) };
+ Options { casei: None, litsep: None, bsesc: Some(true), ealtre: None };
+ const EALTRE: Options = Options {
+ casei: None,
+ litsep: None,
+ bsesc: Some(true),
+ ealtre: Some(true),
+ };
toregex!(re_casei, "a", "(?i)^a$", &CASEI);
@@ -1242,6 +1276,7 @@ mod tests {
toregex!(re32, "/a**", r"^/a.*.*$");
toregex!(re33, "/**a", r"^/.*.*a$");
toregex!(re34, "/a**b", r"^/a.*.*b$");
+ toregex!(re35, "{a,b}", r"^(?:b|a)$");
matches!(match1, "a", "a");
matches!(match2, "a*b", "a_b");
@@ -1326,6 +1361,9 @@ mod tests {
matches!(matchalt11, "{*.foo,*.bar,*.wat}", "test.foo");
matches!(matchalt12, "{*.foo,*.bar,*.wat}", "test.bar");
matches!(matchalt13, "{*.foo,*.bar,*.wat}", "test.wat");
+ matches!(matchalt14, "foo{,.txt}", "foo.txt");
+ nmatches!(matchalt15, "foo{,.txt}", "foo");
+ matches!(matchalt16, "foo{,.txt}", "foo", EALTRE);
matches!(matchslash1, "abc/def", "abc/def", SLASHLIT);
#[cfg(unix)]
@@ -1425,6 +1463,9 @@ mod tests {
if let Some(bsesc) = $options.bsesc {
builder.backslash_escape(bsesc);
}
+ if let Some(ealtre) = $options.ealtre {
+ builder.empty_alternates(ealtre);
+ }
let pat = builder.build().unwrap();
assert_eq!($expect, pat.$which());
}
diff --git a/vendor/globset/src/lib.rs b/vendor/globset/src/lib.rs
index c8072b2db..7a357489b 100644
--- a/vendor/globset/src/lib.rs
+++ b/vendor/globset/src/lib.rs
@@ -498,13 +498,23 @@ impl GlobSetBuilder {
/// Constructing candidates has a very small cost associated with it, so
/// callers may find it beneficial to amortize that cost when matching a single
/// path against multiple globs or sets of globs.
-#[derive(Clone, Debug)]
+#[derive(Clone)]
pub struct Candidate<'a> {
path: Cow<'a, [u8]>,
basename: Cow<'a, [u8]>,
ext: Cow<'a, [u8]>,
}
+impl<'a> std::fmt::Debug for Candidate<'a> {
+ fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
+ f.debug_struct("Candidate")
+ .field("path", &self.path.as_bstr())
+ .field("basename", &self.basename.as_bstr())
+ .field("ext", &self.ext.as_bstr())
+ .finish()
+ }
+}
+
impl<'a> Candidate<'a> {
/// Create a new candidate for matching from the given path.
pub fn new<P: AsRef<Path> + ?Sized>(path: &'a P) -> Candidate<'a> {
@@ -818,7 +828,7 @@ impl MultiStrategyBuilder {
fn prefix(self) -> PrefixStrategy {
PrefixStrategy {
- matcher: AhoCorasick::new_auto_configured(&self.literals),
+ matcher: AhoCorasick::new(&self.literals).unwrap(),
map: self.map,
longest: self.longest,
}
@@ -826,7 +836,7 @@ impl MultiStrategyBuilder {
fn suffix(self) -> SuffixStrategy {
SuffixStrategy {
- matcher: AhoCorasick::new_auto_configured(&self.literals),
+ matcher: AhoCorasick::new(&self.literals).unwrap(),
map: self.map,
longest: self.longest,
}
@@ -870,6 +880,29 @@ impl RequiredExtensionStrategyBuilder {
}
}
+/// Escape meta-characters within the given glob pattern.
+///
+/// The escaping works by surrounding meta-characters with brackets. For
+/// example, `*` becomes `[*]`.
+pub fn escape(s: &str) -> String {
+ let mut escaped = String::with_capacity(s.len());
+ for c in s.chars() {
+ match c {
+ // note that ! does not need escaping because it is only special
+ // inside brackets
+ '?' | '*' | '[' | ']' => {
+ escaped.push('[');
+ escaped.push(c);
+ escaped.push(']');
+ }
+ c => {
+ escaped.push(c);
+ }
+ }
+ }
+ escaped
+}
+
#[cfg(test)]
mod tests {
use super::{GlobSet, GlobSetBuilder};
@@ -909,4 +942,16 @@ mod tests {
assert!(!set.is_match(""));
assert!(!set.is_match("a"));
}
+
+ #[test]
+ fn escape() {
+ use super::escape;
+ assert_eq!("foo", escape("foo"));
+ assert_eq!("foo[*]", escape("foo*"));
+ assert_eq!("[[][]]", escape("[]"));
+ assert_eq!("[*][?]", escape("*?"));
+ assert_eq!("src/[*][*]/[*].rs", escape("src/**/*.rs"));
+ assert_eq!("bar[[]ab[]]baz", escape("bar[ab]baz"));
+ assert_eq!("bar[[]!![]]!baz", escape("bar[!!]!baz"));
+ }
}
diff --git a/vendor/globset/src/pathutil.rs b/vendor/globset/src/pathutil.rs
index 2bd34e1dd..522df3401 100644
--- a/vendor/globset/src/pathutil.rs
+++ b/vendor/globset/src/pathutil.rs
@@ -27,7 +27,7 @@ pub fn file_name<'a>(path: &Cow<'a, [u8]>) -> Option<Cow<'a, [u8]>> {
///
/// Note that this does NOT match the semantics of std::path::Path::extension.
/// Namely, the extension includes the `.` and matching is otherwise more
-/// liberal. Specifically, the extenion is:
+/// liberal. Specifically, the extension is:
///
/// * None, if the file name given is empty;
/// * None, if there is no embedded `.`;
diff --git a/vendor/globset/src/serde_impl.rs b/vendor/globset/src/serde_impl.rs
index 6affc5904..88b1574d7 100644
--- a/vendor/globset/src/serde_impl.rs
+++ b/vendor/globset/src/serde_impl.rs
@@ -1,7 +1,9 @@
-use serde::de::Error;
-use serde::{Deserialize, Deserializer, Serialize, Serializer};
+use serde::{
+ de::{Error, SeqAccess, Visitor},
+ {Deserialize, Deserializer, Serialize, Serializer},
+};
-use crate::Glob;
+use crate::{Glob, GlobSet, GlobSetBuilder};
impl Serialize for Glob {
fn serialize<S: Serializer>(
@@ -12,18 +14,98 @@ impl Serialize for Glob {
}
}
+struct GlobVisitor;
+
+impl<'de> Visitor<'de> for GlobVisitor {
+ type Value = Glob;
+
+ fn expecting(
+ &self,
+ formatter: &mut std::fmt::Formatter,
+ ) -> std::fmt::Result {
+ formatter.write_str("a glob pattern")
+ }
+
+ fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
+ where
+ E: Error,
+ {
+ Glob::new(v).map_err(serde::de::Error::custom)
+ }
+}
+
impl<'de> Deserialize<'de> for Glob {
fn deserialize<D: Deserializer<'de>>(
deserializer: D,
) -> Result<Self, D::Error> {
- let glob = <&str as Deserialize>::deserialize(deserializer)?;
- Glob::new(glob).map_err(D::Error::custom)
+ deserializer.deserialize_str(GlobVisitor)
+ }
+}
+
+struct GlobSetVisitor;
+
+impl<'de> Visitor<'de> for GlobSetVisitor {
+ type Value = GlobSet;
+
+ fn expecting(
+ &self,
+ formatter: &mut std::fmt::Formatter,
+ ) -> std::fmt::Result {
+ formatter.write_str("an array of glob patterns")
+ }
+
+ fn visit_seq<A>(self, mut seq: A) -> Result<Self::Value, A::Error>
+ where
+ A: SeqAccess<'de>,
+ {
+ let mut builder = GlobSetBuilder::new();
+ while let Some(glob) = seq.next_element()? {
+ builder.add(glob);
+ }
+ builder.build().map_err(serde::de::Error::custom)
+ }
+}
+
+impl<'de> Deserialize<'de> for GlobSet {
+ fn deserialize<D: Deserializer<'de>>(
+ deserializer: D,
+ ) -> Result<Self, D::Error> {
+ deserializer.deserialize_seq(GlobSetVisitor)
}
}
#[cfg(test)]
mod tests {
- use Glob;
+ use std::collections::HashMap;
+
+ use crate::{Glob, GlobSet};
+
+ #[test]
+ fn glob_deserialize_borrowed() {
+ let string = r#"{"markdown": "*.md"}"#;
+
+ let map: HashMap<String, Glob> =
+ serde_json::from_str(&string).unwrap();
+ assert_eq!(map["markdown"], Glob::new("*.md").unwrap());
+ }
+
+ #[test]
+ fn glob_deserialize_owned() {
+ let string = r#"{"markdown": "*.md"}"#;
+
+ let v: serde_json::Value = serde_json::from_str(&string).unwrap();
+ let map: HashMap<String, Glob> = serde_json::from_value(v).unwrap();
+ assert_eq!(map["markdown"], Glob::new("*.md").unwrap());
+ }
+
+ #[test]
+ fn glob_deserialize_error() {
+ let string = r#"{"error": "["}"#;
+
+ let map = serde_json::from_str::<HashMap<String, Glob>>(&string);
+
+ assert!(map.is_err());
+ }
#[test]
fn glob_json_works() {
@@ -35,4 +117,12 @@ mod tests {
let de: Glob = serde_json::from_str(&ser).unwrap();
assert_eq!(test_glob, de);
}
+
+ #[test]
+ fn glob_set_deserialize() {
+ let j = r#" ["src/**/*.rs", "README.md"] "#;
+ let set: GlobSet = serde_json::from_str(j).unwrap();
+ assert!(set.is_match("src/lib.rs"));
+ assert!(!set.is_match("Cargo.lock"));
+ }
}