Mercurial > hg > piper-cpp
comparison json11/json11.hpp @ 75:81e1c48e97f9
Rearrange and rename to Piper C++ structure
author | Chris Cannam <c.cannam@qmul.ac.uk> |
---|---|
date | Mon, 10 Oct 2016 16:31:09 +0100 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
74:d45cfa25aaad | 75:81e1c48e97f9 |
---|---|
1 /* json11 | |
2 * | |
3 * json11 is a tiny JSON library for C++11, providing JSON parsing and serialization. | |
4 * | |
5 * The core object provided by the library is json11::Json. A Json object represents any JSON | |
6 * value: null, bool, number (int or double), string (std::string), array (std::vector), or | |
7 * object (std::map). | |
8 * | |
9 * Json objects act like values: they can be assigned, copied, moved, compared for equality or | |
10 * order, etc. There are also helper methods Json::dump, to serialize a Json to a string, and | |
11 * Json::parse (static) to parse a std::string as a Json object. | |
12 * | |
13 * Internally, the various types of Json object are represented by the JsonValue class | |
14 * hierarchy. | |
15 * | |
16 * A note on numbers - JSON specifies the syntax of number formatting but not its semantics, | |
17 * so some JSON implementations distinguish between integers and floating-point numbers, while | |
18 * some don't. In json11, we choose the latter. Because some JSON implementations (namely | |
19 * Javascript itself) treat all numbers as the same type, distinguishing the two leads | |
20 * to JSON that will be *silently* changed by a round-trip through those implementations. | |
21 * Dangerous! To avoid that risk, json11 stores all numbers as double internally, but also | |
22 * provides integer helpers. | |
23 * | |
24 * Fortunately, double-precision IEEE754 ('double') can precisely store any integer in the | |
25 * range +/-2^53, which includes every 'int' on most systems. (Timestamps often use int64 | |
26 * or long long to avoid the Y2038K problem; a double storing microseconds since some epoch | |
27 * will be exact for +/- 275 years.) | |
28 */ | |
29 | |
30 /* Copyright (c) 2013 Dropbox, Inc. | |
31 * | |
32 * Permission is hereby granted, free of charge, to any person obtaining a copy | |
33 * of this software and associated documentation files (the "Software"), to deal | |
34 * in the Software without restriction, including without limitation the rights | |
35 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |
36 * copies of the Software, and to permit persons to whom the Software is | |
37 * furnished to do so, subject to the following conditions: | |
38 * | |
39 * The above copyright notice and this permission notice shall be included in | |
40 * all copies or substantial portions of the Software. | |
41 * | |
42 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |
43 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
44 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |
45 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |
46 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |
47 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | |
48 * THE SOFTWARE. | |
49 */ | |
50 | |
51 #pragma once | |
52 | |
53 #include <string> | |
54 #include <vector> | |
55 #include <map> | |
56 #include <memory> | |
57 #include <initializer_list> | |
58 | |
59 namespace json11 { | |
60 | |
61 enum JsonParse { | |
62 STANDARD, COMMENTS | |
63 }; | |
64 | |
65 class JsonValue; | |
66 | |
67 class Json final { | |
68 public: | |
69 // Types | |
70 enum Type { | |
71 NUL, NUMBER, BOOL, STRING, ARRAY, OBJECT | |
72 }; | |
73 | |
74 // Array and object typedefs | |
75 typedef std::vector<Json> array; | |
76 typedef std::map<std::string, Json> object; | |
77 | |
78 // Constructors for the various types of JSON value. | |
79 Json() noexcept; // NUL | |
80 Json(std::nullptr_t) noexcept; // NUL | |
81 Json(double value); // NUMBER | |
82 Json(int value); // NUMBER | |
83 Json(bool value); // BOOL | |
84 Json(const std::string &value); // STRING | |
85 Json(std::string &&value); // STRING | |
86 Json(const char * value); // STRING | |
87 Json(const array &values); // ARRAY | |
88 Json(array &&values); // ARRAY | |
89 Json(const object &values); // OBJECT | |
90 Json(object &&values); // OBJECT | |
91 | |
92 // Implicit constructor: anything with a to_json() function. | |
93 template <class T, class = decltype(&T::to_json)> | |
94 Json(const T & t) : Json(t.to_json()) {} | |
95 | |
96 // Implicit constructor: map-like objects (std::map, std::unordered_map, etc) | |
97 template <class M, typename std::enable_if< | |
98 std::is_constructible<std::string, typename M::key_type>::value | |
99 && std::is_constructible<Json, typename M::mapped_type>::value, | |
100 int>::type = 0> | |
101 Json(const M & m) : Json(object(m.begin(), m.end())) {} | |
102 | |
103 // Implicit constructor: vector-like objects (std::list, std::vector, std::set, etc) | |
104 template <class V, typename std::enable_if< | |
105 std::is_constructible<Json, typename V::value_type>::value, | |
106 int>::type = 0> | |
107 Json(const V & v) : Json(array(v.begin(), v.end())) {} | |
108 | |
109 // This prevents Json(some_pointer) from accidentally producing a bool. Use | |
110 // Json(bool(some_pointer)) if that behavior is desired. | |
111 Json(void *) = delete; | |
112 | |
113 // Accessors | |
114 Type type() const; | |
115 | |
116 bool is_null() const { return type() == NUL; } | |
117 bool is_number() const { return type() == NUMBER; } | |
118 bool is_bool() const { return type() == BOOL; } | |
119 bool is_string() const { return type() == STRING; } | |
120 bool is_array() const { return type() == ARRAY; } | |
121 bool is_object() const { return type() == OBJECT; } | |
122 | |
123 // Return the enclosed value if this is a number, 0 otherwise. Note that json11 does not | |
124 // distinguish between integer and non-integer numbers - number_value() and int_value() | |
125 // can both be applied to a NUMBER-typed object. | |
126 double number_value() const; | |
127 int int_value() const; | |
128 | |
129 // Return the enclosed value if this is a boolean, false otherwise. | |
130 bool bool_value() const; | |
131 // Return the enclosed string if this is a string, "" otherwise. | |
132 const std::string &string_value() const; | |
133 // Return the enclosed std::vector if this is an array, or an empty vector otherwise. | |
134 const array &array_items() const; | |
135 // Return the enclosed std::map if this is an object, or an empty map otherwise. | |
136 const object &object_items() const; | |
137 | |
138 // Return a reference to arr[i] if this is an array, Json() otherwise. | |
139 const Json & operator[](size_t i) const; | |
140 // Return a reference to obj[key] if this is an object, Json() otherwise. | |
141 const Json & operator[](const std::string &key) const; | |
142 | |
143 // Serialize. | |
144 void dump(std::string &out) const; | |
145 std::string dump() const { | |
146 std::string out; | |
147 dump(out); | |
148 return out; | |
149 } | |
150 | |
151 // Parse. If parse fails, return Json() and assign an error message to err. | |
152 static Json parse(const std::string & in, | |
153 std::string & err, | |
154 JsonParse strategy = JsonParse::STANDARD); | |
155 static Json parse(const char * in, | |
156 std::string & err, | |
157 JsonParse strategy = JsonParse::STANDARD) { | |
158 if (in) { | |
159 return parse(std::string(in), err, strategy); | |
160 } else { | |
161 err = "null input"; | |
162 return nullptr; | |
163 } | |
164 } | |
165 // Parse multiple objects, concatenated or separated by whitespace | |
166 static std::vector<Json> parse_multi( | |
167 const std::string & in, | |
168 std::string & err, | |
169 JsonParse strategy = JsonParse::STANDARD); | |
170 | |
171 bool operator== (const Json &rhs) const; | |
172 bool operator< (const Json &rhs) const; | |
173 bool operator!= (const Json &rhs) const { return !(*this == rhs); } | |
174 bool operator<= (const Json &rhs) const { return !(rhs < *this); } | |
175 bool operator> (const Json &rhs) const { return (rhs < *this); } | |
176 bool operator>= (const Json &rhs) const { return !(*this < rhs); } | |
177 | |
178 /* has_shape(types, err) | |
179 * | |
180 * Return true if this is a JSON object and, for each item in types, has a field of | |
181 * the given type. If not, return false and set err to a descriptive message. | |
182 */ | |
183 typedef std::initializer_list<std::pair<std::string, Type>> shape; | |
184 bool has_shape(const shape & types, std::string & err) const; | |
185 | |
186 private: | |
187 std::shared_ptr<JsonValue> m_ptr; | |
188 }; | |
189 | |
190 // Internal class hierarchy - JsonValue objects are not exposed to users of this API. | |
191 class JsonValue { | |
192 protected: | |
193 friend class Json; | |
194 friend class JsonInt; | |
195 friend class JsonDouble; | |
196 virtual Json::Type type() const = 0; | |
197 virtual bool equals(const JsonValue * other) const = 0; | |
198 virtual bool less(const JsonValue * other) const = 0; | |
199 virtual void dump(std::string &out) const = 0; | |
200 virtual double number_value() const; | |
201 virtual int int_value() const; | |
202 virtual bool bool_value() const; | |
203 virtual const std::string &string_value() const; | |
204 virtual const Json::array &array_items() const; | |
205 virtual const Json &operator[](size_t i) const; | |
206 virtual const Json::object &object_items() const; | |
207 virtual const Json &operator[](const std::string &key) const; | |
208 virtual ~JsonValue() {} | |
209 }; | |
210 | |
211 } // namespace json11 |