#include "ValueBaseBencodeParser.h" #include #include "ValueBase.h" namespace aria2 { class ValueBaseBencodeParserTest:public CppUnit::TestFixture { CPPUNIT_TEST_SUITE(ValueBaseBencodeParserTest); CPPUNIT_TEST(testParseUpdate); CPPUNIT_TEST_SUITE_END(); public: void testParseUpdate(); }; CPPUNIT_TEST_SUITE_REGISTRATION( ValueBaseBencodeParserTest ); namespace { void checkDecodeError(const std::string& src) { bittorrent::ValueBaseBencodeParser parser; ssize_t error; SharedHandle r = parser.parseFinal(src.c_str(), src.size(), error); CPPUNIT_ASSERT(!r); CPPUNIT_ASSERT(error < 0); } } // namespace void ValueBaseBencodeParserTest::testParseUpdate() { bittorrent::ValueBaseBencodeParser parser; ssize_t error; { // empty string std::string src = "0:"; SharedHandle s = parser.parseFinal(src.c_str(), src.size(), error); CPPUNIT_ASSERT_EQUAL(std::string(""), downcast(s)->s()); } { // integer 0 std::string src = "i0e"; SharedHandle s = parser.parseFinal(src.c_str(), src.size(), error); CPPUNIT_ASSERT_EQUAL((int64_t)0, downcast(s)->i()); } { // empty dict std::string src = "de"; SharedHandle d = parser.parseFinal(src.c_str(), src.size(), error); CPPUNIT_ASSERT(downcast(d)->empty()); } { // empty list std::string src = "le"; SharedHandle l = parser.parseFinal(src.c_str(), src.size(), error); CPPUNIT_ASSERT(downcast(l)->empty()); } { // string std::string src = "3:foo"; SharedHandle s = parser.parseFinal(src.c_str(), src.size(), error); CPPUNIT_ASSERT_EQUAL(std::string("foo"), downcast(s)->s()); } { // integer std::string src = "i9223372036854775807e"; SharedHandle s = parser.parseFinal(src.c_str(), src.size(), error); CPPUNIT_ASSERT_EQUAL((int64_t)9223372036854775807LL, downcast(s)->i()); } { // dict, size 1 std::string src = "d3:fooi123ee"; SharedHandle d = parser.parseFinal(src.c_str(), src.size(), error); Dict* dict = downcast(d); CPPUNIT_ASSERT(dict); CPPUNIT_ASSERT(dict->get("foo")); CPPUNIT_ASSERT_EQUAL((int64_t)123, downcast(dict->get("foo"))->i()); } { // dict, size 2 std::string src = "d3:fooi123e3:bar1:ee"; SharedHandle d = parser.parseFinal(src.c_str(), src.size(), error); Dict* dict = downcast(d); CPPUNIT_ASSERT(dict); CPPUNIT_ASSERT_EQUAL((size_t)2, dict->size()); CPPUNIT_ASSERT(dict->get("foo")); CPPUNIT_ASSERT_EQUAL((int64_t)123, downcast(dict->get("foo"))->i()); CPPUNIT_ASSERT(dict->get("bar")); CPPUNIT_ASSERT_EQUAL(std::string("e"), downcast(dict->get("bar"))->s()); } { // list, size 1 std::string src = "l3:fooe"; SharedHandle l = parser.parseFinal(src.c_str(), src.size(), error); List* list = downcast(l); CPPUNIT_ASSERT(list); CPPUNIT_ASSERT_EQUAL((size_t)1, list->size()); CPPUNIT_ASSERT_EQUAL(std::string("foo"), downcast(list->get(0))->s()); } { // list, size 2 std::string src = "l3:fooi123ee"; SharedHandle l = parser.parseFinal(src.c_str(), src.size(), error); List* list = downcast(l); CPPUNIT_ASSERT(list); CPPUNIT_ASSERT_EQUAL((size_t)2, list->size()); CPPUNIT_ASSERT_EQUAL(std::string("foo"), downcast(list->get(0))->s()); CPPUNIT_ASSERT_EQUAL((int64_t)123, downcast(list->get(1))->i()); } { // string, integer and list in dict std::string src = "d4:name5:aria24:sizei12345678900e5:filesl3:bin3:docee"; SharedHandle r = parser.parseFinal(src.c_str(), src.size(), error); const Dict* dict = downcast(r); CPPUNIT_ASSERT(dict); CPPUNIT_ASSERT_EQUAL(std::string("aria2"), downcast(dict->get("name"))->s()); CPPUNIT_ASSERT_EQUAL(static_cast(12345678900LL), downcast(dict->get("size"))->i()); const List* list = downcast(dict->get("files")); CPPUNIT_ASSERT(list); CPPUNIT_ASSERT_EQUAL(static_cast(2), list->size()); CPPUNIT_ASSERT_EQUAL(std::string("bin"), downcast(list->get(0))->s()); CPPUNIT_ASSERT_EQUAL(std::string("doc"), downcast(list->get(1))->s()); } { // dict in list std::string src = "ld1:ki123eee"; SharedHandle r = parser.parseFinal(src.c_str(), src.size(), error); const List* list = downcast(r); CPPUNIT_ASSERT(list); CPPUNIT_ASSERT_EQUAL(static_cast(1), list->size()); const Dict* dict = downcast(list->get(0)); CPPUNIT_ASSERT(dict); CPPUNIT_ASSERT_EQUAL(static_cast(123), downcast(dict->get("k"))->i()); } { // empty key is allowed std::string src = "d0:1:ve"; SharedHandle s = parser.parseFinal(src.c_str(), src.size(), error); } { // empty encoded data std::string src = ""; SharedHandle s = parser.parseFinal(src.c_str(), src.size(), error); CPPUNIT_ASSERT(!s); } // integer, without ending 'e' checkDecodeError("i3"); // dict, without ending 'e' checkDecodeError("d"); // list, without ending 'e' checkDecodeError("l"); // string, less than the specified length. checkDecodeError("3:ab"); // string, but length is invalid checkDecodeError("x:abc"); // string with minus length checkDecodeError("-1:a"); // too deep structure checkDecodeError(std::string(51, 'l')+std::string(51,'e')); checkDecodeError(std::string(50, 'l')+"d3:fooi100ee"+std::string(50,'e')); { // ignore trailing garbage at the end of the input. std::string src = "5:aria2trail"; SharedHandle s = parser.parseFinal(src.c_str(), src.size(), error); CPPUNIT_ASSERT_EQUAL(std::string("aria2"), downcast(s)->s()); // Get trailing garbage position CPPUNIT_ASSERT_EQUAL((ssize_t)7, error); } } } // namespace aria2