2021-10-08 19:08:23 -04:00
|
|
|
/*
|
|
|
|
* RetroShare Friend Server
|
|
|
|
* Copyright (C) 2021-2021 retroshare team <retroshare.project@gmail.com>
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* SPDX-FileCopyrightText: Retroshare Team <contact@retroshare.cc>
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-or-later
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
|
|
|
|
#include "util/rsnet.h"
|
|
|
|
#include "util/rsprint.h"
|
|
|
|
#include "util/rsdebug.h"
|
|
|
|
|
2021-10-19 17:24:50 -04:00
|
|
|
#include "pqi/pqithreadstreamer.h"
|
2021-10-24 11:41:23 -04:00
|
|
|
#include "friend_server/fsbio.h"
|
2021-10-19 17:24:50 -04:00
|
|
|
|
2021-10-08 19:08:23 -04:00
|
|
|
#include "network.h"
|
2021-10-24 14:01:19 -04:00
|
|
|
#include "friend_server/fsitem.h"
|
2021-10-08 19:08:23 -04:00
|
|
|
|
|
|
|
FsNetworkInterface::FsNetworkInterface()
|
2021-10-31 07:02:09 -04:00
|
|
|
: PQInterface(RsPeerId()),mFsNiMtx(std::string("FsNetworkInterface"))
|
2021-10-08 19:08:23 -04:00
|
|
|
{
|
2021-10-09 09:06:06 -04:00
|
|
|
RS_STACK_MUTEX(mFsNiMtx);
|
|
|
|
|
2021-10-08 19:08:23 -04:00
|
|
|
mClintListn = 0;
|
|
|
|
mClintListn = socket(AF_INET, SOCK_STREAM, 0); // creating socket
|
|
|
|
|
2021-10-20 17:06:38 -04:00
|
|
|
int flags = fcntl(mClintListn, F_GETFL);
|
|
|
|
fcntl(mClintListn, F_SETFL, flags | O_NONBLOCK);
|
|
|
|
|
|
|
|
struct sockaddr_in ipOfServer;
|
2021-10-08 19:08:23 -04:00
|
|
|
memset(&ipOfServer, '0', sizeof(ipOfServer));
|
|
|
|
|
|
|
|
ipOfServer.sin_family = AF_INET;
|
|
|
|
ipOfServer.sin_port = htons(2017); // this is the port number of running server
|
2021-10-20 17:06:38 -04:00
|
|
|
ipOfServer.sin_addr.s_addr = htonl(INADDR_ANY);
|
|
|
|
|
|
|
|
if(bind(mClintListn, (struct sockaddr*)&ipOfServer , sizeof(ipOfServer)) < 0)
|
|
|
|
{
|
|
|
|
RsErr() << "Error while binding: errno=" << errno ;
|
|
|
|
return;
|
|
|
|
}
|
2021-10-08 19:08:23 -04:00
|
|
|
|
2021-10-20 17:06:38 -04:00
|
|
|
if(listen(mClintListn , 40) < 0)
|
|
|
|
{
|
|
|
|
RsErr() << "Error while calling listen: errno=" << errno ;
|
|
|
|
return;
|
|
|
|
}
|
2021-10-08 19:08:23 -04:00
|
|
|
|
2021-10-09 09:06:06 -04:00
|
|
|
RsDbg() << "Network interface now listening for TCP on " << sockaddr_storage_tostring( *(sockaddr_storage*)&ipOfServer) ;
|
2021-10-08 19:08:23 -04:00
|
|
|
}
|
|
|
|
|
2021-10-20 17:06:38 -04:00
|
|
|
FsNetworkInterface::~FsNetworkInterface()
|
|
|
|
{
|
2021-10-31 07:02:09 -04:00
|
|
|
RS_STACK_MUTEX(mFsNiMtx);
|
2021-10-20 17:06:38 -04:00
|
|
|
for(auto& it:mConnections)
|
|
|
|
{
|
2021-10-31 07:02:09 -04:00
|
|
|
delete it.second.pqi_thread;
|
2021-10-20 17:06:38 -04:00
|
|
|
std::cerr << "Releasing socket " << it.second.socket << std::endl;
|
|
|
|
close(it.second.socket);
|
|
|
|
}
|
|
|
|
std::cerr << "Releasing listening socket " << mClintListn << std::endl;
|
|
|
|
close(mClintListn);
|
|
|
|
}
|
2021-10-19 17:24:50 -04:00
|
|
|
void FsNetworkInterface::threadTick()
|
2021-10-08 19:08:23 -04:00
|
|
|
{
|
2021-10-19 17:24:50 -04:00
|
|
|
// 1 - check for new connections
|
|
|
|
|
|
|
|
checkForNewConnections();
|
|
|
|
|
|
|
|
// 2 - tick all streamers
|
|
|
|
|
2021-11-04 15:52:38 -04:00
|
|
|
std::list<RsPeerId> to_close;
|
|
|
|
|
2021-10-31 07:02:09 -04:00
|
|
|
RS_STACK_MUTEX(mFsNiMtx);
|
2021-10-19 17:24:50 -04:00
|
|
|
for(auto& it:mConnections)
|
2021-11-04 15:52:38 -04:00
|
|
|
if(it.second.bio->isactive())
|
|
|
|
it.second.pqi_thread->tick();
|
|
|
|
else
|
|
|
|
to_close.push_back(it.first);
|
|
|
|
|
|
|
|
for(const auto& pid:to_close)
|
2021-11-06 12:49:31 -04:00
|
|
|
locked_closeConnection(pid);
|
2021-10-20 17:06:38 -04:00
|
|
|
|
2021-11-04 15:52:38 -04:00
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(200));
|
2021-10-08 19:08:23 -04:00
|
|
|
}
|
|
|
|
|
2021-10-19 17:24:50 -04:00
|
|
|
static RsPeerId makePeerId(int t)
|
|
|
|
{
|
|
|
|
unsigned char s[RsPeerId::SIZE_IN_BYTES];
|
2021-10-31 07:02:09 -04:00
|
|
|
memset(s,0,sizeof(s));
|
|
|
|
|
2021-10-19 17:24:50 -04:00
|
|
|
*reinterpret_cast<int*>(&s) = t;
|
|
|
|
return RsPeerId::fromBufferUnsafe(s);
|
|
|
|
}
|
|
|
|
bool FsNetworkInterface::checkForNewConnections()
|
|
|
|
{
|
|
|
|
// look for incoming data
|
|
|
|
|
|
|
|
struct sockaddr addr;
|
2021-10-20 17:06:38 -04:00
|
|
|
socklen_t addr_len = sizeof(sockaddr);
|
2021-10-19 17:24:50 -04:00
|
|
|
|
2021-10-20 17:06:38 -04:00
|
|
|
int clintConnt = accept(mClintListn, &addr, &addr_len); // accept is a blocking call!
|
2021-10-19 17:24:50 -04:00
|
|
|
|
|
|
|
if(clintConnt < 0)
|
|
|
|
{
|
2021-10-20 17:06:38 -04:00
|
|
|
if(errno == EWOULDBLOCK)
|
|
|
|
;//RsErr()<< "Incoming connection with nothing to read!" << std::endl;
|
|
|
|
else
|
|
|
|
RsErr()<< "Error when accepting connection." << std::endl;
|
|
|
|
|
2021-10-19 17:24:50 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
RsDbg() << "Got incoming connection from " << sockaddr_storage_tostring( *(sockaddr_storage*)&addr);
|
|
|
|
|
2021-10-20 17:06:38 -04:00
|
|
|
// Make the socket non blocking so that we can read from it and return if nothing comes
|
|
|
|
|
|
|
|
int flags = fcntl(clintConnt, F_GETFL);
|
|
|
|
fcntl(clintConnt, F_SETFL, flags | O_NONBLOCK);
|
|
|
|
|
|
|
|
// Create connection info
|
|
|
|
|
2021-10-19 17:24:50 -04:00
|
|
|
ConnectionData c;
|
|
|
|
c.socket = clintConnt;
|
|
|
|
c.client_address = addr;
|
|
|
|
|
|
|
|
RsPeerId pid = makePeerId(clintConnt);
|
|
|
|
|
2021-10-20 17:06:38 -04:00
|
|
|
// Setup a pqistreamer to deserialize whatever comes from this connection
|
|
|
|
|
2021-10-19 17:24:50 -04:00
|
|
|
RsSerialiser *rss = new RsSerialiser ;
|
|
|
|
rss->addSerialType(new FsSerializer) ;
|
|
|
|
|
|
|
|
FsBioInterface *bio = new FsBioInterface(clintConnt);
|
|
|
|
|
2021-10-31 07:02:09 -04:00
|
|
|
auto pqi = new pqithreadstreamer(this,rss, pid, bio,BIN_FLAGS_READABLE | BIN_FLAGS_WRITEABLE);
|
|
|
|
|
|
|
|
c.pqi_thread = pqi;
|
|
|
|
c.bio = bio;
|
2021-10-19 17:24:50 -04:00
|
|
|
|
|
|
|
pqi->start();
|
|
|
|
|
|
|
|
RS_STACK_MUTEX(mFsNiMtx);
|
2021-10-31 07:02:09 -04:00
|
|
|
mConnections[pid] = c;
|
2021-10-19 17:24:50 -04:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-10-31 07:02:09 -04:00
|
|
|
bool FsNetworkInterface::RecvItem(RsItem *item)
|
|
|
|
{
|
|
|
|
RS_STACK_MUTEX(mFsNiMtx);
|
|
|
|
|
|
|
|
auto it = mConnections.find(item->PeerId());
|
|
|
|
|
|
|
|
if(it == mConnections.end())
|
|
|
|
{
|
|
|
|
RsErr() << "Receiving an item for peer ID " << item->PeerId() << " but no connection is known for that peer." << std::endl;
|
|
|
|
delete item;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
it->second.incoming_items.push_back(item);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-10-19 17:24:50 -04:00
|
|
|
RsItem *FsNetworkInterface::GetItem()
|
|
|
|
{
|
|
|
|
RS_STACK_MUTEX(mFsNiMtx);
|
|
|
|
|
|
|
|
for(auto& it:mConnections)
|
|
|
|
{
|
2021-10-31 07:02:09 -04:00
|
|
|
if(!it.second.incoming_items.empty())
|
|
|
|
{
|
|
|
|
RsItem *item = it.second.incoming_items.front();
|
|
|
|
it.second.incoming_items.pop_front();
|
|
|
|
|
2021-10-19 17:24:50 -04:00
|
|
|
return item;
|
2021-10-31 07:02:09 -04:00
|
|
|
}
|
2021-10-19 17:24:50 -04:00
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-10-31 07:02:09 -04:00
|
|
|
int FsNetworkInterface::SendItem(RsItem *item)
|
|
|
|
{
|
|
|
|
RS_STACK_MUTEX(mFsNiMtx);
|
|
|
|
|
|
|
|
const auto& it = mConnections.find(item->PeerId());
|
|
|
|
|
|
|
|
if(it == mConnections.end())
|
|
|
|
{
|
|
|
|
RsErr() << "Cannot send item to peer " << item->PeerId() << ": no pending sockets available." ;
|
|
|
|
delete item;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-11-04 09:24:19 -04:00
|
|
|
uint32_t ss;
|
|
|
|
return it->second.pqi_thread->SendItem(item,ss);
|
2021-10-31 07:02:09 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void FsNetworkInterface::closeConnection(const RsPeerId& peer_id)
|
|
|
|
{
|
2021-10-31 11:46:06 -04:00
|
|
|
RS_STACK_MUTEX(mFsNiMtx);
|
|
|
|
|
2021-11-06 12:49:31 -04:00
|
|
|
locked_closeConnection(peer_id);
|
|
|
|
}
|
|
|
|
void FsNetworkInterface::locked_closeConnection(const RsPeerId& peer_id)
|
|
|
|
{
|
2021-11-04 15:52:38 -04:00
|
|
|
RsDbg() << "Closing connection to virtual peer " << peer_id ;
|
|
|
|
|
2021-10-31 07:02:09 -04:00
|
|
|
const auto& it = mConnections.find(peer_id);
|
|
|
|
|
|
|
|
if(it == mConnections.end())
|
|
|
|
{
|
2021-11-04 15:52:38 -04:00
|
|
|
RsErr() << " Cannot close connection to peer " << peer_id << ": no pending sockets available." ;
|
2021-10-31 07:02:09 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!it->second.incoming_items.empty())
|
|
|
|
{
|
2021-11-07 09:16:24 -05:00
|
|
|
RsErr() << " Trying to close an incoming connection with incoming items still pending! The items will be lost:" << std::endl;
|
2021-10-31 07:02:09 -04:00
|
|
|
|
|
|
|
for(auto& item:it->second.incoming_items)
|
2021-11-07 09:16:24 -05:00
|
|
|
{
|
|
|
|
RsErr() << *item;
|
2021-10-31 07:02:09 -04:00
|
|
|
delete item;
|
2021-11-07 09:16:24 -05:00
|
|
|
}
|
2021-10-31 07:02:09 -04:00
|
|
|
|
|
|
|
it->second.incoming_items.clear();
|
|
|
|
}
|
|
|
|
// Close the socket and delete everything.
|
|
|
|
|
2021-11-07 09:16:24 -05:00
|
|
|
close(it->second.socket);
|
2021-10-31 07:02:09 -04:00
|
|
|
it->second.pqi_thread->fullstop();
|
|
|
|
it->second.bio->close();
|
|
|
|
|
|
|
|
delete it->second.pqi_thread;
|
2021-10-31 11:46:06 -04:00
|
|
|
|
|
|
|
mConnections.erase(it);
|
2021-10-31 07:02:09 -04:00
|
|
|
}
|
|
|
|
|
2021-10-31 13:00:43 -04:00
|
|
|
void FsNetworkInterface::debugPrint()
|
|
|
|
{
|
|
|
|
RsDbg() << " " << mClintListn ; // listening socket
|
|
|
|
RsDbg() << " Connections: " << mConnections.size() ;
|
|
|
|
|
|
|
|
for(auto& it:mConnections)
|
|
|
|
RsDbg() << " " << it.first << ": from \"" << sockaddr_storage_tostring(*(sockaddr_storage*)(&it.second.client_address)) << "\", socket=" << it.second.socket ;
|
|
|
|
|
|
|
|
std::map<RsPeerId,ConnectionData> mConnections;
|
|
|
|
}
|
2021-10-31 07:02:09 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|