1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
|
#include "configparser.h"
#include <hugin.hpp>
#include <pugixml.hpp>
static bool assign(std::string& dest, const std::string& val)
{
dest = val;
return true;
}
template<typename T>
static bool attrcpy(T& dest, const pugi::xml_node& src, const std::string& attr, bool opt = false)
{
const char* val = src.attribute(attr.c_str()).as_string(nullptr);
if(!val)
{
if(!opt)
{
ERR(configparser, "Attribute %s not found in %s, offset %d\n",
attr.data(), src.path().data(), (int)src.offset_debug());
}
return opt;
}
if(!assign(dest, std::string(val)))
{
ERR(configparser, "Attribute %s could not be assigned, offset %d\n",
attr.data(), (int)src.offset_debug());
return false;
}
return true;
}
bool ConfigParser::parseString(const std::string& xml)
{
pugi::xml_document doc;
pugi::xml_parse_result result = doc.load_buffer(xml.data(), xml.size());
if(result.status)
{
ERR(configparser, "XML parse error: %d", (int)result.offset);
return false;
}
pugi::xml_node config_node = doc.child("config");
std::string version = "1.0";
attrcpy(version, config_node, "version", true);
if(version != "1.0")
{
ERR(configparser, "Only config v1.0 XML supported.");
return false;
}
for(pugi::xml_node value_node : config_node.children("value"))
{
auto name = value_node.attribute("name").as_string();
if(std::string(name) == "")
{
continue;
}
auto value = value_node.child_value();
values[name] = value;
}
return true;
}
std::string ConfigParser::value(const std::string& name, const std::string& def)
{
if(values.find(name) == values.end())
{
return def;
}
return values[name];
}
|