2006-08-01 17:27:08 +02:00
|
|
|
/*
|
|
|
|
|
|
|
|
Copyright (c) 2006, 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"
|
|
|
|
|
2006-08-01 17:27:08 +02:00
|
|
|
#include <utility>
|
|
|
|
#include <boost/bind.hpp>
|
2009-11-23 09:38:50 +01:00
|
|
|
#include <boost/function/function1.hpp>
|
2006-08-01 17:27:08 +02:00
|
|
|
|
|
|
|
#include "libtorrent/io.hpp"
|
|
|
|
#include "libtorrent/hasher.hpp"
|
2008-09-20 19:42:25 +02:00
|
|
|
#include "libtorrent/alert_types.hpp"
|
2009-09-20 02:23:36 +02:00
|
|
|
#include "libtorrent/socket.hpp"
|
2008-09-20 19:42:25 +02:00
|
|
|
#include "libtorrent/aux_/session_impl.hpp"
|
2006-08-01 17:27:08 +02:00
|
|
|
#include "libtorrent/kademlia/node_id.hpp"
|
|
|
|
#include "libtorrent/kademlia/rpc_manager.hpp"
|
|
|
|
#include "libtorrent/kademlia/routing_table.hpp"
|
|
|
|
#include "libtorrent/kademlia/node.hpp"
|
|
|
|
|
|
|
|
#include "libtorrent/kademlia/refresh.hpp"
|
|
|
|
#include "libtorrent/kademlia/find_data.hpp"
|
|
|
|
|
|
|
|
namespace libtorrent { namespace dht
|
|
|
|
{
|
|
|
|
|
2009-09-25 18:32:02 +02:00
|
|
|
void incoming_error(entry& e, char const* msg);
|
|
|
|
|
2009-09-20 02:23:36 +02:00
|
|
|
using detail::write_endpoint;
|
|
|
|
|
2009-09-27 05:38:41 +02:00
|
|
|
int search_torrent_entry::match(char const* in_tags[], int num_tags) const
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
for (int i = 0; i < num_tags; ++i)
|
|
|
|
{
|
|
|
|
char const* t = in_tags[i];
|
2009-10-05 01:46:29 +02:00
|
|
|
std::map<std::string, int>::const_iterator j = tags.find(t);
|
|
|
|
if (j == tags.end()) continue;
|
2009-09-27 05:38:41 +02:00
|
|
|
// weigh the score by how popular this tag is in this torrent
|
2009-10-05 01:46:29 +02:00
|
|
|
ret += 100 * j->second / total_tag_points;
|
2009-09-27 05:38:41 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool search_torrent_entry::tick()
|
|
|
|
{
|
|
|
|
int sum = 0;
|
|
|
|
for (std::map<std::string, int>::iterator i = tags.begin()
|
|
|
|
, end(tags.end()); i != end;)
|
|
|
|
{
|
|
|
|
i->second = (i->second * 2) / 3;
|
|
|
|
sum += i->second;
|
|
|
|
if (i->second > 0) { ++i; continue; }
|
|
|
|
tags.erase(i++);
|
|
|
|
}
|
|
|
|
total_tag_points = sum;
|
|
|
|
|
|
|
|
sum = 0;
|
|
|
|
for (std::map<std::string, int>::iterator i = name.begin()
|
|
|
|
, end(name.end()); i != end;)
|
|
|
|
{
|
|
|
|
i->second = (i->second * 2) / 3;
|
|
|
|
sum += i->second;
|
|
|
|
if (i->second > 0) { ++i; continue; }
|
|
|
|
name.erase(i++);
|
|
|
|
}
|
|
|
|
total_name_points = sum;
|
|
|
|
|
|
|
|
return total_tag_points == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void search_torrent_entry::publish(std::string const& torrent_name, char const* in_tags[]
|
|
|
|
, int num_tags)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < num_tags; ++i)
|
|
|
|
{
|
|
|
|
char const* t = in_tags[i];
|
2009-10-05 01:46:29 +02:00
|
|
|
std::map<std::string, int>::iterator j = tags.find(t);
|
|
|
|
if (j != tags.end())
|
|
|
|
++j->second;
|
2009-09-27 05:38:41 +02:00
|
|
|
else
|
|
|
|
tags[t] = 1;
|
|
|
|
++total_tag_points;
|
|
|
|
// TODO: limit the number of tags
|
|
|
|
}
|
|
|
|
|
|
|
|
name[torrent_name] += 1;
|
|
|
|
++total_name_points;
|
|
|
|
|
|
|
|
// TODO: limit the number of names
|
|
|
|
}
|
|
|
|
|
|
|
|
void search_torrent_entry::get_name(std::string& t) const
|
|
|
|
{
|
|
|
|
std::map<std::string, int>::const_iterator max = name.begin();
|
|
|
|
for (std::map<std::string, int>::const_iterator i = name.begin()
|
|
|
|
, end(name.end()); i != end; ++i)
|
|
|
|
{
|
|
|
|
if (i->second > max->second) max = i;
|
|
|
|
}
|
|
|
|
t = max->first;
|
|
|
|
}
|
|
|
|
|
|
|
|
void search_torrent_entry::get_tags(std::string& t) const
|
|
|
|
{
|
|
|
|
for (std::map<std::string, int>::const_iterator i = tags.begin()
|
|
|
|
, end(tags.end()); i != end; ++i)
|
|
|
|
{
|
|
|
|
if (i != tags.begin()) t += " ";
|
|
|
|
t += i->first;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-08-02 00:23:05 +02:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
char rand() { return (char)std::rand(); }
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// TODO: configurable?
|
2006-08-01 17:27:08 +02:00
|
|
|
enum { announce_interval = 30 };
|
|
|
|
|
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
|
|
|
TORRENT_DEFINE_LOG(node)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// remove peers that have timed out
|
|
|
|
void purge_peers(std::set<peer_entry>& peers)
|
|
|
|
{
|
|
|
|
for (std::set<peer_entry>::iterator i = peers.begin()
|
|
|
|
, end(peers.end()); i != end;)
|
|
|
|
{
|
|
|
|
// the peer has timed out
|
2007-04-05 00:27:36 +02:00
|
|
|
if (i->added + minutes(int(announce_interval * 1.5f)) < time_now())
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
2008-11-10 03:08:42 +01:00
|
|
|
TORRENT_LOG(node) << "peer timed out at: " << i->addr;
|
2006-08-01 17:27:08 +02:00
|
|
|
#endif
|
|
|
|
peers.erase(i++);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void nop() {}
|
|
|
|
|
2009-09-27 05:38:41 +02:00
|
|
|
// TODO: the session_impl argument could be an alert reference
|
|
|
|
// instead, and make the dht_tracker less dependent on session_impl
|
|
|
|
// which would make it simpler to unit test
|
2008-09-20 19:42:25 +02:00
|
|
|
node_impl::node_impl(libtorrent::aux::session_impl& ses
|
2009-10-07 22:51:02 +02:00
|
|
|
, bool (*f)(void*, entry const&, udp::endpoint const&, int)
|
2008-11-11 09:33:34 +01:00
|
|
|
, dht_settings const& settings
|
2009-11-23 09:38:50 +01:00
|
|
|
, node_id nid
|
2009-09-20 02:23:36 +02:00
|
|
|
, void* userdata)
|
2006-08-01 17:27:08 +02:00
|
|
|
: m_settings(settings)
|
2010-02-14 05:58:21 +01:00
|
|
|
, m_id(nid == (node_id::min)() ? generate_id() : nid)
|
2006-08-01 17:27:08 +02:00
|
|
|
, m_table(m_id, 8, settings)
|
2009-09-20 02:23:36 +02:00
|
|
|
, m_rpc(m_id, m_table, f, userdata)
|
2007-04-05 00:27:36 +02:00
|
|
|
, m_last_tracker_tick(time_now())
|
2008-09-20 19:42:25 +02:00
|
|
|
, m_ses(ses)
|
2009-09-20 02:23:36 +02:00
|
|
|
, m_send(f)
|
|
|
|
, m_userdata(userdata)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
|
|
|
m_secret[0] = std::rand();
|
|
|
|
m_secret[1] = std::rand();
|
|
|
|
}
|
|
|
|
|
2009-09-20 02:23:36 +02:00
|
|
|
bool node_impl::verify_token(std::string const& token, char const* info_hash
|
|
|
|
, udp::endpoint const& addr)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2007-05-14 19:49:36 +02:00
|
|
|
if (token.length() != 4)
|
|
|
|
{
|
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
|
|
|
TORRENT_LOG(node) << "token of incorrect length: " << token.length();
|
|
|
|
#endif
|
|
|
|
return false;
|
|
|
|
}
|
2006-08-01 17:27:08 +02:00
|
|
|
|
|
|
|
hasher h1;
|
2008-10-21 10:45:42 +02:00
|
|
|
error_code ec;
|
2009-09-20 02:23:36 +02:00
|
|
|
std::string address = addr.address().to_string(ec);
|
2008-10-21 10:45:42 +02:00
|
|
|
if (ec) return false;
|
2006-08-01 17:27:08 +02:00
|
|
|
h1.update(&address[0], address.length());
|
|
|
|
h1.update((char*)&m_secret[0], sizeof(m_secret[0]));
|
2009-09-20 02:23:36 +02:00
|
|
|
h1.update((char*)info_hash, sha1_hash::size);
|
2006-08-01 17:27:08 +02:00
|
|
|
|
|
|
|
sha1_hash h = h1.final();
|
|
|
|
if (std::equal(token.begin(), token.end(), (signed char*)&h[0]))
|
|
|
|
return true;
|
2007-05-14 19:49:36 +02:00
|
|
|
|
2006-08-01 17:27:08 +02:00
|
|
|
hasher h2;
|
|
|
|
h2.update(&address[0], address.length());
|
|
|
|
h2.update((char*)&m_secret[1], sizeof(m_secret[1]));
|
2009-09-20 02:23:36 +02:00
|
|
|
h2.update((char*)info_hash, sha1_hash::size);
|
2006-08-01 17:27:08 +02:00
|
|
|
h = h2.final();
|
|
|
|
if (std::equal(token.begin(), token.end(), (signed char*)&h[0]))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-09-20 02:23:36 +02:00
|
|
|
std::string node_impl::generate_token(udp::endpoint const& addr, char const* info_hash)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
|
|
|
std::string token;
|
|
|
|
token.resize(4);
|
|
|
|
hasher h;
|
2008-10-21 10:45:42 +02:00
|
|
|
error_code ec;
|
2009-09-20 02:23:36 +02:00
|
|
|
std::string address = addr.address().to_string(ec);
|
2008-10-21 10:45:42 +02:00
|
|
|
TORRENT_ASSERT(!ec);
|
2006-08-01 17:27:08 +02:00
|
|
|
h.update(&address[0], address.length());
|
|
|
|
h.update((char*)&m_secret[0], sizeof(m_secret[0]));
|
2009-09-20 02:23:36 +02:00
|
|
|
h.update(info_hash, sha1_hash::size);
|
2006-08-01 17:27:08 +02:00
|
|
|
|
|
|
|
sha1_hash hash = h.final();
|
|
|
|
std::copy(hash.begin(), hash.begin() + 4, (signed char*)&token[0]);
|
2008-11-11 18:51:02 +01:00
|
|
|
return token;
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void node_impl::refresh(node_id const& id
|
2009-09-20 02:23:36 +02:00
|
|
|
, find_data::nodes_callback const& f)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2009-09-20 02:23:36 +02:00
|
|
|
boost::intrusive_ptr<dht::refresh> r(new dht::refresh(*this, id, f));
|
|
|
|
r->start();
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void node_impl::bootstrap(std::vector<udp::endpoint> const& nodes
|
2009-09-20 02:23:36 +02:00
|
|
|
, find_data::nodes_callback const& f)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2009-09-20 02:23:36 +02:00
|
|
|
boost::intrusive_ptr<dht::refresh> r(new dht::refresh(*this, m_id, f));
|
|
|
|
|
2010-02-14 08:46:57 +01:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
|
|
|
int count = 0;
|
|
|
|
#endif
|
|
|
|
|
2008-03-24 03:19:47 +01:00
|
|
|
for (std::vector<udp::endpoint>::const_iterator i = nodes.begin()
|
|
|
|
, end(nodes.end()); i != end; ++i)
|
2009-09-20 02:23:36 +02:00
|
|
|
{
|
2010-02-14 08:46:57 +01:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
|
|
|
++count;
|
|
|
|
#endif
|
2009-09-20 02:23:36 +02:00
|
|
|
r->add_entry(node_id(0), *i, traversal_algorithm::result::initial);
|
|
|
|
}
|
|
|
|
|
2010-02-14 08:46:57 +01:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
|
|
|
TORRENT_LOG(node) << "bootstrapping with " << count << " nodes";
|
|
|
|
#endif
|
2009-09-20 02:23:36 +02:00
|
|
|
r->start();
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
2010-01-03 12:08:39 +01:00
|
|
|
/*
|
2006-08-01 17:27:08 +02:00
|
|
|
void node_impl::refresh()
|
|
|
|
{
|
2009-09-20 02:23:36 +02:00
|
|
|
boost::intrusive_ptr<dht::refresh> r(new dht::refresh(*this, m_id, boost::bind(&nop)));
|
|
|
|
r->start();
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
2010-01-03 12:08:39 +01:00
|
|
|
*/
|
2006-08-01 17:27:08 +02:00
|
|
|
int node_impl::bucket_size(int bucket)
|
|
|
|
{
|
|
|
|
return m_table.bucket_size(bucket);
|
|
|
|
}
|
|
|
|
|
|
|
|
void node_impl::new_write_key()
|
|
|
|
{
|
|
|
|
m_secret[1] = m_secret[0];
|
|
|
|
m_secret[0] = std::rand();
|
|
|
|
}
|
|
|
|
|
2008-05-08 02:22:17 +02:00
|
|
|
void node_impl::unreachable(udp::endpoint const& ep)
|
|
|
|
{
|
|
|
|
m_rpc.unreachable(ep);
|
|
|
|
}
|
|
|
|
|
2006-08-01 17:27:08 +02:00
|
|
|
void node_impl::incoming(msg const& m)
|
|
|
|
{
|
2009-09-20 02:23:36 +02:00
|
|
|
// is this a reply?
|
|
|
|
lazy_entry const* y_ent = m.message.dict_find_string("y");
|
|
|
|
if (!y_ent || y_ent->string_length() == 0)
|
|
|
|
{
|
|
|
|
entry e;
|
|
|
|
incoming_error(e, "missing 'y' entry");
|
|
|
|
m_send(m_userdata, e, m.addr, 0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
char y = *(y_ent->string_ptr());
|
|
|
|
|
|
|
|
switch (y)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2009-09-20 02:23:36 +02:00
|
|
|
case 'r':
|
|
|
|
{
|
2010-01-03 12:08:39 +01:00
|
|
|
node_id id;
|
|
|
|
if (m_rpc.incoming(m, &id))
|
|
|
|
refresh(id, boost::bind(&nop));
|
2009-09-20 02:23:36 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'q':
|
|
|
|
{
|
|
|
|
TORRENT_ASSERT(m.message.dict_find_string_value("y") == "q");
|
|
|
|
entry e;
|
|
|
|
incoming_request(m, e);
|
|
|
|
m_send(m_userdata, e, m.addr, 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'e':
|
|
|
|
{
|
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
|
|
|
lazy_entry const* err = m.message.dict_find_list("e");
|
|
|
|
if (err && err->list_size() >= 2)
|
|
|
|
{
|
|
|
|
TORRENT_LOG(node) << "INCOMING ERROR: " << err->list_string_value_at(1);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
}
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
2008-12-23 21:04:12 +01:00
|
|
|
void announce_fun(std::vector<std::pair<node_entry, std::string> > const& v
|
2009-10-07 22:51:02 +02:00
|
|
|
, node_impl& node, int listen_port, sha1_hash const& ih)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2008-03-24 03:19:47 +01:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
2008-12-23 21:04:12 +01:00
|
|
|
TORRENT_LOG(node) << "sending announce_peer [ ih: " << ih
|
2008-03-24 03:19:47 +01:00
|
|
|
<< " p: " << listen_port
|
|
|
|
<< " nodes: " << v.size() << " ]" ;
|
|
|
|
#endif
|
2009-10-07 22:51:02 +02:00
|
|
|
|
|
|
|
// create a dummy traversal_algorithm
|
|
|
|
boost::intrusive_ptr<traversal_algorithm> algo(
|
2009-10-23 08:41:09 +02:00
|
|
|
new traversal_algorithm(node, (node_id::min)()));
|
2009-10-07 22:51:02 +02:00
|
|
|
|
2008-12-23 21:04:12 +01:00
|
|
|
// store on the first k nodes
|
|
|
|
for (std::vector<std::pair<node_entry, std::string> >::const_iterator i = v.begin()
|
2006-08-01 17:27:08 +02:00
|
|
|
, end(v.end()); i != end; ++i)
|
|
|
|
{
|
2008-05-08 03:17:14 +02:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
2008-12-23 21:04:12 +01:00
|
|
|
TORRENT_LOG(node) << " distance: " << (160 - distance_exp(ih, i->first.id));
|
2008-05-08 03:17:14 +02:00
|
|
|
#endif
|
2008-12-23 21:04:12 +01:00
|
|
|
|
2009-10-07 22:51:02 +02:00
|
|
|
void* ptr = node.m_rpc.allocator().malloc();
|
2009-05-14 00:18:41 +02:00
|
|
|
if (ptr == 0) return;
|
2009-10-07 22:51:02 +02:00
|
|
|
node.m_rpc.allocator().set_next_size(10);
|
|
|
|
observer_ptr o(new (ptr) announce_observer(algo));
|
2008-11-29 22:33:21 +01:00
|
|
|
#ifdef TORRENT_DEBUG
|
2008-02-09 22:04:24 +01:00
|
|
|
o->m_in_constructor = false;
|
|
|
|
#endif
|
2009-09-20 02:23:36 +02:00
|
|
|
entry e;
|
|
|
|
e["y"] = "q";
|
|
|
|
e["q"] = "announce_peer";
|
|
|
|
entry& a = e["a"];
|
2009-10-07 22:51:02 +02:00
|
|
|
a["info_hash"] = ih.to_string();
|
2009-09-20 02:23:36 +02:00
|
|
|
a["port"] = listen_port;
|
|
|
|
a["token"] = i->second;
|
2009-10-07 22:51:02 +02:00
|
|
|
node.m_rpc.invoke(e, i->first.ep(), o);
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-09-27 19:20:18 +02:00
|
|
|
void node_impl::add_router_node(udp::endpoint router)
|
|
|
|
{
|
2008-03-24 03:19:47 +01:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
|
|
|
TORRENT_LOG(node) << "adding router node: " << router;
|
|
|
|
#endif
|
2006-09-27 19:20:18 +02:00
|
|
|
m_table.add_router_node(router);
|
|
|
|
}
|
|
|
|
|
2006-08-01 17:27:08 +02:00
|
|
|
void node_impl::add_node(udp::endpoint node)
|
|
|
|
{
|
|
|
|
// ping the node, and if we get a reply, it
|
|
|
|
// will be added to the routing table
|
2009-05-14 00:18:41 +02:00
|
|
|
void* ptr = m_rpc.allocator().malloc();
|
|
|
|
if (ptr == 0) return;
|
|
|
|
m_rpc.allocator().set_next_size(10);
|
2009-10-07 22:51:02 +02:00
|
|
|
|
|
|
|
// create a dummy traversal_algorithm
|
|
|
|
boost::intrusive_ptr<traversal_algorithm> algo(
|
2009-10-26 02:29:39 +01:00
|
|
|
new traversal_algorithm(*this, (node_id::min)()));
|
2009-10-07 22:51:02 +02:00
|
|
|
observer_ptr o(new (ptr) null_observer(algo));
|
2008-11-29 22:33:21 +01:00
|
|
|
#ifdef TORRENT_DEBUG
|
2008-02-09 22:04:24 +01:00
|
|
|
o->m_in_constructor = false;
|
|
|
|
#endif
|
2009-09-20 02:23:36 +02:00
|
|
|
entry e;
|
|
|
|
e["y"] = "q";
|
|
|
|
e["q"] = "ping";
|
|
|
|
m_rpc.invoke(e, node, o);
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void node_impl::announce(sha1_hash const& info_hash, int listen_port
|
2008-12-23 21:04:12 +01:00
|
|
|
, boost::function<void(std::vector<tcp::endpoint> const&)> f)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2008-03-24 03:19:47 +01:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
|
|
|
TORRENT_LOG(node) << "announcing [ ih: " << info_hash << " p: " << listen_port << " ]" ;
|
|
|
|
#endif
|
2008-12-23 21:04:12 +01:00
|
|
|
// search for nodes with ids close to id or with peers
|
|
|
|
// for info-hash id. then send announce_peer to them.
|
2009-09-20 02:23:36 +02:00
|
|
|
boost::intrusive_ptr<find_data> ta(new find_data(*this, info_hash, f
|
2009-10-07 22:51:02 +02:00
|
|
|
, boost::bind(&announce_fun, _1, boost::ref(*this)
|
2009-09-20 02:23:36 +02:00
|
|
|
, listen_port, info_hash)));
|
|
|
|
ta->start();
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
|
|
|
|
2010-01-03 12:08:39 +01:00
|
|
|
void node_impl::tick()
|
2006-08-06 18:36:00 +02:00
|
|
|
{
|
2010-01-03 12:08:39 +01:00
|
|
|
node_id target;
|
|
|
|
if (m_table.need_refresh(target))
|
|
|
|
refresh(target, boost::bind(&nop));
|
2006-08-06 18:36:00 +02:00
|
|
|
}
|
|
|
|
|
2007-01-07 14:55:27 +01:00
|
|
|
time_duration node_impl::connection_timeout()
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
|
|
|
time_duration d = m_rpc.tick();
|
2008-10-21 19:10:11 +02:00
|
|
|
ptime now(time_now());
|
|
|
|
if (now - m_last_tracker_tick < minutes(10)) return d;
|
|
|
|
m_last_tracker_tick = now;
|
|
|
|
|
|
|
|
// look through all peers and see if any have timed out
|
2009-09-27 05:38:41 +02:00
|
|
|
for (table_t::iterator i = m_map.begin(), end(m_map.end()); i != end;)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2008-10-21 19:10:11 +02:00
|
|
|
torrent_entry& t = i->second;
|
|
|
|
node_id const& key = i->first;
|
|
|
|
++i;
|
|
|
|
purge_peers(t.peers);
|
|
|
|
|
|
|
|
// if there are no more peers, remove the entry altogether
|
|
|
|
if (t.peers.empty())
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2008-10-21 19:10:11 +02:00
|
|
|
table_t::iterator i = m_map.find(key);
|
|
|
|
if (i != m_map.end()) m_map.erase(i);
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
|
|
|
}
|
2007-01-07 14:55:27 +01:00
|
|
|
|
2006-08-01 17:27:08 +02:00
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
2008-09-20 19:42:25 +02:00
|
|
|
void node_impl::status(session_status& s)
|
|
|
|
{
|
2008-10-09 05:33:14 +02:00
|
|
|
mutex_t::scoped_lock l(m_mutex);
|
2009-01-23 11:36:07 +01:00
|
|
|
|
|
|
|
m_table.status(s);
|
|
|
|
s.dht_torrents = int(m_map.size());
|
2008-09-20 19:42:25 +02:00
|
|
|
s.active_requests.clear();
|
|
|
|
for (std::set<traversal_algorithm*>::iterator i = m_running_requests.begin()
|
|
|
|
, end(m_running_requests.end()); i != end; ++i)
|
|
|
|
{
|
|
|
|
s.active_requests.push_back(dht_lookup());
|
|
|
|
dht_lookup& l = s.active_requests.back();
|
|
|
|
(*i)->status(l);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-27 05:38:41 +02:00
|
|
|
bool node_impl::lookup_torrents(sha1_hash const& target
|
|
|
|
, entry& reply, char* tags) const
|
|
|
|
{
|
|
|
|
// if (m_ses.m_alerts.should_post<dht_find_torrents_alert>())
|
|
|
|
// m_ses.m_alerts.post_alert(dht_find_torrents_alert(info_hash));
|
|
|
|
|
|
|
|
search_table_t::const_iterator first, last;
|
2009-10-05 01:46:29 +02:00
|
|
|
first = m_search_map.lower_bound(std::make_pair(target, (sha1_hash::min)()));
|
|
|
|
last = m_search_map.upper_bound(std::make_pair(target, (sha1_hash::max)()));
|
2009-09-27 05:38:41 +02:00
|
|
|
|
|
|
|
if (first == last) return false;
|
|
|
|
|
|
|
|
std::string tags_copy(tags);
|
|
|
|
char const* in_tags[20];
|
|
|
|
int num_tags = 0;
|
|
|
|
num_tags = split_string(in_tags, 20, &tags_copy[0]);
|
|
|
|
|
|
|
|
typedef std::pair<int, search_table_t::const_iterator> sort_item;
|
|
|
|
std::vector<sort_item> result;
|
|
|
|
for (; first != last; ++first)
|
|
|
|
{
|
|
|
|
result.push_back(std::make_pair(
|
|
|
|
first->second.match(in_tags, num_tags), first));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::sort(result.begin(), result.end()
|
|
|
|
, boost::bind(&sort_item::first, _1) > boost::bind(&sort_item::first, _2));
|
|
|
|
int num = (std::min)((int)result.size(), m_settings.max_torrent_search_reply);
|
|
|
|
|
|
|
|
entry::list_type& pe = reply["values"].list();
|
|
|
|
for (int i = 0; i < num; ++i)
|
|
|
|
{
|
|
|
|
pe.push_back(entry());
|
|
|
|
entry::list_type& e = pe.back().list();
|
|
|
|
// push name
|
|
|
|
e.push_back(entry());
|
|
|
|
result[i].second->second.get_name(e.back().string());
|
|
|
|
// push tags
|
|
|
|
e.push_back(entry());
|
|
|
|
result[i].second->second.get_tags(e.back().string());
|
|
|
|
// push info-hash
|
|
|
|
e.push_back(entry());
|
|
|
|
e.back().string() = result[i].second->first.second.to_string();
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool node_impl::lookup_peers(sha1_hash const& info_hash, entry& reply) const
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2008-09-20 19:42:25 +02:00
|
|
|
if (m_ses.m_alerts.should_post<dht_get_peers_alert>())
|
2009-09-20 02:23:36 +02:00
|
|
|
m_ses.m_alerts.post_alert(dht_get_peers_alert(info_hash));
|
2008-09-20 19:42:25 +02:00
|
|
|
|
2009-09-20 02:23:36 +02:00
|
|
|
table_t::const_iterator i = m_map.find(info_hash);
|
2006-08-01 17:27:08 +02:00
|
|
|
if (i == m_map.end()) return false;
|
|
|
|
|
|
|
|
torrent_entry const& v = i->second;
|
2009-10-07 22:51:02 +02:00
|
|
|
if (v.peers.empty()) return false;
|
2006-08-01 17:27:08 +02:00
|
|
|
|
|
|
|
int num = (std::min)((int)v.peers.size(), m_settings.max_peers_reply);
|
2009-09-27 05:38:41 +02:00
|
|
|
int t = 0;
|
|
|
|
int m = 0;
|
2009-10-05 01:46:29 +02:00
|
|
|
std::set<peer_entry>::const_iterator iter = v.peers.begin();
|
2009-09-27 05:38:41 +02:00
|
|
|
entry::list_type& pe = reply["values"].list();
|
|
|
|
std::string endpoint;
|
|
|
|
|
|
|
|
while (m < num)
|
|
|
|
{
|
|
|
|
if ((std::rand() / (RAND_MAX + 1.f)) * (num - t) >= num - m)
|
|
|
|
{
|
|
|
|
++iter;
|
|
|
|
++t;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
endpoint.resize(18);
|
|
|
|
std::string::iterator out = endpoint.begin();
|
|
|
|
write_endpoint(iter->addr, out);
|
|
|
|
endpoint.resize(out - endpoint.begin());
|
|
|
|
pe.push_back(entry(endpoint));
|
|
|
|
|
|
|
|
++iter;
|
|
|
|
++t;
|
|
|
|
++m;
|
|
|
|
}
|
|
|
|
}
|
2006-08-01 17:27:08 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-09-20 02:23:36 +02:00
|
|
|
namespace
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2009-09-20 02:23:36 +02:00
|
|
|
void write_nodes_entry(entry& r, nodes_t const& nodes)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2009-09-20 02:23:36 +02:00
|
|
|
bool ipv6_nodes = false;
|
|
|
|
entry& n = r["nodes"];
|
|
|
|
std::back_insert_iterator<std::string> out(n.string());
|
|
|
|
for (nodes_t::const_iterator i = nodes.begin()
|
|
|
|
, end(nodes.end()); i != end; ++i)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2009-09-20 02:23:36 +02:00
|
|
|
if (!i->addr.is_v4())
|
2008-12-23 21:04:12 +01:00
|
|
|
{
|
2009-09-20 02:23:36 +02:00
|
|
|
ipv6_nodes = true;
|
|
|
|
continue;
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
2009-09-20 02:23:36 +02:00
|
|
|
std::copy(i->id.begin(), i->id.end(), out);
|
|
|
|
write_endpoint(udp::endpoint(i->addr, i->port), out);
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
2009-09-20 02:23:36 +02:00
|
|
|
|
|
|
|
if (ipv6_nodes)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2009-09-20 02:23:36 +02:00
|
|
|
entry& p = r["nodes2"];
|
|
|
|
std::string endpoint;
|
|
|
|
for (nodes_t::const_iterator i = nodes.begin()
|
|
|
|
, end(nodes.end()); i != end; ++i)
|
|
|
|
{
|
|
|
|
if (!i->addr.is_v6()) continue;
|
|
|
|
endpoint.resize(18 + 20);
|
|
|
|
std::string::iterator out = endpoint.begin();
|
|
|
|
std::copy(i->id.begin(), i->id.end(), out);
|
|
|
|
out += 20;
|
|
|
|
write_endpoint(udp::endpoint(i->addr, i->port), out);
|
|
|
|
endpoint.resize(out - endpoint.begin());
|
|
|
|
p.list().push_back(entry(endpoint));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-09-27 19:41:51 +02:00
|
|
|
}
|
2009-09-27 07:27:43 +02:00
|
|
|
|
2009-09-27 19:41:51 +02:00
|
|
|
// verifies that a message has all the required
|
|
|
|
// entries and returns them in ret
|
|
|
|
bool verify_message(lazy_entry const* msg, key_desc_t const desc[], lazy_entry const* ret[]
|
|
|
|
, int size , char* error, int error_size)
|
|
|
|
{
|
|
|
|
// clear the return buffer
|
|
|
|
memset(ret, 0, sizeof(ret[0]) * size);
|
2009-09-27 07:27:43 +02:00
|
|
|
|
2009-09-27 19:41:51 +02:00
|
|
|
if (msg->type() != lazy_entry::dict_t)
|
2009-09-27 07:27:43 +02:00
|
|
|
{
|
2009-09-27 19:41:51 +02:00
|
|
|
snprintf(error, error_size, "not a dictionary");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (int i = 0; i < size; ++i)
|
|
|
|
{
|
|
|
|
key_desc_t const& k = desc[i];
|
|
|
|
ret[i] = msg->dict_find(k.name);
|
|
|
|
if (ret[i] && ret[i]->type() != k.type) ret[i] = 0;
|
|
|
|
if (ret[i] == 0 && (k.flags & key_desc_t::optional) == 0)
|
2009-09-27 07:27:43 +02:00
|
|
|
{
|
2009-09-27 19:41:51 +02:00
|
|
|
// the key was not found, and it's not an optiona key
|
|
|
|
snprintf(error, error_size, "missing '%s' key", k.name);
|
2009-09-27 07:27:43 +02:00
|
|
|
return false;
|
|
|
|
}
|
2009-09-27 19:41:51 +02:00
|
|
|
|
|
|
|
if (k.size > 0
|
|
|
|
&& ret[i]
|
|
|
|
&& k.type == lazy_entry::string_t
|
|
|
|
&& ret[i]->string_length() != k.size)
|
2009-09-27 07:27:43 +02:00
|
|
|
{
|
2009-09-27 19:41:51 +02:00
|
|
|
// the string was not of the required size
|
|
|
|
ret[i] = 0;
|
|
|
|
if ((k.flags & key_desc_t::optional) == 0)
|
2009-09-27 07:27:43 +02:00
|
|
|
{
|
2009-09-27 19:41:51 +02:00
|
|
|
snprintf(error, error_size, "invalid value for '%s'", k.name);
|
2009-09-27 07:27:43 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-09-27 19:41:51 +02:00
|
|
|
return true;
|
2009-09-20 02:23:36 +02:00
|
|
|
}
|
2006-08-01 17:27:08 +02:00
|
|
|
|
2009-09-20 02:23:36 +02:00
|
|
|
void incoming_error(entry& e, char const* msg)
|
|
|
|
{
|
|
|
|
e["y"] = "e";
|
|
|
|
entry::list_type& l = e["e"].list();
|
|
|
|
l.push_back(entry(203));
|
|
|
|
l.push_back(entry(msg));
|
|
|
|
}
|
|
|
|
|
|
|
|
// build response
|
|
|
|
void node_impl::incoming_request(msg const& m, entry& e)
|
|
|
|
{
|
|
|
|
e = entry(entry::dictionary_t);
|
|
|
|
e["y"] = "r";
|
|
|
|
e["t"] = m.message.dict_find_string_value("t");
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
key_desc_t top_desc[] = {
|
|
|
|
{"q", lazy_entry::string_t, 0, 0},
|
|
|
|
{"a", lazy_entry::dict_t, 0, 0},
|
|
|
|
};
|
|
|
|
|
|
|
|
lazy_entry const* top_level[2];
|
|
|
|
char error_string[200];
|
|
|
|
if (!verify_message(&m.message, top_desc, top_level, 2, error_string, sizeof(error_string)))
|
2009-09-20 02:23:36 +02:00
|
|
|
{
|
2009-09-27 07:27:43 +02:00
|
|
|
incoming_error(e, error_string);
|
2009-09-20 02:23:36 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
char const* query = top_level[0]->string_cstr();
|
2009-09-20 02:23:36 +02:00
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
lazy_entry const* arg_ent = top_level[1];
|
2009-09-20 02:23:36 +02:00
|
|
|
|
|
|
|
lazy_entry const* node_id_ent = arg_ent->dict_find_string("id");
|
|
|
|
if (node_id_ent == 0 || node_id_ent->string_length() != 20)
|
|
|
|
{
|
|
|
|
incoming_error(e, "missing 'id' key");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
node_id id(node_id_ent->string_ptr());
|
|
|
|
|
|
|
|
m_table.heard_about(id, m.addr);
|
|
|
|
|
|
|
|
entry& reply = e["r"];
|
|
|
|
m_rpc.add_our_id(reply);
|
|
|
|
|
|
|
|
if (strcmp(query, "ping") == 0)
|
|
|
|
{
|
|
|
|
// we already have 't' and 'id' in the response
|
|
|
|
// no more left to add
|
|
|
|
}
|
|
|
|
else if (strcmp(query, "get_peers") == 0)
|
|
|
|
{
|
2009-09-27 07:27:43 +02:00
|
|
|
key_desc_t msg_desc[] = {
|
|
|
|
{"info_hash", lazy_entry::string_t, 20, 0},
|
|
|
|
};
|
|
|
|
|
|
|
|
lazy_entry const* msg_keys[1];
|
|
|
|
if (!verify_message(arg_ent, msg_desc, msg_keys, 1, error_string, sizeof(error_string)))
|
2009-09-20 02:23:36 +02:00
|
|
|
{
|
2009-09-27 07:27:43 +02:00
|
|
|
incoming_error(e, error_string);
|
2009-09-20 02:23:36 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
reply["token"] = generate_token(m.addr, msg_keys[0]->string_ptr());
|
2009-09-20 02:23:36 +02:00
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
sha1_hash info_hash(msg_keys[0]->string_ptr());
|
2009-09-20 02:23:36 +02:00
|
|
|
nodes_t n;
|
|
|
|
// always return nodes as well as peers
|
|
|
|
m_table.find_node(info_hash, n, 0);
|
|
|
|
write_nodes_entry(reply, n);
|
|
|
|
|
2009-10-07 22:51:02 +02:00
|
|
|
bool ret = lookup_peers(info_hash, reply);
|
2009-09-20 02:23:36 +02:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
2009-10-07 22:51:02 +02:00
|
|
|
if (ret) TORRENT_LOG(node) << " values: " << reply["values"].list().size();
|
2006-08-01 17:27:08 +02:00
|
|
|
#endif
|
2009-09-20 02:23:36 +02:00
|
|
|
}
|
|
|
|
else if (strcmp(query, "find_node") == 0)
|
|
|
|
{
|
2009-09-27 07:27:43 +02:00
|
|
|
key_desc_t msg_desc[] = {
|
|
|
|
{"target", lazy_entry::string_t, 20, 0},
|
|
|
|
};
|
|
|
|
|
|
|
|
lazy_entry const* msg_keys[1];
|
|
|
|
if (!verify_message(arg_ent, msg_desc, msg_keys, 1, error_string, sizeof(error_string)))
|
2009-09-20 02:23:36 +02:00
|
|
|
{
|
2009-09-27 07:27:43 +02:00
|
|
|
incoming_error(e, error_string);
|
2009-09-20 02:23:36 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
sha1_hash target(msg_keys[0]->string_ptr());
|
|
|
|
|
|
|
|
// TODO: find_node should write directly to the response entry
|
2009-09-20 02:23:36 +02:00
|
|
|
nodes_t n;
|
|
|
|
m_table.find_node(target, n, 0);
|
|
|
|
write_nodes_entry(reply, n);
|
|
|
|
}
|
|
|
|
else if (strcmp(query, "announce_peer") == 0)
|
|
|
|
{
|
2009-10-03 20:29:47 +02:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
2009-09-29 19:06:08 +02:00
|
|
|
extern int g_failed_announces;
|
2009-10-03 20:29:47 +02:00
|
|
|
#endif
|
2009-09-27 07:27:43 +02:00
|
|
|
key_desc_t msg_desc[] = {
|
|
|
|
{"info_hash", lazy_entry::string_t, 20, 0},
|
|
|
|
{"port", lazy_entry::int_t, 0, 0},
|
|
|
|
{"token", lazy_entry::string_t, 0, 0},
|
|
|
|
};
|
|
|
|
|
|
|
|
lazy_entry const* msg_keys[3];
|
|
|
|
if (!verify_message(arg_ent, msg_desc, msg_keys, 3, error_string, sizeof(error_string)))
|
2009-09-20 02:23:36 +02:00
|
|
|
{
|
2009-10-03 20:29:47 +02:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
2009-09-29 19:06:08 +02:00
|
|
|
++g_failed_announces;
|
2009-10-03 20:29:47 +02:00
|
|
|
#endif
|
2009-09-27 07:27:43 +02:00
|
|
|
incoming_error(e, error_string);
|
2009-09-20 02:23:36 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
int port = msg_keys[1]->int_value();
|
2009-09-20 02:23:36 +02:00
|
|
|
if (port < 0 || port >= 65536)
|
|
|
|
{
|
2009-10-03 20:29:47 +02:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
2009-09-29 19:06:08 +02:00
|
|
|
++g_failed_announces;
|
2009-10-03 20:29:47 +02:00
|
|
|
#endif
|
2009-09-20 02:23:36 +02:00
|
|
|
incoming_error(e, "invalid 'port' in announce");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
sha1_hash info_hash(msg_keys[0]->string_ptr());
|
2009-09-20 02:23:36 +02:00
|
|
|
|
|
|
|
if (m_ses.m_alerts.should_post<dht_announce_alert>())
|
|
|
|
m_ses.m_alerts.post_alert(dht_announce_alert(
|
|
|
|
m.addr.address(), port, info_hash));
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
if (!verify_token(msg_keys[2]->string_value(), msg_keys[0]->string_ptr(), m.addr))
|
2009-09-20 02:23:36 +02:00
|
|
|
{
|
2009-10-03 20:29:47 +02:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
2009-09-29 19:06:08 +02:00
|
|
|
++g_failed_announces;
|
2009-10-03 20:29:47 +02:00
|
|
|
#endif
|
2009-09-20 02:23:36 +02:00
|
|
|
incoming_error(e, "invalid token in announce");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// the token was correct. That means this
|
|
|
|
// node is not spoofing its address. So, let
|
|
|
|
// the table get a chance to add it.
|
|
|
|
m_table.node_seen(id, m.addr);
|
|
|
|
|
|
|
|
torrent_entry& v = m_map[info_hash];
|
|
|
|
peer_entry e;
|
|
|
|
e.addr = tcp::endpoint(m.addr.address(), port);
|
|
|
|
e.added = time_now();
|
|
|
|
std::set<peer_entry>::iterator i = v.peers.find(e);
|
|
|
|
if (i != v.peers.end()) v.peers.erase(i++);
|
|
|
|
v.peers.insert(i, e);
|
2009-10-03 20:29:47 +02:00
|
|
|
#ifdef TORRENT_DHT_VERBOSE_LOGGING
|
2009-09-29 19:06:08 +02:00
|
|
|
extern int g_announces;
|
|
|
|
++g_announces;
|
2009-10-03 20:29:47 +02:00
|
|
|
#endif
|
2009-09-20 02:23:36 +02:00
|
|
|
}
|
2009-09-27 05:38:41 +02:00
|
|
|
else if (strcmp(query, "find_torrent") == 0)
|
|
|
|
{
|
2009-09-27 07:27:43 +02:00
|
|
|
key_desc_t msg_desc[] = {
|
|
|
|
{"target", lazy_entry::string_t, 20, 0},
|
|
|
|
{"tags", lazy_entry::string_t, 0, 0},
|
|
|
|
};
|
2009-09-27 05:38:41 +02:00
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
lazy_entry const* msg_keys[2];
|
|
|
|
if (!verify_message(arg_ent, msg_desc, msg_keys, 2, error_string, sizeof(error_string)))
|
2009-09-27 05:38:41 +02:00
|
|
|
{
|
2009-09-27 07:27:43 +02:00
|
|
|
incoming_error(e, error_string);
|
2009-09-27 05:38:41 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
reply["token"] = generate_token(m.addr, msg_keys[0]->string_ptr());
|
2009-09-27 05:38:41 +02:00
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
sha1_hash target(msg_keys[0]->string_ptr());
|
2009-09-27 05:38:41 +02:00
|
|
|
nodes_t n;
|
|
|
|
// always return nodes as well as torrents
|
|
|
|
m_table.find_node(target, n, 0);
|
|
|
|
write_nodes_entry(reply, n);
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
lookup_torrents(target, reply, (char*)msg_keys[1]->string_cstr());
|
2009-09-27 05:38:41 +02:00
|
|
|
}
|
|
|
|
else if (strcmp(query, "announce_torrent") == 0)
|
|
|
|
{
|
2009-09-27 07:27:43 +02:00
|
|
|
key_desc_t msg_desc[] = {
|
|
|
|
{"target", lazy_entry::string_t, 20, 0},
|
|
|
|
{"info_hash", lazy_entry::string_t, 20, 0},
|
|
|
|
{"name", lazy_entry::string_t, 0, 0},
|
|
|
|
{"tags", lazy_entry::string_t, 0, 0},
|
|
|
|
{"token", lazy_entry::string_t, 0, 0},
|
|
|
|
};
|
|
|
|
|
2010-03-12 04:18:17 +01:00
|
|
|
lazy_entry const* msg_keys[5];
|
|
|
|
if (!verify_message(arg_ent, msg_desc, msg_keys, 5, error_string, sizeof(error_string)))
|
2009-09-27 05:38:41 +02:00
|
|
|
{
|
2009-09-27 07:27:43 +02:00
|
|
|
incoming_error(e, error_string);
|
2009-09-27 05:38:41 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if (m_ses.m_alerts.should_post<dht_announce_torrent_alert>())
|
|
|
|
// m_ses.m_alerts.post_alert(dht_announce_torrent_alert(
|
|
|
|
// m.addr.address(), name, tags, info_hash));
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
if (!verify_token(msg_keys[4]->string_value(), msg_keys[0]->string_ptr(), m.addr))
|
2009-09-27 05:38:41 +02:00
|
|
|
{
|
|
|
|
incoming_error(e, "invalid token in announce");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
sha1_hash target(msg_keys[0]->string_ptr());
|
|
|
|
sha1_hash info_hash(msg_keys[1]->string_ptr());
|
2009-09-27 05:38:41 +02:00
|
|
|
|
|
|
|
// the token was correct. That means this
|
|
|
|
// node is not spoofing its address. So, let
|
|
|
|
// the table get a chance to add it.
|
|
|
|
m_table.node_seen(id, m.addr);
|
|
|
|
|
|
|
|
search_table_t::iterator i = m_search_map.find(std::make_pair(target, info_hash));
|
|
|
|
if (i == m_search_map.end())
|
|
|
|
{
|
|
|
|
boost::tie(i, boost::tuples::ignore)
|
|
|
|
= m_search_map.insert(std::make_pair(std::make_pair(target, info_hash)
|
|
|
|
, search_torrent_entry()));
|
|
|
|
}
|
|
|
|
|
|
|
|
char const* in_tags[20];
|
|
|
|
int num_tags = 0;
|
2009-09-27 07:27:43 +02:00
|
|
|
num_tags = split_string(in_tags, 20, (char*)msg_keys[3]->string_cstr());
|
2009-09-27 05:38:41 +02:00
|
|
|
|
2009-09-27 07:27:43 +02:00
|
|
|
i->second.publish(msg_keys[2]->string_value(), in_tags, num_tags);
|
2009-09-27 05:38:41 +02:00
|
|
|
}
|
2009-09-20 02:23:36 +02:00
|
|
|
else
|
|
|
|
{
|
2009-09-27 02:40:05 +02:00
|
|
|
// if we don't recognize the message but there's a
|
|
|
|
// 'target' or 'info_hash' in the arguments, treat it
|
|
|
|
// as find_node to be future compatible
|
|
|
|
lazy_entry const* target_ent = arg_ent->dict_find_string("target");
|
|
|
|
if (target_ent == 0 || target_ent->string_length() != 20)
|
|
|
|
{
|
|
|
|
target_ent = arg_ent->dict_find_string("info_hash");
|
|
|
|
if (target_ent == 0 || target_ent->string_length() != 20)
|
|
|
|
{
|
|
|
|
incoming_error(e, "unknown message");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sha1_hash target(target_ent->string_ptr());
|
|
|
|
nodes_t n;
|
|
|
|
// always return nodes as well as peers
|
|
|
|
m_table.find_node(target, n, 0);
|
|
|
|
write_nodes_entry(reply, n);
|
2009-09-20 02:23:36 +02:00
|
|
|
return;
|
|
|
|
}
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} } // namespace libtorrent::dht
|
2009-09-20 02:23:36 +02:00
|
|
|
|