2015-04-10 14:54:13 +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.
|
|
|
|
|
|
|
|
#include "unittest.h"
|
|
|
|
#include "rapidjson/pointer.h"
|
|
|
|
#include "rapidjson/stringbuffer.h"
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
using namespace rapidjson;
|
|
|
|
|
2015-04-10 17:43:11 +08:00
|
|
|
static const char kJson[] = "{\n"
|
2015-04-10 14:54:13 +08:00
|
|
|
" \"foo\":[\"bar\", \"baz\"],\n"
|
|
|
|
" \"\" : 0,\n"
|
|
|
|
" \"a/b\" : 1,\n"
|
|
|
|
" \"c%d\" : 2,\n"
|
|
|
|
" \"e^f\" : 3,\n"
|
|
|
|
" \"g|h\" : 4,\n"
|
|
|
|
" \"i\\\\j\" : 5,\n"
|
|
|
|
" \"k\\\"l\" : 6,\n"
|
|
|
|
" \" \" : 7,\n"
|
|
|
|
" \"m~n\" : 8\n"
|
|
|
|
"}";
|
|
|
|
|
|
|
|
TEST(Pointer, Parse) {
|
|
|
|
{
|
|
|
|
Pointer p("");
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(0u, p.GetTokenCount());
|
2015-04-10 14:54:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
Pointer p("/foo");
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokenCount());
|
|
|
|
EXPECT_EQ(3u, p.GetTokens()[0].length);
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("foo", p.GetTokens()[0].name);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
Pointer p("/foo/0");
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(2u, p.GetTokenCount());
|
|
|
|
EXPECT_EQ(3u, p.GetTokens()[0].length);
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("foo", p.GetTokens()[0].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokens()[1].length);
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("0", p.GetTokens()[1].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(0u, p.GetTokens()[1].index);
|
2015-04-10 14:54:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
// Unescape ~1
|
|
|
|
Pointer p("/a~1b");
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokenCount());
|
|
|
|
EXPECT_EQ(3u, p.GetTokens()[0].length);
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("a/b", p.GetTokens()[0].name);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
// Unescape ~0
|
|
|
|
Pointer p("/m~0n");
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokenCount());
|
|
|
|
EXPECT_EQ(3u, p.GetTokens()[0].length);
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("m~n", p.GetTokens()[0].name);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
// empty name
|
|
|
|
Pointer p("/");
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokenCount());
|
|
|
|
EXPECT_EQ(0u, p.GetTokens()[0].length);
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("", p.GetTokens()[0].name);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
// empty and non-empty name
|
|
|
|
Pointer p("//a");
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(2u, p.GetTokenCount());
|
|
|
|
EXPECT_EQ(0u, p.GetTokens()[0].length);
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("", p.GetTokens()[0].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokens()[1].length);
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("a", p.GetTokens()[1].name);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
// Null characters
|
|
|
|
Pointer p("/\0\0", 3);
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokenCount());
|
|
|
|
EXPECT_EQ(2u, p.GetTokens()[0].length);
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_EQ('\0', p.GetTokens()[0].name[0]);
|
|
|
|
EXPECT_EQ('\0', p.GetTokens()[0].name[1]);
|
|
|
|
EXPECT_EQ('\0', p.GetTokens()[0].name[2]);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
// Valid index
|
|
|
|
Pointer p("/123");
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokenCount());
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("123", p.GetTokens()[0].name);
|
|
|
|
EXPECT_EQ(123, p.GetTokens()[0].index);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
// Invalid index (with leading zero)
|
|
|
|
Pointer p("/01");
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokenCount());
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("01", p.GetTokens()[0].name);
|
|
|
|
EXPECT_EQ(Pointer::kInvalidIndex, p.GetTokens()[0].index);
|
|
|
|
}
|
|
|
|
|
2015-04-10 17:43:11 +08:00
|
|
|
if (sizeof(SizeType) == 4) {
|
2015-04-10 14:54:13 +08:00
|
|
|
// Invalid index (overflow)
|
|
|
|
Pointer p("/4294967296");
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokenCount());
|
2015-04-10 14:54:13 +08:00
|
|
|
EXPECT_STREQ("4294967296", p.GetTokens()[0].name);
|
|
|
|
EXPECT_EQ(Pointer::kInvalidIndex, p.GetTokens()[0].index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, Stringify) {
|
|
|
|
// Test by roundtrip
|
|
|
|
const char* sources[] = {
|
|
|
|
"",
|
|
|
|
"/foo",
|
|
|
|
"/foo/0",
|
|
|
|
"/",
|
|
|
|
"/a~1b",
|
|
|
|
"/c%d",
|
|
|
|
"/e^f",
|
|
|
|
"/g|h",
|
|
|
|
"/i\\j",
|
|
|
|
"/k\"l",
|
|
|
|
"/ ",
|
|
|
|
"/m~0n"
|
|
|
|
};
|
|
|
|
|
|
|
|
for (size_t i = 0; i < sizeof(sources) / sizeof(sources[0]); i++) {
|
|
|
|
Pointer p(sources[i]);
|
|
|
|
StringBuffer s;
|
|
|
|
p.Stringify(s);
|
|
|
|
EXPECT_STREQ(sources[i], s.GetString());
|
|
|
|
}
|
|
|
|
}
|
2015-04-10 17:43:11 +08:00
|
|
|
|
2015-04-10 18:25:02 +08:00
|
|
|
// Construct a Pointer with static tokens, no dynamic allocation involved.
|
|
|
|
#define NAME(s) { s, sizeof(s) / sizeof(s[0]) - 1, Pointer::kInvalidIndex }
|
|
|
|
#define INDEX(i) { #i, sizeof(#i) - 1, i }
|
|
|
|
|
|
|
|
static const Pointer::Token kTokens[] = { NAME("foo"), INDEX(0) }; // equivalent to "/foo/0"
|
|
|
|
|
|
|
|
#undef NAME
|
|
|
|
#undef INDEX
|
|
|
|
|
|
|
|
TEST(Pointer, ConstructorWithToken) {
|
|
|
|
Pointer p(kTokens, sizeof(kTokens) / sizeof(kTokens[0]));
|
|
|
|
EXPECT_TRUE(p.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(2u, p.GetTokenCount());
|
|
|
|
EXPECT_EQ(3u, p.GetTokens()[0].length);
|
2015-04-10 18:25:02 +08:00
|
|
|
EXPECT_STREQ("foo", p.GetTokens()[0].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, p.GetTokens()[1].length);
|
2015-04-10 18:25:02 +08:00
|
|
|
EXPECT_STREQ("0", p.GetTokens()[1].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(0u, p.GetTokens()[1].index);
|
2015-04-10 18:25:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, CopyConstructor) {
|
|
|
|
{
|
|
|
|
Pointer p("/foo/0");
|
|
|
|
Pointer q(p);
|
|
|
|
EXPECT_TRUE(q.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(2u, q.GetTokenCount());
|
|
|
|
EXPECT_EQ(3u, q.GetTokens()[0].length);
|
2015-04-10 18:25:02 +08:00
|
|
|
EXPECT_STREQ("foo", q.GetTokens()[0].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, q.GetTokens()[1].length);
|
2015-04-10 18:25:02 +08:00
|
|
|
EXPECT_STREQ("0", q.GetTokens()[1].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(0u, q.GetTokens()[1].index);
|
2015-04-10 18:25:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Static tokens
|
|
|
|
{
|
|
|
|
Pointer p(kTokens, sizeof(kTokens) / sizeof(kTokens[0]));
|
|
|
|
Pointer q(p);
|
|
|
|
EXPECT_TRUE(q.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(2u, q.GetTokenCount());
|
|
|
|
EXPECT_EQ(3u, q.GetTokens()[0].length);
|
2015-04-10 18:25:02 +08:00
|
|
|
EXPECT_STREQ("foo", q.GetTokens()[0].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, q.GetTokens()[1].length);
|
2015-04-10 18:25:02 +08:00
|
|
|
EXPECT_STREQ("0", q.GetTokens()[1].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(0u, q.GetTokens()[1].index);
|
2015-04-10 18:25:02 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, Assignment) {
|
|
|
|
{
|
|
|
|
Pointer p("/foo/0");
|
|
|
|
Pointer q;
|
|
|
|
q = p;
|
|
|
|
EXPECT_TRUE(q.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(2u, q.GetTokenCount());
|
|
|
|
EXPECT_EQ(3u, q.GetTokens()[0].length);
|
2015-04-10 18:25:02 +08:00
|
|
|
EXPECT_STREQ("foo", q.GetTokens()[0].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, q.GetTokens()[1].length);
|
2015-04-10 18:25:02 +08:00
|
|
|
EXPECT_STREQ("0", q.GetTokens()[1].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(0u, q.GetTokens()[1].index);
|
2015-04-10 18:25:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Static tokens
|
|
|
|
{
|
|
|
|
Pointer p(kTokens, sizeof(kTokens) / sizeof(kTokens[0]));
|
|
|
|
Pointer q;
|
|
|
|
q = p;
|
|
|
|
EXPECT_TRUE(q.IsValid());
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(2u, q.GetTokenCount());
|
|
|
|
EXPECT_EQ(3u, q.GetTokens()[0].length);
|
2015-04-10 18:25:02 +08:00
|
|
|
EXPECT_STREQ("foo", q.GetTokens()[0].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(1u, q.GetTokens()[1].length);
|
2015-04-10 18:25:02 +08:00
|
|
|
EXPECT_STREQ("0", q.GetTokens()[1].name);
|
2015-04-10 22:12:59 +08:00
|
|
|
EXPECT_EQ(0u, q.GetTokens()[1].index);
|
2015-04-10 18:25:02 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-10 17:43:11 +08:00
|
|
|
TEST(Pointer, Create) {
|
|
|
|
Document d;
|
2015-04-10 19:28:12 +08:00
|
|
|
{
|
|
|
|
Value* v = &Pointer("").Create(d, d.GetAllocator());
|
|
|
|
EXPECT_EQ(&d, v);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
Value* v = &Pointer("/foo").Create(d, d.GetAllocator());
|
|
|
|
EXPECT_EQ(&d["foo"], v);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
Value* v = &Pointer("/foo/0").Create(d, d.GetAllocator());
|
|
|
|
EXPECT_EQ(&d["foo"][0], v);
|
|
|
|
}
|
2015-04-10 17:43:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, Get) {
|
|
|
|
Document d;
|
|
|
|
d.Parse(kJson);
|
|
|
|
|
|
|
|
EXPECT_EQ(&d, Pointer("").Get(d));
|
|
|
|
EXPECT_EQ(&d["foo"], Pointer("/foo").Get(d));
|
|
|
|
EXPECT_EQ(&d["foo"][0], Pointer("/foo/0").Get(d));
|
|
|
|
EXPECT_EQ(&d[""], Pointer("/").Get(d));
|
|
|
|
EXPECT_EQ(&d["a/b"], Pointer("/a~1b").Get(d));
|
|
|
|
EXPECT_EQ(&d["c%d"], Pointer("/c%d").Get(d));
|
|
|
|
EXPECT_EQ(&d["e^f"], Pointer("/e^f").Get(d));
|
|
|
|
EXPECT_EQ(&d["g|h"], Pointer("/g|h").Get(d));
|
|
|
|
EXPECT_EQ(&d["i\\j"], Pointer("/i\\j").Get(d));
|
|
|
|
EXPECT_EQ(&d["k\"l"], Pointer("/k\"l").Get(d));
|
|
|
|
EXPECT_EQ(&d[" "], Pointer("/ ").Get(d));
|
|
|
|
EXPECT_EQ(&d["m~n"], Pointer("/m~0n").Get(d));
|
2015-04-10 19:28:12 +08:00
|
|
|
EXPECT_TRUE(Pointer("/abc").Get(d) == 0);
|
2015-04-10 17:43:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, GetWithDefault) {
|
|
|
|
Document d;
|
|
|
|
d.Parse(kJson);
|
|
|
|
|
|
|
|
Document::AllocatorType& a = d.GetAllocator();
|
|
|
|
const Value v("qux");
|
|
|
|
EXPECT_TRUE(Value("bar") == Pointer("/foo/0").GetWithDefault(d, v, a));
|
|
|
|
EXPECT_TRUE(Value("baz") == Pointer("/foo/1").GetWithDefault(d, v, a));
|
|
|
|
EXPECT_TRUE(Value("qux") == Pointer("/foo/2").GetWithDefault(d, v, a));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, Set) {
|
|
|
|
Document d;
|
|
|
|
d.Parse(kJson);
|
|
|
|
Document::AllocatorType& a = d.GetAllocator();
|
|
|
|
|
|
|
|
Pointer("/foo/0").Set(d, Value(123).Move(), a);
|
|
|
|
EXPECT_EQ(123, d["foo"][0].GetInt());
|
|
|
|
|
|
|
|
Pointer("/foo/2").Set(d, Value(456).Move(), a);
|
|
|
|
EXPECT_EQ(456, d["foo"][2].GetInt());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, Swap) {
|
|
|
|
Document d;
|
|
|
|
d.Parse(kJson);
|
|
|
|
Document::AllocatorType& a = d.GetAllocator();
|
|
|
|
Pointer("/foo/0").Swap(d, *Pointer("/foo/1").Get(d), a);
|
|
|
|
EXPECT_STREQ("baz", d["foo"][0].GetString());
|
|
|
|
EXPECT_STREQ("bar", d["foo"][1].GetString());
|
|
|
|
}
|
2015-04-10 19:28:12 +08:00
|
|
|
|
|
|
|
TEST(Pointer, CreateValueByPointer) {
|
|
|
|
Document d;
|
|
|
|
Document::AllocatorType& a = d.GetAllocator();
|
|
|
|
|
|
|
|
{
|
|
|
|
Value& v = CreateValueByPointer(d, Pointer("/foo/0"), a);
|
|
|
|
EXPECT_EQ(&d["foo"][0], &v);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
Value& v = CreateValueByPointer(d, "/foo/1", a);
|
|
|
|
EXPECT_EQ(&d["foo"][1], &v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, GetValueByPointer) {
|
|
|
|
Document d;
|
|
|
|
d.Parse(kJson);
|
|
|
|
|
|
|
|
EXPECT_EQ(&d["foo"][0], GetValueByPointer(d, Pointer("/foo/0")));
|
|
|
|
EXPECT_EQ(&d["foo"][0], GetValueByPointer(d, "/foo/0"));
|
|
|
|
|
|
|
|
// const version
|
|
|
|
const Value& v = d;
|
|
|
|
EXPECT_EQ(&d["foo"][0], GetValueByPointer(v, Pointer("/foo/0")));
|
|
|
|
EXPECT_EQ(&d["foo"][0], GetValueByPointer(v, "/foo/0"));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, GetValueByPointerWithDefault) {
|
|
|
|
Document d;
|
|
|
|
d.Parse(kJson);
|
|
|
|
|
|
|
|
Document::AllocatorType& a = d.GetAllocator();
|
|
|
|
const Value v("qux");
|
|
|
|
EXPECT_TRUE(Value("bar") == GetValueByPointerWithDefault(d, Pointer("/foo/0"), v, a));
|
|
|
|
EXPECT_TRUE(Value("baz") == GetValueByPointerWithDefault(d, "/foo/1", v, a));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, SetValueByPointer) {
|
|
|
|
Document d;
|
|
|
|
d.Parse(kJson);
|
|
|
|
Document::AllocatorType& a = d.GetAllocator();
|
|
|
|
|
|
|
|
SetValueByPointer(d, Pointer("/foo/0"), Value(123).Move(), a);
|
|
|
|
EXPECT_EQ(123, d["foo"][0].GetInt());
|
|
|
|
|
|
|
|
SetValueByPointer(d, "/foo/2", Value(456).Move(), a);
|
|
|
|
EXPECT_EQ(456, d["foo"][2].GetInt());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Pointer, SwapValueByPointer) {
|
|
|
|
Document d;
|
|
|
|
d.Parse(kJson);
|
|
|
|
Document::AllocatorType& a = d.GetAllocator();
|
|
|
|
SwapValueByPointer(d, Pointer("/foo/0"), *GetValueByPointer(d, "/foo/1"), a);
|
|
|
|
EXPECT_STREQ("baz", d["foo"][0].GetString());
|
|
|
|
EXPECT_STREQ("bar", d["foo"][1].GetString());
|
|
|
|
|
|
|
|
SwapValueByPointer(d, "/foo/0", *GetValueByPointer(d, "/foo/1"), a);
|
|
|
|
EXPECT_STREQ("bar", d["foo"][0].GetString());
|
|
|
|
EXPECT_STREQ("baz", d["foo"][1].GetString());
|
|
|
|
}
|