Home | History | Annotate | Download | only in unittest
      1 // Tencent is pleased to support the open source community by making RapidJSON available.
      2 //
      3 // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
      4 //
      5 // Licensed under the MIT License (the "License"); you may not use this file except
      6 // in compliance with the License. You may obtain a copy of the License at
      7 //
      8 // http://opensource.org/licenses/MIT
      9 //
     10 // Unless required by applicable law or agreed to in writing, software distributed
     11 // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
     12 // CONDITIONS OF ANY KIND, either express or implied. See the License for the
     13 // specific language governing permissions and limitations under the License.
     14 
     15 #include "unittest.h"
     16 
     17 #include "rapidjson/document.h"
     18 #include "rapidjson/reader.h"
     19 #include "rapidjson/writer.h"
     20 #include "rapidjson/stringbuffer.h"
     21 
     22 using namespace rapidjson;
     23 
     24 TEST(Writer, Compact) {
     25     StringStream s("{ \"hello\" : \"world\", \"t\" : true , \"f\" : false, \"n\": null, \"i\":123, \"pi\": 3.1416, \"a\":[1, 2, 3] } ");
     26     StringBuffer buffer;
     27     Writer<StringBuffer> writer(buffer);
     28     buffer.ShrinkToFit();
     29     Reader reader;
     30     reader.Parse<0>(s, writer);
     31     EXPECT_STREQ("{\"hello\":\"world\",\"t\":true,\"f\":false,\"n\":null,\"i\":123,\"pi\":3.1416,\"a\":[1,2,3]}", buffer.GetString());
     32     EXPECT_EQ(77u, buffer.GetSize());
     33     EXPECT_TRUE(writer.IsComplete());
     34 }
     35 
     36 // json -> parse -> writer -> json
     37 #define TEST_ROUNDTRIP(json) \
     38     { \
     39         StringStream s(json); \
     40         StringBuffer buffer; \
     41         Writer<StringBuffer> writer(buffer); \
     42         Reader reader; \
     43         reader.Parse<kParseFullPrecisionFlag>(s, writer); \
     44         EXPECT_STREQ(json, buffer.GetString()); \
     45         EXPECT_TRUE(writer.IsComplete()); \
     46     }
     47 
     48 TEST(Writer, Root) {
     49     TEST_ROUNDTRIP("null");
     50     TEST_ROUNDTRIP("true");
     51     TEST_ROUNDTRIP("false");
     52     TEST_ROUNDTRIP("0");
     53     TEST_ROUNDTRIP("\"foo\"");
     54     TEST_ROUNDTRIP("[]");
     55     TEST_ROUNDTRIP("{}");
     56 }
     57 
     58 TEST(Writer, Int) {
     59     TEST_ROUNDTRIP("[-1]");
     60     TEST_ROUNDTRIP("[-123]");
     61     TEST_ROUNDTRIP("[-2147483648]");
     62 }
     63 
     64 TEST(Writer, UInt) {
     65     TEST_ROUNDTRIP("[0]");
     66     TEST_ROUNDTRIP("[1]");
     67     TEST_ROUNDTRIP("[123]");
     68     TEST_ROUNDTRIP("[2147483647]");
     69     TEST_ROUNDTRIP("[4294967295]");
     70 }
     71 
     72 TEST(Writer, Int64) {
     73     TEST_ROUNDTRIP("[-1234567890123456789]");
     74     TEST_ROUNDTRIP("[-9223372036854775808]");
     75 }
     76 
     77 TEST(Writer, Uint64) {
     78     TEST_ROUNDTRIP("[1234567890123456789]");
     79     TEST_ROUNDTRIP("[9223372036854775807]");
     80 }
     81 
     82 TEST(Writer, String) {
     83     TEST_ROUNDTRIP("[\"Hello\"]");
     84     TEST_ROUNDTRIP("[\"Hello\\u0000World\"]");
     85     TEST_ROUNDTRIP("[\"\\\"\\\\/\\b\\f\\n\\r\\t\"]");
     86 
     87 #if RAPIDJSON_HAS_STDSTRING
     88     {
     89         StringBuffer buffer;
     90         Writer<StringBuffer> writer(buffer);
     91         writer.String(std::string("Hello\n"));
     92         EXPECT_STREQ("\"Hello\\n\"", buffer.GetString());
     93     }
     94 #endif
     95 }
     96 
     97 TEST(Writer, Double) {
     98     TEST_ROUNDTRIP("[1.2345,1.2345678,0.123456789012,1234567.8]");
     99     TEST_ROUNDTRIP("0.0");
    100     TEST_ROUNDTRIP("-0.0"); // Issue #289
    101     TEST_ROUNDTRIP("1e30");
    102     TEST_ROUNDTRIP("1.0");
    103     TEST_ROUNDTRIP("5e-324"); // Min subnormal positive double
    104     TEST_ROUNDTRIP("2.225073858507201e-308"); // Max subnormal positive double
    105     TEST_ROUNDTRIP("2.2250738585072014e-308"); // Min normal positive double
    106     TEST_ROUNDTRIP("1.7976931348623157e308"); // Max double
    107 
    108 }
    109 
    110 TEST(Writer, Transcode) {
    111     const char json[] = "{\"hello\":\"world\",\"t\":true,\"f\":false,\"n\":null,\"i\":123,\"pi\":3.1416,\"a\":[1,2,3],\"dollar\":\"\x24\",\"cents\":\"\xC2\xA2\",\"euro\":\"\xE2\x82\xAC\",\"gclef\":\"\xF0\x9D\x84\x9E\"}";
    112 
    113     // UTF8 -> UTF16 -> UTF8
    114     {
    115         StringStream s(json);
    116         StringBuffer buffer;
    117         Writer<StringBuffer, UTF16<>, UTF8<> > writer(buffer);
    118         GenericReader<UTF8<>, UTF16<> > reader;
    119         reader.Parse(s, writer);
    120         EXPECT_STREQ(json, buffer.GetString());
    121     }
    122 
    123     // UTF8 -> UTF8 -> ASCII -> UTF8 -> UTF8
    124     {
    125         StringStream s(json);
    126         StringBuffer buffer;
    127         Writer<StringBuffer, UTF8<>, ASCII<> > writer(buffer);
    128         Reader reader;
    129         reader.Parse(s, writer);
    130 
    131         StringBuffer buffer2;
    132         Writer<StringBuffer> writer2(buffer2);
    133         GenericReader<ASCII<>, UTF8<> > reader2;
    134         StringStream s2(buffer.GetString());
    135         reader2.Parse(s2, writer2);
    136 
    137         EXPECT_STREQ(json, buffer2.GetString());
    138     }
    139 }
    140 
    141 #include <sstream>
    142 
    143 class OStreamWrapper {
    144 public:
    145     typedef char Ch;
    146 
    147     OStreamWrapper(std::ostream& os) : os_(os) {}
    148 
    149     Ch Peek() const { assert(false); return '\0'; }
    150     Ch Take() { assert(false); return '\0'; }
    151     size_t Tell() const { return 0; }
    152 
    153     Ch* PutBegin() { assert(false); return 0; }
    154     void Put(Ch c) { os_.put(c); }
    155     void Flush() { os_.flush(); }
    156     size_t PutEnd(Ch*) { assert(false); return 0; }
    157 
    158 private:
    159     OStreamWrapper(const OStreamWrapper&);
    160     OStreamWrapper& operator=(const OStreamWrapper&);
    161 
    162     std::ostream& os_;
    163 };
    164 
    165 TEST(Writer, OStreamWrapper) {
    166     StringStream s("{ \"hello\" : \"world\", \"t\" : true , \"f\" : false, \"n\": null, \"i\":123, \"pi\": 3.1416, \"a\":[1, 2, 3], \"u64\": 1234567890123456789, \"i64\":-1234567890123456789 } ");
    167 
    168     std::stringstream ss;
    169     OStreamWrapper os(ss);
    170 
    171     Writer<OStreamWrapper> writer(os);
    172 
    173     Reader reader;
    174     reader.Parse<0>(s, writer);
    175 
    176     std::string actual = ss.str();
    177     EXPECT_STREQ("{\"hello\":\"world\",\"t\":true,\"f\":false,\"n\":null,\"i\":123,\"pi\":3.1416,\"a\":[1,2,3],\"u64\":1234567890123456789,\"i64\":-1234567890123456789}", actual.c_str());
    178 }
    179 
    180 TEST(Writer, AssertRootMayBeAnyValue) {
    181 #define T(x)\
    182     {\
    183         StringBuffer buffer;\
    184         Writer<StringBuffer> writer(buffer);\
    185         EXPECT_TRUE(x);\
    186     }
    187     T(writer.Bool(false));
    188     T(writer.Bool(true));
    189     T(writer.Null());
    190     T(writer.Int(0));
    191     T(writer.Uint(0));
    192     T(writer.Int64(0));
    193     T(writer.Uint64(0));
    194     T(writer.Double(0));
    195     T(writer.String("foo"));
    196 #undef T
    197 }
    198 
    199 TEST(Writer, AssertIncorrectObjectLevel) {
    200     StringBuffer buffer;
    201     Writer<StringBuffer> writer(buffer);
    202     writer.StartObject();
    203     writer.EndObject();
    204     ASSERT_THROW(writer.EndObject(), AssertException);
    205 }
    206 
    207 TEST(Writer, AssertIncorrectArrayLevel) {
    208     StringBuffer buffer;
    209     Writer<StringBuffer> writer(buffer);
    210     writer.StartArray();
    211     writer.EndArray();
    212     ASSERT_THROW(writer.EndArray(), AssertException);
    213 }
    214 
    215 TEST(Writer, AssertIncorrectEndObject) {
    216     StringBuffer buffer;
    217     Writer<StringBuffer> writer(buffer);
    218     writer.StartObject();
    219     ASSERT_THROW(writer.EndArray(), AssertException);
    220 }
    221 
    222 TEST(Writer, AssertIncorrectEndArray) {
    223     StringBuffer buffer;
    224     Writer<StringBuffer> writer(buffer);
    225     writer.StartObject();
    226     ASSERT_THROW(writer.EndArray(), AssertException);
    227 }
    228 
    229 TEST(Writer, AssertObjectKeyNotString) {
    230 #define T(x)\
    231     {\
    232         StringBuffer buffer;\
    233         Writer<StringBuffer> writer(buffer);\
    234         writer.StartObject();\
    235         ASSERT_THROW(x, AssertException); \
    236     }
    237     T(writer.Bool(false));
    238     T(writer.Bool(true));
    239     T(writer.Null());
    240     T(writer.Int(0));
    241     T(writer.Uint(0));
    242     T(writer.Int64(0));
    243     T(writer.Uint64(0));
    244     T(writer.Double(0));
    245     T(writer.StartObject());
    246     T(writer.StartArray());
    247 #undef T
    248 }
    249 
    250 TEST(Writer, AssertMultipleRoot) {
    251     StringBuffer buffer;
    252     Writer<StringBuffer> writer(buffer);
    253 
    254     writer.StartObject();
    255     writer.EndObject();
    256     ASSERT_THROW(writer.StartObject(), AssertException);
    257 
    258     writer.Reset(buffer);
    259     writer.Null();
    260     ASSERT_THROW(writer.Int(0), AssertException);
    261 
    262     writer.Reset(buffer);
    263     writer.String("foo");
    264     ASSERT_THROW(writer.StartArray(), AssertException);
    265 
    266     writer.Reset(buffer);
    267     writer.StartArray();
    268     writer.EndArray();
    269     //ASSERT_THROW(writer.Double(3.14), AssertException);
    270 }
    271 
    272 TEST(Writer, RootObjectIsComplete) {
    273     StringBuffer buffer;
    274     Writer<StringBuffer> writer(buffer);
    275     EXPECT_FALSE(writer.IsComplete());
    276     writer.StartObject();
    277     EXPECT_FALSE(writer.IsComplete());
    278     writer.String("foo");
    279     EXPECT_FALSE(writer.IsComplete());
    280     writer.Int(1);
    281     EXPECT_FALSE(writer.IsComplete());
    282     writer.EndObject();
    283     EXPECT_TRUE(writer.IsComplete());
    284 }
    285 
    286 TEST(Writer, RootArrayIsComplete) {
    287     StringBuffer buffer;
    288     Writer<StringBuffer> writer(buffer);
    289     EXPECT_FALSE(writer.IsComplete());
    290     writer.StartArray();
    291     EXPECT_FALSE(writer.IsComplete());
    292     writer.String("foo");
    293     EXPECT_FALSE(writer.IsComplete());
    294     writer.Int(1);
    295     EXPECT_FALSE(writer.IsComplete());
    296     writer.EndArray();
    297     EXPECT_TRUE(writer.IsComplete());
    298 }
    299 
    300 TEST(Writer, RootValueIsComplete) {
    301 #define T(x)\
    302     {\
    303         StringBuffer buffer;\
    304         Writer<StringBuffer> writer(buffer);\
    305         EXPECT_FALSE(writer.IsComplete()); \
    306         x; \
    307         EXPECT_TRUE(writer.IsComplete()); \
    308     }
    309     T(writer.Null());
    310     T(writer.Bool(true));
    311     T(writer.Bool(false));
    312     T(writer.Int(0));
    313     T(writer.Uint(0));
    314     T(writer.Int64(0));
    315     T(writer.Uint64(0));
    316     T(writer.Double(0));
    317     T(writer.String(""));
    318 #undef T
    319 }
    320 
    321 TEST(Writer, InvalidEncoding) {
    322     // Fail in decoding invalid UTF-8 sequence http://www.cl.cam.ac.uk/~mgk25/ucs/examples/UTF-8-test.txt
    323     {
    324         GenericStringBuffer<UTF16<> > buffer;
    325         Writer<GenericStringBuffer<UTF16<> >, UTF8<>, UTF16<> > writer(buffer);
    326         writer.StartArray();
    327         EXPECT_FALSE(writer.String("\xfe"));
    328         EXPECT_FALSE(writer.String("\xff"));
    329         EXPECT_FALSE(writer.String("\xfe\xfe\xff\xff"));
    330         writer.EndArray();
    331     }
    332 
    333     // Fail in encoding
    334     {
    335         StringBuffer buffer;
    336         Writer<StringBuffer, UTF32<> > writer(buffer);
    337         static const UTF32<>::Ch s[] = { 0x110000, 0 }; // Out of U+0000 to U+10FFFF
    338         EXPECT_FALSE(writer.String(s));
    339     }
    340 
    341     // Fail in unicode escaping in ASCII output
    342     {
    343         StringBuffer buffer;
    344         Writer<StringBuffer, UTF32<>, ASCII<> > writer(buffer);
    345         static const UTF32<>::Ch s[] = { 0x110000, 0 }; // Out of U+0000 to U+10FFFF
    346         EXPECT_FALSE(writer.String(s));
    347     }
    348 }
    349 
    350 TEST(Writer, InvalidEventSequence) {
    351     // {]
    352     {
    353         StringBuffer buffer;
    354         Writer<StringBuffer> writer(buffer);
    355         writer.StartObject();
    356         EXPECT_THROW(writer.EndArray(), AssertException);
    357         EXPECT_FALSE(writer.IsComplete());
    358     }
    359 
    360     // [}
    361     {
    362         StringBuffer buffer;
    363         Writer<StringBuffer> writer(buffer);
    364         writer.StartArray();
    365         EXPECT_THROW(writer.EndObject(), AssertException);
    366         EXPECT_FALSE(writer.IsComplete());
    367     }
    368 
    369     // { 1:
    370     {
    371         StringBuffer buffer;
    372         Writer<StringBuffer> writer(buffer);
    373         writer.StartObject();
    374         EXPECT_THROW(writer.Int(1), AssertException);
    375         EXPECT_FALSE(writer.IsComplete());
    376     }
    377 }
    378