2013-02-24 15:56:49 +00:00
|
|
|
#include "UDPTrackerClient.h"
|
|
|
|
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
#include <cppunit/extensions/HelperMacros.h>
|
|
|
|
|
|
|
|
#include "TestUtil.h"
|
|
|
|
#include "UDPTrackerRequest.h"
|
|
|
|
#include "bittorrent_helper.h"
|
|
|
|
#include "wallclock.h"
|
|
|
|
|
|
|
|
namespace aria2 {
|
|
|
|
|
2015-12-27 09:39:47 +00:00
|
|
|
class UDPTrackerClientTest : public CppUnit::TestFixture {
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
CPPUNIT_TEST_SUITE(UDPTrackerClientTest);
|
|
|
|
CPPUNIT_TEST(testCreateUDPTrackerConnect);
|
|
|
|
CPPUNIT_TEST(testCreateUDPTrackerAnnounce);
|
|
|
|
CPPUNIT_TEST(testConnectFollowedByAnnounce);
|
|
|
|
CPPUNIT_TEST(testRequestFailure);
|
|
|
|
CPPUNIT_TEST(testTimeout);
|
|
|
|
CPPUNIT_TEST_SUITE_END();
|
2015-12-27 09:39:47 +00:00
|
|
|
|
2013-02-24 15:56:49 +00:00
|
|
|
public:
|
2015-12-27 09:39:47 +00:00
|
|
|
void setUp() {}
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
void testCreateUDPTrackerConnect();
|
|
|
|
void testCreateUDPTrackerAnnounce();
|
|
|
|
void testConnectFollowedByAnnounce();
|
|
|
|
void testRequestFailure();
|
|
|
|
void testTimeout();
|
|
|
|
};
|
|
|
|
|
2015-12-27 09:39:47 +00:00
|
|
|
CPPUNIT_TEST_SUITE_REGISTRATION(UDPTrackerClientTest);
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
namespace {
|
2013-06-21 16:10:38 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> createAnnounce(const std::string& remoteAddr,
|
2015-12-27 09:39:47 +00:00
|
|
|
uint16_t remotePort,
|
2016-01-18 14:47:31 +00:00
|
|
|
uint32_t transactionId)
|
2013-02-24 15:56:49 +00:00
|
|
|
{
|
2013-06-21 16:10:38 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req(new UDPTrackerRequest());
|
2016-01-18 14:47:31 +00:00
|
|
|
req->connectionId = std::numeric_limits<uint64_t>::max();
|
2013-02-24 15:56:49 +00:00
|
|
|
req->action = UDPT_ACT_ANNOUNCE;
|
|
|
|
req->remoteAddr = remoteAddr;
|
|
|
|
req->remotePort = remotePort;
|
|
|
|
req->transactionId = transactionId;
|
|
|
|
req->infohash = "bittorrent-infohash-";
|
2015-12-27 09:39:47 +00:00
|
|
|
req->peerId = "bittorrent-peer-id--";
|
2013-02-24 15:56:49 +00:00
|
|
|
req->downloaded = INT64_MAX - 1;
|
|
|
|
req->left = INT64_MAX - 2;
|
|
|
|
req->uploaded = INT64_MAX - 3;
|
|
|
|
req->event = UDPT_EVT_STARTED;
|
|
|
|
req->ip = 0;
|
|
|
|
req->key = 1000000007;
|
|
|
|
req->numWant = 50;
|
|
|
|
req->port = 6889;
|
|
|
|
req->extensions = 0;
|
|
|
|
return req;
|
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace {
|
2016-01-18 14:47:31 +00:00
|
|
|
ssize_t createErrorReply(unsigned char* data, size_t len,
|
|
|
|
uint32_t transactionId, const std::string& errorString)
|
2013-02-24 15:56:49 +00:00
|
|
|
{
|
|
|
|
bittorrent::setIntParam(data, UDPT_ACT_ERROR);
|
2015-12-27 09:39:47 +00:00
|
|
|
bittorrent::setIntParam(data + 4, transactionId);
|
|
|
|
memcpy(data + 8, errorString.c_str(), errorString.size());
|
|
|
|
return 8 + errorString.size();
|
2013-02-24 15:56:49 +00:00
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
ssize_t createConnectReply(unsigned char* data, size_t len,
|
2016-01-18 14:47:31 +00:00
|
|
|
uint64_t connectionId, uint32_t transactionId)
|
2013-02-24 15:56:49 +00:00
|
|
|
{
|
|
|
|
bittorrent::setIntParam(data, UDPT_ACT_CONNECT);
|
2015-12-27 09:39:47 +00:00
|
|
|
bittorrent::setIntParam(data + 4, transactionId);
|
|
|
|
bittorrent::setLLIntParam(data + 8, connectionId);
|
2013-02-24 15:56:49 +00:00
|
|
|
return 16;
|
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace {
|
2015-12-27 09:39:47 +00:00
|
|
|
ssize_t createAnnounceReply(unsigned char* data, size_t len,
|
2016-01-18 14:47:31 +00:00
|
|
|
uint32_t transactionId, int numPeers = 0)
|
2013-02-24 15:56:49 +00:00
|
|
|
{
|
|
|
|
bittorrent::setIntParam(data, UDPT_ACT_ANNOUNCE);
|
2015-12-27 09:39:47 +00:00
|
|
|
bittorrent::setIntParam(data + 4, transactionId);
|
|
|
|
bittorrent::setIntParam(data + 8, 1800);
|
|
|
|
bittorrent::setIntParam(data + 12, 100);
|
|
|
|
bittorrent::setIntParam(data + 16, 256);
|
|
|
|
for (int i = 0; i < numPeers; ++i) {
|
|
|
|
bittorrent::packcompact(data + 20 + 6 * i,
|
|
|
|
"192.168.0." + util::uitos(i + 1), 6990 + i);
|
2013-02-24 15:56:49 +00:00
|
|
|
}
|
|
|
|
return 20 + 6 * numPeers;
|
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
void UDPTrackerClientTest::testCreateUDPTrackerConnect()
|
|
|
|
{
|
|
|
|
unsigned char data[16];
|
|
|
|
std::string remoteAddr;
|
|
|
|
uint16_t remotePort = 0;
|
2013-06-21 16:10:38 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req(new UDPTrackerRequest());
|
2013-02-24 15:56:49 +00:00
|
|
|
req->action = UDPT_ACT_CONNECT;
|
|
|
|
req->remoteAddr = "192.168.0.1";
|
|
|
|
req->remotePort = 6991;
|
|
|
|
req->transactionId = 1000000009;
|
2015-12-27 09:39:47 +00:00
|
|
|
ssize_t rv =
|
|
|
|
createUDPTrackerConnect(data, sizeof(data), remoteAddr, remotePort, req);
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)16, rv);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req->remoteAddr, remoteAddr);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req->remotePort, remotePort);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int64_t)UDPT_INITIAL_CONNECTION_ID,
|
|
|
|
(int64_t)bittorrent::getLLIntParam(data, 0));
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)req->action, (int)bittorrent::getIntParam(data, 8));
|
2016-01-18 14:47:31 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(req->transactionId, bittorrent::getIntParam(data, 12));
|
2013-02-24 15:56:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void UDPTrackerClientTest::testCreateUDPTrackerAnnounce()
|
|
|
|
{
|
|
|
|
unsigned char data[100];
|
|
|
|
std::string remoteAddr;
|
|
|
|
uint16_t remotePort = 0;
|
2015-12-27 09:39:47 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 1000000009));
|
|
|
|
ssize_t rv =
|
|
|
|
createUDPTrackerAnnounce(data, sizeof(data), remoteAddr, remotePort, req);
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)100, rv);
|
2016-01-18 14:47:31 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(req->connectionId, bittorrent::getLLIntParam(data, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL((int)req->action, (int)bittorrent::getIntParam(data, 8));
|
2016-01-18 14:47:31 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(req->transactionId, bittorrent::getIntParam(data, 12));
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(req->infohash, std::string(&data[16], &data[36]));
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req->peerId, std::string(&data[36], &data[56]));
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req->downloaded,
|
|
|
|
(int64_t)bittorrent::getLLIntParam(data, 56));
|
2015-12-27 09:39:47 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(req->left, (int64_t)bittorrent::getLLIntParam(data, 64));
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(req->uploaded,
|
|
|
|
(int64_t)bittorrent::getLLIntParam(data, 72));
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req->event, (int32_t)bittorrent::getIntParam(data, 80));
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req->ip, bittorrent::getIntParam(data, 84));
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req->key, bittorrent::getIntParam(data, 88));
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req->numWant,
|
|
|
|
(int32_t)bittorrent::getIntParam(data, 92));
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req->port, bittorrent::getShortIntParam(data, 96));
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req->extensions, bittorrent::getShortIntParam(data, 98));
|
|
|
|
}
|
|
|
|
|
|
|
|
void UDPTrackerClientTest::testConnectFollowedByAnnounce()
|
|
|
|
{
|
|
|
|
ssize_t rv;
|
|
|
|
UDPTrackerClient tr;
|
|
|
|
unsigned char data[100];
|
|
|
|
std::string remoteAddr;
|
|
|
|
uint16_t remotePort;
|
|
|
|
Timer now;
|
2016-01-19 02:24:57 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> recvReq;
|
2013-02-24 15:56:49 +00:00
|
|
|
|
2015-12-27 09:39:47 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req1(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
|
|
|
std::shared_ptr<UDPTrackerRequest> req2(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
req2->infohash = "bittorrent-infohash2";
|
|
|
|
|
|
|
|
tr.addRequest(req1);
|
|
|
|
tr.addRequest(req2);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)2, tr.getPendingRequests().size());
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
// CONNECT request was inserted
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)3, tr.getPendingRequests().size());
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)16, rv);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req1->remoteAddr, remoteAddr);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(req1->remotePort, remotePort);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int64_t)UDPT_INITIAL_CONNECTION_ID,
|
|
|
|
(int64_t)bittorrent::getLLIntParam(data, 0));
|
2016-01-18 14:47:31 +00:00
|
|
|
uint32_t transactionId = bittorrent::getIntParam(data, 12);
|
2013-02-24 15:56:49 +00:00
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
// Duplicate CONNECT request was not inserted
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)3, tr.getPendingRequests().size());
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)16, rv);
|
|
|
|
|
|
|
|
tr.requestSent(now);
|
|
|
|
// CONNECT request was moved to inflight
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)2, tr.getPendingRequests().size());
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
// Now all pending requests were moved to connect
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)-1, rv);
|
|
|
|
CPPUNIT_ASSERT(tr.getPendingRequests().empty());
|
|
|
|
|
2016-01-18 14:47:31 +00:00
|
|
|
uint64_t connectionId = 12345;
|
2013-02-24 15:56:49 +00:00
|
|
|
rv = createConnectReply(data, sizeof(data), connectionId, transactionId);
|
2016-01-19 02:24:57 +00:00
|
|
|
rv = tr.receiveReply(recvReq, data, rv, req1->remoteAddr, req1->remotePort,
|
|
|
|
now);
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(0, (int)rv);
|
2016-01-19 02:24:57 +00:00
|
|
|
if (rv == 0) {
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int32_t)UDPT_ACT_CONNECT, recvReq->action);
|
|
|
|
}
|
2013-02-24 15:56:49 +00:00
|
|
|
// Now 2 requests get back to pending
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)2, tr.getPendingRequests().size());
|
|
|
|
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
// Creates announce for req1
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)100, rv);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)2, tr.getPendingRequests().size());
|
2016-01-18 14:47:31 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(connectionId, bittorrent::getLLIntParam(data, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_ANNOUNCE,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
2015-12-27 09:39:47 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(req1->infohash, std::string(&data[16], &data[36]));
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
// Don't duplicate same request data
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)100, rv);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)2, tr.getPendingRequests().size());
|
2016-01-18 14:47:31 +00:00
|
|
|
uint32_t transactionId1 = bittorrent::getIntParam(data, 12);
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
tr.requestSent(now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)1, tr.getPendingRequests().size());
|
|
|
|
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
2016-01-18 14:47:31 +00:00
|
|
|
uint32_t transactionId2 = bittorrent::getIntParam(data, 12);
|
2013-02-24 15:56:49 +00:00
|
|
|
// Creates announce for req2
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)100, rv);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)1, tr.getPendingRequests().size());
|
2016-01-18 14:47:31 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(connectionId, bittorrent::getLLIntParam(data, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_ANNOUNCE,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
2015-12-27 09:39:47 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(req2->infohash, std::string(&data[16], &data[36]));
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
tr.requestSent(now);
|
|
|
|
// Now all requests are inflight
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)0, tr.getPendingRequests().size());
|
|
|
|
|
|
|
|
// Reply for req2
|
|
|
|
rv = createAnnounceReply(data, sizeof(data), transactionId2);
|
2016-01-19 02:24:57 +00:00
|
|
|
rv = tr.receiveReply(recvReq, data, rv, req2->remoteAddr, req2->remotePort,
|
|
|
|
now);
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(0, (int)rv);
|
2016-01-19 02:24:57 +00:00
|
|
|
if (rv == 0) {
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int32_t)UDPT_ACT_ANNOUNCE, recvReq->action);
|
|
|
|
}
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_STA_COMPLETE, req2->state);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ERR_SUCCESS, req2->error);
|
|
|
|
|
|
|
|
// Reply for req1
|
|
|
|
rv = createAnnounceReply(data, sizeof(data), transactionId1, 2);
|
2016-01-19 02:24:57 +00:00
|
|
|
rv = tr.receiveReply(recvReq, data, rv, req1->remoteAddr, req1->remotePort,
|
|
|
|
now);
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(0, (int)rv);
|
2016-01-19 02:24:57 +00:00
|
|
|
if (rv == 0) {
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int32_t)UDPT_ACT_ANNOUNCE, recvReq->action);
|
|
|
|
}
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_STA_COMPLETE, req1->state);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ERR_SUCCESS, req1->error);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)2, req1->reply->peers.size());
|
2015-12-27 09:39:47 +00:00
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
|
|
CPPUNIT_ASSERT_EQUAL("192.168.0." + util::uitos(i + 1),
|
2013-02-24 15:56:49 +00:00
|
|
|
req1->reply->peers[i].first);
|
2015-12-27 09:39:47 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL((uint16_t)(6990 + i), req1->reply->peers[i].second);
|
2013-02-24 15:56:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Since we have connection ID, next announce request can be sent
|
|
|
|
// immediately
|
2015-12-27 09:39:47 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req3(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
req3->infohash = "bittorrent-infohash3";
|
|
|
|
tr.addRequest(req3);
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)100, rv);
|
2015-12-27 09:39:47 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(req3->infohash, std::string(&data[16], &data[36]));
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
tr.requestSent(now);
|
|
|
|
|
2015-12-27 09:39:47 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req4(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
req4->infohash = "bittorrent-infohash4";
|
|
|
|
tr.addRequest(req4);
|
|
|
|
Timer future = now;
|
2015-06-21 06:17:00 +00:00
|
|
|
future.advance(1_h);
|
2015-12-27 09:39:47 +00:00
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, future);
|
2013-02-24 15:56:49 +00:00
|
|
|
// connection ID is stale because of the timeout
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)16, rv);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int64_t)UDPT_INITIAL_CONNECTION_ID,
|
|
|
|
(int64_t)bittorrent::getLLIntParam(data, 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
void UDPTrackerClientTest::testRequestFailure()
|
|
|
|
{
|
|
|
|
ssize_t rv;
|
|
|
|
UDPTrackerClient tr;
|
|
|
|
unsigned char data[100];
|
|
|
|
std::string remoteAddr;
|
|
|
|
uint16_t remotePort;
|
|
|
|
Timer now;
|
2016-01-19 02:24:57 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> recvReq;
|
|
|
|
|
2013-02-24 15:56:49 +00:00
|
|
|
{
|
2015-12-27 09:39:47 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req1(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
|
|
|
std::shared_ptr<UDPTrackerRequest> req2(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
tr.addRequest(req1);
|
|
|
|
tr.addRequest(req2);
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_CONNECT,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
|
|
|
tr.requestFail(UDPT_ERR_NETWORK);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_STA_COMPLETE, req1->state);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ERR_NETWORK, req1->error);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_STA_COMPLETE, req2->state);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ERR_NETWORK, req2->error);
|
|
|
|
CPPUNIT_ASSERT(tr.getConnectRequests().empty());
|
|
|
|
CPPUNIT_ASSERT(tr.getPendingRequests().empty());
|
|
|
|
CPPUNIT_ASSERT(tr.getInflightRequests().empty());
|
|
|
|
}
|
|
|
|
{
|
2015-12-27 09:39:47 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req1(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
|
|
|
std::shared_ptr<UDPTrackerRequest> req2(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
tr.addRequest(req1);
|
|
|
|
tr.addRequest(req2);
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_CONNECT,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
2016-01-18 14:47:31 +00:00
|
|
|
uint32_t transactionId = bittorrent::getIntParam(data, 12);
|
2013-02-24 15:56:49 +00:00
|
|
|
tr.requestSent(now);
|
|
|
|
|
|
|
|
rv = createErrorReply(data, sizeof(data), transactionId, "error");
|
2016-01-19 02:24:57 +00:00
|
|
|
rv = tr.receiveReply(recvReq, data, rv, req1->remoteAddr, req1->remotePort,
|
|
|
|
now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)0, rv);
|
|
|
|
if (rv == 0) {
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int32_t)UDPT_ACT_CONNECT, recvReq->action);
|
|
|
|
}
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_STA_COMPLETE, req1->state);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ERR_TRACKER, req1->error);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_STA_COMPLETE, req2->state);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ERR_TRACKER, req2->error);
|
|
|
|
CPPUNIT_ASSERT(tr.getConnectRequests().empty());
|
|
|
|
CPPUNIT_ASSERT(tr.getPendingRequests().empty());
|
|
|
|
CPPUNIT_ASSERT(tr.getInflightRequests().empty());
|
|
|
|
}
|
|
|
|
{
|
2015-12-27 09:39:47 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req1(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
tr.addRequest(req1);
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)16, rv);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_CONNECT,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
2016-01-18 14:47:31 +00:00
|
|
|
uint32_t transactionId = bittorrent::getIntParam(data, 12);
|
2013-02-24 15:56:49 +00:00
|
|
|
tr.requestSent(now);
|
|
|
|
|
2016-01-18 14:47:31 +00:00
|
|
|
uint64_t connectionId = 12345;
|
2013-02-24 15:56:49 +00:00
|
|
|
rv = createConnectReply(data, sizeof(data), connectionId, transactionId);
|
2016-01-19 02:24:57 +00:00
|
|
|
rv = tr.receiveReply(recvReq, data, rv, req1->remoteAddr, req1->remotePort,
|
|
|
|
now);
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(0, (int)rv);
|
|
|
|
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_ANNOUNCE,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
|
|
|
transactionId = bittorrent::getIntParam(data, 12);
|
|
|
|
tr.requestSent(now);
|
|
|
|
|
|
|
|
rv = createErrorReply(data, sizeof(data), transactionId, "announce error");
|
2016-01-19 02:24:57 +00:00
|
|
|
rv = tr.receiveReply(recvReq, data, rv, req1->remoteAddr, req1->remotePort,
|
|
|
|
now);
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_STA_COMPLETE, req1->state);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ERR_TRACKER, req1->error);
|
|
|
|
CPPUNIT_ASSERT(tr.getConnectRequests().empty());
|
|
|
|
CPPUNIT_ASSERT(tr.getPendingRequests().empty());
|
|
|
|
CPPUNIT_ASSERT(tr.getInflightRequests().empty());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void UDPTrackerClientTest::testTimeout()
|
|
|
|
{
|
|
|
|
ssize_t rv;
|
|
|
|
unsigned char data[100];
|
|
|
|
std::string remoteAddr;
|
|
|
|
uint16_t remotePort;
|
|
|
|
Timer now;
|
|
|
|
UDPTrackerClient tr;
|
2016-01-19 02:24:57 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> recvReq;
|
|
|
|
|
2013-02-24 15:56:49 +00:00
|
|
|
{
|
2015-12-27 09:39:47 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req1(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
|
|
|
std::shared_ptr<UDPTrackerRequest> req2(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
tr.addRequest(req1);
|
|
|
|
tr.addRequest(req2);
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_CONNECT,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
|
|
|
tr.requestSent(now);
|
2015-06-21 06:17:00 +00:00
|
|
|
now.advance(20_s);
|
2013-02-24 15:56:49 +00:00
|
|
|
// 15 seconds 1st stage timeout passed
|
|
|
|
tr.handleTimeout(now);
|
|
|
|
CPPUNIT_ASSERT(tr.getConnectRequests().empty());
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)3, tr.getPendingRequests().size());
|
|
|
|
CPPUNIT_ASSERT(tr.getInflightRequests().empty());
|
|
|
|
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
// CONNECT request was inserted
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_CONNECT,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
|
|
|
tr.requestSent(now);
|
2015-06-21 06:17:00 +00:00
|
|
|
now.advance(65_s);
|
2013-02-24 15:56:49 +00:00
|
|
|
// 60 seconds 2nd stage timeout passed
|
|
|
|
tr.handleTimeout(now);
|
|
|
|
CPPUNIT_ASSERT(tr.getConnectRequests().empty());
|
|
|
|
CPPUNIT_ASSERT(tr.getPendingRequests().empty());
|
|
|
|
CPPUNIT_ASSERT(tr.getInflightRequests().empty());
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_STA_COMPLETE, req1->state);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ERR_TIMEOUT, req1->error);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_STA_COMPLETE, req2->state);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ERR_TIMEOUT, req2->error);
|
|
|
|
}
|
|
|
|
{
|
2015-12-27 09:39:47 +00:00
|
|
|
std::shared_ptr<UDPTrackerRequest> req1(
|
|
|
|
createAnnounce("192.168.0.1", 6991, 0));
|
2013-02-24 15:56:49 +00:00
|
|
|
|
|
|
|
tr.addRequest(req1);
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((ssize_t)16, rv);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_CONNECT,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
2016-01-18 14:47:31 +00:00
|
|
|
uint32_t transactionId = bittorrent::getIntParam(data, 12);
|
2013-02-24 15:56:49 +00:00
|
|
|
tr.requestSent(now);
|
|
|
|
|
2016-01-18 14:47:31 +00:00
|
|
|
uint64_t connectionId = 12345;
|
2013-02-24 15:56:49 +00:00
|
|
|
rv = createConnectReply(data, sizeof(data), connectionId, transactionId);
|
2016-01-19 02:24:57 +00:00
|
|
|
rv = tr.receiveReply(recvReq, data, rv, req1->remoteAddr, req1->remotePort,
|
|
|
|
now);
|
2013-02-24 15:56:49 +00:00
|
|
|
CPPUNIT_ASSERT_EQUAL(0, (int)rv);
|
|
|
|
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_ANNOUNCE,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
|
|
|
tr.requestSent(now);
|
2015-06-21 06:17:00 +00:00
|
|
|
now.advance(20_s);
|
2013-02-24 15:56:49 +00:00
|
|
|
// 15 seconds 1st stage timeout passed
|
|
|
|
tr.handleTimeout(now);
|
|
|
|
CPPUNIT_ASSERT(tr.getConnectRequests().empty());
|
|
|
|
CPPUNIT_ASSERT_EQUAL((size_t)1, tr.getPendingRequests().size());
|
|
|
|
CPPUNIT_ASSERT(tr.getInflightRequests().empty());
|
|
|
|
|
|
|
|
rv = tr.createRequest(data, sizeof(data), remoteAddr, remotePort, now);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ACT_ANNOUNCE,
|
|
|
|
(int)bittorrent::getIntParam(data, 8));
|
|
|
|
tr.requestSent(now);
|
2015-06-21 06:17:00 +00:00
|
|
|
now.advance(65_s);
|
2013-02-24 15:56:49 +00:00
|
|
|
// 60 seconds 2nd stage timeout passed
|
|
|
|
tr.handleTimeout(now);
|
|
|
|
CPPUNIT_ASSERT(tr.getConnectRequests().empty());
|
|
|
|
CPPUNIT_ASSERT(tr.getPendingRequests().empty());
|
|
|
|
CPPUNIT_ASSERT(tr.getInflightRequests().empty());
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_STA_COMPLETE, req1->state);
|
|
|
|
CPPUNIT_ASSERT_EQUAL((int)UDPT_ERR_TIMEOUT, req1->error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace aria2
|