GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/json_document.cc
Date: 2025-12-21 02:39:23
Exec Total Coverage
Lines: 118 148 79.7%
Branches: 101 202 50.0%

Line Branch Exec Source
1 /**
2 * This file is part of the CernVM File System.
3 */
4
5 #include "json_document.h"
6
7 #include <cassert>
8 #include <cstdlib>
9 #include <cstring>
10
11 #include "util/exception.h"
12 #include "util/logging.h"
13 #include "util/pointer.h"
14 #include "util/string.h"
15
16 using namespace std; // NOLINT
17
18 1494 JsonDocument *JsonDocument::Create(const string &text) {
19
3/6
✓ Branch 1 taken 1494 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1494 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 1494 times.
✗ Branch 8 not taken.
1494 UniquePtr<JsonDocument> json(new JsonDocument());
20
1/2
✓ Branch 2 taken 1494 times.
✗ Branch 3 not taken.
1494 const bool retval = json->Parse(text);
21
2/2
✓ Branch 0 taken 103 times.
✓ Branch 1 taken 1391 times.
1494 if (!retval)
22 103 return NULL;
23 1391 return json.Release();
24 1494 }
25
26 602 string JsonDocument::EscapeString(const string &input) {
27 602 string escaped;
28
1/2
✓ Branch 2 taken 602 times.
✗ Branch 3 not taken.
602 escaped.reserve(input.length());
29
30
2/2
✓ Branch 1 taken 4386 times.
✓ Branch 2 taken 602 times.
4988 for (unsigned i = 0, s = input.length(); i < s; ++i) {
31
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 4386 times.
4386 if (input[i] == '\\') {
32 escaped.push_back('\\');
33 escaped.push_back('\\');
34
2/2
✓ Branch 1 taken 86 times.
✓ Branch 2 taken 4300 times.
4386 } else if (input[i] == '"') {
35
1/2
✓ Branch 1 taken 86 times.
✗ Branch 2 not taken.
86 escaped.push_back('\\');
36
1/2
✓ Branch 1 taken 86 times.
✗ Branch 2 not taken.
86 escaped.push_back('"');
37 } else {
38
1/2
✓ Branch 2 taken 4300 times.
✗ Branch 3 not taken.
4300 escaped.push_back(input[i]);
39 }
40 }
41 602 return escaped;
42 }
43
44 1494 JsonDocument::JsonDocument()
45 1494 : allocator_(kDefaultBlockSize), root_(NULL), raw_text_(NULL) { }
46
47 1494 JsonDocument::~JsonDocument() {
48
1/2
✓ Branch 0 taken 1494 times.
✗ Branch 1 not taken.
1494 if (raw_text_)
49 1494 free(raw_text_);
50 1494 }
51
52 /**
53 * Parses a JSON string in text.
54 *
55 * @return true if parsing was successful
56 */
57 1494 bool JsonDocument::Parse(const string &text) {
58
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1494 times.
1494 assert(root_ == NULL);
59
60 // The used JSON library 'vjson' is a destructive parser and therefore
61 // alters the content of the provided buffer. The buffer must persist as
62 // name and string values from JSON nodes just point into it.
63 1494 raw_text_ = strdup(text.c_str());
64
65 1494 char *error_pos = 0;
66 1494 char *error_desc = 0;
67 1494 int error_line = 0;
68
1/2
✓ Branch 1 taken 1494 times.
✗ Branch 2 not taken.
1494 JSON *root = json_parse(raw_text_, &error_pos, &error_desc, &error_line,
69 &allocator_);
70
71 // check if the json string was parsed successfully
72
2/2
✓ Branch 0 taken 103 times.
✓ Branch 1 taken 1391 times.
1494 if (!root) {
73
1/2
✓ Branch 1 taken 103 times.
✗ Branch 2 not taken.
103 LogCvmfs(kLogUtility, kLogDebug,
74 "Failed to parse json string. Error at line %d: %s (%s)",
75 error_line, error_desc, error_pos);
76 103 return false;
77 }
78
79 1391 root_ = root;
80 1391 return true;
81 }
82
83 129 string JsonDocument::PrintArray(JSON *first_child, PrintOptions print_options) {
84
1/2
✓ Branch 2 taken 129 times.
✗ Branch 3 not taken.
129 string result = "[";
85
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 129 times.
129 if (print_options.with_whitespace) {
86 result += "\n";
87 print_options.num_indent += 2;
88 }
89 129 JSON *value = first_child;
90
2/2
✓ Branch 0 taken 86 times.
✓ Branch 1 taken 43 times.
129 if (value != NULL) {
91
2/4
✓ Branch 1 taken 86 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 86 times.
✗ Branch 5 not taken.
86 result += PrintValue(value, print_options);
92 86 value = value->next_sibling;
93 }
94
2/2
✓ Branch 0 taken 430 times.
✓ Branch 1 taken 129 times.
559 while (value != NULL) {
95
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 430 times.
✓ Branch 3 taken 430 times.
✗ Branch 4 not taken.
430 result += print_options.with_whitespace ? ",\n" : ",";
96
2/4
✓ Branch 1 taken 430 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 430 times.
✗ Branch 5 not taken.
430 result += PrintValue(value, print_options);
97 430 value = value->next_sibling;
98 }
99
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 129 times.
129 if (print_options.with_whitespace) {
100 result += "\n";
101 for (unsigned i = 2; i < print_options.num_indent; ++i)
102 result.push_back(' ');
103 }
104
1/2
✓ Branch 1 taken 129 times.
✗ Branch 2 not taken.
258 return result + "]";
105 129 }
106
107 /**
108 * JSON string in a canonical format:
109 * - No whitespaces
110 * - Variable names and strings in quotes
111 *
112 * Can be used as a canonical representation to sign or encrypt a JSON text.
113 */
114 129 string JsonDocument::PrintCanonical() {
115
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 129 times.
129 if (!root_)
116 return "";
117 129 const PrintOptions print_options;
118
1/2
✓ Branch 1 taken 129 times.
✗ Branch 2 not taken.
129 return PrintObject(root_->first_child, print_options);
119 }
120
121 301 string JsonDocument::PrintObject(JSON *first_child,
122 PrintOptions print_options) {
123
1/2
✓ Branch 2 taken 301 times.
✗ Branch 3 not taken.
301 string result = "{";
124
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 301 times.
301 if (print_options.with_whitespace) {
125 result += "\n";
126 print_options.num_indent += 2;
127 }
128 301 JSON *value = first_child;
129
2/2
✓ Branch 0 taken 172 times.
✓ Branch 1 taken 129 times.
301 if (value != NULL) {
130
2/4
✓ Branch 1 taken 172 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 172 times.
✗ Branch 5 not taken.
172 result += PrintValue(value, print_options);
131 172 value = value->next_sibling;
132 }
133
2/2
✓ Branch 0 taken 387 times.
✓ Branch 1 taken 301 times.
688 while (value != NULL) {
134
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 387 times.
✓ Branch 3 taken 387 times.
✗ Branch 4 not taken.
387 result += print_options.with_whitespace ? ",\n" : ",";
135
2/4
✓ Branch 1 taken 387 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 387 times.
✗ Branch 5 not taken.
387 result += PrintValue(value, print_options);
136 387 value = value->next_sibling;
137 }
138
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 301 times.
301 if (print_options.with_whitespace) {
139 result += "\n";
140 for (unsigned i = 2; i < print_options.num_indent; ++i)
141 result.push_back(' ');
142 }
143
1/2
✓ Branch 1 taken 301 times.
✗ Branch 2 not taken.
602 return result + "}";
144 301 }
145
146 /**
147 * JSON string for humans.
148 */
149 string JsonDocument::PrintPretty() {
150 if (!root_)
151 return "";
152 PrintOptions print_options;
153 print_options.with_whitespace = true;
154 return PrintObject(root_->first_child, print_options);
155 }
156
157 1075 std::string JsonDocument::PrintValue(JSON *value, PrintOptions print_options) {
158
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1075 times.
1075 assert(value);
159
160 1075 string result;
161
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1075 times.
1075 for (unsigned i = 0; i < print_options.num_indent; ++i)
162 result.push_back(' ');
163
2/2
✓ Branch 0 taken 473 times.
✓ Branch 1 taken 602 times.
1075 if (value->name) {
164
5/10
✓ Branch 2 taken 473 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 473 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 473 times.
✗ Branch 9 not taken.
✓ Branch 11 taken 473 times.
✗ Branch 12 not taken.
✓ Branch 14 taken 473 times.
✗ Branch 15 not taken.
473 result += "\"" + EscapeString(value->name) + "\":";
165
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 473 times.
473 if (print_options.with_whitespace)
166 result += " ";
167 }
168
7/8
✓ Branch 0 taken 86 times.
✓ Branch 1 taken 172 times.
✓ Branch 2 taken 129 times.
✓ Branch 3 taken 129 times.
✓ Branch 4 taken 344 times.
✓ Branch 5 taken 86 times.
✓ Branch 6 taken 129 times.
✗ Branch 7 not taken.
1075 switch (value->type) {
169 86 case JSON_NULL:
170
1/2
✓ Branch 1 taken 86 times.
✗ Branch 2 not taken.
86 result += "null";
171 86 break;
172 172 case JSON_OBJECT:
173
2/4
✓ Branch 1 taken 172 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 172 times.
✗ Branch 5 not taken.
172 result += PrintObject(value->first_child, print_options);
174 172 break;
175 129 case JSON_ARRAY:
176
2/4
✓ Branch 1 taken 129 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 129 times.
✗ Branch 5 not taken.
129 result += PrintArray(value->first_child, print_options);
177 129 break;
178 129 case JSON_STRING:
179
5/10
✓ Branch 2 taken 129 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 129 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 129 times.
✗ Branch 9 not taken.
✓ Branch 11 taken 129 times.
✗ Branch 12 not taken.
✓ Branch 14 taken 129 times.
✗ Branch 15 not taken.
129 result += "\"" + EscapeString(value->string_value) + "\"";
180 129 break;
181 344 case JSON_INT:
182
2/4
✓ Branch 1 taken 344 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 344 times.
✗ Branch 5 not taken.
344 result += StringifyInt(value->int_value);
183 344 break;
184 86 case JSON_FLOAT:
185
2/4
✓ Branch 1 taken 86 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 86 times.
✗ Branch 5 not taken.
86 result += StringifyDouble(value->float_value);
186 86 break;
187 129 case JSON_BOOL:
188
3/4
✓ Branch 0 taken 86 times.
✓ Branch 1 taken 43 times.
✓ Branch 3 taken 129 times.
✗ Branch 4 not taken.
129 result += value->int_value ? "true" : "false";
189 129 break;
190 default:
191 PANIC(NULL);
192 }
193 1075 return result;
194 }
195
196 3129 JSON *JsonDocument::SearchInObject(const JSON *json_object, const string &name,
197 const json_type type) {
198
4/4
✓ Branch 0 taken 3086 times.
✓ Branch 1 taken 43 times.
✓ Branch 2 taken 43 times.
✓ Branch 3 taken 3043 times.
3129 if (!json_object || (json_object->type != JSON_OBJECT))
199 86 return NULL;
200
201 3043 JSON *walker = json_object->first_child;
202
2/2
✓ Branch 0 taken 5614 times.
✓ Branch 1 taken 310 times.
5924 while (walker != NULL) {
203
3/6
✓ Branch 2 taken 5614 times.
✗ Branch 3 not taken.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✓ Branch 7 taken 2733 times.
✓ Branch 8 taken 2881 times.
5614 if (string(walker->name) == name) {
204
2/2
✓ Branch 0 taken 2668 times.
✓ Branch 1 taken 65 times.
2733 return (walker->type == type) ? walker : NULL;
205 }
206 2881 walker = walker->next_sibling;
207 }
208 310 return NULL;
209 }
210
211 template<>
212 18 bool GetFromJSON<std::string>(const JSON *object, const std::string &name,
213 std::string *value) {
214 18 const JSON *o = JsonDocument::SearchInObject(object, name, JSON_STRING);
215
216
2/2
✓ Branch 0 taken 3 times.
✓ Branch 1 taken 15 times.
18 if (o == NULL) {
217 3 return false;
218 }
219
220
1/2
✓ Branch 0 taken 15 times.
✗ Branch 1 not taken.
15 if (value) {
221 15 *value = o->string_value;
222 }
223
224 15 return true;
225 }
226
227 template<>
228 6 bool GetFromJSON<int>(const JSON *object, const std::string &name, int *value) {
229 6 const JSON *o = JsonDocument::SearchInObject(object, name, JSON_INT);
230
231
2/4
✓ Branch 0 taken 6 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 6 times.
6 if (o == NULL || value == NULL) {
232 return false;
233 }
234
235 6 *value = o->int_value;
236
237 6 return true;
238 }
239
240 template<>
241 bool GetFromJSON<float>(const JSON *object, const std::string &name,
242 float *value) {
243 const JSON *o = JsonDocument::SearchInObject(object, name, JSON_FLOAT);
244
245 if (o == NULL || value == NULL) {
246 return false;
247 }
248
249 *value = o->float_value;
250
251 return true;
252 }
253