| Directory: | cvmfs/ |
|---|---|
| File: | cvmfs/pathspec/pathspec_pattern.cc |
| Date: | 2026-01-11 02:35:46 |
| Exec | Total | Coverage | |
|---|---|---|---|
| Lines: | 136 | 138 | 98.6% |
| Branches: | 134 | 182 | 73.6% |
| Line | Branch | Exec | Source |
|---|---|---|---|
| 1 | /** | ||
| 2 | * This file is part of the CernVM File System. | ||
| 3 | */ | ||
| 4 | |||
| 5 | |||
| 6 | #include "pathspec_pattern.h" | ||
| 7 | |||
| 8 | #include <cassert> | ||
| 9 | |||
| 10 | #include "pathspec/pathspec.h" | ||
| 11 | |||
| 12 | 14489 | PathspecElementPattern::PathspecElementPattern( | |
| 13 | const std::string::const_iterator begin, | ||
| 14 | 14489 | const std::string::const_iterator &end) | |
| 15 | 14489 | : valid_(true) { | |
| 16 |
1/2✓ Branch 1 taken 14489 times.
✗ Branch 2 not taken.
|
14489 | Parse(begin, end); |
| 17 | 14489 | } | |
| 18 | |||
| 19 | 38110 | PathspecElementPattern::PathspecElementPattern( | |
| 20 | 38110 | const PathspecElementPattern &other) | |
| 21 | 38110 | : valid_(other.valid_) { | |
| 22 |
1/2✓ Branch 2 taken 38110 times.
✗ Branch 3 not taken.
|
38110 | subpatterns_.reserve(other.subpatterns_.size()); |
| 23 | 38110 | SubPatterns::const_iterator i = other.subpatterns_.begin(); | |
| 24 | 38110 | const SubPatterns::const_iterator iend = other.subpatterns_.end(); | |
| 25 |
2/2✓ Branch 1 taken 43523 times.
✓ Branch 2 taken 38110 times.
|
81633 | for (; i != iend; ++i) { |
| 26 |
2/4✓ Branch 2 taken 43523 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 43523 times.
✗ Branch 6 not taken.
|
43523 | subpatterns_.push_back((*i)->Clone()); |
| 27 | } | ||
| 28 | 38110 | } | |
| 29 | |||
| 30 | 322 | PathspecElementPattern &PathspecElementPattern::operator=( | |
| 31 | const PathspecElementPattern &other) { | ||
| 32 |
1/2✓ Branch 0 taken 322 times.
✗ Branch 1 not taken.
|
322 | if (this != &other) { |
| 33 | 322 | valid_ = other.valid_; | |
| 34 | 322 | subpatterns_.clear(); | |
| 35 |
1/2✓ Branch 2 taken 322 times.
✗ Branch 3 not taken.
|
322 | subpatterns_.reserve(other.subpatterns_.size()); |
| 36 | 322 | SubPatterns::const_iterator i = other.subpatterns_.begin(); | |
| 37 | 322 | const SubPatterns::const_iterator iend = other.subpatterns_.end(); | |
| 38 |
2/2✓ Branch 1 taken 506 times.
✓ Branch 2 taken 322 times.
|
828 | for (; i != iend; ++i) { |
| 39 |
2/4✓ Branch 2 taken 506 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 506 times.
✗ Branch 6 not taken.
|
506 | subpatterns_.push_back((*i)->Clone()); |
| 40 | } | ||
| 41 | } | ||
| 42 | |||
| 43 | 322 | return *this; | |
| 44 | } | ||
| 45 | |||
| 46 | |||
| 47 | 52593 | PathspecElementPattern::~PathspecElementPattern() { | |
| 48 | 52593 | SubPatterns::const_iterator i = subpatterns_.begin(); | |
| 49 | 52593 | const SubPatterns::const_iterator iend = subpatterns_.end(); | |
| 50 |
2/2✓ Branch 2 taken 61353 times.
✓ Branch 3 taken 52593 times.
|
113946 | for (; i != iend; ++i) { |
| 51 |
1/2✓ Branch 1 taken 61353 times.
✗ Branch 2 not taken.
|
61353 | delete *i; |
| 52 | } | ||
| 53 | 52593 | subpatterns_.clear(); | |
| 54 | 52593 | } | |
| 55 | |||
| 56 | |||
| 57 | 14489 | void PathspecElementPattern::Parse(const std::string::const_iterator &begin, | |
| 58 | const std::string::const_iterator &end) { | ||
| 59 | 14489 | std::string::const_iterator i = begin; | |
| 60 |
2/2✓ Branch 1 taken 17690 times.
✓ Branch 2 taken 14489 times.
|
32179 | while (i != end) { |
| 61 |
3/4✓ Branch 2 taken 4138 times.
✓ Branch 3 taken 13552 times.
✓ Branch 5 taken 4138 times.
✗ Branch 6 not taken.
|
17690 | SubPattern *next = (Pathspec::IsSpecialChar(*i)) ? ParseSpecialChar(end, &i) |
| 62 |
1/2✓ Branch 1 taken 13552 times.
✗ Branch 2 not taken.
|
13552 | : ParsePlaintext(end, &i); |
| 63 |
3/4✓ Branch 1 taken 17690 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 38 times.
✓ Branch 4 taken 17652 times.
|
17690 | if (next->IsEmpty()) { |
| 64 | 38 | valid_ = false; | |
| 65 |
1/2✓ Branch 0 taken 38 times.
✗ Branch 1 not taken.
|
38 | delete next; |
| 66 | } else { | ||
| 67 |
1/2✓ Branch 1 taken 17652 times.
✗ Branch 2 not taken.
|
17652 | subpatterns_.push_back(next); |
| 68 | } | ||
| 69 | } | ||
| 70 | 14489 | } | |
| 71 | |||
| 72 | |||
| 73 | 13552 | PathspecElementPattern::SubPattern *PathspecElementPattern::ParsePlaintext( | |
| 74 | const std::string::const_iterator &end, std::string::const_iterator *i) { | ||
| 75 | 13552 | PlaintextSubPattern *pattern = new PlaintextSubPattern(); | |
| 76 | 13552 | bool next_is_escaped = false; | |
| 77 | |||
| 78 |
2/2✓ Branch 1 taken 58918 times.
✓ Branch 2 taken 12318 times.
|
71236 | while (*i < end) { |
| 79 |
6/6✓ Branch 2 taken 1842 times.
✓ Branch 3 taken 57076 times.
✓ Branch 4 taken 1234 times.
✓ Branch 5 taken 608 times.
✓ Branch 6 taken 1234 times.
✓ Branch 7 taken 57684 times.
|
58918 | if (Pathspec::IsSpecialChar(**i) && !next_is_escaped) { |
| 80 | 1234 | break; | |
| 81 | } | ||
| 82 | |||
| 83 |
6/6✓ Branch 1 taken 1067 times.
✓ Branch 2 taken 56617 times.
✓ Branch 3 taken 880 times.
✓ Branch 4 taken 187 times.
✓ Branch 5 taken 880 times.
✓ Branch 6 taken 56804 times.
|
57684 | if (**i == Pathspec::kEscaper && !next_is_escaped) { |
| 84 | 880 | next_is_escaped = true; | |
| 85 |
2/2✓ Branch 0 taken 1144 times.
✓ Branch 1 taken 55660 times.
|
56804 | } else if (next_is_escaped) { |
| 86 |
6/6✓ Branch 2 taken 536 times.
✓ Branch 3 taken 608 times.
✓ Branch 5 taken 187 times.
✓ Branch 6 taken 349 times.
✓ Branch 7 taken 795 times.
✓ Branch 8 taken 349 times.
|
1144 | if (Pathspec::IsSpecialChar(**i) || **i == Pathspec::kEscaper) { |
| 87 | 795 | pattern->AddChar(**i); | |
| 88 | 795 | next_is_escaped = false; | |
| 89 | } else { | ||
| 90 | 349 | valid_ = false; | |
| 91 | } | ||
| 92 | } else { | ||
| 93 |
1/2✗ Branch 2 not taken.
✓ Branch 3 taken 55660 times.
|
55660 | assert(!Pathspec::IsSpecialChar(**i)); |
| 94 | 55660 | pattern->AddChar(**i); | |
| 95 | } | ||
| 96 | |||
| 97 | 57684 | ++(*i); | |
| 98 | } | ||
| 99 | |||
| 100 | 13552 | return pattern; | |
| 101 | } | ||
| 102 | |||
| 103 | 4138 | PathspecElementPattern::SubPattern *PathspecElementPattern::ParseSpecialChar( | |
| 104 | const std::string::const_iterator &end, std::string::const_iterator *i) { | ||
| 105 |
1/2✗ Branch 2 not taken.
✓ Branch 3 taken 4138 times.
|
4138 | assert(Pathspec::IsSpecialChar(**i)); |
| 106 | 4138 | const char chr = **i; | |
| 107 | 4138 | ++(*i); | |
| 108 | |||
| 109 |
2/3✓ Branch 0 taken 2322 times.
✓ Branch 1 taken 1816 times.
✗ Branch 2 not taken.
|
4138 | switch (chr) { |
| 110 | 2322 | case Pathspec::kWildcard: | |
| 111 | 2322 | return new WildcardSubPattern(); | |
| 112 | 1816 | case Pathspec::kPlaceholder: | |
| 113 | 1816 | return new PlaceholderSubPattern(); | |
| 114 | ✗ | default: | |
| 115 | ✗ | assert(false && "unrecognized special character"); | |
| 116 | } | ||
| 117 | } | ||
| 118 | |||
| 119 | 8658 | std::string PathspecElementPattern::GenerateRegularExpression( | |
| 120 | const bool is_relaxed) const { | ||
| 121 | 8658 | std::string result; | |
| 122 | 8658 | SubPatterns::const_iterator i = subpatterns_.begin(); | |
| 123 | 8658 | const SubPatterns::const_iterator iend = subpatterns_.end(); | |
| 124 |
2/2✓ Branch 1 taken 11226 times.
✓ Branch 2 taken 8658 times.
|
19884 | for (; i != iend; ++i) { |
| 125 |
2/4✓ Branch 2 taken 11226 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 11226 times.
✗ Branch 6 not taken.
|
11226 | result += (*i)->GenerateRegularExpression(is_relaxed); |
| 126 | } | ||
| 127 | 17316 | return result; | |
| 128 | } | ||
| 129 | |||
| 130 | 1590 | std::string PathspecElementPattern::GenerateGlobString() const { | |
| 131 | 1590 | std::string result; | |
| 132 | 1590 | SubPatterns::const_iterator i = subpatterns_.begin(); | |
| 133 | 1590 | const SubPatterns::const_iterator iend = subpatterns_.end(); | |
| 134 |
2/2✓ Branch 1 taken 1869 times.
✓ Branch 2 taken 1590 times.
|
3459 | for (; i != iend; ++i) { |
| 135 |
2/4✓ Branch 2 taken 1869 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 1869 times.
✗ Branch 6 not taken.
|
1869 | result += (*i)->GenerateGlobString(); |
| 136 | } | ||
| 137 | 3180 | return result; | |
| 138 | } | ||
| 139 | |||
| 140 | 4368 | bool PathspecElementPattern::operator==( | |
| 141 | const PathspecElementPattern &other) const { | ||
| 142 | 4368 | if (subpatterns_.size() != other.subpatterns_.size() | |
| 143 |
5/6✓ Branch 0 taken 3934 times.
✓ Branch 1 taken 434 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 3934 times.
✓ Branch 6 taken 434 times.
✓ Branch 7 taken 3934 times.
|
4368 | || IsValid() != other.IsValid()) { |
| 144 | 434 | return false; | |
| 145 | } | ||
| 146 | |||
| 147 | 3934 | SubPatterns::const_iterator i = subpatterns_.begin(); | |
| 148 | 3934 | const SubPatterns::const_iterator iend = subpatterns_.end(); | |
| 149 | 3934 | SubPatterns::const_iterator j = other.subpatterns_.begin(); | |
| 150 | 3934 | const SubPatterns::const_iterator jend = other.subpatterns_.end(); | |
| 151 | |||
| 152 |
5/6✓ Branch 3 taken 4592 times.
✓ Branch 4 taken 3036 times.
✓ Branch 6 taken 4592 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 4592 times.
✓ Branch 9 taken 3036 times.
|
7628 | for (; i != iend && j != jend; ++i, ++j) { |
| 153 |
3/4✓ Branch 3 taken 4592 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 898 times.
✓ Branch 6 taken 3694 times.
|
4592 | if (!(*i)->Compare(*j)) { |
| 154 | 898 | return false; | |
| 155 | } | ||
| 156 | } | ||
| 157 | |||
| 158 | 3036 | return true; | |
| 159 | } | ||
| 160 | |||
| 161 | |||
| 162 | //------------------------------------------------------------------------------ | ||
| 163 | |||
| 164 | |||
| 165 | 56455 | void PathspecElementPattern::PlaintextSubPattern::AddChar(const char chr) { | |
| 166 | 56455 | chars_.push_back(chr); | |
| 167 | 56455 | } | |
| 168 | |||
| 169 | std::string | ||
| 170 | 8555 | PathspecElementPattern::PlaintextSubPattern::GenerateRegularExpression( | |
| 171 | const bool is_relaxed) const { | ||
| 172 | // Note: strict and relaxed regex are the same! | ||
| 173 | 8555 | std::string::const_iterator i = chars_.begin(); | |
| 174 | 8555 | const std::string::const_iterator iend = chars_.end(); | |
| 175 | 8555 | std::string regex; | |
| 176 |
2/2✓ Branch 2 taken 37169 times.
✓ Branch 3 taken 8555 times.
|
45724 | for (; i != iend; ++i) { |
| 177 |
2/2✓ Branch 2 taken 2518 times.
✓ Branch 3 taken 34651 times.
|
37169 | if (IsSpecialRegexCharacter(*i)) { |
| 178 |
1/2✓ Branch 1 taken 2518 times.
✗ Branch 2 not taken.
|
2518 | regex += "\\"; |
| 179 | } | ||
| 180 |
1/2✓ Branch 2 taken 37169 times.
✗ Branch 3 not taken.
|
37169 | regex += *i; |
| 181 | } | ||
| 182 | 17110 | return regex; | |
| 183 | } | ||
| 184 | |||
| 185 | |||
| 186 | 1683 | std::string PathspecElementPattern::PlaintextSubPattern::GenerateGlobString() | |
| 187 | const { | ||
| 188 | 1683 | std::string::const_iterator i = chars_.begin(); | |
| 189 | 1683 | const std::string::const_iterator iend = chars_.end(); | |
| 190 | 1683 | std::string glob_string; | |
| 191 |
2/2✓ Branch 2 taken 8608 times.
✓ Branch 3 taken 1683 times.
|
10291 | for (; i != iend; ++i) { |
| 192 |
2/2✓ Branch 2 taken 188 times.
✓ Branch 3 taken 8420 times.
|
8608 | if (Pathspec::IsSpecialChar(*i)) { |
| 193 |
1/2✓ Branch 1 taken 188 times.
✗ Branch 2 not taken.
|
188 | glob_string += "\\"; |
| 194 | } | ||
| 195 |
1/2✓ Branch 2 taken 8608 times.
✗ Branch 3 not taken.
|
8608 | glob_string += *i; |
| 196 | } | ||
| 197 | 3366 | return glob_string; | |
| 198 | } | ||
| 199 | |||
| 200 | 37169 | bool PathspecElementPattern::PlaintextSubPattern::IsSpecialRegexCharacter( | |
| 201 | const char chr) const { | ||
| 202 |
8/8✓ Branch 0 taken 35541 times.
✓ Branch 1 taken 93 times.
✓ Branch 2 taken 35401 times.
✓ Branch 3 taken 140 times.
✓ Branch 4 taken 35215 times.
✓ Branch 5 taken 186 times.
✓ Branch 6 taken 35121 times.
✓ Branch 7 taken 94 times.
|
35634 | return (chr == '.' || chr == '\\' || chr == '*' || chr == '?' || chr == '[' |
| 203 |
10/10✓ Branch 0 taken 35027 times.
✓ Branch 1 taken 94 times.
✓ Branch 2 taken 34980 times.
✓ Branch 3 taken 47 times.
✓ Branch 4 taken 34933 times.
✓ Branch 5 taken 47 times.
✓ Branch 6 taken 34886 times.
✓ Branch 7 taken 47 times.
✓ Branch 8 taken 34839 times.
✓ Branch 9 taken 47 times.
|
35121 | || chr == ']' || chr == '(' || chr == ')' || chr == '{' || chr == '}' |
| 204 |
8/8✓ Branch 0 taken 35634 times.
✓ Branch 1 taken 1535 times.
✓ Branch 2 taken 34792 times.
✓ Branch 3 taken 47 times.
✓ Branch 4 taken 34745 times.
✓ Branch 5 taken 47 times.
✓ Branch 6 taken 94 times.
✓ Branch 7 taken 34651 times.
|
72803 | || chr == '^' || chr == '$' || chr == '+'); |
| 205 | } | ||
| 206 | |||
| 207 | 3415 | bool PathspecElementPattern::PlaintextSubPattern::Compare( | |
| 208 | const SubPattern *other) const { | ||
| 209 |
2/2✓ Branch 1 taken 47 times.
✓ Branch 2 taken 3368 times.
|
3415 | if (!other->IsPlaintext()) { |
| 210 | 47 | return false; | |
| 211 | } | ||
| 212 | |||
| 213 | const PlaintextSubPattern | ||
| 214 |
1/2✓ Branch 0 taken 3368 times.
✗ Branch 1 not taken.
|
3368 | *pt_other = dynamic_cast<const PlaintextSubPattern *>(other); |
| 215 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 3368 times.
|
3368 | assert(pt_other != NULL); |
| 216 | 3368 | return chars_ == pt_other->chars_; | |
| 217 | } | ||
| 218 | |||
| 219 | |||
| 220 | std::string | ||
| 221 | 1696 | PathspecElementPattern::WildcardSubPattern::GenerateRegularExpression( | |
| 222 | const bool is_relaxed) const { | ||
| 223 | return (is_relaxed) ? std::string(".*") | ||
| 224 |
14/27✓ Branch 0 taken 540 times.
✓ Branch 1 taken 1156 times.
✓ Branch 4 taken 540 times.
✗ Branch 5 not taken.
✓ Branch 8 taken 1156 times.
✗ Branch 9 not taken.
✓ Branch 11 taken 1156 times.
✗ Branch 12 not taken.
✓ Branch 14 taken 1156 times.
✗ Branch 15 not taken.
✓ Branch 16 taken 1156 times.
✓ Branch 17 taken 540 times.
✗ Branch 18 not taken.
✓ Branch 19 taken 1156 times.
✓ Branch 20 taken 540 times.
✓ Branch 22 taken 1156 times.
✓ Branch 23 taken 540 times.
✓ Branch 25 taken 540 times.
✓ Branch 26 taken 1156 times.
✗ Branch 28 not taken.
✗ Branch 29 not taken.
✗ Branch 31 not taken.
✗ Branch 32 not taken.
✗ Branch 34 not taken.
✗ Branch 35 not taken.
✗ Branch 37 not taken.
✗ Branch 38 not taken.
|
3392 | : std::string("[^") + Pathspec::kSeparator + "]*"; |
| 225 | } | ||
| 226 | |||
| 227 | |||
| 228 | 93 | std::string PathspecElementPattern::WildcardSubPattern::GenerateGlobString() | |
| 229 | const { | ||
| 230 |
1/2✓ Branch 2 taken 93 times.
✗ Branch 3 not taken.
|
93 | return "*"; |
| 231 | } | ||
| 232 | |||
| 233 | |||
| 234 | 613 | bool PathspecElementPattern::WildcardSubPattern::Compare( | |
| 235 | const SubPattern *other) const { | ||
| 236 | 613 | return other->IsWildcard(); | |
| 237 | } | ||
| 238 | |||
| 239 | |||
| 240 | std::string | ||
| 241 | 975 | PathspecElementPattern::PlaceholderSubPattern::GenerateRegularExpression( | |
| 242 | const bool is_relaxed) const { | ||
| 243 | // Note: strict and relaxed regex are the same! | ||
| 244 |
3/6✓ Branch 2 taken 975 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 975 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 975 times.
✗ Branch 9 not taken.
|
1950 | return std::string("[^") + Pathspec::kSeparator + "]"; |
| 245 | } | ||
| 246 | |||
| 247 | 93 | std::string PathspecElementPattern::PlaceholderSubPattern::GenerateGlobString() | |
| 248 | const { | ||
| 249 |
1/2✓ Branch 2 taken 93 times.
✗ Branch 3 not taken.
|
93 | return "?"; |
| 250 | } | ||
| 251 | |||
| 252 | 564 | bool PathspecElementPattern::PlaceholderSubPattern::Compare( | |
| 253 | const SubPattern *other) const { | ||
| 254 | 564 | return other->IsPlaceholder(); | |
| 255 | } | ||
| 256 |