AnnounceListTest.cc 13 KB

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