RetroShare/libretroshare/src/pqi/pqipersongrp.cc

411 lines
8.1 KiB
C++
Raw Normal View History

/*
* libretroshare/src/pqi: pqipersongrp.cc
*
* 3P/PQI network interface for RetroShare.
*
* Copyright 2004-2008 by Robert Fernie.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License Version 2 as published by the Free Software Foundation.
*
* This library 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
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
* USA.
*
* Please report all bugs and problems to "retroshare@lunamutt.com".
*
*/
#include "pqi/pqipersongrp.h"
#include "pqi/p3connmgr.h"
#include "pqi/pqidebug.h"
#include <sstream>
const int pqipersongrpzone = 354;
// handle the tunnel services.
int pqipersongrp::tickServiceRecv()
{
RsRawItem *pqi = NULL;
int i = 0;
{
std::ostringstream out;
out << "pqipersongrp::tickTunnelServer()";
pqioutput(PQL_DEBUG_ALL, pqipersongrpzone, out.str());
}
//p3ServiceServer::tick();
while(NULL != (pqi = GetRsRawItem()))
{
++i;
pqioutput(PQL_DEBUG_BASIC, pqipersongrpzone,
"pqipersongrp::tickTunnelServer() Incoming TunnelItem");
incoming(pqi);
}
if (0 < i)
{
return 1;
}
return 0;
}
// handle the tunnel services.
int pqipersongrp::tickServiceSend()
{
RsRawItem *pqi = NULL;
int i = 0;
{
std::ostringstream out;
out << "pqipersongrp::tickServiceSend()";
pqioutput(PQL_DEBUG_ALL, pqipersongrpzone, out.str());
}
p3ServiceServer::tick();
while(NULL != (pqi = outgoing()))
{
++i;
pqioutput(PQL_DEBUG_BASIC, pqipersongrpzone,
"pqipersongrp::tickTunnelServer() OutGoing RsItem");
SendRsRawItem(pqi);
}
if (0 < i)
{
return 1;
}
return 0;
}
// init
pqipersongrp::pqipersongrp(SecurityPolicy *glob, unsigned long flags)
:pqihandler(glob), pqil(NULL), config(NULL), initFlags(flags)
{
}
int pqipersongrp::tick()
{
/* could limit the ticking of listener / tunnels to 1/sec...
* but not to important.
*/
if (pqil)
{
pqil -> tick();
}
int i = 0;
if (tickServiceSend()) i = 1;
if (pqihandler::tick()) i = 1; /* does actual Send/Recv */
if (tickServiceRecv()) i = 1;
return 1;
}
int pqipersongrp::status()
{
if (pqil)
{
pqil -> status();
}
return pqihandler::status();
}
/* Initialise pqilistener */
int pqipersongrp::init_listener()
{
/* extract our information from the p3ConnectMgr */
if (initFlags & PQIPERSON_NO_LISTENER)
{
pqil = NULL;
}
else
{
/* extract details from
*/
peerConnectState state;
mConnMgr->getOwnNetStatus(state);
pqil = createListener(state.localaddr);
}
return 1;
}
int pqipersongrp::restart_listener()
{
// stop it,
// change the address.
// restart.
if (pqil)
{
peerConnectState state;
mConnMgr->getOwnNetStatus(state);
pqil -> resetlisten();
pqil -> setListenAddr(state.localaddr);
pqil -> setuplisten();
}
return 1;
}
int pqipersongrp::setConfig(p3GeneralConfig *cfg)
{
config = cfg;
}
static const std::string pqih_ftr("PQIH_FTR");
int pqipersongrp::save_config()
{
char line[512];
sprintf(line, "%f %f %f %f", getMaxRate(true), getMaxRate(false),
getMaxIndivRate(true), getMaxIndivRate(false));
if (config)
{
config -> setSetting(pqih_ftr, std::string(line));
}
return 1;
}
int pqipersongrp::load_config()
{
std::string line;
if (config)
{
line = config -> getSetting(pqih_ftr);
}
float mri, mro, miri, miro;
if (4 == sscanf(line.c_str(), "%f %f %f %f", &mri, &mro, &miri, &miro))
{
setMaxRate(true, mri);
setMaxRate(false, mro);
setMaxIndivRate(true, miri);
setMaxIndivRate(false, miro);
}
else
{
pqioutput(PQL_DEBUG_BASIC, pqipersongrpzone,
"pqipersongrp::load_config() Loading Default Rates!");
setMaxRate(true, 50.0);
setMaxRate(false,50.0);
setMaxIndivRate(true, 50.0);
setMaxIndivRate(false, 50.0);
}
return 1;
}
void pqipersongrp::statusChange(const std::list<pqipeer> &plist)
{
/* iterate through, only worry about the friends */
std::list<pqipeer>::const_iterator it;
for(it = plist.begin(); it != plist.end(); it++)
{
if (it->state & RS_PEER_S_FRIEND)
{
/* now handle add/remove */
if ((it->actions & RS_PEER_NEW)
|| (it->actions & RS_PEER_MOVED))
{
addPeer(it->id);
}
if (it->actions & RS_PEER_CONNECT_REQ)
{
connectPeer(it->id);
}
}
else /* Not Friend */
{
if (it->actions & RS_PEER_MOVED)
{
removePeer(it->id);
}
}
}
}
int pqipersongrp::addPeer(std::string id)
{
{
std::ostringstream out;
out << "pqipersongrp::addPeer() PeerId: " << id;
pqioutput(PQL_DEBUG_BASIC, pqipersongrpzone, out.str());
}
std::cerr << " pqipersongrp::addPeer() id: " << id;
std::cerr << std::endl;
SearchModule *sm = NULL;
std::map<std::string, SearchModule *>::iterator it;
it = mods.find(id);
if (it != mods.end())
{
pqioutput(PQL_DEBUG_BASIC, pqipersongrpzone,
"pqipersongrp::addPeer() Peer already in Use!");
return -1;
}
pqiperson *pqip = createPerson(id, pqil);
// attach to pqihandler
sm = new SearchModule();
sm -> peerid = id;
sm -> pqi = pqip;
sm -> sp = secpolicy_create();
// reset it to start it working.
pqip -> reset();
return AddSearchModule(sm);
}
int pqipersongrp::removePeer(std::string id)
{
std::map<std::string, SearchModule *>::iterator it;
std::cerr << " pqipersongrp::removePeer() id: " << id;
std::cerr << std::endl;
it = mods.find(id);
if (it != mods.end())
{
SearchModule *mod = it->second;
RemoveSearchModule(mod);
secpolicy_delete(mod -> sp);
pqiperson *p = (pqiperson *) mod -> pqi;
p -> reset();
delete p;
mods.erase(it);
}
return 1;
}
int pqipersongrp::connectPeer(std::string id)
{
/* get status from p3connectMgr */
std::cerr << " pqipersongrp::connectPeer() id: " << id << " does nothing yet! ";
std::cerr << std::endl;
std::map<std::string, SearchModule *>::iterator it;
it = mods.find(id);
if (it == mods.end())
{
return 0;
}
/* get the connect attempt details from the p3connmgr... */
SearchModule *mod = it->second;
pqiperson *p = (pqiperson *) mod -> pqi;
/* get address from p3connmgr */
if (!mConnMgr)
return 0;
struct sockaddr_in addr;
uint32_t type;
mConnMgr->connectAttempt(id, addr, type);
uint32_t ptype;
if (type & RS_NET_CONN_TCP_ALL)
{
ptype = PQI_CONNECT_TCP;
}
else if (type & RS_NET_CONN_UDP_ALL)
{
ptype = PQI_CONNECT_UDP;
}
else
return 0;
p->connect(ptype, addr);
/* */
return 1;
}
bool pqipersongrp::notifyConnect(std::string id, bool success)
{
if (mConnMgr)
mConnMgr->connectResult(id, success, 0);
return (NULL != mConnMgr);
}
/******************************** DUMMY Specific features ***************************/
#include "pqi/pqibin.h"
pqilistener * pqipersongrpDummy::createListener(struct sockaddr_in laddr)
{
pqilistener *listener = new pqilistener();
return listener;
}
pqiperson * pqipersongrpDummy::createPerson(std::string id, pqilistener *listener)
{
{
std::ostringstream out;
out << "pqipersongrpDummy::createPerson() PeerId: " << id;
pqioutput(PQL_DEBUG_BASIC, pqipersongrpzone, out.str());
}
pqiperson *pqip = new pqiperson(id, this);
// TCP
NetBinDummy *d1 = new NetBinDummy(pqip, id, PQI_CONNECT_TCP);
RsSerialiser *rss = new RsSerialiser();
rss->addSerialType(new RsFileItemSerialiser());
rss->addSerialType(new RsCacheItemSerialiser());
rss->addSerialType(new RsServiceSerialiser());
pqiconnect *pqic = new pqiconnect(rss, d1);
pqip -> addChildInterface(PQI_CONNECT_TCP, pqic);
// UDP.
NetBinDummy *d2 = new NetBinDummy(pqip, id, PQI_CONNECT_UDP);
RsSerialiser *rss2 = new RsSerialiser();
rss2->addSerialType(new RsFileItemSerialiser());
rss2->addSerialType(new RsCacheItemSerialiser());
rss2->addSerialType(new RsServiceSerialiser());
pqiconnect *pqic2 = new pqiconnect(rss2, d2);
pqip -> addChildInterface(PQI_CONNECT_UDP, pqic2);
return pqip;
}
/******************************** DUMMY Specific features ***************************/