GCC Code Coverage Report
Directory: ./ Exec Total Coverage
File: tmp_project/FileParser/src/DicoValue.cpp Lines: 168 171 98.2 %
Date: 2024-09-10 03:06:26 Branches: 224 247 90.7 %

Line Branch Exec Source
1
/***************************************
2
	Auteur : Pierre Aubert
3
	Mail : pierre.aubert@lapp.in2p3.fr
4
	Licence : CeCILL-C
5
****************************************/
6
7
#include <fstream>
8
#include "DicoValue.h"
9
10
///Constructor of class DicoValue
11
835
DicoValue::DicoValue(){
12
13
835
}
14
15
///Copy Constructor of class DicoValue
16
/**	@param other : DicoValue we want ot copy
17
*/
18
542
DicoValue::DicoValue(const DicoValue & other){
19
542
	copyDicoValue(other);
20
542
}
21
22
///Destructor of class DicoValue
23
2754
DicoValue::~DicoValue(){
24
25
}
26
27
///Operator = of class DicoValue
28
/**	@param other : DicoValue we want ot copy
29
 * 	@return copied class DicoValue
30
*/
31
325
DicoValue & DicoValue::operator = (const DicoValue & other){
32
325
	copyDicoValue(other);
33
325
	return *this;
34
}
35
36
///Load the DicoValue with a text file
37
/**	@param fileName : name of the file to be loaded
38
 * 	@return true on success, false otherwise
39
*/
40
19
bool DicoValue::load(const std::string & fileName){
41
38
	PFileParser parser;
42
19
	if(!parser.open(fileName)){
43

7
		std::cerr << "DicoValue::load : cannot load the file '"<<fileName<<"'" << std::endl;
44
7
		return false;
45
	}
46
12
	return loadParser(parser);
47
}
48
49
///Save the DicoValue with a text file
50
/**	@param fileName : name of the file to be saved
51
 * 	@return true on success, false otherwise
52
*/
53
9
bool DicoValue::save(const std::string & fileName) const{
54
18
	std::string out("\n{");
55
9
	out += saveRecurse("\t");
56
9
	out += "\n}\n\n";
57
18
	return saveFileContent(fileName, out);
58
}
59
60
///Convert the DicoValue into a string
61
/**	@param indentation : indentation of the current DicoValue
62
 * 	@param isAfterDash : true if the toString is performed after a dash, false otherwise
63
 * 	@return string
64
*/
65
368
std::string DicoValue::toString(const std::string & indentation, int isAfterDash) const{
66
1104
	std::string out(""), newIndentation(indentation), currentIndentation(indentation);
67
368
	if(isAfterDash > 0){
68
80
		currentIndentation = "";
69
	}
70
368
	if(p_key != ""){
71
326
		newIndentation = indentation + "\t";
72
	}
73
368
	if(p_mapChild.size() != 0lu){
74

74
		if(p_key != ""){out += currentIndentation + p_key + ":\n";}
75
400
		for(MapDicoValue::const_iterator it(p_mapChild.begin()); it != p_mapChild.end(); ++it){
76
326
			out += it->second.toString(newIndentation, isAfterDash);
77
		}
78
294
	}else if(p_vecChild.size() != 0lu){
79
44
		std::string dash("- ");
80
22
		if(p_key != ""){
81
22
			out += currentIndentation + p_key + ":\n";
82
		}
83
58
		for(VecDicoValue::const_iterator it(p_vecChild.begin()); it != p_vecChild.end(); ++it){
84

36
			out += newIndentation + dash + it->toString(newIndentation, 2);
85
		}
86
	}else{
87
272
		if(p_key != ""){
88

248
			out += indentation + p_key + ": "+p_value+"\n";
89
		}else{
90
24
			out += p_value + "\n";
91
		}
92
	}
93
368
	--isAfterDash;
94
736
	return out;
95
}
96
97
///Print the DicoValue
98
6
void DicoValue::print() const{
99

6
	std::cout << toString() << std::endl;
100
6
}
101
102
///Say if the DicoValue has a key
103
/**	@return true if the DicoValue has a key, false otherwise
104
*/
105
156
bool DicoValue::hasKey() const{return p_key != "";}
106
107
///Say if the DicoValue has a map of children
108
/**	@return true if the DicoValue has a map of children, false otherwise
109
*/
110
210
bool DicoValue::hasMap() const{return p_mapChild.size() != 0lu;}
111
112
///Say if the DicoValue has a vector of children
113
/**	@return true if the DicoValue has a vector of children, false otherwise
114
*/
115
168
bool DicoValue::hasVec() const{return p_vecChild.size() != 0lu;}
116
117
///Say if the given key exists in the map of children
118
/**	@param key : key to be checked
119
 * 	@return true if the given key exists in the map of children, false otherwise
120
*/
121
1
bool DicoValue::isKeyExist(const std::string & key) const{
122
1
	std::map<std::string, DicoValue>::const_iterator it(p_mapChild.find(key));
123
1
	return it != p_mapChild.end();
124
}
125
126
///Get a DicoValue in the map of the current one
127
/**	@param key : name of the DicoValue to get
128
 * 	@return pointer to the found DicoValue if it exists, NULL otherwise
129
*/
130
54
const DicoValue * DicoValue::getMap(const std::string & key) const{
131
54
	std::map<std::string, DicoValue>::const_iterator it(p_mapChild.find(key));
132
54
	if(it != p_mapChild.end()){
133
46
		return &(it->second);
134
	}else{
135
8
		return NULL;
136
	}
137
}
138
139
///Get a DicoValue in the map of the current one
140
/**	@param key : name of the DicoValue to get
141
 * 	@return pointer to the found DicoValue if it exists, NULL otherwise
142
*/
143
73
DicoValue * DicoValue::getMap(const std::string & key){
144
73
	std::map<std::string, DicoValue>::const_iterator it(p_mapChild.find(key));
145
73
	if(it != p_mapChild.end()){
146
72
		return (DicoValue *)&(it->second);
147
	}else{
148
1
		return NULL;
149
	}
150
}
151
152
///Sets the value of the DicoValue
153
/**	@param value : value of the DicoValue
154
*/
155
343
void DicoValue::setValue(const std::string & value){
156
343
	p_value = value;
157
343
}
158
159
///Sets the key of the DicoValue
160
/**	@param key : key of the DicoValue
161
*/
162
322
void DicoValue::setKey(const std::string & key){
163
322
	p_key = key;
164
322
}
165
166
///Sets the vecChild of the DicoValue
167
/**	@param vecChild : vecChild of the DicoValue
168
*/
169
1
void DicoValue::setVecChild(const std::vector<DicoValue> & vecChild){
170
1
	p_vecChild = vecChild;
171
1
}
172
173
///Sets the mapChild of the DicoValue
174
/**	@param mapChild : mapChild of the DicoValue
175
*/
176
1
void DicoValue::setMapChild(const std::map<std::string, DicoValue> & mapChild){
177
1
	p_mapChild = mapChild;
178
1
}
179
180
///Gets the value of the DicoValue
181
/**	@return value of the DicoValue
182
*/
183
82
const std::string & DicoValue::getValue() const{
184
82
	return p_value;
185
}
186
187
///Gets the value of the DicoValue
188
/**	@return value of the DicoValue
189
*/
190
73
std::string & DicoValue::getValue(){
191
73
	return p_value;
192
}
193
194
///Get a string value without the first and/or last quote or double quote in there are some
195
/**	@return value without the first and/or last quote or double quote in there are some
196
*/
197
156
std::string DicoValue::getString() const{
198
156
	return eraseFirstLastChars(p_value, "\"\'");
199
}
200
201
///Gets the key of the DicoValue
202
/**	@return key of the DicoValue
203
*/
204
2
const std::string & DicoValue::getKey() const{
205
2
	return p_key;
206
}
207
208
///Gets the key of the DicoValue
209
/**	@return key of the DicoValue
210
*/
211
172
std::string & DicoValue::getKey(){
212
172
	return p_key;
213
}
214
215
///Gets the vecChild of the DicoValue
216
/**	@return vecChild of the DicoValue
217
*/
218
23
const std::vector<DicoValue> & DicoValue::getVecChild() const{
219
23
	return p_vecChild;
220
}
221
222
///Gets the vecChild of the DicoValue
223
/**	@return vecChild of the DicoValue
224
*/
225
86
std::vector<DicoValue> & DicoValue::getVecChild(){
226
86
	return p_vecChild;
227
}
228
229
///Gets the mapChild of the DicoValue
230
/**	@return mapChild of the DicoValue
231
*/
232
4
const std::map<std::string, DicoValue> & DicoValue::getMapChild() const{
233
4
	return p_mapChild;
234
}
235
236
///Gets the mapChild of the DicoValue
237
/**	@return mapChild of the DicoValue
238
*/
239
364
std::map<std::string, DicoValue> & DicoValue::getMapChild(){
240
364
	return p_mapChild;
241
}
242
243
///Copy Function of class DicoValue
244
/**	@param other : DicoValue we want ot copy
245
*/
246
867
void DicoValue::copyDicoValue(const DicoValue & other){
247
867
	p_value = other.p_value;
248
867
	p_key = other.p_key;
249
867
	p_vecChild = other.p_vecChild;
250
867
	p_mapChild = other.p_mapChild;
251
867
}
252
253
///Load the DicoValue with a parser
254
/**	@param[out] parser : parser to be used
255
 * 	@return true on success, false otherwise
256
*/
257
12
bool DicoValue::loadParser(PFileParser & parser){
258
12
	parser.setEscapeChar('\\');
259
12
	parser.setWhiteSpace(" \t\n");
260
12
	parser.setSeparator(",:{}\"");
261
12
	bool isRunning(true);
262

25
	while(!parser.isEndOfFile() && isRunning){
263
13
		if(parseDicoValue(parser, isRunning)){
264
13
			parser.skipWhiteSpace();
265
		}else{
266
			errorAt(parser, isRunning, "Cannot parse dico value");
267
		}
268
	}
269
12
	return isRunning;
270
}
271
272
///Parse a DicoValue with a text file
273
/**	@param[out] parser : parser to be used
274
 * 	@param[out] isRunning : true to continue the parsing, false to stop it
275
 * 	@return true on success, false otherwise
276
*/
277
86
bool DicoValue::parseDicoValue(PFileParser & parser, bool & isRunning){
278
86
	if(parseListOrMap(parser, isRunning)){return true;}
279
	else{
280
66
		std::string nextKeyOrValue(parseString(parser));
281
66
		if(nextKeyOrValue == ""){
282
48
			nextKeyOrValue = parser.getStrComposedOf("abcdefghijklmnopqsrtuvwxyzABCDEFGHIJKLMNOPQSRTUVWXYZ0123456789._-");
283
48
			if(nextKeyOrValue == ""){
284
4
				return errorAt(parser, isRunning,
285
2
					"Expecting a string or a keywork composed of letters, number, underscore, slash or minus");
286
			}
287
		}
288

64
		if(parser.isMatch(":")){	//It was a key for a dictionnary
289
// 			std::cerr << "DicoValue::parseDicoValue : find key '"<<nextKeyOrValue<<"'" << std::endl;
290
26
			p_key = nextKeyOrValue;
291
26
			if(!parseDicoValue(parser, isRunning)){
292
				return errorAt(parser, isRunning, "Cannot parse value");
293
			}
294
		}else{			//It was a value
295
38
			p_value = nextKeyOrValue;
296
		}
297
64
		parser.skipWhiteSpace();
298
	}
299
64
	return true;
300
}
301
302
///Parse a list or a map
303
/**	@param[out] parser : parser to be used
304
 * 	@param[out] isRunning : true to continue the parsing, false to stop it
305
 * 	@return true on success, false otherwise
306
*/
307
86
bool DicoValue::parseListOrMap(PFileParser & parser, bool & isRunning){
308

86
	if(!parser.isMatch("{")){return false;}	//If this is not a {, then it is not a list or a map
309
310




63
	while(!parser.isEndOfFile() && !parser.isMatch("}") && isRunning){
311
47
		DicoValue dv;
312
47
		if(dv.parseDicoValue(parser, isRunning)){
313
47
			if(dv.p_key != ""){	//It is a dico entry
314
// 				std::cerr << "DicoValue::parseListOrMap : loadParser add DicoValue with key '"<<dv.p_key<<"'" << std::endl;
315
26
				p_mapChild[dv.p_key] = dv;
316
			}else{	//It is a value
317
21
				p_vecChild.push_back(dv);
318
			}
319
		}else{
320
			errorAt(parser, isRunning, "Cannot parse dico value");
321
		}
322

47
		if(parser.isMatch(",")){}
323

19
		else if(parser.isMatchRewind("}")){}
324
		else{
325
4
			return errorAt(parser, isRunning, "Expect ',' or '}' after value");
326
		}
327
	}
328
329
16
	return true;
330
}
331
332
///Parse a string
333
/**	@param[out] parser : parser to be used
334
 * 	@return true on success, false otherwise
335
*/
336
66
std::string DicoValue::parseString(PFileParser & parser){
337

66
	if(parser.isMatch("\"")){
338
16
		return parser.getUntilKeyWithoutPatern("\"");
339

50
	}else if(parser.isMatch("'")){
340
2
		return parser.getUntilKeyWithoutPatern("'");
341
	}
342
48
	return "";
343
}
344
345
///Print the parsing error
346
/**	@param[out] parser : parser to be used
347
 * 	@param[out] isRunning : true to continue the parsing, false to stop it
348
 * 	@param errorMsg : error message
349
 * 	@return true and stop the parsing with isRunning
350
*/
351
6
bool DicoValue::errorAt(PFileParser & parser, bool & isRunning, const std::string & errorMsg){
352
6
	isRunning = false;
353
6
	std::cerr << "DicoValue::errorAt : " << parser.getLocation() << std::endl;
354
6
	std::cerr << "\t" << errorMsg << std::endl;
355
6
	return true;
356
}
357
358
///Save the DicoValue with a text file
359
/**	@param indentation : indentation of the current DicoValue
360
 * 	@return file content
361
*/
362
52
std::string DicoValue::saveRecurse(const std::string & indentation) const{
363
156
	std::string out(""), newIndentation("");
364
52
	if(p_key != ""){
365
25
		newIndentation = indentation + "\t";
366
	}
367
52
	if(p_mapChild.size() != 0lu){
368

9
		if(p_key != ""){out += "\n" + indentation +p_key + ": {";}
369
18
		std::string comma("");
370
34
		for(MapDicoValue::const_iterator it(p_mapChild.begin()); it != p_mapChild.end(); ++it){
371
25
			out += comma;
372
25
			out += it->second.saveRecurse(newIndentation);
373
25
			comma = ",";
374
		}
375

9
		if(p_key != ""){out += "\n"+indentation+"}";}
376
43
	}else if(p_vecChild.size() != 0lu){
377

6
		if(p_key != ""){out += "\n" + indentation + p_key + ": {";}
378
12
		std::string comma("");
379
24
		for(VecDicoValue::const_iterator it(p_vecChild.begin()); it != p_vecChild.end(); ++it){
380
18
			out += comma;
381
18
			out += it->saveRecurse(newIndentation);
382
18
			comma = ", ";
383
		}
384

6
		if(p_key != ""){out += "}";}
385
	}else{
386
74
		std::string valueToSave(p_value);
387

37
		if(findCharsInString(p_value, " \t\n':/")){
388
3
			valueToSave = "\"" + p_value + "\"";
389
		}
390


37
		if(p_key != ""){out += "\n" + indentation + p_key + ": "+valueToSave;}
391
21
		else{out += valueToSave;}
392
	}
393
394
104
	return out;
395
}
396
397