DHTMessageFactoryImplTest.cc 15 KB

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