merge branch samy
This commit is contained in:
commit
15cceeecd2
7 changed files with 337 additions and 226 deletions
80
sources/core/PollManager.cpp
Normal file
80
sources/core/PollManager.cpp
Normal file
|
|
@ -0,0 +1,80 @@
|
|||
#include "PollManager.hpp"
|
||||
#include <iostream>
|
||||
#include <unistd.h>
|
||||
#include <cstring>
|
||||
#include <netinet/in.h>
|
||||
#include <sys/socket.h>
|
||||
#include <poll.h>
|
||||
#include <cstdlib>
|
||||
|
||||
PollManager::PollManager() {}
|
||||
|
||||
PollManager::~PollManager() {
|
||||
for (size_t i = 0; i < _fds.size(); ++i) {
|
||||
close(_fds[i].fd);
|
||||
}
|
||||
}
|
||||
|
||||
void PollManager::pollLoop(int server_fd) {
|
||||
|
||||
struct pollfd server_pollfd;
|
||||
server_pollfd.fd = server_fd;
|
||||
server_pollfd.events = POLLIN;
|
||||
_fds.push_back(server_pollfd);
|
||||
|
||||
std::cout << "Serveur prêt à accepter des connexions..." << std::endl;
|
||||
|
||||
while (true) {
|
||||
int poll_count = poll(&_fds[0], _fds.size(), -1);
|
||||
if (poll_count == -1) {
|
||||
std::cerr << "poll error\n" << std::endl;
|
||||
continue;
|
||||
}
|
||||
for (size_t i = 0; i < _fds.size(); ++i) {
|
||||
int fd = _fds[i].fd;
|
||||
|
||||
if ((fd == server_fd) && (_fds[i].revents & POLLIN)) {
|
||||
int client_fd = accept(server_fd, NULL, NULL);
|
||||
if (client_fd == -1) {
|
||||
std::cerr << "Error accept()" << std::endl;
|
||||
continue;
|
||||
}
|
||||
addClient(client_fd);
|
||||
}
|
||||
else if (_fds[i].revents & POLLIN) {
|
||||
char buffer[1024];
|
||||
ssize_t bytes = recv(fd, buffer, sizeof(buffer) -1, 0);
|
||||
if (bytes > 0) {
|
||||
buffer[bytes] = '\0';
|
||||
_buffers[fd] += buffer;
|
||||
std::cout << "Client " << fd << " send : " << buffer;
|
||||
} else {
|
||||
std::cout << "Client " << fd << " disconected." << std::endl;
|
||||
// removeClient(fd);
|
||||
--i;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void PollManager::addClient(int fd) {
|
||||
struct pollfd pfd;
|
||||
pfd.fd = fd;
|
||||
pfd.events = POLLIN;
|
||||
_fds.push_back(pfd);
|
||||
_buffers[fd] = "";
|
||||
std::cout << "Client connecté (fd " << fd << ")" << std::endl;
|
||||
}
|
||||
|
||||
void PollManager::removeClient(int fd) {
|
||||
for (size_t i = 0; i < _fds.size(); ++i) {
|
||||
if (_fds[i].fd == fd) {
|
||||
_fds.erase(_fds.begin() + i);
|
||||
break;
|
||||
}
|
||||
}
|
||||
_buffers.erase(fd);
|
||||
close(fd);
|
||||
std::cout << "Client retiré (fd " << fd << ")" << std::endl;
|
||||
}
|
||||
101
sources/core/Server.cpp
Normal file
101
sources/core/Server.cpp
Normal file
|
|
@ -0,0 +1,101 @@
|
|||
/* ************************************************************************** */
|
||||
/* */
|
||||
/* ::: :::::::: */
|
||||
/* Server.cpp :+: :+: :+: */
|
||||
/* +:+ +:+ +:+ */
|
||||
/* By: sben-tay <sben-tay@student.42.fr> +#+ +:+ +#+ */
|
||||
/* +#+#+#+#+#+ +#+ */
|
||||
/* Created: 2025/05/13 11:11:07 by rparodi #+# #+# */
|
||||
/* Updated: 2025/05/20 17:11:06 by sben-tay ### ########.fr */
|
||||
/* */
|
||||
/* ************************************************************************** */
|
||||
|
||||
#include "color.hpp"
|
||||
#include "server.hpp"
|
||||
#include "core.hpp"
|
||||
#include <iostream>
|
||||
#include <sys/socket.h>
|
||||
#include <netinet/in.h>
|
||||
#include <unistd.h>
|
||||
#include <cstdlib>
|
||||
#include <cstring>
|
||||
|
||||
/**
|
||||
* @brief The default constructor of the Server class.
|
||||
*/
|
||||
Server::Server() : _port(0), _password("") {
|
||||
if (DEBUG)
|
||||
std::cerr << CLR_MAGENTA << "Debug: Thanks to use the constructor with port and password !" << std::endl;
|
||||
std::cout << CLR_GREY << "Info: Server default constructor called" << CLR_RESET << std::endl;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief The constructor of the Server class.
|
||||
*
|
||||
* @param port The port given by the user (argv[1])
|
||||
* @param password The password given by the user (argv[2])
|
||||
*
|
||||
* @note Thanks to check the port / password before give them to the constructor.
|
||||
*/
|
||||
Server::Server(int port, const std::string &password) : _port(port), _password(password) {
|
||||
std::cout << CLR_GREY << "Info: Server constructor called" << CLR_RESET << std::endl;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief The default destructor of the Server class.
|
||||
*/
|
||||
Server::~Server() {
|
||||
std::cout << CLR_GREY << "Info: Server destructor called" << CLR_RESET << std::endl;
|
||||
}
|
||||
|
||||
|
||||
void Server::start() {
|
||||
_serverFd = socket(AF_INET, SOCK_STREAM, 0);
|
||||
if (_serverFd == -1) {
|
||||
std::cerr << "Erreur socket" << std::endl;
|
||||
return;
|
||||
}
|
||||
|
||||
sockaddr_in addr;
|
||||
std::memset(&addr, 0, sizeof(addr));
|
||||
addr.sin_family = AF_INET;
|
||||
addr.sin_addr.s_addr = INADDR_ANY;
|
||||
addr.sin_port = htons(_port);
|
||||
|
||||
if (bind(_serverFd, (sockaddr*)&addr, sizeof(addr)) == -1 ||
|
||||
listen(_serverFd, 10) == -1) {
|
||||
std::cerr << "Erreur bind/listen" << std::endl;
|
||||
close(_serverFd);
|
||||
return;
|
||||
}
|
||||
|
||||
std::cout << "Serveur lancé sur le port " << _port << std::endl;
|
||||
|
||||
_poll.pollLoop(_serverFd);
|
||||
close(_serverFd);
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Show the server settings.
|
||||
*
|
||||
* @note This function is used to show the server settings.
|
||||
* It is used for debug purpose.
|
||||
*/
|
||||
|
||||
void Server::showInfo() const {
|
||||
std::cout << std::endl;
|
||||
std::cout << CLR_BLUE << "IRCSettings:" << CLR_RESET << std::endl;
|
||||
std::cout << CLR_BLUE << "\t- Port:\t\t" << CLR_GOLD << this->_port << CLR_RESET << std::endl;
|
||||
std::cout << CLR_BLUE << "\t- Password:\t" << CLR_GOLD << this->_password << CLR_RESET << std::endl;
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief The getter of the port.
|
||||
*
|
||||
* @return the port of the server
|
||||
*/
|
||||
unsigned short int Server::getPort() const {
|
||||
return this->_port;
|
||||
}
|
||||
|
||||
|
|
@ -1,144 +0,0 @@
|
|||
/******************************************************************************/
|
||||
/* */
|
||||
/* ::: :::::::: */
|
||||
/* server.cpp :+: :+: :+: */
|
||||
/* +:+ +:+ +:+ */
|
||||
/* By: omoudni <omoudni@student.42paris.fr> +#+ +:+ +#+ */
|
||||
/* +#+#+#+#+#+ +#+ */
|
||||
/* Created: 2025/05/13 11:11:07 by rparodi #+# #+# */
|
||||
/* Updated: 2025/05/19 20:16:00 by omoudni ### ########.fr */
|
||||
/* */
|
||||
/******************************************************************************/
|
||||
|
||||
#include "color.hpp"
|
||||
#include "server.hpp"
|
||||
#include "core.hpp"
|
||||
#include <iostream>
|
||||
#include <sys/socket.h>
|
||||
#include <netinet/in.h>
|
||||
#include <unistd.h>
|
||||
/**
|
||||
* @brief The default constructor of the Server class.
|
||||
*/
|
||||
Server::Server() : _port(0), _password("") {
|
||||
if (DEBUG)
|
||||
std::cerr << CLR_MAGENTA << "Debug: Thanks to use the constructor with port and password !" << std::endl;
|
||||
std::cout << CLR_GREY << "Info: Server default constructor called" << CLR_RESET << std::endl;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief The constructor of the Server class.
|
||||
*
|
||||
* @param port The port given by the user (argv[1])
|
||||
* @param password The password given by the user (argv[2])
|
||||
*
|
||||
* @note Thanks to check the port / password before give them to the constructor.
|
||||
*/
|
||||
Server::Server(int port, const std::string &password) : _port(port), _password(password) {
|
||||
std::cout << CLR_GREY << "Info: Server constructor called" << CLR_RESET << std::endl;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief The default destructor of the Server class.
|
||||
*/
|
||||
Server::~Server() {
|
||||
std::cout << CLR_GREY << "Info: Server destructor called" << CLR_RESET << std::endl;
|
||||
}
|
||||
|
||||
|
||||
void Server::start() {
|
||||
int fd = socket(AF_INET, SOCK_STREAM, 0);
|
||||
if (server_fd == -1) {
|
||||
std::cerr << CLR_RED << "Error: Failed to create socket" << CLR_RESET << std::endl;
|
||||
return;
|
||||
}
|
||||
setServerFd(fd);
|
||||
struct sockaddr_in server_addr;
|
||||
server_addr.sin_family = AF_INET;
|
||||
server_addr.sin_addr.s_addr = INADDR_ANY;
|
||||
server_addr.sin_port = htons(this->_port);
|
||||
if (bind(server_fd, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1) {
|
||||
std::cerr << CLR_RED << "Error: Failed to bind socket" << CLR_RESET << std::endl;
|
||||
close(server_fd);
|
||||
return;
|
||||
}
|
||||
if (listen(server_fd, 5) == -1) {
|
||||
std::cerr << CLR_RED << "Error: Failed to listen on socket" << CLR_RESET << std::endl;
|
||||
close(server_fd);
|
||||
return;
|
||||
}
|
||||
_pollManager.addFd(server_fd, POLLIN);
|
||||
std::cout << CLR_GREEN << "Server started on port " << this->_port << CLR_RESET << std::endl;
|
||||
std::cout << CLR_GREEN << "Waiting for clients..." << CLR_RESET << std::endl;
|
||||
while (true) {
|
||||
int n = _pollManager.pollEvents();
|
||||
if (n < 0) {
|
||||
std::cerr << CLR_RED << "Poll error" << CLR_RESET << std::endl;
|
||||
break;
|
||||
}
|
||||
const std::vector<struct pollfd>& fds = _pollManager.getFds();
|
||||
for (size_t i = 0; i < fds.size(); ++i) {
|
||||
if (fds[i].revents & POLLIN) {
|
||||
if (fds[i].fd == server_fd) {
|
||||
// Accept new client
|
||||
int client_fd = accept(server_fd, NULL, NULL);
|
||||
if (client_fd != -1) {
|
||||
_pollManager.addFd(client_fd, POLLIN);
|
||||
std::cout << CLR_GREEN << "Client connected" << CLR_RESET << std::endl;
|
||||
std::cout << "Welcome to IRC. Your client_id is: " << client_fd << std::endl;
|
||||
} else {
|
||||
std::cerr << CLR_RED << "Error: Failed to accept client" << CLR_RESET << std::endl;
|
||||
|
||||
}
|
||||
} else {
|
||||
// Handle client communication
|
||||
char buffer[1024];
|
||||
ssize_t bytes_received = recv(fds[i].fd, buffer, sizeof(buffer) - 1, 0);
|
||||
if (bytes_received <= 0) {
|
||||
// Client disconnected
|
||||
// if it's negative, it means an error occurred, maybe print it with perror?
|
||||
std::cout << CLR_RED << "Client disconnected" << CLR_RESET << std::endl;
|
||||
close(fds[i].fd);
|
||||
_pollManager.removeFd(fds[i].fd);
|
||||
} else {
|
||||
buffer[bytes_received] = '\0';
|
||||
std::cout << "Received: " << buffer << std::endl;
|
||||
// Optionally: send response to client here
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
close(server_fd);
|
||||
std::cout << CLR_GREEN << "Server stopped" << CLR_RESET << std::endl;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Show the server settings.
|
||||
*
|
||||
* @note This function is used to show the server settings.
|
||||
* It is used for debug purpose.
|
||||
*/
|
||||
|
||||
void Server::showInfo() const {
|
||||
std::cout << std::endl;
|
||||
std::cout << CLR_BLUE << "IRCSettings:" << CLR_RESET << std::endl;
|
||||
std::cout << CLR_BLUE << "\t- Port:\t\t" << CLR_GOLD << this->_port << CLR_RESET << std::endl;
|
||||
std::cout << CLR_BLUE << "\t- Password:\t" << CLR_GOLD << this->_password << CLR_RESET << std::endl;
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief The getter of the port.
|
||||
*
|
||||
* @return the port of the server
|
||||
*/
|
||||
unsigned short int Server::getPort() const {
|
||||
return this->_port;
|
||||
}
|
||||
|
||||
void Server::setServerFd(int fd) {
|
||||
this->server_fd = fd;
|
||||
}
|
||||
|
||||
|
||||
Loading…
Add table
Add a link
Reference in a new issue