BencodeTest.cc 8.3 KB

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