mirror of https://github.com/aria2/aria2
				
				
				
			
		
			
				
	
	
		
			516 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
			
		
		
	
	
			516 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
| #include "DHTMessageFactoryImpl.h"
 | |
| 
 | |
| #include <cstring>
 | |
| #include <iostream>
 | |
| 
 | |
| #include <cppunit/extensions/HelperMacros.h>
 | |
| 
 | |
| #include "RecoverableException.h"
 | |
| #include "util.h"
 | |
| #include "DHTNode.h"
 | |
| #include "DHTRoutingTable.h"
 | |
| #include "Peer.h"
 | |
| #include "bittorrent_helper.h"
 | |
| #include "DHTBucket.h"
 | |
| #include "DHTPingMessage.h"
 | |
| #include "DHTPingReplyMessage.h"
 | |
| #include "DHTFindNodeMessage.h"
 | |
| #include "DHTFindNodeReplyMessage.h"
 | |
| #include "DHTGetPeersMessage.h"
 | |
| #include "DHTGetPeersReplyMessage.h"
 | |
| #include "DHTAnnouncePeerMessage.h"
 | |
| #include "DHTAnnouncePeerReplyMessage.h"
 | |
| #include "bencode2.h"
 | |
| 
 | |
| namespace aria2 {
 | |
| 
 | |
| class DHTMessageFactoryImplTest:public CppUnit::TestFixture {
 | |
| 
 | |
|   CPPUNIT_TEST_SUITE(DHTMessageFactoryImplTest);
 | |
|   CPPUNIT_TEST(testCreatePingMessage);
 | |
|   CPPUNIT_TEST(testCreatePingReplyMessage);
 | |
|   CPPUNIT_TEST(testCreateFindNodeMessage);
 | |
|   CPPUNIT_TEST(testCreateFindNodeReplyMessage);
 | |
|   CPPUNIT_TEST(testCreateFindNodeReplyMessage6);
 | |
|   CPPUNIT_TEST(testCreateGetPeersMessage);
 | |
|   CPPUNIT_TEST(testCreateGetPeersReplyMessage);
 | |
|   CPPUNIT_TEST(testCreateGetPeersReplyMessage6);
 | |
|   CPPUNIT_TEST(testCreateAnnouncePeerMessage);
 | |
|   CPPUNIT_TEST(testCreateAnnouncePeerReplyMessage);
 | |
|   CPPUNIT_TEST(testReceivedErrorMessage);
 | |
|   CPPUNIT_TEST_SUITE_END();
 | |
| public:
 | |
|   SharedHandle<DHTMessageFactoryImpl> factory;
 | |
| 
 | |
|   SharedHandle<DHTRoutingTable> routingTable;
 | |
| 
 | |
|   SharedHandle<DHTNode> localNode;
 | |
| 
 | |
|   unsigned char transactionID[DHT_TRANSACTION_ID_LENGTH];
 | |
| 
 | |
|   unsigned char remoteNodeID[DHT_ID_LENGTH];
 | |
| 
 | |
|   void setUp()
 | |
|   {
 | |
|     localNode.reset(new DHTNode());
 | |
|     factory.reset(new DHTMessageFactoryImpl(AF_INET));
 | |
|     factory->setLocalNode(localNode);
 | |
|     memset(transactionID, 0xff, DHT_TRANSACTION_ID_LENGTH);
 | |
|     memset(remoteNodeID, 0x0f, DHT_ID_LENGTH);
 | |
|     routingTable.reset(new DHTRoutingTable(localNode));
 | |
|     factory->setRoutingTable(routingTable.get());
 | |
|   }
 | |
| 
 | |
|   void tearDown() {}
 | |
| 
 | |
|   void testCreatePingMessage();
 | |
|   void testCreatePingReplyMessage();
 | |
|   void testCreateFindNodeMessage();
 | |
|   void testCreateFindNodeReplyMessage();
 | |
|   void testCreateFindNodeReplyMessage6();
 | |
|   void testCreateGetPeersMessage();
 | |
|   void testCreateGetPeersReplyMessage();
 | |
|   void testCreateGetPeersReplyMessage6();
 | |
|   void testCreateAnnouncePeerMessage();
 | |
|   void testCreateAnnouncePeerReplyMessage();
 | |
|   void testReceivedErrorMessage();
 | |
| };
 | |
| 
 | |
| 
 | |
| CPPUNIT_TEST_SUITE_REGISTRATION(DHTMessageFactoryImplTest);
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testCreatePingMessage()
 | |
| {
 | |
|   Dict dict;
 | |
|   dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|   dict.put("y", "q");
 | |
|   dict.put("q", "ping");
 | |
|   SharedHandle<Dict> aDict = Dict::g();
 | |
|   aDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));
 | |
|   dict.put("a", aDict);
 | |
|   
 | |
|   SharedHandle<DHTPingMessage> m
 | |
|     (dynamic_pointer_cast<DHTPingMessage>
 | |
|      (factory->createQueryMessage(&dict, "192.168.0.1", 6881)));
 | |
|   SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|   remoteNode->setIPAddress("192.168.0.1");
 | |
|   remoteNode->setPort(6881);
 | |
| 
 | |
|   CPPUNIT_ASSERT(*localNode == *m->getLocalNode());
 | |
|   CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());
 | |
|   CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
 | |
|                        util::toHex(m->getTransactionID()));
 | |
| }
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testCreatePingReplyMessage()
 | |
| {
 | |
|   Dict dict;
 | |
|   dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|   dict.put("y", "r");
 | |
|   SharedHandle<Dict> rDict = Dict::g();
 | |
|   rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));
 | |
|   dict.put("r", rDict);
 | |
| 
 | |
|   SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|   remoteNode->setIPAddress("192.168.0.1");
 | |
|   remoteNode->setPort(6881);
 | |
|   
 | |
|   SharedHandle<DHTPingReplyMessage> m
 | |
|     (dynamic_pointer_cast<DHTPingReplyMessage>
 | |
|      (factory->createResponseMessage("ping", &dict,
 | |
|                                      remoteNode->getIPAddress(),
 | |
|                                      remoteNode->getPort())));
 | |
| 
 | |
|   CPPUNIT_ASSERT(*localNode == *m->getLocalNode());
 | |
|   CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());
 | |
|   CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
 | |
|                        util::toHex(m->getTransactionID()));
 | |
| }
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testCreateFindNodeMessage()
 | |
| {
 | |
|   Dict dict;
 | |
|   dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|   dict.put("y", "q");
 | |
|   dict.put("q", "find_node");
 | |
|   SharedHandle<Dict> aDict = Dict::g();
 | |
|   aDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));
 | |
|   unsigned char targetNodeID[DHT_ID_LENGTH];
 | |
|   memset(targetNodeID, 0x11, DHT_ID_LENGTH);
 | |
|   aDict->put("target", String::g(targetNodeID, DHT_ID_LENGTH));
 | |
|   dict.put("a", aDict);
 | |
|   
 | |
|   SharedHandle<DHTFindNodeMessage> m
 | |
|     (dynamic_pointer_cast<DHTFindNodeMessage>
 | |
|      (factory->createQueryMessage(&dict, "192.168.0.1", 6881)));
 | |
|   SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|   remoteNode->setIPAddress("192.168.0.1");
 | |
|   remoteNode->setPort(6881);
 | |
| 
 | |
|   CPPUNIT_ASSERT(*localNode == *m->getLocalNode());
 | |
|   CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());
 | |
|   CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
 | |
|                        util::toHex(m->getTransactionID()));
 | |
|   CPPUNIT_ASSERT_EQUAL(util::toHex(targetNodeID, DHT_ID_LENGTH),
 | |
|                        util::toHex(m->getTargetNodeID(), DHT_ID_LENGTH));
 | |
| }
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testCreateFindNodeReplyMessage()
 | |
| {
 | |
|   try {
 | |
|     Dict dict;
 | |
|     dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|     dict.put("y", "r");
 | |
|     SharedHandle<Dict> rDict = Dict::g();
 | |
|     rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));
 | |
|     std::string compactNodeInfo;
 | |
|     SharedHandle<DHTNode> nodes[8];
 | |
|     for(size_t i = 0; i < DHTBucket::K; ++i) {
 | |
|       nodes[i].reset(new DHTNode());
 | |
|       nodes[i]->setIPAddress("192.168.0."+util::uitos(i+1));
 | |
|       nodes[i]->setPort(6881+i);
 | |
| 
 | |
|       unsigned char buf[COMPACT_LEN_IPV6];
 | |
|       CPPUNIT_ASSERT_EQUAL
 | |
|         (COMPACT_LEN_IPV4,
 | |
|          bittorrent::packcompact
 | |
|          (buf, nodes[i]->getIPAddress(), nodes[i]->getPort()));
 | |
|       compactNodeInfo +=
 | |
|         std::string(&nodes[i]->getID()[0], &nodes[i]->getID()[DHT_ID_LENGTH])+
 | |
|         std::string(&buf[0], &buf[COMPACT_LEN_IPV4]);
 | |
|     }
 | |
|     rDict->put("nodes", compactNodeInfo);
 | |
|     dict.put("r", rDict);
 | |
| 
 | |
|     SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|     remoteNode->setIPAddress("192.168.0.1");
 | |
|     remoteNode->setPort(6881);
 | |
|   
 | |
|     SharedHandle<DHTFindNodeReplyMessage> m
 | |
|       (dynamic_pointer_cast<DHTFindNodeReplyMessage>
 | |
|        (factory->createResponseMessage("find_node", &dict,
 | |
|                                        remoteNode->getIPAddress(),
 | |
|                                        remoteNode->getPort())));
 | |
| 
 | |
|     CPPUNIT_ASSERT(*localNode == *m->getLocalNode());
 | |
|     CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());
 | |
|     CPPUNIT_ASSERT_EQUAL((size_t)DHTBucket::K, m->getClosestKNodes().size());
 | |
|     CPPUNIT_ASSERT(*nodes[0] == *m->getClosestKNodes()[0]);
 | |
|     CPPUNIT_ASSERT(*nodes[7] == *m->getClosestKNodes()[7]);
 | |
|     CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
 | |
|                          util::toHex(m->getTransactionID()));
 | |
|   } catch(Exception& e) {
 | |
|     CPPUNIT_FAIL(e.stackTrace());
 | |
|   }
 | |
| }
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testCreateFindNodeReplyMessage6()
 | |
| {
 | |
|   factory.reset(new DHTMessageFactoryImpl(AF_INET6));
 | |
|   factory->setLocalNode(localNode);
 | |
|   factory->setRoutingTable(routingTable.get());
 | |
|   try {
 | |
|     Dict dict;
 | |
|     dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|     dict.put("y", "r");
 | |
|     SharedHandle<Dict> rDict = Dict::g();
 | |
|     rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));
 | |
|     std::string compactNodeInfo;
 | |
|     SharedHandle<DHTNode> nodes[8];
 | |
|     for(size_t i = 0; i < DHTBucket::K; ++i) {
 | |
|       nodes[i].reset(new DHTNode());
 | |
|       nodes[i]->setIPAddress("2001::000"+util::uitos(i+1));
 | |
|       nodes[i]->setPort(6881+i);
 | |
| 
 | |
|       unsigned char buf[COMPACT_LEN_IPV6];
 | |
|       CPPUNIT_ASSERT_EQUAL
 | |
|         (COMPACT_LEN_IPV6,
 | |
|          bittorrent::packcompact
 | |
|          (buf, nodes[i]->getIPAddress(), nodes[i]->getPort()));
 | |
|       compactNodeInfo +=
 | |
|         std::string(&nodes[i]->getID()[0], &nodes[i]->getID()[DHT_ID_LENGTH])+
 | |
|         std::string(&buf[0], &buf[COMPACT_LEN_IPV6]);
 | |
|     }
 | |
|     rDict->put("nodes6", compactNodeInfo);
 | |
|     dict.put("r", rDict);
 | |
| 
 | |
|     SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|     remoteNode->setIPAddress("2001::2001");
 | |
|     remoteNode->setPort(6881);
 | |
|   
 | |
|     SharedHandle<DHTFindNodeReplyMessage> m
 | |
|       (dynamic_pointer_cast<DHTFindNodeReplyMessage>
 | |
|        (factory->createResponseMessage("find_node", &dict,
 | |
|                                        remoteNode->getIPAddress(),
 | |
|                                        remoteNode->getPort())));
 | |
| 
 | |
|     CPPUNIT_ASSERT(*localNode == *m->getLocalNode());
 | |
|     CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());
 | |
|     CPPUNIT_ASSERT_EQUAL((size_t)DHTBucket::K, m->getClosestKNodes().size());
 | |
|     CPPUNIT_ASSERT(*nodes[0] == *m->getClosestKNodes()[0]);
 | |
|     CPPUNIT_ASSERT(*nodes[7] == *m->getClosestKNodes()[7]);
 | |
|     CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
 | |
|                          util::toHex(m->getTransactionID()));
 | |
|   } catch(Exception& e) {
 | |
|     CPPUNIT_FAIL(e.stackTrace());
 | |
|   }
 | |
| }
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testCreateGetPeersMessage()
 | |
| {
 | |
|   Dict dict;
 | |
|   dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|   dict.put("y", "q");
 | |
|   dict.put("q", "get_peers");
 | |
|   SharedHandle<Dict> aDict = Dict::g();
 | |
|   aDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));
 | |
|   unsigned char infoHash[DHT_ID_LENGTH];
 | |
|   memset(infoHash, 0x11, DHT_ID_LENGTH);
 | |
|   aDict->put("info_hash", String::g(infoHash, DHT_ID_LENGTH));
 | |
|   dict.put("a", aDict);
 | |
|   
 | |
|   SharedHandle<DHTGetPeersMessage> m
 | |
|     (dynamic_pointer_cast<DHTGetPeersMessage>
 | |
|      (factory->createQueryMessage(&dict, "192.168.0.1", 6881)));
 | |
|   SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|   remoteNode->setIPAddress("192.168.0.1");
 | |
|   remoteNode->setPort(6881);
 | |
| 
 | |
|   CPPUNIT_ASSERT(*localNode == *m->getLocalNode());
 | |
|   CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());
 | |
|   CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
 | |
|                        util::toHex(m->getTransactionID()));
 | |
|   CPPUNIT_ASSERT_EQUAL(util::toHex(infoHash, DHT_ID_LENGTH),
 | |
|                        util::toHex(m->getInfoHash(), DHT_ID_LENGTH));
 | |
| }
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testCreateGetPeersReplyMessage()
 | |
| {
 | |
|   try {
 | |
|     Dict dict;
 | |
|     dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|     dict.put("y", "r");
 | |
|     SharedHandle<Dict> rDict = Dict::g();
 | |
|     rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));
 | |
|     std::string compactNodeInfo;
 | |
|     SharedHandle<DHTNode> nodes[8];
 | |
|     for(size_t i = 0; i < DHTBucket::K; ++i) {
 | |
|       nodes[i].reset(new DHTNode());
 | |
|       nodes[i]->setIPAddress("192.168.0."+util::uitos(i+1));
 | |
|       nodes[i]->setPort(6881+i);
 | |
| 
 | |
|       unsigned char buf[COMPACT_LEN_IPV6];
 | |
|       CPPUNIT_ASSERT_EQUAL
 | |
|         (COMPACT_LEN_IPV4,
 | |
|          bittorrent::packcompact
 | |
|          (buf, nodes[i]->getIPAddress(), nodes[i]->getPort()));
 | |
|       compactNodeInfo +=
 | |
|         std::string(&nodes[i]->getID()[0], &nodes[i]->getID()[DHT_ID_LENGTH])+
 | |
|         std::string(&buf[0], &buf[COMPACT_LEN_IPV4]);
 | |
|     }
 | |
|     rDict->put("nodes", compactNodeInfo);
 | |
| 
 | |
|     std::deque<SharedHandle<Peer> > peers;
 | |
|     SharedHandle<List> valuesList = List::g();
 | |
|     for(size_t i = 0; i < 4; ++i) {
 | |
|       SharedHandle<Peer> peer(new Peer("192.168.0."+util::uitos(i+1), 6881+i));
 | |
|       unsigned char buffer[COMPACT_LEN_IPV6];
 | |
|       CPPUNIT_ASSERT_EQUAL
 | |
|         (COMPACT_LEN_IPV4,
 | |
|          bittorrent::packcompact
 | |
|          (buffer, peer->getIPAddress(), peer->getPort()));
 | |
|       valuesList->append(String::g(buffer, COMPACT_LEN_IPV4));
 | |
|       peers.push_back(peer);
 | |
|     }
 | |
|     rDict->put("values", valuesList);
 | |
| 
 | |
|     rDict->put("token", "token");
 | |
|     dict.put("r", rDict);
 | |
| 
 | |
|     SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|     remoteNode->setIPAddress("192.168.0.1");
 | |
|     remoteNode->setPort(6881);
 | |
|   
 | |
|     SharedHandle<DHTGetPeersReplyMessage> m
 | |
|       (dynamic_pointer_cast<DHTGetPeersReplyMessage>
 | |
|        (factory->createResponseMessage("get_peers", &dict,
 | |
|                                        remoteNode->getIPAddress(),
 | |
|                                        remoteNode->getPort())));
 | |
| 
 | |
|     CPPUNIT_ASSERT(*localNode == *m->getLocalNode());
 | |
|     CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());
 | |
|     CPPUNIT_ASSERT_EQUAL(std::string("token"), m->getToken());
 | |
|     CPPUNIT_ASSERT_EQUAL((size_t)DHTBucket::K, m->getClosestKNodes().size());
 | |
|     CPPUNIT_ASSERT(*nodes[0] == *m->getClosestKNodes()[0]);
 | |
|     CPPUNIT_ASSERT(*nodes[7] == *m->getClosestKNodes()[7]);
 | |
|     CPPUNIT_ASSERT_EQUAL((size_t)4, m->getValues().size());
 | |
|     CPPUNIT_ASSERT(*peers[0] == *m->getValues()[0]);
 | |
|     CPPUNIT_ASSERT(*peers[3] == *m->getValues()[3]);
 | |
|     CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
 | |
|                          util::toHex(m->getTransactionID()));
 | |
|   } catch(Exception& e) {
 | |
|     CPPUNIT_FAIL(e.stackTrace());
 | |
|   }
 | |
| }
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testCreateGetPeersReplyMessage6()
 | |
| {
 | |
|   factory.reset(new DHTMessageFactoryImpl(AF_INET6));
 | |
|   factory->setLocalNode(localNode);
 | |
|   factory->setRoutingTable(routingTable.get());
 | |
|   try {
 | |
|     Dict dict;
 | |
|     dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|     dict.put("y", "r");
 | |
|     SharedHandle<Dict> rDict = Dict::g();
 | |
|     rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));
 | |
|     std::string compactNodeInfo;
 | |
|     SharedHandle<DHTNode> nodes[8];
 | |
|     for(size_t i = 0; i < DHTBucket::K; ++i) {
 | |
|       nodes[i].reset(new DHTNode());
 | |
|       nodes[i]->setIPAddress("2001::000"+util::uitos(i+1));
 | |
|       nodes[i]->setPort(6881+i);
 | |
| 
 | |
|       unsigned char buf[COMPACT_LEN_IPV6];
 | |
|       CPPUNIT_ASSERT_EQUAL
 | |
|         (COMPACT_LEN_IPV6,
 | |
|          bittorrent::packcompact
 | |
|          (buf, nodes[i]->getIPAddress(), nodes[i]->getPort()));
 | |
|       compactNodeInfo +=
 | |
|         std::string(&nodes[i]->getID()[0], &nodes[i]->getID()[DHT_ID_LENGTH])+
 | |
|         std::string(&buf[0], &buf[COMPACT_LEN_IPV6]);
 | |
|     }
 | |
|     rDict->put("nodes6", compactNodeInfo);
 | |
| 
 | |
|     std::deque<SharedHandle<Peer> > peers;
 | |
|     SharedHandle<List> valuesList = List::g();
 | |
|     for(size_t i = 0; i < 4; ++i) {
 | |
|       SharedHandle<Peer> peer(new Peer("2001::100"+util::uitos(i+1), 6881+i));
 | |
|       unsigned char buffer[COMPACT_LEN_IPV6];
 | |
|       CPPUNIT_ASSERT_EQUAL
 | |
|         (COMPACT_LEN_IPV6,
 | |
|          bittorrent::packcompact
 | |
|          (buffer, peer->getIPAddress(), peer->getPort()));
 | |
|       valuesList->append(String::g(buffer, COMPACT_LEN_IPV6));
 | |
|       peers.push_back(peer);
 | |
|     }
 | |
|     rDict->put("values", valuesList);
 | |
| 
 | |
|     rDict->put("token", "token");
 | |
|     dict.put("r", rDict);
 | |
| 
 | |
|     SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|     remoteNode->setIPAddress("2001::2001");
 | |
|     remoteNode->setPort(6881);
 | |
|   
 | |
|     SharedHandle<DHTGetPeersReplyMessage> m
 | |
|       (dynamic_pointer_cast<DHTGetPeersReplyMessage>
 | |
|        (factory->createResponseMessage("get_peers", &dict,
 | |
|                                        remoteNode->getIPAddress(),
 | |
|                                        remoteNode->getPort())));
 | |
| 
 | |
|     CPPUNIT_ASSERT(*localNode == *m->getLocalNode());
 | |
|     CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());
 | |
|     CPPUNIT_ASSERT_EQUAL(std::string("token"), m->getToken());
 | |
|     CPPUNIT_ASSERT_EQUAL((size_t)DHTBucket::K, m->getClosestKNodes().size());
 | |
|     CPPUNIT_ASSERT(*nodes[0] == *m->getClosestKNodes()[0]);
 | |
|     CPPUNIT_ASSERT(*nodes[7] == *m->getClosestKNodes()[7]);
 | |
|     CPPUNIT_ASSERT_EQUAL((size_t)4, m->getValues().size());
 | |
|     CPPUNIT_ASSERT(*peers[0] == *m->getValues()[0]);
 | |
|     CPPUNIT_ASSERT(*peers[3] == *m->getValues()[3]);
 | |
|     CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
 | |
|                          util::toHex(m->getTransactionID()));
 | |
|   } catch(Exception& e) {
 | |
|     CPPUNIT_FAIL(e.stackTrace());
 | |
|   }
 | |
| }
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testCreateAnnouncePeerMessage()
 | |
| {
 | |
|   try {
 | |
|     Dict dict;
 | |
|     dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|     dict.put("y", "q");
 | |
|     dict.put("q", "announce_peer");
 | |
|     SharedHandle<Dict> aDict = Dict::g();
 | |
|     aDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));
 | |
|     unsigned char infoHash[DHT_ID_LENGTH];
 | |
|     memset(infoHash, 0x11, DHT_ID_LENGTH);
 | |
|     aDict->put("info_hash", String::g(infoHash, DHT_ID_LENGTH));
 | |
|     std::string token = "ffff";
 | |
|     uint16_t port = 6881;
 | |
|     aDict->put("port", Integer::g(port));
 | |
|     aDict->put("token", token);
 | |
|     dict.put("a", aDict);
 | |
|   
 | |
|     SharedHandle<DHTAnnouncePeerMessage> m
 | |
|       (dynamic_pointer_cast<DHTAnnouncePeerMessage>
 | |
|        (factory->createQueryMessage(&dict, "192.168.0.1", 6882)));
 | |
|     SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|     remoteNode->setIPAddress("192.168.0.1");
 | |
|     remoteNode->setPort(6882);
 | |
| 
 | |
|     CPPUNIT_ASSERT(*localNode == *m->getLocalNode());
 | |
|     CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());
 | |
|     CPPUNIT_ASSERT_EQUAL(token, m->getToken());
 | |
|     CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
 | |
|                          util::toHex(m->getTransactionID()));
 | |
|     CPPUNIT_ASSERT_EQUAL(util::toHex(infoHash, DHT_ID_LENGTH),
 | |
|                          util::toHex(m->getInfoHash(), DHT_ID_LENGTH));
 | |
|     CPPUNIT_ASSERT_EQUAL(port, m->getTCPPort());
 | |
|   } catch(Exception& e) {
 | |
|     CPPUNIT_FAIL(e.stackTrace());
 | |
|   }
 | |
| }
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testCreateAnnouncePeerReplyMessage()
 | |
| {
 | |
|   Dict dict;
 | |
|   dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|   dict.put("y", "r");
 | |
|   SharedHandle<Dict> rDict = Dict::g();
 | |
|   rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));
 | |
|   dict.put("r", rDict);
 | |
| 
 | |
|   SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|   remoteNode->setIPAddress("192.168.0.1");
 | |
|   remoteNode->setPort(6881);
 | |
|   
 | |
|   SharedHandle<DHTAnnouncePeerReplyMessage> m
 | |
|     (dynamic_pointer_cast<DHTAnnouncePeerReplyMessage>
 | |
|      (factory->createResponseMessage("announce_peer", &dict,
 | |
|                                      remoteNode->getIPAddress(),
 | |
|                                      remoteNode->getPort())));
 | |
| 
 | |
|   CPPUNIT_ASSERT(*localNode == *m->getLocalNode());
 | |
|   CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());
 | |
|   CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),
 | |
|                        util::toHex(m->getTransactionID()));
 | |
| }
 | |
| 
 | |
| void DHTMessageFactoryImplTest::testReceivedErrorMessage()
 | |
| {
 | |
|   Dict dict;
 | |
|   dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));
 | |
|   dict.put("y", "e");
 | |
|   SharedHandle<List> list = List::g();
 | |
|   list->append(Integer::g(404));
 | |
|   list->append("Not found");
 | |
|   dict.put("e", list);
 | |
| 
 | |
|   SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));
 | |
|   remoteNode->setIPAddress("192.168.0.1");
 | |
|   remoteNode->setPort(6881);
 | |
| 
 | |
|   try {
 | |
|     factory->createResponseMessage("announce_peer", &dict,
 | |
|                                    remoteNode->getIPAddress(),
 | |
|                                    remoteNode->getPort());
 | |
|     CPPUNIT_FAIL("exception must be thrown.");
 | |
|   } catch(RecoverableException& e) {
 | |
|     std::cerr << e.stackTrace() << std::endl;
 | |
|   }
 | |
| }
 | |
| 
 | |
| } // namespace aria2
 |