1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78


/// matches % as a wildcard operator
pub struct Wildcard
{
  w: String,
}

impl Wildcard
{
  pub fn new(w: &str) -> Wildcard
  {
    Wildcard
    {
      w: w.to_owned()
    }
  }

  pub fn prefix(&self) -> &str
  {
    if let Some(o) = self.w.find('%')
    {
      &self.w[0 .. o]
    }
    else
    {
      &self.w[..]
    }
  }

  /// returns true if this search can only match
  /// a single key
  pub fn is_exact(&self) -> bool
  {
    self.w.find('%').is_none()
  }

  /// Returns the regex that matches my wildcard,
  /// or None if the prefix is all that's needed
  pub fn as_regex(&self) -> Option<regex::Regex>
  {
    let mut re = String::with_capacity(self.w.len()+4);
    re += "^";

    let mut haspct = false;
    let mut needre = false;

    for c in self.w.chars()
    {
      if haspct { needre = true; }

      match c
      {
        '%' =>
        {
          re += ".*";
          haspct=true;
        }
        a @ '.' | a@'(' | a@')'
          | a@'{' | a@'}' | a@'\\' | a@'|'
          | a@'^' | a@'$'
          | a@'[' | a@']'
          =>
        {
          re.push('\\'); re.push(a);
        },
        a => re.push(a),
      }
    }

    re += "$";

    if needre || !haspct
      { Some(regex::Regex::new(&re).unwrap()) }
    else
      { None }
  }
}