You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
ktorrent/libktorrent/kademlia/dht.cpp

379 lines
9.0 KiB

/***************************************************************************
* Copyright (C) 2005 by Joris Guisson *
* joris.guisson@gmail.com *
* *
* 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. *
***************************************************************************/
#include <tqmap.h>
#include <kresolver.h>
#include <util/log.h>
#include <util/array.h>
#include <util/functions.h>
#include <torrent/bnode.h>
#include <torrent/globals.h>
#include <ksocketaddress.h>
#include "announcetask.h"
#include "dht.h"
#include "node.h"
#include "rpcserver.h"
#include "rpcmsg.h"
#include "kclosestnodessearch.h"
#include "database.h"
#include "taskmanager.h"
#include "nodelookup.h"
using namespace bt;
using namespace KNetwork;
namespace dht
{
DHT::DHT() : node(0),srv(0),db(0),tman(0)
{
connect(&update_timer,TQT_SIGNAL(timeout()),this,TQT_SLOT(update()));
}
DHT::~DHT()
{
if (running)
stop();
}
void DHT::start(const TQString & table,const TQString & key_file,bt::Uint16 port)
{
if (running)
return;
if (port == 0)
port = 6881;
table_file = table;
this->port = port;
Out(SYS_DHT|LOG_NOTICE) << "DHT: Starting on port " << port << endl;
srv = new RPCServer(this,port);
node = new Node(srv,key_file);
db = new Database();
tman = new TaskManager();
expire_timer.update();
running = true;
srv->start();
node->loadTable(table);
update_timer.start(1000);
started();
}
void DHT::stop()
{
if (!running)
return;
update_timer.stop();
Out(SYS_DHT|LOG_NOTICE) << "DHT: Stopping " << endl;
srv->stop();
node->saveTable(table_file);
running = false;
stopped();
delete tman; tman = 0;
delete db; db = 0;
delete node; node = 0;
delete srv; srv = 0;
}
void DHT::ping(PingReq* r)
{
if (!running)
return;
// ignore requests we get from ourself
if (r->getID() == node->getOurID())
return;
Out(SYS_DHT|LOG_NOTICE) << "DHT: Sending ping response" << endl;
PingRsp rsp(r->getMTID(),node->getOurID());
rsp.setOrigin(r->getOrigin());
srv->sendMsg(&rsp);
node->recieved(this,r);
}
void DHT::findNode(FindNodeReq* r)
{
if (!running)
return;
// ignore requests we get from ourself
if (r->getID() == node->getOurID())
return;
Out(SYS_DHT|LOG_DEBUG) << "DHT: got findNode request" << endl;
node->recieved(this,r);
// find the K closest nodes and pack them
KClosestNodesSearch kns(r->getTarget(),K);
node->findKClosestNodes(kns);
Uint32 rs = kns.requiredSpace();
// create the data
TQByteArray nodes(rs);
// pack the found nodes in a byte array
if (rs > 0)
kns.pack(nodes);
FindNodeRsp fnr(r->getMTID(),node->getOurID(),nodes);
fnr.setOrigin(r->getOrigin());
srv->sendMsg(&fnr);
}
void DHT::announce(AnnounceReq* r)
{
if (!running)
return;
// ignore requests we get from ourself
if (r->getID() == node->getOurID())
return;
Out(SYS_DHT|LOG_DEBUG) << "DHT: got announce request" << endl;
node->recieved(this,r);
// first check if the token is OK
dht::Key token = r->getToken();
if (!db->checkToken(token,r->getOrigin().ipAddress().IPv4Addr(),r->getOrigin().port()))
return;
// everything OK, so store the value
Uint8 tdata[6];
bt::WriteUint32(tdata,0,r->getOrigin().ipAddress().IPv4Addr());
bt::WriteUint16(tdata,4,r->getPort());
db->store(r->getInfoHash(),DBItem(tdata));
// send a proper response to indicate everything is OK
AnnounceRsp rsp(r->getMTID(),node->getOurID());
rsp.setOrigin(r->getOrigin());
srv->sendMsg(&rsp);
}
void DHT::getPeers(GetPeersReq* r)
{
if (!running)
return;
// ignore requests we get from ourself
if (r->getID() == node->getOurID())
return;
Out(SYS_DHT|LOG_DEBUG) << "DHT: got getPeers request" << endl;
node->recieved(this,r);
DBItemList dbl;
db->sample(r->getInfoHash(),dbl,50);
// generate a token
dht::Key token = db->genToken(r->getOrigin().ipAddress().IPv4Addr(),r->getOrigin().port());
if (dbl.count() == 0)
{
// if data is null do the same as when we have a findNode request
// find the K closest nodes and pack them
KClosestNodesSearch kns(r->getInfoHash(),K);
node->findKClosestNodes(kns);
Uint32 rs = kns.requiredSpace();
// create the data
TQByteArray nodes(rs);
// pack the found nodes in a byte array
if (rs > 0)
kns.pack(nodes);
GetPeersRsp fnr(r->getMTID(),node->getOurID(),nodes,token);
fnr.setOrigin(r->getOrigin());
srv->sendMsg(&fnr);
}
else
{
// send a get peers response
GetPeersRsp fvr(r->getMTID(),node->getOurID(),dbl,token);
fvr.setOrigin(r->getOrigin());
srv->sendMsg(&fvr);
}
}
void DHT::response(MsgBase* r)
{
if (!running)
return;
node->recieved(this,r);
}
void DHT::error(ErrMsg* )
{}
void DHT::portRecieved(const TQString & ip,bt::Uint16 port)
{
if (!running)
return;
Out(SYS_DHT|LOG_DEBUG) << "Sending ping request to " << ip << ":" << port << endl;
PingReq* r = new PingReq(node->getOurID());
r->setOrigin(KInetSocketAddress(ip,port));
srv->doCall(r);
}
bool DHT::canStartTask() const
{
// we can start a task if we have less then 7 runnning and
// there are at least 16 RPC slots available
if (tman->getNumTasks() >= 7)
return false;
else if (256 - srv->getNumActiveRPCCalls() <= 16)
return false;
return true;
}
AnnounceTask* DHT::announce(const bt::SHA1Hash & info_hash,bt::Uint16 port)
{
if (!running)
return 0;
KClosestNodesSearch kns(info_hash,K);
node->findKClosestNodes(kns);
if (kns.getNumEntries() > 0)
{
Out(SYS_DHT|LOG_NOTICE) << "DHT: Doing announce " << endl;
AnnounceTask* at = new AnnounceTask(db,srv,node,info_hash,port);
at->start(kns,!canStartTask());
tman->addTask(at);
if (!db->tqcontains(info_hash))
db->insert(info_hash);
return at;
}
return 0;
}
NodeLookup* DHT::refreshBucket(const dht::Key & id,KBucket & bucket)
{
if (!running)
return 0;
KClosestNodesSearch kns(id,K);
bucket.findKClosestNodes(kns);
bucket.updateRefreshTimer();
if (kns.getNumEntries() > 0)
{
Out(SYS_DHT|LOG_DEBUG) << "DHT: refreshing bucket " << endl;
NodeLookup* nl = new NodeLookup(id,srv,node);
nl->start(kns,!canStartTask());
tman->addTask(nl);
return nl;
}
return 0;
}
NodeLookup* DHT::findNode(const dht::Key & id)
{
if (!running)
return 0;
KClosestNodesSearch kns(id,K);
node->findKClosestNodes(kns);
if (kns.getNumEntries() > 0)
{
Out(SYS_DHT|LOG_DEBUG) << "DHT: finding node " << endl;
NodeLookup* at = new NodeLookup(id,srv,node);
at->start(kns,!canStartTask());
tman->addTask(at);
return at;
}
return 0;
}
void DHT::update()
{
if (!running)
return;
if (expire_timer.getElapsedSinceUpdate() > 5*60*1000)
{
db->expire(bt::GetCurrentTime());
expire_timer.update();
}
node->refreshBuckets(this);
tman->removeFinishedTasks(this);
stats.num_tasks = tman->getNumTasks() + tman->getNumQueuedTasks();
stats.num_peers = node->getNumEntriesInRoutingTable();
}
void DHT::timeout(const MsgBase* r)
{
node->onTimeout(r);
}
void DHT::addDHTNode(const TQString & host,Uint16 hport)
{
if (!running)
return;
KResolverResults res = KResolver::resolve(host,TQString::number(hport));
if (res.count() > 0)
{
srv->ping(node->getOurID(),res.front().address());
}
}
TQMap<TQString, int> DHT::getClosestGoodNodes(int maxNodes)
{
TQMap<TQString, int> map;
if(!node)
return map;
int max = 0;
KClosestNodesSearch kns(node->getOurID(), maxNodes*2);
node->findKClosestNodes(kns);
KClosestNodesSearch::Itr it;
for(it = kns.begin(); it != kns.end(); ++it)
{
KBucketEntry e = it->second;
if(!e.isGood())
continue;
KInetSocketAddress a = e.getAddress();
map.insert(a.ipAddress().toString(), a.port());
if(++max >= maxNodes)
break;
}
return map;
}
}
#include "dht.moc"