/* * RetroShare Friend Server * Copyright (C) 2021-2021 retroshare team * * 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 . * * SPDX-FileCopyrightText: Retroshare Team * SPDX-License-Identifier: AGPL-3.0-or-later */ #include #include #include #include #include #include #include #include #include #include #include "util/rsnet.h" #include "util/rsprint.h" #include "util/rsdebug.h" #include "pqi/pqithreadstreamer.h" #include "pqi/pqifdbin.h" #include "network.h" #include "friend_server/fsitem.h" FsNetworkInterface::FsNetworkInterface(const std::string& listening_address,uint16_t listening_port) : PQInterface(RsPeerId()),mFsNiMtx(std::string("FsNetworkInterface")),mListeningAddress(listening_address),mListeningPort(listening_port) { RS_STACK_MUTEX(mFsNiMtx); mClintListn = 0; mClintListn = socket(AF_INET, SOCK_STREAM, 0); // creating socket int flags = fcntl(mClintListn, F_GETFL); fcntl(mClintListn, F_SETFL, flags | O_NONBLOCK); struct sockaddr_in ipOfServer; memset(&ipOfServer, '0', sizeof(ipOfServer)); assert(mListeningPort > 1024); ipOfServer.sin_family = AF_INET; ipOfServer.sin_port = htons(mListeningPort); // this is the port number of running server int addr[4]; if(sscanf(listening_address.c_str(),"%d.%d.%d.%d",&addr[0],&addr[1],&addr[2],&addr[3]) != 4) throw std::runtime_error("Cannot parse a proper IPv4 address in \""+listening_address+"\""); for(int i=0;i<4;++i) if(addr[i] < 0 || addr[i] > 255) throw std::runtime_error("Cannot parse a proper IPv4 address in \""+listening_address+"\""); ipOfServer.sin_addr.s_addr = htonl( (addr[0] << 24) + (addr[1] << 16) + (addr[2] << 8) + addr[3] ); if(bind(mClintListn, (struct sockaddr*)&ipOfServer , sizeof(ipOfServer)) < 0) { RsErr() << "Error while binding: errno=" << errno ; return; } if(listen(mClintListn , 40) < 0) { RsErr() << "Error while calling listen: errno=" << errno ; return; } RsDbg() << "Network interface now listening for TCP on " << sockaddr_storage_tostring( *(sockaddr_storage*)&ipOfServer) ; } FsNetworkInterface::~FsNetworkInterface() { RS_STACK_MUTEX(mFsNiMtx); for(auto& it:mConnections) { delete it.second.pqi_thread; std::cerr << "Releasing socket " << it.second.socket << std::endl; close(it.second.socket); } std::cerr << "Releasing listening socket " << mClintListn << std::endl; close(mClintListn); } void FsNetworkInterface::threadTick() { // 1 - check for new connections checkForNewConnections(); // 2 - tick all streamers std::list to_close; RS_STACK_MUTEX(mFsNiMtx); for(auto& it:mConnections) if(it.second.bio->isactive()) it.second.pqi_thread->tick(); else to_close.push_back(it.first); for(const auto& pid:to_close) locked_closeConnection(pid); std::this_thread::sleep_for(std::chrono::milliseconds(200)); } static RsPeerId makePeerId(int t) { unsigned char s[RsPeerId::SIZE_IN_BYTES]; memset(s,0,sizeof(s)); *reinterpret_cast(&s) = t; return RsPeerId::fromBufferUnsafe(s); } bool FsNetworkInterface::checkForNewConnections() { // look for incoming data struct sockaddr addr; socklen_t addr_len = sizeof(sockaddr); int clintConnt = accept(mClintListn, &addr, &addr_len); // accept is a blocking call! if(clintConnt < 0) { if(errno == EWOULDBLOCK) ;//RsErr()<< "Incoming connection with nothing to read!" << std::endl; else RsErr()<< "Error when accepting connection." << std::endl; return false; } RsDbg() << "Got incoming connection from " << sockaddr_storage_tostring( *(sockaddr_storage*)&addr); // 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 ConnectionData c; c.socket = clintConnt; c.client_address = addr; RsPeerId pid = makePeerId(clintConnt); // Setup a pqistreamer to deserialize whatever comes from this connection RsSerialiser *rss = new RsSerialiser ; rss->addSerialType(new FsSerializer) ; RsFdBinInterface *bio = new RsFdBinInterface(clintConnt,true); auto pqi = new pqithreadstreamer(this,rss, pid, bio,BIN_FLAGS_READABLE | BIN_FLAGS_WRITEABLE); c.pqi_thread = pqi; c.bio = bio; pqi->start(); RS_STACK_MUTEX(mFsNiMtx); mConnections[pid] = c; return true; } 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; } RsItem *FsNetworkInterface::GetItem() { RS_STACK_MUTEX(mFsNiMtx); for(auto& it:mConnections) { if(!it.second.incoming_items.empty()) { RsItem *item = it.second.incoming_items.front(); it.second.incoming_items.pop_front(); return item; } } return nullptr; } 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; } uint32_t ss; return it->second.pqi_thread->SendItem(item,ss); } void FsNetworkInterface::closeConnection(const RsPeerId& peer_id) { RS_STACK_MUTEX(mFsNiMtx); locked_closeConnection(peer_id); } void FsNetworkInterface::locked_closeConnection(const RsPeerId& peer_id) { RsDbg() << "Closing connection to virtual peer " << peer_id ; const auto& it = mConnections.find(peer_id); if(it == mConnections.end()) { RsErr() << " Cannot close connection to peer " << peer_id << ": no pending sockets available." ; return; } if(!it->second.incoming_items.empty()) { RsErr() << " Trying to close an incoming connection with incoming items still pending! The items will be lost:" << std::endl; for(auto& item:it->second.incoming_items) { RsErr() << *item; delete item; } it->second.incoming_items.clear(); } // Close the socket and delete everything. close(it->second.socket); it->second.pqi_thread->fullstop(); it->second.bio->close(); delete it->second.pqi_thread; mConnections.erase(it); } 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 mConnections; }