GCC Code Coverage Report


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