1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2018 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkData.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRefCnt.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkScalar.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkStream.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkString.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkArenaAlloc.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/utils/SkJSON.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
16*c8dee2aaSAndroid Build Coastguard Worker
17*c8dee2aaSAndroid Build Coastguard Worker #include <cstring>
18*c8dee2aaSAndroid Build Coastguard Worker #include <string_view>
19*c8dee2aaSAndroid Build Coastguard Worker
20*c8dee2aaSAndroid Build Coastguard Worker using namespace skjson;
21*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(JSON_Parse,reporter)22*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(JSON_Parse, reporter) {
23*c8dee2aaSAndroid Build Coastguard Worker static constexpr struct {
24*c8dee2aaSAndroid Build Coastguard Worker const char* in;
25*c8dee2aaSAndroid Build Coastguard Worker const char* out;
26*c8dee2aaSAndroid Build Coastguard Worker } g_tests[] = {
27*c8dee2aaSAndroid Build Coastguard Worker { "" , nullptr },
28*c8dee2aaSAndroid Build Coastguard Worker { "[" , nullptr },
29*c8dee2aaSAndroid Build Coastguard Worker { "]" , nullptr },
30*c8dee2aaSAndroid Build Coastguard Worker { "[[]" , nullptr },
31*c8dee2aaSAndroid Build Coastguard Worker { "[]]" , nullptr },
32*c8dee2aaSAndroid Build Coastguard Worker { "[]f" , nullptr },
33*c8dee2aaSAndroid Build Coastguard Worker { "{" , nullptr },
34*c8dee2aaSAndroid Build Coastguard Worker { "}" , nullptr },
35*c8dee2aaSAndroid Build Coastguard Worker { "{{}" , nullptr },
36*c8dee2aaSAndroid Build Coastguard Worker { "{}}" , nullptr },
37*c8dee2aaSAndroid Build Coastguard Worker { "{}f" , nullptr },
38*c8dee2aaSAndroid Build Coastguard Worker { "{]" , nullptr },
39*c8dee2aaSAndroid Build Coastguard Worker { "[}" , nullptr },
40*c8dee2aaSAndroid Build Coastguard Worker { "{\"}" , nullptr },
41*c8dee2aaSAndroid Build Coastguard Worker { "[\"]" , nullptr },
42*c8dee2aaSAndroid Build Coastguard Worker { "1" , nullptr },
43*c8dee2aaSAndroid Build Coastguard Worker { "true" , nullptr },
44*c8dee2aaSAndroid Build Coastguard Worker { "false", nullptr },
45*c8dee2aaSAndroid Build Coastguard Worker { "null" , nullptr },
46*c8dee2aaSAndroid Build Coastguard Worker
47*c8dee2aaSAndroid Build Coastguard Worker { "[nulll]" , nullptr },
48*c8dee2aaSAndroid Build Coastguard Worker { "[false2]", nullptr },
49*c8dee2aaSAndroid Build Coastguard Worker { "[true:]" , nullptr },
50*c8dee2aaSAndroid Build Coastguard Worker
51*c8dee2aaSAndroid Build Coastguard Worker { "[1 2]" , nullptr },
52*c8dee2aaSAndroid Build Coastguard Worker { "[1,,2]" , nullptr },
53*c8dee2aaSAndroid Build Coastguard Worker { "[1,2,]" , nullptr },
54*c8dee2aaSAndroid Build Coastguard Worker { "[,1,2]" , nullptr },
55*c8dee2aaSAndroid Build Coastguard Worker
56*c8dee2aaSAndroid Build Coastguard Worker { "[ \"foo" , nullptr },
57*c8dee2aaSAndroid Build Coastguard Worker { "[ \"fo\0o\" ]" , nullptr },
58*c8dee2aaSAndroid Build Coastguard Worker
59*c8dee2aaSAndroid Build Coastguard Worker { "{\"\":{}" , nullptr },
60*c8dee2aaSAndroid Build Coastguard Worker { "{ null }" , nullptr },
61*c8dee2aaSAndroid Build Coastguard Worker { "{ \"k\" : }" , nullptr },
62*c8dee2aaSAndroid Build Coastguard Worker { "{ : null }" , nullptr },
63*c8dee2aaSAndroid Build Coastguard Worker { "{ \"k\" : : null }" , nullptr },
64*c8dee2aaSAndroid Build Coastguard Worker { "{ \"k\" : null , }" , nullptr },
65*c8dee2aaSAndroid Build Coastguard Worker { "{ \"k\" : null \"k\" : 1 }", nullptr },
66*c8dee2aaSAndroid Build Coastguard Worker
67*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\)zzz" , nullptr},
68*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\])zzz" , nullptr},
69*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\"])zzz" , nullptr},
70*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\z"])zzz" , nullptr},
71*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\u"])zzz" , nullptr},
72*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\u0"])zzz" , nullptr},
73*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\u00"])zzz" , nullptr},
74*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\u000"])zzz", nullptr},
75*c8dee2aaSAndroid Build Coastguard Worker
76*c8dee2aaSAndroid Build Coastguard Worker { "[]" , "[]" },
77*c8dee2aaSAndroid Build Coastguard Worker { " \n\r\t [ \n\r\t ] \n\r\t " , "[]" },
78*c8dee2aaSAndroid Build Coastguard Worker { "[[]]" , "[[]]" },
79*c8dee2aaSAndroid Build Coastguard Worker { "[ null ]" , "[null]" },
80*c8dee2aaSAndroid Build Coastguard Worker { "[ true ]" , "[true]" },
81*c8dee2aaSAndroid Build Coastguard Worker { "[ false ]" , "[false]" },
82*c8dee2aaSAndroid Build Coastguard Worker { "[ 0 ]" , "[0]" },
83*c8dee2aaSAndroid Build Coastguard Worker { "[ 1 ]" , "[1]" },
84*c8dee2aaSAndroid Build Coastguard Worker { "[ 1.248 ]" , "[1.248]" },
85*c8dee2aaSAndroid Build Coastguard Worker { "[ \"\" ]" , "[\"\"]" },
86*c8dee2aaSAndroid Build Coastguard Worker { "[ \"foo{bar}baz\" ]" , "[\"foo{bar}baz\"]" },
87*c8dee2aaSAndroid Build Coastguard Worker { "[ \" f o o \" ]" , "[\" f o o \"]" },
88*c8dee2aaSAndroid Build Coastguard Worker { "[ \"123456\" ]" , "[\"123456\"]" },
89*c8dee2aaSAndroid Build Coastguard Worker { "[ \"1234567\" ]" , "[\"1234567\"]" },
90*c8dee2aaSAndroid Build Coastguard Worker { "[ \"12345678\" ]" , "[\"12345678\"]" },
91*c8dee2aaSAndroid Build Coastguard Worker { "[ \"123456789\" ]" , "[\"123456789\"]" },
92*c8dee2aaSAndroid Build Coastguard Worker { "[ null , true, false,0,12.8 ]", "[null,true,false,0,12.8]" },
93*c8dee2aaSAndroid Build Coastguard Worker
94*c8dee2aaSAndroid Build Coastguard Worker { "{}" , "{}" },
95*c8dee2aaSAndroid Build Coastguard Worker { " \n\r\t { \n\r\t } \n\r\t " , "{}" },
96*c8dee2aaSAndroid Build Coastguard Worker { "{ \"k\" : null }" , "{\"k\":null}" },
97*c8dee2aaSAndroid Build Coastguard Worker { "{ \"foo{\" : \"bar}baz\" }" , "{\"foo{\":\"bar}baz\"}" },
98*c8dee2aaSAndroid Build Coastguard Worker { "{ \"k1\" : null, \"k2 \":0 }", "{\"k1\":null,\"k2 \":0}" },
99*c8dee2aaSAndroid Build Coastguard Worker { "{ \"k1\" : null, \"k1\":0 }" , "{\"k1\":null,\"k1\":0}" },
100*c8dee2aaSAndroid Build Coastguard Worker
101*c8dee2aaSAndroid Build Coastguard Worker { "{ \"k1\" : null, \n\
102*c8dee2aaSAndroid Build Coastguard Worker \"k2\" : 0, \n\
103*c8dee2aaSAndroid Build Coastguard Worker \"k3\" : [ \n\
104*c8dee2aaSAndroid Build Coastguard Worker true, \r\n\
105*c8dee2aaSAndroid Build Coastguard Worker { \"kk1\" : \"foo\" , \n\
106*c8dee2aaSAndroid Build Coastguard Worker \"kk2\" : \"bar\" , \n\
107*c8dee2aaSAndroid Build Coastguard Worker \"kk3\" : 1.28 , \n\
108*c8dee2aaSAndroid Build Coastguard Worker \"kk4\" : [ 42 ] \n\
109*c8dee2aaSAndroid Build Coastguard Worker } , \n\
110*c8dee2aaSAndroid Build Coastguard Worker \"boo\" , \n\
111*c8dee2aaSAndroid Build Coastguard Worker null \n\
112*c8dee2aaSAndroid Build Coastguard Worker ] \n\
113*c8dee2aaSAndroid Build Coastguard Worker }",
114*c8dee2aaSAndroid Build Coastguard Worker "{\"k1\":null,\"k2\":0,\"k3\":[true,"
115*c8dee2aaSAndroid Build Coastguard Worker "{\"kk1\":\"foo\",\"kk2\":\"bar\",\"kk3\":1.28,\"kk4\":[42]},\"boo\",null]}" },
116*c8dee2aaSAndroid Build Coastguard Worker
117*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\""])zzz" , "[\"\"\"]"},
118*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\\"])zzz" , "[\"\\\"]"},
119*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\/"])zzz" , "[\"/\"]" },
120*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\b"])zzz" , "[\"\b\"]"},
121*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\f"])zzz" , "[\"\f\"]"},
122*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\n"])zzz" , "[\"\n\"]"},
123*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\r"])zzz" , "[\"\r\"]"},
124*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\t"])zzz" , "[\"\t\"]"},
125*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["\u1234"])zzz", "[\"\u1234\"]"},
126*c8dee2aaSAndroid Build Coastguard Worker
127*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["foo\"bar"])zzz" , "[\"foo\"bar\"]"},
128*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["foo\\bar"])zzz" , "[\"foo\\bar\"]"},
129*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["foo\/bar"])zzz" , "[\"foo/bar\"]" },
130*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["foo\bbar"])zzz" , "[\"foo\bbar\"]"},
131*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["foo\fbar"])zzz" , "[\"foo\fbar\"]"},
132*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["foo\nbar"])zzz" , "[\"foo\nbar\"]"},
133*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["foo\rbar"])zzz" , "[\"foo\rbar\"]"},
134*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["foo\tbar"])zzz" , "[\"foo\tbar\"]"},
135*c8dee2aaSAndroid Build Coastguard Worker {R"zzz(["foo\u1234bar"])zzz", "[\"foo\u1234bar\"]"},
136*c8dee2aaSAndroid Build Coastguard Worker };
137*c8dee2aaSAndroid Build Coastguard Worker
138*c8dee2aaSAndroid Build Coastguard Worker for (const auto& tst : g_tests) {
139*c8dee2aaSAndroid Build Coastguard Worker DOM dom(tst.in, strlen(tst.in));
140*c8dee2aaSAndroid Build Coastguard Worker const auto success = !dom.root().is<NullValue>();
141*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, success == (tst.out != nullptr));
142*c8dee2aaSAndroid Build Coastguard Worker if (!success) continue;
143*c8dee2aaSAndroid Build Coastguard Worker
144*c8dee2aaSAndroid Build Coastguard Worker SkDynamicMemoryWStream str;
145*c8dee2aaSAndroid Build Coastguard Worker dom.write(&str);
146*c8dee2aaSAndroid Build Coastguard Worker str.write8('\0');
147*c8dee2aaSAndroid Build Coastguard Worker
148*c8dee2aaSAndroid Build Coastguard Worker auto data = str.detachAsData();
149*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !strcmp(tst.out, static_cast<const char*>(data->data())));
150*c8dee2aaSAndroid Build Coastguard Worker }
151*c8dee2aaSAndroid Build Coastguard Worker
152*c8dee2aaSAndroid Build Coastguard Worker }
153*c8dee2aaSAndroid Build Coastguard Worker
154*c8dee2aaSAndroid Build Coastguard Worker template <typename T, typename VT>
check_primitive(skiatest::Reporter * reporter,const Value & v,T pv,bool is_type)155*c8dee2aaSAndroid Build Coastguard Worker static void check_primitive(skiatest::Reporter* reporter, const Value& v, T pv,
156*c8dee2aaSAndroid Build Coastguard Worker bool is_type) {
157*c8dee2aaSAndroid Build Coastguard Worker
158*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, v.is<VT>() == is_type);
159*c8dee2aaSAndroid Build Coastguard Worker const VT* cast_t = v;
160*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, (cast_t != nullptr) == is_type);
161*c8dee2aaSAndroid Build Coastguard Worker
162*c8dee2aaSAndroid Build Coastguard Worker if (is_type) {
163*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, &v.as<VT>() == cast_t);
164*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, *v.as<VT>() == pv);
165*c8dee2aaSAndroid Build Coastguard Worker }
166*c8dee2aaSAndroid Build Coastguard Worker }
167*c8dee2aaSAndroid Build Coastguard Worker
168*c8dee2aaSAndroid Build Coastguard Worker template <typename T>
check_vector(skiatest::Reporter * reporter,const Value & v,size_t expected_size,bool is_vector)169*c8dee2aaSAndroid Build Coastguard Worker static void check_vector(skiatest::Reporter* reporter, const Value& v, size_t expected_size,
170*c8dee2aaSAndroid Build Coastguard Worker bool is_vector) {
171*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, v.is<T>() == is_vector);
172*c8dee2aaSAndroid Build Coastguard Worker const T* cast_t = v;
173*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, (cast_t != nullptr) == is_vector);
174*c8dee2aaSAndroid Build Coastguard Worker
175*c8dee2aaSAndroid Build Coastguard Worker if (is_vector) {
176*c8dee2aaSAndroid Build Coastguard Worker const auto& vec = v.as<T>();
177*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, &vec == cast_t);
178*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, vec.size() == expected_size);
179*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, vec.begin() != nullptr);
180*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, vec.end() == vec.begin() + expected_size);
181*c8dee2aaSAndroid Build Coastguard Worker }
182*c8dee2aaSAndroid Build Coastguard Worker }
183*c8dee2aaSAndroid Build Coastguard Worker
check_string(skiatest::Reporter * reporter,const Value & v,const char * s)184*c8dee2aaSAndroid Build Coastguard Worker static void check_string(skiatest::Reporter* reporter, const Value& v, const char* s) {
185*c8dee2aaSAndroid Build Coastguard Worker check_vector<StringValue>(reporter, v, s ? strlen(s) : 0, !!s);
186*c8dee2aaSAndroid Build Coastguard Worker if (s) {
187*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, v.as<StringValue>().str() == s);
188*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !strcmp(v.as<StringValue>().begin(), s));
189*c8dee2aaSAndroid Build Coastguard Worker }
190*c8dee2aaSAndroid Build Coastguard Worker }
191*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(JSON_DOM_visit,reporter)192*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(JSON_DOM_visit, reporter) {
193*c8dee2aaSAndroid Build Coastguard Worker static constexpr char json[] = "{ \n\
194*c8dee2aaSAndroid Build Coastguard Worker \"k1\": null, \n\
195*c8dee2aaSAndroid Build Coastguard Worker \"k2\": false, \n\
196*c8dee2aaSAndroid Build Coastguard Worker \"k3\": true, \n\
197*c8dee2aaSAndroid Build Coastguard Worker \"k4\": 42, \n\
198*c8dee2aaSAndroid Build Coastguard Worker \"k5\": .75, \n\
199*c8dee2aaSAndroid Build Coastguard Worker \"k6\": \"foo\", \n\
200*c8dee2aaSAndroid Build Coastguard Worker \"k6b\": \"this string is long\", \n\
201*c8dee2aaSAndroid Build Coastguard Worker \"k7\": [ 1, true, \"bar\" ], \n\
202*c8dee2aaSAndroid Build Coastguard Worker \"k8\": { \"kk1\": 2, \"kk2\": false, \"kk1\": \"baz\" } \n\
203*c8dee2aaSAndroid Build Coastguard Worker }";
204*c8dee2aaSAndroid Build Coastguard Worker
205*c8dee2aaSAndroid Build Coastguard Worker DOM dom(json, strlen(json));
206*c8dee2aaSAndroid Build Coastguard Worker
207*c8dee2aaSAndroid Build Coastguard Worker const auto& jroot = dom.root().as<ObjectValue>();
208*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, jroot.is<ObjectValue>());
209*c8dee2aaSAndroid Build Coastguard Worker
210*c8dee2aaSAndroid Build Coastguard Worker {
211*c8dee2aaSAndroid Build Coastguard Worker const auto& v = jroot["k1"];
212*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, v.is<NullValue>());
213*c8dee2aaSAndroid Build Coastguard Worker
214*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v, false, false);
215*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, v, 0, false);
216*c8dee2aaSAndroid Build Coastguard Worker
217*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, v, nullptr);
218*c8dee2aaSAndroid Build Coastguard Worker check_vector<ArrayValue >(reporter, v, 0, false);
219*c8dee2aaSAndroid Build Coastguard Worker check_vector<ObjectValue>(reporter, v, 0, false);
220*c8dee2aaSAndroid Build Coastguard Worker }
221*c8dee2aaSAndroid Build Coastguard Worker
222*c8dee2aaSAndroid Build Coastguard Worker {
223*c8dee2aaSAndroid Build Coastguard Worker const auto& v = jroot["k2"];
224*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !v.is<NullValue>());
225*c8dee2aaSAndroid Build Coastguard Worker
226*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v, false, true);
227*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, v, 0, false);
228*c8dee2aaSAndroid Build Coastguard Worker
229*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, v, nullptr);
230*c8dee2aaSAndroid Build Coastguard Worker check_vector<ArrayValue >(reporter, v, 0, false);
231*c8dee2aaSAndroid Build Coastguard Worker check_vector<ObjectValue>(reporter, v, 0, false);
232*c8dee2aaSAndroid Build Coastguard Worker }
233*c8dee2aaSAndroid Build Coastguard Worker
234*c8dee2aaSAndroid Build Coastguard Worker {
235*c8dee2aaSAndroid Build Coastguard Worker const auto& v = jroot["k3"];
236*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !v.is<NullValue>());
237*c8dee2aaSAndroid Build Coastguard Worker
238*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v, true, true);
239*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, v, 0, false);
240*c8dee2aaSAndroid Build Coastguard Worker
241*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, v, nullptr);
242*c8dee2aaSAndroid Build Coastguard Worker check_vector<ArrayValue >(reporter, v, 0, false);
243*c8dee2aaSAndroid Build Coastguard Worker check_vector<ObjectValue>(reporter, v, 0, false);
244*c8dee2aaSAndroid Build Coastguard Worker }
245*c8dee2aaSAndroid Build Coastguard Worker
246*c8dee2aaSAndroid Build Coastguard Worker {
247*c8dee2aaSAndroid Build Coastguard Worker const auto& v = jroot["k4"];
248*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !v.is<NullValue>());
249*c8dee2aaSAndroid Build Coastguard Worker
250*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v, false, false);
251*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, v, 42, true);
252*c8dee2aaSAndroid Build Coastguard Worker
253*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, v, nullptr);
254*c8dee2aaSAndroid Build Coastguard Worker check_vector<ArrayValue >(reporter, v, 0, false);
255*c8dee2aaSAndroid Build Coastguard Worker check_vector<ObjectValue>(reporter, v, 0, false);
256*c8dee2aaSAndroid Build Coastguard Worker }
257*c8dee2aaSAndroid Build Coastguard Worker
258*c8dee2aaSAndroid Build Coastguard Worker {
259*c8dee2aaSAndroid Build Coastguard Worker const auto& v = jroot["k5"];
260*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !v.is<NullValue>());
261*c8dee2aaSAndroid Build Coastguard Worker
262*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v, false, false);
263*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, v, .75f, true);
264*c8dee2aaSAndroid Build Coastguard Worker
265*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, v, nullptr);
266*c8dee2aaSAndroid Build Coastguard Worker check_vector<ArrayValue >(reporter, v, 0, false);
267*c8dee2aaSAndroid Build Coastguard Worker check_vector<ObjectValue>(reporter, v, 0, false);
268*c8dee2aaSAndroid Build Coastguard Worker }
269*c8dee2aaSAndroid Build Coastguard Worker
270*c8dee2aaSAndroid Build Coastguard Worker {
271*c8dee2aaSAndroid Build Coastguard Worker const auto& v = jroot["k6"];
272*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !v.is<NullValue>());
273*c8dee2aaSAndroid Build Coastguard Worker
274*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v, false, false);
275*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, v, 0, false);
276*c8dee2aaSAndroid Build Coastguard Worker
277*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, v, "foo");
278*c8dee2aaSAndroid Build Coastguard Worker check_vector<ArrayValue >(reporter, v, 0, false);
279*c8dee2aaSAndroid Build Coastguard Worker check_vector<ObjectValue>(reporter, v, 0, false);
280*c8dee2aaSAndroid Build Coastguard Worker }
281*c8dee2aaSAndroid Build Coastguard Worker
282*c8dee2aaSAndroid Build Coastguard Worker {
283*c8dee2aaSAndroid Build Coastguard Worker const auto& v = jroot["k6b"];
284*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !v.is<NullValue>());
285*c8dee2aaSAndroid Build Coastguard Worker
286*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v, false, false);
287*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, v, 0, false);
288*c8dee2aaSAndroid Build Coastguard Worker
289*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, v, "this string is long");
290*c8dee2aaSAndroid Build Coastguard Worker check_vector<ArrayValue >(reporter, v, 0, false);
291*c8dee2aaSAndroid Build Coastguard Worker check_vector<ObjectValue>(reporter, v, 0, false);
292*c8dee2aaSAndroid Build Coastguard Worker }
293*c8dee2aaSAndroid Build Coastguard Worker
294*c8dee2aaSAndroid Build Coastguard Worker {
295*c8dee2aaSAndroid Build Coastguard Worker const auto& v = jroot["k7"];
296*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !v.is<NullValue>());
297*c8dee2aaSAndroid Build Coastguard Worker
298*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v, false, false);
299*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, v, 0, false);
300*c8dee2aaSAndroid Build Coastguard Worker
301*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, v, nullptr);
302*c8dee2aaSAndroid Build Coastguard Worker check_vector<ObjectValue>(reporter, v, 0, false);
303*c8dee2aaSAndroid Build Coastguard Worker
304*c8dee2aaSAndroid Build Coastguard Worker check_vector<ArrayValue >(reporter, v, 3, true);
305*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, v.as<ArrayValue>()[0], 1, true);
306*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v.as<ArrayValue>()[1], true, true);
307*c8dee2aaSAndroid Build Coastguard Worker check_vector<StringValue>(reporter, v.as<ArrayValue>()[2], 3, true);
308*c8dee2aaSAndroid Build Coastguard Worker }
309*c8dee2aaSAndroid Build Coastguard Worker
310*c8dee2aaSAndroid Build Coastguard Worker {
311*c8dee2aaSAndroid Build Coastguard Worker const auto& v = jroot["k8"];
312*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !v.is<NullValue>());
313*c8dee2aaSAndroid Build Coastguard Worker
314*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v, false, false);
315*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, v, 0, false);
316*c8dee2aaSAndroid Build Coastguard Worker
317*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, v, nullptr);
318*c8dee2aaSAndroid Build Coastguard Worker check_vector<ArrayValue >(reporter, v, 0, false);
319*c8dee2aaSAndroid Build Coastguard Worker
320*c8dee2aaSAndroid Build Coastguard Worker check_vector<ObjectValue>(reporter, v, 3, true);
321*c8dee2aaSAndroid Build Coastguard Worker
322*c8dee2aaSAndroid Build Coastguard Worker const auto& m0 = v.as<ObjectValue>().begin()[0];
323*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, m0.fKey, "kk1");
324*c8dee2aaSAndroid Build Coastguard Worker check_primitive<float, NumberValue>(reporter, m0.fValue, 2, true);
325*c8dee2aaSAndroid Build Coastguard Worker
326*c8dee2aaSAndroid Build Coastguard Worker const auto& m1 = v.as<ObjectValue>().begin()[1];
327*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, m1.fKey, "kk2");
328*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, m1.fValue, false, true);
329*c8dee2aaSAndroid Build Coastguard Worker
330*c8dee2aaSAndroid Build Coastguard Worker const auto& m2 = v.as<ObjectValue>().begin()[2];
331*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, m2.fKey, "kk1");
332*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, m2.fValue, "baz");
333*c8dee2aaSAndroid Build Coastguard Worker
334*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, v.as<ObjectValue>()[""].is<NullValue>());
335*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, v.as<ObjectValue>()["nosuchkey"].is<NullValue>());
336*c8dee2aaSAndroid Build Coastguard Worker check_string(reporter, v.as<ObjectValue>()["kk1"], "baz");
337*c8dee2aaSAndroid Build Coastguard Worker check_primitive<bool, BoolValue>(reporter, v.as<ObjectValue>()["kk2"], false, true);
338*c8dee2aaSAndroid Build Coastguard Worker }
339*c8dee2aaSAndroid Build Coastguard Worker }
340*c8dee2aaSAndroid Build Coastguard Worker
341*c8dee2aaSAndroid Build Coastguard Worker template <typename T>
check_value(skiatest::Reporter * reporter,const Value & v,const char * expected_string)342*c8dee2aaSAndroid Build Coastguard Worker void check_value(skiatest::Reporter* reporter, const Value& v, const char* expected_string) {
343*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, v.is<T>());
344*c8dee2aaSAndroid Build Coastguard Worker
345*c8dee2aaSAndroid Build Coastguard Worker const T* cast_t = v;
346*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, cast_t == &v.as<T>());
347*c8dee2aaSAndroid Build Coastguard Worker
348*c8dee2aaSAndroid Build Coastguard Worker const auto vstr = v.toString();
349*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, 0 == strcmp(expected_string, vstr.c_str()));
350*c8dee2aaSAndroid Build Coastguard Worker }
351*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(JSON_DOM_build,reporter)352*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(JSON_DOM_build, reporter) {
353*c8dee2aaSAndroid Build Coastguard Worker SkArenaAlloc alloc(4096);
354*c8dee2aaSAndroid Build Coastguard Worker
355*c8dee2aaSAndroid Build Coastguard Worker const auto v0 = NullValue();
356*c8dee2aaSAndroid Build Coastguard Worker check_value<NullValue>(reporter, v0, "null");
357*c8dee2aaSAndroid Build Coastguard Worker
358*c8dee2aaSAndroid Build Coastguard Worker const auto v1 = BoolValue(true);
359*c8dee2aaSAndroid Build Coastguard Worker check_value<BoolValue>(reporter, v1, "true");
360*c8dee2aaSAndroid Build Coastguard Worker
361*c8dee2aaSAndroid Build Coastguard Worker const auto v2 = BoolValue(false);
362*c8dee2aaSAndroid Build Coastguard Worker check_value<BoolValue>(reporter, v2, "false");
363*c8dee2aaSAndroid Build Coastguard Worker
364*c8dee2aaSAndroid Build Coastguard Worker const auto v3 = NumberValue(0);
365*c8dee2aaSAndroid Build Coastguard Worker check_value<NumberValue>(reporter, v3, "0");
366*c8dee2aaSAndroid Build Coastguard Worker
367*c8dee2aaSAndroid Build Coastguard Worker const auto v4 = NumberValue(42);
368*c8dee2aaSAndroid Build Coastguard Worker check_value<NumberValue>(reporter, v4, "42");
369*c8dee2aaSAndroid Build Coastguard Worker
370*c8dee2aaSAndroid Build Coastguard Worker const auto v5 = NumberValue(42.75f);
371*c8dee2aaSAndroid Build Coastguard Worker check_value<NumberValue>(reporter, v5, "42.75");
372*c8dee2aaSAndroid Build Coastguard Worker
373*c8dee2aaSAndroid Build Coastguard Worker const auto v6 = StringValue(nullptr, 0, alloc);
374*c8dee2aaSAndroid Build Coastguard Worker check_value<StringValue>(reporter, v6, "\"\"");
375*c8dee2aaSAndroid Build Coastguard Worker
376*c8dee2aaSAndroid Build Coastguard Worker const auto v7 = StringValue(" foo ", 5, alloc);
377*c8dee2aaSAndroid Build Coastguard Worker check_value<StringValue>(reporter, v7, "\" foo \"");
378*c8dee2aaSAndroid Build Coastguard Worker
379*c8dee2aaSAndroid Build Coastguard Worker const auto v8 = StringValue(" foo bar baz ", 13, alloc);
380*c8dee2aaSAndroid Build Coastguard Worker check_value<StringValue>(reporter, v8, "\" foo bar baz \"");
381*c8dee2aaSAndroid Build Coastguard Worker
382*c8dee2aaSAndroid Build Coastguard Worker const auto v9 = ArrayValue(nullptr, 0, alloc);
383*c8dee2aaSAndroid Build Coastguard Worker check_value<ArrayValue>(reporter, v9, "[]");
384*c8dee2aaSAndroid Build Coastguard Worker
385*c8dee2aaSAndroid Build Coastguard Worker const Value values0[] = { v0, v3, v9 };
386*c8dee2aaSAndroid Build Coastguard Worker const auto v10 = ArrayValue(values0, std::size(values0), alloc);
387*c8dee2aaSAndroid Build Coastguard Worker check_value<ArrayValue>(reporter, v10, "[null,0,[]]");
388*c8dee2aaSAndroid Build Coastguard Worker
389*c8dee2aaSAndroid Build Coastguard Worker const auto v11 = ObjectValue(nullptr, 0, alloc);
390*c8dee2aaSAndroid Build Coastguard Worker check_value<ObjectValue>(reporter, v11, "{}");
391*c8dee2aaSAndroid Build Coastguard Worker
392*c8dee2aaSAndroid Build Coastguard Worker const Member members0[] = {
393*c8dee2aaSAndroid Build Coastguard Worker { StringValue("key_0", 5, alloc), v1 },
394*c8dee2aaSAndroid Build Coastguard Worker { StringValue("key_1", 5, alloc), v4 },
395*c8dee2aaSAndroid Build Coastguard Worker { StringValue("key_2", 5, alloc), v11 },
396*c8dee2aaSAndroid Build Coastguard Worker };
397*c8dee2aaSAndroid Build Coastguard Worker const auto v12 = ObjectValue(members0, std::size(members0), alloc);
398*c8dee2aaSAndroid Build Coastguard Worker check_value<ObjectValue>(reporter, v12, "{"
399*c8dee2aaSAndroid Build Coastguard Worker "\"key_0\":true,"
400*c8dee2aaSAndroid Build Coastguard Worker "\"key_1\":42,"
401*c8dee2aaSAndroid Build Coastguard Worker "\"key_2\":{}"
402*c8dee2aaSAndroid Build Coastguard Worker "}");
403*c8dee2aaSAndroid Build Coastguard Worker
404*c8dee2aaSAndroid Build Coastguard Worker const Value values1[] = { v2, v6, v12 };
405*c8dee2aaSAndroid Build Coastguard Worker const auto v13 = ArrayValue(values1, std::size(values1), alloc);
406*c8dee2aaSAndroid Build Coastguard Worker check_value<ArrayValue>(reporter, v13, "["
407*c8dee2aaSAndroid Build Coastguard Worker "false,"
408*c8dee2aaSAndroid Build Coastguard Worker "\"\","
409*c8dee2aaSAndroid Build Coastguard Worker "{"
410*c8dee2aaSAndroid Build Coastguard Worker "\"key_0\":true,"
411*c8dee2aaSAndroid Build Coastguard Worker "\"key_1\":42,"
412*c8dee2aaSAndroid Build Coastguard Worker "\"key_2\":{}"
413*c8dee2aaSAndroid Build Coastguard Worker "}"
414*c8dee2aaSAndroid Build Coastguard Worker "]");
415*c8dee2aaSAndroid Build Coastguard Worker
416*c8dee2aaSAndroid Build Coastguard Worker const Member members1[] = {
417*c8dee2aaSAndroid Build Coastguard Worker { StringValue("key_00", 6, alloc), v5 },
418*c8dee2aaSAndroid Build Coastguard Worker { StringValue("key_01", 6, alloc), v7 },
419*c8dee2aaSAndroid Build Coastguard Worker { StringValue("key_02", 6, alloc), v13 },
420*c8dee2aaSAndroid Build Coastguard Worker };
421*c8dee2aaSAndroid Build Coastguard Worker const auto v14 = ObjectValue(members1, std::size(members1), alloc);
422*c8dee2aaSAndroid Build Coastguard Worker check_value<ObjectValue>(reporter, v14, "{"
423*c8dee2aaSAndroid Build Coastguard Worker "\"key_00\":42.75,"
424*c8dee2aaSAndroid Build Coastguard Worker "\"key_01\":\" foo \","
425*c8dee2aaSAndroid Build Coastguard Worker "\"key_02\":["
426*c8dee2aaSAndroid Build Coastguard Worker "false,"
427*c8dee2aaSAndroid Build Coastguard Worker "\"\","
428*c8dee2aaSAndroid Build Coastguard Worker "{"
429*c8dee2aaSAndroid Build Coastguard Worker "\"key_0\":true,"
430*c8dee2aaSAndroid Build Coastguard Worker "\"key_1\":42,"
431*c8dee2aaSAndroid Build Coastguard Worker "\"key_2\":{}"
432*c8dee2aaSAndroid Build Coastguard Worker "}"
433*c8dee2aaSAndroid Build Coastguard Worker "]"
434*c8dee2aaSAndroid Build Coastguard Worker "}");
435*c8dee2aaSAndroid Build Coastguard Worker }
436*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(JSON_ParseNumber,reporter)437*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(JSON_ParseNumber, reporter) {
438*c8dee2aaSAndroid Build Coastguard Worker static constexpr struct {
439*c8dee2aaSAndroid Build Coastguard Worker const char* string;
440*c8dee2aaSAndroid Build Coastguard Worker SkScalar value,
441*c8dee2aaSAndroid Build Coastguard Worker tolerance;
442*c8dee2aaSAndroid Build Coastguard Worker } gTests[] = {
443*c8dee2aaSAndroid Build Coastguard Worker { "0", 0, 0 },
444*c8dee2aaSAndroid Build Coastguard Worker { "1", 1, 0 },
445*c8dee2aaSAndroid Build Coastguard Worker
446*c8dee2aaSAndroid Build Coastguard Worker { "00000000", 0, 0 },
447*c8dee2aaSAndroid Build Coastguard Worker { "00000001", 1, 0 },
448*c8dee2aaSAndroid Build Coastguard Worker
449*c8dee2aaSAndroid Build Coastguard Worker { "0.001", 0.001f, 0 },
450*c8dee2aaSAndroid Build Coastguard Worker { "1.001", 1.001f, 0 },
451*c8dee2aaSAndroid Build Coastguard Worker
452*c8dee2aaSAndroid Build Coastguard Worker { "0.000001" , 0.000001f, 0 },
453*c8dee2aaSAndroid Build Coastguard Worker { "1.000001" , 1.000001f, 0 },
454*c8dee2aaSAndroid Build Coastguard Worker { "1000.000001", 1000.000001f, 0 },
455*c8dee2aaSAndroid Build Coastguard Worker
456*c8dee2aaSAndroid Build Coastguard Worker { "0.0000000001" , 0.0000000001f, 0 },
457*c8dee2aaSAndroid Build Coastguard Worker { "1.0000000001" , 1.0000000001f, 0 },
458*c8dee2aaSAndroid Build Coastguard Worker { "1000.0000000001", 1000.0000000001f, 0 },
459*c8dee2aaSAndroid Build Coastguard Worker
460*c8dee2aaSAndroid Build Coastguard Worker { "20.001111814444444444444444444444444444444444444444444444444444444444444444444444444444444444444444444444444444444444444444473",
461*c8dee2aaSAndroid Build Coastguard Worker 20.001f, 0.001f },
462*c8dee2aaSAndroid Build Coastguard Worker };
463*c8dee2aaSAndroid Build Coastguard Worker
464*c8dee2aaSAndroid Build Coastguard Worker for (const auto& test : gTests) {
465*c8dee2aaSAndroid Build Coastguard Worker const auto json = SkStringPrintf("{ \"key\": %s }", test.string);
466*c8dee2aaSAndroid Build Coastguard Worker const DOM dom(json.c_str(), json.size());
467*c8dee2aaSAndroid Build Coastguard Worker const ObjectValue* jroot = dom.root();
468*c8dee2aaSAndroid Build Coastguard Worker
469*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, jroot);
470*c8dee2aaSAndroid Build Coastguard Worker
471*c8dee2aaSAndroid Build Coastguard Worker const NumberValue* jnumber = (*jroot)["key"];
472*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, jnumber);
473*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, SkScalarNearlyEqual(**jnumber, test.value, test.tolerance));
474*c8dee2aaSAndroid Build Coastguard Worker }
475*c8dee2aaSAndroid Build Coastguard Worker }
476*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(JSON_Lookup,r)477*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(JSON_Lookup, r) {
478*c8dee2aaSAndroid Build Coastguard Worker const char* json = R"({"foo": { "bar": { "baz": 100 }}})";
479*c8dee2aaSAndroid Build Coastguard Worker const DOM dom(json, strlen(json));
480*c8dee2aaSAndroid Build Coastguard Worker const Value& root = dom.root();
481*c8dee2aaSAndroid Build Coastguard Worker
482*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root.is<ObjectValue>());
483*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["foo"].is<ObjectValue>());
484*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["foo"]["bar"].is<ObjectValue>());
485*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["foo"]["bar"]["baz"].is<NumberValue>());
486*c8dee2aaSAndroid Build Coastguard Worker
487*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["foozz"].is<NullValue>());
488*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["foozz"]["barzz"].is<NullValue>());
489*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["foozz"]["barzz"]["bazzz"].is<NullValue>());
490*c8dee2aaSAndroid Build Coastguard Worker }
491*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(JSON_Writable,r)492*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(JSON_Writable, r) {
493*c8dee2aaSAndroid Build Coastguard Worker const char* json = R"({"null": null, "num": 100})";
494*c8dee2aaSAndroid Build Coastguard Worker const DOM dom(json, strlen(json));
495*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, dom.root().is<ObjectValue>());
496*c8dee2aaSAndroid Build Coastguard Worker const ObjectValue& root = dom.root().as<ObjectValue>();
497*c8dee2aaSAndroid Build Coastguard Worker
498*c8dee2aaSAndroid Build Coastguard Worker SkArenaAlloc alloc(4096);
499*c8dee2aaSAndroid Build Coastguard Worker
500*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["null"].is<NullValue>());
501*c8dee2aaSAndroid Build Coastguard Worker Value& w1 = root.writable("null", alloc);
502*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, w1.is<NullValue>());
503*c8dee2aaSAndroid Build Coastguard Worker w1 = NumberValue(42);
504*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["null"].is<NumberValue>());
505*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, *root["null"].as<NumberValue>() == 42);
506*c8dee2aaSAndroid Build Coastguard Worker
507*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["num"].is<NumberValue>());
508*c8dee2aaSAndroid Build Coastguard Worker Value& w2 = root.writable("num", alloc);
509*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, w2.is<NumberValue>());
510*c8dee2aaSAndroid Build Coastguard Worker w2 = StringValue("foo", 3, alloc);
511*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["num"].is<StringValue>());
512*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["num"].as<StringValue>().str() == "foo");
513*c8dee2aaSAndroid Build Coastguard Worker
514*c8dee2aaSAndroid Build Coastguard Worker // new/insert semantics
515*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["new"].is<NullValue>());
516*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root.size() == 2u);
517*c8dee2aaSAndroid Build Coastguard Worker Value& w3 = root.writable("new", alloc);
518*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, w3.is<NullValue>());
519*c8dee2aaSAndroid Build Coastguard Worker w3 = BoolValue(true);
520*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root.size() == 3u);
521*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["new"].is<BoolValue>());
522*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, *root["new"].as<BoolValue>());
523*c8dee2aaSAndroid Build Coastguard Worker
524*c8dee2aaSAndroid Build Coastguard Worker root.writable("newobj", alloc) = ObjectValue(nullptr, 0, alloc);
525*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root.size() == 4u);
526*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["newobj"].is<ObjectValue>());
527*c8dee2aaSAndroid Build Coastguard Worker const ObjectValue& newobj = root["newobj"].as<ObjectValue>();
528*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, newobj.size() == 0u);
529*c8dee2aaSAndroid Build Coastguard Worker
530*c8dee2aaSAndroid Build Coastguard Worker newobj.writable("newprop", alloc) = NumberValue(-1);
531*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, newobj.size() == 1u);
532*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root["newobj"]["newprop"].is<NumberValue>());
533*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, *root["newobj"]["newprop"].as<NumberValue>() == -1);
534*c8dee2aaSAndroid Build Coastguard Worker
535*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, root.toString() ==
536*c8dee2aaSAndroid Build Coastguard Worker SkString(R"({"null":42,"num":"foo","new":true,"newobj":{"newprop":-1}})"));
537*c8dee2aaSAndroid Build Coastguard Worker }
538