#include "DHTGetPeersMessage.h" #include "DHTNode.h" #include "DHTUtil.h" #include "BencodeVisitor.h" #include "Dictionary.h" #include "Data.h" #include "Exception.h" #include "Util.h" #include "MockDHTMessageFactory.h" #include "MockDHTMessage.h" #include "MockDHTMessageDispatcher.h" #include "DHTTokenTracker.h" #include "DHTPeerAnnounceStorage.h" #include "DHTRoutingTable.h" #include namespace aria2 { class DHTGetPeersMessageTest:public CppUnit::TestFixture { CPPUNIT_TEST_SUITE(DHTGetPeersMessageTest); CPPUNIT_TEST(testGetBencodedMessage); CPPUNIT_TEST(testDoReceivedAction); CPPUNIT_TEST_SUITE_END(); public: void setUp() {} void tearDown() {} void testGetBencodedMessage(); void testDoReceivedAction(); class MockDHTMessageFactory2:public MockDHTMessageFactory { public: virtual SharedHandle createGetPeersReplyMessage(const SharedHandle& remoteNode, const std::deque >& peers, const std::string& token, const std::string& transactionID) { SharedHandle m (new MockDHTMessage(_localNode, remoteNode, "get_peers", transactionID)); m->_peers = peers; m->_token = token; return m; } virtual SharedHandle createGetPeersReplyMessage(const SharedHandle& remoteNode, const std::deque >& closestKNodes, const std::string& token, const std::string& transactionID) { SharedHandle m (new MockDHTMessage(_localNode, remoteNode, "get_peers", transactionID)); m->_nodes = closestKNodes; m->_token = token; return m; } }; }; CPPUNIT_TEST_SUITE_REGISTRATION(DHTGetPeersMessageTest); void DHTGetPeersMessageTest::testGetBencodedMessage() { SharedHandle localNode(new DHTNode()); SharedHandle remoteNode(new DHTNode()); unsigned char tid[DHT_TRANSACTION_ID_LENGTH]; DHTUtil::generateRandomData(tid, DHT_TRANSACTION_ID_LENGTH); std::string transactionID(&tid[0], &tid[DHT_TRANSACTION_ID_LENGTH]); unsigned char infoHash[DHT_ID_LENGTH]; DHTUtil::generateRandomData(infoHash, DHT_ID_LENGTH); DHTGetPeersMessage msg(localNode, remoteNode, infoHash, transactionID); std::string msgbody = msg.getBencodedMessage(); SharedHandle cm(new Dictionary()); cm->put("t", new Data(transactionID)); cm->put("y", new Data("q")); cm->put("q", new Data("get_peers")); Dictionary* a = new Dictionary(); cm->put("a", a); a->put("id", new Data(localNode->getID(), DHT_ID_LENGTH)); a->put("info_hash", new Data(infoHash, DHT_ID_LENGTH)); BencodeVisitor v; cm->accept(&v); CPPUNIT_ASSERT_EQUAL(Util::urlencode(v.getBencodedData()), Util::urlencode(msgbody)); } void DHTGetPeersMessageTest::testDoReceivedAction() { SharedHandle localNode(new DHTNode()); SharedHandle remoteNode(new DHTNode()); remoteNode->setIPAddress("192.168.0.1"); remoteNode->setPort(6881); unsigned char tid[DHT_TRANSACTION_ID_LENGTH]; DHTUtil::generateRandomData(tid, DHT_TRANSACTION_ID_LENGTH); std::string transactionID(&tid[0], &tid[DHT_TRANSACTION_ID_LENGTH]); unsigned char infoHash[DHT_ID_LENGTH]; DHTUtil::generateRandomData(infoHash, DHT_ID_LENGTH); DHTTokenTracker tokenTracker; MockDHTMessageDispatcher dispatcher; MockDHTMessageFactory2 factory; factory.setLocalNode(localNode); DHTGetPeersMessage msg(localNode, remoteNode, infoHash, transactionID); msg.setTokenTracker(WeakHandle (&tokenTracker)); msg.setMessageDispatcher(WeakHandle (&dispatcher)); msg.setMessageFactory(WeakHandle (&factory)); { // localhost has peer contact information for that infohash. DHTPeerAnnounceStorage peerAnnounceStorage; peerAnnounceStorage.addPeerAnnounce(infoHash, "192.168.0.100", 6888); peerAnnounceStorage.addPeerAnnounce(infoHash, "192.168.0.101", 6889); msg.setPeerAnnounceStorage(WeakHandle (&peerAnnounceStorage)); msg.doReceivedAction(); CPPUNIT_ASSERT_EQUAL((size_t)1, dispatcher._messageQueue.size()); SharedHandle m (dynamic_pointer_cast(dispatcher._messageQueue[0]._message)); CPPUNIT_ASSERT(localNode == m->getLocalNode()); CPPUNIT_ASSERT(remoteNode == m->getRemoteNode()); CPPUNIT_ASSERT_EQUAL(std::string("get_peers"), m->getMessageType()); CPPUNIT_ASSERT_EQUAL(msg.getTransactionID(), m->getTransactionID()); CPPUNIT_ASSERT_EQUAL(tokenTracker.generateToken(infoHash, remoteNode->getIPAddress(), remoteNode->getPort()), m->_token); CPPUNIT_ASSERT_EQUAL((size_t)0, m->_nodes.size()); CPPUNIT_ASSERT_EQUAL((size_t)2, m->_peers.size()); { SharedHandle peer = m->_peers[0]; CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.100"), peer->ipaddr); CPPUNIT_ASSERT_EQUAL((uint16_t)6888, peer->port); } { SharedHandle peer = m->_peers[1]; CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.101"), peer->ipaddr); CPPUNIT_ASSERT_EQUAL((uint16_t)6889, peer->port); } } dispatcher._messageQueue.clear(); { // localhost doesn't have peer contact information for that infohash. DHTPeerAnnounceStorage peerAnnounceStorage; DHTRoutingTable routingTable(localNode); SharedHandle returnNode1(new DHTNode()); routingTable.addNode(returnNode1); msg.setPeerAnnounceStorage(WeakHandle (&peerAnnounceStorage)); msg.setRoutingTable(WeakHandle (&routingTable)); msg.doReceivedAction(); CPPUNIT_ASSERT_EQUAL((size_t)1, dispatcher._messageQueue.size()); SharedHandle m (dynamic_pointer_cast(dispatcher._messageQueue[0]._message)); CPPUNIT_ASSERT(localNode == m->getLocalNode()); CPPUNIT_ASSERT(remoteNode == m->getRemoteNode()); CPPUNIT_ASSERT_EQUAL(std::string("get_peers"), m->getMessageType()); CPPUNIT_ASSERT_EQUAL(msg.getTransactionID(), m->getTransactionID()); CPPUNIT_ASSERT_EQUAL(tokenTracker.generateToken(infoHash, remoteNode->getIPAddress(), remoteNode->getPort()), m->_token); CPPUNIT_ASSERT_EQUAL((size_t)1, m->_nodes.size()); CPPUNIT_ASSERT(returnNode1 == m->_nodes[0]); CPPUNIT_ASSERT_EQUAL((size_t)0, m->_peers.size()); } } } // namespace aria2