-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathstringKeyMap.h
181 lines (160 loc) · 5.27 KB
/
stringKeyMap.h
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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
/*
@copyright Russell Standish 2016
@author Russell Standish
This file is part of Classdesc
Open source licensed under the MIT license. See LICENSE for details.
*/
#ifndef CLASSDESC_STRINGKEYMAP_H
#define CLASSDESC_STRINGKEYMAP_H
#include <map>
#include <string>
#include "classdesc.h"
#include "json_pack_base.h"
#include "xml_pack_base.h"
#include "xml_unpack_base.h"
#include "xsd_generate_base.h"
namespace classdesc
{
/// A specialisation of a map with string keys, allowing JSON and
/// XML objects to be mapped directly
template <class T>
struct StringKeyMap: public std::map<std::string, T> {};
template <class T>
struct is_associative_container<classdesc::StringKeyMap<T> >: public true_type {};
template <class T> struct is_stringKeyMap<classdesc::StringKeyMap<T> >: public true_type {};
}
namespace classdesc
{
template <class T>
void convert(StringKeyMap<T>& map, const json_pack_t& j)
{if (j.type()!=RESTProcessType::null) j>>map;}
template <class T>
void json_pack_stringKeyMap(json_pack_t& j,const string& d, const StringKeyMap<T>& a)
{
try
{
json5_parser::mValue& parent=json_find(j,head(d));
if (parent.type()!=json5_parser::obj_type)
throw json_pack_error("attempt to pack an array member into a non-object");
else
{
typename StringKeyMap<T>::const_iterator i=a.begin();
for (; i!=a.end(); ++i)
json_pack(j,d+"."+i->first, i->second);
}
}
catch (json_pack_error&)
{
if (j.throw_on_error) throw;
}
}
template <class T>
void json_unpack_stringKeyMap(json_pack_t& j,const string& d, StringKeyMap<T>& a)
{
try
{
const json5_parser::mValue& val=json_find(j,d);
switch (val.type())
{
case json5_parser::obj_type:
{
const json5_parser::mObject& arr=val.get_obj();
a.clear();
for (json5_parser::mObject::const_iterator i=arr.begin(); i!=arr.end(); ++i)
json_unpack(j,d+"."+i->first,a[i->first]);
}
break;
case json5_parser::null_type:
a.clear();
break;
default:
throw json_pack_error("%s is not an array",d.c_str());
}
}
catch (json_pack_error&)
{
if (j.throw_on_error) throw;
}
}
template <class T>
void xsd_generate(xsd_generate_t& g, const string& d, const StringKeyMap<T>& a)
{
std::ostringstream os;
// element name is given by the type name
string eName=typeName<T>().c_str();
eName=eName.substr(0,eName.find('<')); //trim off any template args
// strip leading namespace and qualifiers
const char *el=eName.c_str()+eName.length();
while (el!=eName.c_str() && *(el-1)!=' ' && *(el-1)!=':') el--;
string type=transformTypeName(typeName<StringKeyMap<T> >());
os << " <xs:complexType name=\"" << type << "\">\n";
os << " <xs:sequence minOccurs=\"0\" maxOccurs=\"unbounded\">\n";
// There doesn't seem to be any way of specifying the element type
// here without also constraining the element name!
// os << " <xs:element name=\".*\" type=\""<<xsd_typeName<T>()<<"\"/>\n";
os << " <xs:any processContents=\"lax\"/>\n";
os << " </xs:sequence>\n";
os << " </xs:complexType>\n";
g.addMember(tail(d), xsd_typeName<StringKeyMap<T> >());
g.defineType(type, os.str());
g.addDependency(type, xsd_typeName<T>());
// ensure that the value type as a definition also
xsd_generate(g,"",T());
}
}
using classdesc::xsd_generate;
namespace classdesc_access
{
namespace cd=classdesc;
template <class T>
struct access_json_pack<classdesc::StringKeyMap<T> >
{
template <class U>
void operator()(cd::json_pack_t& j,const cd::string& d,U& x)
{json_pack_stringKeyMap(j,d,x);}
};
template <class T>
struct access_json_unpack<classdesc::StringKeyMap<T> >
{
void operator()(cd::json_unpack_t& j,const cd::string& d,cd::StringKeyMap<T>& x)
{json_unpack_stringKeyMap(j,d,x);}
void operator()(cd::json_unpack_t& j,const cd::string& d,const cd::StringKeyMap<T>& x)
{}
};
template <class T>
struct access_xml_pack<classdesc::StringKeyMap<T> >
{
template <class U>
void operator()(cd::xml_pack_t& x,const cd::string& d,U& arg)
{
cd::xml_pack_t::Tag tag(x,d);
for (typename cd::StringKeyMap<T>::const_iterator i=arg.begin();
i!=arg.end(); ++i)
::xml_pack(x,d+"."+i->first,i->second);
}
};
template <class T>
struct access_xml_unpack<classdesc::StringKeyMap<T> >
{
template <class U>
void operator()(cd::xml_unpack_t& x,const cd::string& d,U& arg)
{
for (cd::xml_unpack_t::ContentMap::const_iterator i=x.firstToken(d);
i!=x.endToken(d); ++i)
{
std::string key=i->first;
size_t p=i->first.rfind('.');
if (p!=std::string::npos) key=key.substr(p);
::xml_unpack(x,i->first,arg[key]);
}
}
};
}
#ifdef _CLASSDESC
#pragma omit json_pack classdesc::StringKeyMap
#pragma omit json_unpack classdesc::StringKeyMap
#pragma omit xml_pack classdesc::StringKeyMap
#pragma omit xml_unpack classdesc::StringKeyMap
#pragma omit xsd_generate classdesc::StringKeyMap
#endif
#endif