2014-08-11 22:26:45 +08:00
|
|
|
// Copyright (C) 2011 Milo Yip
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
// THE SOFTWARE.
|
|
|
|
|
2011-11-18 17:01:23 +00:00
|
|
|
#include "unittest.h"
|
2014-07-13 13:16:03 +08:00
|
|
|
|
2014-04-01 16:49:24 +02:00
|
|
|
#include "rapidjson/document.h"
|
2011-11-18 17:01:23 +00:00
|
|
|
#include "rapidjson/reader.h"
|
|
|
|
#include "rapidjson/writer.h"
|
|
|
|
#include "rapidjson/stringbuffer.h"
|
|
|
|
|
|
|
|
using namespace rapidjson;
|
|
|
|
|
|
|
|
TEST(Writer, Compact) {
|
2014-08-11 22:26:45 +08:00
|
|
|
StringStream s("{ \"hello\" : \"world\", \"t\" : true , \"f\" : false, \"n\": null, \"i\":123, \"pi\": 3.1416, \"a\":[1, 2, 3] } ");
|
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
2014-08-17 18:32:08 +08:00
|
|
|
buffer.ShrinkToFit();
|
2014-08-11 22:26:45 +08:00
|
|
|
Reader reader;
|
|
|
|
reader.Parse<0>(s, writer);
|
|
|
|
EXPECT_STREQ("{\"hello\":\"world\",\"t\":true,\"f\":false,\"n\":null,\"i\":123,\"pi\":3.1416,\"a\":[1,2,3]}", buffer.GetString());
|
|
|
|
EXPECT_EQ(77u, buffer.GetSize());
|
|
|
|
EXPECT_TRUE(writer.IsComplete());
|
2011-11-18 17:01:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// json -> parse -> writer -> json
|
|
|
|
#define TEST_ROUNDTRIP(json) \
|
2014-08-11 22:26:45 +08:00
|
|
|
{ \
|
|
|
|
StringStream s(json); \
|
|
|
|
StringBuffer buffer; \
|
|
|
|
Writer<StringBuffer> writer(buffer); \
|
|
|
|
Reader reader; \
|
|
|
|
reader.Parse<0>(s, writer); \
|
|
|
|
EXPECT_STREQ(json, buffer.GetString()); \
|
|
|
|
EXPECT_TRUE(writer.IsComplete()); \
|
|
|
|
}
|
2011-11-18 17:01:23 +00:00
|
|
|
|
2014-08-11 15:26:44 +02:00
|
|
|
TEST(Writer, Root) {
|
2014-08-11 17:11:59 +02:00
|
|
|
TEST_ROUNDTRIP("null");
|
|
|
|
TEST_ROUNDTRIP("true");
|
|
|
|
TEST_ROUNDTRIP("false");
|
|
|
|
TEST_ROUNDTRIP("0");
|
|
|
|
TEST_ROUNDTRIP("\"foo\"");
|
|
|
|
TEST_ROUNDTRIP("[]");
|
|
|
|
TEST_ROUNDTRIP("{}");
|
2014-08-11 15:26:44 +02:00
|
|
|
}
|
|
|
|
|
2011-11-18 17:01:23 +00:00
|
|
|
TEST(Writer, Int) {
|
2014-08-11 22:26:45 +08:00
|
|
|
TEST_ROUNDTRIP("[-1]");
|
|
|
|
TEST_ROUNDTRIP("[-123]");
|
|
|
|
TEST_ROUNDTRIP("[-2147483648]");
|
2011-11-18 17:01:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, UInt) {
|
2014-08-11 22:26:45 +08:00
|
|
|
TEST_ROUNDTRIP("[0]");
|
|
|
|
TEST_ROUNDTRIP("[1]");
|
|
|
|
TEST_ROUNDTRIP("[123]");
|
|
|
|
TEST_ROUNDTRIP("[2147483647]");
|
|
|
|
TEST_ROUNDTRIP("[4294967295]");
|
2011-11-18 17:01:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, Int64) {
|
2014-08-11 22:26:45 +08:00
|
|
|
TEST_ROUNDTRIP("[-1234567890123456789]");
|
|
|
|
TEST_ROUNDTRIP("[-9223372036854775808]");
|
2011-11-18 17:01:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, Uint64) {
|
2014-08-11 22:26:45 +08:00
|
|
|
TEST_ROUNDTRIP("[1234567890123456789]");
|
|
|
|
TEST_ROUNDTRIP("[9223372036854775807]");
|
2011-11-18 17:01:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, String) {
|
2014-08-11 22:26:45 +08:00
|
|
|
TEST_ROUNDTRIP("[\"Hello\"]");
|
|
|
|
TEST_ROUNDTRIP("[\"Hello\\u0000World\"]");
|
|
|
|
TEST_ROUNDTRIP("[\"\\\"\\\\/\\b\\f\\n\\r\\t\"]");
|
2015-04-13 14:25:05 +08:00
|
|
|
|
|
|
|
#if RAPIDJSON_HAS_STDSTRING
|
|
|
|
{
|
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
|
|
|
writer.String(std::string("Hello\n"));
|
|
|
|
EXPECT_STREQ("\"Hello\\n\"", buffer.GetString());
|
|
|
|
}
|
|
|
|
#endif
|
2011-11-18 17:01:23 +00:00
|
|
|
}
|
2011-12-03 04:17:07 +00:00
|
|
|
|
2014-08-09 21:12:58 +08:00
|
|
|
TEST(Writer, Double) {
|
2014-08-11 22:26:45 +08:00
|
|
|
TEST_ROUNDTRIP("[1.2345,1.2345678,0.123456789012,1234567.8]");
|
2015-04-10 21:24:29 +08:00
|
|
|
TEST_ROUNDTRIP("[-0.0]"); // Issue #289
|
2014-04-01 16:49:24 +02:00
|
|
|
}
|
|
|
|
|
2011-12-03 04:17:07 +00:00
|
|
|
TEST(Writer, Transcode) {
|
2014-08-11 22:26:45 +08:00
|
|
|
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\"}";
|
|
|
|
|
|
|
|
// UTF8 -> UTF16 -> UTF8
|
|
|
|
{
|
|
|
|
StringStream s(json);
|
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer, UTF16<>, UTF8<> > writer(buffer);
|
|
|
|
GenericReader<UTF8<>, UTF16<> > reader;
|
|
|
|
reader.Parse(s, writer);
|
|
|
|
EXPECT_STREQ(json, buffer.GetString());
|
|
|
|
}
|
|
|
|
|
|
|
|
// UTF8 -> UTF8 -> ASCII -> UTF8 -> UTF8
|
|
|
|
{
|
|
|
|
StringStream s(json);
|
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer, UTF8<>, ASCII<> > writer(buffer);
|
|
|
|
Reader reader;
|
|
|
|
reader.Parse(s, writer);
|
|
|
|
|
|
|
|
StringBuffer buffer2;
|
|
|
|
Writer<StringBuffer> writer2(buffer2);
|
|
|
|
GenericReader<ASCII<>, UTF8<> > reader2;
|
|
|
|
StringStream s2(buffer.GetString());
|
|
|
|
reader2.Parse(s2, writer2);
|
|
|
|
|
|
|
|
EXPECT_STREQ(json, buffer2.GetString());
|
|
|
|
}
|
2012-11-13 08:02:22 +00:00
|
|
|
}
|
2014-07-04 16:39:09 +08:00
|
|
|
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
class OStreamWrapper {
|
|
|
|
public:
|
2014-08-11 22:26:45 +08:00
|
|
|
typedef char Ch;
|
2014-07-04 16:39:09 +08:00
|
|
|
|
2014-08-11 22:26:45 +08:00
|
|
|
OStreamWrapper(std::ostream& os) : os_(os) {}
|
2014-07-04 16:39:09 +08:00
|
|
|
|
2014-08-11 22:26:45 +08:00
|
|
|
Ch Peek() const { assert(false); return '\0'; }
|
|
|
|
Ch Take() { assert(false); return '\0'; }
|
|
|
|
size_t Tell() const { return 0; }
|
2014-07-04 16:39:09 +08:00
|
|
|
|
2014-08-11 22:26:45 +08:00
|
|
|
Ch* PutBegin() { assert(false); return 0; }
|
|
|
|
void Put(Ch c) { os_.put(c); }
|
|
|
|
void Flush() { os_.flush(); }
|
|
|
|
size_t PutEnd(Ch*) { assert(false); return 0; }
|
2014-07-04 16:39:09 +08:00
|
|
|
|
|
|
|
private:
|
2014-08-11 22:26:45 +08:00
|
|
|
OStreamWrapper(const OStreamWrapper&);
|
|
|
|
OStreamWrapper& operator=(const OStreamWrapper&);
|
2014-07-04 16:39:09 +08:00
|
|
|
|
2014-08-11 22:26:45 +08:00
|
|
|
std::ostream& os_;
|
2014-07-04 16:39:09 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
TEST(Writer, OStreamWrapper) {
|
2015-04-13 13:41:56 +08:00
|
|
|
StringStream s("{ \"hello\" : \"world\", \"t\" : true , \"f\" : false, \"n\": null, \"i\":123, \"pi\": 3.1416, \"a\":[1, 2, 3], \"u64\": 1234567890123456789, \"i64\":-1234567890123456789 } ");
|
2014-08-11 22:26:45 +08:00
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
OStreamWrapper os(ss);
|
|
|
|
|
|
|
|
Writer<OStreamWrapper> writer(os);
|
|
|
|
|
|
|
|
Reader reader;
|
|
|
|
reader.Parse<0>(s, writer);
|
|
|
|
|
|
|
|
std::string actual = ss.str();
|
2015-04-13 13:41:56 +08:00
|
|
|
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());
|
2014-07-04 16:39:09 +08:00
|
|
|
}
|
2014-07-13 13:16:03 +08:00
|
|
|
|
2014-08-11 15:17:39 +02:00
|
|
|
TEST(Writer, AssertRootMayBeAnyValue) {
|
2014-07-13 13:16:03 +08:00
|
|
|
#define T(x)\
|
2014-08-11 22:26:45 +08:00
|
|
|
{\
|
|
|
|
StringBuffer buffer;\
|
|
|
|
Writer<StringBuffer> writer(buffer);\
|
2014-08-11 17:11:59 +02:00
|
|
|
EXPECT_TRUE(x);\
|
2014-08-11 22:26:45 +08:00
|
|
|
}
|
|
|
|
T(writer.Bool(false));
|
|
|
|
T(writer.Bool(true));
|
|
|
|
T(writer.Null());
|
|
|
|
T(writer.Int(0));
|
|
|
|
T(writer.Uint(0));
|
|
|
|
T(writer.Int64(0));
|
|
|
|
T(writer.Uint64(0));
|
|
|
|
T(writer.Double(0));
|
|
|
|
T(writer.String("foo"));
|
2014-07-13 13:16:03 +08:00
|
|
|
#undef T
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, AssertIncorrectObjectLevel) {
|
2014-08-11 22:26:45 +08:00
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
|
|
|
writer.StartObject();
|
|
|
|
writer.EndObject();
|
|
|
|
ASSERT_THROW(writer.EndObject(), AssertException);
|
2014-07-13 13:16:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, AssertIncorrectArrayLevel) {
|
2014-08-11 22:26:45 +08:00
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
|
|
|
writer.StartArray();
|
|
|
|
writer.EndArray();
|
|
|
|
ASSERT_THROW(writer.EndArray(), AssertException);
|
2014-07-13 13:16:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, AssertIncorrectEndObject) {
|
2014-08-11 22:26:45 +08:00
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
|
|
|
writer.StartObject();
|
|
|
|
ASSERT_THROW(writer.EndArray(), AssertException);
|
2014-07-13 13:16:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, AssertIncorrectEndArray) {
|
2014-08-11 22:26:45 +08:00
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
|
|
|
writer.StartObject();
|
|
|
|
ASSERT_THROW(writer.EndArray(), AssertException);
|
2014-07-13 13:16:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, AssertObjectKeyNotString) {
|
|
|
|
#define T(x)\
|
2014-08-11 22:26:45 +08:00
|
|
|
{\
|
|
|
|
StringBuffer buffer;\
|
|
|
|
Writer<StringBuffer> writer(buffer);\
|
|
|
|
writer.StartObject();\
|
|
|
|
ASSERT_THROW(x, AssertException); \
|
|
|
|
}
|
|
|
|
T(writer.Bool(false));
|
|
|
|
T(writer.Bool(true));
|
|
|
|
T(writer.Null());
|
|
|
|
T(writer.Int(0));
|
|
|
|
T(writer.Uint(0));
|
|
|
|
T(writer.Int64(0));
|
|
|
|
T(writer.Uint64(0));
|
|
|
|
T(writer.Double(0));
|
|
|
|
T(writer.StartObject());
|
|
|
|
T(writer.StartArray());
|
2014-07-13 13:16:03 +08:00
|
|
|
#undef T
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, AssertMultipleRoot) {
|
2014-08-11 22:26:45 +08:00
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
2014-08-11 15:17:39 +02:00
|
|
|
|
2014-08-11 22:26:45 +08:00
|
|
|
writer.StartObject();
|
|
|
|
writer.EndObject();
|
|
|
|
ASSERT_THROW(writer.StartObject(), AssertException);
|
2014-08-11 15:17:39 +02:00
|
|
|
|
2014-08-11 17:11:59 +02:00
|
|
|
writer.Reset(buffer);
|
|
|
|
writer.Null();
|
|
|
|
ASSERT_THROW(writer.Int(0), AssertException);
|
2014-08-11 15:17:39 +02:00
|
|
|
|
2014-08-11 17:11:59 +02:00
|
|
|
writer.Reset(buffer);
|
|
|
|
writer.String("foo");
|
|
|
|
ASSERT_THROW(writer.StartArray(), AssertException);
|
2014-08-11 15:17:39 +02:00
|
|
|
|
2014-08-11 17:11:59 +02:00
|
|
|
writer.Reset(buffer);
|
|
|
|
writer.StartArray();
|
|
|
|
writer.EndArray();
|
2014-09-14 10:52:33 +08:00
|
|
|
//ASSERT_THROW(writer.Double(3.14), AssertException);
|
2014-07-13 13:16:03 +08:00
|
|
|
}
|
2014-07-13 13:21:25 +08:00
|
|
|
|
|
|
|
TEST(Writer, RootObjectIsComplete) {
|
2014-08-11 22:26:45 +08:00
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
writer.StartObject();
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
writer.String("foo");
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
writer.Int(1);
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
writer.EndObject();
|
|
|
|
EXPECT_TRUE(writer.IsComplete());
|
2014-07-13 13:21:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, RootArrayIsComplete) {
|
2014-08-11 22:26:45 +08:00
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
writer.StartArray();
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
writer.String("foo");
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
writer.Int(1);
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
writer.EndArray();
|
|
|
|
EXPECT_TRUE(writer.IsComplete());
|
2014-07-15 09:20:19 +08:00
|
|
|
}
|
2014-08-11 15:17:39 +02:00
|
|
|
|
|
|
|
TEST(Writer, RootValueIsComplete) {
|
|
|
|
#define T(x)\
|
2014-08-11 17:11:59 +02:00
|
|
|
{\
|
|
|
|
StringBuffer buffer;\
|
|
|
|
Writer<StringBuffer> writer(buffer);\
|
|
|
|
EXPECT_FALSE(writer.IsComplete()); \
|
|
|
|
x; \
|
|
|
|
EXPECT_TRUE(writer.IsComplete()); \
|
|
|
|
}
|
|
|
|
T(writer.Null());
|
|
|
|
T(writer.Bool(true));
|
|
|
|
T(writer.Bool(false));
|
|
|
|
T(writer.Int(0));
|
|
|
|
T(writer.Uint(0));
|
|
|
|
T(writer.Int64(0));
|
|
|
|
T(writer.Uint64(0));
|
|
|
|
T(writer.Double(0));
|
|
|
|
T(writer.String(""));
|
2014-08-11 15:17:39 +02:00
|
|
|
#undef T
|
|
|
|
}
|
2015-04-13 14:05:43 +08:00
|
|
|
|
|
|
|
TEST(Writer, InvalidEncoding) {
|
|
|
|
// Fail in decoding invalid UTF-8 sequence http://www.cl.cam.ac.uk/~mgk25/ucs/examples/UTF-8-test.txt
|
|
|
|
{
|
|
|
|
GenericStringBuffer<UTF16<> > buffer;
|
|
|
|
Writer<GenericStringBuffer<UTF16<> >, UTF8<>, UTF16<> > writer(buffer);
|
|
|
|
writer.StartArray();
|
|
|
|
EXPECT_FALSE(writer.String("\xfe"));
|
|
|
|
EXPECT_FALSE(writer.String("\xff"));
|
|
|
|
EXPECT_FALSE(writer.String("\xfe\xfe\xff\xff"));
|
|
|
|
writer.EndArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fail in encoding
|
|
|
|
{
|
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer, UTF32<> > writer(buffer);
|
|
|
|
static const UTF32<>::Ch s[] = { 0x110000, 0 }; // Out of U+0000 to U+10FFFF
|
|
|
|
EXPECT_FALSE(writer.String(s));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fail in unicode escaping in ASCII output
|
|
|
|
{
|
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer, UTF32<>, ASCII<> > writer(buffer);
|
|
|
|
static const UTF32<>::Ch s[] = { 0x110000, 0 }; // Out of U+0000 to U+10FFFF
|
|
|
|
EXPECT_FALSE(writer.String(s));
|
|
|
|
}
|
2015-04-13 14:50:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Writer, InvalidEventSequence) {
|
|
|
|
// {]
|
|
|
|
{
|
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
|
|
|
writer.StartObject();
|
|
|
|
EXPECT_THROW(writer.EndArray(), AssertException);
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
}
|
|
|
|
|
|
|
|
// [}
|
|
|
|
{
|
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
|
|
|
writer.StartArray();
|
|
|
|
EXPECT_THROW(writer.EndObject(), AssertException);
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
}
|
|
|
|
|
|
|
|
// { 1:
|
|
|
|
{
|
|
|
|
StringBuffer buffer;
|
|
|
|
Writer<StringBuffer> writer(buffer);
|
|
|
|
writer.StartObject();
|
|
|
|
EXPECT_THROW(writer.Int(1), AssertException);
|
|
|
|
EXPECT_FALSE(writer.IsComplete());
|
|
|
|
}
|
|
|
|
}
|