GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/options.cc
Date: 2025-11-02 02:35:35
Exec Total Coverage
Lines: 274 361 75.9%
Branches: 280 675 41.5%

Line Branch Exec Source
1 /**
2 * This file is part of the CernVM File System.
3 *
4 * Fills an internal map of key-value pairs from ASCII files in key=value
5 * style. Parameters can be overwritten. Used to read configuration from
6 * /etc/cvmfs/...
7 */
8
9
10 #include "options.h"
11
12 #include <fcntl.h>
13 #include <sys/wait.h>
14 #include <unistd.h>
15
16 #include <cassert>
17 #include <cstdio>
18 #include <cstdlib>
19 #include <utility>
20
21 #include "sanitizer.h"
22 #include "util/exception.h"
23 #include "util/logging.h"
24 #include "util/posix.h"
25 #include "util/string.h"
26
27 using namespace std; // NOLINT
28
29 #ifdef CVMFS_NAMESPACE_GUARD
30 namespace CVMFS_NAMESPACE_GUARD {
31 #endif
32
33
34 14415 static string EscapeShell(const std::string &raw) {
35
2/2
✓ Branch 1 taken 212650 times.
✓ Branch 2 taken 14315 times.
226965 for (unsigned i = 0, l = raw.length(); i < l; ++i) {
36
6/6
✓ Branch 1 taken 187525 times.
✓ Branch 2 taken 25125 times.
✓ Branch 4 taken 168941 times.
✓ Branch 5 taken 18584 times.
✓ Branch 6 taken 100 times.
✓ Branch 7 taken 212550 times.
406716 if (!(((raw[i] >= '0') && (raw[i] <= '9'))
37
4/4
✓ Branch 1 taken 168246 times.
✓ Branch 2 taken 25820 times.
✓ Branch 4 taken 149326 times.
✓ Branch 5 taken 18920 times.
194066 || ((raw[i] >= 'A') && (raw[i] <= 'Z'))
38
5/6
✓ Branch 1 taken 139230 times.
✓ Branch 2 taken 35916 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 139230 times.
✓ Branch 7 taken 19274 times.
✓ Branch 8 taken 16642 times.
175146 || ((raw[i] >= 'a') && (raw[i] <= 'z')) || (raw[i] == '/')
39
6/6
✓ Branch 1 taken 18647 times.
✓ Branch 2 taken 627 times.
✓ Branch 4 taken 10452 times.
✓ Branch 5 taken 8195 times.
✓ Branch 7 taken 356 times.
✓ Branch 8 taken 10096 times.
19274 || (raw[i] == ':') || (raw[i] == '.') || (raw[i] == '_')
40
3/4
✓ Branch 1 taken 100 times.
✓ Branch 2 taken 256 times.
✓ Branch 4 taken 100 times.
✗ Branch 5 not taken.
356 || (raw[i] == '-') || (raw[i] == ','))) {
41 100 goto escape_shell_quote;
42 }
43 }
44
1/2
✓ Branch 1 taken 14315 times.
✗ Branch 2 not taken.
14315 return raw;
45
46 100 escape_shell_quote:
47
1/2
✓ Branch 2 taken 100 times.
✗ Branch 3 not taken.
200 string result = "'";
48
2/2
✓ Branch 1 taken 3564 times.
✓ Branch 2 taken 100 times.
3664 for (unsigned i = 0, l = raw.length(); i < l; ++i) {
49
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 3564 times.
3564 if (raw[i] == '\'')
50 result += "\\";
51
1/2
✓ Branch 2 taken 3564 times.
✗ Branch 3 not taken.
3564 result += raw[i];
52 }
53
1/2
✓ Branch 1 taken 100 times.
✗ Branch 2 not taken.
100 result += "'";
54 100 return result;
55 }
56
57
58 10296 string OptionsManager::TrimParameter(const string &parameter) {
59 10296 string result = Trim(parameter);
60 // Strip "readonly"
61
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 10296 times.
10296 if (result.find("readonly ") == 0) {
62 result = result.substr(9);
63 result = Trim(result);
64
2/2
✓ Branch 1 taken 432 times.
✓ Branch 2 taken 9864 times.
10296 } else if (result.find("export ") == 0) {
65
1/2
✓ Branch 1 taken 432 times.
✗ Branch 2 not taken.
432 result = result.substr(7);
66
1/2
✓ Branch 1 taken 432 times.
✗ Branch 2 not taken.
432 result = Trim(result);
67
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 9864 times.
9864 } else if (result.find("eval ") == 0) {
68 result = result.substr(5);
69 result = Trim(result);
70 }
71 10296 return result;
72 }
73
74 11592 string OptionsManager::SanitizeParameterAssignment(string *line,
75 vector<string> *tokens) {
76 11592 const size_t comment_idx = line->find("#");
77
2/2
✓ Branch 0 taken 864 times.
✓ Branch 1 taken 10728 times.
11592 if (comment_idx != string::npos)
78
1/2
✓ Branch 1 taken 864 times.
✗ Branch 2 not taken.
864 *line = line->substr(0, comment_idx);
79
1/2
✓ Branch 1 taken 11592 times.
✗ Branch 2 not taken.
11592 *line = Trim(*line);
80
2/2
✓ Branch 1 taken 864 times.
✓ Branch 2 taken 10728 times.
11592 if (line->empty())
81
1/2
✓ Branch 2 taken 864 times.
✗ Branch 3 not taken.
864 return "";
82
1/2
✓ Branch 1 taken 10728 times.
✗ Branch 2 not taken.
10728 *tokens = SplitString(*line, '=');
83
2/2
✓ Branch 1 taken 432 times.
✓ Branch 2 taken 10296 times.
10728 if (tokens->size() < 2)
84
1/2
✓ Branch 2 taken 432 times.
✗ Branch 3 not taken.
432 return "";
85
1/2
✓ Branch 2 taken 10296 times.
✗ Branch 3 not taken.
10296 string parameter = TrimParameter((*tokens)[0]);
86
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 10296 times.
10296 if (parameter.find(" ") != string::npos)
87 return "";
88 10296 return parameter;
89 10296 }
90
91 361 void OptionsManager::SwitchTemplateManager(
92 OptionsTemplateManager *opt_templ_mgr_param) {
93
1/2
✓ Branch 0 taken 361 times.
✗ Branch 1 not taken.
361 delete opt_templ_mgr_;
94
1/2
✓ Branch 0 taken 361 times.
✗ Branch 1 not taken.
361 if (opt_templ_mgr_param != NULL) {
95 361 opt_templ_mgr_ = opt_templ_mgr_param;
96 } else {
97 opt_templ_mgr_ = new OptionsTemplateManager();
98 }
99 361 for (std::map<std::string, std::string>::iterator it =
100 361 templatable_values_.begin();
101
2/2
✓ Branch 2 taken 144 times.
✓ Branch 3 taken 361 times.
505 it != templatable_values_.end();
102 144 it++) {
103
2/4
✓ Branch 3 taken 144 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 144 times.
✗ Branch 7 not taken.
144 config_[it->first].value = it->second;
104
2/4
✓ Branch 2 taken 144 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 144 times.
✗ Branch 6 not taken.
144 opt_templ_mgr_->ParseString(&(config_[it->first].value));
105
2/4
✓ Branch 2 taken 144 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 144 times.
✗ Branch 6 not taken.
144 UpdateEnvironment(it->first, config_[it->first]);
106 }
107 361 }
108
109 360 bool SimpleOptionsParser::TryParsePath(const string &config_file) {
110
1/2
✓ Branch 2 taken 360 times.
✗ Branch 3 not taken.
360 LogCvmfs(kLogCvmfs, kLogDebug, "Fast-parsing config file %s",
111 config_file.c_str());
112 360 string line;
113
1/2
✓ Branch 2 taken 360 times.
✗ Branch 3 not taken.
360 FILE *fconfig = fopen(config_file.c_str(), "r");
114
2/2
✓ Branch 0 taken 40 times.
✓ Branch 1 taken 320 times.
360 if (fconfig == NULL)
115 40 return false;
116
117 // Read line by line and extract parameters
118
3/4
✓ Branch 1 taken 4960 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 4640 times.
✓ Branch 4 taken 320 times.
4960 while (GetLineFile(fconfig, &line)) {
119 4640 vector<string> tokens;
120
1/2
✓ Branch 1 taken 4640 times.
✗ Branch 2 not taken.
4640 const string parameter = SanitizeParameterAssignment(&line, &tokens);
121
2/2
✓ Branch 1 taken 1200 times.
✓ Branch 2 taken 3440 times.
4640 if (parameter.empty())
122 1200 continue;
123
124 // Strip quotes from value
125
1/2
✓ Branch 3 taken 3440 times.
✗ Branch 4 not taken.
3440 tokens.erase(tokens.begin());
126
3/6
✓ Branch 2 taken 3440 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 3440 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 3440 times.
✗ Branch 9 not taken.
6880 string value = Trim(JoinStrings(tokens, "="));
127 3440 const unsigned value_length = value.length();
128
2/2
✓ Branch 0 taken 2480 times.
✓ Branch 1 taken 960 times.
3440 if (value_length > 2) {
129
3/7
✓ Branch 1 taken 2480 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 240 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✓ Branch 7 taken 240 times.
2720 if (((value[0] == '"') && ((value[value_length - 1] == '"')))
130
9/13
✓ Branch 0 taken 240 times.
✓ Branch 1 taken 2240 times.
✓ Branch 3 taken 2240 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 240 times.
✓ Branch 6 taken 2000 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 240 times.
✗ Branch 9 not taken.
✓ Branch 10 taken 240 times.
✗ Branch 11 not taken.
✓ Branch 12 taken 480 times.
✓ Branch 13 taken 2000 times.
2720 || ((value[0] == '\'') && ((value[value_length - 1] == '\'')))) {
131
1/2
✓ Branch 1 taken 480 times.
✗ Branch 2 not taken.
480 value = value.substr(1, value_length - 2);
132 }
133 }
134
135 3440 ConfigValue config_value;
136
1/2
✓ Branch 1 taken 3440 times.
✗ Branch 2 not taken.
3440 config_value.source = config_file;
137
1/2
✓ Branch 1 taken 3440 times.
✗ Branch 2 not taken.
3440 config_value.value = value;
138
2/4
✓ Branch 1 taken 3440 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 3440 times.
✗ Branch 5 not taken.
3440 PopulateParameter(parameter, config_value);
139
4/4
✓ Branch 3 taken 3440 times.
✓ Branch 4 taken 1200 times.
✓ Branch 6 taken 3440 times.
✓ Branch 7 taken 1200 times.
5840 }
140
1/2
✓ Branch 1 taken 320 times.
✗ Branch 2 not taken.
320 fclose(fconfig);
141 320 return true;
142 360 }
143
144 648 void BashOptionsManager::ParsePath(const string &config_file,
145 const bool external) {
146
1/2
✓ Branch 2 taken 648 times.
✗ Branch 3 not taken.
648 LogCvmfs(kLogCvmfs, kLogDebug, "Parsing config file %s", config_file.c_str());
147 int retval;
148 int pipe_open[2];
149 int pipe_quit[2];
150 648 pid_t pid_child = 0;
151
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 648 times.
648 if (external) {
152 // cvmfs can run in the process group of automount in which case
153 // autofs won't mount an additional config repository. We create a
154 // short-lived process that detaches from the process group and triggers
155 // autofs to mount the config repository, if necessary. It holds a file
156 // handle to the config file until the main process opened the file, too.
157 MakePipe(pipe_open);
158 MakePipe(pipe_quit);
159 switch (pid_child = fork()) {
160 case -1:
161 PANIC(NULL);
162 case 0: { // Child
163 close(pipe_open[0]);
164 close(pipe_quit[1]);
165 // If this is not a process group leader, create a new session
166 if (getpgrp() != getpid()) {
167 const pid_t new_session = setsid();
168 assert(new_session != (pid_t)-1);
169 }
170 (void)open(config_file.c_str(), O_RDONLY);
171 char ready = 'R';
172 WritePipe(pipe_open[1], &ready, 1);
173 retval = read(pipe_quit[0], &ready, 1);
174 _exit(retval); // Don't flush shared file descriptors
175 }
176 }
177 // Parent
178 close(pipe_open[1]);
179 close(pipe_quit[0]);
180 char ready = 0;
181 ReadPipe(pipe_open[0], &ready, 1);
182 assert(ready == 'R');
183 close(pipe_open[0]);
184 }
185
1/2
✓ Branch 1 taken 648 times.
✗ Branch 2 not taken.
648 const string config_path = GetParentPath(config_file);
186
1/2
✓ Branch 2 taken 648 times.
✗ Branch 3 not taken.
648 FILE *fconfig = fopen(config_file.c_str(), "r");
187
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 648 times.
648 if (pid_child > 0) {
188 char c = 'C';
189 WritePipe(pipe_quit[1], &c, 1);
190 int statloc;
191 waitpid(pid_child, &statloc, 0);
192 close(pipe_quit[1]);
193 }
194
2/2
✓ Branch 0 taken 32 times.
✓ Branch 1 taken 616 times.
648 if (!fconfig) {
195
2/8
✗ Branch 0 not taken.
✓ Branch 1 taken 32 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✗ Branch 7 not taken.
✓ Branch 8 taken 32 times.
32 if (external && !DirectoryExists(config_path)) {
196 string repo_required;
197 if (GetValue("CVMFS_CONFIG_REPO_REQUIRED", &repo_required)
198 && IsOn(repo_required)) {
199 LogCvmfs(
200 kLogCvmfs, kLogStderr | kLogSyslogErr,
201 "required configuration repository directory does not exist: %s",
202 config_path.c_str());
203 // Do not crash as in abort(), which can trigger core file creation
204 // from the mount helper
205 exit(1);
206 }
207
208 LogCvmfs(kLogCvmfs, kLogDebug | kLogSyslogWarn,
209 "configuration repository directory does not exist: %s",
210 config_path.c_str());
211 }
212 32 return;
213 }
214
215 int fd_stdin;
216 int fd_stdout;
217 int fd_stderr;
218
1/2
✓ Branch 1 taken 616 times.
✗ Branch 2 not taken.
616 retval = Shell(&fd_stdin, &fd_stdout, &fd_stderr);
219
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 616 times.
616 assert(retval);
220
221 // Let the shell read the file
222 1232 string line;
223
1/2
✓ Branch 2 taken 616 times.
✗ Branch 3 not taken.
1232 const string newline = "\n";
224
5/15
✗ Branch 1 not taken.
✓ Branch 2 taken 616 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✓ Branch 8 taken 616 times.
✗ Branch 9 not taken.
✗ Branch 10 not taken.
✓ Branch 11 taken 616 times.
✗ Branch 12 not taken.
✓ Branch 14 taken 616 times.
✗ Branch 15 not taken.
✗ Branch 18 not taken.
✓ Branch 19 taken 616 times.
✗ Branch 20 not taken.
✗ Branch 21 not taken.
1232 const string cd = "cd \"" + ((config_path == "") ? "/" : config_path) + "\""
225
1/2
✓ Branch 1 taken 616 times.
✗ Branch 2 not taken.
1232 + newline;
226
1/2
✓ Branch 3 taken 616 times.
✗ Branch 4 not taken.
616 WritePipe(fd_stdin, cd.data(), cd.length());
227
3/4
✓ Branch 1 taken 7568 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 6952 times.
✓ Branch 4 taken 616 times.
7568 while (GetLineFile(fconfig, &line)) {
228
1/2
✓ Branch 3 taken 6952 times.
✗ Branch 4 not taken.
6952 WritePipe(fd_stdin, line.data(), line.length());
229
1/2
✓ Branch 3 taken 6952 times.
✗ Branch 4 not taken.
6952 WritePipe(fd_stdin, newline.data(), newline.length());
230 }
231
1/2
✓ Branch 1 taken 616 times.
✗ Branch 2 not taken.
616 rewind(fconfig);
232
233 // Read line by line and extract parameters
234
3/4
✓ Branch 1 taken 7568 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 6952 times.
✓ Branch 4 taken 616 times.
7568 while (GetLineFile(fconfig, &line)) {
235 6952 vector<string> tokens;
236
1/2
✓ Branch 1 taken 6952 times.
✗ Branch 2 not taken.
6952 const string parameter = SanitizeParameterAssignment(&line, &tokens);
237
2/2
✓ Branch 1 taken 960 times.
✓ Branch 2 taken 5992 times.
6952 if (parameter.empty())
238 960 continue;
239
240 5992 ConfigValue value;
241
1/2
✓ Branch 1 taken 5992 times.
✗ Branch 2 not taken.
5992 value.source = config_file;
242
2/4
✓ Branch 1 taken 5992 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5992 times.
✗ Branch 5 not taken.
5992 const string sh_echo = "echo $" + parameter + "\n";
243
1/2
✓ Branch 3 taken 5992 times.
✗ Branch 4 not taken.
5992 WritePipe(fd_stdin, sh_echo.data(), sh_echo.length());
244
1/2
✓ Branch 1 taken 5992 times.
✗ Branch 2 not taken.
5992 GetLineFd(fd_stdout, &value.value);
245
2/4
✓ Branch 1 taken 5992 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5992 times.
✗ Branch 5 not taken.
5992 PopulateParameter(parameter, value);
246
4/4
✓ Branch 3 taken 5992 times.
✓ Branch 4 taken 960 times.
✓ Branch 6 taken 5992 times.
✓ Branch 7 taken 960 times.
7912 }
247
248
1/2
✓ Branch 1 taken 616 times.
✗ Branch 2 not taken.
616 close(fd_stderr);
249
1/2
✓ Branch 1 taken 616 times.
✗ Branch 2 not taken.
616 close(fd_stdout);
250
1/2
✓ Branch 1 taken 616 times.
✗ Branch 2 not taken.
616 close(fd_stdin);
251
1/2
✓ Branch 1 taken 616 times.
✗ Branch 2 not taken.
616 fclose(fconfig);
252 }
253
254
255 1231 bool OptionsManager::HasConfigRepository(const string &fqrn,
256 string *config_path) {
257 1231 string cvmfs_mount_dir;
258
3/6
✓ Branch 2 taken 1231 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 1231 times.
✗ Branch 6 not taken.
✗ Branch 9 not taken.
✓ Branch 10 taken 1231 times.
1231 if (!GetValue("CVMFS_MOUNT_DIR", &cvmfs_mount_dir)) {
259 LogCvmfs(kLogCvmfs, kLogDebug | kLogSyslogErr, "CVMFS_MOUNT_DIR missing");
260 return false;
261 }
262
263 1231 string config_repository;
264
4/6
✓ Branch 2 taken 1231 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 1231 times.
✗ Branch 6 not taken.
✓ Branch 9 taken 96 times.
✓ Branch 10 taken 1135 times.
1231 if (GetValue("CVMFS_CONFIG_REPOSITORY", &config_repository)) {
265
3/6
✓ Branch 1 taken 96 times.
✗ Branch 2 not taken.
✗ Branch 4 not taken.
✓ Branch 5 taken 96 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 96 times.
96 if (config_repository.empty() || (config_repository == fqrn))
266 return false;
267
1/2
✓ Branch 1 taken 96 times.
✗ Branch 2 not taken.
96 const sanitizer::RepositorySanitizer repository_sanitizer;
268
2/4
✓ Branch 1 taken 96 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 96 times.
96 if (!repository_sanitizer.IsValid(config_repository)) {
269 LogCvmfs(kLogCvmfs, kLogDebug | kLogSyslogErr,
270 "invalid CVMFS_CONFIG_REPOSITORY: %s",
271 config_repository.c_str());
272 return false;
273 }
274
3/6
✓ Branch 1 taken 96 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 96 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 96 times.
✗ Branch 8 not taken.
96 *config_path = cvmfs_mount_dir + "/" + config_repository + "/etc/cvmfs/";
275 96 return true;
276 96 }
277 1135 return false;
278 1231 }
279
280 void OptionsManager::SetupGlobalEnvironmentParams() {
281 SetValue("CVMFS_ARCH", GetArch());
282
283 // Set CVMFS_VERSION environment variable
284 SetValue("CVMFS_VERSION", CVMFS_VERSION);
285
286 // Calculate and set CVMFS_VERSION_NUMERIC
287 // Format: major * 10000 + minor * 100 + patch
288 // Example: 2.13.2 becomes 21302
289 const int version_numeric = CVMFS_VERSION_MAJOR * 10000
290 + CVMFS_VERSION_MINOR * 100 + CVMFS_VERSION_PATCH;
291 char version_numeric_str[16];
292 snprintf(version_numeric_str, sizeof(version_numeric_str), "%d",
293 version_numeric);
294 SetValue("CVMFS_VERSION_NUMERIC", version_numeric_str);
295 }
296
297 void OptionsManager::ParseDefault(const string &fqrn) {
298 if (taint_environment_) {
299 const int retval = setenv("CVMFS_FQRN", fqrn.c_str(), 1);
300 assert(retval == 0);
301 }
302
303 SetupGlobalEnvironmentParams();
304 protected_parameters_.clear();
305 ParsePath("/etc/cvmfs/default.conf", false);
306 vector<string> dist_defaults = FindFilesBySuffix("/etc/cvmfs/default.d",
307 ".conf");
308 for (unsigned i = 0; i < dist_defaults.size(); ++i) {
309 ParsePath(dist_defaults[i], false);
310 }
311 ProtectParameter("CVMFS_CONFIG_REPOSITORY");
312 string external_config_path;
313 if ((fqrn != "") && HasConfigRepository(fqrn, &external_config_path))
314 ParsePath(external_config_path + "default.conf", true);
315 ParsePath("/etc/cvmfs/default.local", false);
316
317 if (fqrn != "") {
318 string domain;
319 vector<string> tokens = SplitString(fqrn, '.');
320 assert(tokens.size() > 1);
321 tokens.erase(tokens.begin());
322 domain = JoinStrings(tokens, ".");
323
324 if (HasConfigRepository(fqrn, &external_config_path))
325 ParsePath(external_config_path + "domain.d/" + domain + ".conf", true);
326 ParsePath("/etc/cvmfs/domain.d/" + domain + ".conf", false);
327 ParsePath("/etc/cvmfs/domain.d/" + domain + ".local", false);
328
329 if (HasConfigRepository(fqrn, &external_config_path))
330 ParsePath(external_config_path + "config.d/" + fqrn + ".conf", true);
331 ParsePath("/etc/cvmfs/config.d/" + fqrn + ".conf", false);
332 ParsePath("/etc/cvmfs/config.d/" + fqrn + ".local", false);
333 }
334 }
335
336
337 21746 void OptionsManager::PopulateParameter(const string &param, ConfigValue val) {
338
1/2
✓ Branch 1 taken 21746 times.
✗ Branch 2 not taken.
21746 const map<string, string>::const_iterator iter = protected_parameters_.find(
339 21746 param);
340
5/6
✓ Branch 3 taken 72 times.
✓ Branch 4 taken 21674 times.
✓ Branch 7 taken 72 times.
✗ Branch 8 not taken.
✓ Branch 9 taken 72 times.
✓ Branch 10 taken 21674 times.
21746 if ((iter != protected_parameters_.end()) && (iter->second != val.value)) {
341
1/3
✓ Branch 4 taken 72 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
144 LogCvmfs(kLogCvmfs, kLogDebug | kLogSyslogErr,
342 "error in cvmfs configuration: attempt to change protected %s "
343 "from %s to %s",
344 72 param.c_str(), iter->second.c_str(), val.value.c_str());
345 72 return;
346 }
347
2/4
✓ Branch 1 taken 21674 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 21674 times.
✗ Branch 5 not taken.
21674 ParseValue(param, &val);
348
2/4
✓ Branch 1 taken 21674 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 21674 times.
✗ Branch 5 not taken.
21674 config_[param] = val;
349
1/2
✓ Branch 1 taken 21674 times.
✗ Branch 2 not taken.
21674 UpdateEnvironment(param, val);
350 }
351
352 21818 void OptionsManager::UpdateEnvironment(const string &param, ConfigValue val) {
353
2/2
✓ Branch 0 taken 18058 times.
✓ Branch 1 taken 3760 times.
21818 if (taint_environment_) {
354 18058 const int retval = setenv(param.c_str(), val.value.c_str(), 1);
355
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 18058 times.
18058 assert(retval == 0);
356 }
357 21818 }
358
359 21674 void OptionsManager::ParseValue(std::string param, ConfigValue *val) {
360
1/2
✓ Branch 1 taken 21674 times.
✗ Branch 2 not taken.
21674 const string orig = val->value;
361
1/2
✓ Branch 1 taken 21674 times.
✗ Branch 2 not taken.
21674 const bool has_templ = opt_templ_mgr_->ParseString(&(val->value));
362
2/2
✓ Branch 0 taken 504 times.
✓ Branch 1 taken 21170 times.
21674 if (has_templ) {
363
2/4
✓ Branch 1 taken 504 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 504 times.
✗ Branch 5 not taken.
504 templatable_values_[param] = orig;
364 }
365 21674 }
366
367
368 72 void OptionsManager::ProtectParameter(const string &param) {
369 72 string value;
370 // We don't care about the result. If param does not yet exists, we lock it
371 // to the empty string.
372
1/2
✓ Branch 1 taken 72 times.
✗ Branch 2 not taken.
72 (void)GetValue(param, &value);
373
2/4
✓ Branch 1 taken 72 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 72 times.
✗ Branch 5 not taken.
72 protected_parameters_[param] = value;
374 72 }
375
376
377 72 void OptionsManager::ClearConfig() { config_.clear(); }
378
379
380 13301 bool OptionsManager::IsDefined(const std::string &key) {
381
1/2
✓ Branch 1 taken 13301 times.
✗ Branch 2 not taken.
13301 const map<string, ConfigValue>::const_iterator iter = config_.find(key);
382 13301 return iter != config_.end();
383 }
384
385
386 147379 bool OptionsManager::GetValue(const string &key, string *value) const {
387
1/2
✓ Branch 1 taken 147379 times.
✗ Branch 2 not taken.
147379 const map<string, ConfigValue>::const_iterator iter = config_.find(key);
388
2/2
✓ Branch 2 taken 36246 times.
✓ Branch 3 taken 111133 times.
147379 if (iter != config_.end()) {
389
1/2
✓ Branch 2 taken 36246 times.
✗ Branch 3 not taken.
36246 *value = iter->second.value;
390 36246 return true;
391 }
392
1/2
✓ Branch 1 taken 111133 times.
✗ Branch 2 not taken.
111133 *value = "";
393 111133 return false;
394 }
395
396
397 90 std::string OptionsManager::GetValueOrDie(const string &key) {
398 90 std::string value;
399
1/2
✓ Branch 1 taken 90 times.
✗ Branch 2 not taken.
90 const bool retval = GetValue(key, &value);
400
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 90 times.
90 if (!retval) {
401 PANIC(kLogStderr | kLogDebug, "%s configuration parameter missing",
402 key.c_str());
403 }
404 90 return value;
405 }
406
407
408 14847 bool OptionsManager::GetSource(const string &key, string *value) {
409
1/2
✓ Branch 1 taken 14847 times.
✗ Branch 2 not taken.
14847 const map<string, ConfigValue>::const_iterator iter = config_.find(key);
410
1/2
✓ Branch 3 taken 14847 times.
✗ Branch 4 not taken.
14847 if (iter != config_.end()) {
411
1/2
✓ Branch 2 taken 14847 times.
✗ Branch 3 not taken.
14847 *value = iter->second.source;
412 14847 return true;
413 }
414 *value = "";
415 return false;
416 }
417
418
419 2719 bool OptionsManager::IsOn(const std::string &param_value) const {
420
1/2
✓ Branch 1 taken 2719 times.
✗ Branch 2 not taken.
2719 const string uppercase = ToUpper(param_value);
421
2/4
✓ Branch 2 taken 2463 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 2463 times.
✗ Branch 6 not taken.
5182 return ((uppercase == "YES") || (uppercase == "ON") || (uppercase == "1")
422
3/4
✓ Branch 0 taken 2463 times.
✓ Branch 1 taken 256 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 2463 times.
7901 || (uppercase == "TRUE"));
423 2719 }
424
425
426 56 bool OptionsManager::IsOff(const std::string &param_value) const {
427
1/2
✓ Branch 1 taken 56 times.
✗ Branch 2 not taken.
56 const string uppercase = ToUpper(param_value);
428
3/4
✓ Branch 2 taken 28 times.
✓ Branch 3 taken 28 times.
✓ Branch 5 taken 28 times.
✗ Branch 6 not taken.
112 return ((uppercase == "NO") || (uppercase == "OFF") || (uppercase == "0")
429
2/4
✓ Branch 0 taken 56 times.
✗ Branch 1 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 28 times.
168 || (uppercase == "FALSE"));
430 56 }
431
432
433 2480 vector<string> OptionsManager::GetAllKeys() {
434 2480 vector<string> result;
435 2480 for (map<string, ConfigValue>::const_iterator i = config_.begin(),
436 2480 iEnd = config_.end();
437
2/2
✓ Branch 1 taken 15423 times.
✓ Branch 2 taken 2480 times.
17903 i != iEnd;
438 15423 ++i) {
439
1/2
✓ Branch 2 taken 15423 times.
✗ Branch 3 not taken.
15423 result.push_back(i->first);
440 }
441 2480 return result;
442 }
443
444
445 292 vector<string> OptionsManager::GetEnvironmentSubset(const string &key_prefix,
446 bool strip_prefix) {
447 292 vector<string> result;
448 292 for (map<string, ConfigValue>::const_iterator i = config_.begin(),
449 292 iEnd = config_.end();
450
2/2
✓ Branch 1 taken 4032 times.
✓ Branch 2 taken 292 times.
4324 i != iEnd;
451 4032 ++i) {
452 4032 const bool ignore_prefix = false;
453
3/4
✓ Branch 2 taken 4032 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 504 times.
✓ Branch 5 taken 3528 times.
4032 if (HasPrefix(i->first, key_prefix, ignore_prefix)) {
454 const string output_key = strip_prefix
455 72 ? i->first.substr(key_prefix.length())
456
4/6
✓ Branch 0 taken 72 times.
✓ Branch 1 taken 432 times.
✓ Branch 4 taken 72 times.
✗ Branch 5 not taken.
✓ Branch 8 taken 432 times.
✗ Branch 9 not taken.
576 : i->first;
457
3/6
✓ Branch 2 taken 504 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 504 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 504 times.
✗ Branch 9 not taken.
504 result.push_back(output_key + "=" + i->second.value);
458 504 }
459 }
460 292 return result;
461 }
462
463
464 2336 string OptionsManager::Dump() {
465 2336 string result;
466
1/2
✓ Branch 1 taken 2336 times.
✗ Branch 2 not taken.
2336 vector<string> keys = GetAllKeys();
467
2/2
✓ Branch 1 taken 14415 times.
✓ Branch 2 taken 2336 times.
16751 for (unsigned i = 0, l = keys.size(); i < l; ++i) {
468 bool retval;
469 14415 string value;
470 14415 string source;
471
472
1/2
✓ Branch 2 taken 14415 times.
✗ Branch 3 not taken.
14415 retval = GetValue(keys[i], &value);
473
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 14415 times.
14415 assert(retval);
474
1/2
✓ Branch 2 taken 14415 times.
✗ Branch 3 not taken.
14415 retval = GetSource(keys[i], &source);
475
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 14415 times.
14415 assert(retval);
476
5/10
✓ Branch 1 taken 14415 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 14415 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 14415 times.
✗ Branch 9 not taken.
✓ Branch 11 taken 14415 times.
✗ Branch 12 not taken.
✓ Branch 14 taken 14415 times.
✗ Branch 15 not taken.
28830 result += keys[i] + "=" + EscapeShell(value) + " # from " + source
477
2/4
✓ Branch 1 taken 14415 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 14415 times.
✗ Branch 5 not taken.
14415 + "\n";
478 14415 }
479 4672 return result;
480 2336 }
481
482 void OptionsManager::SetValueFromTalk(const string &key, const string &value) {
483 ConfigValue config_value;
484 config_value.source = "cvmfs_talk";
485 config_value.value = value;
486 PopulateParameter(key, config_value);
487 }
488
489 12314 void OptionsManager::SetValue(const string &key, const string &value) {
490 12314 ConfigValue config_value;
491
1/2
✓ Branch 1 taken 12314 times.
✗ Branch 2 not taken.
12314 config_value.source = "@INTERNAL@";
492
1/2
✓ Branch 1 taken 12314 times.
✗ Branch 2 not taken.
12314 config_value.value = value;
493
2/4
✓ Branch 1 taken 12314 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 12314 times.
✗ Branch 5 not taken.
12314 PopulateParameter(key, config_value);
494 12314 }
495
496
497 686 void OptionsManager::UnsetValue(const string &key) {
498 686 protected_parameters_.erase(key);
499 686 config_.erase(key);
500
2/2
✓ Branch 0 taken 614 times.
✓ Branch 1 taken 72 times.
686 if (taint_environment_)
501 614 unsetenv(key.c_str());
502 686 }
503
504 const char *DefaultOptionsTemplateManager ::kTemplateIdentFqrn = "fqrn";
505
506 const char *DefaultOptionsTemplateManager ::kTemplateIdentOrg = "org";
507
508 761 DefaultOptionsTemplateManager::DefaultOptionsTemplateManager(std::string fqrn) {
509
3/6
✓ Branch 1 taken 761 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 761 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 761 times.
✗ Branch 9 not taken.
761 SetTemplate(kTemplateIdentFqrn, fqrn);
510
1/2
✓ Branch 1 taken 761 times.
✗ Branch 2 not taken.
761 vector<string> fqrn_parts = SplitString(fqrn, '.');
511
3/6
✓ Branch 2 taken 761 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 761 times.
✗ Branch 7 not taken.
✓ Branch 9 taken 761 times.
✗ Branch 10 not taken.
761 SetTemplate(kTemplateIdentOrg, fqrn_parts[0]);
512 761 }
513
514 1754 void OptionsTemplateManager::SetTemplate(std::string name, std::string val) {
515 1754 templates_[name] = val;
516 1754 }
517
518 2136 std::string OptionsTemplateManager::GetTemplate(std::string name) {
519
2/2
✓ Branch 1 taken 536 times.
✓ Branch 2 taken 1600 times.
2136 if (templates_.count(name)) {
520 536 return templates_[name];
521 } else {
522
2/4
✓ Branch 1 taken 1600 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1600 times.
✗ Branch 5 not taken.
1600 std::string var_name = "@" + name + "@";
523
1/2
✓ Branch 2 taken 1600 times.
✗ Branch 3 not taken.
1600 LogCvmfs(kLogCvmfs, kLogDebug, "Undeclared variable: %s", var_name.c_str());
524 1600 return var_name;
525 1600 }
526 }
527
528 21978 bool OptionsTemplateManager::ParseString(std::string *input) {
529 21978 std::string result;
530
1/2
✓ Branch 1 taken 21978 times.
✗ Branch 2 not taken.
21978 std::string in = *input;
531 21978 bool has_vars = false;
532 21978 int mode = 0;
533 21978 std::string stock;
534
2/2
✓ Branch 1 taken 490857 times.
✓ Branch 2 taken 21978 times.
512835 for (std::string::size_type i = 0; i < in.size(); i++) {
535
2/3
✓ Branch 0 taken 479513 times.
✓ Branch 1 taken 11344 times.
✗ Branch 2 not taken.
490857 switch (mode) {
536 479513 case 0:
537
3/4
✓ Branch 1 taken 479513 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 2368 times.
✓ Branch 4 taken 477145 times.
479513 if (in[i] == '@') {
538 2368 mode = 1;
539 } else {
540
2/4
✓ Branch 1 taken 477145 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 477145 times.
✗ Branch 5 not taken.
477145 result += in[i];
541 }
542 479513 break;
543 11344 case 1:
544
3/4
✓ Branch 1 taken 11344 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 1856 times.
✓ Branch 4 taken 9488 times.
11344 if (in[i] == '@') {
545 1856 mode = 0;
546
3/6
✓ Branch 1 taken 1856 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1856 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 1856 times.
✗ Branch 8 not taken.
1856 result += GetTemplate(stock);
547
1/2
✓ Branch 1 taken 1856 times.
✗ Branch 2 not taken.
1856 stock = "";
548 1856 has_vars = true;
549 } else {
550
2/4
✓ Branch 1 taken 9488 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 9488 times.
✗ Branch 5 not taken.
9488 stock += in[i];
551 }
552 11344 break;
553 }
554 }
555
2/2
✓ Branch 0 taken 512 times.
✓ Branch 1 taken 21466 times.
21978 if (mode == 1) {
556
2/4
✓ Branch 1 taken 512 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 512 times.
✗ Branch 5 not taken.
512 result += "@" + stock;
557 }
558
1/2
✓ Branch 1 taken 21978 times.
✗ Branch 2 not taken.
21978 *input = result;
559 21978 return has_vars;
560 21978 }
561
562 280 bool OptionsTemplateManager::HasTemplate(std::string name) {
563 280 return templates_.count(name);
564 }
565
566 #ifdef CVMFS_NAMESPACE_GUARD
567 } // namespace CVMFS_NAMESPACE_GUARD
568 #endif
569