| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285 | #include "ValueBaseJsonParser.h"#include <cppunit/extensions/HelperMacros.h>#include "RecoverableException.h"#include "array_fun.h"#include "ValueBase.h"namespace aria2 {class ValueBaseJsonParserTest : public CppUnit::TestFixture {  CPPUNIT_TEST_SUITE(ValueBaseJsonParserTest);  CPPUNIT_TEST(testParseUpdate);  CPPUNIT_TEST(testParseUpdate_error);  CPPUNIT_TEST_SUITE_END();private:public:  void testParseUpdate();  void testParseUpdate_error();};CPPUNIT_TEST_SUITE_REGISTRATION(ValueBaseJsonParserTest);void ValueBaseJsonParserTest::testParseUpdate(){  json::ValueBaseJsonParser parser;  ssize_t error;  {    // empty object    std::string src = "{}";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto dict = downcast<Dict>(r);    CPPUNIT_ASSERT(dict);  }  {    // empty object    std::string src = "{  }";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto dict = downcast<Dict>(r);    CPPUNIT_ASSERT(dict);  }  {    // empty array    std::string src = "[]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);  }  {    // empty array    std::string src = "[ ]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);  }  {    // empty string    std::string src = "[\"\"]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    auto s = downcast<String>(list->get(0));    CPPUNIT_ASSERT_EQUAL(std::string(), s->s());  }  {    // string    std::string src = "[\"foobar\"]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    auto s = downcast<String>(list->get(0));    CPPUNIT_ASSERT_EQUAL(std::string("foobar"), s->s());  }  {    // string with escape    std::string src = "[\"\\\\foo\\\"\\\"bar\"]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    auto s = downcast<String>(list->get(0));    CPPUNIT_ASSERT_EQUAL(std::string("\\foo\"\"bar"), s->s());  }  {    // string with escape    std::string src = "[\"foo\\\"\"]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    auto s = downcast<String>(list->get(0));    CPPUNIT_ASSERT_EQUAL(std::string("foo\""), s->s());  }  {    // string: utf-8 1 to 3 bytes.    std::string src = "[\"\\u0024\\u00A2\\u20AC\"]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    auto s = downcast<String>(list->get(0));    CPPUNIT_ASSERT_EQUAL(std::string("$¢€"), s->s());  }  {    // string: utf-8 4 bytes    std::string src = "[\"\\uD852\\uDF62\"]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    auto s = downcast<String>(list->get(0));    const unsigned char arr[] = {0xF0u, 0xA4u, 0xADu, 0xA2u};    CPPUNIT_ASSERT_EQUAL(std::string(std::begin(arr), std::end(arr)), s->s());  }  {    // null    std::string src = "[null]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    const Null* s = downcast<Null>(list->get(0));    CPPUNIT_ASSERT(s);  }  {    // true, false    std::string src = "[true, false]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    const Bool* trueValue = downcast<Bool>(list->get(0));    CPPUNIT_ASSERT(trueValue);    CPPUNIT_ASSERT(trueValue->val());    const Bool* falseValue = downcast<Bool>(list->get(1));    CPPUNIT_ASSERT(falseValue);    CPPUNIT_ASSERT(!falseValue->val());  }  {    // object: 1 member    std::string src = "{\"foo\":[\"bar\"]}";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    const Dict* dict = downcast<Dict>(r);    CPPUNIT_ASSERT(dict);    auto list = downcast<List>(dict->get("foo"));    CPPUNIT_ASSERT(list);    auto s = downcast<String>(list->get(0));    CPPUNIT_ASSERT_EQUAL(std::string("bar"), s->s());  }  {    // object: 2 members    // TODO ValueBaseJsonParser does not allow empty dict key    std::string src = "{\"foo\":[\"bar\"], \"alpha\" : \"bravo\"}";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    const Dict* dict = downcast<Dict>(r);    CPPUNIT_ASSERT(dict);    auto list = downcast<List>(dict->get("foo"));    CPPUNIT_ASSERT(list);    auto s = downcast<String>(list->get(0));    CPPUNIT_ASSERT_EQUAL(std::string("bar"), s->s());    auto str = downcast<String>(dict->get("alpha"));    CPPUNIT_ASSERT_EQUAL(std::string("bravo"), str->s());  }  {    // array: 2 values    std::string src = "[\"foo\", {}]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    auto s = downcast<String>(list->get(0));    CPPUNIT_ASSERT_EQUAL(std::string("foo"), s->s());    const Dict* dict = downcast<Dict>(list->get(1));    CPPUNIT_ASSERT(dict);  }  {    // Number: currently we ignore frac and exp    std::string src = "[0,-1,1.2,-1.2e-10,-1e10]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    const Integer* i = downcast<Integer>(list->get(0));    CPPUNIT_ASSERT_EQUAL((Integer::ValueType)0, i->i());    const Integer* i1 = downcast<Integer>(list->get(1));    CPPUNIT_ASSERT_EQUAL((Integer::ValueType)-1, i1->i());    const Integer* i2 = downcast<Integer>(list->get(2));    CPPUNIT_ASSERT_EQUAL((Integer::ValueType)1, i2->i());    const Integer* i3 = downcast<Integer>(list->get(3));    CPPUNIT_ASSERT_EQUAL((Integer::ValueType)-1, i3->i());    const Integer* i4 = downcast<Integer>(list->get(4));    CPPUNIT_ASSERT_EQUAL((Integer::ValueType)-1, i4->i());  }  {    // escape chars: ", \, /, \b, \f, \n, \r, \t    std::string src = "[\"\\\"\\\\\\/\\b\\f\\n\\r\\t\"]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    auto s = downcast<String>(list->get(0));    CPPUNIT_ASSERT_EQUAL(std::string("\"\\/\b\f\n\r\t"), s->s());  }  {    // string: literal + escaped chars.    std::string src = "[\"foo\\u0024b\\u00A2\\u20ACbaz\"]";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    auto s = downcast<String>(list->get(0));    CPPUNIT_ASSERT_EQUAL(std::string("foo$b¢€baz"), s->s());  }  {    // ignore garbage at the end of the input.    std::string src = "[]trail";    auto r = parser.parseFinal(src.c_str(), src.size(), error);    auto list = downcast<List>(r);    CPPUNIT_ASSERT(list);    CPPUNIT_ASSERT_EQUAL((ssize_t)2, error);  }}namespace {void checkDecodeError(const std::string& src){  json::ValueBaseJsonParser parser;  ssize_t error;  auto r = parser.parseFinal(src.c_str(), src.size(), error);  CPPUNIT_ASSERT(!r);  CPPUNIT_ASSERT(error < 0);}} // namespacevoid ValueBaseJsonParserTest::testParseUpdate_error(){  // object  checkDecodeError("{");  // object  checkDecodeError("}");  // object  checkDecodeError("{\"\":");  // object  checkDecodeError("{\"\":\"\",");  // array  checkDecodeError("[");  // array  checkDecodeError("]");  // array  checkDecodeError("[\"\"");  // array  checkDecodeError("[\"\",");  // string  checkDecodeError("[\"foo]");  // string  checkDecodeError("[\"\\u\"]");  // string  checkDecodeError("[\"\\u");  // string  checkDecodeError("[\"\\u000\"]");  // string  checkDecodeError("[\"\\u000");  // string  checkDecodeError("[\"\\uD852foo\"]");  // string  checkDecodeError("[\"\\uD852");  // string  checkDecodeError("[\"\\uD852\\u\"]");  // string  checkDecodeError("[\"\\uD852\\u");  // string  checkDecodeError("[\"\\uD852\\u0000\"]");  // string  checkDecodeError("[\"\\uD852\\uDF62");  // object  checkDecodeError("{:\"\"}");  // object  checkDecodeError("{\"foo\":}");  // number  // TODO ValueBaseJsonParser allows leading zeros  // checkDecodeError("[00]");  // number  checkDecodeError("[1.]");  // number  checkDecodeError("[1.1e]");  // bool  checkDecodeError("[t");  // too deep structure  checkDecodeError(std::string(51, '[') + std::string(51, ']'));  checkDecodeError(std::string(50, '[') + "{\"foo\":100}" +                   std::string(50, ']'));}} // namespace aria2
 |