123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617 |
- /* <!-- copyright */
- /*
- * aria2 - The high speed download utility
- *
- * Copyright (C) 2013 Tatsuhiro Tsujikawa
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- *
- * In addition, as a special exception, the copyright holders give
- * permission to link the code of portions of this program with the
- * OpenSSL library under certain conditions as described in each
- * individual source file, and distribute linked combinations
- * including the two.
- * You must obey the GNU General Public License in all respects
- * for all of the code used other than OpenSSL. If you modify
- * file(s) with this exception, you may extend this exception to your
- * version of the file(s), but you are not obligated to do so. If you
- * do not wish to do so, delete this exception statement from your
- * version. If you delete this exception statement from all source
- * files in the program, then also delete it here.
- */
- /* copyright --> */
- #include "UDPTrackerClient.h"
- #include "UDPTrackerRequest.h"
- #include "bittorrent_helper.h"
- #include "util.h"
- #include "LogFactory.h"
- #include "SimpleRandomizer.h"
- #include "fmt.h"
- namespace aria2 {
- UDPTrackerClient::UDPTrackerClient()
- : numWatchers_(0)
- {}
- namespace {
- template<typename InputIterator>
- void failRequest(InputIterator first, InputIterator last, int error)
- {
- for(; first != last; ++first) {
- (*first)->state = UDPT_STA_COMPLETE;
- (*first)->error = error;
- }
- }
- } // namespace
- namespace {
- int32_t generateTransactionId()
- {
- return SimpleRandomizer::getInstance()->getRandomNumber(INT32_MAX);
- }
- } // namespace
- namespace {
- void logInvalidLength(const std::string& remoteAddr, uint16_t remotePort,
- int action, unsigned long expected, unsigned long actual)
- {
- A2_LOG_INFO(fmt("UDPT received %s reply from %s:%u invalid length "
- "expected:%lu, actual:%lu",
- getUDPTrackerActionStr(action),
- remoteAddr.c_str(), remotePort, expected, actual));
- }
- } // namespace
- namespace {
- void logInvalidTransaction(const std::string& remoteAddr, uint16_t remotePort,
- int action, int32_t transactionId)
- {
- A2_LOG_INFO(fmt("UDPT received %s reply from %s:%u invalid transaction_id=%d",
- getUDPTrackerActionStr(action),
- remoteAddr.c_str(), remotePort, transactionId));
- }
- } // namespace
- namespace {
- void logTooShortLength(const std::string& remoteAddr, uint16_t remotePort,
- int action,
- unsigned long minLength, unsigned long actual)
- {
- A2_LOG_INFO(fmt("UDPT received %s reply from %s:%u length too short "
- "min:%lu, actual:%lu",
- getUDPTrackerActionStr(action),
- remoteAddr.c_str(), remotePort, minLength, actual));
- }
- } // namespace
- UDPTrackerClient::~UDPTrackerClient()
- {
- // Make all contained requests fail
- int error = UDPT_ERR_SHUTDOWN;
- failRequest(inflightRequests_.begin(), inflightRequests_.end(), error);
- failRequest(pendingRequests_.begin(), pendingRequests_.end(), error);
- failRequest(connectRequests_.begin(), connectRequests_.end(), error);
- }
- namespace {
- struct CollectAddrPortMatch {
- bool operator()(const std::shared_ptr<UDPTrackerRequest>& req) const
- {
- if(req->remoteAddr == remoteAddr && req->remotePort == remotePort) {
- dest.push_back(req);
- return true;
- } else {
- return false;
- }
- }
- std::vector<std::shared_ptr<UDPTrackerRequest> >& dest;
- std::string remoteAddr;
- uint16_t remotePort;
- CollectAddrPortMatch(std::vector<std::shared_ptr<UDPTrackerRequest> >& dest,
- std::string remoteAddr, uint16_t remotePort)
- : dest(dest), remoteAddr(std::move(remoteAddr)), remotePort(remotePort)
- {}
- };
- } // namespace
- int UDPTrackerClient::receiveReply
- (const unsigned char* data, size_t length, const std::string& remoteAddr,
- uint16_t remotePort, const Timer& now)
- {
- int32_t action = bittorrent::getIntParam(data, 0);
- switch(action) {
- case UDPT_ACT_CONNECT: {
- if(length != 16) {
- logInvalidLength(remoteAddr, remotePort, action, 16, length);
- return -1;
- }
- int32_t transactionId = bittorrent::getIntParam(data, 4);
- std::shared_ptr<UDPTrackerRequest> req =
- findInflightRequest(remoteAddr, remotePort, transactionId, true);
- if(!req) {
- logInvalidTransaction(remoteAddr, remotePort, action, transactionId);
- return -1;
- }
- req->state = UDPT_STA_COMPLETE;
- int64_t connectionId = bittorrent::getLLIntParam(data, 8);
- A2_LOG_INFO(fmt("UDPT received CONNECT reply from %s:%u transaction_id=%u,"
- "connection_id=%" PRId64, remoteAddr.c_str(), remotePort,
- transactionId, connectionId));
- UDPTrackerConnection c(UDPT_CST_CONNECTED, connectionId, now);
- connectionIdCache_[std::make_pair(remoteAddr, remotePort)] = c;
- // Now we have connection ID, push requests which are waiting for
- // it.
- std::vector<std::shared_ptr<UDPTrackerRequest> > reqs;
- connectRequests_.erase(std::remove_if
- (connectRequests_.begin(), connectRequests_.end(),
- CollectAddrPortMatch(reqs, remoteAddr, remotePort)),
- connectRequests_.end());
- pendingRequests_.insert(pendingRequests_.begin(),
- reqs.begin(), reqs.end());
- break;
- }
- case UDPT_ACT_ANNOUNCE: {
- if(length < 20) {
- logTooShortLength(remoteAddr, remotePort, action, 20, length);
- return - 1;
- }
- int32_t transactionId = bittorrent::getIntParam(data, 4);
- std::shared_ptr<UDPTrackerRequest> req =
- findInflightRequest(remoteAddr, remotePort, transactionId, true);
- if(!req) {
- logInvalidTransaction(remoteAddr, remotePort, action, transactionId);
- return -1;
- }
- req->state = UDPT_STA_COMPLETE;
- req->reply = std::make_shared<UDPTrackerReply>();
- req->reply->action = action;
- req->reply->transactionId = transactionId;
- req->reply->interval = bittorrent::getIntParam(data, 8);
- req->reply->leechers = bittorrent::getIntParam(data, 12);
- req->reply->seeders = bittorrent::getIntParam(data, 16);
- int numPeers = 0;
- for(size_t i = 20; i < length; i += 6) {
- std::pair<std::string, uint16_t> hostport =
- bittorrent::unpackcompact(data+i, AF_INET);
- if(!hostport.first.empty()) {
- req->reply->peers.push_back(hostport);
- ++numPeers;
- }
- }
- A2_LOG_INFO(fmt("UDPT received ANNOUNCE reply from %s:%u transaction_id=%u,"
- "connection_id=%" PRId64 ", event=%s, infohash=%s, "
- "interval=%d, leechers=%d, "
- "seeders=%d, num_peers=%d", remoteAddr.c_str(), remotePort,
- transactionId, req->connectionId,
- getUDPTrackerEventStr(req->event),
- util::toHex(req->infohash).c_str(),
- req->reply->interval, req->reply->leechers,
- req->reply->seeders, numPeers));
- break;
- }
- case UDPT_ACT_ERROR: {
- if(length < 8) {
- logTooShortLength(remoteAddr, remotePort, action, 8, length);
- return -1;
- }
- int32_t transactionId = bittorrent::getIntParam(data, 4);
- std::shared_ptr<UDPTrackerRequest> req =
- findInflightRequest(remoteAddr, remotePort, transactionId, true);
- if(!req) {
- logInvalidTransaction(remoteAddr, remotePort, action, transactionId);
- return -1;
- }
- std::string errorString(data+8, data+length);
- errorString = util::encodeNonUtf8(errorString);
- req->state = UDPT_STA_COMPLETE;
- req->error = UDPT_ERR_TRACKER;
- A2_LOG_INFO(fmt("UDPT received ERROR reply from %s:%u transaction_id=%u,"
- "connection_id=%" PRId64 ", action=%d, error_string=%s",
- remoteAddr.c_str(),
- remotePort, transactionId, req->connectionId, action,
- errorString.c_str()));
- if(req->action == UDPT_ACT_CONNECT) {
- failConnect(req->remoteAddr, req->remotePort, UDPT_ERR_TRACKER);
- }
- break;
- }
- case UDPT_ACT_SCRAPE:
- A2_LOG_INFO(fmt("unexpected scrape action reply from %s:%u",
- remoteAddr.c_str(), remotePort));
- return -1;
- default:
- A2_LOG_INFO(fmt("unknown action reply from %s:%u",
- remoteAddr.c_str(), remotePort));
- return -1;
- }
- return 0;
- }
- ssize_t UDPTrackerClient::createRequest
- (unsigned char* data, size_t length, std::string& remoteAddr,
- uint16_t& remotePort, const Timer& now)
- {
- if(pendingRequests_.empty()) {
- return -1;
- }
- while(!pendingRequests_.empty()) {
- const std::shared_ptr<UDPTrackerRequest>& req = pendingRequests_.front();
- if(req->action == UDPT_ACT_CONNECT) {
- ssize_t rv;
- rv = createUDPTrackerConnect(data, length, remoteAddr, remotePort, req);
- return rv;
- }
- UDPTrackerConnection* c = getConnectionId(req->remoteAddr,
- req->remotePort,
- now);
- if(!c) {
- auto creq = std::make_shared<UDPTrackerRequest>();
- creq->action = UDPT_ACT_CONNECT;
- creq->remoteAddr = req->remoteAddr;
- creq->remotePort = req->remotePort;
- creq->transactionId = generateTransactionId();
- pendingRequests_.push_front(creq);
- ssize_t rv;
- rv = createUDPTrackerConnect(data, length, remoteAddr, remotePort, creq);
- return rv;
- }
- if(c->state == UDPT_CST_CONNECTING) {
- connectRequests_.push_back(req);
- pendingRequests_.pop_front();
- continue;
- }
- req->connectionId = c->connectionId;
- req->transactionId = generateTransactionId();
- ssize_t rv;
- rv = createUDPTrackerAnnounce(data, length, remoteAddr, remotePort, req);
- return rv;
- }
- return -1;
- }
- void UDPTrackerClient::requestSent(const Timer& now)
- {
- if(pendingRequests_.empty()) {
- A2_LOG_WARN("pendingRequests_ is empty");
- return;
- }
- const std::shared_ptr<UDPTrackerRequest>& req = pendingRequests_.front();
- switch(req->action) {
- case UDPT_ACT_CONNECT:
- A2_LOG_INFO(fmt("UDPT sent CONNECT to %s:%u transaction_id=%u",
- req->remoteAddr.c_str(), req->remotePort,
- req->transactionId));
- break;
- case UDPT_ACT_ANNOUNCE:
- A2_LOG_INFO(fmt("UDPT sent ANNOUNCE to %s:%u transaction_id=%u, "
- "connection_id=%" PRId64 ", event=%s, infohash=%s",
- req->remoteAddr.c_str(), req->remotePort,
- req->transactionId, req->connectionId,
- getUDPTrackerEventStr(req->event),
- util::toHex(req->infohash).c_str()));
- break;
- default:
- // unreachable
- assert(0);
- }
- req->dispatched = now;
- switch(req->action) {
- case UDPT_ACT_CONNECT: {
- connectionIdCache_[std::make_pair(req->remoteAddr, req->remotePort)]
- = UDPTrackerConnection();
- break;
- }
- }
- inflightRequests_.push_back(req);
- pendingRequests_.pop_front();
- }
- void UDPTrackerClient::requestFail(int error)
- {
- if(pendingRequests_.empty()) {
- A2_LOG_WARN("pendingRequests_ is empty");
- return;
- }
- const std::shared_ptr<UDPTrackerRequest>& req = pendingRequests_.front();
- switch(req->action) {
- case UDPT_ACT_CONNECT:
- A2_LOG_INFO(fmt("UDPT fail CONNECT to %s:%u transaction_id=%u",
- req->remoteAddr.c_str(), req->remotePort,
- req->transactionId));
- failConnect(req->remoteAddr, req->remotePort, error);
- break;
- case UDPT_ACT_ANNOUNCE:
- A2_LOG_INFO(fmt("UDPT fail ANNOUNCE to %s:%u transaction_id=%u, "
- "connection_id=%" PRId64 ", event=%s, infohash=%s",
- req->remoteAddr.c_str(), req->remotePort,
- req->transactionId, req->connectionId,
- getUDPTrackerEventStr(req->event),
- util::toHex(req->infohash).c_str()));
- break;
- default:
- // unreachable
- assert(0);
- }
- req->state = UDPT_STA_COMPLETE;
- req->error = error;
- pendingRequests_.pop_front();
- }
- void UDPTrackerClient::addRequest(const std::shared_ptr<UDPTrackerRequest>& req)
- {
- req->state = UDPT_STA_PENDING;
- req->error = UDPT_ERR_SUCCESS;
- pendingRequests_.push_back(req);
- }
- namespace {
- struct TimeoutCheck {
- bool operator()(const std::shared_ptr<UDPTrackerRequest>& req) const
- {
- int t = req->dispatched.difference(now);
- if(req->failCount == 0) {
- if(t >= 15) {
- switch(req->action) {
- case UDPT_ACT_CONNECT:
- A2_LOG_INFO(fmt("UDPT resend CONNECT to %s:%u transaction_id=%u",
- req->remoteAddr.c_str(), req->remotePort,
- req->transactionId));
- break;
- case UDPT_ACT_ANNOUNCE:
- A2_LOG_INFO(fmt("UDPT resend ANNOUNCE to %s:%u transaction_id=%u, "
- "connection_id=%" PRId64 ", event=%s, infohash=%s",
- req->remoteAddr.c_str(), req->remotePort,
- req->transactionId, req->connectionId,
- getUDPTrackerEventStr(req->event),
- util::toHex(req->infohash).c_str()));
- break;
- default:
- // unreachable
- assert(0);
- }
- ++req->failCount;
- dest.push_back(req);
- return true;
- } else {
- return false;
- }
- } else {
- if(t >= 60) {
- switch(req->action) {
- case UDPT_ACT_CONNECT:
- A2_LOG_INFO(fmt("UDPT timeout CONNECT to %s:%u transaction_id=%u",
- req->remoteAddr.c_str(), req->remotePort,
- req->transactionId));
- client->failConnect(req->remoteAddr, req->remotePort,
- UDPT_ERR_TIMEOUT);
- break;
- case UDPT_ACT_ANNOUNCE:
- A2_LOG_INFO(fmt("UDPT timeout ANNOUNCE to %s:%u transaction_id=%u, "
- "connection_id=%" PRId64 ", event=%s, infohash=%s",
- req->remoteAddr.c_str(), req->remotePort,
- req->transactionId, req->connectionId,
- getUDPTrackerEventStr(req->event),
- util::toHex(req->infohash).c_str()));
- break;
- default:
- // unreachable
- assert(0);
- }
- ++req->failCount;
- req->state = UDPT_STA_COMPLETE;
- req->error = UDPT_ERR_TIMEOUT;
- return true;
- } else {
- return false;
- }
- }
- }
- std::vector<std::shared_ptr<UDPTrackerRequest> >& dest;
- UDPTrackerClient* client;
- const Timer& now;
- TimeoutCheck(std::vector<std::shared_ptr<UDPTrackerRequest> >& dest,
- UDPTrackerClient* client,
- const Timer& now)
- : dest(dest), client(client), now(now)
- {}
- };
- } // namespace
- void UDPTrackerClient::handleTimeout(const Timer& now)
- {
- std::vector<std::shared_ptr<UDPTrackerRequest> > dest;
- inflightRequests_.erase(std::remove_if(inflightRequests_.begin(),
- inflightRequests_.end(),
- TimeoutCheck(dest, this, now)),
- inflightRequests_.end());
- pendingRequests_.insert(pendingRequests_.begin(), dest.begin(), dest.end());
- }
- std::shared_ptr<UDPTrackerRequest> UDPTrackerClient::findInflightRequest
- (const std::string& remoteAddr, uint16_t remotePort, int32_t transactionId,
- bool remove)
- {
- std::shared_ptr<UDPTrackerRequest> res;
- for(auto i = inflightRequests_.begin(), eoi = inflightRequests_.end();
- i != eoi;
- ++i) {
- if((*i)->remoteAddr == remoteAddr && (*i)->remotePort == remotePort &&
- (*i)->transactionId == transactionId) {
- res = *i;
- if(remove) {
- inflightRequests_.erase(i);
- }
- break;
- }
- }
- return res;
- }
- UDPTrackerConnection* UDPTrackerClient::getConnectionId
- (const std::string& remoteAddr, uint16_t remotePort, const Timer& now)
- {
- auto i = connectionIdCache_.find(std::make_pair(remoteAddr, remotePort));
- if(i == connectionIdCache_.end()) {
- return nullptr;
- }
- if((*i).second.state == UDPT_CST_CONNECTED &&
- (*i).second.lastUpdated.difference(now) > 60) {
- connectionIdCache_.erase(i);
- return nullptr;
- } else {
- return &(*i).second;
- }
- }
- namespace {
- struct FailConnectDelete {
- bool operator()(const std::shared_ptr<UDPTrackerRequest>& req) const
- {
- if(req->action == UDPT_ACT_ANNOUNCE &&
- req->remoteAddr == remoteAddr && req->remotePort == remotePort) {
- A2_LOG_INFO(fmt("Force fail infohash=%s",
- util::toHex(req->infohash).c_str()));
- req->state = UDPT_STA_COMPLETE;
- req->error = error;
- return true;
- } else {
- return false;
- }
- }
- std::string remoteAddr;
- uint16_t remotePort;
- int error;
- FailConnectDelete(std::string remoteAddr, uint16_t remotePort,
- int error)
- : remoteAddr(std::move(remoteAddr)), remotePort(remotePort), error(error)
- {}
- };
- } // namespace
- void UDPTrackerClient::failConnect(const std::string& remoteAddr,
- uint16_t remotePort, int error)
- {
- connectionIdCache_.erase(std::make_pair(remoteAddr, remotePort));
- // Fail all requests which are waiting for connection ID of the host.
- connectRequests_.erase(std::remove_if(connectRequests_.begin(),
- connectRequests_.end(),
- FailConnectDelete
- (remoteAddr, remotePort, error)),
- connectRequests_.end());
- pendingRequests_.erase(std::remove_if(pendingRequests_.begin(),
- pendingRequests_.end(),
- FailConnectDelete
- (remoteAddr, remotePort, error)),
- pendingRequests_.end());
- }
- void UDPTrackerClient::failAll()
- {
- int error = UDPT_ERR_SHUTDOWN;
- failRequest(inflightRequests_.begin(), inflightRequests_.end(), error);
- failRequest(pendingRequests_.begin(), pendingRequests_.end(), error);
- failRequest(connectRequests_.begin(), connectRequests_.end(), error);
- }
- void UDPTrackerClient::increaseWatchers()
- {
- ++numWatchers_;
- }
- void UDPTrackerClient::decreaseWatchers()
- {
- --numWatchers_;
- }
- ssize_t createUDPTrackerConnect
- (unsigned char* data, size_t length,
- std::string& remoteAddr, uint16_t& remotePort,
- const std::shared_ptr<UDPTrackerRequest>& req)
- {
- assert(length >= 16);
- remoteAddr = req->remoteAddr;
- remotePort = req->remotePort;
- bittorrent::setLLIntParam(data, UDPT_INITIAL_CONNECTION_ID);
- bittorrent::setIntParam(data+8, req->action);
- bittorrent::setIntParam(data+12, req->transactionId);
- return 16;
- }
- ssize_t createUDPTrackerAnnounce
- (unsigned char* data, size_t length,
- std::string& remoteAddr, uint16_t& remotePort,
- const std::shared_ptr<UDPTrackerRequest>& req)
- {
- assert(length >= 100);
- remoteAddr = req->remoteAddr;
- remotePort = req->remotePort;
- bittorrent::setLLIntParam(data, req->connectionId);
- bittorrent::setIntParam(data+8, req->action);
- bittorrent::setIntParam(data+12, req->transactionId);
- memcpy(data+16, req->infohash.c_str(), req->infohash.size());
- memcpy(data+36, req->peerId.c_str(), req->peerId.size());
- bittorrent::setLLIntParam(data+56, req->downloaded);
- bittorrent::setLLIntParam(data+64, req->left);
- bittorrent::setLLIntParam(data+72, req->uploaded);
- bittorrent::setIntParam(data+80, req->event);
- // ip is already network-byte order
- memcpy(data+84, &req->ip, sizeof(req->ip));
- bittorrent::setIntParam(data+88, req->key);
- bittorrent::setIntParam(data+92, req->numWant);
- bittorrent::setShortIntParam(data+96, req->port);
- // extensions is always 0
- bittorrent::setShortIntParam(data+98, 0);
- return 100;
- }
- const char* getUDPTrackerActionStr(int action)
- {
- switch(action) {
- case UDPT_ACT_CONNECT:
- return "CONNECT";
- case UDPT_ACT_ANNOUNCE:
- return "ANNOUNCE";
- case UDPT_ACT_ERROR:
- return "ERROR";
- default:
- return "(unknown)";
- }
- }
- const char* getUDPTrackerEventStr(int event)
- {
- switch(event) {
- case UDPT_EVT_NONE:
- return "NONE";
- case UDPT_EVT_COMPLETED:
- return "COMPLETED";
- case UDPT_EVT_STARTED:
- return "STARTED";
- case UDPT_EVT_STOPPED:
- return "STOPPED";
- default:
- return "(unknown)";
- }
- }
- } // namespace aria2
|