523 lines
13 KiB
C++
523 lines
13 KiB
C++
|
/*
|
||
|
|
||
|
Copyright (c) 2003, Arvid Norberg
|
||
|
All rights reserved.
|
||
|
|
||
|
Redistribution and use in source and binary forms, with or without
|
||
|
modification, are permitted provided that the following conditions
|
||
|
are met:
|
||
|
|
||
|
* Redistributions of source code must retain the above copyright
|
||
|
notice, this list of conditions and the following disclaimer.
|
||
|
* Redistributions in binary form must reproduce the above copyright
|
||
|
notice, this list of conditions and the following disclaimer in
|
||
|
the documentation and/or other materials provided with the distribution.
|
||
|
* Neither the name of the author nor the names of its
|
||
|
contributors may be used to endorse or promote products derived
|
||
|
from this software without specific prior written permission.
|
||
|
|
||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||
|
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
||
|
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
||
|
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
||
|
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||
|
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||
|
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||
|
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||
|
POSSIBILITY OF SUCH DAMAGE.
|
||
|
|
||
|
*/
|
||
|
|
||
|
#include <vector>
|
||
|
#include <iostream>
|
||
|
#include <cctype>
|
||
|
#include <iomanip>
|
||
|
#include <sstream>
|
||
|
|
||
|
#include "zlib.h"
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
#pragma warning(push, 1)
|
||
|
#endif
|
||
|
|
||
|
#include <boost/bind.hpp>
|
||
|
#include <boost/lexical_cast.hpp>
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
#pragma warning(pop)
|
||
|
#endif
|
||
|
|
||
|
#include "libtorrent/tracker_manager.hpp"
|
||
|
#include "libtorrent/udp_tracker_connection.hpp"
|
||
|
#include "libtorrent/io.hpp"
|
||
|
|
||
|
namespace
|
||
|
{
|
||
|
enum
|
||
|
{
|
||
|
udp_connection_retries = 4,
|
||
|
udp_announce_retries = 15,
|
||
|
udp_connect_timeout = 15,
|
||
|
udp_announce_timeout = 10,
|
||
|
udp_buffer_size = 2048
|
||
|
};
|
||
|
}
|
||
|
|
||
|
using namespace boost::posix_time;
|
||
|
using boost::bind;
|
||
|
using boost::lexical_cast;
|
||
|
|
||
|
namespace libtorrent
|
||
|
{
|
||
|
|
||
|
udp_tracker_connection::udp_tracker_connection(
|
||
|
demuxer& d
|
||
|
, tracker_manager& man
|
||
|
, tracker_request const& req
|
||
|
, std::string const& hostname
|
||
|
, unsigned short port
|
||
|
, boost::weak_ptr<request_callback> c
|
||
|
, session_settings const& stn)
|
||
|
: tracker_connection(man, req, d, c)
|
||
|
, m_man(man)
|
||
|
, m_name_lookup(d)
|
||
|
, m_port(port)
|
||
|
, m_transaction_id(0)
|
||
|
, m_connection_id(0)
|
||
|
, m_settings(stn)
|
||
|
, m_attempts(0)
|
||
|
{
|
||
|
m_socket.reset(new datagram_socket(d));
|
||
|
tcp::resolver::query q(hostname, "0");
|
||
|
m_name_lookup.async_resolve(q
|
||
|
, boost::bind(&udp_tracker_connection::name_lookup, self(), _1, _2));
|
||
|
set_timeout(m_settings.tracker_completion_timeout
|
||
|
, m_settings.tracker_receive_timeout);
|
||
|
}
|
||
|
|
||
|
void udp_tracker_connection::name_lookup(asio::error const& error
|
||
|
, tcp::resolver::iterator i) try
|
||
|
{
|
||
|
if (error == asio::error::operation_aborted) return;
|
||
|
if (!m_socket) return; // the operation was aborted
|
||
|
if (error || i == tcp::resolver::iterator())
|
||
|
{
|
||
|
fail(-1, error.what());
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
#if defined(TORRENT_VERBOSE_LOGGING) || defined(TORRENT_LOGGING)
|
||
|
if (has_requester()) requester().debug_log("udp tracker name lookup successful");
|
||
|
#endif
|
||
|
restart_read_timeout();
|
||
|
m_target = udp::endpoint(i->endpoint().address(), m_port);
|
||
|
if (has_requester()) requester().m_tracker_address
|
||
|
= tcp::endpoint(i->endpoint().address(), m_port);
|
||
|
m_socket->connect(m_target);
|
||
|
send_udp_connect();
|
||
|
}
|
||
|
catch (std::exception& e)
|
||
|
{
|
||
|
fail(-1, e.what());
|
||
|
};
|
||
|
|
||
|
void udp_tracker_connection::on_timeout()
|
||
|
{
|
||
|
m_socket.reset();
|
||
|
m_name_lookup.cancel();
|
||
|
fail_timeout();
|
||
|
}
|
||
|
|
||
|
void udp_tracker_connection::send_udp_connect()
|
||
|
{
|
||
|
#if defined(TORRENT_VERBOSE_LOGGING) || defined(TORRENT_LOGGING)
|
||
|
if (has_requester())
|
||
|
{
|
||
|
requester().debug_log("==> UDP_TRACKER_CONNECT ["
|
||
|
+ lexical_cast<std::string>(tracker_req().info_hash) + "]");
|
||
|
}
|
||
|
#endif
|
||
|
if (!m_socket) return; // the operation was aborted
|
||
|
|
||
|
char send_buf[16];
|
||
|
char* ptr = send_buf;
|
||
|
|
||
|
if (m_transaction_id == 0)
|
||
|
m_transaction_id = rand() ^ (rand() << 16);
|
||
|
|
||
|
// connection_id
|
||
|
detail::write_uint32(0x417, ptr);
|
||
|
detail::write_uint32(0x27101980, ptr);
|
||
|
// action (connect)
|
||
|
detail::write_int32(action_connect, ptr);
|
||
|
// transaction_id
|
||
|
detail::write_int32(m_transaction_id, ptr);
|
||
|
|
||
|
m_socket->send(asio::buffer((void*)send_buf, 16), 0);
|
||
|
++m_attempts;
|
||
|
m_buffer.resize(udp_buffer_size);
|
||
|
m_socket->async_receive_from(asio::buffer(m_buffer), m_sender
|
||
|
, boost::bind(&udp_tracker_connection::connect_response, self(), _1, _2));
|
||
|
}
|
||
|
|
||
|
void udp_tracker_connection::connect_response(asio::error const& error
|
||
|
, std::size_t bytes_transferred) try
|
||
|
{
|
||
|
if (error == asio::error::operation_aborted) return;
|
||
|
if (!m_socket) return; // the operation was aborted
|
||
|
if (error)
|
||
|
{
|
||
|
fail(-1, error.what());
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (m_target != m_sender)
|
||
|
{
|
||
|
// this packet was not received from the tracker
|
||
|
m_socket->async_receive_from(asio::buffer(m_buffer), m_sender
|
||
|
, boost::bind(&udp_tracker_connection::connect_response, self(), _1, _2));
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (bytes_transferred >= udp_buffer_size)
|
||
|
{
|
||
|
fail(-1, "udp response too big");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (bytes_transferred < 8)
|
||
|
{
|
||
|
fail(-1, "got a message with size < 8");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
restart_read_timeout();
|
||
|
|
||
|
const char* ptr = &m_buffer[0];
|
||
|
int action = detail::read_int32(ptr);
|
||
|
int transaction = detail::read_int32(ptr);
|
||
|
|
||
|
if (action == action_error)
|
||
|
{
|
||
|
fail(-1, std::string(ptr, bytes_transferred - 8).c_str());
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (action != action_connect)
|
||
|
{
|
||
|
fail(-1, "invalid action in connect reply");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (m_transaction_id != transaction)
|
||
|
{
|
||
|
fail(-1, "incorrect transaction id");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (bytes_transferred < 16)
|
||
|
{
|
||
|
fail(-1, "udp_tracker_connection: "
|
||
|
"got a message with size < 16");
|
||
|
return;
|
||
|
}
|
||
|
// reset transaction
|
||
|
m_transaction_id = 0;
|
||
|
m_attempts = 0;
|
||
|
m_connection_id = detail::read_int64(ptr);
|
||
|
|
||
|
#if defined(TORRENT_VERBOSE_LOGGING) || defined(TORRENT_LOGGING)
|
||
|
if (has_requester())
|
||
|
{
|
||
|
requester().debug_log("<== UDP_TRACKER_CONNECT_RESPONSE ["
|
||
|
+ lexical_cast<std::string>(m_connection_id) + "]");
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if (tracker_req().kind == tracker_request::announce_request)
|
||
|
send_udp_announce();
|
||
|
else if (tracker_req().kind == tracker_request::scrape_request)
|
||
|
send_udp_scrape();
|
||
|
}
|
||
|
catch (std::exception& e)
|
||
|
{
|
||
|
fail(-1, e.what());
|
||
|
}
|
||
|
|
||
|
void udp_tracker_connection::send_udp_announce()
|
||
|
{
|
||
|
if (m_transaction_id == 0)
|
||
|
m_transaction_id = rand() ^ (rand() << 16);
|
||
|
|
||
|
if (!m_socket) return; // the operation was aborted
|
||
|
|
||
|
std::vector<char> buf;
|
||
|
std::back_insert_iterator<std::vector<char> > out(buf);
|
||
|
|
||
|
tracker_request const& req = tracker_req();
|
||
|
|
||
|
// connection_id
|
||
|
detail::write_int64(m_connection_id, out);
|
||
|
// action (announce)
|
||
|
detail::write_int32(action_announce, out);
|
||
|
// transaction_id
|
||
|
detail::write_int32(m_transaction_id, out);
|
||
|
// info_hash
|
||
|
std::copy(req.info_hash.begin(), req.info_hash.end(), out);
|
||
|
// peer_id
|
||
|
std::copy(req.pid.begin(), req.pid.end(), out);
|
||
|
// downloaded
|
||
|
detail::write_int64(req.downloaded, out);
|
||
|
// left
|
||
|
detail::write_int64(req.left, out);
|
||
|
// uploaded
|
||
|
detail::write_int64(req.uploaded, out);
|
||
|
// event
|
||
|
detail::write_int32(req.event, out);
|
||
|
// ip address
|
||
|
detail::write_int32(0, out);
|
||
|
// key
|
||
|
detail::write_int32(req.key, out);
|
||
|
// num_want
|
||
|
detail::write_int32(req.num_want, out);
|
||
|
// port
|
||
|
detail::write_uint16(req.listen_port, out);
|
||
|
// extensions
|
||
|
detail::write_uint16(0, out);
|
||
|
|
||
|
#if defined(TORRENT_VERBOSE_LOGGING) || defined(TORRENT_LOGGING)
|
||
|
if (has_requester())
|
||
|
{
|
||
|
requester().debug_log("==> UDP_TRACKER_ANNOUNCE ["
|
||
|
+ lexical_cast<std::string>(req.info_hash) + "]");
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
m_socket->send(asio::buffer(buf), 0);
|
||
|
++m_attempts;
|
||
|
|
||
|
m_socket->async_receive_from(asio::buffer(m_buffer), m_sender
|
||
|
, bind(&udp_tracker_connection::announce_response, self(), _1, _2));
|
||
|
}
|
||
|
|
||
|
void udp_tracker_connection::send_udp_scrape()
|
||
|
{
|
||
|
if (m_transaction_id == 0)
|
||
|
m_transaction_id = rand() ^ (rand() << 16);
|
||
|
|
||
|
if (!m_socket) return; // the operation was aborted
|
||
|
|
||
|
std::vector<char> buf;
|
||
|
std::back_insert_iterator<std::vector<char> > out(buf);
|
||
|
|
||
|
// connection_id
|
||
|
detail::write_int64(m_connection_id, out);
|
||
|
// action (scrape)
|
||
|
detail::write_int32(action_scrape, out);
|
||
|
// transaction_id
|
||
|
detail::write_int32(m_transaction_id, out);
|
||
|
// info_hash
|
||
|
std::copy(tracker_req().info_hash.begin(), tracker_req().info_hash.end(), out);
|
||
|
|
||
|
m_socket->send(asio::buffer(&buf[0], buf.size()), 0);
|
||
|
++m_attempts;
|
||
|
|
||
|
m_socket->async_receive_from(asio::buffer(m_buffer), m_sender
|
||
|
, bind(&udp_tracker_connection::scrape_response, self(), _1, _2));
|
||
|
}
|
||
|
|
||
|
void udp_tracker_connection::announce_response(asio::error const& error
|
||
|
, std::size_t bytes_transferred) try
|
||
|
{
|
||
|
if (error == asio::error::operation_aborted) return;
|
||
|
if (!m_socket) return; // the operation was aborted
|
||
|
if (error)
|
||
|
{
|
||
|
fail(-1, error.what());
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (m_target != m_sender)
|
||
|
{
|
||
|
// this packet was not received from the tracker
|
||
|
m_socket->async_receive_from(asio::buffer(m_buffer), m_sender
|
||
|
, bind(&udp_tracker_connection::connect_response, self(), _1, _2));
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (bytes_transferred >= udp_buffer_size)
|
||
|
{
|
||
|
fail(-1, "udp response too big");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (bytes_transferred < 8)
|
||
|
{
|
||
|
fail(-1, "got a message with size < 8");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
restart_read_timeout();
|
||
|
char* buf = &m_buffer[0];
|
||
|
int action = detail::read_int32(buf);
|
||
|
int transaction = detail::read_int32(buf);
|
||
|
|
||
|
if (transaction != m_transaction_id)
|
||
|
{
|
||
|
fail(-1, "incorrect transaction id");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (action == action_error)
|
||
|
{
|
||
|
fail(-1, std::string(buf, bytes_transferred - 8).c_str());
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (action != action_announce)
|
||
|
{
|
||
|
fail(-1, "invalid action in announce response");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (bytes_transferred < 20)
|
||
|
{
|
||
|
fail(-1, "got a message with size < 20");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
int interval = detail::read_int32(buf);
|
||
|
int incomplete = detail::read_int32(buf);
|
||
|
int complete = detail::read_int32(buf);
|
||
|
int num_peers = (bytes_transferred - 20) / 6;
|
||
|
if ((bytes_transferred - 20) % 6 != 0)
|
||
|
{
|
||
|
fail(-1, "invalid udp tracker response length");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
#if defined(TORRENT_VERBOSE_LOGGING) || defined(TORRENT_LOGGING)
|
||
|
if (has_requester())
|
||
|
{
|
||
|
requester().debug_log("<== UDP_TRACKER_ANNOUNCE_RESPONSE");
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if (!has_requester())
|
||
|
{
|
||
|
m_man.remove_request(this);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
std::vector<peer_entry> peer_list;
|
||
|
for (int i = 0; i < num_peers; ++i)
|
||
|
{
|
||
|
peer_entry e;
|
||
|
std::stringstream s;
|
||
|
s << (int)detail::read_uint8(buf) << ".";
|
||
|
s << (int)detail::read_uint8(buf) << ".";
|
||
|
s << (int)detail::read_uint8(buf) << ".";
|
||
|
s << (int)detail::read_uint8(buf);
|
||
|
e.ip = s.str();
|
||
|
e.port = detail::read_uint16(buf);
|
||
|
e.pid.clear();
|
||
|
peer_list.push_back(e);
|
||
|
}
|
||
|
|
||
|
requester().tracker_response(tracker_req(), peer_list, interval
|
||
|
, complete, incomplete);
|
||
|
|
||
|
m_man.remove_request(this);
|
||
|
return;
|
||
|
}
|
||
|
catch (std::exception& e)
|
||
|
{
|
||
|
fail(-1, e.what());
|
||
|
}; // msvc 7.1 seems to require this
|
||
|
|
||
|
void udp_tracker_connection::scrape_response(asio::error const& error
|
||
|
, std::size_t bytes_transferred) try
|
||
|
{
|
||
|
if (error == asio::error::operation_aborted) return;
|
||
|
if (!m_socket) return; // the operation was aborted
|
||
|
if (error)
|
||
|
{
|
||
|
fail(-1, error.what());
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (m_target != m_sender)
|
||
|
{
|
||
|
// this packet was not received from the tracker
|
||
|
m_socket->async_receive_from(asio::buffer(m_buffer), m_sender
|
||
|
, bind(&udp_tracker_connection::connect_response, self(), _1, _2));
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (bytes_transferred >= udp_buffer_size)
|
||
|
{
|
||
|
fail(-1, "udp response too big");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (bytes_transferred < 8)
|
||
|
{
|
||
|
fail(-1, "got a message with size < 8");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
restart_read_timeout();
|
||
|
char* buf = &m_buffer[0];
|
||
|
int action = detail::read_int32(buf);
|
||
|
int transaction = detail::read_int32(buf);
|
||
|
|
||
|
if (transaction != m_transaction_id)
|
||
|
{
|
||
|
fail(-1, "incorrect transaction id");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (action == action_error)
|
||
|
{
|
||
|
fail(-1, std::string(buf, bytes_transferred - 8).c_str());
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (action != action_scrape)
|
||
|
{
|
||
|
fail(-1, "invalid action in announce response");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (bytes_transferred < 20)
|
||
|
{
|
||
|
fail(-1, "got a message with size < 20");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
int complete = detail::read_int32(buf);
|
||
|
/*int downloaded = */detail::read_int32(buf);
|
||
|
int incomplete = detail::read_int32(buf);
|
||
|
|
||
|
if (!has_requester())
|
||
|
{
|
||
|
m_man.remove_request(this);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
std::vector<peer_entry> peer_list;
|
||
|
requester().tracker_response(tracker_req(), peer_list, 0
|
||
|
, complete, incomplete);
|
||
|
|
||
|
m_man.remove_request(this);
|
||
|
}
|
||
|
catch (std::exception& e)
|
||
|
{
|
||
|
fail(-1, e.what());
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|