GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/util/string.cc
Date: 2026-06-28 02:36:10
Exec Total Coverage
Lines: 319 332 96.1%
Branches: 273 392 69.6%

Line Branch Exec Source
1 /**
2 * This file is part of the CernVM File System.
3 *
4 * Some common functions.
5 */
6
7 #include "string.h"
8
9 #include <cctype>
10 #include <cstdio>
11 #include <cstdlib>
12 #include <cstring>
13 #include <ctime>
14 #include <algorithm>
15 #include <vector>
16 #include <map>
17 #include <stdio.h>
18 #include <errno.h>
19 #include <fcntl.h>
20 #include <inttypes.h>
21 #include <stdint.h>
22 #include <time.h>
23 #include <string>
24 #include <unistd.h>
25
26
27 using namespace std; // NOLINT
28
29 #ifdef CVMFS_NAMESPACE_GUARD
30 namespace CVMFS_NAMESPACE_GUARD {
31 #endif
32
33 const char b64_table[] = {'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K',
34 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V',
35 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f', 'g',
36 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r',
37 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '0', '1', '2',
38 '3', '4', '5', '6', '7', '8', '9', '+', '/'};
39
40 /**
41 * Decode Base64 and Base64Url
42 */
43 const int8_t db64_table[] = {
44 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
45 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
46 -1, -1, -1, -1, -1, 62, -1, 62, -1, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60,
47 61, -1, -1, -1, 0, -1, -1, -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10,
48 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, -1, -1, -1, -1,
49 63, -1, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42,
50 43, 44, 45, 46, 47, 48, 49, 50, 51, -1, -1, -1, -1, -1,
51
52 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
53 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
54 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
55 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
56 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
57 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
58 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
59 };
60
61 namespace {
62
63 /**
64 * Used for cas insensitive HasSuffix
65 */
66 struct IgnoreCaseComperator {
67 27809 IgnoreCaseComperator() { }
68 489 bool operator()(const std::string::value_type a,
69 const std::string::value_type b) const {
70 489 return std::tolower(a) == std::tolower(b);
71 }
72 };
73
74 } // anonymous namespace
75
76
3/4
✓ Branch 1 taken 120 times.
✓ Branch 2 taken 2924 times.
✓ Branch 4 taken 3044 times.
✗ Branch 5 not taken.
3044 string StringifyBool(const bool value) { return value ? "yes" : "no"; }
77
78 44919204 string StringifyInt(const int64_t value) {
79 char buffer[48];
80 44919204 snprintf(buffer, sizeof(buffer), "%" PRId64, value);
81
1/2
✓ Branch 2 taken 44702340 times.
✗ Branch 3 not taken.
44919204 return string(buffer);
82 }
83
84 367089 std::string StringifyUint(const uint64_t value) {
85 char buffer[48];
86 367089 snprintf(buffer, sizeof(buffer), "%" PRIu64, value);
87
1/2
✓ Branch 2 taken 367089 times.
✗ Branch 3 not taken.
367089 return string(buffer);
88 }
89
90 96 string StringifyByteAsHex(const unsigned char value) {
91 char buffer[3];
92 96 snprintf(buffer, sizeof(buffer), "%02x", value);
93
1/2
✓ Branch 2 taken 96 times.
✗ Branch 3 not taken.
96 return string(buffer);
94 }
95
96 82 string StringifyDouble(const double value) {
97 char buffer[64];
98 82 snprintf(buffer, sizeof(buffer), "%.03f", value);
99
1/2
✓ Branch 2 taken 82 times.
✗ Branch 3 not taken.
82 return string(buffer);
100 }
101
102 /**
103 * Converts seconds since UTC 0 into something readable
104 */
105 1216 string StringifyTime(const time_t seconds, const bool utc) {
106 struct tm timestamp;
107
2/2
✓ Branch 0 taken 631 times.
✓ Branch 1 taken 585 times.
1216 if (utc) {
108 631 localtime_r(&seconds, &timestamp);
109 } else {
110 585 gmtime_r(&seconds, &timestamp);
111 }
112
113 1216 const char *months[] = {"Jan", "Feb", "Mar", "Apr", "May", "Jun",
114 "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"};
115 char buffer[21];
116 1216 snprintf(buffer, sizeof(buffer), "%d %s %d %02d:%02d:%02d", timestamp.tm_mday,
117 1216 months[timestamp.tm_mon], timestamp.tm_year + 1900,
118 timestamp.tm_hour, timestamp.tm_min, timestamp.tm_sec);
119
120
1/2
✓ Branch 2 taken 1216 times.
✗ Branch 3 not taken.
1216 return string(buffer);
121 }
122
123 /**
124 * Converts seconds since UTC 0 into something like 12 Sep 14:59:37 CDT
125 */
126 47 string StringifyLocalTime(const time_t seconds) {
127 struct tm timestamp;
128 47 localtime_r(&seconds, &timestamp);
129
130 47 const char *months[] = {"Jan", "Feb", "Mar", "Apr", "May", "Jun",
131 "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"};
132 char buffer[26];
133 (void)/* cast to void ignores return and placates clang-tidy */
134 47 snprintf(buffer, sizeof(buffer), "%d %s %d %02d:%02d:%02d %s",
135 47 timestamp.tm_mday, months[timestamp.tm_mon],
136 47 timestamp.tm_year + 1900, timestamp.tm_hour, timestamp.tm_min,
137 timestamp.tm_sec, timestamp.tm_zone);
138
139
1/2
✓ Branch 2 taken 47 times.
✗ Branch 3 not taken.
47 return string(buffer);
140 }
141
142
143 /**
144 * Current time in format Wed, 01 Mar 2006 12:00:00 GMT
145 */
146 59427 std::string RfcTimestamp() {
147 59427 const char *months[] = {"Jan", "Feb", "Mar", "Apr", "May", "Jun",
148 "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"};
149 59427 const char *day_of_week[] = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"};
150
151 struct tm timestamp;
152 59427 const time_t now = time(NULL);
153 59427 gmtime_r(&now, &timestamp);
154
155 char buffer[30];
156 59427 snprintf(buffer, sizeof(buffer), "%s, %02d %s %d %02d:%02d:%02d %s",
157 59427 day_of_week[timestamp.tm_wday], timestamp.tm_mday,
158 59427 months[timestamp.tm_mon], timestamp.tm_year + 1900,
159 timestamp.tm_hour, timestamp.tm_min, timestamp.tm_sec,
160 timestamp.tm_zone);
161
1/2
✓ Branch 2 taken 59427 times.
✗ Branch 3 not taken.
59427 return string(buffer);
162 }
163
164
165 /**
166 * Current time in format YYYYMMDDTHHMMSSZ. Used in AWS4 requests.
167 */
168 27 std::string IsoTimestamp() {
169 struct tm timestamp;
170 27 const time_t now = time(NULL);
171 27 gmtime_r(&now, &timestamp);
172
173 char buffer[17];
174 27 snprintf(buffer, sizeof(buffer), "%04d%02d%02dT%02d%02d%02dZ",
175 27 timestamp.tm_year + 1900, timestamp.tm_mon + 1, timestamp.tm_mday,
176 timestamp.tm_hour, timestamp.tm_min, timestamp.tm_sec);
177
1/2
✓ Branch 2 taken 27 times.
✗ Branch 3 not taken.
27 return string(buffer);
178 }
179
180
181 /**
182 * UTC time in format YYYYMMDDHHMMSS. Used in cvmfs whitelists.
183 */
184 294 std::string WhitelistTimestamp(time_t when) {
185 struct tm timestamp;
186 294 gmtime_r(&when, &timestamp);
187
188 char buffer[15];
189 294 snprintf(buffer, sizeof(buffer), "%04d%02d%02d%02d%02d%02d",
190 294 timestamp.tm_year + 1900, timestamp.tm_mon + 1, timestamp.tm_mday,
191 timestamp.tm_hour, timestamp.tm_min, timestamp.tm_sec);
192
1/2
✓ Branch 2 taken 294 times.
✗ Branch 3 not taken.
294 return string(buffer);
193 }
194
195
196 17886264 string StringifyTimeval(const timeval value) {
197 char buffer[64];
198 17886264 int64_t msec = value.tv_sec * 1000;
199 17886264 msec += value.tv_usec / 1000;
200 17886264 snprintf(buffer, sizeof(buffer), "%" PRId64 ".%03d", msec,
201 17886264 static_cast<int>(value.tv_usec % 1000));
202
1/2
✓ Branch 2 taken 17886264 times.
✗ Branch 3 not taken.
17886264 return string(buffer);
203 }
204
205 /**
206 * Parses a timestamp of the form YYYY-MM-DDTHH:MM:SSZ
207 * Return 0 on error
208 */
209 1218 time_t IsoTimestamp2UtcTime(const std::string &iso8601) {
210 1218 time_t utc_time = 0;
211 1218 const unsigned length = iso8601.length();
212
213
2/2
✓ Branch 0 taken 30 times.
✓ Branch 1 taken 1188 times.
1218 if (length != 20)
214 30 return utc_time;
215
2/4
✓ Branch 2 taken 1188 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 1188 times.
✗ Branch 6 not taken.
2376 if ((iso8601[4] != '-') || (iso8601[7] != '-') || (iso8601[10] != 'T')
216
5/10
✓ Branch 0 taken 1188 times.
✗ Branch 1 not taken.
✓ Branch 3 taken 1188 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 1188 times.
✗ Branch 7 not taken.
✗ Branch 9 not taken.
✓ Branch 10 taken 1188 times.
✗ Branch 11 not taken.
✓ Branch 12 taken 1188 times.
2376 || (iso8601[13] != ':') || (iso8601[16] != ':') || (iso8601[19] != 'Z')) {
217 return utc_time;
218 }
219
220 struct tm tm_wl;
221 1188 memset(&tm_wl, 0, sizeof(struct tm));
222
2/4
✓ Branch 1 taken 1188 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1188 times.
✗ Branch 5 not taken.
1188 tm_wl.tm_year = static_cast<int>(String2Int64(iso8601.substr(0, 4))) - 1900;
223
2/4
✓ Branch 1 taken 1188 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1188 times.
✗ Branch 5 not taken.
1188 tm_wl.tm_mon = static_cast<int>(String2Int64(iso8601.substr(5, 2))) - 1;
224
2/4
✓ Branch 1 taken 1188 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1188 times.
✗ Branch 5 not taken.
1188 tm_wl.tm_mday = static_cast<int>(String2Int64(iso8601.substr(8, 2)));
225
2/4
✓ Branch 1 taken 1188 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1188 times.
✗ Branch 5 not taken.
1188 tm_wl.tm_hour = static_cast<int>(String2Int64(iso8601.substr(11, 2)));
226
2/4
✓ Branch 1 taken 1188 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1188 times.
✗ Branch 5 not taken.
1188 tm_wl.tm_min = static_cast<int>(String2Int64(iso8601.substr(14, 2)));
227
2/4
✓ Branch 1 taken 1188 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1188 times.
✗ Branch 5 not taken.
1188 tm_wl.tm_sec = static_cast<int>(String2Int64(iso8601.substr(17, 2)));
228 1188 utc_time = timegm(&tm_wl);
229
2/2
✓ Branch 0 taken 15 times.
✓ Branch 1 taken 1173 times.
1188 if (utc_time < 0)
230 15 return 0;
231
232 1173 return utc_time;
233 }
234
235 37907 int64_t String2Int64(const string &value) {
236 int64_t result;
237 37907 sscanf(value.c_str(), "%" PRId64, &result);
238 37907 return result;
239 }
240
241 306567 uint64_t String2Uint64(const string &value) {
242 uint64_t result;
243
2/2
✓ Branch 1 taken 306403 times.
✓ Branch 2 taken 164 times.
306567 if (sscanf(value.c_str(), "%" PRIu64, &result) == 1) {
244 306403 return result;
245 }
246 164 return 0;
247 }
248
249 /**
250 * Parse a string into a a uint64_t.
251 *
252 * Unlike String2Uint64, this:
253 * - Checks to make sure the full string is parsed
254 * - Can indicate an error occurred.
255 *
256 * If an error occurs, this returns false and sets errno appropriately.
257 */
258 3464 bool String2Uint64Parse(const std::string &value, uint64_t *result) {
259 3464 char *endptr = NULL;
260 3464 errno = 0;
261 3464 long long myval = strtoll(value.c_str(), &endptr, 10); // NOLINT
262
2/2
✓ Branch 3 taken 2477 times.
✓ Branch 4 taken 782 times.
6723 if ((value.size() == 0) || (endptr != (value.c_str() + value.size()))
263
6/6
✓ Branch 0 taken 3259 times.
✓ Branch 1 taken 205 times.
✓ Branch 2 taken 13 times.
✓ Branch 3 taken 2464 times.
✓ Branch 4 taken 1000 times.
✓ Branch 5 taken 2464 times.
6723 || (myval < 0)) {
264 1000 errno = EINVAL;
265 1000 return false;
266 }
267
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2464 times.
2464 if (errno) {
268 return false;
269 }
270
2/2
✓ Branch 0 taken 2451 times.
✓ Branch 1 taken 13 times.
2464 if (result) {
271 2451 *result = myval;
272 }
273 2464 return true;
274 }
275
276 120 void String2Uint64Pair(const string &value, uint64_t *a, uint64_t *b) {
277 120 sscanf(value.c_str(), "%" PRIu64 " %" PRIu64, a, b);
278 120 }
279
280 272371 bool HasPrefix(const string &str, const string &prefix,
281 const bool ignore_case) {
282
2/2
✓ Branch 2 taken 64066 times.
✓ Branch 3 taken 208305 times.
272371 if (prefix.length() > str.length())
283 64066 return false;
284
285
2/2
✓ Branch 1 taken 721466 times.
✓ Branch 2 taken 70848 times.
792314 for (unsigned i = 0, l = prefix.length(); i < l; ++i) {
286
2/2
✓ Branch 0 taken 74746 times.
✓ Branch 1 taken 646720 times.
721466 if (ignore_case) {
287
2/2
✓ Branch 2 taken 47567 times.
✓ Branch 3 taken 27179 times.
74746 if (toupper(str[i]) != toupper(prefix[i]))
288 47567 return false;
289 } else {
290
2/2
✓ Branch 2 taken 89890 times.
✓ Branch 3 taken 556830 times.
646720 if (str[i] != prefix[i])
291 89890 return false;
292 }
293 }
294 70848 return true;
295 }
296
297 28191 bool HasSuffix(const std::string &str, const std::string &suffix,
298 const bool ignore_case) {
299
2/2
✓ Branch 2 taken 382 times.
✓ Branch 3 taken 27809 times.
28191 if (suffix.size() > str.size())
300 382 return false;
301 27809 const IgnoreCaseComperator icmp;
302 return (ignore_case)
303
3/4
✓ Branch 0 taken 234 times.
✓ Branch 1 taken 27575 times.
✓ Branch 6 taken 234 times.
✗ Branch 7 not taken.
27809 ? std::equal(suffix.rbegin(), suffix.rend(), str.rbegin(), icmp)
304
1/2
✓ Branch 4 taken 27575 times.
✗ Branch 5 not taken.
27809 : std::equal(suffix.rbegin(), suffix.rend(), str.rbegin());
305 }
306
307 42004 vector<string> SplitString(const string &str, char delim) {
308 42004 return SplitStringBounded(0, str, delim);
309 }
310
311 42178 vector<string> SplitStringBounded(unsigned max_chunks, const string &str,
312 char delim) {
313 42178 vector<string> result;
314
315 // edge case... one chunk is always the whole string
316
2/2
✓ Branch 0 taken 29 times.
✓ Branch 1 taken 42149 times.
42178 if (1 == max_chunks) {
317
1/2
✓ Branch 1 taken 29 times.
✗ Branch 2 not taken.
29 result.push_back(str);
318 29 return result;
319 }
320
321 // split the string
322 42149 const unsigned size = str.size();
323 42149 unsigned marker = 0;
324 42149 unsigned chunks = 1;
325 unsigned i;
326
2/2
✓ Branch 0 taken 1369192 times.
✓ Branch 1 taken 42091 times.
1411283 for (i = 0; i < size; ++i) {
327
2/2
✓ Branch 1 taken 53062 times.
✓ Branch 2 taken 1316130 times.
1369192 if (str[i] == delim) {
328
2/4
✓ Branch 1 taken 53062 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 53062 times.
✗ Branch 5 not taken.
53062 result.push_back(str.substr(marker, i - marker));
329 53062 marker = i + 1;
330
331 // we got what we want... good bye
332
2/2
✓ Branch 0 taken 58 times.
✓ Branch 1 taken 53004 times.
53062 if (++chunks == max_chunks)
333 58 break;
334 }
335 }
336
337 // push the remainings of the string and return
338
2/4
✓ Branch 1 taken 42149 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 42149 times.
✗ Branch 5 not taken.
42149 result.push_back(str.substr(marker));
339 42149 return result;
340 }
341
342 186 vector<string> SplitStringMultiChar(const string &str, const string &delim) {
343 186 size_t pos_start = 0, pos_end = 0, delim_len = delim.length();
344 186 std::string substring;
345 186 std::vector<std::string> result;
346
347
2/2
✓ Branch 1 taken 332 times.
✓ Branch 2 taken 186 times.
518 while ((pos_end = str.find(delim, pos_start)) != string::npos) {
348
1/2
✓ Branch 1 taken 332 times.
✗ Branch 2 not taken.
332 substring = str.substr(pos_start, pos_end - pos_start);
349 332 pos_start = pos_end + delim_len;
350
1/2
✓ Branch 1 taken 332 times.
✗ Branch 2 not taken.
332 result.push_back(substring);
351 }
352
353
2/4
✓ Branch 1 taken 186 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 186 times.
✗ Branch 5 not taken.
186 result.push_back(str.substr(pos_start));
354 372 return result;
355 186 }
356
357 7465 string JoinStrings(const vector<string> &strings, const string &joint) {
358
1/2
✓ Branch 2 taken 7465 times.
✗ Branch 3 not taken.
7465 string result = "";
359 7465 const unsigned size = strings.size();
360
361
2/2
✓ Branch 0 taken 6066 times.
✓ Branch 1 taken 1399 times.
7465 if (size > 0) {
362
1/2
✓ Branch 2 taken 6066 times.
✗ Branch 3 not taken.
6066 result = strings[0];
363
2/2
✓ Branch 0 taken 724 times.
✓ Branch 1 taken 6066 times.
6790 for (unsigned i = 1; i < size; ++i)
364
2/4
✓ Branch 2 taken 724 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 724 times.
✗ Branch 6 not taken.
724 result += joint + strings[i];
365 }
366
367 7465 return result;
368 }
369
370 2977 void ParseKeyvalMem(const unsigned char *buffer, const unsigned buffer_size,
371 map<char, string> *content) {
372 2977 string line;
373 2977 unsigned pos = 0;
374
2/2
✓ Branch 0 taken 178660 times.
✓ Branch 1 taken 86 times.
178746 while (pos < buffer_size) {
375
2/2
✓ Branch 0 taken 16559 times.
✓ Branch 1 taken 162101 times.
178660 if (static_cast<char>(buffer[pos]) == '\n') {
376
2/2
✓ Branch 1 taken 2891 times.
✓ Branch 2 taken 13668 times.
16559 if (line == "--")
377 2891 return;
378
379
1/2
✓ Branch 1 taken 13668 times.
✗ Branch 2 not taken.
13668 if (line != "") {
380
3/11
✗ Branch 1 not taken.
✓ Branch 2 taken 13668 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✗ Branch 7 not taken.
✓ Branch 8 taken 13668 times.
✗ Branch 9 not taken.
✗ Branch 10 not taken.
✓ Branch 11 taken 13668 times.
✗ Branch 12 not taken.
✗ Branch 13 not taken.
13668 const string tail = (line.length() == 1) ? "" : line.substr(1);
381 // Special handling of 'Z' key because it can exist multiple times
382
3/4
✓ Branch 1 taken 13668 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 13578 times.
✓ Branch 4 taken 90 times.
13668 if (line[0] != 'Z') {
383
3/6
✓ Branch 1 taken 13578 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 13578 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 13578 times.
✗ Branch 8 not taken.
13578 (*content)[line[0]] = tail;
384 } else {
385
4/7
✓ Branch 2 taken 90 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✓ Branch 5 taken 90 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 45 times.
✓ Branch 9 taken 45 times.
90 if (content->find(line[0]) == content->end()) {
386
3/6
✓ Branch 1 taken 45 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 45 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 45 times.
✗ Branch 8 not taken.
45 (*content)[line[0]] = tail;
387 } else {
388
6/12
✓ Branch 1 taken 45 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 45 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 45 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 45 times.
✗ Branch 11 not taken.
✓ Branch 13 taken 45 times.
✗ Branch 14 not taken.
✓ Branch 16 taken 45 times.
✗ Branch 17 not taken.
45 (*content)[line[0]] = (*content)[line[0]] + "|" + tail;
389 }
390 }
391 13668 }
392
1/2
✓ Branch 1 taken 13668 times.
✗ Branch 2 not taken.
13668 line = "";
393 } else {
394
1/2
✓ Branch 1 taken 162101 times.
✗ Branch 2 not taken.
162101 line += static_cast<char>(buffer[pos]);
395 }
396 175769 pos++;
397 }
398
2/2
✓ Branch 1 taken 86 times.
✓ Branch 2 taken 2891 times.
2977 }
399
400 463 bool ParseKeyvalPath(const string &filename, map<char, string> *content) {
401
1/2
✓ Branch 2 taken 463 times.
✗ Branch 3 not taken.
463 const int fd = open(filename.c_str(), O_RDONLY);
402
2/2
✓ Branch 0 taken 30 times.
✓ Branch 1 taken 433 times.
463 if (fd < 0)
403 30 return false;
404
405 unsigned char buffer[4096];
406
1/2
✓ Branch 1 taken 433 times.
✗ Branch 2 not taken.
433 const ssize_t num_bytes = read(fd, buffer, sizeof(buffer));
407
1/2
✓ Branch 1 taken 433 times.
✗ Branch 2 not taken.
433 close(fd);
408
409
3/4
✓ Branch 0 taken 403 times.
✓ Branch 1 taken 30 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 403 times.
433 if ((num_bytes <= 0) || (unsigned(num_bytes) >= sizeof(buffer)))
410 30 return false;
411
412
1/2
✓ Branch 1 taken 403 times.
✗ Branch 2 not taken.
403 ParseKeyvalMem(buffer, unsigned(num_bytes), content);
413 403 return true;
414 }
415
416 246403 string GetLineMem(const char *text, const int text_size) {
417 246403 int pos = 0;
418
4/4
✓ Branch 0 taken 10283335 times.
✓ Branch 1 taken 841 times.
✓ Branch 2 taken 10037773 times.
✓ Branch 3 taken 245562 times.
10284176 while ((pos < text_size) && (text[pos] != '\n'))
419 10037773 pos++;
420
1/2
✓ Branch 2 taken 246403 times.
✗ Branch 3 not taken.
246403 return string(text, pos);
421 }
422
423 17921478 bool GetLineFile(FILE *f, std::string *line) {
424 int retval;
425 17921478 line->clear();
426 while (true) {
427 1278112694 retval = fgetc(f);
428
2/6
✗ Branch 1 not taken.
✓ Branch 2 taken 1278112694 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✗ Branch 5 not taken.
✓ Branch 6 taken 1278112694 times.
1278112694 if (ferror(f) && (errno == EINTR)) {
429 clearerr(f);
430 continue;
431
2/2
✓ Branch 0 taken 5337 times.
✓ Branch 1 taken 1278107357 times.
1278112694 } else if (retval == EOF) {
432 5337 break;
433 }
434 1278107357 const char c = static_cast<char>(retval);
435
2/2
✓ Branch 0 taken 17916141 times.
✓ Branch 1 taken 1260191216 times.
1278107357 if (c == '\n')
436 17916141 break;
437 1260191216 line->push_back(c);
438 1260191216 }
439
4/4
✓ Branch 0 taken 5337 times.
✓ Branch 1 taken 17916141 times.
✓ Branch 3 taken 968 times.
✓ Branch 4 taken 4369 times.
17921478 return (retval != EOF) || !line->empty();
440 }
441
442 9075 bool GetLineFd(const int fd, std::string *line) {
443 ssize_t retval;
444 char c;
445 9075 line->clear();
446 while (true) {
447
1/2
✓ Branch 1 taken 426138 times.
✗ Branch 2 not taken.
426138 retval = read(fd, &c, 1);
448
2/2
✓ Branch 0 taken 89 times.
✓ Branch 1 taken 426049 times.
426138 if (retval == 0) {
449 89 break;
450 }
451
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 426049 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
426049 if ((retval == -1) && (errno == EINTR)) {
452 continue;
453 }
454
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 426049 times.
426049 if (retval == -1) {
455 break;
456 }
457
2/2
✓ Branch 0 taken 8986 times.
✓ Branch 1 taken 417063 times.
426049 if (c == '\n')
458 8986 break;
459
1/2
✓ Branch 1 taken 417063 times.
✗ Branch 2 not taken.
417063 line->push_back(c);
460 }
461
4/4
✓ Branch 0 taken 89 times.
✓ Branch 1 taken 8986 times.
✓ Branch 3 taken 62 times.
✓ Branch 4 taken 27 times.
9075 return (retval == 1) || !line->empty();
462 }
463
464 /**
465 * Removes leading and trailing whitespaces.
466 */
467 42414 string Trim(const string &raw, bool trim_newline) {
468
2/2
✓ Branch 1 taken 4193 times.
✓ Branch 2 taken 38221 times.
42414 if (raw.empty())
469
1/2
✓ Branch 2 taken 4193 times.
✗ Branch 3 not taken.
4193 return "";
470
471 38221 unsigned start_pos = 0;
472 40360 for (; (start_pos < raw.length())
473
7/8
✓ Branch 0 taken 40264 times.
✓ Branch 1 taken 96 times.
✓ Branch 3 taken 38340 times.
✓ Branch 4 taken 1924 times.
✓ Branch 6 taken 38340 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 2139 times.
✓ Branch 9 taken 38221 times.
78700 && (raw[start_pos] == ' ' || raw[start_pos] == '\t'
474
2/2
✓ Branch 0 taken 1576 times.
✓ Branch 1 taken 36764 times.
38340 || (trim_newline
475
4/4
✓ Branch 1 taken 1480 times.
✓ Branch 2 taken 96 times.
✓ Branch 4 taken 119 times.
✓ Branch 5 taken 1361 times.
1576 && (raw[start_pos] == '\n' || raw[start_pos] == '\r')));
476 ++start_pos) {
477 }
478 38221 unsigned end_pos = raw.length() - 1; // at least one character in raw
479 42224 for (;
480 (end_pos >= start_pos)
481
7/8
✓ Branch 0 taken 42128 times.
✓ Branch 1 taken 96 times.
✓ Branch 3 taken 39999 times.
✓ Branch 4 taken 2129 times.
✓ Branch 6 taken 39999 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 4003 times.
✓ Branch 9 taken 38221 times.
82223 && (raw[end_pos] == ' ' || raw[end_pos] == '\t'
482
6/6
✓ Branch 0 taken 3235 times.
✓ Branch 1 taken 36764 times.
✓ Branch 3 taken 2042 times.
✓ Branch 4 taken 1193 times.
✓ Branch 6 taken 681 times.
✓ Branch 7 taken 1361 times.
39999 || (trim_newline && (raw[end_pos] == '\n' || raw[end_pos] == '\r')));
483 --end_pos) {
484 }
485
486 38221 return raw.substr(start_pos, end_pos - start_pos + 1);
487 }
488
489 4 std::string TrimString(const std::string &path,
490 const std::string &toTrim,
491 const int trimMode) {
492 4 std::string trimmed = path;
493
1/2
✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
4 if (trimmed != toTrim) {
494
3/4
✓ Branch 1 taken 8 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 6 times.
✓ Branch 4 taken 2 times.
12 while ((trimMode & kTrimLeading) && HasPrefix(trimmed, toTrim, true)
495
5/6
✓ Branch 0 taken 8 times.
✓ Branch 1 taken 2 times.
✓ Branch 4 taken 6 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 6 times.
✓ Branch 7 taken 4 times.
18 && (trimmed.size() > toTrim.size())) {
496
1/2
✓ Branch 2 taken 6 times.
✗ Branch 3 not taken.
6 trimmed = trimmed.substr(toTrim.size());
497 }
498
3/4
✓ Branch 1 taken 6 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 4 times.
✓ Branch 4 taken 2 times.
10 while ((trimMode & kTrimTrailing) && HasSuffix(trimmed, toTrim, true)
499
5/6
✓ Branch 0 taken 6 times.
✓ Branch 1 taken 2 times.
✓ Branch 4 taken 4 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 4 times.
✓ Branch 7 taken 4 times.
14 && (trimmed.size() > toTrim.size())) {
500
1/2
✓ Branch 3 taken 4 times.
✗ Branch 4 not taken.
4 trimmed = trimmed.substr(0, trimmed.size() - toTrim.size());
501 }
502 }
503 4 return trimmed;
504 }
505
506 /**
507 * Converts all characters to upper case
508 */
509 1009 string ToUpper(const string &mixed_case) {
510 1009 string result(mixed_case);
511
2/2
✓ Branch 1 taken 3156 times.
✓ Branch 2 taken 1009 times.
4165 for (unsigned i = 0, l = result.length(); i < l; ++i) {
512
2/4
✓ Branch 1 taken 3156 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 3156 times.
✗ Branch 5 not taken.
3156 result[i] = static_cast<char>(toupper(result[i]));
513 }
514 1009 return result;
515 }
516
517 4516 string ReplaceAll(const string &haystack, const string &needle,
518 const string &replace_by) {
519 4516 string result(haystack);
520 4516 size_t pos = 0;
521 4516 const unsigned needle_size = needle.size();
522
2/2
✓ Branch 1 taken 18 times.
✓ Branch 2 taken 4498 times.
4516 if (needle == "")
523 18 return result;
524
525
2/2
✓ Branch 1 taken 2732 times.
✓ Branch 2 taken 4498 times.
7230 while ((pos = result.find(needle, pos)) != string::npos)
526
1/2
✓ Branch 1 taken 2732 times.
✗ Branch 2 not taken.
2732 result.replace(pos, needle_size, replace_by);
527 4498 return result;
528 }
529
530 665551 static inline void Base64Block(const unsigned char input[3], const char *table,
531 char output[4]) {
532 665551 output[0] = table[(input[0] & 0xFD) >> 2];
533 665551 output[1] = table[((input[0] & 0x03) << 4) | ((input[1] & 0xF0) >> 4)];
534 665551 output[2] = table[((input[1] & 0x0F) << 2) | ((input[2] & 0xD0) >> 6)];
535 665551 output[3] = table[input[2] & 0x3F];
536 665551 }
537
538 81968 string Base64(const string &data) {
539 81968 string result;
540
1/2
✓ Branch 2 taken 81968 times.
✗ Branch 3 not taken.
81968 result.reserve((data.length() + 3) * 4 / 3);
541 81968 unsigned pos = 0;
542 const unsigned char *data_ptr = reinterpret_cast<const unsigned char *>(
543 81968 data.data());
544 81968 const unsigned length = data.length();
545
2/2
✓ Branch 0 taken 583883 times.
✓ Branch 1 taken 81968 times.
665851 while (pos + 2 < length) {
546 char encoded_block[4];
547 583883 Base64Block(data_ptr + pos, b64_table, encoded_block);
548
1/2
✓ Branch 1 taken 583883 times.
✗ Branch 2 not taken.
583883 result.append(encoded_block, 4);
549 583883 pos += 3;
550 }
551
2/2
✓ Branch 0 taken 81668 times.
✓ Branch 1 taken 300 times.
81968 if (length % 3 != 0) {
552 unsigned char input[3];
553 81668 input[0] = data_ptr[pos];
554
2/2
✓ Branch 0 taken 54346 times.
✓ Branch 1 taken 27322 times.
81668 input[1] = ((length % 3) == 2) ? data_ptr[pos + 1] : 0;
555 81668 input[2] = 0;
556 char encoded_block[4];
557 81668 Base64Block(input, b64_table, encoded_block);
558
1/2
✓ Branch 1 taken 81668 times.
✗ Branch 2 not taken.
81668 result.append(encoded_block, 2);
559
3/4
✓ Branch 0 taken 54346 times.
✓ Branch 1 taken 27322 times.
✓ Branch 3 taken 81668 times.
✗ Branch 4 not taken.
81668 result.push_back(((length % 3) == 2) ? encoded_block[2] : '=');
560
1/2
✓ Branch 1 taken 81668 times.
✗ Branch 2 not taken.
81668 result.push_back('=');
561 }
562
563 81968 return result;
564 }
565
566 /**
567 * Safe encoding for URIs and path names: replace + by - and / by _
568 */
569 140 string Base64Url(const string &data) {
570 140 string base64 = Base64(data);
571
2/2
✓ Branch 1 taken 6640 times.
✓ Branch 2 taken 140 times.
6780 for (unsigned i = 0, l = base64.length(); i < l; ++i) {
572
3/4
✓ Branch 1 taken 6640 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 72 times.
✓ Branch 4 taken 6568 times.
6640 if (base64[i] == '+') {
573
1/2
✓ Branch 1 taken 72 times.
✗ Branch 2 not taken.
72 base64[i] = '-';
574
3/4
✓ Branch 1 taken 6568 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 48 times.
✓ Branch 4 taken 6520 times.
6568 } else if (base64[i] == '/') {
575
1/2
✓ Branch 1 taken 48 times.
✗ Branch 2 not taken.
48 base64[i] = '_';
576 }
577 }
578 140 return base64;
579 }
580
581 531330 static bool Debase64Block(const unsigned char input[4],
582 unsigned char output[3]) {
583 int32_t dec[4];
584
2/2
✓ Branch 0 taken 2125284 times.
✓ Branch 1 taken 531318 times.
2656602 for (int i = 0; i < 4; ++i) {
585 2125284 dec[i] = db64_table[input[i]];
586
2/2
✓ Branch 0 taken 12 times.
✓ Branch 1 taken 2125272 times.
2125284 if (dec[i] < 0)
587 12 return false;
588 }
589
590 531318 output[0] = (dec[0] << 2) | (dec[1] >> 4);
591 531318 output[1] = ((dec[1] & 0x0F) << 4) | (dec[2] >> 2);
592 531318 output[2] = ((dec[2] & 0x03) << 6) | dec[3];
593 531318 return true;
594 }
595
596 /**
597 * Can decode both base64 and base64url
598 */
599 2107 bool Debase64(const string &data, string *decoded) {
600 2107 decoded->clear();
601 2107 decoded->reserve((data.length() + 4) * 3 / 4);
602 2107 unsigned pos = 0;
603 const unsigned char *data_ptr = reinterpret_cast<const unsigned char *>(
604 2107 data.data());
605 2107 const unsigned length = data.length();
606
2/2
✓ Branch 0 taken 12 times.
✓ Branch 1 taken 2095 times.
2107 if (length == 0)
607 12 return true;
608
2/2
✓ Branch 0 taken 24 times.
✓ Branch 1 taken 2071 times.
2095 if ((length % 4) != 0)
609 24 return false;
610
611
2/2
✓ Branch 0 taken 531330 times.
✓ Branch 1 taken 2059 times.
533389 while (pos < length) {
612 unsigned char decoded_block[3];
613 531330 const bool retval = Debase64Block(data_ptr + pos, decoded_block);
614
2/2
✓ Branch 0 taken 12 times.
✓ Branch 1 taken 531318 times.
531330 if (!retval)
615 12 return false;
616
1/2
✓ Branch 1 taken 531318 times.
✗ Branch 2 not taken.
531318 decoded->append(reinterpret_cast<char *>(decoded_block), 3);
617 531318 pos += 4;
618 }
619
620
2/2
✓ Branch 0 taken 4118 times.
✓ Branch 1 taken 2059 times.
6177 for (int i = 0; i < 2; ++i) {
621 4118 pos--;
622
2/2
✓ Branch 1 taken 630 times.
✓ Branch 2 taken 3488 times.
4118 if (data[pos] == '=')
623 630 decoded->erase(decoded->length() - 1);
624 }
625 2059 return true;
626 }
627
628 /**
629 * Assumes that source is terminated by a newline
630 */
631 210 string Tail(const string &source, unsigned num_lines) {
632
6/6
✓ Branch 1 taken 150 times.
✓ Branch 2 taken 60 times.
✓ Branch 3 taken 30 times.
✓ Branch 4 taken 120 times.
✓ Branch 5 taken 90 times.
✓ Branch 6 taken 120 times.
210 if (source.empty() || (num_lines == 0))
633
1/2
✓ Branch 2 taken 90 times.
✗ Branch 3 not taken.
90 return "";
634
635 120 const int l = static_cast<int>(source.length());
636 120 int i = l - 1;
637
2/2
✓ Branch 0 taken 480 times.
✓ Branch 1 taken 90 times.
570 for (; i >= 0; --i) {
638 480 const char c = source.data()[i];
639
2/2
✓ Branch 0 taken 150 times.
✓ Branch 1 taken 330 times.
480 if (c == '\n') {
640
2/2
✓ Branch 0 taken 30 times.
✓ Branch 1 taken 120 times.
150 if (num_lines == 0) {
641 30 return source.substr(i + 1);
642 }
643 120 num_lines--;
644 }
645 }
646 90 return source;
647 }
648
649 /**
650 * Get UTC Time.
651 *
652 * @param format format if timestamp (YYYY-MM-DD HH:MM:SS by default)
653 * @return a timestamp string on success, empty string on failure
654 */
655 std::string GetGMTimestamp(const std::string &format) {
656 struct tm time_ptr;
657 char date_and_time[100];
658 const time_t t = time(NULL);
659 gmtime_r(&t, &time_ptr); // take UTC
660 // return empty string if formatting fails
661 if (!strftime(date_and_time, 100, format.c_str(), &time_ptr)) {
662 return "";
663 }
664 std::string timestamp(date_and_time);
665 return timestamp;
666 }
667
668 #ifdef CVMFS_NAMESPACE_GUARD
669 } // namespace CVMFS_NAMESPACE_GUARD
670 #endif
671