GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/path_filters/dirtab.cc
Date: 2025-09-14 02:35:40
Exec Total Coverage
Lines: 87 95 91.6%
Branches: 68 92 73.9%

Line Branch Exec Source
1 /**
2 * This file is part of the CernVM File System.
3 */
4
5 #include "dirtab.h"
6
7 #include <cassert>
8 #include <cerrno>
9 #include <cstdio>
10 #include <cstdlib>
11
12 #include "util/logging.h"
13 #include "util/posix.h"
14 #include "util/string.h"
15
16 namespace catalog {
17
18 1042 Dirtab::Dirtab() : valid_(true) { }
19
20
21 37 bool Dirtab::Open(const std::string &dirtab_path) {
22
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 37 times.
37 if (!FileExists(dirtab_path)) {
23 LogCvmfs(kLogCatalog, kLogStderr, "Cannot find dirtab at '%s'",
24 dirtab_path.c_str());
25 valid_ = false;
26 return valid_;
27 }
28
29 37 FILE *dirtab_file = fopen(dirtab_path.c_str(), "r");
30
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 37 times.
37 if (dirtab_file == NULL) {
31 LogCvmfs(kLogCatalog, kLogStderr,
32 "Cannot open dirtab for reading at '%s' "
33 "(errno: %d)",
34 dirtab_path.c_str(), errno);
35 valid_ = false;
36 return valid_;
37 }
38
39 37 valid_ = Parse(dirtab_file);
40 37 fclose(dirtab_file);
41 37 return valid_;
42 }
43
44 894 bool Dirtab::Parse(const std::string &dirtab) {
45 894 valid_ = true;
46 894 off_t line_offset = 0;
47
2/2
✓ Branch 1 taken 4160 times.
✓ Branch 2 taken 894 times.
5054 while (line_offset < static_cast<off_t>(dirtab.size())) {
48 8320 const std::string line = GetLineMem(dirtab.c_str() + line_offset,
49
1/2
✓ Branch 3 taken 4160 times.
✗ Branch 4 not taken.
4160 dirtab.size() - line_offset);
50 4160 line_offset += line.size() + 1; // +1 == skipped \n
51
3/4
✓ Branch 1 taken 4160 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 96 times.
✓ Branch 4 taken 4064 times.
4160 if (!ParseLine(line)) {
52 96 valid_ = false;
53 }
54 4160 }
55
4/4
✓ Branch 0 taken 798 times.
✓ Branch 1 taken 96 times.
✓ Branch 3 taken 750 times.
✓ Branch 4 taken 48 times.
894 valid_ = valid_ && CheckRuleValidity();
56 894 return valid_;
57 }
58
59
60 74 bool Dirtab::Parse(FILE *dirtab_file) {
61 74 valid_ = true;
62 74 std::string line;
63
3/4
✓ Branch 1 taken 148 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 74 times.
✓ Branch 4 taken 74 times.
148 while (GetLineFile(dirtab_file, &line)) {
64
2/4
✓ Branch 1 taken 74 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 74 times.
74 if (!ParseLine(line)) {
65 valid_ = false;
66 }
67 }
68
3/6
✓ Branch 0 taken 74 times.
✗ Branch 1 not taken.
✓ Branch 3 taken 74 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 74 times.
✗ Branch 6 not taken.
74 valid_ = valid_ && CheckRuleValidity();
69 74 return valid_;
70 74 }
71
72
73 4234 bool Dirtab::ParseLine(const std::string &line) {
74 // line parsing is done using std::string iterators. Each parsing method ex-
75 // pects an iterator and the end iterator. While parsing itr is constantly
76 // incremented to walk through the given .cvmfsdirtab line.
77 4234 std::string::const_iterator itr = line.begin();
78 4234 const std::string::const_iterator iend = line.end();
79 4234 bool negation = false;
80
81 // parse preamble
82 4234 SkipWhitespace(iend, &itr);
83
2/2
✓ Branch 1 taken 1260 times.
✓ Branch 2 taken 2974 times.
4234 if (*itr == Dirtab::kCommentMarker) {
84 1260 return true;
85
2/2
✓ Branch 1 taken 1234 times.
✓ Branch 2 taken 1740 times.
2974 } else if (*itr == Dirtab::kNegationMarker) {
86 1234 negation = true;
87 1234 ++itr;
88 1234 SkipWhitespace(iend, &itr);
89 }
90
91 // extract and parse pathspec
92
2/4
✓ Branch 2 taken 2974 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 2974 times.
✗ Branch 6 not taken.
5948 const std::string pathspec_str(Trim(std::string(itr, iend)));
93
1/2
✓ Branch 1 taken 2974 times.
✗ Branch 2 not taken.
2974 return this->ParsePathspec(pathspec_str, negation);
94 2974 }
95
96 3344 bool Dirtab::ParsePathspec(const std::string &pathspec_str, bool negation) {
97
2/2
✓ Branch 1 taken 480 times.
✓ Branch 2 taken 2864 times.
3344 if (pathspec_str.empty()) {
98 480 return true;
99 }
100
1/2
✓ Branch 1 taken 2864 times.
✗ Branch 2 not taken.
2864 const Pathspec pathspec(pathspec_str);
101
102 // all generated Pathspecs need to be valid and positive rules must be
103 // absolute. Otherwise the .cvmfsdirtab is not valid.
104
8/8
✓ Branch 1 taken 2816 times.
✓ Branch 2 taken 48 times.
✓ Branch 3 taken 1630 times.
✓ Branch 4 taken 1186 times.
✓ Branch 6 taken 48 times.
✓ Branch 7 taken 1582 times.
✓ Branch 8 taken 96 times.
✓ Branch 9 taken 2768 times.
2864 if (!pathspec.IsValid() || (!negation && !pathspec.IsAbsolute())) {
105 96 return false;
106 }
107
108 // create a new dirtab rule
109
1/2
✓ Branch 1 taken 2768 times.
✗ Branch 2 not taken.
2768 const Rule rule(pathspec, negation);
110
1/2
✓ Branch 1 taken 2768 times.
✗ Branch 2 not taken.
2768 AddRule(rule);
111 2768 return true;
112 2864 }
113
114
115 2768 void Dirtab::AddRule(const Rule &rule) {
116
2/2
✓ Branch 0 taken 1186 times.
✓ Branch 1 taken 1582 times.
2768 if (rule.is_negation) {
117 1186 negative_rules_.push_back(rule);
118 } else {
119 1582 positive_rules_.push_back(rule);
120 }
121 2768 }
122
123
124 872 bool Dirtab::CheckRuleValidity() const {
125 // check if there are contradicting positive and negative rules
126 872 Rules::const_iterator p = positive_rules_.begin();
127 872 const Rules::const_iterator pend = positive_rules_.end();
128
2/2
✓ Branch 2 taken 1534 times.
✓ Branch 3 taken 824 times.
2358 for (; p != pend; ++p) {
129
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 1534 times.
1534 assert(!p->is_negation);
130 1534 Rules::const_iterator n = negative_rules_.begin();
131 1534 const Rules::const_iterator nend = negative_rules_.end();
132
2/2
✓ Branch 2 taken 2321 times.
✓ Branch 3 taken 1486 times.
3807 for (; n != nend; ++n) {
133
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 2321 times.
2321 assert(n->is_negation);
134
3/4
✓ Branch 3 taken 2321 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 48 times.
✓ Branch 6 taken 2273 times.
2321 if (p->pathspec == n->pathspec) {
135 48 return false;
136 }
137 }
138 }
139
140 824 return true;
141 }
142
143
144 3445 bool Dirtab::IsMatching(const std::string &path) const {
145 // check if path has a positive match
146 3445 bool has_positive_match = false;
147 3445 Rules::const_iterator p = positive_rules_.begin();
148 3445 const Rules::const_iterator pend = positive_rules_.end();
149
2/2
✓ Branch 2 taken 6966 times.
✓ Branch 3 taken 1601 times.
8567 for (; p != pend; ++p) {
150
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 6966 times.
6966 assert(!p->is_negation);
151
3/4
✓ Branch 2 taken 6966 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 1844 times.
✓ Branch 5 taken 5122 times.
6966 if (p->pathspec.IsMatching(path)) {
152 1844 has_positive_match = true;
153 1844 break;
154 }
155 }
156
157
5/6
✓ Branch 0 taken 1844 times.
✓ Branch 1 taken 1601 times.
✓ Branch 3 taken 1844 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 1508 times.
✓ Branch 6 taken 336 times.
3445 return has_positive_match && !IsOpposing(path);
158 }
159
160
161 6455 bool Dirtab::IsOpposing(const std::string &path) const {
162 6455 Rules::const_iterator n = negative_rules_.begin();
163 6455 const Rules::const_iterator nend = negative_rules_.end();
164
2/2
✓ Branch 2 taken 11055 times.
✓ Branch 3 taken 5177 times.
16232 for (; n != nend; ++n) {
165
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 11055 times.
11055 assert(n->is_negation);
166
3/4
✓ Branch 2 taken 11055 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 1278 times.
✓ Branch 5 taken 9777 times.
11055 if (n->pathspec.IsMatchingRelaxed(path)) {
167 1278 return true;
168 }
169 }
170
171 5177 return false;
172 }
173
174 } // namespace catalog
175