1 /*
2 __ _____ _____ _____
3 __| | __| | | | JSON for Modern C++ (test suite)
4 | | |__ | | | | | | version 3.9.1
5 |_____|_____|_____|_|___| https://github.com/nlohmann/json
6
7 Licensed under the MIT License <http://opensource.org/licenses/MIT>.
8 SPDX-License-Identifier: MIT
9 Copyright (c) 2013-2019 Niels Lohmann <http://nlohmann.me>.
10
11 Permission is hereby granted, free of charge, to any person obtaining a copy
12 of this software and associated documentation files (the "Software"), to deal
13 in the Software without restriction, including without limitation the rights
14 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
15 copies of the Software, and to permit persons to whom the Software is
16 furnished to do so, subject to the following conditions:
17
18 The above copyright notice and this permission notice shall be included in all
19 copies or substantial portions of the Software.
20
21 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
22 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
23 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
24 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
25 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
26 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
27 SOFTWARE.
28 */
29
30 #include "doctest_compatibility.h"
31
32 #include <nlohmann/json.hpp>
33 using nlohmann::json;
34
35 namespace
36 {
37 // helper function to check std::less<json::value_t>
38 // see https://en.cppreference.com/w/cpp/utility/functional/less
39 template <typename A, typename B, typename U = std::less<json::value_t>>
f(A a,B b,U u=U ())40 bool f(A a, B b, U u = U())
41 {
42 return u(a, b);
43 }
44 }
45
46 TEST_CASE("lexicographical comparison operators")
47 {
48 SECTION("types")
49 {
50 std::vector<json::value_t> j_types =
51 {
52 json::value_t::null,
53 json::value_t::boolean,
54 json::value_t::number_integer,
55 json::value_t::number_unsigned,
56 json::value_t::number_float,
57 json::value_t::object,
58 json::value_t::array,
59 json::value_t::string,
60 json::value_t::binary
61 };
62
63 SECTION("comparison: less")
64 {
65 std::vector<std::vector<bool>> expected =
66 {
67 {false, true, true, true, true, true, true, true, true},
68 {false, false, true, true, true, true, true, true, true},
69 {false, false, false, false, false, true, true, true, true},
70 {false, false, false, false, false, true, true, true, true},
71 {false, false, false, false, false, true, true, true, true},
72 {false, false, false, false, false, false, true, true, true},
73 {false, false, false, false, false, false, false, true, true},
74 {false, false, false, false, false, false, false, false, true},
75 {false, false, false, false, false, false, false, false, false}
76 };
77
78 for (size_t i = 0; i < j_types.size(); ++i)
79 {
80 for (size_t j = 0; j < j_types.size(); ++j)
81 {
82 CAPTURE(i)
83 CAPTURE(j)
84 // check precomputed values
85 CHECK(operator<(j_types[i], j_types[j]) == expected[i][j]);
86 CHECK(f(j_types[i], j_types[j]) == expected[i][j]);
87 }
88 }
89 }
90 }
91
92 SECTION("values")
93 {
94 json j_values =
95 {
96 nullptr, nullptr,
97 -17, 42,
98 8u, 13u,
99 3.14159, 23.42,
100 "foo", "bar",
101 true, false,
102 {1, 2, 3}, {"one", "two", "three"},
103 {{"first", 1}, {"second", 2}}, {{"a", "A"}, {"b", {"B"}}},
104 json::binary({1, 2, 3}), json::binary({1, 2, 4})
105 };
106
107 SECTION("comparison: equal")
108 {
109 std::vector<std::vector<bool>> expected =
110 {
111 {true, true, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false},
112 {true, true, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false},
113 {false, false, true, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false},
114 {false, false, false, true, false, false, false, false, false, false, false, false, false, false, false, false, false, false},
115 {false, false, false, false, true, false, false, false, false, false, false, false, false, false, false, false, false, false},
116 {false, false, false, false, false, true, false, false, false, false, false, false, false, false, false, false, false, false},
117 {false, false, false, false, false, false, true, false, false, false, false, false, false, false, false, false, false, false},
118 {false, false, false, false, false, false, false, true, false, false, false, false, false, false, false, false, false, false},
119 {false, false, false, false, false, false, false, false, true, false, false, false, false, false, false, false, false, false},
120 {false, false, false, false, false, false, false, false, false, true, false, false, false, false, false, false, false, false},
121 {false, false, false, false, false, false, false, false, false, false, true, false, false, false, false, false, false, false},
122 {false, false, false, false, false, false, false, false, false, false, false, true, false, false, false, false, false, false},
123 {false, false, false, false, false, false, false, false, false, false, false, false, true, false, false, false, false, false},
124 {false, false, false, false, false, false, false, false, false, false, false, false, false, true, false, false, false, false},
125 {false, false, false, false, false, false, false, false, false, false, false, false, false, false, true, false, false, false},
126 {false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, true, false, false},
127 {false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, true, false},
128 {false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, true}
129 };
130
131 for (size_t i = 0; i < j_values.size(); ++i)
132 {
133 for (size_t j = 0; j < j_values.size(); ++j)
134 {
135 CAPTURE(i)
136 CAPTURE(j)
137 CAPTURE(j_values[i])
138 CAPTURE(j_values[j])
139 // check precomputed values
140 CHECK( (j_values[i] == j_values[j]) == expected[i][j] );
141 }
142 }
143
144 // comparison with discarded elements
145 json j_discarded(json::value_t::discarded);
146 for (size_t i = 0; i < j_values.size(); ++i)
147 {
148 CHECK( (j_values[i] == j_discarded) == false);
149 CHECK( (j_discarded == j_values[i]) == false);
150 CHECK( (j_discarded == j_discarded) == false);
151 }
152
153 // compare with null pointer
154 json j_null;
155 CHECK(j_null == nullptr);
156 CHECK(nullptr == j_null);
157 }
158
159 SECTION("comparison: not equal")
160 {
161 for (size_t i = 0; i < j_values.size(); ++i)
162 {
163 for (size_t j = 0; j < j_values.size(); ++j)
164 {
165 CAPTURE(i)
166 CAPTURE(j)
167 // check definition
168 CHECK( (j_values[i] != j_values[j]) == !(j_values[i] == j_values[j]) );
169 }
170 }
171
172 // compare with null pointer
173 json j_null;
174 CHECK( (j_null != nullptr) == false);
175 CHECK( (nullptr != j_null) == false);
176 CHECK( (j_null != nullptr) == !(j_null == nullptr));
177 CHECK( (nullptr != j_null) == !(nullptr == j_null));
178 }
179
180 SECTION("comparison: less")
181 {
182 std::vector<std::vector<bool>> expected =
183 {
184 {false, false, true, true, true, true, true, true, true, true, true, true, true, true, true, true, true, true},
185 {false, false, true, true, true, true, true, true, true, true, true, true, true, true, true, true, true, true},
186 {false, false, false, true, true, true, true, true, true, true, false, false, true, true, true, true, true, true},
187 {false, false, false, false, false, false, false, false, true, true, false, false, true, true, true, true, true, true},
188 {false, false, false, true, false, true, false, true, true, true, false, false, true, true, true, true, true, true},
189 {false, false, false, true, false, false, false, true, true, true, false, false, true, true, true, true, true, true},
190 {false, false, false, true, true, true, false, true, true, true, false, false, true, true, true, true, true, true},
191 {false, false, false, true, false, false, false, false, true, true, false, false, true, true, true, true, true, true},
192 {false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, true, true},
193 {false, false, false, false, false, false, false, false, true, false, false, false, false, false, false, false, true, true},
194 {false, false, true, true, true, true, true, true, true, true, false, false, true, true, true, true, true, true},
195 {false, false, true, true, true, true, true, true, true, true, true, false, true, true, true, true, true, true},
196 {false, false, false, false, false, false, false, false, true, true, false, false, false, true, false, false, true, true},
197 {false, false, false, false, false, false, false, false, true, true, false, false, false, false, false, false, true, true},
198 {false, false, false, false, false, false, false, false, true, true, false, false, true, true, false, false, true, true},
199 {false, false, false, false, false, false, false, false, true, true, false, false, true, true, true, false, true, true},
200 {false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, true},
201 {false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false}
202 };
203
204 for (size_t i = 0; i < j_values.size(); ++i)
205 {
206 for (size_t j = 0; j < j_values.size(); ++j)
207 {
208 CAPTURE(i)
209 CAPTURE(j)
210 CAPTURE(j_values[i])
211 CAPTURE(j_values[j])
212 // check precomputed values
213 CHECK( (j_values[i] < j_values[j]) == expected[i][j] );
214 }
215 }
216
217 // comparison with discarded elements
218 json j_discarded(json::value_t::discarded);
219 for (size_t i = 0; i < j_values.size(); ++i)
220 {
221 CAPTURE(i)
222 CHECK( (j_values[i] < j_discarded) == false);
223 CHECK( (j_discarded < j_values[i]) == false);
224 CHECK( (j_discarded < j_discarded) == false);
225 }
226 }
227
228 SECTION("comparison: less than or equal equal")
229 {
230 for (size_t i = 0; i < j_values.size(); ++i)
231 {
232 for (size_t j = 0; j < j_values.size(); ++j)
233 {
234 CAPTURE(i)
235 CAPTURE(j)
236 // check definition
237 CHECK( (j_values[i] <= j_values[j]) == !(j_values[j] < j_values[i]) );
238 }
239 }
240 }
241
242 SECTION("comparison: greater than")
243 {
244 for (size_t i = 0; i < j_values.size(); ++i)
245 {
246 for (size_t j = 0; j < j_values.size(); ++j)
247 {
248 CAPTURE(i)
249 CAPTURE(j)
250 // check definition
251 CHECK( (j_values[i] > j_values[j]) == (j_values[j] < j_values[i]) );
252 }
253 }
254 }
255
256 SECTION("comparison: greater than or equal")
257 {
258 for (size_t i = 0; i < j_values.size(); ++i)
259 {
260 for (size_t j = 0; j < j_values.size(); ++j)
261 {
262 CAPTURE(i)
263 CAPTURE(j)
264 // check definition
265 CHECK( (j_values[i] >= j_values[j]) == !(j_values[i] < j_values[j]) );
266 }
267 }
268 }
269 }
270 }
271