BencodeTest.cc 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. #include "bencode.h"
  2. #include <cstring>
  3. #include <iostream>
  4. #include <cppunit/extensions/HelperMacros.h>
  5. namespace aria2 {
  6. class BencodeTest:public CppUnit::TestFixture {
  7. CPPUNIT_TEST_SUITE(BencodeTest);
  8. CPPUNIT_TEST(testString);
  9. CPPUNIT_TEST(testInteger);
  10. CPPUNIT_TEST(testDict);
  11. CPPUNIT_TEST(testDictIter);
  12. CPPUNIT_TEST(testList);
  13. CPPUNIT_TEST(testListIter);
  14. CPPUNIT_TEST(testDecode);
  15. CPPUNIT_TEST(testEncode);
  16. CPPUNIT_TEST_SUITE_END();
  17. private:
  18. public:
  19. void testString();
  20. void testInteger();
  21. void testDict();
  22. void testDictIter();
  23. void testList();
  24. void testListIter();
  25. void testDecode();
  26. void testEncode();
  27. };
  28. CPPUNIT_TEST_SUITE_REGISTRATION( BencodeTest );
  29. void BencodeTest::testString()
  30. {
  31. bencode::BDE s(std::string("aria2"));
  32. CPPUNIT_ASSERT_EQUAL(std::string("aria2"), s.s());
  33. unsigned char dataWithNull[] = { 0xf0, '\0', 0x0f };
  34. bencode::BDE sWithNull(dataWithNull, sizeof(dataWithNull));
  35. CPPUNIT_ASSERT(memcmp(dataWithNull, sWithNull.s().c_str(),
  36. sizeof(dataWithNull)) == 0);
  37. bencode::BDE zero("");
  38. CPPUNIT_ASSERT_EQUAL(std::string(""), zero.s());
  39. }
  40. void BencodeTest::testInteger()
  41. {
  42. bencode::BDE integer(INT64_MAX);
  43. CPPUNIT_ASSERT_EQUAL(INT64_MAX, integer.i());
  44. }
  45. void BencodeTest::testDict()
  46. {
  47. bencode::BDE dict = bencode::BDE::dict();
  48. CPPUNIT_ASSERT(dict.empty());
  49. dict["ki"] = 7;
  50. dict["ks"] = std::string("abc");
  51. CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(2), dict.size());
  52. CPPUNIT_ASSERT(dict.containsKey("ki"));
  53. CPPUNIT_ASSERT_EQUAL(static_cast<bencode::BDE::Integer>(7), dict["ki"].i());
  54. CPPUNIT_ASSERT(dict.containsKey("ks"));
  55. CPPUNIT_ASSERT_EQUAL(std::string("abc"), dict["ks"].s());
  56. CPPUNIT_ASSERT(dict["kn"].isNone()); // This adds kn key with default value.
  57. CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(3), dict.size());
  58. CPPUNIT_ASSERT(dict.containsKey("kn"));
  59. const bencode::BDE& ref = dict;
  60. ref["kn2"]; // This doesn't add kn2 key.
  61. CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(3), ref.size());
  62. CPPUNIT_ASSERT(!ref.containsKey("kn2"));
  63. }
  64. void BencodeTest::testDictIter()
  65. {
  66. bencode::BDE dict = bencode::BDE::dict();
  67. dict["alpha2"] = std::string("alpha2");
  68. dict["charlie"] = std::string("charlie");
  69. dict["bravo"] = std::string("bravo");
  70. dict["alpha"] = std::string("alpha");
  71. bencode::BDE::Dict::iterator i = dict.dictBegin();
  72. CPPUNIT_ASSERT_EQUAL(std::string("alpha"), (*i++).first);
  73. CPPUNIT_ASSERT_EQUAL(std::string("alpha2"), (*i++).first);
  74. CPPUNIT_ASSERT_EQUAL(std::string("bravo"), (*i++).first);
  75. CPPUNIT_ASSERT_EQUAL(std::string("charlie"), (*i++).first);
  76. CPPUNIT_ASSERT(dict.dictEnd() == i);
  77. const bencode::BDE& ref = dict;
  78. bencode::BDE::Dict::const_iterator ci = ref.dictBegin();
  79. CPPUNIT_ASSERT_EQUAL(std::string("alpha"), (*ci++).first);
  80. std::advance(ci, 3);
  81. CPPUNIT_ASSERT(ref.dictEnd() == ci);
  82. }
  83. void BencodeTest::testList()
  84. {
  85. bencode::BDE list = bencode::BDE::list();
  86. CPPUNIT_ASSERT(list.empty());
  87. list << 7;
  88. list << std::string("aria2");
  89. CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(2), list.size());
  90. CPPUNIT_ASSERT_EQUAL(static_cast<bencode::BDE::Integer>(7), list[0].i());
  91. CPPUNIT_ASSERT_EQUAL(std::string("aria2"), list[1].s());
  92. const bencode::BDE& ref = list;
  93. CPPUNIT_ASSERT_EQUAL(static_cast<bencode::BDE::Integer>(7), ref[0].i());
  94. CPPUNIT_ASSERT_EQUAL(std::string("aria2"), ref[1].s());
  95. }
  96. void BencodeTest::testListIter()
  97. {
  98. bencode::BDE list = bencode::BDE::list();
  99. list << std::string("alpha2");
  100. list << std::string("charlie");
  101. list << std::string("bravo");
  102. list << std::string("alpha");
  103. bencode::BDE::List::iterator i = list.listBegin();
  104. CPPUNIT_ASSERT_EQUAL(std::string("alpha2"), (*i++).s());
  105. CPPUNIT_ASSERT_EQUAL(std::string("charlie"), (*i++).s());
  106. CPPUNIT_ASSERT_EQUAL(std::string("bravo"), (*i++).s());
  107. CPPUNIT_ASSERT_EQUAL(std::string("alpha"), (*i++).s());
  108. CPPUNIT_ASSERT(list.listEnd() == i);
  109. const bencode::BDE& ref = list;
  110. bencode::BDE::List::const_iterator ci = ref.listBegin();
  111. CPPUNIT_ASSERT_EQUAL(std::string("alpha2"), (*ci++).s());
  112. std::advance(ci, 3);
  113. CPPUNIT_ASSERT(ref.listEnd() == ci);
  114. }
  115. void BencodeTest::testDecode()
  116. {
  117. {
  118. // string, integer and list in dict
  119. bencode::BDE dict =
  120. bencode::decode("d4:name5:aria24:sizei12345678900e5:filesl3:bin3:docee");
  121. CPPUNIT_ASSERT(dict.isDict());
  122. CPPUNIT_ASSERT_EQUAL(std::string("aria2"), dict["name"].s());
  123. CPPUNIT_ASSERT_EQUAL(static_cast<bencode::BDE::Integer>(12345678900LL),
  124. dict["size"].i());
  125. bencode::BDE list = dict["files"];
  126. CPPUNIT_ASSERT(list.isList());
  127. CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(2), list.size());
  128. CPPUNIT_ASSERT_EQUAL(std::string("bin"), list[0].s());
  129. CPPUNIT_ASSERT_EQUAL(std::string("doc"), list[1].s());
  130. }
  131. {
  132. // dict in list
  133. bencode::BDE list = bencode::decode("ld1:ki123eee");
  134. CPPUNIT_ASSERT(list.isList());
  135. CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(1), list.size());
  136. bencode::BDE dict = list[0];
  137. CPPUNIT_ASSERT(dict.isDict());
  138. CPPUNIT_ASSERT_EQUAL(static_cast<bencode::BDE::Integer>(123),
  139. dict["k"].i());
  140. }
  141. {
  142. // empty key is allowed
  143. bencode::BDE s = bencode::decode("d0:1:ve");
  144. }
  145. {
  146. // empty string
  147. bencode::BDE s = bencode::decode("0:");
  148. CPPUNIT_ASSERT_EQUAL(std::string(""), s.s());
  149. }
  150. {
  151. // empty dict
  152. bencode::BDE d = bencode::decode("de");
  153. CPPUNIT_ASSERT(d.empty());
  154. }
  155. {
  156. // empty list
  157. bencode::BDE l = bencode::decode("le");
  158. CPPUNIT_ASSERT(l.empty());
  159. }
  160. {
  161. // integer, without ending 'e'
  162. try {
  163. bencode::decode("i3");
  164. CPPUNIT_FAIL("exception must be thrown.");
  165. } catch(RecoverableException& e) {
  166. CPPUNIT_ASSERT_EQUAL(std::string("Delimiter 'e' not found."),
  167. std::string(e.what()));
  168. }
  169. }
  170. {
  171. // dict, without ending 'e'
  172. try {
  173. bencode::decode("d");
  174. CPPUNIT_FAIL("exception must be thrown.");
  175. } catch(RecoverableException& e) {
  176. CPPUNIT_ASSERT_EQUAL(std::string("Unexpected EOF in dict context."
  177. " 'e' expected."),
  178. std::string(e.what()));
  179. }
  180. }
  181. {
  182. // list, without ending 'e'
  183. try {
  184. bencode::decode("l");
  185. CPPUNIT_FAIL("exception must be thrown.");
  186. } catch(RecoverableException& e) {
  187. CPPUNIT_ASSERT_EQUAL(std::string("Unexpected EOF in list context."
  188. " 'e' expected."),
  189. std::string(e.what()));
  190. }
  191. }
  192. {
  193. // string, less than the specified length.
  194. try {
  195. bencode::decode("3:ab");
  196. CPPUNIT_FAIL("exception must be thrown.");
  197. } catch(RecoverableException& e) {
  198. CPPUNIT_ASSERT_EQUAL(std::string("Expected 3 bytes of data,"
  199. " but only 2 read."),
  200. std::string(e.what()));
  201. }
  202. }
  203. {
  204. // string, but length is invalid
  205. try {
  206. bencode::decode("x:abc");
  207. CPPUNIT_FAIL("exception must be thrown.");
  208. } catch(RecoverableException& e) {
  209. CPPUNIT_ASSERT_EQUAL(std::string("Integer expected but none found."),
  210. std::string(e.what()));
  211. }
  212. }
  213. {
  214. // empty encoded data
  215. CPPUNIT_ASSERT(bencode::decode("").isNone());
  216. }
  217. {
  218. // ignore trailing garbage at the end of the input.
  219. bencode::BDE s = bencode::decode("5:aria2trail");
  220. CPPUNIT_ASSERT_EQUAL(std::string("aria2"), s.s());
  221. }
  222. }
  223. void BencodeTest::testEncode()
  224. {
  225. {
  226. bencode::BDE dict = bencode::BDE::dict();
  227. dict["name"] = std::string("aria2");
  228. dict["loc"] = 80000;
  229. dict["files"] = bencode::BDE::list();
  230. dict["files"] << std::string("aria2c");
  231. dict["attrs"] = bencode::BDE::dict();
  232. dict["attrs"]["license"] = std::string("GPL");
  233. CPPUNIT_ASSERT_EQUAL(std::string("d"
  234. "5:attrsd7:license3:GPLe"
  235. "5:filesl6:aria2ce"
  236. "3:loci80000e"
  237. "4:name5:aria2"
  238. "e"),
  239. bencode::encode(dict));
  240. }
  241. }
  242. } // namespace aria2