// Copyright (C) 2016-2017 Internet Systems Consortium, Inc. ("ISC") // // This Source Code Form is subject to the terms of the Mozilla Public // License, v. 2.0. If a copy of the MPL was not distributed with this // file, You can obtain one at http://mozilla.org/MPL/2.0/. #include #include #include #include using namespace isc::data; using namespace std; using namespace isc::test; namespace isc { namespace dhcp { namespace test { /// @brief compares two JSON trees /// /// If differences are discovered, gtest failure is reported (using EXPECT_EQ) /// /// @param a first to be compared /// @param b second to be compared void compareJSON(ConstElementPtr a, ConstElementPtr b) { ASSERT_TRUE(a); ASSERT_TRUE(b); EXPECT_EQ(a->str(), b->str()); } /// @brief Tests if the input string can be parsed with specific parser /// /// The input text will be passed to bison parser of specified type. /// Then the same input text is passed to legacy JSON parser and outputs /// from both parsers are compared. The legacy comparison can be disabled, /// if the feature tested is not supported by the old parser (e.g. /// new comment styles) /// /// @param txt text to be compared /// @param parser_type bison parser type to be instantiated /// @param compare whether to compare the output with legacy JSON parser void testParser(const std::string& txt, Parser6Context::ParserType parser_type, bool compare = true) { ConstElementPtr test_json; ASSERT_NO_THROW({ try { Parser6Context ctx; test_json = ctx.parseString(txt, parser_type); } catch (const std::exception &e) { cout << "EXCEPTION: " << e.what() << endl; throw; } }); if (!compare) { return; } // Now compare if both representations are the same. ElementPtr reference_json; ASSERT_NO_THROW(reference_json = Element::fromJSON(txt, true)); compareJSON(reference_json, test_json); } TEST(ParserTest, mapInMap) { string txt = "{ \"xyzzy\": { \"foo\": 123, \"baz\": 456 } }"; testParser(txt, Parser6Context::PARSER_JSON); } TEST(ParserTest, listInList) { string txt = "[ [ \"Britain\", \"Wales\", \"Scotland\" ], " "[ \"Pomorze\", \"Wielkopolska\", \"Tatry\"] ]"; testParser(txt, Parser6Context::PARSER_JSON); } TEST(ParserTest, nestedMaps) { string txt = "{ \"europe\": { \"UK\": { \"London\": { \"street\": \"221B Baker\" }}}}"; testParser(txt, Parser6Context::PARSER_JSON); } TEST(ParserTest, nestedLists) { string txt = "[ \"half\", [ \"quarter\", [ \"eighth\", [ \"sixteenth\" ]]]]"; testParser(txt, Parser6Context::PARSER_JSON); } TEST(ParserTest, listsInMaps) { string txt = "{ \"constellations\": { \"orion\": [ \"rigel\", \"betelgeuse\" ], " "\"cygnus\": [ \"deneb\", \"albireo\"] } }"; testParser(txt, Parser6Context::PARSER_JSON); } TEST(ParserTest, mapsInLists) { string txt = "[ { \"body\": \"earth\", \"gravity\": 1.0 }," " { \"body\": \"mars\", \"gravity\": 0.376 } ]"; testParser(txt, Parser6Context::PARSER_JSON); } TEST(ParserTest, types) { string txt = "{ \"string\": \"foo\"," "\"integer\": 42," "\"boolean\": true," "\"map\": { \"foo\": \"bar\" }," "\"list\": [ 1, 2, 3 ]," "\"null\": null }"; testParser(txt, Parser6Context::PARSER_JSON); } TEST(ParserTest, keywordJSON) { string txt = "{ \"name\": \"user\"," "\"type\": \"password\"," "\"user\": \"name\"," "\"password\": \"type\" }"; testParser(txt, Parser6Context::PARSER_JSON); } TEST(ParserTest, keywordDhcp6) { string txt = "{ \"Dhcp6\": { \"interfaces-config\": {" " \"interfaces\": [ \"type\", \"htype\" ] },\n" "\"preferred-lifetime\": 3000,\n" "\"rebind-timer\": 2000, \n" "\"renew-timer\": 1000, \n" "\"subnet6\": [ { " " \"pools\": [ { \"pool\": \"2001:db8:1::/64\" } ]," " \"subnet\": \"2001:db8:1::/48\", " " \"interface\": \"test\" } ],\n" "\"valid-lifetime\": 4000 } }"; testParser(txt, Parser6Context::PARSER_DHCP6); } // Tests if bash (#) comments are supported. That's the only comment type that // was supported by the old parser. TEST(ParserTest, bashComments) { string txt= "{ \"Dhcp6\": { \"interfaces-config\": {" " \"interfaces\": [ \"*\" ]" "},\n" "\"preferred-lifetime\": 3000,\n" "# this is a comment\n" "\"rebind-timer\": 2000, \n" "# lots of comments here\n" "# and here\n" "\"renew-timer\": 1000, \n" "\"subnet6\": [ { " " \"pools\": [ { \"pool\": \"2001:db8:1::/64\" } ]," " \"subnet\": \"2001:db8:1::/48\", " " \"interface\": \"eth0\"" " } ]," "\"valid-lifetime\": 4000 } }"; testParser(txt, Parser6Context::PARSER_DHCP6); } // Tests if C++ (//) comments can start anywhere, not just in the first line. TEST(ParserTest, cppComments) { string txt= "{ \"Dhcp6\": { \"interfaces-config\": {" " \"interfaces\": [ \"*\" ]" "},\n" "\"preferred-lifetime\": 3000, // this is a comment \n" "\"rebind-timer\": 2000, // everything after // is ignored\n" "\"renew-timer\": 1000, // this will be ignored, too\n" "\"subnet6\": [ { " " \"pools\": [ { \"pool\": \"2001:db8:1::/64\" } ]," " \"subnet\": \"2001:db8:1::/48\", " " \"interface\": \"eth0\"" " } ]," "\"valid-lifetime\": 4000 } }"; testParser(txt, Parser6Context::PARSER_DHCP6, false); } // Tests if bash (#) comments can start anywhere, not just in the first line. TEST(ParserTest, bashCommentsInline) { string txt= "{ \"Dhcp6\": { \"interfaces-config\": {" " \"interfaces\": [ \"*\" ]" "},\n" "\"preferred-lifetime\": 3000, # this is a comment \n" "\"rebind-timer\": 2000, # everything after # is ignored\n" "\"renew-timer\": 1000, # this will be ignored, too\n" "\"subnet6\": [ { " " \"pools\": [ { \"pool\": \"2001:db8:1::/64\" } ]," " \"subnet\": \"2001:db8:1::/48\", " " \"interface\": \"eth0\"" " } ]," "\"valid-lifetime\": 4000 } }"; testParser(txt, Parser6Context::PARSER_DHCP6, false); } // Tests if multi-line C style comments are handled correctly. TEST(ParserTest, multilineComments) { string txt= "{ \"Dhcp6\": { \"interfaces-config\": {" " \"interfaces\": [ \"*\" ]" "},\n" "\"preferred-lifetime\": 3000, /* this is a C style comment\n" "that\n can \n span \n multiple \n lines */ \n" "\"rebind-timer\": 2000,\n" "\"renew-timer\": 1000, \n" "\"subnet6\": [ { " " \"pools\": [ { \"pool\": \"2001:db8:1::/64\" } ]," " \"subnet\": \"2001:db8:1::/48\", " " \"interface\": \"eth0\"" " } ]," "\"valid-lifetime\": 4000 } }"; testParser(txt, Parser6Context::PARSER_DHCP6, false); } /// @brief Loads specified example config file /// /// This test loads specified example file twice: first, using the legacy /// JSON file and then second time using bison parser. Two created Element /// trees are then compared. The input is decommented before it is passed /// to legacy parser (as legacy support for comments is very limited). /// /// @param fname name of the file to be loaded void testFile(const std::string& fname) { ElementPtr reference_json; ConstElementPtr test_json; string decommented = decommentJSONfile(fname); cout << "Parsing file " << fname << "(" << decommented << ")" << endl; EXPECT_NO_THROW(reference_json = Element::fromJSONFile(decommented, true)); // remove the temporary file EXPECT_NO_THROW(::remove(decommented.c_str())); EXPECT_NO_THROW( try { Parser6Context ctx; test_json = ctx.parseFile(fname, Parser6Context::PARSER_DHCP6); } catch (const std::exception &x) { cout << "EXCEPTION: " << x.what() << endl; throw; }); ASSERT_TRUE(reference_json); ASSERT_TRUE(test_json); compareJSON(reference_json, test_json); } // This test loads all available existing files. Each config is loaded // twice: first with the existing Element::fromJSONFile() and then // the second time with Parser6. Both JSON trees are then compared. TEST(ParserTest, file) { vector configs; configs.push_back("advanced.json"); configs.push_back("backends.json"); configs.push_back("classify.json"); configs.push_back("dhcpv4-over-dhcpv6.json"); configs.push_back("duid.json"); configs.push_back("hooks.json"); configs.push_back("leases-expiration.json"); configs.push_back("multiple-options.json"); configs.push_back("mysql-reservations.json"); configs.push_back("pgsql-reservations.json"); configs.push_back("reservations.json"); configs.push_back("several-subnets.json"); configs.push_back("simple.json"); configs.push_back("stateless.json"); configs.push_back("with-ddns.json"); for (int i = 0; i:1.1: syntax error, unexpected end of file"); testError(" ", Parser6Context::PARSER_JSON, ":1.2: syntax error, unexpected end of file"); testError("\n", Parser6Context::PARSER_JSON, ":2.1: syntax error, unexpected end of file"); testError("\t", Parser6Context::PARSER_JSON, ":1.2: syntax error, unexpected end of file"); testError("\r", Parser6Context::PARSER_JSON, ":1.2: syntax error, unexpected end of file"); // comments testError("# nothing\n", Parser6Context::PARSER_JSON, ":2.1: syntax error, unexpected end of file"); testError(" #\n", Parser6Context::PARSER_JSON, ":2.1: syntax error, unexpected end of file"); testError("// nothing\n", Parser6Context::PARSER_JSON, ":2.1: syntax error, unexpected end of file"); testError("/* nothing */\n", Parser6Context::PARSER_JSON, ":2.1: syntax error, unexpected end of file"); testError("/* no\nthing */\n", Parser6Context::PARSER_JSON, ":3.1: syntax error, unexpected end of file"); testError("/* no\nthing */\n\n", Parser6Context::PARSER_JSON, ":4.1: syntax error, unexpected end of file"); testError("/* nothing\n", Parser6Context::PARSER_JSON, "Comment not closed. (/* in line 1"); testError("\n\n\n/* nothing\n", Parser6Context::PARSER_JSON, "Comment not closed. (/* in line 4"); testError("{ /* */*/ }\n", Parser6Context::PARSER_JSON, ":1.3-8: Invalid character: *"); testError("{ /* // *// }\n", Parser6Context::PARSER_JSON, ":1.3-11: Invalid character: /"); testError("{ /* // */// }\n", Parser6Context::PARSER_JSON, ":2.1: syntax error, unexpected end of file, " "expecting }"); // includes testError("/n", Parser6Context::PARSER_JSON, "Can't open include file /foo/bar"); // JSON keywords testError("{ \"foo\": True }", Parser6Context::PARSER_JSON, ":1.10-13: JSON true reserved keyword is lower case only"); testError("{ \"foo\": False }", Parser6Context::PARSER_JSON, ":1.10-14: JSON false reserved keyword is lower case only"); testError("{ \"foo\": NULL }", Parser6Context::PARSER_JSON, ":1.10-13: JSON null reserved keyword is lower case only"); testError("{ \"foo\": Tru }", Parser6Context::PARSER_JSON, ":1.10: Invalid character: T"); testError("{ \"foo\": nul }", Parser6Context::PARSER_JSON, ":1.10: Invalid character: n"); // numbers testError("123", Parser6Context::PARSER_DHCP6, ":1.1-3: syntax error, unexpected integer, " "expecting {"); testError("-456", Parser6Context::PARSER_DHCP6, ":1.1-4: syntax error, unexpected integer, " "expecting {"); testError("-0001", Parser6Context::PARSER_DHCP6, ":1.1-5: syntax error, unexpected integer, " "expecting {"); testError("1234567890123456789012345678901234567890", Parser6Context::PARSER_JSON, ":1.1-40: Failed to convert " "1234567890123456789012345678901234567890" " to an integer."); testError("-3.14e+0", Parser6Context::PARSER_DHCP6, ":1.1-8: syntax error, unexpected floating point, " "expecting {"); testError("1e50000", Parser6Context::PARSER_JSON, ":1.1-7: Failed to convert 1e50000 " "to a floating point."); // strings testError("\"aabb\"", Parser6Context::PARSER_DHCP6, ":1.1-6: syntax error, unexpected constant string, " "expecting {"); testError("{ \"aabb\"err", Parser6Context::PARSER_JSON, ":1.9: Invalid character: e"); testError("{ err\"aabb\"", Parser6Context::PARSER_JSON, ":1.3: Invalid character: e"); testError("\"a\n\tb\"", Parser6Context::PARSER_JSON, ":1.1-6: Invalid control in \"a\n\tb\""); testError("\"a\\n\\tb\"", Parser6Context::PARSER_DHCP6, ":1.1-8: syntax error, unexpected constant string, " "expecting {"); testError("\"a\\x01b\"", Parser6Context::PARSER_JSON, ":1.1-8: Bad escape in \"a\\x01b\""); testError("\"a\\u0162\"", Parser6Context::PARSER_JSON, ":1.1-9: Unsupported unicode escape in \"a\\u0162\""); testError("\"a\\u062z\"", Parser6Context::PARSER_JSON, ":1.1-9: Bad escape in \"a\\u062z\""); testError("\"abc\\\"", Parser6Context::PARSER_JSON, ":1.1-6: Overflow escape in \"abc\\\""); // from data_unittest.c testError("\\a", Parser6Context::PARSER_JSON, ":1.1: Invalid character: \\"); testError("\\", Parser6Context::PARSER_JSON, ":1.1: Invalid character: \\"); testError("\\\"\\\"", Parser6Context::PARSER_JSON, ":1.1: Invalid character: \\"); // want a map testError("[]\n", Parser6Context::PARSER_DHCP6, ":1.1: syntax error, unexpected [, " "expecting {"); testError("[]\n", Parser6Context::PARSER_DHCP6, ":1.1: syntax error, unexpected [, " "expecting {"); testError("{ 123 }\n", Parser6Context::PARSER_JSON, ":1.3-5: syntax error, unexpected integer, " "expecting }"); testError("{ 123 }\n", Parser6Context::PARSER_DHCP6, ":1.3-5: syntax error, unexpected integer"); testError("{ \"foo\" }\n", Parser6Context::PARSER_JSON, ":1.9: syntax error, unexpected }, " "expecting :"); testError("{ \"foo\" }\n", Parser6Context::PARSER_DHCP6, ":1.9: syntax error, unexpected }, expecting :"); testError("{ \"foo\":null }\n", Parser6Context::PARSER_DHCP6, ":1.3-7: got unexpected keyword " "\"foo\" in toplevel map."); testError("{ \"Dhcp6\" }\n", Parser6Context::PARSER_DHCP6, ":1.11: syntax error, unexpected }, " "expecting :"); testError("{ \"Dhcp4\":[]\n", Parser6Context::PARSER_DHCP6, ":2.1: syntax error, unexpected end of file, " "expecting \",\" or }"); testError("{}{}\n", Parser6Context::PARSER_JSON, ":1.3: syntax error, unexpected {, " "expecting end of file"); // bad commas testError("{ , }\n", Parser6Context::PARSER_JSON, ":1.3: syntax error, unexpected \",\", " "expecting }"); testError("{ , \"foo\":true }\n", Parser6Context::PARSER_JSON, ":1.3: syntax error, unexpected \",\", " "expecting }"); testError("{ \"foo\":true, }\n", Parser6Context::PARSER_JSON, ":1.15: syntax error, unexpected }, " "expecting constant string"); // bad type testError("{ \"Dhcp6\":{\n" " \"preferred-lifetime\":false }}\n", Parser6Context::PARSER_DHCP6, ":2.24-28: syntax error, unexpected boolean, " "expecting integer"); // unknown keyword testError("{ \"Dhcp6\":{\n" " \"preferred_lifetime\":600 }}\n", Parser6Context::PARSER_DHCP6, ":2.2-21: got unexpected keyword " "\"preferred_lifetime\" in Dhcp6 map."); // missing parameter testError("{ \"name\": \"foo\",\n" " \"code\": 123 }\n", Parser6Context::PARSER_OPTION_DEF, "missing parameter 'type' (:1:1) " "[option-def map between :1:1 and :2:15]"); } // Check unicode escapes TEST(ParserTest, unicodeEscapes) { ConstElementPtr result; string json; // check we can reread output for (char c = -128; c < 127; ++c) { string ins(" "); ins[1] = c; ConstElementPtr e(new StringElement(ins)); json = e->str(); ASSERT_NO_THROW( try { Parser6Context ctx; result = ctx.parseString(json, Parser6Context::PARSER_JSON); } catch (const std::exception &x) { cout << "EXCEPTION: " << x.what() << endl; throw; }); ASSERT_EQ(Element::string, result->getType()); EXPECT_EQ(ins, result->stringValue()); } } // This test checks that all representations of a slash is recognized properly. TEST(ParserTest, unicodeSlash) { // check the 4 possible encodings of solidus '/' ConstElementPtr result; string json = "\"/\\/\\u002f\\u002F\""; ASSERT_NO_THROW( try { Parser6Context ctx; result = ctx.parseString(json, Parser6Context::PARSER_JSON); } catch (const std::exception &x) { cout << "EXCEPTION: " << x.what() << endl; throw; }); ASSERT_EQ(Element::string, result->getType()); EXPECT_EQ("////", result->stringValue()); } }; }; };