DefaultBtMessageDispatcherTest.cc 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428
  1. #include "DefaultBtMessageDispatcher.h"
  2. #include <cassert>
  3. #include <cppunit/extensions/HelperMacros.h>
  4. #include "Util.h"
  5. #include "Exception.h"
  6. #include "MockPieceStorage.h"
  7. #include "MockPeerStorage.h"
  8. #include "DefaultBtContext.h"
  9. #include "MockBtMessage.h"
  10. #include "MockBtMessageFactory.h"
  11. #include "prefs.h"
  12. #include "BtCancelSendingPieceEvent.h"
  13. #include "BtHandshakeMessage.h"
  14. #include "Option.h"
  15. #include "RequestGroupMan.h"
  16. #include "ServerStatMan.h"
  17. #include "RequestGroup.h"
  18. namespace aria2 {
  19. class DefaultBtMessageDispatcherTest:public CppUnit::TestFixture {
  20. CPPUNIT_TEST_SUITE(DefaultBtMessageDispatcherTest);
  21. CPPUNIT_TEST(testAddMessage);
  22. CPPUNIT_TEST(testSendMessages);
  23. CPPUNIT_TEST(testSendMessages_underUploadLimit);
  24. // See the comment on the definition
  25. //CPPUNIT_TEST(testSendMessages_overUploadLimit);
  26. CPPUNIT_TEST(testSendMessages_sendingInProgress);
  27. CPPUNIT_TEST(testDoCancelSendingPieceAction);
  28. CPPUNIT_TEST(testCheckRequestSlotAndDoNecessaryThing);
  29. CPPUNIT_TEST(testCheckRequestSlotAndDoNecessaryThing_timeout);
  30. CPPUNIT_TEST(testCheckRequestSlotAndDoNecessaryThing_completeBlock);
  31. CPPUNIT_TEST(testIsSendingInProgress);
  32. CPPUNIT_TEST(testCountOutstandingRequest);
  33. CPPUNIT_TEST(testIsOutstandingRequest);
  34. CPPUNIT_TEST(testGetOutstandingRequest);
  35. CPPUNIT_TEST(testRemoveOutstandingRequest);
  36. CPPUNIT_TEST_SUITE_END();
  37. private:
  38. SharedHandle<DefaultBtContext> btContext;
  39. SharedHandle<Peer> peer;
  40. SharedHandle<DefaultBtMessageDispatcher> btMessageDispatcher;
  41. SharedHandle<MockPeerStorage> peerStorage;
  42. SharedHandle<MockPieceStorage> pieceStorage;
  43. SharedHandle<MockBtMessageFactory> _messageFactory;
  44. SharedHandle<RequestGroupMan> _rgman;
  45. SharedHandle<Option> _option;
  46. SharedHandle<RequestGroup> _rg;
  47. public:
  48. void tearDown() {}
  49. void testAddMessage();
  50. void testSendMessages();
  51. void testSendMessages_underUploadLimit();
  52. void testSendMessages_overUploadLimit();
  53. void testSendMessages_sendingInProgress();
  54. void testDoCancelSendingPieceAction();
  55. void testCheckRequestSlotAndDoNecessaryThing();
  56. void testCheckRequestSlotAndDoNecessaryThing_timeout();
  57. void testCheckRequestSlotAndDoNecessaryThing_completeBlock();
  58. void testIsSendingInProgress();
  59. void testCountOutstandingRequest();
  60. void testIsOutstandingRequest();
  61. void testGetOutstandingRequest();
  62. void testRemoveOutstandingRequest();
  63. class MockBtMessage2 : public MockBtMessage {
  64. private:
  65. bool onQueuedCalled;
  66. bool sendCalled;
  67. bool doCancelActionCalled;
  68. public:
  69. std::string type;
  70. public:
  71. MockBtMessage2():onQueuedCalled(false),
  72. sendCalled(false),
  73. doCancelActionCalled(false)
  74. {}
  75. virtual ~MockBtMessage2() {}
  76. virtual void onQueued() {
  77. onQueuedCalled = true;
  78. }
  79. bool isOnQueuedCalled() const {
  80. return onQueuedCalled;
  81. }
  82. virtual void send() {
  83. sendCalled = true;
  84. }
  85. bool isSendCalled() const {
  86. return sendCalled;
  87. }
  88. virtual void onCancelSendingPieceEvent
  89. (const BtCancelSendingPieceEvent& event)
  90. {
  91. doCancelActionCalled = true;
  92. }
  93. bool isDoCancelActionCalled() const {
  94. return doCancelActionCalled;
  95. }
  96. };
  97. class MockPieceStorage2 : public MockPieceStorage {
  98. private:
  99. SharedHandle<Piece> piece;
  100. public:
  101. virtual SharedHandle<Piece> getPiece(size_t index) {
  102. return piece;
  103. }
  104. void setPiece(const SharedHandle<Piece>& piece) {
  105. this->piece = piece;
  106. }
  107. };
  108. class MockBtMessageFactory2 : public MockBtMessageFactory {
  109. public:
  110. virtual SharedHandle<BtMessage>
  111. createCancelMessage(size_t index, uint32_t begin, size_t length) {
  112. SharedHandle<MockBtMessage2> btMsg(new MockBtMessage2());
  113. btMsg->type = "cancel";
  114. return btMsg;
  115. }
  116. };
  117. void setUp() {
  118. _option.reset(new Option());
  119. _rg.reset(new RequestGroup(_option, std::deque<std::string>()));
  120. btContext.reset(new DefaultBtContext());
  121. btContext->load("test.torrent");
  122. btContext->setOwnerRequestGroup(_rg.get());
  123. peer.reset(new Peer("192.168.0.1", 6969));
  124. peer->allocateSessionResource(btContext->getPieceLength(),
  125. btContext->getTotalLength());
  126. peerStorage.reset(new MockPeerStorage());
  127. pieceStorage.reset(new MockPieceStorage());
  128. _messageFactory.reset(new MockBtMessageFactory2());
  129. _rgman.reset(new RequestGroupMan(std::deque<SharedHandle<RequestGroup> >(),
  130. 0, _option.get()));
  131. btMessageDispatcher.reset(new DefaultBtMessageDispatcher());
  132. btMessageDispatcher->setPeer(peer);
  133. btMessageDispatcher->setBtContext(btContext);
  134. btMessageDispatcher->setPieceStorage(pieceStorage);
  135. btMessageDispatcher->setPeerStorage(peerStorage);
  136. btMessageDispatcher->setBtMessageFactory(_messageFactory);
  137. btMessageDispatcher->setCuid(1);
  138. btMessageDispatcher->setRequestGroupMan(_rgman);
  139. }
  140. };
  141. CPPUNIT_TEST_SUITE_REGISTRATION(DefaultBtMessageDispatcherTest);
  142. void DefaultBtMessageDispatcherTest::testAddMessage() {
  143. SharedHandle<MockBtMessage2> msg(new MockBtMessage2());
  144. CPPUNIT_ASSERT_EQUAL(false, msg->isOnQueuedCalled());
  145. btMessageDispatcher->addMessageToQueue(msg);
  146. CPPUNIT_ASSERT_EQUAL(true, msg->isOnQueuedCalled());
  147. CPPUNIT_ASSERT_EQUAL((size_t)1,
  148. btMessageDispatcher->getMessageQueue().size());
  149. }
  150. void DefaultBtMessageDispatcherTest::testSendMessages() {
  151. TransferStat stat;
  152. stat.setUploadSpeed(0);
  153. peerStorage->setStat(stat);
  154. SharedHandle<MockBtMessage2> msg1(new MockBtMessage2());
  155. msg1->setSendingInProgress(false);
  156. msg1->setUploading(false);
  157. SharedHandle<MockBtMessage2> msg2(new MockBtMessage2());
  158. msg2->setSendingInProgress(false);
  159. msg2->setUploading(false);
  160. btMessageDispatcher->addMessageToQueue(msg1);
  161. btMessageDispatcher->addMessageToQueue(msg2);
  162. btMessageDispatcher->sendMessages();
  163. CPPUNIT_ASSERT(msg1->isSendCalled());
  164. CPPUNIT_ASSERT(msg2->isSendCalled());
  165. }
  166. void DefaultBtMessageDispatcherTest::testSendMessages_underUploadLimit() {
  167. TransferStat stat;
  168. stat.setUploadSpeed(0);
  169. peerStorage->setStat(stat);
  170. SharedHandle<MockBtMessage2> msg1(new MockBtMessage2());
  171. msg1->setSendingInProgress(false);
  172. msg1->setUploading(true);
  173. SharedHandle<MockBtMessage2> msg2(new MockBtMessage2());
  174. msg2->setSendingInProgress(false);
  175. msg2->setUploading(true);
  176. btMessageDispatcher->addMessageToQueue(msg1);
  177. btMessageDispatcher->addMessageToQueue(msg2);
  178. btMessageDispatcher->sendMessages();
  179. CPPUNIT_ASSERT(msg1->isSendCalled());
  180. CPPUNIT_ASSERT(msg2->isSendCalled());
  181. }
  182. // TODO Because we no longer directly use PeerStorage::calculateStat()
  183. // and Neither RequestGroup nor RequestGroupMan can be stubbed, this
  184. // test is commented out for now.
  185. //
  186. // void DefaultBtMessageDispatcherTest::testSendMessages_overUploadLimit() {
  187. // btMessageDispatcher->setMaxUploadSpeedLimit(100);
  188. // TransferStat stat;
  189. // stat.setUploadSpeed(150);
  190. // peerStorage->setStat(stat);
  191. // SharedHandle<MockBtMessage2> msg1(new MockBtMessage2());
  192. // msg1->setSendingInProgress(false);
  193. // msg1->setUploading(true);
  194. // SharedHandle<MockBtMessage2> msg2(new MockBtMessage2());
  195. // msg2->setSendingInProgress(false);
  196. // msg2->setUploading(true);
  197. // SharedHandle<MockBtMessage2> msg3(new MockBtMessage2());
  198. // msg3->setSendingInProgress(false);
  199. // msg3->setUploading(false);
  200. // btMessageDispatcher->addMessageToQueue(msg1);
  201. // btMessageDispatcher->addMessageToQueue(msg2);
  202. // btMessageDispatcher->addMessageToQueue(msg3);
  203. // btMessageDispatcher->sendMessages();
  204. // CPPUNIT_ASSERT(!msg1->isSendCalled());
  205. // CPPUNIT_ASSERT(!msg2->isSendCalled());
  206. // CPPUNIT_ASSERT(msg3->isSendCalled());
  207. // CPPUNIT_ASSERT_EQUAL((size_t)2,
  208. // btMessageDispatcher->getMessageQueue().size());
  209. // }
  210. void DefaultBtMessageDispatcherTest::testSendMessages_sendingInProgress() {
  211. SharedHandle<MockBtMessage2> msg1(new MockBtMessage2());
  212. msg1->setSendingInProgress(false);
  213. msg1->setUploading(false);
  214. SharedHandle<MockBtMessage2> msg2(new MockBtMessage2());
  215. msg2->setSendingInProgress(true);
  216. msg2->setUploading(false);
  217. SharedHandle<MockBtMessage2> msg3(new MockBtMessage2());
  218. msg3->setSendingInProgress(false);
  219. msg3->setUploading(false);
  220. btMessageDispatcher->addMessageToQueue(msg1);
  221. btMessageDispatcher->addMessageToQueue(msg2);
  222. btMessageDispatcher->addMessageToQueue(msg3);
  223. btMessageDispatcher->sendMessages();
  224. CPPUNIT_ASSERT(msg1->isSendCalled());
  225. CPPUNIT_ASSERT(msg2->isSendCalled());
  226. CPPUNIT_ASSERT(!msg3->isSendCalled());
  227. CPPUNIT_ASSERT_EQUAL((size_t)2,
  228. btMessageDispatcher->getMessageQueue().size());
  229. }
  230. void DefaultBtMessageDispatcherTest::testDoCancelSendingPieceAction() {
  231. SharedHandle<MockBtMessage2> msg1(new MockBtMessage2());
  232. SharedHandle<MockBtMessage2> msg2(new MockBtMessage2());
  233. btMessageDispatcher->addMessageToQueue(msg1);
  234. btMessageDispatcher->addMessageToQueue(msg2);
  235. btMessageDispatcher->doCancelSendingPieceAction(0, 0, 0);
  236. CPPUNIT_ASSERT_EQUAL(true, msg1->isDoCancelActionCalled());
  237. CPPUNIT_ASSERT_EQUAL(true, msg2->isDoCancelActionCalled());
  238. }
  239. int MY_PIECE_LENGTH = 16*1024;
  240. void DefaultBtMessageDispatcherTest::testCheckRequestSlotAndDoNecessaryThing() {
  241. SharedHandle<Piece> piece(new Piece(0, MY_PIECE_LENGTH));
  242. RequestSlot slot(0, 0, MY_PIECE_LENGTH, 0, piece);
  243. size_t index;
  244. CPPUNIT_ASSERT(piece->getMissingUnusedBlockIndex(index));
  245. CPPUNIT_ASSERT_EQUAL((size_t)0, index);
  246. SharedHandle<MockPieceStorage2> pieceStorage(new MockPieceStorage2());
  247. pieceStorage->setPiece(piece);
  248. btMessageDispatcher->setRequestTimeout(60);
  249. btMessageDispatcher->setPieceStorage(pieceStorage);
  250. btMessageDispatcher->addOutstandingRequest(slot);
  251. btMessageDispatcher->checkRequestSlotAndDoNecessaryThing();
  252. CPPUNIT_ASSERT_EQUAL((size_t)0,
  253. btMessageDispatcher->getMessageQueue().size());
  254. CPPUNIT_ASSERT_EQUAL((size_t)1,
  255. btMessageDispatcher->getRequestSlots().size());
  256. }
  257. void DefaultBtMessageDispatcherTest::testCheckRequestSlotAndDoNecessaryThing_timeout() {
  258. SharedHandle<Piece> piece(new Piece(0, MY_PIECE_LENGTH));
  259. RequestSlot slot(0, 0, MY_PIECE_LENGTH, 0, piece);
  260. // make this slot timeout
  261. slot.setDispatchedTime(0);
  262. size_t index;
  263. CPPUNIT_ASSERT(piece->getMissingUnusedBlockIndex(index));
  264. CPPUNIT_ASSERT_EQUAL((size_t)0, index);
  265. SharedHandle<MockPieceStorage2> pieceStorage(new MockPieceStorage2());
  266. pieceStorage->setPiece(piece);
  267. btMessageDispatcher->setRequestTimeout(60);
  268. btMessageDispatcher->setPieceStorage(pieceStorage);
  269. btMessageDispatcher->addOutstandingRequest(slot);
  270. btMessageDispatcher->checkRequestSlotAndDoNecessaryThing();
  271. CPPUNIT_ASSERT_EQUAL((size_t)0,
  272. btMessageDispatcher->getMessageQueue().size());
  273. CPPUNIT_ASSERT_EQUAL((size_t)0,
  274. btMessageDispatcher->getRequestSlots().size());
  275. CPPUNIT_ASSERT_EQUAL(false, piece->isBlockUsed(0));
  276. CPPUNIT_ASSERT_EQUAL(true, peer->snubbing());
  277. }
  278. void DefaultBtMessageDispatcherTest::testCheckRequestSlotAndDoNecessaryThing_completeBlock() {
  279. SharedHandle<Piece> piece(new Piece(0, MY_PIECE_LENGTH));
  280. piece->completeBlock(0);
  281. RequestSlot slot(0, 0, MY_PIECE_LENGTH, 0, piece);
  282. SharedHandle<MockPieceStorage2> pieceStorage(new MockPieceStorage2());
  283. pieceStorage->setPiece(piece);
  284. btMessageDispatcher->setRequestTimeout(60);
  285. btMessageDispatcher->setPieceStorage(pieceStorage);
  286. btMessageDispatcher->addOutstandingRequest(slot);
  287. btMessageDispatcher->checkRequestSlotAndDoNecessaryThing();
  288. CPPUNIT_ASSERT_EQUAL((size_t)1,
  289. btMessageDispatcher->getMessageQueue().size());
  290. CPPUNIT_ASSERT_EQUAL((size_t)0,
  291. btMessageDispatcher->getRequestSlots().size());
  292. }
  293. void DefaultBtMessageDispatcherTest::testIsSendingInProgress() {
  294. CPPUNIT_ASSERT(!btMessageDispatcher->isSendingInProgress());
  295. SharedHandle<MockBtMessage2> msg(new MockBtMessage2());
  296. msg->setSendingInProgress(false);
  297. btMessageDispatcher->addMessageToQueue(msg);
  298. CPPUNIT_ASSERT(!btMessageDispatcher->isSendingInProgress());
  299. msg->setSendingInProgress(true);
  300. CPPUNIT_ASSERT(btMessageDispatcher->isSendingInProgress());
  301. }
  302. void DefaultBtMessageDispatcherTest::testCountOutstandingRequest() {
  303. RequestSlot slot(0, 0, MY_PIECE_LENGTH, 0);
  304. btMessageDispatcher->addOutstandingRequest(slot);
  305. CPPUNIT_ASSERT_EQUAL((size_t)1,
  306. btMessageDispatcher->countOutstandingRequest());
  307. }
  308. void DefaultBtMessageDispatcherTest::testIsOutstandingRequest() {
  309. RequestSlot slot(0, 0, MY_PIECE_LENGTH, 0);
  310. btMessageDispatcher->addOutstandingRequest(slot);
  311. CPPUNIT_ASSERT(btMessageDispatcher->isOutstandingRequest(0, 0));
  312. CPPUNIT_ASSERT(!btMessageDispatcher->isOutstandingRequest(0, 1));
  313. CPPUNIT_ASSERT(!btMessageDispatcher->isOutstandingRequest(1, 0));
  314. CPPUNIT_ASSERT(!btMessageDispatcher->isOutstandingRequest(1, 1));
  315. }
  316. void DefaultBtMessageDispatcherTest::testGetOutstandingRequest() {
  317. RequestSlot slot(1, 1024, 16*1024, 10);
  318. btMessageDispatcher->addOutstandingRequest(slot);
  319. RequestSlot s2 = btMessageDispatcher->getOutstandingRequest(1, 1024, 16*1024);
  320. CPPUNIT_ASSERT(!RequestSlot::isNull(s2));
  321. RequestSlot s3 = btMessageDispatcher->getOutstandingRequest(1, 1024, 17*1024);
  322. CPPUNIT_ASSERT(RequestSlot::isNull(s3));
  323. RequestSlot s4 =
  324. btMessageDispatcher->getOutstandingRequest(1, 2*1024, 16*1024);
  325. CPPUNIT_ASSERT(RequestSlot::isNull(s4));
  326. RequestSlot s5 = btMessageDispatcher->getOutstandingRequest(2, 1024, 16*1024);
  327. CPPUNIT_ASSERT(RequestSlot::isNull(s5));
  328. }
  329. void DefaultBtMessageDispatcherTest::testRemoveOutstandingRequest() {
  330. SharedHandle<Piece> piece(new Piece(1, 1024*1024));
  331. size_t blockIndex = 0;
  332. CPPUNIT_ASSERT(piece->getMissingUnusedBlockIndex(blockIndex));
  333. uint32_t begin = blockIndex*piece->getBlockLength();
  334. size_t length = piece->getBlockLength(blockIndex);
  335. RequestSlot slot(piece->getIndex(), begin, length, blockIndex, piece);
  336. btMessageDispatcher->addOutstandingRequest(slot);
  337. RequestSlot s2 = btMessageDispatcher->getOutstandingRequest
  338. (piece->getIndex(), begin, length);
  339. CPPUNIT_ASSERT(!RequestSlot::isNull(s2));
  340. CPPUNIT_ASSERT(piece->isBlockUsed(blockIndex));
  341. btMessageDispatcher->removeOutstandingRequest(s2);
  342. RequestSlot s3 = btMessageDispatcher->getOutstandingRequest
  343. (piece->getIndex(), begin, length);
  344. CPPUNIT_ASSERT(RequestSlot::isNull(s3));
  345. CPPUNIT_ASSERT(!piece->isBlockUsed(blockIndex));
  346. }
  347. } // namespace aria2