AnnounceListTest.cc 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. #include "AnnounceList.h"
  2. #include <cppunit/extensions/HelperMacros.h>
  3. #include "Exception.h"
  4. #include "bencode2.h"
  5. namespace aria2 {
  6. class AnnounceListTest : public CppUnit::TestFixture {
  7. CPPUNIT_TEST_SUITE(AnnounceListTest);
  8. CPPUNIT_TEST(testSingleElementList);
  9. CPPUNIT_TEST(testMultiElementList);
  10. CPPUNIT_TEST(testSingleAndMulti);
  11. CPPUNIT_TEST(testNoGroup);
  12. CPPUNIT_TEST(testEvent);
  13. CPPUNIT_TEST(testNextEventIfAfterStarted);
  14. CPPUNIT_TEST(testCountStoppedAllowedTier);
  15. CPPUNIT_TEST(testCountCompletedAllowedTier);
  16. CPPUNIT_TEST(testMoveToStoppedAllowedTier);
  17. CPPUNIT_TEST(testMoveToCompletedAllowedTier);
  18. CPPUNIT_TEST_SUITE_END();
  19. private:
  20. public:
  21. void setUp() {}
  22. void testSingleElementList();
  23. void testMultiElementList();
  24. void testSingleAndMulti();
  25. void testNoGroup();
  26. void testEvent();
  27. void testNextEventIfAfterStarted();
  28. void testCountStoppedAllowedTier();
  29. void testCountCompletedAllowedTier();
  30. void testMoveToStoppedAllowedTier();
  31. void testMoveToCompletedAllowedTier();
  32. };
  33. CPPUNIT_TEST_SUITE_REGISTRATION(AnnounceListTest);
  34. namespace {
  35. std::vector<std::vector<std::string>> toVector(const List* announceList)
  36. {
  37. std::vector<std::vector<std::string>> dest;
  38. for (List::ValueType::const_iterator tierIter = announceList->begin(),
  39. eoi = announceList->end();
  40. tierIter != eoi; ++tierIter) {
  41. std::vector<std::string> ntier;
  42. const List* tier = downcast<List>(*tierIter);
  43. for (List::ValueType::const_iterator uriIter = tier->begin(),
  44. eoi2 = tier->end();
  45. uriIter != eoi2; ++uriIter) {
  46. const String* uri = downcast<String>(*uriIter);
  47. ntier.push_back(uri->s());
  48. }
  49. dest.push_back(ntier);
  50. }
  51. return dest;
  52. }
  53. } // namespace
  54. void AnnounceListTest::testSingleElementList()
  55. {
  56. std::string peersString = "ll8:tracker1el8:tracker2el8:tracker3ee";
  57. std::shared_ptr<ValueBase> announcesList = bencode2::decode(peersString);
  58. // ANNOUNCE_LIST
  59. // [ [ tracker1 ], [ tracker2 ], [ tracker3 ] ]
  60. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  61. CPPUNIT_ASSERT(!announceList.allTiersFailed());
  62. std::string url = announceList.getAnnounce();
  63. std::string event = announceList.getEventString();
  64. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  65. CPPUNIT_ASSERT_EQUAL(std::string("started"), event);
  66. announceList.announceFailure();
  67. url = announceList.getAnnounce();
  68. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  69. announceList.announceFailure();
  70. url = announceList.getAnnounce();
  71. CPPUNIT_ASSERT_EQUAL(std::string("tracker3"), url);
  72. announceList.announceFailure();
  73. CPPUNIT_ASSERT(announceList.allTiersFailed());
  74. announceList.resetTier();
  75. CPPUNIT_ASSERT(!announceList.allTiersFailed());
  76. // back to the first list
  77. url = announceList.getAnnounce();
  78. event = announceList.getEventString();
  79. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  80. CPPUNIT_ASSERT_EQUAL(std::string("started"), event);
  81. announceList.announceFailure();
  82. url = announceList.getAnnounce();
  83. event = announceList.getEventString();
  84. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  85. CPPUNIT_ASSERT_EQUAL(std::string("started"), event);
  86. announceList.announceSuccess();
  87. // back to the first list because announce to tracker2 succeeded.
  88. url = announceList.getAnnounce();
  89. event = announceList.getEventString();
  90. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  91. CPPUNIT_ASSERT_EQUAL(std::string("started"), event);
  92. announceList.announceFailure();
  93. url = announceList.getAnnounce();
  94. event = announceList.getEventString();
  95. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  96. CPPUNIT_ASSERT_EQUAL(std::string(""), event);
  97. }
  98. void AnnounceListTest::testMultiElementList()
  99. {
  100. std::string peersString = "ll8:tracker18:tracker28:tracker3ee";
  101. std::shared_ptr<ValueBase> announcesList = bencode2::decode(peersString);
  102. // ANNOUNCE_LIST
  103. // [ [ tracker1, tracker2, tracker3 ] ]
  104. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  105. CPPUNIT_ASSERT(!announceList.allTiersFailed());
  106. std::string url = announceList.getAnnounce();
  107. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  108. announceList.announceFailure();
  109. url = announceList.getAnnounce();
  110. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  111. announceList.announceSuccess();
  112. url = announceList.getAnnounce();
  113. // tracker2 returns because tracker2 is now first.
  114. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  115. announceList.announceFailure();
  116. url = announceList.getAnnounce();
  117. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  118. announceList.announceFailure();
  119. url = announceList.getAnnounce();
  120. CPPUNIT_ASSERT_EQUAL(std::string("tracker3"), url);
  121. announceList.announceFailure();
  122. CPPUNIT_ASSERT(announceList.allTiersFailed());
  123. announceList.resetTier();
  124. CPPUNIT_ASSERT(!announceList.allTiersFailed());
  125. // back to the first list because there is no other list.
  126. url = announceList.getAnnounce();
  127. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  128. }
  129. void AnnounceListTest::testSingleAndMulti()
  130. {
  131. std::string peersString = "ll8:tracker18:tracker2el8:tracker3ee";
  132. std::shared_ptr<ValueBase> announcesList = bencode2::decode(peersString);
  133. // ANNOUNCE_LIST
  134. // [ [ tracker1, tracker2 ], [ tracker3 ] ]
  135. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  136. std::string url = announceList.getAnnounce();
  137. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  138. announceList.announceSuccess();
  139. url = announceList.getAnnounce();
  140. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  141. announceList.announceFailure();
  142. url = announceList.getAnnounce();
  143. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  144. announceList.announceFailure();
  145. url = announceList.getAnnounce();
  146. CPPUNIT_ASSERT_EQUAL(std::string("tracker3"), url);
  147. announceList.announceSuccess();
  148. url = announceList.getAnnounce();
  149. // tracker1 returns because after the announce to tracker3 succeeds, list
  150. // pointer points to the first list.
  151. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  152. }
  153. void AnnounceListTest::testNoGroup()
  154. {
  155. std::string peersString = "llee";
  156. std::shared_ptr<ValueBase> announcesList = bencode2::decode(peersString);
  157. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  158. CPPUNIT_ASSERT(announceList.countTier() == 0);
  159. }
  160. void AnnounceListTest::testNextEventIfAfterStarted()
  161. {
  162. std::string peersString = "ll8:tracker1ee";
  163. std::shared_ptr<ValueBase> announcesList = bencode2::decode(peersString);
  164. // ANNOUNCE_LIST
  165. // [ [ tracker1 ] ]
  166. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  167. announceList.setEvent(AnnounceTier::STOPPED);
  168. announceList.announceFailure();
  169. announceList.resetTier();
  170. CPPUNIT_ASSERT_EQUAL(std::string(""),
  171. std::string(announceList.getEventString()));
  172. CPPUNIT_ASSERT_EQUAL(AnnounceTier::HALTED, announceList.getEvent());
  173. announceList.setEvent(AnnounceTier::COMPLETED);
  174. announceList.announceFailure();
  175. announceList.resetTier();
  176. CPPUNIT_ASSERT_EQUAL(std::string(""),
  177. std::string(announceList.getEventString()));
  178. CPPUNIT_ASSERT_EQUAL(AnnounceTier::SEEDING, announceList.getEvent());
  179. }
  180. void AnnounceListTest::testEvent()
  181. {
  182. std::string peersString = "ll8:tracker1el8:tracker2el8:tracker3ee";
  183. std::shared_ptr<ValueBase> announcesList = bencode2::decode(peersString);
  184. // ANNOUNCE_LIST
  185. // [ [ tracker1 ], [ tracker2 ], [ tracker3 ] ]
  186. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  187. announceList.setEvent(AnnounceTier::STOPPED);
  188. announceList.announceSuccess();
  189. CPPUNIT_ASSERT_EQUAL(std::string(""),
  190. std::string(announceList.getEventString()));
  191. CPPUNIT_ASSERT_EQUAL(AnnounceTier::HALTED, announceList.getEvent());
  192. announceList.setEvent(AnnounceTier::COMPLETED);
  193. announceList.announceSuccess();
  194. CPPUNIT_ASSERT_EQUAL(std::string(""),
  195. std::string(announceList.getEventString()));
  196. CPPUNIT_ASSERT_EQUAL(AnnounceTier::SEEDING, announceList.getEvent());
  197. announceList.setEvent(AnnounceTier::STARTED_AFTER_COMPLETION);
  198. CPPUNIT_ASSERT_EQUAL(std::string("started"),
  199. std::string(announceList.getEventString()));
  200. announceList.announceSuccess();
  201. CPPUNIT_ASSERT_EQUAL(AnnounceTier::SEEDING, announceList.getEvent());
  202. }
  203. void AnnounceListTest::testCountStoppedAllowedTier()
  204. {
  205. std::string peersString = "ll8:tracker1el8:tracker2el8:tracker3ee";
  206. std::shared_ptr<ValueBase> announcesList = bencode2::decode(peersString);
  207. // ANNOUNCE_LIST
  208. // [ [ tracker1 ], [ tracker2 ], [ tracker3 ] ]
  209. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  210. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countStoppedAllowedTier());
  211. announceList.setEvent(AnnounceTier::STARTED);
  212. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countStoppedAllowedTier());
  213. announceList.setEvent(AnnounceTier::STARTED_AFTER_COMPLETION);
  214. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countStoppedAllowedTier());
  215. announceList.setEvent(AnnounceTier::HALTED);
  216. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countStoppedAllowedTier());
  217. announceList.setEvent(AnnounceTier::DOWNLOADING);
  218. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countStoppedAllowedTier());
  219. announceList.setEvent(AnnounceTier::STOPPED);
  220. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countStoppedAllowedTier());
  221. announceList.setEvent(AnnounceTier::COMPLETED);
  222. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countStoppedAllowedTier());
  223. announceList.setEvent(AnnounceTier::SEEDING);
  224. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countStoppedAllowedTier());
  225. }
  226. void AnnounceListTest::testCountCompletedAllowedTier()
  227. {
  228. std::string peersString = "ll8:tracker1el8:tracker2el8:tracker3ee";
  229. std::shared_ptr<ValueBase> announcesList = bencode2::decode(peersString);
  230. // ANNOUNCE_LIST
  231. // [ [ tracker1 ], [ tracker2 ], [ tracker3 ] ]
  232. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  233. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  234. announceList.setEvent(AnnounceTier::STARTED);
  235. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  236. announceList.setEvent(AnnounceTier::STARTED_AFTER_COMPLETION);
  237. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  238. announceList.setEvent(AnnounceTier::STOPPED);
  239. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  240. announceList.setEvent(AnnounceTier::SEEDING);
  241. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  242. announceList.setEvent(AnnounceTier::HALTED);
  243. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  244. announceList.setEvent(AnnounceTier::DOWNLOADING);
  245. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countCompletedAllowedTier());
  246. announceList.setEvent(AnnounceTier::COMPLETED);
  247. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countCompletedAllowedTier());
  248. }
  249. std::deque<std::string> createUrls(const std::string& url)
  250. {
  251. std::deque<std::string> urls;
  252. urls.push_back(url);
  253. return urls;
  254. }
  255. void AnnounceListTest::testMoveToStoppedAllowedTier()
  256. {
  257. std::shared_ptr<AnnounceTier> t1(new AnnounceTier(createUrls("tracker1")));
  258. std::shared_ptr<AnnounceTier> t2(new AnnounceTier(createUrls("tracker2")));
  259. t2->event = AnnounceTier::COMPLETED;
  260. std::shared_ptr<AnnounceTier> t3(new AnnounceTier(createUrls("tracker3")));
  261. std::deque<std::shared_ptr<AnnounceTier>> tiers;
  262. tiers.push_back(t1);
  263. tiers.push_back(t2);
  264. tiers.push_back(t3);
  265. AnnounceList announceList(tiers);
  266. CPPUNIT_ASSERT(!announceList.currentTierAcceptsStoppedEvent());
  267. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), announceList.getAnnounce());
  268. announceList.moveToStoppedAllowedTier();
  269. CPPUNIT_ASSERT(announceList.currentTierAcceptsStoppedEvent());
  270. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), announceList.getAnnounce());
  271. announceList.announceFailure();
  272. CPPUNIT_ASSERT(!announceList.currentTierAcceptsStoppedEvent());
  273. CPPUNIT_ASSERT_EQUAL(std::string("tracker3"), announceList.getAnnounce());
  274. // test wrapped search
  275. announceList.moveToStoppedAllowedTier();
  276. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), announceList.getAnnounce());
  277. }
  278. void AnnounceListTest::testMoveToCompletedAllowedTier()
  279. {
  280. std::shared_ptr<AnnounceTier> t1(new AnnounceTier(createUrls("tracker1")));
  281. std::shared_ptr<AnnounceTier> t2(new AnnounceTier(createUrls("tracker2")));
  282. t2->event = AnnounceTier::COMPLETED;
  283. std::shared_ptr<AnnounceTier> t3(new AnnounceTier(createUrls("tracker3")));
  284. std::deque<std::shared_ptr<AnnounceTier>> tiers;
  285. tiers.push_back(t1);
  286. tiers.push_back(t2);
  287. tiers.push_back(t3);
  288. AnnounceList announceList(tiers);
  289. CPPUNIT_ASSERT(!announceList.currentTierAcceptsCompletedEvent());
  290. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), announceList.getAnnounce());
  291. announceList.moveToStoppedAllowedTier();
  292. CPPUNIT_ASSERT(announceList.currentTierAcceptsCompletedEvent());
  293. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), announceList.getAnnounce());
  294. announceList.announceFailure();
  295. CPPUNIT_ASSERT(!announceList.currentTierAcceptsCompletedEvent());
  296. CPPUNIT_ASSERT_EQUAL(std::string("tracker3"), announceList.getAnnounce());
  297. // test wrapped search
  298. announceList.moveToStoppedAllowedTier();
  299. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), announceList.getAnnounce());
  300. }
  301. } // namespace aria2