AnnounceListTest.cc 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  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. }
  23. void testSingleElementList();
  24. void testMultiElementList();
  25. void testSingleAndMulti();
  26. void testNoGroup();
  27. void testEvent();
  28. void testNextEventIfAfterStarted();
  29. void testCountStoppedAllowedTier();
  30. void testCountCompletedAllowedTier();
  31. void testMoveToStoppedAllowedTier();
  32. void testMoveToCompletedAllowedTier();
  33. };
  34. CPPUNIT_TEST_SUITE_REGISTRATION( AnnounceListTest );
  35. namespace {
  36. std::vector<std::vector<std::string> > toVector
  37. (const List* announceList)
  38. {
  39. std::vector<std::vector<std::string> > dest;
  40. for(List::ValueType::const_iterator tierIter = announceList->begin(),
  41. eoi = announceList->end(); tierIter != eoi; ++tierIter) {
  42. std::vector<std::string> ntier;
  43. const List* tier = downcast<List>(*tierIter);
  44. for(List::ValueType::const_iterator uriIter = tier->begin(),
  45. eoi2 = tier->end(); 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. std::string peersString = "ll8:tracker1el8:tracker2el8:tracker3ee";
  56. SharedHandle<ValueBase> announcesList =
  57. 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. std::string peersString = "ll8:tracker18:tracker28:tracker3ee";
  100. SharedHandle<ValueBase> announcesList = bencode2::decode(peersString);
  101. // ANNOUNCE_LIST
  102. // [ [ tracker1, tracker2, tracker3 ] ]
  103. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  104. CPPUNIT_ASSERT(!announceList.allTiersFailed());
  105. std::string url = announceList.getAnnounce();
  106. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  107. announceList.announceFailure();
  108. url = announceList.getAnnounce();
  109. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  110. announceList.announceSuccess();
  111. url = announceList.getAnnounce();
  112. // tracker2 returns because tracker2 is now first.
  113. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  114. announceList.announceFailure();
  115. url = announceList.getAnnounce();
  116. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  117. announceList.announceFailure();
  118. url = announceList.getAnnounce();
  119. CPPUNIT_ASSERT_EQUAL(std::string("tracker3"), url);
  120. announceList.announceFailure();
  121. CPPUNIT_ASSERT(announceList.allTiersFailed());
  122. announceList.resetTier();
  123. CPPUNIT_ASSERT(!announceList.allTiersFailed());
  124. // back to the first list because there is no other list.
  125. url = announceList.getAnnounce();
  126. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  127. }
  128. void AnnounceListTest::testSingleAndMulti() {
  129. std::string peersString = "ll8:tracker18:tracker2el8:tracker3ee";
  130. SharedHandle<ValueBase> announcesList = bencode2::decode(peersString);
  131. // ANNOUNCE_LIST
  132. // [ [ tracker1, tracker2 ], [ tracker3 ] ]
  133. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  134. std::string url = announceList.getAnnounce();
  135. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  136. announceList.announceSuccess();
  137. url = announceList.getAnnounce();
  138. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  139. announceList.announceFailure();
  140. url = announceList.getAnnounce();
  141. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), url);
  142. announceList.announceFailure();
  143. url = announceList.getAnnounce();
  144. CPPUNIT_ASSERT_EQUAL(std::string("tracker3"), url);
  145. announceList.announceSuccess();
  146. url = announceList.getAnnounce();
  147. // tracker1 returns because after the announce to tracker3 succeeds, list
  148. // pointer points to the first list.
  149. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), url);
  150. }
  151. void AnnounceListTest::testNoGroup() {
  152. std::string peersString = "llee";
  153. SharedHandle<ValueBase> announcesList = bencode2::decode(peersString);
  154. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  155. CPPUNIT_ASSERT(announceList.countTier() == 0);
  156. }
  157. void AnnounceListTest::testNextEventIfAfterStarted() {
  158. std::string peersString = "ll8:tracker1ee";
  159. SharedHandle<ValueBase> announcesList = bencode2::decode(peersString);
  160. // ANNOUNCE_LIST
  161. // [ [ tracker1 ] ]
  162. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  163. announceList.setEvent(AnnounceTier::STOPPED);
  164. announceList.announceFailure();
  165. announceList.resetTier();
  166. CPPUNIT_ASSERT_EQUAL(std::string(""), announceList.getEventString());
  167. CPPUNIT_ASSERT_EQUAL(AnnounceTier::HALTED, announceList.getEvent());
  168. announceList.setEvent(AnnounceTier::COMPLETED);
  169. announceList.announceFailure();
  170. announceList.resetTier();
  171. CPPUNIT_ASSERT_EQUAL(std::string(""), announceList.getEventString());
  172. CPPUNIT_ASSERT_EQUAL(AnnounceTier::SEEDING, announceList.getEvent());
  173. }
  174. void AnnounceListTest::testEvent() {
  175. std::string peersString = "ll8:tracker1el8:tracker2el8:tracker3ee";
  176. SharedHandle<ValueBase> announcesList = bencode2::decode(peersString);
  177. // ANNOUNCE_LIST
  178. // [ [ tracker1 ], [ tracker2 ], [ tracker3 ] ]
  179. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  180. announceList.setEvent(AnnounceTier::STOPPED);
  181. announceList.announceSuccess();
  182. CPPUNIT_ASSERT_EQUAL(std::string(""), announceList.getEventString());
  183. CPPUNIT_ASSERT_EQUAL(AnnounceTier::HALTED, announceList.getEvent());
  184. announceList.setEvent(AnnounceTier::COMPLETED);
  185. announceList.announceSuccess();
  186. CPPUNIT_ASSERT_EQUAL(std::string(""), announceList.getEventString());
  187. CPPUNIT_ASSERT_EQUAL(AnnounceTier::SEEDING, announceList.getEvent());
  188. announceList.setEvent(AnnounceTier::STARTED_AFTER_COMPLETION);
  189. CPPUNIT_ASSERT_EQUAL(std::string("started"), announceList.getEventString());
  190. announceList.announceSuccess();
  191. CPPUNIT_ASSERT_EQUAL(AnnounceTier::SEEDING, announceList.getEvent());
  192. }
  193. void AnnounceListTest::testCountStoppedAllowedTier() {
  194. std::string peersString = "ll8:tracker1el8:tracker2el8:tracker3ee";
  195. SharedHandle<ValueBase> announcesList = bencode2::decode(peersString);
  196. // ANNOUNCE_LIST
  197. // [ [ tracker1 ], [ tracker2 ], [ tracker3 ] ]
  198. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  199. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countStoppedAllowedTier());
  200. announceList.setEvent(AnnounceTier::STARTED);
  201. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countStoppedAllowedTier());
  202. announceList.setEvent(AnnounceTier::STARTED_AFTER_COMPLETION);
  203. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countStoppedAllowedTier());
  204. announceList.setEvent(AnnounceTier::HALTED);
  205. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countStoppedAllowedTier());
  206. announceList.setEvent(AnnounceTier::DOWNLOADING);
  207. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countStoppedAllowedTier());
  208. announceList.setEvent(AnnounceTier::STOPPED);
  209. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countStoppedAllowedTier());
  210. announceList.setEvent(AnnounceTier::COMPLETED);
  211. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countStoppedAllowedTier());
  212. announceList.setEvent(AnnounceTier::SEEDING);
  213. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countStoppedAllowedTier());
  214. }
  215. void AnnounceListTest::testCountCompletedAllowedTier() {
  216. std::string peersString = "ll8:tracker1el8:tracker2el8:tracker3ee";
  217. SharedHandle<ValueBase> announcesList = bencode2::decode(peersString);
  218. // ANNOUNCE_LIST
  219. // [ [ tracker1 ], [ tracker2 ], [ tracker3 ] ]
  220. AnnounceList announceList(toVector(downcast<List>(announcesList)));
  221. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  222. announceList.setEvent(AnnounceTier::STARTED);
  223. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  224. announceList.setEvent(AnnounceTier::STARTED_AFTER_COMPLETION);
  225. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  226. announceList.setEvent(AnnounceTier::STOPPED);
  227. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  228. announceList.setEvent(AnnounceTier::SEEDING);
  229. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  230. announceList.setEvent(AnnounceTier::HALTED);
  231. CPPUNIT_ASSERT_EQUAL((size_t)0, announceList.countCompletedAllowedTier());
  232. announceList.setEvent(AnnounceTier::DOWNLOADING);
  233. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countCompletedAllowedTier());
  234. announceList.setEvent(AnnounceTier::COMPLETED);
  235. CPPUNIT_ASSERT_EQUAL((size_t)1, announceList.countCompletedAllowedTier());
  236. }
  237. std::deque<std::string> createUrls(const std::string& url) {
  238. std::deque<std::string> urls;
  239. urls.push_back(url);
  240. return urls;
  241. }
  242. void AnnounceListTest::testMoveToStoppedAllowedTier() {
  243. SharedHandle<AnnounceTier> t1(new AnnounceTier(createUrls("tracker1")));
  244. SharedHandle<AnnounceTier> t2(new AnnounceTier(createUrls("tracker2")));
  245. t2->event = AnnounceTier::COMPLETED;
  246. SharedHandle<AnnounceTier> t3(new AnnounceTier(createUrls("tracker3")));
  247. std::deque<SharedHandle<AnnounceTier> > tiers;
  248. tiers.push_back(t1);
  249. tiers.push_back(t2);
  250. tiers.push_back(t3);
  251. AnnounceList announceList(tiers);
  252. CPPUNIT_ASSERT(!announceList.currentTierAcceptsStoppedEvent());
  253. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), announceList.getAnnounce());
  254. announceList.moveToStoppedAllowedTier();
  255. CPPUNIT_ASSERT(announceList.currentTierAcceptsStoppedEvent());
  256. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), announceList.getAnnounce());
  257. announceList.announceFailure();
  258. CPPUNIT_ASSERT(!announceList.currentTierAcceptsStoppedEvent());
  259. CPPUNIT_ASSERT_EQUAL(std::string("tracker3"), announceList.getAnnounce());
  260. // test wrapped search
  261. announceList.moveToStoppedAllowedTier();
  262. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), announceList.getAnnounce());
  263. }
  264. void AnnounceListTest::testMoveToCompletedAllowedTier() {
  265. SharedHandle<AnnounceTier> t1(new AnnounceTier(createUrls("tracker1")));
  266. SharedHandle<AnnounceTier> t2(new AnnounceTier(createUrls("tracker2")));
  267. t2->event = AnnounceTier::COMPLETED;
  268. SharedHandle<AnnounceTier> t3(new AnnounceTier(createUrls("tracker3")));
  269. std::deque<SharedHandle<AnnounceTier> > tiers;
  270. tiers.push_back(t1);
  271. tiers.push_back(t2);
  272. tiers.push_back(t3);
  273. AnnounceList announceList(tiers);
  274. CPPUNIT_ASSERT(!announceList.currentTierAcceptsCompletedEvent());
  275. CPPUNIT_ASSERT_EQUAL(std::string("tracker1"), announceList.getAnnounce());
  276. announceList.moveToStoppedAllowedTier();
  277. CPPUNIT_ASSERT(announceList.currentTierAcceptsCompletedEvent());
  278. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), announceList.getAnnounce());
  279. announceList.announceFailure();
  280. CPPUNIT_ASSERT(!announceList.currentTierAcceptsCompletedEvent());
  281. CPPUNIT_ASSERT_EQUAL(std::string("tracker3"), announceList.getAnnounce());
  282. // test wrapped search
  283. announceList.moveToStoppedAllowedTier();
  284. CPPUNIT_ASSERT_EQUAL(std::string("tracker2"), announceList.getAnnounce());
  285. }
  286. } // namespace aria2