/* * 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 "util/rsdebug.h" #include const int pqipersongrpzone = 354; /**** *#define PGRP_DEBUG 1 ****/ // 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; #ifdef PGRP_DEBUG std::cerr << "pqipersongrp::tick() moreToTick from tickServiceSend()" << std::endl; #endif } if (pqihandler::tick()) /* does actual Send/Recv */ { i = 1; #ifdef PGRP_DEBUG std::cerr << "pqipersongrp::tick() moreToTick from pqihandler::tick()" << std::endl; #endif } if (tickServiceRecv()) { i = 1; #ifdef PGRP_DEBUG std::cerr << "pqipersongrp::tick() moreToTick from tickServiceRecv()" << std::endl; #endif } return i; } 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; return 1; } 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, 500.0); setMaxRate(false, 500.0); setMaxIndivRate(true, 100.0); setMaxIndivRate(false, 100.0); } return 1; } void pqipersongrp::statusChange(const std::list &plist) { /* iterate through, only worry about the friends */ std::list::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()); } #ifdef PGRP_DEBUG std::cerr << " pqipersongrp::addPeer() id: " << id; std::cerr << std::endl; #endif SearchModule *sm = NULL; std::map::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(); pqip -> listen(); return AddSearchModule(sm); } int pqipersongrp::removePeer(std::string id) { std::map::iterator it; #ifdef PGRP_DEBUG std::cerr << " pqipersongrp::removePeer() id: " << id; std::cerr << std::endl; #endif it = mods.find(id); if (it != mods.end()) { SearchModule *mod = it->second; // Don't duplicate remove!!! //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 */ #ifdef PGRP_DEBUG std::cerr << " pqipersongrp::connectPeer() id: " << id << " does nothing yet! "; std::cerr << std::endl; #endif std::map::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 delay; uint32_t period; uint32_t timeout; uint32_t type; if (!mConnMgr->connectAttempt(id, addr, delay, period, type)) { #ifdef PGRP_DEBUG std::cerr << " pqipersongrp::connectPeer() No Net Address"; std::cerr << std::endl; #endif return 0; } #ifdef PGRP_DEBUG std::cerr << " pqipersongrp::connectPeer() connectAttempt data id: " << id; std::cerr << " addr: " << inet_ntoa(addr.sin_addr) << ":" << ntohs(addr.sin_port); std::cerr << " delay: " << delay; std::cerr << " period: " << period; std::cerr << " type: " << type; std::cerr << std::endl; #endif uint32_t ptype; if (type & RS_NET_CONN_TCP_ALL) { ptype = PQI_CONNECT_TCP; timeout = RS_TCP_STD_TIMEOUT_PERIOD; #ifdef PGRP_DEBUG std::cerr << " pqipersongrp::connectPeer() connecting with TCP: Timeout :" << timeout; std::cerr << std::endl; #endif } else if (type & RS_NET_CONN_UDP_ALL) { ptype = PQI_CONNECT_UDP; timeout = period * 2; #ifdef PGRP_DEBUG std::cerr << " pqipersongrp::connectPeer() connecting with UDP: Timeout :" << timeout; std::cerr << std::endl; #endif } else return 0; p->connect(ptype, addr, delay, period, timeout); /* */ return 1; } bool pqipersongrp::notifyConnect(std::string id, uint32_t ptype, bool success) { uint32_t type = 0; if (ptype == PQI_CONNECT_TCP) { type = RS_NET_CONN_TCP_ALL; } else { type = RS_NET_CONN_UDP_ALL; } if (mConnMgr) mConnMgr->connectResult(id, success, type); 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 ***************************/