GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/sanitizer.cc
Date: 2025-09-14 02:35:40
Exec Total Coverage
Lines: 69 71 97.2%
Branches: 40 49 81.6%

Line Branch Exec Source
1 /**
2 * This file is part of the CernVM File System.
3 *
4 * Provides input data sanitizer in the form of whitelist of character ranges.
5 */
6
7
8 #include "sanitizer.h"
9
10 #include <cassert>
11
12 using namespace std; // NOLINT
13
14 #ifdef CVMFS_NAMESPACE_GUARD
15 namespace CVMFS_NAMESPACE_GUARD {
16 #endif
17
18 namespace sanitizer {
19
20 47428 CharRange::CharRange(const char range_begin, const char range_end) {
21 47428 range_begin_ = range_begin;
22 47428 range_end_ = range_end;
23 47428 }
24
25
26 41526 CharRange::CharRange(const char single_char) {
27 41526 range_begin_ = range_end_ = single_char;
28 41526 }
29
30
31 575501 bool CharRange::InRange(const char c) const {
32
4/4
✓ Branch 0 taken 471753 times.
✓ Branch 1 taken 103748 times.
✓ Branch 2 taken 307906 times.
✓ Branch 3 taken 163847 times.
575501 return (c >= range_begin_) && (c <= range_end_);
33 }
34
35
36 //------------------------------------------------------------------------------
37
38
39 41384 InputSanitizer::InputSanitizer(const string &whitelist) : max_length_(-1) {
40
1/2
✓ Branch 1 taken 41384 times.
✗ Branch 2 not taken.
41384 InitValidRanges(whitelist);
41 41384 }
42
43
44 191 InputSanitizer::InputSanitizer(const string &whitelist, int max_length)
45 191 : max_length_(max_length) {
46
1/2
✓ Branch 1 taken 191 times.
✗ Branch 2 not taken.
191 InitValidRanges(whitelist);
47 191 }
48
49
50 41575 void InputSanitizer::InitValidRanges(const std::string &whitelist) {
51 // Parse the whitelist
52 41575 const unsigned length = whitelist.length();
53 41575 unsigned pickup_pos = 0;
54
2/2
✓ Branch 0 taken 136382 times.
✓ Branch 1 taken 41575 times.
177957 for (unsigned i = 0; i < length; ++i) {
55
7/8
✓ Branch 0 taken 94844 times.
✓ Branch 1 taken 41538 times.
✓ Branch 3 taken 47428 times.
✓ Branch 4 taken 47416 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 47428 times.
✓ Branch 7 taken 88954 times.
✓ Branch 8 taken 47428 times.
136382 if ((i + 1 >= length) || (whitelist[i + 1] == ' ') || (i == length - 1)) {
56
1/2
✓ Branch 1 taken 88954 times.
✗ Branch 2 not taken.
88954 const string range = whitelist.substr(pickup_pos, i - pickup_pos + 1);
57
2/3
✓ Branch 1 taken 41526 times.
✓ Branch 2 taken 47428 times.
✗ Branch 3 not taken.
88954 switch (range.length()) {
58 41526 case 1:
59
1/2
✓ Branch 3 taken 41526 times.
✗ Branch 4 not taken.
41526 valid_ranges_.push_back(CharRange(range[0]));
60 41526 break;
61 47428 case 2:
62
1/2
✓ Branch 4 taken 47428 times.
✗ Branch 5 not taken.
47428 valid_ranges_.push_back(CharRange(range[0], range[1]));
63 47428 break;
64 default:
65 assert(false);
66 }
67 88954 ++i;
68 88954 pickup_pos = i + 1;
69 88954 }
70 }
71 41575 }
72
73
74 43310 bool InputSanitizer::Sanitize(std::string::const_iterator begin,
75 std::string::const_iterator end,
76 std::string *filtered_output) const {
77 43310 int pos = 0;
78 43310 bool is_sane = true;
79
2/2
✓ Branch 2 taken 385347 times.
✓ Branch 3 taken 43162 times.
428509 for (; begin != end; ++begin) {
80
2/2
✓ Branch 2 taken 307906 times.
✓ Branch 3 taken 77441 times.
385347 if (CheckRanges(*begin)) {
81
4/4
✓ Branch 0 taken 1879 times.
✓ Branch 1 taken 306027 times.
✓ Branch 2 taken 148 times.
✓ Branch 3 taken 1731 times.
307906 if ((max_length_ >= 0) && (pos >= max_length_)) {
82 148 is_sane = false;
83 148 break;
84 }
85 307758 filtered_output->push_back(*begin);
86 307758 pos++;
87 } else {
88 77441 is_sane = false;
89 }
90 }
91 43310 return is_sane;
92 }
93
94
95 385347 bool InputSanitizer::CheckRanges(const char chr) const {
96
2/2
✓ Branch 1 taken 575501 times.
✓ Branch 2 taken 77441 times.
652942 for (unsigned j = 0; j < valid_ranges_.size(); ++j) {
97
2/2
✓ Branch 2 taken 307906 times.
✓ Branch 3 taken 267595 times.
575501 if (valid_ranges_[j].InRange(chr)) {
98 307906 return true;
99 }
100 }
101 77441 return false;
102 }
103
104
105 480 string InputSanitizer::Filter(const std::string &input) const {
106 480 string filtered_output;
107
1/2
✓ Branch 1 taken 480 times.
✗ Branch 2 not taken.
480 Sanitize(input, &filtered_output);
108 480 return filtered_output;
109 }
110
111
112 42904 bool InputSanitizer::IsValid(const std::string &input) const {
113 42904 string dummy;
114
1/2
✓ Branch 1 taken 42904 times.
✗ Branch 2 not taken.
85808 return Sanitize(input, &dummy);
115 42904 }
116
117
118 1114 bool IntegerSanitizer::Sanitize(std::string::const_iterator begin,
119 std::string::const_iterator end,
120 std::string *filtered_output) const {
121
2/2
✓ Branch 1 taken 37 times.
✓ Branch 2 taken 1077 times.
1114 if (std::distance(begin, end) == 0) {
122 37 return false;
123 }
124
125
2/2
✓ Branch 1 taken 74 times.
✓ Branch 2 taken 1003 times.
1077 if (*begin == '-') {
126 // minus is allowed as the first character!
127 74 filtered_output->push_back('-');
128 74 begin++;
129 }
130
131 1077 return InputSanitizer::Sanitize(begin, end, filtered_output);
132 }
133
134
135 370 bool PositiveIntegerSanitizer::Sanitize(std::string::const_iterator begin,
136 std::string::const_iterator end,
137 std::string *filtered_output) const {
138
2/2
✓ Branch 1 taken 37 times.
✓ Branch 2 taken 333 times.
370 if (std::distance(begin, end) == 0) {
139 37 return false;
140 }
141
142 333 return InputSanitizer::Sanitize(begin, end, filtered_output);
143 }
144
145 } // namespace sanitizer
146
147 #ifdef CVMFS_NAMESPACE_GUARD
148 } // namespace CVMFS_NAMESPACE_GUARD
149 #endif
150