clusterizer : server and client can now communicate

This commit is contained in:
2025-02-22 02:07:01 +01:00
parent 960c4eb99c
commit 05ba0447d2
6 changed files with 238 additions and 24 deletions

View File

@ -6,13 +6,13 @@
/* By: tomoron <tomoron@student.42angouleme.fr> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2025/02/20 18:24:39 by tomoron #+# #+# */
/* Updated: 2025/02/20 22:42:42 by tomoron ### ########.fr */
/* Updated: 2025/02/21 21:45:24 by tomoron ### ########.fr */
/* */
/* ************************************************************************** */
#include "RT.hpp"
Clusterizer::Clusterizer(Arguments args)
Clusterizer::Clusterizer(Arguments &args)
{
_isActive = 1;
_isServer = 0;
@ -27,8 +27,7 @@ Clusterizer::Clusterizer(Arguments args)
else if(args.getBoolean("client"))
{
_isServer = 0;
_serverIp = *args.getString("client");
initClient();
initClient(*args.getString("client"));
}
else
_isActive = 0;

View File

@ -6,18 +6,109 @@
/* By: tomoron <tomoron@student.42angouleme.fr> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2025/02/20 21:08:38 by tomoron #+# #+# */
/* Updated: 2025/02/20 21:21:45 by tomoron ### ########.fr */
/* Updated: 2025/02/22 02:02:32 by tomoron ### ########.fr */
/* */
/* ************************************************************************** */
#include "RT.hpp"
void Clusterizer::initClient(void)
void Clusterizer::initClient(std::string &dest)
{
_serverFd = 0;
if(dest.find(":") == std::string::npos)
std::cerr << "Client Initialisation error : invalid ip provided format must be <ip>:<port>" << std::endl;
_serverIp = dest.substr(0, dest.find(":"));
_serverPort = stoi(dest.substr(dest.find(":") + 1));
try
{
openClientConnection(_serverIp.c_str(), _serverPort);
}
catch(std::exception &e)
{
if(_error)
std::cerr << "\033[31mClient initialisation error : " << e.what() << std::endl;
}
}
void Clusterizer::openClientConnection(const char *ip, int port)
{
struct sockaddr_in serv_addr;
uint8_t sendBuffer;
if(port > 65535 || port < 0)
{
_error = 1;
throw std::runtime_error("invalid port provided");
}
_serverFd = socket(AF_INET, SOCK_STREAM, 0);
if (_serverFd < 0)
{
_error = 1;
throw std::runtime_error("can't create socket");
}
bzero(&serv_addr, sizeof(struct sockaddr_in));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = inet_addr(ip);
serv_addr.sin_port = htons(port);
if(connect(_serverFd, (struct sockaddr *)&serv_addr, sizeof(struct sockaddr_in)))
{
close(_serverFd);
_serverFd = 0;
}
sendBuffer = RDY;
(void)write(_serverFd, &sendBuffer, 1);
}
void Clusterizer::clientGetJob(std::vector<uint8_t> &sendBuf)
{
if(_receiveBuffer.size() < sizeof(t_job) + 1)
return ;
_currentJob = *(t_job *)(_receiveBuffer.data() + 1);
_receiveBuffer.erase(_receiveBuffer.begin(), _receiveBuffer.begin() + sizeof(t_job) + 1);
sendBuf.push_back(ACK);
}
void Clusterizer::clientHandleBuffer(void)
{
std::vector<uint8_t> sendBuf;
if(_receiveBuffer[0] == JOB)
clientGetJob(sendBuf);
if(sendBuf.size())
(void)write(1, sendBuf.data(), sendBuf.size());
}
void Clusterizer::clientReceive(void)
{
uint8_t buffer[512];
size_t ret;
ret = recv(_serverFd, buffer, 512, MSG_DONTWAIT);
if(ret == (size_t)-1)
return;
if(!ret)
{
close(_serverFd);
_serverFd = 0;
return ;
}
_receiveBuffer.insert(_receiveBuffer.end(), buffer, buffer + ret);
}
void Clusterizer::updateClient(void)
{
if(!_serverFd)
{
std::cout << "server isn't connected, waiting for connection" << std::endl;
while(!_serverFd)
openClientConnection(_serverIp.c_str(), _serverPort);
std::cout << "server reconnected." << std::endl;
}
clientReceive();
if(_receiveBuffer.size())
clientHandleBuffer();
}

View File

@ -6,7 +6,7 @@
/* By: tomoron <tomoron@student.42angouleme.fr> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2025/02/20 21:08:38 by tomoron #+# #+# */
/* Updated: 2025/02/20 22:43:00 by tomoron ### ########.fr */
/* Updated: 2025/02/21 22:19:20 by tomoron ### ########.fr */
/* */
/* ************************************************************************** */
@ -14,9 +14,12 @@
void Clusterizer::initServer(std::string port)
{
_pollfds = 0;
try
{
initServerSocket(stoi(port));
std::cout << "server initialized with socket " << _serverSocket << std::endl;
}
catch(std::exception &e)
{
@ -25,24 +28,107 @@ void Clusterizer::initServer(std::string port)
}
}
void Clusterizer::initServerSocket(uint16_t port)
void Clusterizer::initServerSocket(int port)
{
struct sockaddr_in s_addr;
if(port > 65535 || port < 0)
throw std::runtime_error("invalid port");
_serverSocket = socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK, 0);
if (_serverSocket < 0)
throw std::runtime_error("can't create socket");
s_addr.sin_family = AF_INET;
s_addr.sin_addr.s_addr = INADDR_ANY;
s_addr.sin_port = port >> 8 | port << 8;
s_addr.sin_port = htons(port);
if (bind(_serverSocket, (struct sockaddr *)&s_addr, \
sizeof(struct sockaddr_in)) < 0)
throw std::runtime_error("can't bind socket");
if (::listen(_serverSocket, 50) < 0)
throw std::runtime_error("can't listen on socket");
if(_serverSocket == -1)
throw std::runtime_error("unknown error");
}
void Clusterizer::updatePollfds(void)
{
if(_pollfds)
delete[] _pollfds;
_pollfds = new struct pollfd[_clients.size()];
for(auto it = _clients.begin(); it != _clients.end(); it++)
{
_pollfds[std::distance(it, _clients.begin())].fd = it->first;
_pollfds[std::distance(it, _clients.begin())].events = POLLIN;
_pollfds[std::distance(it, _clients.begin())].revents = 0;
}
}
void Clusterizer::acceptClients(void)
{
int fd;
fd = accept(_serverSocket, 0, 0);
if (fd != -1) {
std::cout << "new client :" << fd << std::endl;
_clients[fd].ready = 0;
_clients[fd].curJob = 0;
updatePollfds();
}
}
void Clusterizer::deleteClient(int fd)
{
std::map<int, t_client>::iterator it;
std::cout << "client disconnected" << std::endl;
it = _clients.find(fd);
if(it == _clients.end())
return;
_clients.erase(it);
updatePollfds();
close(fd);
}
void Clusterizer::handleBuffer(int fd, std::vector<uint8_t> &buf)
{
std::vector<uint8_t> sendBuffer;
if(buf[0] == RDY)
{
_clients[fd].ready = 1;
sendBuffer.push_back(ACK);
}
if(sendBuffer.size())
(void)write(fd, sendBuffer.data(), sendBuffer.size());
}
int Clusterizer::updateBuffer(int fd)
{
uint8_t buffer[512];
size_t ret;
ret = recv(fd, buffer, 512, 0);
if(!ret)
return(1);
_clients[fd].buffer.insert(_clients[fd].buffer.end(), buffer, buffer + ret);
handleBuffer(fd, _clients[fd].buffer);
return(0);
}
void Clusterizer::updateServer(void)
{
int recv;
acceptClients();
recv = poll(_pollfds, _clients.size(), 1);
if(!recv)
return ;
for(auto it = _clients.begin(); it != _clients.end(); it++)
{
if(_pollfds[std::distance(it, _clients.begin())].revents & POLLIN)
{
if (updateBuffer(it->first))
deleteClient(it->first);
}
}
}