Compare commits

...

32 Commits

Author SHA1 Message Date
b201de3796 Merge pull request 'Channel class init' (#14) from init-channel-class into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/14
2026-05-08 15:24:58 +00:00
1c6c9be6ec Channel class init 2026-05-08 17:24:06 +02:00
d1577cdb08 Merge pull request 'User constructor with fd' (#13) from user-constructor into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/13
2026-05-08 14:40:22 +00:00
86e3034a0a User constructor with fd 2026-05-08 16:37:51 +02:00
3614237b8b Merge pull request 'Fixed syntax and operator=' (#12) from fix-user-syntax into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/12
2026-05-08 14:27:39 +00:00
ffd650da20 Fixed syntax and operator= 2026-05-08 16:27:00 +02:00
63ebb5b9ba Merge pull request 'SRC points to User/User.cpp, instead of Client/Client.cpp' (#11) from fix-makefile into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/11
2026-05-08 14:21:40 +00:00
b688d04fde SRC points to User/User.cpp, instead of Client/Client.cpp 2026-05-08 16:20:57 +02:00
cd4eb82af2 Merge pull request 'echo-server-implementation' (#10) from echo-server-implementation into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/10
2026-05-07 12:22:12 +00:00
iherman-
752eb89e72 Improved Server::run to now properly handle multiple packets 2026-05-07 14:18:52 +02:00
iherman-
4beabead01 Added a basic Server::run function which listens for one connection and sends the data back, added listen call to server constructor 2026-05-07 13:52:42 +02:00
iherman-
0c29c50e23 Created a functional basic constructor for Server class 2026-05-07 13:01:57 +02:00
3a43f9d2bd Merge pull request 'client-class-init' (#9) from client-class-init into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/9
2026-05-07 08:42:26 +00:00
5051c6cfcd Info about user added 2026-05-07 10:41:28 +02:00
f70a46dde0 Client.cpp added to makefile 2026-05-07 10:41:03 +02:00
def1787cb3 User class created 2026-05-07 10:40:05 +02:00
38823adb9d Merge pull request 'Fixed a really stupid mistake wow thats crazy' (#8) from fixing-idiot-mistake into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/8
2026-05-06 16:27:50 +00:00
iherman-
fd7a24de3d Fixed a really stupid mistake wow thats crazy 2026-05-06 18:27:10 +02:00
635335831c Merge pull request 'Removed executable and added a check to the Server class to check whether the port is in a valid range' (#7) from remove-executable into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/7
2026-05-06 16:20:25 +00:00
iherman-
cbc6c7fc41 Removed executable and added a check to the Server class to check whether the port is in a valid range 2026-05-06 18:19:17 +02:00
f2bbec366b Merge pull request 'Made a basic arg parser and shell implementation of the Server class' (#6) from simple-main into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/6
2026-05-06 16:09:48 +00:00
iherman-
b47c7dc9b8 Made a basic arg parser and shell implementation of the Server class 2026-05-06 18:07:17 +02:00
d2e02440b3 Merge pull request 'Finished makefile' (#5) from Improving-Makefile into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/5
2026-05-06 15:09:10 +00:00
Igor Herman Perik
e08ef37928 Finished makefile 2026-05-05 20:51:49 +02:00
bbeaae5923 Merge pull request 'tasks-file' (#4) from tasks-file into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/4
2026-05-05 17:54:02 +00:00
Angel Ortigosa Perez
c31c69c0fa Tasks will continue... 2026-05-05 19:52:58 +02:00
Angel Ortigosa Perez
206ff02eb4 Stages that project must follow 2026-05-05 19:50:33 +02:00
ddfe1202fd Merge pull request 'Created makefile' (#3) from Makefile-creation into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/3
2026-05-05 17:42:40 +00:00
Igor Herman Perik
77e3166f15 Created makefile 2026-05-05 19:41:11 +02:00
Angel Ortigosa Perez
c7bc6f90ca First line in italic 2026-05-05 19:40:10 +02:00
279a71aa31 Merge pull request 'Added thanks at the end' (#2) from readme-improving into main
Reviewed-on: http://gitea.hadi.es/aortigos/ft_irc/pulls/2
2026-05-05 17:37:56 +00:00
Angel Ortigosa Perez
b520ef7c34 Added thanks at the end 2026-05-05 19:36:53 +02:00
10 changed files with 549 additions and 1 deletions

55
Channel/Channel.cpp Normal file
View File

@@ -0,0 +1,55 @@
/* ************************************************************************** */
/* */
/* ::: :::::::: */
/* Channel.cpp :+: :+: :+: */
/* +:+ +:+ +:+ */
/* By: aortigos <aortigos@student.42malaga.com> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2026/05/08 16:41:40 by aortigos #+# #+# */
/* Updated: 2026/05/08 16:41:40 by aortigos ### ########.fr */
/* */
/* ************************************************************************** */
#include "Channel.hpp"
//////////////////
// Constructors //
//////////////////
Channel::Channel() { /* std::cout << "Channel default constructor called" << std::endl; */}
Channel::~Channel() { /* std::cout << "Channel destructor called" << std::endl; */}
Channel::Channel(const Channel &other)
{
*this = other;
// std::cout << "Channel copy constructor called" << std::endl;
}
Channel& Channel::operator=(const Channel &other)
{
if (this != &other)
{
name_ = other.name_;
members_ = other.members_;
operators_ = other.operators_;
}
// std::cout << "Channel copy assignment operator called" << std::endl;
return (*this);
}
// Constructor with name
Channel::Channel(std::string name) : name_(name) { /* std::cout << "Channel with name constructor called" << std::endl; */ }
// Members
void Channel::addMember(int fd) { this->members_.insert(fd); }
void Channel::removeMember(int fd) { this->members_.erase(fd); }
bool Channel::hasMember(int fd) const { return (this->members_.count(fd) > 0); }
// Operators
void Channel::addOperator(int fd) { this->operators_.insert(fd); }
void Channel::removeOperator(int fd) { this->operators_.erase(fd); }
bool Channel::hasOperator(int fd) const { return (this->operators_.count(fd) > 0); }

47
Channel/Channel.hpp Normal file
View File

@@ -0,0 +1,47 @@
/* ************************************************************************** */
/* */
/* ::: :::::::: */
/* Channel.hpp :+: :+: :+: */
/* +:+ +:+ +:+ */
/* By: aortigos <aortigos@student.42malaga.com> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2026/05/08 16:41:40 by aortigos #+# #+# */
/* Updated: 2026/05/08 16:41:40 by aortigos ### ########.fr */
/* */
/* ************************************************************************** */
#ifndef CHANNEL_HPP
# define CHANNEL_HPP
# include <iostream>
# include <set>
class Channel
{
private:
std::string name_;
std::set<int> members_;
std::set<int> operators_;
public:
Channel();
Channel(std::string name);
Channel(const Channel &other);
Channel& operator=(const Channel &other);
~Channel();
// Users
void addMember(int fd);
void removeMember(int fd);
bool hasMember(int fd) const;
// Operators
void addOperator(int fd);
void removeOperator(int fd);
bool hasOperator(int fd) const;
};
#endif

44
Makefile Normal file
View File

@@ -0,0 +1,44 @@
NAME = ircserv
SRC = main.cpp Server/Server.cpp User/User.cpp
HEADERS = Server/Server.hpp
OBJ = $(SRC:.cpp=.o)
CC = c++
CFLAGS = -Wall -Wextra -Werror -std=c++98
GREEN = \033[0;32m
RED = \033[0;31m
RESET = \033[0m
TOTAL := $(words $(SRC))
COUNT = 0
all: $(NAME)
$(NAME): $(OBJ) $(HEADERS)
@rm -f .build_start
@$(CC) $(CFLAGS) $(OBJ) -o $(NAME)
@printf "\n$(GREEN)✔ Listo (100%%)\n$(RESET)"
%.o: %.cpp
@if [ ! -f .build_start ]; then printf "$(GREEN)Compiling objects...\n$(RESET)"; touch .build_start; fi
@$(eval COUNT = $(shell echo $$(($(COUNT)+1))))
@PERCENT=$$(($(COUNT)*100/$(TOTAL))); \
BAR=$$(printf "%0.s#" $$(seq 1 $$((PERCENT/5)))); \
SPACE=$$(printf "%0.s " $$(seq 1 $$((20-PERCENT/5)))); \
printf "\r [$$BAR$$SPACE] %3d%% (%d/%d) $< " $$PERCENT $(COUNT) $(TOTAL)
@$(CC) $(CFLAGS) -c $< -o $@
clean:
@printf "$(RED)Removing objects...\n$(RESET)"
@rm -f $(OBJ)
fclean: clean
@rm -f $(NAME)
re: fclean all
.PHONY: all clean fclean re

View File

@@ -1,4 +1,4 @@
**This project has been created as part of the 42 curriculum by iherman- and aortigos** *This project has been created as part of the 42 curriculum by iherman- and aortigos*
# Description # Description
@@ -8,3 +8,5 @@ The goal of this project is to do an IRC Server
# Resources # Resources
Thanks for reading :)

99
Server/Server.cpp Normal file
View File

@@ -0,0 +1,99 @@
/* ************************************************************************** */
/* */
/* ::: :::::::: */
/* Server.cpp :+: :+: :+: */
/* +:+ +:+ +:+ */
/* By: iherman- <iherman-@student.42malaga.com +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2026/05/06 17:19:12 by iherman- #+# #+# */
/* Updated: 2026/05/07 14:08:48 by iherman- ### ########.fr */
/* */
/* ************************************************************************** */
#include "Server.hpp"
const int Server::kConnectionQueueLimit = 10;
Server::Server() :
port_(PORT_DEFAULT),
password_("")
{
// std::cout << "Hello from server" << std::endl;
}
Server::Server(int port, const std::string& password) :
port_(port),
password_(password)
{
if (port_ < 1 || port_ > 65535)
throw std::runtime_error("Invalid port");
if (password_.empty())
throw std::runtime_error("Empty password");
serverSocket_ = socket(AF_INET, SOCK_STREAM, 0);
if (serverSocket_ < 0)
throw std::runtime_error("Failed to create socket");
struct sockaddr_in addr;
std::memset(&addr, 0, sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_port = htons(port_);
addr.sin_addr.s_addr = INADDR_ANY;
if (bind(serverSocket_, (struct sockaddr*)&addr, sizeof(addr)))
throw std::runtime_error("Failed to bind");
if (listen(serverSocket_, kConnectionQueueLimit))
throw std::runtime_error("Failed to listen");
std::cout << "Server port: " << port_
<< "\nServer Password: " << password_
<< std::endl;
}
Server::~Server()
{
close(serverSocket_);
}
Server &Server::operator=(const Server& other)
{
if (this != &other)
{
port_ = other.port_;
serverSocket_ = other.serverSocket_;
password_ = other.password_;
}
return *this;
}
void Server::run()
{
const std::size_t kBufferSize = 1024;
char buffer[kBufferSize] = {0};
struct sockaddr_in client_addr;
socklen_t client_addr_size = sizeof(client_addr);
int clientSocket = accept(serverSocket_, (struct sockaddr*)&client_addr, &client_addr_size);
std::string message;
while (true)
{
int recv_amount = recv(clientSocket, buffer, kBufferSize, 0);
if (recv_amount <= 0)
break ;
message.append(buffer, recv_amount);
if (message.find('\n') != std::string::npos)
break ;
}
message = "Server received: " + message;
send(clientSocket, message.c_str(), message.size(), 0);
close(clientSocket);
}

63
Server/Server.hpp Normal file
View File

@@ -0,0 +1,63 @@
/* ************************************************************************** */
/* */
/* ::: :::::::: */
/* Server.hpp :+: :+: :+: */
/* +:+ +:+ +:+ */
/* By: iherman- <iherman-@student.42malaga.com +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2026/05/06 17:18:11 by iherman- #+# #+# */
/* Updated: 2026/05/07 13:29:00 by iherman- ### ########.fr */
/* */
/* ************************************************************************** */
#ifndef SERVER_HPP
# define SERVER_HPP
// C lib functions
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <signal.h>
#include <poll.h>
// C++ lib functions
# include <iostream>
# include <string>
# include <cstring>
# define PORT_DEFAULT 9898
class Server
{
private:
static const int kConnectionQueueLimit;
int port_;
int serverSocket_;
std::string password_;
public:
Server();
Server(int port, const std::string& password);
~Server();
Server &operator=(const Server& other);
void run();
};
#endif // SERVER_HPP

76
User/User.cpp Normal file
View File

@@ -0,0 +1,76 @@
/* ************************************************************************** */
/* */
/* ::: :::::::: */
/* User.cpp :+: :+: :+: */
/* +:+ +:+ +:+ */
/* By: aortigos <aortigos@student.42malaga.com> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2026/05/07 10:22:52 by aortigos #+# #+# */
/* Updated: 2026/05/07 10:22:52 by aortigos ### ########.fr */
/* */
/* ************************************************************************** */
#include "User.hpp"
//////////////////
// Constructors //
//////////////////
User::User() : fd(-1), nick(""), username(""), realname(""), buffer(""), authenticated(false), registered(false)
{
// std::cout << "User default constructor called" << std::endl;
}
User::User(int fd) : fd(fd), nick(""), username(""), realname(""), buffer(""), authenticated(false), registered(false)
{
// std::cout << "User with fd constructor called" << std::endl;
}
User::User(const User &other)
{
*this = other;
// std::cout << "User copy constructor called" << std::endl;
}
User& User::operator=(const User &other)
{
if (this != &other)
{
fd = other.fd;
nick = other.nick;
username = other.username;
realname = other.realname;
buffer = other.buffer;
authenticated = other.authenticated;
registered = other.registered;
}
// std::cout << "User copy assignment operator called" << std::endl;
return (*this);
}
User::~User()
{
// std::cout << "User destructor called" << std::endl;
}
// Getters
int User::getFd() const { return (this->fd); };
std::string User::getNick() const { return (this->nick); };
std::string User::getUsername() const { return (this->username); };
std::string User::getRealname() const { return (this->realname); };
std::string &User::getBuffer() { return (this->buffer); };
bool User::isAuthenticated() const { return (this->authenticated); };
bool User::isRegistered() const { return (this->registered); };
// Setters
void User::setNick(std::string nick) { this->nick = nick; }
void User::setUsername(std::string username) { this->username = username; }
void User::setRealname(std::string realname) { this->realname = realname; }
void User::appendBuffer(std::string buff) { this->buffer = buff; }
void User::clearBuffer() { this->buffer.clear(); }
void User::setAuthenticated(bool value) { this->authenticated = value; }
void User::setRegistered(bool value) { this->registered = value; }

61
User/User.hpp Normal file
View File

@@ -0,0 +1,61 @@
/* ************************************************************************** */
/* */
/* ::: :::::::: */
/* User.hpp :+: :+: :+: */
/* +:+ +:+ +:+ */
/* By: aortigos <aortigos@student.42malaga.com> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2026/05/07 10:22:52 by aortigos #+# #+# */
/* Updated: 2026/05/07 10:22:52 by aortigos ### ########.fr */
/* */
/* ************************************************************************** */
#ifndef USER_HPP
# define USER_HPP
# include <iostream>
class User
{
private:
int fd;
std::string nick;
std::string username;
std::string realname;
std::string buffer;
bool authenticated;
bool registered;
public:
User();
User(int fd);
User(const User &other);
User& operator=(const User &other);
~User();
// Getters
int getFd() const;
std::string getNick() const;
std::string getUsername() const;
std::string getRealname() const;
std::string &getBuffer();
bool isAuthenticated() const;
bool isRegistered() const;
// Setters
void setNick(std::string nick);
void setUsername(std::string username);
void setRealname(std::string realname);
void appendBuffer(std::string buff);
void clearBuffer();
void setAuthenticated(bool value);
void setRegistered(bool value);
};
#endif

49
main.cpp Normal file
View File

@@ -0,0 +1,49 @@
/* ************************************************************************** */
/* */
/* ::: :::::::: */
/* main.cpp :+: :+: :+: */
/* +:+ +:+ +:+ */
/* By: iherman- <iherman-@student.42malaga.com +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2026/05/06 17:57:53 by iherman- #+# #+# */
/* Updated: 2026/05/07 13:28:09 by iherman- ### ########.fr */
/* */
/* ************************************************************************** */
#include "Server/Server.hpp"
#include <sstream>
int get_port(const char* arg)
{
std::stringstream input(arg);
int port;
if (!(input >> port))
throw std::runtime_error("Invalid port");
if (input.peek() != EOF)
throw std::runtime_error("Malformed port");
return port;
}
int main(int argc, char *argv[])
{
if (argc != 3)
{
std::cerr << "Usage: ./ircserv <port> <password>" << std::endl;
return 1;
}
try
{
int port = get_port(argv[1]);
Server server(port, argv[2]);
server.run();
}
catch (std::exception& e)
{
std::cerr << e.what() << std::endl;
return 1;
}
}

52
tasks.md Normal file
View File

@@ -0,0 +1,52 @@
# Tasks for IRC
### Useful info
You can try connect IRC server with
```bash
nc localhost <port>
```
---
### User
Represents a connected IRC client.
| Attribute | Type | Description |
|-----------------|----------|------------------------------------------------|
| `fd` | `int` | TCP s-ocket file descriptor |
| `nick` | `string` | IRC nickname |
| `username` | `string` | Username |
| `realname` | `string` | Real name |
| `buffer` | `string` | Incoming data buffer, accumulates until `\r\n` |
| `authenticated` | `bool` | `true` after correct `PASS` |
| `registered` | `bool` | `true` after `NICK` + `USER` received |
**Registration flow:** `PASS` → authenticated → `NICK` + `USER` → registered → client ready
---
### First stage
- Server starts
- It accepts two params, ./ircserv <port> <password>
- The server accepts one user, user can send messages and we can see from server
- Only 1 poll()
### Second stage
- Server can handle multiple clients simultaneously
### Third stage
- Client has nickname and username
- Client needs password for authenticate
- Client can be regular user or operator (admin)
### Fourth stage
- Client can create/connect to channels
- Client can send message inside channel (only people inside this channel can read)
*It will continue...*