DHTMessageFactoryImplTest.cc 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413
  1. #include "DHTMessageFactoryImpl.h"
  2. #include <cstring>
  3. #include <iostream>
  4. #include <cppunit/extensions/HelperMacros.h>
  5. #include "RecoverableException.h"
  6. #include "util.h"
  7. #include "DHTNode.h"
  8. #include "DHTRoutingTable.h"
  9. #include "Peer.h"
  10. #include "bittorrent_helper.h"
  11. #include "DHTBucket.h"
  12. #include "DHTPingMessage.h"
  13. #include "DHTPingReplyMessage.h"
  14. #include "DHTFindNodeMessage.h"
  15. #include "DHTFindNodeReplyMessage.h"
  16. #include "DHTGetPeersMessage.h"
  17. #include "DHTGetPeersReplyMessage.h"
  18. #include "DHTAnnouncePeerMessage.h"
  19. #include "DHTAnnouncePeerReplyMessage.h"
  20. #include "bencode.h"
  21. namespace aria2 {
  22. class DHTMessageFactoryImplTest:public CppUnit::TestFixture {
  23. CPPUNIT_TEST_SUITE(DHTMessageFactoryImplTest);
  24. CPPUNIT_TEST(testCreatePingMessage);
  25. CPPUNIT_TEST(testCreatePingReplyMessage);
  26. CPPUNIT_TEST(testCreateFindNodeMessage);
  27. CPPUNIT_TEST(testCreateFindNodeReplyMessage);
  28. CPPUNIT_TEST(testCreateGetPeersMessage);
  29. CPPUNIT_TEST(testCreateGetPeersReplyMessage_nodes);
  30. CPPUNIT_TEST(testCreateGetPeersReplyMessage_values);
  31. CPPUNIT_TEST(testCreateAnnouncePeerMessage);
  32. CPPUNIT_TEST(testCreateAnnouncePeerReplyMessage);
  33. CPPUNIT_TEST(testReceivedErrorMessage);
  34. CPPUNIT_TEST_SUITE_END();
  35. public:
  36. SharedHandle<DHTMessageFactoryImpl> factory;
  37. SharedHandle<DHTRoutingTable> routingTable;
  38. SharedHandle<DHTNode> localNode;
  39. unsigned char transactionID[DHT_TRANSACTION_ID_LENGTH];
  40. unsigned char remoteNodeID[DHT_ID_LENGTH];
  41. void setUp()
  42. {
  43. localNode.reset(new DHTNode());
  44. factory.reset(new DHTMessageFactoryImpl());
  45. factory->setLocalNode(localNode);
  46. memset(transactionID, 0xff, DHT_TRANSACTION_ID_LENGTH);
  47. memset(remoteNodeID, 0x0f, DHT_ID_LENGTH);
  48. routingTable.reset(new DHTRoutingTable(localNode));
  49. factory->setRoutingTable(routingTable);
  50. }
  51. void tearDown() {}
  52. void testCreatePingMessage();
  53. void testCreatePingReplyMessage();
  54. void testCreateFindNodeMessage();
  55. void testCreateFindNodeReplyMessage();
  56. void testCreateGetPeersMessage();
  57. void testCreateGetPeersReplyMessage_nodes();
  58. void testCreateGetPeersReplyMessage_values();
  59. void testCreateAnnouncePeerMessage();
  60. void testCreateAnnouncePeerReplyMessage();
  61. void testReceivedErrorMessage();
  62. };
  63. CPPUNIT_TEST_SUITE_REGISTRATION(DHTMessageFactoryImplTest);
  64. void DHTMessageFactoryImplTest::testCreatePingMessage()
  65. {
  66. BDE dict = BDE::dict();
  67. dict["t"] = BDE(transactionID, DHT_TRANSACTION_ID_LENGTH);
  68. dict["y"] = BDE("q");
  69. dict["q"] = BDE("ping");
  70. BDE aDict = BDE::dict();
  71. aDict["id"] = BDE(remoteNodeID, DHT_ID_LENGTH);
  72. dict["a"] = aDict;
  73. SharedHandle<DHTPingMessage> m
  74. (dynamic_pointer_cast<DHTPingMessage>
  75. (factory->createQueryMessage(dict, "192.168.0.1", 6881)));
  76. SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
  77. remoteNode->setIPAddress("192.168.0.1");
  78. remoteNode->setPort(6881);
  79. CPPUNIT_ASSERT(localNode == m->getLocalNode());
  80. CPPUNIT_ASSERT(remoteNode == m->getRemoteNode());
  81. CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
  82. util::toHex(m->getTransactionID()));
  83. }
  84. void DHTMessageFactoryImplTest::testCreatePingReplyMessage()
  85. {
  86. BDE dict = BDE::dict();
  87. dict["t"] = BDE(transactionID, DHT_TRANSACTION_ID_LENGTH);
  88. dict["y"] = BDE("r");
  89. BDE rDict = BDE::dict();
  90. rDict["id"] = BDE(remoteNodeID, DHT_ID_LENGTH);
  91. dict["r"] = rDict;
  92. SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
  93. remoteNode->setIPAddress("192.168.0.1");
  94. remoteNode->setPort(6881);
  95. SharedHandle<DHTPingReplyMessage> m
  96. (dynamic_pointer_cast<DHTPingReplyMessage>
  97. (factory->createResponseMessage("ping", dict,
  98. remoteNode->getIPAddress(),
  99. remoteNode->getPort())));
  100. CPPUNIT_ASSERT(localNode == m->getLocalNode());
  101. CPPUNIT_ASSERT(remoteNode == m->getRemoteNode());
  102. CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
  103. util::toHex(m->getTransactionID()));
  104. }
  105. void DHTMessageFactoryImplTest::testCreateFindNodeMessage()
  106. {
  107. BDE dict = BDE::dict();
  108. dict["t"] = BDE(transactionID, DHT_TRANSACTION_ID_LENGTH);
  109. dict["y"] = BDE("q");
  110. dict["q"] = BDE("find_node");
  111. BDE aDict = BDE::dict();
  112. aDict["id"] = BDE(remoteNodeID, DHT_ID_LENGTH);
  113. unsigned char targetNodeID[DHT_ID_LENGTH];
  114. memset(targetNodeID, 0x11, DHT_ID_LENGTH);
  115. aDict["target"] = BDE(targetNodeID, DHT_ID_LENGTH);
  116. dict["a"] = aDict;
  117. SharedHandle<DHTFindNodeMessage> m
  118. (dynamic_pointer_cast<DHTFindNodeMessage>
  119. (factory->createQueryMessage(dict, "192.168.0.1", 6881)));
  120. SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
  121. remoteNode->setIPAddress("192.168.0.1");
  122. remoteNode->setPort(6881);
  123. CPPUNIT_ASSERT(localNode == m->getLocalNode());
  124. CPPUNIT_ASSERT(remoteNode == m->getRemoteNode());
  125. CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
  126. util::toHex(m->getTransactionID()));
  127. CPPUNIT_ASSERT_EQUAL(util::toHex(targetNodeID, DHT_ID_LENGTH),
  128. util::toHex(m->getTargetNodeID(), DHT_ID_LENGTH));
  129. }
  130. void DHTMessageFactoryImplTest::testCreateFindNodeReplyMessage()
  131. {
  132. try {
  133. BDE dict = BDE::dict();
  134. dict["t"] = BDE(transactionID, DHT_TRANSACTION_ID_LENGTH);
  135. dict["y"] = BDE("r");
  136. BDE rDict = BDE::dict();
  137. rDict["id"] = BDE(remoteNodeID, DHT_ID_LENGTH);
  138. std::string compactNodeInfo;
  139. SharedHandle<DHTNode> nodes[8];
  140. for(size_t i = 0; i < DHTBucket::K; ++i) {
  141. nodes[i].reset(new DHTNode());
  142. nodes[i]->setIPAddress("192.168.0."+util::uitos(i+1));
  143. nodes[i]->setPort(6881+i);
  144. unsigned char buf[6];
  145. CPPUNIT_ASSERT(bittorrent::createcompact
  146. (buf, nodes[i]->getIPAddress(), nodes[i]->getPort()));
  147. compactNodeInfo +=
  148. std::string(&nodes[i]->getID()[0], &nodes[i]->getID()[DHT_ID_LENGTH])+
  149. std::string(&buf[0], &buf[sizeof(buf)]);
  150. }
  151. rDict["nodes"] = compactNodeInfo;
  152. dict["r"] = rDict;
  153. SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
  154. remoteNode->setIPAddress("192.168.0.1");
  155. remoteNode->setPort(6881);
  156. SharedHandle<DHTFindNodeReplyMessage> m
  157. (dynamic_pointer_cast<DHTFindNodeReplyMessage>
  158. (factory->createResponseMessage("find_node", dict,
  159. remoteNode->getIPAddress(),
  160. remoteNode->getPort())));
  161. CPPUNIT_ASSERT(localNode == m->getLocalNode());
  162. CPPUNIT_ASSERT(remoteNode == m->getRemoteNode());
  163. CPPUNIT_ASSERT_EQUAL((size_t)DHTBucket::K, m->getClosestKNodes().size());
  164. CPPUNIT_ASSERT(nodes[0] == m->getClosestKNodes()[0]);
  165. CPPUNIT_ASSERT(nodes[7] == m->getClosestKNodes()[7]);
  166. CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
  167. util::toHex(m->getTransactionID()));
  168. } catch(Exception& e) {
  169. CPPUNIT_FAIL(e.stackTrace());
  170. }
  171. }
  172. void DHTMessageFactoryImplTest::testCreateGetPeersMessage()
  173. {
  174. BDE dict = BDE::dict();
  175. dict["t"] = BDE(transactionID, DHT_TRANSACTION_ID_LENGTH);
  176. dict["y"] = BDE("q");
  177. dict["q"] = BDE("get_peers");
  178. BDE aDict = BDE::dict();
  179. aDict["id"] = BDE(remoteNodeID, DHT_ID_LENGTH);
  180. unsigned char infoHash[DHT_ID_LENGTH];
  181. memset(infoHash, 0x11, DHT_ID_LENGTH);
  182. aDict["info_hash"] = BDE(infoHash, DHT_ID_LENGTH);
  183. dict["a"] = aDict;
  184. SharedHandle<DHTGetPeersMessage> m
  185. (dynamic_pointer_cast<DHTGetPeersMessage>
  186. (factory->createQueryMessage(dict, "192.168.0.1", 6881)));
  187. SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
  188. remoteNode->setIPAddress("192.168.0.1");
  189. remoteNode->setPort(6881);
  190. CPPUNIT_ASSERT(localNode == m->getLocalNode());
  191. CPPUNIT_ASSERT(remoteNode == m->getRemoteNode());
  192. CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
  193. util::toHex(m->getTransactionID()));
  194. CPPUNIT_ASSERT_EQUAL(util::toHex(infoHash, DHT_ID_LENGTH),
  195. util::toHex(m->getInfoHash(), DHT_ID_LENGTH));
  196. }
  197. void DHTMessageFactoryImplTest::testCreateGetPeersReplyMessage_nodes()
  198. {
  199. try {
  200. BDE dict = BDE::dict();
  201. dict["t"] = BDE(transactionID, DHT_TRANSACTION_ID_LENGTH);
  202. dict["y"] = BDE("r");
  203. BDE rDict = BDE::dict();
  204. rDict["id"] = BDE(remoteNodeID, DHT_ID_LENGTH);
  205. std::string compactNodeInfo;
  206. SharedHandle<DHTNode> nodes[8];
  207. for(size_t i = 0; i < DHTBucket::K; ++i) {
  208. nodes[i].reset(new DHTNode());
  209. nodes[i]->setIPAddress("192.168.0."+util::uitos(i+1));
  210. nodes[i]->setPort(6881+i);
  211. unsigned char buf[6];
  212. CPPUNIT_ASSERT(bittorrent::createcompact
  213. (buf, nodes[i]->getIPAddress(), nodes[i]->getPort()));
  214. compactNodeInfo +=
  215. std::string(&nodes[i]->getID()[0], &nodes[i]->getID()[DHT_ID_LENGTH])+
  216. std::string(&buf[0], &buf[sizeof(buf)]);
  217. }
  218. rDict["nodes"] = compactNodeInfo;
  219. rDict["token"] = BDE("token");
  220. dict["r"] = rDict;
  221. SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
  222. remoteNode->setIPAddress("192.168.0.1");
  223. remoteNode->setPort(6881);
  224. SharedHandle<DHTGetPeersReplyMessage> m
  225. (dynamic_pointer_cast<DHTGetPeersReplyMessage>
  226. (factory->createResponseMessage("get_peers", dict,
  227. remoteNode->getIPAddress(),
  228. remoteNode->getPort())));
  229. CPPUNIT_ASSERT(localNode == m->getLocalNode());
  230. CPPUNIT_ASSERT(remoteNode == m->getRemoteNode());
  231. CPPUNIT_ASSERT_EQUAL(std::string("token"), m->getToken());
  232. CPPUNIT_ASSERT_EQUAL((size_t)DHTBucket::K, m->getClosestKNodes().size());
  233. CPPUNIT_ASSERT(nodes[0] == m->getClosestKNodes()[0]);
  234. CPPUNIT_ASSERT(nodes[7] == m->getClosestKNodes()[7]);
  235. CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
  236. util::toHex(m->getTransactionID()));
  237. } catch(Exception& e) {
  238. CPPUNIT_FAIL(e.stackTrace());
  239. }
  240. }
  241. void DHTMessageFactoryImplTest::testCreateGetPeersReplyMessage_values()
  242. {
  243. try {
  244. BDE dict = BDE::dict();
  245. dict["t"] = BDE(transactionID, DHT_TRANSACTION_ID_LENGTH);
  246. dict["y"] = BDE("r");
  247. BDE rDict = BDE::dict();
  248. rDict["id"] = BDE(remoteNodeID, DHT_ID_LENGTH);
  249. std::deque<SharedHandle<Peer> > peers;
  250. BDE valuesList = BDE::list();
  251. for(size_t i = 0; i < 4; ++i) {
  252. SharedHandle<Peer> peer(new Peer("192.168.0."+util::uitos(i+1), 6881+i));
  253. unsigned char buffer[6];
  254. CPPUNIT_ASSERT(bittorrent::createcompact
  255. (buffer, peer->getIPAddress(), peer->getPort()));
  256. valuesList << BDE(buffer, sizeof(buffer));
  257. peers.push_back(peer);
  258. }
  259. rDict["values"] = valuesList;
  260. rDict["token"] = BDE("token");
  261. dict["r"] = rDict;
  262. SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
  263. remoteNode->setIPAddress("192.168.0.1");
  264. remoteNode->setPort(6881);
  265. SharedHandle<DHTGetPeersReplyMessage> m
  266. (dynamic_pointer_cast<DHTGetPeersReplyMessage>
  267. (factory->createResponseMessage("get_peers", dict,
  268. remoteNode->getIPAddress(),
  269. remoteNode->getPort())));
  270. CPPUNIT_ASSERT(localNode == m->getLocalNode());
  271. CPPUNIT_ASSERT(remoteNode == m->getRemoteNode());
  272. CPPUNIT_ASSERT_EQUAL(std::string("token"), m->getToken());
  273. CPPUNIT_ASSERT_EQUAL((size_t)4, m->getValues().size());
  274. CPPUNIT_ASSERT(peers[0] == m->getValues()[0]);
  275. CPPUNIT_ASSERT(peers[3] == m->getValues()[3]);
  276. CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
  277. util::toHex(m->getTransactionID()));
  278. } catch(Exception& e) {
  279. CPPUNIT_FAIL(e.stackTrace());
  280. }
  281. }
  282. void DHTMessageFactoryImplTest::testCreateAnnouncePeerMessage()
  283. {
  284. try {
  285. BDE dict = BDE::dict();
  286. dict["t"] = BDE(transactionID, DHT_TRANSACTION_ID_LENGTH);
  287. dict["y"] = BDE("q");
  288. dict["q"] = BDE("announce_peer");
  289. BDE aDict = BDE::dict();
  290. aDict["id"] = BDE(remoteNodeID, DHT_ID_LENGTH);
  291. unsigned char infoHash[DHT_ID_LENGTH];
  292. memset(infoHash, 0x11, DHT_ID_LENGTH);
  293. aDict["info_hash"] = BDE(infoHash, DHT_ID_LENGTH);
  294. std::string token = "ffff";
  295. uint16_t port = 6881;
  296. aDict["port"] = port;
  297. aDict["token"] = token;
  298. dict["a"] = aDict;
  299. SharedHandle<DHTAnnouncePeerMessage> m
  300. (dynamic_pointer_cast<DHTAnnouncePeerMessage>
  301. (factory->createQueryMessage(dict, "192.168.0.1", 6882)));
  302. SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
  303. remoteNode->setIPAddress("192.168.0.1");
  304. remoteNode->setPort(6882);
  305. CPPUNIT_ASSERT(localNode == m->getLocalNode());
  306. CPPUNIT_ASSERT(remoteNode == m->getRemoteNode());
  307. CPPUNIT_ASSERT_EQUAL(token, m->getToken());
  308. CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
  309. util::toHex(m->getTransactionID()));
  310. CPPUNIT_ASSERT_EQUAL(util::toHex(infoHash, DHT_ID_LENGTH),
  311. util::toHex(m->getInfoHash(), DHT_ID_LENGTH));
  312. CPPUNIT_ASSERT_EQUAL(port, m->getTCPPort());
  313. } catch(Exception& e) {
  314. CPPUNIT_FAIL(e.stackTrace());
  315. }
  316. }
  317. void DHTMessageFactoryImplTest::testCreateAnnouncePeerReplyMessage()
  318. {
  319. BDE dict = BDE::dict();
  320. dict["t"] = BDE(transactionID, DHT_TRANSACTION_ID_LENGTH);
  321. dict["y"] = BDE("r");
  322. BDE rDict = BDE::dict();
  323. rDict["id"] = BDE(remoteNodeID, DHT_ID_LENGTH);
  324. dict["r"] = rDict;
  325. SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
  326. remoteNode->setIPAddress("192.168.0.1");
  327. remoteNode->setPort(6881);
  328. SharedHandle<DHTAnnouncePeerReplyMessage> m
  329. (dynamic_pointer_cast<DHTAnnouncePeerReplyMessage>
  330. (factory->createResponseMessage("announce_peer", dict,
  331. remoteNode->getIPAddress(),
  332. remoteNode->getPort())));
  333. CPPUNIT_ASSERT(localNode == m->getLocalNode());
  334. CPPUNIT_ASSERT(remoteNode == m->getRemoteNode());
  335. CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
  336. util::toHex(m->getTransactionID()));
  337. }
  338. void DHTMessageFactoryImplTest::testReceivedErrorMessage()
  339. {
  340. BDE dict = BDE::dict();
  341. dict["t"] = BDE(transactionID, DHT_TRANSACTION_ID_LENGTH);
  342. dict["y"] = BDE("e");
  343. BDE list = BDE::list();
  344. list << 404;
  345. list << BDE("Not found");
  346. dict["e"] = list;
  347. SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
  348. remoteNode->setIPAddress("192.168.0.1");
  349. remoteNode->setPort(6881);
  350. try {
  351. factory->createResponseMessage("announce_peer", dict,
  352. remoteNode->getIPAddress(),
  353. remoteNode->getPort());
  354. CPPUNIT_FAIL("exception must be thrown.");
  355. } catch(RecoverableException& e) {
  356. std::cerr << e.stackTrace() << std::endl;
  357. }
  358. }
  359. } // namespace aria2