2004-01-31 11:46:15 +01:00
|
|
|
/*
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2007-03-17 18:15:16 +01:00
|
|
|
#include "libtorrent/pch.hpp"
|
|
|
|
|
2004-01-31 11:46:15 +01:00
|
|
|
#include <vector>
|
|
|
|
#include <cctype>
|
|
|
|
|
|
|
|
#include "zlib.h"
|
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(push, 1)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <boost/bind.hpp>
|
2009-01-28 06:49:21 +01:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(pop)
|
|
|
|
#endif
|
|
|
|
|
2004-01-31 11:46:15 +01:00
|
|
|
#include "libtorrent/tracker_manager.hpp"
|
2008-05-17 16:19:34 +02:00
|
|
|
#include "libtorrent/parse_url.hpp"
|
2004-01-31 11:46:15 +01:00
|
|
|
#include "libtorrent/udp_tracker_connection.hpp"
|
|
|
|
#include "libtorrent/io.hpp"
|
2008-10-22 21:40:32 +02:00
|
|
|
#include "libtorrent/aux_/session_impl.hpp"
|
2009-01-27 07:17:55 +01:00
|
|
|
#include "libtorrent/escape_string.hpp"
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
using boost::bind;
|
2004-10-14 03:17:04 +02:00
|
|
|
|
2004-01-31 11:46:15 +01:00
|
|
|
namespace libtorrent
|
|
|
|
{
|
|
|
|
|
|
|
|
udp_tracker_connection::udp_tracker_connection(
|
2008-01-08 06:47:43 +01:00
|
|
|
io_service& ios
|
2008-02-05 07:32:10 +01:00
|
|
|
, connection_queue& cc
|
2006-04-25 23:04:48 +02:00
|
|
|
, tracker_manager& man
|
|
|
|
, tracker_request const& req
|
2004-09-16 03:14:16 +02:00
|
|
|
, boost::weak_ptr<request_callback> c
|
2008-10-22 21:40:32 +02:00
|
|
|
, aux::session_impl const& ses
|
2008-02-05 07:32:10 +01:00
|
|
|
, proxy_settings const& proxy)
|
2009-05-15 23:23:41 +02:00
|
|
|
: tracker_connection(man, req, ios, c)
|
2006-04-25 23:04:48 +02:00
|
|
|
, m_man(man)
|
2008-01-08 06:47:43 +01:00
|
|
|
, m_name_lookup(ios)
|
2008-04-22 19:30:28 +02:00
|
|
|
, m_socket(ios, boost::bind(&udp_tracker_connection::on_receive, self(), _1, _2, _3, _4), cc)
|
2004-01-31 11:46:15 +01:00
|
|
|
, m_transaction_id(0)
|
|
|
|
, m_connection_id(0)
|
2008-10-22 21:40:32 +02:00
|
|
|
, m_ses(ses)
|
2004-01-31 11:46:15 +01:00
|
|
|
, m_attempts(0)
|
2008-02-05 07:32:10 +01:00
|
|
|
, m_state(action_error)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2008-02-05 07:32:10 +01:00
|
|
|
m_socket.set_proxy_settings(proxy);
|
2008-09-07 12:03:59 +02:00
|
|
|
}
|
2008-02-05 07:32:10 +01:00
|
|
|
|
2008-09-07 12:03:59 +02:00
|
|
|
void udp_tracker_connection::start()
|
|
|
|
{
|
2008-02-05 07:32:10 +01:00
|
|
|
std::string hostname;
|
|
|
|
int port;
|
2009-06-12 18:40:38 +02:00
|
|
|
error_code ec;
|
2008-02-05 07:32:10 +01:00
|
|
|
|
|
|
|
using boost::tuples::ignore;
|
2009-06-12 18:40:38 +02:00
|
|
|
boost::tie(ignore, ignore, hostname, port, ignore)
|
|
|
|
= parse_url_components(tracker_req().url, ec);
|
2008-05-17 16:19:34 +02:00
|
|
|
|
2009-06-12 18:40:38 +02:00
|
|
|
if (ec)
|
2008-05-17 16:19:34 +02:00
|
|
|
{
|
2009-06-12 18:40:38 +02:00
|
|
|
fail(-1, ec.message().c_str());
|
2008-05-17 16:19:34 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-10-22 21:40:32 +02:00
|
|
|
session_settings const& settings = m_ses.settings();
|
|
|
|
|
2009-01-27 07:17:55 +01:00
|
|
|
udp::resolver::query q(hostname, to_string(port).elems);
|
2006-05-20 17:30:40 +02:00
|
|
|
m_name_lookup.async_resolve(q
|
2008-01-08 06:47:43 +01:00
|
|
|
, boost::bind(
|
|
|
|
&udp_tracker_connection::name_lookup, self(), _1, _2));
|
2008-09-07 12:03:59 +02:00
|
|
|
set_timeout(tracker_req().event == tracker_request::stopped
|
2008-10-22 21:40:32 +02:00
|
|
|
? settings.stop_tracker_timeout
|
|
|
|
: settings.tracker_completion_timeout
|
|
|
|
, settings.tracker_receive_timeout);
|
2008-02-17 23:51:03 +01:00
|
|
|
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING
|
2008-01-06 23:14:00 +01:00
|
|
|
boost::shared_ptr<request_callback> cb = requester();
|
|
|
|
if (cb) cb->debug_log(("*** UDP_TRACKER [ initiating name lookup: " + hostname + " ]").c_str());
|
|
|
|
#endif
|
2004-01-31 11:46:15 +01:00
|
|
|
}
|
|
|
|
|
2008-05-03 18:05:42 +02:00
|
|
|
void udp_tracker_connection::name_lookup(error_code const& error
|
2007-12-30 02:57:57 +01:00
|
|
|
, udp::resolver::iterator i)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2006-04-25 23:04:48 +02:00
|
|
|
if (error == asio::error::operation_aborted) return;
|
2007-03-02 19:40:02 +01:00
|
|
|
if (error || i == udp::resolver::iterator())
|
2006-04-25 23:04:48 +02:00
|
|
|
{
|
2006-11-14 01:08:16 +01:00
|
|
|
fail(-1, error.message().c_str());
|
2006-04-25 23:04:48 +02:00
|
|
|
return;
|
|
|
|
}
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2007-09-14 04:54:15 +02:00
|
|
|
boost::shared_ptr<request_callback> cb = requester();
|
2008-02-17 23:51:03 +01:00
|
|
|
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING
|
2008-01-06 23:14:00 +01:00
|
|
|
if (cb) cb->debug_log("*** UDP_TRACKER [ name lookup successful ]");
|
2006-04-25 23:04:48 +02:00
|
|
|
#endif
|
|
|
|
restart_read_timeout();
|
2007-03-02 19:40:02 +01:00
|
|
|
|
|
|
|
// look for an address that has the same kind as the one
|
|
|
|
// we're listening on. To make sure the tracker get our
|
|
|
|
// correct listening address.
|
2009-05-15 23:23:41 +02:00
|
|
|
|
|
|
|
std::transform(i, udp::resolver::iterator(), std::back_inserter(m_endpoints)
|
|
|
|
, boost::bind(&udp::resolver::iterator::value_type::endpoint, _1));
|
|
|
|
|
|
|
|
// remove endpoints that are filtered by the IP filter
|
|
|
|
for (std::list<udp::endpoint>::iterator i = m_endpoints.begin();
|
|
|
|
i != m_endpoints.end();)
|
2007-03-02 19:40:02 +01:00
|
|
|
{
|
2009-05-15 23:23:41 +02:00
|
|
|
if (m_ses.m_ip_filter.access(i->address()) == ip_filter::blocked)
|
|
|
|
i = m_endpoints.erase(i);
|
|
|
|
else
|
|
|
|
++i;
|
2007-03-02 19:40:02 +01:00
|
|
|
}
|
2008-10-22 21:40:32 +02:00
|
|
|
|
2009-05-15 23:23:41 +02:00
|
|
|
if (m_endpoints.empty())
|
2008-10-22 21:40:32 +02:00
|
|
|
{
|
|
|
|
fail(-1, "blocked by IP filter");
|
|
|
|
return;
|
|
|
|
}
|
2007-03-02 19:40:02 +01:00
|
|
|
|
2009-05-15 23:23:41 +02:00
|
|
|
std::list<udp::endpoint>::iterator iter = m_endpoints.begin();
|
|
|
|
m_target = *iter;
|
|
|
|
|
|
|
|
if (bind_interface() != address_v4::any())
|
2007-12-30 02:57:57 +01:00
|
|
|
{
|
2009-05-15 23:23:41 +02:00
|
|
|
// find first endpoint that matches our bind interface type
|
|
|
|
for (; iter != m_endpoints.end() && iter->address().is_v4()
|
|
|
|
!= bind_interface().is_v4(); ++iter);
|
|
|
|
|
|
|
|
if (iter == m_endpoints.end())
|
|
|
|
{
|
|
|
|
TORRENT_ASSERT(m_target.address().is_v4() != bind_interface().is_v4());
|
|
|
|
if (cb)
|
|
|
|
{
|
|
|
|
char const* tracker_address_type = m_target.address().is_v4() ? "IPv4" : "IPv6";
|
|
|
|
char const* bind_address_type = bind_interface().is_v4() ? "IPv4" : "IPv6";
|
|
|
|
char msg[200];
|
|
|
|
snprintf(msg, sizeof(msg)
|
|
|
|
, "the tracker only resolves to an %s address, and you're "
|
|
|
|
"listening on an %s socket. This may prevent you from receiving "
|
|
|
|
"incoming connections."
|
|
|
|
, tracker_address_type, bind_address_type);
|
|
|
|
|
|
|
|
cb->tracker_warning(tracker_req(), msg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_target = *iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cb) cb->m_tracker_address = tcp::endpoint(m_target.address(), m_target.port());
|
|
|
|
|
|
|
|
if (bind_interface() != address_v4::any())
|
|
|
|
{
|
|
|
|
error_code ec;
|
|
|
|
m_socket.bind(udp::endpoint(bind_interface(), 0), ec);
|
|
|
|
if (ec)
|
|
|
|
{
|
|
|
|
fail(-1, ec.message().c_str());
|
|
|
|
return;
|
|
|
|
}
|
2007-12-30 02:57:57 +01:00
|
|
|
}
|
2006-04-25 23:04:48 +02:00
|
|
|
send_udp_connect();
|
|
|
|
}
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
void udp_tracker_connection::on_timeout()
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2008-02-17 23:51:03 +01:00
|
|
|
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING || defined TORRENT_ERROR_LOGGING
|
2008-01-06 23:14:00 +01:00
|
|
|
boost::shared_ptr<request_callback> cb = requester();
|
2009-04-04 11:52:25 +02:00
|
|
|
char msg[200];
|
|
|
|
snprintf(msg, 200, "*** UDP_TRACKER [ timed out url: %s ]", tracker_req().url.c_str());
|
|
|
|
if (cb) cb->debug_log(msg);
|
2008-01-06 23:14:00 +01:00
|
|
|
#endif
|
2008-02-05 07:32:10 +01:00
|
|
|
m_socket.close();
|
2006-04-25 23:04:48 +02:00
|
|
|
m_name_lookup.cancel();
|
|
|
|
fail_timeout();
|
|
|
|
}
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2007-10-26 09:14:19 +02:00
|
|
|
void udp_tracker_connection::close()
|
|
|
|
{
|
2008-05-03 18:05:42 +02:00
|
|
|
error_code ec;
|
2008-02-05 07:32:10 +01:00
|
|
|
m_socket.close();
|
2007-10-26 09:14:19 +02:00
|
|
|
m_name_lookup.cancel();
|
|
|
|
tracker_connection::close();
|
|
|
|
}
|
|
|
|
|
2008-05-03 18:05:42 +02:00
|
|
|
void udp_tracker_connection::on_receive(error_code const& e
|
2008-03-25 05:46:18 +01:00
|
|
|
, udp::endpoint const& ep, char const* buf, int size)
|
2006-04-25 23:04:48 +02:00
|
|
|
{
|
2008-02-05 07:32:10 +01:00
|
|
|
// ignore resposes before we've sent any requests
|
|
|
|
if (m_state == action_error) return;
|
2006-04-25 23:04:48 +02:00
|
|
|
|
2007-10-22 06:17:26 +02:00
|
|
|
if (!m_socket.is_open()) return; // the operation was aborted
|
2005-08-09 01:32:38 +02:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
// ignore packet not sent from the tracker
|
|
|
|
if (m_target != ep) return;
|
2008-03-25 05:46:18 +01:00
|
|
|
|
2008-09-22 02:15:05 +02:00
|
|
|
received_bytes(size + 28); // assuming UDP/IP header
|
2008-03-25 05:46:18 +01:00
|
|
|
if (e) fail(-1, e.message().c_str());
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2008-02-17 23:51:03 +01:00
|
|
|
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING
|
2008-02-05 07:32:10 +01:00
|
|
|
boost::shared_ptr<request_callback> cb = requester();
|
|
|
|
if (cb)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2009-04-04 11:52:25 +02:00
|
|
|
char msg[200];
|
|
|
|
snprintf(msg, 200, "<== UDP_TRACKER_PACKET [ size: %d ]", size);
|
|
|
|
cb->debug_log(msg);
|
2006-04-25 23:04:48 +02:00
|
|
|
}
|
2008-02-05 07:32:10 +01:00
|
|
|
#endif
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
// ignore packets smaller than 8 bytes
|
|
|
|
if (size < 8) return;
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
restart_read_timeout();
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
const char* ptr = buf;
|
2006-04-25 23:04:48 +02:00
|
|
|
int action = detail::read_int32(ptr);
|
|
|
|
int transaction = detail::read_int32(ptr);
|
|
|
|
|
2008-02-17 23:51:03 +01:00
|
|
|
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING
|
2008-02-05 07:32:10 +01:00
|
|
|
if (cb)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2009-04-04 11:52:25 +02:00
|
|
|
char msg[200];
|
|
|
|
snprintf(msg, 200, "*** UDP_TRACKER_PACKET [ action: %d ]", action);
|
|
|
|
cb->debug_log(msg);
|
2004-01-31 11:46:15 +01:00
|
|
|
}
|
2008-02-05 07:32:10 +01:00
|
|
|
#endif
|
2006-04-25 23:04:48 +02:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
// ignore packets with incorrect transaction id
|
|
|
|
if (m_transaction_id != transaction) return;
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
if (action == action_error)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2008-02-05 07:32:10 +01:00
|
|
|
fail(-1, std::string(ptr, size - 8).c_str());
|
2006-04-25 23:04:48 +02:00
|
|
|
return;
|
2004-01-31 11:46:15 +01:00
|
|
|
}
|
2006-04-25 23:04:48 +02:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
// ignore packets that's not a response to our message
|
|
|
|
if (action != m_state) return;
|
2006-04-25 23:04:48 +02:00
|
|
|
|
2008-02-17 23:51:03 +01:00
|
|
|
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING
|
2007-09-14 04:54:15 +02:00
|
|
|
if (cb)
|
2005-04-12 13:34:40 +02:00
|
|
|
{
|
2009-04-04 11:52:25 +02:00
|
|
|
char msg[200];
|
|
|
|
snprintf(msg, 200, "*** UDP_TRACKER_RESPONSE [ cid: %x%x ]"
|
|
|
|
, int(m_connection_id >> 32), int(m_connection_id & 0xffffffff));
|
|
|
|
cb->debug_log(msg);
|
2005-04-12 13:34:40 +02:00
|
|
|
}
|
2006-04-25 23:04:48 +02:00
|
|
|
#endif
|
2005-04-12 13:34:40 +02:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
switch (m_state)
|
|
|
|
{
|
|
|
|
case action_connect:
|
|
|
|
on_connect_response(buf, size);
|
|
|
|
break;
|
|
|
|
case action_announce:
|
|
|
|
on_announce_response(buf, size);
|
|
|
|
break;
|
|
|
|
case action_scrape:
|
|
|
|
on_scrape_response(buf, size);
|
|
|
|
break;
|
|
|
|
default: break;
|
|
|
|
}
|
2006-04-25 23:04:48 +02:00
|
|
|
}
|
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
void udp_tracker_connection::on_connect_response(char const* buf, int size)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2008-02-05 07:32:10 +01:00
|
|
|
// ignore packets smaller than 16 bytes
|
|
|
|
if (size < 16) return;
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
restart_read_timeout();
|
|
|
|
buf += 8; // skip header
|
2006-10-03 00:19:21 +02:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
// reset transaction
|
|
|
|
m_transaction_id = 0;
|
|
|
|
m_attempts = 0;
|
|
|
|
m_connection_id = detail::read_int64(buf);
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
if (tracker_req().kind == tracker_request::announce_request)
|
|
|
|
send_udp_announce();
|
|
|
|
else if (tracker_req().kind == tracker_request::scrape_request)
|
|
|
|
send_udp_scrape();
|
|
|
|
}
|
2005-04-05 09:55:27 +02:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
void udp_tracker_connection::send_udp_connect()
|
|
|
|
{
|
2008-02-17 23:51:03 +01:00
|
|
|
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING
|
2007-09-14 04:54:15 +02:00
|
|
|
boost::shared_ptr<request_callback> cb = requester();
|
|
|
|
if (cb)
|
2006-04-25 23:04:48 +02:00
|
|
|
{
|
2009-04-04 11:52:25 +02:00
|
|
|
char hex_ih[41];
|
|
|
|
to_hex((char const*)&tracker_req().info_hash[0], 20, hex_ih);
|
|
|
|
char msg[200];
|
|
|
|
snprintf(msg, 200, "==> UDP_TRACKER_CONNECT [%s]", hex_ih);
|
|
|
|
cb->debug_log(msg);
|
2006-04-25 23:04:48 +02:00
|
|
|
}
|
|
|
|
#endif
|
2008-02-05 07:32:10 +01:00
|
|
|
if (!m_socket.is_open()) return; // the operation was aborted
|
|
|
|
|
|
|
|
char buf[16];
|
|
|
|
char* ptr = buf;
|
|
|
|
|
|
|
|
if (m_transaction_id == 0)
|
2009-01-27 09:24:48 +01:00
|
|
|
m_transaction_id = std::rand() ^ (std::rand() << 16);
|
2008-02-05 07:32:10 +01:00
|
|
|
|
|
|
|
detail::write_uint32(0x417, ptr);
|
|
|
|
detail::write_uint32(0x27101980, ptr); // connection_id
|
|
|
|
detail::write_int32(action_connect, ptr); // action (connect)
|
|
|
|
detail::write_int32(m_transaction_id, ptr); // transaction_id
|
|
|
|
TORRENT_ASSERT(ptr - buf == sizeof(buf));
|
2006-04-25 23:04:48 +02:00
|
|
|
|
2008-05-03 18:05:42 +02:00
|
|
|
error_code ec;
|
2008-02-05 07:32:10 +01:00
|
|
|
m_socket.send(m_target, buf, 16, ec);
|
|
|
|
m_state = action_connect;
|
2008-09-22 02:15:05 +02:00
|
|
|
sent_bytes(16 + 28); // assuming UDP/IP header
|
2004-01-31 11:46:15 +01:00
|
|
|
++m_attempts;
|
2007-12-30 02:57:57 +01:00
|
|
|
if (ec)
|
|
|
|
{
|
|
|
|
fail(-1, ec.message().c_str());
|
|
|
|
return;
|
|
|
|
}
|
2004-01-31 11:46:15 +01:00
|
|
|
}
|
|
|
|
|
2005-04-12 13:34:40 +02:00
|
|
|
void udp_tracker_connection::send_udp_scrape()
|
|
|
|
{
|
|
|
|
if (m_transaction_id == 0)
|
2009-01-27 09:24:48 +01:00
|
|
|
m_transaction_id = std::rand() ^ (std::rand() << 16);
|
2005-04-12 13:34:40 +02:00
|
|
|
|
2007-10-22 06:17:26 +02:00
|
|
|
if (!m_socket.is_open()) return; // the operation was aborted
|
2006-10-03 00:19:21 +02:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
char buf[8 + 4 + 4 + 20];
|
|
|
|
char* out = buf;
|
2005-04-12 13:34:40 +02:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
detail::write_int64(m_connection_id, out); // connection_id
|
|
|
|
detail::write_int32(action_scrape, out); // action (scrape)
|
|
|
|
detail::write_int32(m_transaction_id, out); // transaction_id
|
2005-04-12 13:34:40 +02:00
|
|
|
// info_hash
|
2006-04-25 23:04:48 +02:00
|
|
|
std::copy(tracker_req().info_hash.begin(), tracker_req().info_hash.end(), out);
|
2008-02-25 09:50:07 +01:00
|
|
|
out += 20;
|
|
|
|
TORRENT_ASSERT(out - buf == sizeof(buf));
|
2005-04-12 13:34:40 +02:00
|
|
|
|
2008-05-03 18:05:42 +02:00
|
|
|
error_code ec;
|
2008-02-05 07:32:10 +01:00
|
|
|
m_socket.send(m_target, buf, sizeof(buf), ec);
|
|
|
|
m_state = action_scrape;
|
2008-09-22 02:15:05 +02:00
|
|
|
sent_bytes(sizeof(buf) + 28); // assuming UDP/IP header
|
2005-04-12 13:34:40 +02:00
|
|
|
++m_attempts;
|
2007-12-30 02:57:57 +01:00
|
|
|
if (ec)
|
|
|
|
{
|
|
|
|
fail(-1, ec.message().c_str());
|
|
|
|
return;
|
|
|
|
}
|
2005-04-12 13:34:40 +02:00
|
|
|
}
|
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
void udp_tracker_connection::on_announce_response(char const* buf, int size)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2008-02-05 07:32:10 +01:00
|
|
|
if (size < 20) return;
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
restart_read_timeout();
|
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
buf += 8; // skip header
|
|
|
|
restart_read_timeout();
|
2004-01-31 11:46:15 +01:00
|
|
|
int interval = detail::read_int32(buf);
|
2005-02-21 14:59:24 +01:00
|
|
|
int incomplete = detail::read_int32(buf);
|
|
|
|
int complete = detail::read_int32(buf);
|
2008-02-05 07:32:10 +01:00
|
|
|
int num_peers = (size - 20) / 6;
|
|
|
|
if ((size - 20) % 6 != 0)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2006-04-25 23:04:48 +02:00
|
|
|
fail(-1, "invalid udp tracker response length");
|
|
|
|
return;
|
2004-01-31 11:46:15 +01:00
|
|
|
}
|
|
|
|
|
2007-09-14 04:54:15 +02:00
|
|
|
boost::shared_ptr<request_callback> cb = requester();
|
2008-02-17 23:51:03 +01:00
|
|
|
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING
|
2007-09-14 04:54:15 +02:00
|
|
|
if (cb)
|
2006-04-25 23:04:48 +02:00
|
|
|
{
|
2008-12-30 18:37:38 +01:00
|
|
|
boost::shared_ptr<request_callback> cb = requester();
|
2009-04-04 11:52:25 +02:00
|
|
|
char msg[200];
|
|
|
|
snprintf(msg, 200, "<== UDP_TRACKER_RESPONSE [ url: %s ]", tracker_req().url.c_str());
|
|
|
|
cb->debug_log(msg);
|
2006-04-25 23:04:48 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2007-09-14 04:54:15 +02:00
|
|
|
if (!cb)
|
2006-04-25 23:04:48 +02:00
|
|
|
{
|
|
|
|
m_man.remove_request(this);
|
|
|
|
return;
|
|
|
|
}
|
2004-01-31 11:46:15 +01:00
|
|
|
|
|
|
|
std::vector<peer_entry> peer_list;
|
|
|
|
for (int i = 0; i < num_peers; ++i)
|
|
|
|
{
|
2008-02-05 07:32:10 +01:00
|
|
|
// TODO: don't use a string here
|
2004-01-31 11:46:15 +01:00
|
|
|
peer_entry e;
|
2009-04-04 11:52:25 +02:00
|
|
|
char ip_string[100];
|
|
|
|
unsigned int a = detail::read_uint8(buf);
|
|
|
|
unsigned int b = detail::read_uint8(buf);
|
|
|
|
unsigned int c = detail::read_uint8(buf);
|
|
|
|
unsigned int d = detail::read_uint8(buf);
|
|
|
|
snprintf(ip_string, 100, "%u.%u.%u.%u", a, b, c, d);
|
|
|
|
e.ip = ip_string;
|
2004-01-31 11:46:15 +01:00
|
|
|
e.port = detail::read_uint16(buf);
|
2006-04-25 23:04:48 +02:00
|
|
|
e.pid.clear();
|
2004-01-31 11:46:15 +01:00
|
|
|
peer_list.push_back(e);
|
|
|
|
}
|
|
|
|
|
2009-05-15 23:23:41 +02:00
|
|
|
std::list<address> ip_list;
|
|
|
|
std::transform(m_endpoints.begin()
|
|
|
|
, m_endpoints.end()
|
|
|
|
, std::back_inserter(ip_list)
|
|
|
|
, boost::bind(&udp::endpoint::address, _1));
|
|
|
|
|
|
|
|
cb->tracker_response(tracker_req(), m_target.address(), ip_list
|
|
|
|
, peer_list, interval, complete, incomplete, address());
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
m_man.remove_request(this);
|
2007-10-26 09:14:19 +02:00
|
|
|
close();
|
2006-04-25 23:04:48 +02:00
|
|
|
}
|
2005-04-12 13:34:40 +02:00
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
void udp_tracker_connection::on_scrape_response(char const* buf, int size)
|
2006-04-25 23:04:48 +02:00
|
|
|
{
|
2008-02-05 07:32:10 +01:00
|
|
|
buf += 8; // skip header
|
2005-04-12 13:34:40 +02:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
restart_read_timeout();
|
|
|
|
int action = detail::read_int32(buf);
|
|
|
|
int transaction = detail::read_int32(buf);
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
if (transaction != m_transaction_id)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2006-04-25 23:04:48 +02:00
|
|
|
fail(-1, "incorrect transaction id");
|
|
|
|
return;
|
2004-01-31 11:46:15 +01:00
|
|
|
}
|
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
if (action == action_error)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2008-02-05 07:32:10 +01:00
|
|
|
fail(-1, std::string(buf, size - 8).c_str());
|
2006-04-25 23:04:48 +02:00
|
|
|
return;
|
2004-01-31 11:46:15 +01:00
|
|
|
}
|
2006-04-25 23:04:48 +02:00
|
|
|
|
|
|
|
if (action != action_scrape)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2006-04-25 23:04:48 +02:00
|
|
|
fail(-1, "invalid action in announce response");
|
|
|
|
return;
|
2004-01-31 11:46:15 +01:00
|
|
|
}
|
|
|
|
|
2008-02-05 07:32:10 +01:00
|
|
|
if (size < 20)
|
2004-01-31 11:46:15 +01:00
|
|
|
{
|
2006-04-25 23:04:48 +02:00
|
|
|
fail(-1, "got a message with size < 20");
|
|
|
|
return;
|
2004-01-31 11:46:15 +01:00
|
|
|
}
|
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
int complete = detail::read_int32(buf);
|
2007-11-20 23:46:27 +01:00
|
|
|
int downloaded = detail::read_int32(buf);
|
2006-04-25 23:04:48 +02:00
|
|
|
int incomplete = detail::read_int32(buf);
|
2005-04-12 13:34:40 +02:00
|
|
|
|
2007-09-14 04:54:15 +02:00
|
|
|
boost::shared_ptr<request_callback> cb = requester();
|
|
|
|
if (!cb)
|
2006-04-25 23:04:48 +02:00
|
|
|
{
|
2007-10-26 09:14:19 +02:00
|
|
|
close();
|
2006-04-25 23:04:48 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-11-20 23:46:27 +01:00
|
|
|
cb->tracker_scrape_response(tracker_req()
|
|
|
|
, complete, incomplete, downloaded);
|
2004-01-31 11:46:15 +01:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
m_man.remove_request(this);
|
2007-10-26 09:14:19 +02:00
|
|
|
close();
|
2006-04-25 23:04:48 +02:00
|
|
|
}
|
2008-02-05 07:32:10 +01:00
|
|
|
|
|
|
|
void udp_tracker_connection::send_udp_announce()
|
|
|
|
{
|
|
|
|
if (m_transaction_id == 0)
|
2009-01-27 09:24:48 +01:00
|
|
|
m_transaction_id = std::rand() ^ (std::rand() << 16);
|
2008-02-05 07:32:10 +01:00
|
|
|
|
|
|
|
if (!m_socket.is_open()) return; // the operation was aborted
|
|
|
|
|
|
|
|
char buf[8 + 4 + 4 + 20 + 20 + 8 + 8 + 8 + 4 + 4 + 4 + 4 + 2 + 2];
|
|
|
|
char* out = buf;
|
|
|
|
|
|
|
|
tracker_request const& req = tracker_req();
|
2008-10-22 21:40:32 +02:00
|
|
|
session_settings const& settings = m_ses.settings();
|
2008-02-05 07:32:10 +01:00
|
|
|
|
|
|
|
detail::write_int64(m_connection_id, out); // connection_id
|
|
|
|
detail::write_int32(action_announce, out); // action (announce)
|
|
|
|
detail::write_int32(m_transaction_id, out); // transaction_id
|
|
|
|
std::copy(req.info_hash.begin(), req.info_hash.end(), out); // info_hash
|
2008-02-25 09:50:07 +01:00
|
|
|
out += 20;
|
2008-02-05 07:32:10 +01:00
|
|
|
std::copy(req.pid.begin(), req.pid.end(), out); // peer_id
|
2008-02-25 09:50:07 +01:00
|
|
|
out += 20;
|
2008-02-05 07:32:10 +01:00
|
|
|
detail::write_int64(req.downloaded, out); // downloaded
|
|
|
|
detail::write_int64(req.left, out); // left
|
|
|
|
detail::write_int64(req.uploaded, out); // uploaded
|
|
|
|
detail::write_int32(req.event, out); // event
|
|
|
|
// ip address
|
2008-10-22 21:40:32 +02:00
|
|
|
if (settings.announce_ip != address() && settings.announce_ip.is_v4())
|
|
|
|
detail::write_uint32(settings.announce_ip.to_v4().to_ulong(), out);
|
2008-02-05 07:32:10 +01:00
|
|
|
else
|
|
|
|
detail::write_int32(0, out);
|
|
|
|
detail::write_int32(req.key, out); // key
|
|
|
|
detail::write_int32(req.num_want, out); // num_want
|
|
|
|
detail::write_uint16(req.listen_port, out); // port
|
|
|
|
detail::write_uint16(0, out); // extensions
|
|
|
|
|
|
|
|
TORRENT_ASSERT(out - buf == sizeof(buf));
|
|
|
|
|
2008-02-17 23:51:03 +01:00
|
|
|
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING
|
2008-02-05 07:32:10 +01:00
|
|
|
boost::shared_ptr<request_callback> cb = requester();
|
2009-04-24 19:04:20 +02:00
|
|
|
if (cb)
|
|
|
|
{
|
|
|
|
char hex_ih[41];
|
|
|
|
to_hex((char const*)&req.info_hash[0], 20, hex_ih);
|
|
|
|
char msg[200];
|
|
|
|
snprintf(msg, 200, "==> UDP_TRACKER_ANNOUNCE [%s]", hex_ih);
|
|
|
|
cb->debug_log(msg);
|
|
|
|
}
|
2008-02-05 07:32:10 +01:00
|
|
|
#endif
|
|
|
|
|
2008-05-03 18:05:42 +02:00
|
|
|
error_code ec;
|
2008-02-05 07:32:10 +01:00
|
|
|
m_socket.send(m_target, buf, sizeof(buf), ec);
|
|
|
|
m_state = action_announce;
|
2008-09-22 02:15:05 +02:00
|
|
|
sent_bytes(sizeof(buf) + 28); // assuming UDP/IP header
|
2008-02-05 07:32:10 +01:00
|
|
|
++m_attempts;
|
|
|
|
if (ec)
|
|
|
|
{
|
|
|
|
fail(-1, ec.message().c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-31 11:46:15 +01:00
|
|
|
}
|
2005-03-19 13:22:40 +01:00
|
|
|
|