/* * Bittorrent Client using Qt4 and libtorrent. * Copyright (C) 2015 Vladimir Golovnev * Copyright (C) 2006 Christophe Dumez * * 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 this program with the OpenSSL project's "OpenSSL" library (or with * modified versions of it that use the same license as the "OpenSSL" library), * and distribute the linked executables. 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), 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. */ #include #include #include #include "base/global.h" #include "base/http/server.h" #include "base/preferences.h" #include "base/utils/bytearray.h" #include "base/utils/string.h" #include "tracker.h" // static limits static const int MAX_TORRENTS = 100; static const int MAX_PEERS_PER_TORRENT = 1000; static const int ANNOUNCE_INTERVAL = 1800; // 30min using namespace BitTorrent; // Peer bool Peer::operator!=(const Peer &other) const { return uid() != other.uid(); } bool Peer::operator==(const Peer &other) const { return uid() == other.uid(); } QString Peer::uid() const { return ip + ":" + QString::number(port); } libtorrent::entry Peer::toEntry(bool noPeerId) const { libtorrent::entry::dictionary_type peerMap; if (!noPeerId) peerMap["id"] = libtorrent::entry(peerId.toStdString()); peerMap["ip"] = libtorrent::entry(ip.toStdString()); peerMap["port"] = libtorrent::entry(port); return libtorrent::entry(peerMap); } // Tracker Tracker::Tracker(QObject *parent) : QObject(parent) , m_server(new Http::Server(this, this)) { } Tracker::~Tracker() { if (m_server->isListening()) qDebug("Shutting down the embedded tracker..."); // TODO: Store the torrent list } bool Tracker::start() { const int listenPort = Preferences::instance()->getTrackerPort(); if (m_server->isListening()) { if (m_server->serverPort() == listenPort) { // Already listening on the right port, just return return true; } // Wrong port, closing the server m_server->close(); } qDebug("Starting the embedded tracker..."); // Listen on the predefined port return m_server->listen(QHostAddress::Any, listenPort); } Http::Response Tracker::processRequest(const Http::Request &request, const Http::Environment &env) { clear(); // clear response //qDebug("Tracker received the following request:\n%s", qUtf8Printable(parser.toString())); // Is request a GET request? if (request.method != "GET") { qDebug("Tracker: Unsupported HTTP request: %s", qUtf8Printable(request.method)); status(100, "Invalid request type"); } else if (!request.path.startsWith("/announce", Qt::CaseInsensitive)) { qDebug("Tracker: Unrecognized path: %s", qUtf8Printable(request.path)); status(100, "Invalid request type"); } else { // OK, this is a GET request m_request = request; m_env = env; respondToAnnounceRequest(); } return response(); } void Tracker::respondToAnnounceRequest() { QMap queryParams; // Parse GET parameters using namespace Utils::ByteArray; for (const QByteArray ¶m : copyAsConst(splitToViews(m_request.query, "&"))) { const int sepPos = param.indexOf('='); if (sepPos <= 0) continue; // ignores params without name const QString paramName {QString::fromUtf8(param.constData(), sepPos)}; const QByteArray paramValue {param.mid(sepPos + 1)}; queryParams[paramName] = paramValue; } TrackerAnnounceRequest annonceReq; // IP annonceReq.peer.ip = m_env.clientAddress.toString(); // 1. Get info_hash if (!queryParams.contains("info_hash")) { qDebug("Tracker: Missing info_hash"); status(101, "Missing info_hash"); return; } annonceReq.infoHash = queryParams.value("info_hash"); // info_hash cannot be longer than 20 bytes /*if (annonce_req.info_hash.toLatin1().length() > 20) { qDebug("Tracker: Info_hash is not 20 byte long: %s (%d)", qUtf8Printable(annonce_req.info_hash), annonce_req.info_hash.toLatin1().length()); status(150, "Invalid infohash"); return; }*/ // 2. Get peer ID if (!queryParams.contains("peer_id")) { qDebug("Tracker: Missing peer_id"); status(102, "Missing peer_id"); return; } annonceReq.peer.peerId = queryParams.value("peer_id"); // peer_id cannot be longer than 20 bytes /*if (annonce_req.peer.peer_id.length() > 20) { qDebug("Tracker: peer_id is not 20 byte long: %s", qUtf8Printable(annonce_req.peer.peer_id)); status(151, "Invalid peerid"); return; }*/ // 3. Get port if (!queryParams.contains("port")) { qDebug("Tracker: Missing port"); status(103, "Missing port"); return; } bool ok = false; annonceReq.peer.port = queryParams.value("port").toInt(&ok); if (!ok || annonceReq.peer.port < 1 || annonceReq.peer.port > 65535) { qDebug("Tracker: Invalid port number (%d)", annonceReq.peer.port); status(103, "Missing port"); return; } // 4. Get event annonceReq.event = ""; if (queryParams.contains("event")) { annonceReq.event = queryParams.value("event"); qDebug("Tracker: event is %s", qUtf8Printable(annonceReq.event)); } // 5. Get numwant annonceReq.numwant = 50; if (queryParams.contains("numwant")) { int tmp = queryParams.value("numwant").toInt(); if (tmp > 0) { qDebug("Tracker: numwant = %d", tmp); annonceReq.numwant = tmp; } } // 6. no_peer_id (extension) annonceReq.noPeerId = false; if (queryParams.contains("no_peer_id")) annonceReq.noPeerId = true; // 7. TODO: support "compact" extension // Done parsing, now let's reply if (m_torrents.contains(annonceReq.infoHash)) { if (annonceReq.event == "stopped") { qDebug("Tracker: Peer stopped downloading, deleting it from the list"); m_torrents[annonceReq.infoHash].remove(annonceReq.peer.uid()); return; } } else { // Unknown torrent if (m_torrents.size() == MAX_TORRENTS) { // Reached max size, remove a random torrent m_torrents.erase(m_torrents.begin()); } } // Register the user PeerList peers = m_torrents.value(annonceReq.infoHash); if (peers.size() == MAX_PEERS_PER_TORRENT) { // Too many peers, remove a random one peers.erase(peers.begin()); } peers[annonceReq.peer.uid()] = annonceReq.peer; m_torrents[annonceReq.infoHash] = peers; // Reply replyWithPeerList(annonceReq); } void Tracker::replyWithPeerList(const TrackerAnnounceRequest &annonceReq) { // Prepare the entry for bencoding libtorrent::entry::dictionary_type replyDict; replyDict["interval"] = libtorrent::entry(ANNOUNCE_INTERVAL); QList peers = m_torrents.value(annonceReq.infoHash).values(); libtorrent::entry::list_type peerList; foreach (const Peer &p, peers) { //if (p != annonce_req.peer) peerList.push_back(p.toEntry(annonceReq.noPeerId)); } replyDict["peers"] = libtorrent::entry(peerList); libtorrent::entry replyEntry(replyDict); // bencode std::vector buf; libtorrent::bencode(std::back_inserter(buf), replyEntry); QByteArray reply(&buf[0], static_cast(buf.size())); qDebug("Tracker: reply with the following bencoded data:\n %s", reply.constData()); // HTTP reply print(reply, Http::CONTENT_TYPE_TXT); }