DefaultBtContextTest.cc 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. #include "DefaultBtContext.h"
  2. #include "Util.h"
  3. #include "RecoverableException.h"
  4. #include "AnnounceTier.h"
  5. #include "FixedNumberRandomizer.h"
  6. #include "FileEntry.h"
  7. #include "array_fun.h"
  8. #include <cstring>
  9. #include <iostream>
  10. #include <cppunit/extensions/HelperMacros.h>
  11. namespace aria2 {
  12. class DefaultBtContextTest:public CppUnit::TestFixture {
  13. CPPUNIT_TEST_SUITE(DefaultBtContextTest);
  14. CPPUNIT_TEST(testGetInfoHash);
  15. CPPUNIT_TEST(testGetPieceHash);
  16. CPPUNIT_TEST(testGetFileEntries);
  17. CPPUNIT_TEST(testGetTotalLength);
  18. CPPUNIT_TEST(testGetFileEntriesSingle);
  19. CPPUNIT_TEST(testGetTotalLengthSingle);
  20. CPPUNIT_TEST(testGetFileModeMulti);
  21. CPPUNIT_TEST(testGetFileModeSingle);
  22. CPPUNIT_TEST(testGetNameMulti);
  23. CPPUNIT_TEST(testGetNameSingle);
  24. CPPUNIT_TEST(testGetAnnounceTier);
  25. CPPUNIT_TEST(testGetAnnounceTierAnnounceList);
  26. CPPUNIT_TEST(testGetPieceLength);
  27. CPPUNIT_TEST(testGetInfoHashAsString);
  28. CPPUNIT_TEST(testGetPeerId);
  29. CPPUNIT_TEST(testComputeFastSet);
  30. CPPUNIT_TEST(testGetFileEntries_multiFileUrlList);
  31. CPPUNIT_TEST(testGetFileEntries_singleFileUrlList);
  32. CPPUNIT_TEST(testLoadFromMemory);
  33. CPPUNIT_TEST(testLoadFromMemory_somethingMissing);
  34. CPPUNIT_TEST(testGetNodes);
  35. CPPUNIT_TEST_SUITE_END();
  36. public:
  37. void setUp() {
  38. }
  39. void testGetInfoHash();
  40. void testGetPieceHash();
  41. void testGetFileEntries();
  42. void testGetTotalLength();
  43. void testGetFileEntriesSingle();
  44. void testGetTotalLengthSingle();
  45. void testGetFileModeMulti();
  46. void testGetFileModeSingle();
  47. void testGetNameMulti();
  48. void testGetNameSingle();
  49. void testGetAnnounceTier();
  50. void testGetAnnounceTierAnnounceList();
  51. void testGetPieceLength();
  52. void testGetInfoHashAsString();
  53. void testGetPeerId();
  54. void testComputeFastSet();
  55. void testGetFileEntries_multiFileUrlList();
  56. void testGetFileEntries_singleFileUrlList();
  57. void testLoadFromMemory();
  58. void testLoadFromMemory_somethingMissing();
  59. void testGetNodes();
  60. };
  61. CPPUNIT_TEST_SUITE_REGISTRATION(DefaultBtContextTest);
  62. void DefaultBtContextTest::testGetInfoHash() {
  63. DefaultBtContext btContext;
  64. btContext.load("test.torrent");
  65. std::string correctHash = "248d0a1cd08284299de78d5c1ed359bb46717d8c";
  66. CPPUNIT_ASSERT_EQUAL((size_t)20, btContext.getInfoHashLength());
  67. CPPUNIT_ASSERT_EQUAL(correctHash, Util::toHex(btContext.getInfoHash(),
  68. btContext.getInfoHashLength()));
  69. }
  70. void DefaultBtContextTest::testGetPieceHash() {
  71. DefaultBtContext btContext;
  72. btContext.load("test.torrent");
  73. CPPUNIT_ASSERT_EQUAL(Util::toHex((const unsigned char*)"AAAAAAAAAAAAAAAAAAAA", 20),
  74. btContext.getPieceHash(0));
  75. CPPUNIT_ASSERT_EQUAL(Util::toHex((const unsigned char*)"BBBBBBBBBBBBBBBBBBBB", 20),
  76. btContext.getPieceHash(1));
  77. CPPUNIT_ASSERT_EQUAL(Util::toHex((const unsigned char*)"CCCCCCCCCCCCCCCCCCCC", 20),
  78. btContext.getPieceHash(2));
  79. CPPUNIT_ASSERT_EQUAL(std::string(""),
  80. btContext.getPieceHash(-1));
  81. CPPUNIT_ASSERT_EQUAL(std::string(""),
  82. btContext.getPieceHash(3));
  83. }
  84. void DefaultBtContextTest::testGetFileEntries() {
  85. DefaultBtContext btContext;
  86. btContext.load("test.torrent");
  87. // This is multi-file torrent.
  88. std::deque<SharedHandle<FileEntry> > fileEntries = btContext.getFileEntries();
  89. // There are 2 file entries.
  90. CPPUNIT_ASSERT_EQUAL((size_t)2, fileEntries.size());
  91. std::deque<SharedHandle<FileEntry> >::iterator itr = fileEntries.begin();
  92. SharedHandle<FileEntry> fileEntry1 = *itr;
  93. CPPUNIT_ASSERT_EQUAL(std::string("aria2/src/aria2c"),
  94. fileEntry1->getPath());
  95. itr++;
  96. SharedHandle<FileEntry> fileEntry2 = *itr;
  97. CPPUNIT_ASSERT_EQUAL(std::string("aria2-0.2.2.tar.bz2"),
  98. fileEntry2->getPath());
  99. }
  100. void DefaultBtContextTest::testGetFileEntriesSingle() {
  101. DefaultBtContext btContext;
  102. btContext.load("single.torrent");
  103. // This is multi-file torrent.
  104. std::deque<SharedHandle<FileEntry> > fileEntries = btContext.getFileEntries();
  105. // There is 1 file entry.
  106. CPPUNIT_ASSERT_EQUAL((size_t)1, fileEntries.size());
  107. std::deque<SharedHandle<FileEntry> >::iterator itr = fileEntries.begin();
  108. SharedHandle<FileEntry> fileEntry1 = *itr;
  109. CPPUNIT_ASSERT_EQUAL(std::string("aria2-0.8.2.tar.bz2"),
  110. fileEntry1->getPath());
  111. }
  112. void DefaultBtContextTest::testGetTotalLength() {
  113. DefaultBtContext btContext;
  114. btContext.load("test.torrent");
  115. CPPUNIT_ASSERT_EQUAL((uint64_t)384ULL, btContext.getTotalLength());
  116. }
  117. void DefaultBtContextTest::testGetTotalLengthSingle() {
  118. DefaultBtContext btContext;
  119. btContext.load("single.torrent");
  120. CPPUNIT_ASSERT_EQUAL((uint64_t)384ULL, btContext.getTotalLength());
  121. }
  122. void DefaultBtContextTest::testGetFileModeMulti() {
  123. DefaultBtContext btContext;
  124. btContext.load("test.torrent");
  125. CPPUNIT_ASSERT_EQUAL(BtContext::MULTI, btContext.getFileMode());
  126. }
  127. void DefaultBtContextTest::testGetFileModeSingle() {
  128. DefaultBtContext btContext;
  129. btContext.load("single.torrent");
  130. CPPUNIT_ASSERT_EQUAL(BtContext::SINGLE, btContext.getFileMode());
  131. }
  132. void DefaultBtContextTest::testGetNameMulti() {
  133. DefaultBtContext btContext;
  134. btContext.load("test.torrent");
  135. CPPUNIT_ASSERT_EQUAL(std::string("aria2-test"), btContext.getName());
  136. }
  137. void DefaultBtContextTest::testGetNameSingle() {
  138. DefaultBtContext btContext;
  139. btContext.load("single.torrent");
  140. CPPUNIT_ASSERT_EQUAL(std::string("aria2-0.8.2.tar.bz2"), btContext.getName());
  141. }
  142. void DefaultBtContextTest::testGetAnnounceTier() {
  143. DefaultBtContext btContext;
  144. btContext.load("single.torrent");
  145. std::deque<SharedHandle<AnnounceTier> > tiers = btContext.getAnnounceTiers();
  146. // There is 1 tier.
  147. CPPUNIT_ASSERT_EQUAL((size_t)1, tiers.size());
  148. std::deque<SharedHandle<AnnounceTier> >::iterator itr = tiers.begin();
  149. SharedHandle<AnnounceTier> tier1 = *itr;
  150. CPPUNIT_ASSERT_EQUAL((size_t)1, tier1->urls.size());
  151. CPPUNIT_ASSERT_EQUAL(std::string("http://aria.rednoah.com/announce.php"),
  152. tier1->urls.at(0));
  153. }
  154. void DefaultBtContextTest::testGetAnnounceTierAnnounceList() {
  155. DefaultBtContext btContext;
  156. btContext.load("test.torrent");
  157. std::deque<SharedHandle<AnnounceTier> > tiers = btContext.getAnnounceTiers();
  158. // There are 3 tiers.
  159. CPPUNIT_ASSERT_EQUAL((size_t)3, tiers.size());
  160. SharedHandle<AnnounceTier> tier1 = tiers.at(0);
  161. CPPUNIT_ASSERT_EQUAL((size_t)1, tier1->urls.size());
  162. CPPUNIT_ASSERT_EQUAL(std::string("http://tracker1"),
  163. tier1->urls.at(0));
  164. SharedHandle<AnnounceTier> tier2 = tiers.at(1);
  165. CPPUNIT_ASSERT_EQUAL((size_t)1, tier2->urls.size());
  166. CPPUNIT_ASSERT_EQUAL(std::string("http://tracker2"), tier2->urls.at(0));
  167. SharedHandle<AnnounceTier> tier3 = tiers.at(2);
  168. CPPUNIT_ASSERT_EQUAL((size_t)1, tier3->urls.size());
  169. CPPUNIT_ASSERT_EQUAL(std::string("http://tracker3"), tier3->urls.at(0));
  170. }
  171. void DefaultBtContextTest::testGetPieceLength() {
  172. DefaultBtContext btContext;
  173. btContext.load("test.torrent");
  174. CPPUNIT_ASSERT_EQUAL((size_t)128, btContext.getPieceLength());
  175. }
  176. void DefaultBtContextTest::testGetInfoHashAsString() {
  177. DefaultBtContext btContext;
  178. btContext.load("test.torrent");
  179. CPPUNIT_ASSERT_EQUAL(std::string("248d0a1cd08284299de78d5c1ed359bb46717d8c"),
  180. btContext.getInfoHashAsString());
  181. }
  182. void DefaultBtContextTest::testGetPeerId() {
  183. DefaultBtContext btContext;
  184. btContext.setRandomizer(SharedHandle<Randomizer>(new FixedNumberRandomizer()));
  185. CPPUNIT_ASSERT_EQUAL(std::string("%2daria2%2dAAAAAAAAAAAAA"), Util::torrentUrlencode(btContext.getPeerId(), 20));
  186. }
  187. void DefaultBtContextTest::testComputeFastSet()
  188. {
  189. std::string ipaddr = "192.168.0.1";
  190. unsigned char infoHash[20];
  191. memset(infoHash, 0, sizeof(infoHash));
  192. infoHash[0] = 0xff;
  193. int fastSetSize = 10;
  194. DefaultBtContext btContext;
  195. btContext.setInfoHash(infoHash);
  196. btContext.setNumPieces(1000);
  197. {
  198. std::deque<size_t> fastSet;
  199. btContext.computeFastSet(fastSet, ipaddr, fastSetSize);
  200. size_t ans[] = { 686, 459, 278, 200, 404, 834, 64, 203, 760, 950 };
  201. std::deque<size_t> ansSet(&ans[0], &ans[arrayLength(ans)]);
  202. CPPUNIT_ASSERT(std::equal(fastSet.begin(), fastSet.end(), ansSet.begin()));
  203. }
  204. ipaddr = "10.0.0.1";
  205. {
  206. std::deque<size_t> fastSet;
  207. btContext.computeFastSet(fastSet, ipaddr, fastSetSize);
  208. size_t ans[] = { 568, 188, 466, 452, 550, 662, 109, 226, 398, 11 };
  209. std::deque<size_t> ansSet(&ans[0], &ans[arrayLength(ans)]);
  210. CPPUNIT_ASSERT(std::equal(fastSet.begin(), fastSet.end(), ansSet.begin()));
  211. }
  212. // See when pieces < fastSetSize
  213. btContext.setNumPieces(9);
  214. {
  215. std::deque<size_t> fastSet;
  216. btContext.computeFastSet(fastSet, ipaddr, fastSetSize);
  217. size_t ans[] = { 8, 6, 7, 5, 1, 4, 0, 2, 3 };
  218. std::deque<size_t> ansSet(&ans[0], &ans[arrayLength(ans)]);
  219. CPPUNIT_ASSERT(std::equal(fastSet.begin(), fastSet.end(), ansSet.begin()));
  220. }
  221. }
  222. void DefaultBtContextTest::testGetFileEntries_multiFileUrlList() {
  223. DefaultBtContext btContext;
  224. btContext.load("url-list-multiFile.torrent");
  225. // This is multi-file torrent.
  226. std::deque<SharedHandle<FileEntry> > fileEntries = btContext.getFileEntries();
  227. // There are 2 file entries.
  228. CPPUNIT_ASSERT_EQUAL((size_t)2, fileEntries.size());
  229. std::deque<SharedHandle<FileEntry> >::iterator itr = fileEntries.begin();
  230. SharedHandle<FileEntry> fileEntry1 = *itr;
  231. CPPUNIT_ASSERT_EQUAL(std::string("aria2/src/aria2c"),
  232. fileEntry1->getPath());
  233. std::deque<std::string> uris1 = fileEntry1->getAssociatedUris();
  234. CPPUNIT_ASSERT_EQUAL((size_t)2, uris1.size());
  235. CPPUNIT_ASSERT_EQUAL(std::string("http://localhost/dist/aria2-test/aria2/src/aria2c"),
  236. uris1[0]);
  237. CPPUNIT_ASSERT_EQUAL(std::string("http://mirror/dist/aria2-test/aria2/src/aria2c"),
  238. uris1[1]);
  239. itr++;
  240. SharedHandle<FileEntry> fileEntry2 = *itr;
  241. CPPUNIT_ASSERT_EQUAL(std::string("aria2-0.2.2.tar.bz2"),
  242. fileEntry2->getPath());
  243. std::deque<std::string> uris2 = fileEntry2->getAssociatedUris();
  244. CPPUNIT_ASSERT_EQUAL((size_t)2, uris2.size());
  245. CPPUNIT_ASSERT_EQUAL(std::string("http://localhost/dist/aria2-test/aria2-0.2.2.tar.bz2"),
  246. uris2[0]);
  247. CPPUNIT_ASSERT_EQUAL(std::string("http://mirror/dist/aria2-test/aria2-0.2.2.tar.bz2"),
  248. uris2[1]);
  249. }
  250. void DefaultBtContextTest::testGetFileEntries_singleFileUrlList() {
  251. DefaultBtContext btContext;
  252. btContext.load("url-list-singleFile.torrent");
  253. // This is multi-file torrent.
  254. std::deque<SharedHandle<FileEntry> > fileEntries = btContext.getFileEntries();
  255. // There are 1 file entries.
  256. CPPUNIT_ASSERT_EQUAL((size_t)1, fileEntries.size());
  257. SharedHandle<FileEntry> fileEntry1 = fileEntries.front();
  258. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"),
  259. fileEntry1->getPath());
  260. std::deque<std::string> uris1 = fileEntry1->getAssociatedUris();
  261. CPPUNIT_ASSERT_EQUAL((size_t)1, uris1.size());
  262. CPPUNIT_ASSERT_EQUAL(std::string("http://localhost/dist/aria2.tar.bz2"),
  263. uris1[0]);
  264. }
  265. void DefaultBtContextTest::testLoadFromMemory()
  266. {
  267. std::string memory = "d8:announce36:http://aria.rednoah.com/announce.php13:announce-listll16:http://tracker1 el15:http://tracker2el15:http://tracker3ee7:comment17:REDNOAH.COM RULES13:creation datei1123456789e4:infod5:filesld6:lengthi284e4:pathl5:aria23:src6:aria2ceed6:lengthi100e4:pathl19:aria2-0.2.2.tar.bz2eee4:name10:aria2-test12:piece lengthi128e6:pieces60:AAAAAAAAAAAAAAAAAAAABBBBBBBBBBBBBBBBBBBBCCCCCCCCCCCCCCCCCCCCee";
  268. DefaultBtContext btContext;
  269. btContext.loadFromMemory(memory, "default");
  270. std::string correctHash = "248d0a1cd08284299de78d5c1ed359bb46717d8c";
  271. CPPUNIT_ASSERT_EQUAL((size_t)20, btContext.getInfoHashLength());
  272. CPPUNIT_ASSERT_EQUAL(correctHash, Util::toHex(btContext.getInfoHash(),
  273. btContext.getInfoHashLength()));
  274. }
  275. void DefaultBtContextTest::testLoadFromMemory_somethingMissing()
  276. {
  277. // pieces missing
  278. try {
  279. std::string memory = "d8:announce36:http://aria.rednoah.com/announce.php4:infod4:name13:aria2.tar.bz26:lengthi262144eee";
  280. DefaultBtContext btContext;
  281. btContext.loadFromMemory(memory, "default");
  282. CPPUNIT_FAIL("exception must be thrown.");
  283. } catch(Exception& e) {
  284. std::cerr << e.stackTrace() << std::endl;
  285. }
  286. }
  287. void DefaultBtContextTest::testGetNodes()
  288. {
  289. {
  290. std::string memory =
  291. "d5:nodesl"
  292. "l11:192.168.0.1i6881ee"
  293. "l11:192.168.0.24:6882e"
  294. "e4:infod4:name13:aria2.tar.bz26:lengthi262144e"
  295. "12:piece lengthi262144e"
  296. "6:pieces20:AAAAAAAAAAAAAAAAAAAA"
  297. "ee";
  298. DefaultBtContext btContext;
  299. btContext.loadFromMemory(memory, "default");
  300. const std::deque<std::pair<std::string, uint16_t> >& nodes =
  301. btContext.getNodes();
  302. CPPUNIT_ASSERT_EQUAL((size_t)2, nodes.size());
  303. CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.1"), nodes[0].first);
  304. CPPUNIT_ASSERT_EQUAL((uint16_t)6881, nodes[0].second);
  305. CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), nodes[1].first);
  306. CPPUNIT_ASSERT_EQUAL((uint16_t)6882, nodes[1].second);
  307. }
  308. {
  309. // empty hostname
  310. std::string memory =
  311. "d5:nodesl"
  312. "l1: i6881ee"
  313. "l11:192.168.0.24:6882e"
  314. "e4:infod4:name13:aria2.tar.bz26:lengthi262144e"
  315. "12:piece lengthi262144e"
  316. "6:pieces20:AAAAAAAAAAAAAAAAAAAA"
  317. "ee";
  318. DefaultBtContext btContext;
  319. btContext.loadFromMemory(memory, "default");
  320. const std::deque<std::pair<std::string, uint16_t> >& nodes =
  321. btContext.getNodes();
  322. CPPUNIT_ASSERT_EQUAL((size_t)1, nodes.size());
  323. CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), nodes[0].first);
  324. CPPUNIT_ASSERT_EQUAL((uint16_t)6882, nodes[0].second);
  325. }
  326. {
  327. // bad port
  328. std::string memory =
  329. "d5:nodesl"
  330. "l11:192.168.0.11:xe"
  331. "l11:192.168.0.24:6882e"
  332. "e4:infod4:name13:aria2.tar.bz26:lengthi262144e"
  333. "12:piece lengthi262144e"
  334. "6:pieces20:AAAAAAAAAAAAAAAAAAAA"
  335. "ee";
  336. DefaultBtContext btContext;
  337. btContext.loadFromMemory(memory, "default");
  338. const std::deque<std::pair<std::string, uint16_t> >& nodes =
  339. btContext.getNodes();
  340. CPPUNIT_ASSERT_EQUAL((size_t)1, nodes.size());
  341. CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), nodes[0].first);
  342. CPPUNIT_ASSERT_EQUAL((uint16_t)6882, nodes[0].second);
  343. }
  344. {
  345. // port missing
  346. std::string memory =
  347. "d5:nodesl"
  348. "l11:192.168.0.1e"
  349. "l11:192.168.0.24:6882e"
  350. "e4:infod4:name13:aria2.tar.bz26:lengthi262144e"
  351. "12:piece lengthi262144e"
  352. "6:pieces20:AAAAAAAAAAAAAAAAAAAA"
  353. "ee";
  354. DefaultBtContext btContext;
  355. btContext.loadFromMemory(memory, "default");
  356. const std::deque<std::pair<std::string, uint16_t> >& nodes =
  357. btContext.getNodes();
  358. CPPUNIT_ASSERT_EQUAL((size_t)1, nodes.size());
  359. CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), nodes[0].first);
  360. CPPUNIT_ASSERT_EQUAL((uint16_t)6882, nodes[0].second);
  361. }
  362. {
  363. // nodes is not a list
  364. std::string memory =
  365. "d5:nodes"
  366. "l11:192.168.0.1e"
  367. "4:infod4:name13:aria2.tar.bz26:lengthi262144e"
  368. "12:piece lengthi262144e"
  369. "6:pieces20:AAAAAAAAAAAAAAAAAAAA"
  370. "ee";
  371. DefaultBtContext btContext;
  372. btContext.loadFromMemory(memory, "default");
  373. const std::deque<std::pair<std::string, uint16_t> >& nodes =
  374. btContext.getNodes();
  375. CPPUNIT_ASSERT_EQUAL((size_t)0, nodes.size());
  376. }
  377. {
  378. // the element of node is not Data
  379. std::string memory =
  380. "d5:nodesl"
  381. "ll11:192.168.0.1i6881eee"
  382. "l11:192.168.0.24:6882e"
  383. "e4:infod4:name13:aria2.tar.bz26:lengthi262144e"
  384. "12:piece lengthi262144e"
  385. "6:pieces20:AAAAAAAAAAAAAAAAAAAA"
  386. "ee";
  387. DefaultBtContext btContext;
  388. btContext.loadFromMemory(memory, "default");
  389. const std::deque<std::pair<std::string, uint16_t> >& nodes =
  390. btContext.getNodes();
  391. CPPUNIT_ASSERT_EQUAL((size_t)1, nodes.size());
  392. CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), nodes[0].first);
  393. CPPUNIT_ASSERT_EQUAL((uint16_t)6882, nodes[0].second);
  394. }
  395. }
  396. } // namespace aria2