DHTMessageFactoryImplTest.cc 14 KB

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