Initial commit of XnoIRCd codebase

This commit is contained in:
Xnoe 2021-06-19 14:48:55 +01:00
commit 033456e949
Signed by: xnoe
GPG Key ID: 45AC398F44F0DAFE
17 changed files with 874 additions and 0 deletions

2
.gitignore vendored Normal file
View File

@ -0,0 +1,2 @@
*.o
xnoircd

28
Makefile Normal file
View File

@ -0,0 +1,28 @@
FLAGS = -pthread
xnoircd: user.o nameable.o channel.o connection.o ircserver.o irccommand.o userconnection.o main.o
g++ $(FLAGS) user.o nameable.o channel.o connection.o ircserver.o irccommand.o userconnection.o main.o -o $@
user.o: user.cpp user.h
g++ -g -c $*.cpp
connection.o: connection.cpp connection.h
g++ -g -c $*.cpp
ircserver.o: ircserver.cpp ircserver.h
g++ -g -c $*.cpp
irccommand.o: irccommand.cpp irccommand.h
g++ -g -c $*.cpp
userconnection.o: userconnection.cpp userconnection.h
g++ -g -c $*.cpp
nameable.o: nameable.cpp nameable.h
g++ -g -c $*.cpp
channel.o: channel.cpp channel.h
g++ -g -c $*.cpp
main.o: main.cpp
g++ -g -c $*.cpp

28
channel.cpp Normal file
View File

@ -0,0 +1,28 @@
#include "channel.h"
Channel::Channel(IRCServer* server, std::string channel_name): Nameable(server) {
this->channel_name = channel_name;
server->register_addressable_name(channel_name, this);
}
Channel::~Channel() {
remove_addressable_name(channel_name);
}
void Channel::send_direct(std::string message) {
send_relay(message);
}
std::string Channel::get_topic() {
return topic;
}
void Channel::set_topic(std::string topic) {
this->topic = topic;
}
bool Channel::has_topic() {
return (this->topic!="");
}
NameableType Channel::what_are_you() {
return NT_Channel;
}

26
channel.h Normal file
View File

@ -0,0 +1,26 @@
#ifndef CHANNEL_H
#define CHANNEL_H
#include <string>
#include "nameable.h"
#include "ircserver.h"
class Channel: public Nameable {
private:
std::string channel_name;
std::string topic;
public:
Channel(IRCServer* server, std::string channel_name);
~Channel();
void send_direct(std::string message);
std::string get_topic();
bool has_topic();
void set_topic(std::string topic);
NameableType what_are_you();
};
#endif

42
connection.cpp Normal file
View File

@ -0,0 +1,42 @@
#include "connection.h"
#include <iostream>
Connection::Connection(int fd) {
this->fd = fd;
std::thread(&Connection::conn_hdlr, this).detach();
}
Connection::~Connection() {
close(this->fd);
}
void Connection::conn_hdlr() {
buffer = new char[512];
while (alive) {
if (read(this->fd, buffer, 511) == 0) {
destroy();
return;
}
std::string message(buffer);
int index;
if ((index = message.find_first_of("\r\n")) == std::string::npos)
continue;
else
message = message.substr(0, index);
process_cmd(message);
}
}
void Connection::process_cmd(std::string command) {}
void Connection::destroy() {}
void Connection::kill() {
this->alive = false;
}
void Connection::send(std::string to_write) {
to_write = to_write + "\r\n";
const char* buffer = to_write.c_str();
int len = strlen(buffer);
write(this->fd, buffer, len);
}

29
connection.h Normal file
View File

@ -0,0 +1,29 @@
#ifndef CONNECTION_H
#define CONNECTION_H
#include <string>
#include <unistd.h>
#include <thread>
#include <string.h>
class Connection {
private:
int fd;
int alive = true;
char* buffer;
public:
Connection(int fd);
~Connection();
void conn_hdlr();
virtual void process_cmd(std::string command);
virtual void destroy();
void kill();
void send(std::string to_write);
};
#endif

94
irccommand.cpp Normal file
View File

@ -0,0 +1,94 @@
#include "irccommand.h"
#include <iostream>
IRCCommand::IRCCommand(std::string raw_command) {
if (raw_command[0] == '@') {
int index = raw_command.find_first_of(" ");
std::string tag_section = raw_command.substr(1, index);
raw_command = raw_command.substr(index+1);
// Todo: Further process tags down
}
if (raw_command[0] != 0)
raw_command.substr(1);
else
return;
if (raw_command[0] == ':') {
int index = raw_command.find_first_of(" ");
source = raw_command.substr(1, index);
raw_command = raw_command.substr(index+1);
}
if (raw_command[0] != 0)
raw_command.substr(1);
else
return;
int index = raw_command.find_first_of(" ");
command = raw_command.substr(0, index);
raw_command = raw_command.substr(index+1);
if (raw_command[0] != 0)
raw_command.substr(1);
else
return;
std::vector<std::string> payloads;
index = raw_command.find_first_of(" ");
while (index != std::string::npos && raw_command[0] != ':') {
payloads.push_back(raw_command.substr(0, index));
raw_command = raw_command.substr(index + 1);
index = raw_command.find_first_of(" ");
}
if (raw_command[0] == ':')
payloads.push_back(raw_command.substr(1));
else
payloads.push_back(raw_command);
this->payload = payloads;
}
bool IRCCommand::has_source() {
return (source!="");
}
std::string IRCCommand::get_source() {
return source;
}
bool IRCCommand::has_command() {
return (command!="");
}
std::string IRCCommand::get_command() {
return command;
}
bool IRCCommand::has_payload() {
return payload.size() > 0;
}
std::string IRCCommand::get_payload(int i) {
return payload.at(i);
}
int IRCCommand::get_payload_count() {
return payload.size();
}
std::string IRCCommand::to_string() {
std::string myself;
if (has_source())
myself += "Source: " + get_source() + "\n";
if (has_command())
myself += "Command: " + get_command() + "\n";
for (std::string p : payload)
myself += "Parameters: " + p + "\n";
return myself;
}

33
irccommand.h Normal file
View File

@ -0,0 +1,33 @@
#ifndef IRCCOMMAND_H
#define IRCCOMMAND_H
#include <unordered_map>
#include <string>
#include <vector>
class IRCCommand {
private:
std::unordered_map<std::string, std::string> tags;
std::string source, command = "";
std::vector<std::string> payload;
public:
IRCCommand(std::string raw_command);
bool has_source();
std::string get_source();
bool has_command();
std::string get_command();
bool has_payload();
std::string get_payload(int i);
int get_payload_count();
std::string to_string();
};
#endif

30
ircserver.cpp Normal file
View File

@ -0,0 +1,30 @@
#include "ircserver.h"
IRCServer::IRCServer(std::string hostname) {
this->hostname = hostname;
this->nameable_map = {};
}
bool IRCServer::is_addressable_name_free(std::string name) {
return nameable_map.find(name) == nameable_map.end();
}
bool IRCServer::has_addressable_name(std::string name) {
return !is_addressable_name_free(name);
}
void IRCServer::remove_addressable_name(std::string name) {
nameable_map.erase(name);
}
void IRCServer::register_addressable_name(std::string name, Nameable* bind) {
nameable_map[name] = bind;
}
Nameable* IRCServer::resolve_addressable_name(std::string name) {
return nameable_map[name];
}
std::string IRCServer::get_hostname() {
return hostname;
}

24
ircserver.h Normal file
View File

@ -0,0 +1,24 @@
#ifndef IRCSERVER_H
#define IRCSERVER_H
#include <string>
#include <unordered_map>
class Nameable;
class IRCServer {
private:
std::unordered_map<std::string, Nameable*> nameable_map;
std::string hostname;
public:
IRCServer(std::string);
void register_addressable_name(std::string name, Nameable* bind);
bool is_addressable_name_free(std::string name);
bool has_addressable_name(std::string name);
void remove_addressable_name(std::string name);
Nameable* resolve_addressable_name(std::string name);
std::string get_hostname();
};
#endif

42
main.cpp Normal file
View File

@ -0,0 +1,42 @@
#include <sys/socket.h>
#include <netinet/in.h>
#include <iostream>
#include "ircserver.h"
#include "userconnection.h"
int main() {
int socketfd = socket(AF_INET, SOCK_STREAM, 0);
if (socketfd == 0) {
std::cout << "Socket Create Failed!\n" << std::flush;
return 1;
}
int opt = 1;
if (setsockopt(socketfd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT, &opt, sizeof(opt))) {
std::cout << "Setting Socket Options Failed!\n" << std::flush;
return 1;
}
struct sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_addr.s_addr = INADDR_ANY;
addr.sin_port = htons(6667);
unsigned int addrsize = sizeof(addr);
if (bind(socketfd, (struct sockaddr*)&addr, addrsize) < 0) {
std::cout << "Socket Bind Failed!\n" << std::flush;
return 1;
}
if (listen(socketfd, 5) < 0) {
std::cout << "Socket Listen Failed!\n" << std::flush;
return 1;
}
IRCServer* server = new IRCServer("xnoircd");
while (true) {
int sockfd = accept(socketfd, (struct sockaddr*)&addr, &addrsize);
new UserConnection(sockfd, server);
}
}

71
nameable.cpp Normal file
View File

@ -0,0 +1,71 @@
#include "nameable.h"
Nameable::Nameable(IRCServer* server) {
this->server = server;
}
Nameable::~Nameable() {
this->associates.clear();
this->associates.shrink_to_fit();
}
std::string Nameable::get_addressable_name() {
return this->addressable_name;
}
bool Nameable::set_addressable_name(std::string name) {
if (this->server->is_addressable_name_free(name)) {
this->addressable_name = name;
server->register_addressable_name(name, this);
return true;
} else {
return false;
}
}
bool Nameable::has_addressable_name() {
return (this->addressable_name!="");
}
void Nameable::remove_addressable_name(std::string name) {
server->remove_addressable_name(name);
}
bool Nameable::rename(std::string name) {
if (server->is_addressable_name_free(name)) {
server->remove_addressable_name(name);
server->register_addressable_name(name, this);
this->addressable_name = name;
return true;
} else {
return false;
}
}
std::vector<Nameable*> Nameable::get_associates() {
return this->associates;
}
void Nameable::add_associate(Nameable* associate) {
std::cout << "Adding: " << associate->get_addressable_name() << "\n";
this->associates.push_back(associate);
}
bool Nameable::has_associate(Nameable* associate) {
return !(std::find(associates.begin(), associates.end(), associate) == associates.end());
}
void Nameable::remove_associate(Nameable* associate) {
auto index = std::find(this->associates.begin(), this->associates.end(), associate);
if (index != this->associates.end()) {
this->associates.erase(index);
}
}
std::string Nameable::associates_as_string() {
std::string associate_list;
for (Nameable* associate : associates) {
associate_list += associate->get_addressable_name() + " ";
}
return associate_list;
}
void Nameable::send_direct(std::string message) {}
void Nameable::send_relay(std::string message) {
for (Nameable* associate : this->associates) {
associate->send_direct(message);
}
}
NameableType Nameable::what_are_you() {}

44
nameable.h Normal file
View File

@ -0,0 +1,44 @@
#ifndef NAMEABLE_H
#define NAMEABLE_H
#include <string>
#include <vector>
#include <algorithm>
#include <iostream>
#include "connection.h"
#include "ircserver.h"
enum NameableType {
NT_User,
NT_Channel
};
class Nameable {
private:
std::string addressable_name = "";
std::vector<Nameable*> associates;
public:
IRCServer* server;
Nameable(IRCServer* server);
~Nameable();
std::string get_addressable_name();
bool set_addressable_name(std::string name);
bool has_addressable_name();
void remove_addressable_name(std::string name);
bool rename(std::string to);
std::vector<Nameable*> get_associates();
void add_associate(Nameable* associate);
bool has_associate(Nameable* associate);
void remove_associate(Nameable* associate);
std::string associates_as_string();
virtual void send_direct(std::string message);
void send_relay(std::string message);
virtual NameableType what_are_you();
};
#endif

233
user.cpp Normal file
View File

@ -0,0 +1,233 @@
#include "user.h"
#include <iostream>
std::vector<std::string> split_string(std::string str, std::string delim) {
std::vector<std::string> to_return;
int delim_len = delim.length();
int index = str.find_first_of(delim);
while (index != std::string::npos) {
to_return.push_back(str.substr(0, index));
str = str.substr(index+delim_len);
index = str.find_first_of(delim);
}
to_return.push_back(str);
return to_return;
}
std::unordered_map<std::string, std::tuple<int, std::function<void(User*, IRCCommand*)>>> User::commands = {
{"NICK", {0, &User::nick_cmd}},
{"PING", {1, &User::ping_cmd}},
{"WHO", {1, &User::who_cmd}},
{"WHOIS", {1, &User::whois_cmd}},
{"JOIN", {1, &User::join_cmd}},
{"PART", {1, &User::part_cmd}},
{"QUIT", {0, &User::quit_cmd}},
{"PRIVMSG", {2, &User::privmsg_cmd}},
{"TOPIC", {2, &User::topic_cmd}},
{"NAMES", {1, &User::names_cmd}}
};
User::User(IRCServer* server, UserConnection* conn): Nameable(server) {
this->connection = conn;
}
User::~User() {
this->server->remove_addressable_name(this->get_nick());
this->send_relay(":" + this->get_nick() + " QUIT :Quit: ");
for (Nameable* associate : this->get_associates())
associate->remove_associate(this);
delete connection;
}
void User::nick_cmd(IRCCommand* cmd) {
set_nick(cmd->get_payload(0));
}
void User::ping_cmd(IRCCommand* cmd) {
send_direct(":" + server->get_hostname() + " PONG " + server->get_hostname() + " " + cmd->get_payload(0));
}
void User::join_cmd(IRCCommand* cmd) {
std::string channels_parameter = cmd->get_payload(0);
std::vector<std::string> named_channels = split_string(channels_parameter, ",");
std::cout << named_channels.size() << "\n";
for (std::string named_channel : named_channels) {
Nameable* channel;
if (server->has_addressable_name(named_channel)) {
channel = server->resolve_addressable_name(named_channel);
} else {
channel = new Channel(server, named_channel);
}
if (channel->what_are_you() != NT_Channel)
continue;
Channel* chan = (Channel*)channel;
this->add_associate(chan);
chan->add_associate(this);
chan->send_direct(":" + this->get_nick() + " JOIN " + named_channel);
std::vector<Nameable*> channel_members = chan->get_associates();
this->send_direct(":" + server->get_hostname() + " 353 " + this->get_nick() + " " + named_channel + " :" + chan->associates_as_string());
if (chan->has_topic())
this->send_direct(":" + server->get_hostname() + " 332 " + this->get_nick() + " " + named_channel + " :" + chan->get_topic());
}
}
void User::part_cmd(IRCCommand* cmd) {
std::string channels_parameter = cmd->get_payload(0);
std::vector<std::string> named_channels = split_string(channels_parameter, ",");
for (std::string named_channel : named_channels) {
if (server->has_addressable_name(named_channel)) {
Nameable* channel = server->resolve_addressable_name(named_channel);
this->remove_associate(channel);
channel->remove_associate(this);
channel->send_direct(":" + this->get_nick() + " PART " + named_channel);
}
}
}
void User::privmsg_cmd(IRCCommand* cmd) {
std::string named_target = cmd->get_payload(0);
if (server->has_addressable_name(named_target)) {
Nameable* target = server->resolve_addressable_name(named_target);
for (Nameable* associate : target->get_associates()) {
if (associate->get_addressable_name() != this->get_nick())
associate->send_direct(":" + this->get_nick() + " PRIVMSG " + named_target + " :" + cmd->get_payload(1));
}
}
}
void User::quit_cmd(IRCCommand* cmd) {
destroy();
}
void User::who_cmd(IRCCommand* cmd) {
if (!server->has_addressable_name(cmd->get_payload(0)))
return;
Nameable* target = server->resolve_addressable_name(cmd->get_payload(0));
if (target->what_are_you() != NT_User)
return;
User* user = (User*)target;
send_direct(":" + server->get_hostname() + " 352 " + this->get_nick() + " * " + user->get_username() + " " /*+ user->get_hostname()*/ + " " + server->get_hostname() + " H :0 " + user->get_realname());
send_direct(":" + server->get_hostname() + " 315 " + this->get_nick() + " " + user->get_nick() + " :End of WHO listing.");
}
void User::whois_cmd(IRCCommand* cmd) {
if (!server->has_addressable_name(cmd->get_payload(0)))
return;
Nameable* target = server->resolve_addressable_name(cmd->get_payload(0));
if (target->what_are_you() != NT_User)
return;
User* user = (User*)target;
}
void User::topic_cmd(IRCCommand* cmd) {
if (!server->has_addressable_name(cmd->get_payload(0)))
return;
Nameable* target = server->resolve_addressable_name(cmd->get_payload(0));
if (target->what_are_you() != NT_Channel)
return;
((Channel*)target)->set_topic(cmd->get_payload(1));
}
void User::names_cmd(IRCCommand* cmd) {
std::string named_channel = cmd->get_payload(0);
if (!server->has_addressable_name(named_channel))
return;
Nameable* resolved = server->resolve_addressable_name(named_channel);
if (resolved->what_are_you() != NT_Channel)
return;
Channel* chan = (Channel*)resolved;
this->send_direct(":" + server->get_hostname() + " 353 " + this->get_nick() + " " + named_channel + " :" + chan->associates_as_string());
}
void User::process_cmd(IRCCommand* cmd) {
auto command = commands.find(cmd->get_command());
if (command == commands.end()) {
send_direct(":" + server->get_hostname() + " 421 " + get_nick() + " " + cmd->get_command() + " :Unknown Command.");
return;
}
if (cmd->get_payload_count() < std::get<0>(command->second)) {
send_direct(":" + server->get_hostname() + " 461 " + get_nick() + " " + cmd->get_command() + " :Too few parameters.");
return;
}
std::get<1>(command->second)(this, cmd);
}
void User::set_nick(std::string nick) {
bool success;
std::string old_nick = this->get_nick();
if (this->has_nick())
success = this->rename(nick);
else
success = this->set_addressable_name(nick);
if (success) {
if (old_nick != "")
send_direct(":" + old_nick + " NICK :" + this->get_nick());
} else {
send_direct(":" + server->get_hostname() + " 433 * " + nick + " Nickname is already in use!");
}
}
void User::set_username(std::string username) {
this->username = username;
}
void User::set_realname(std::string realname) {
this->realname = realname;
}
bool User::has_nick() {
return has_addressable_name();
}
bool User::has_username(){
return (username!="");
}
bool User::has_realname(){
return (realname!="");
}
std::string User::get_nick(){
return get_addressable_name();
}
std::string User::get_username(){
return username;
}
std::string User::get_realname(){
return realname;
}
void User::send_direct(std::string message) {
std::cout << message << "\n";
this->connection->send(message);
}
NameableType User::what_are_you() {
return NT_User;
}
void User::destroy() {
Connection* conn = connection;
delete this;
conn->kill();
}

69
user.h Normal file
View File

@ -0,0 +1,69 @@
#ifndef USER_H
#define USER_H
#include <string>
#include <unordered_map>
#include <tuple>
#include <functional>
#include "irccommand.h"
#include "ircserver.h"
#include "userconnection.h"
#include "nameable.h"
#include "channel.h"
class UserConnection;
class User: public Nameable {
private:
std::string username, realname = "";
Connection* connection;
void nick_cmd(IRCCommand* cmd);
void ping_cmd(IRCCommand* cmd);
void who_cmd(IRCCommand* cmd);
void whois_cmd(IRCCommand* cmd);
void join_cmd(IRCCommand* cmd);
void part_cmd(IRCCommand* cmd);
void quit_cmd(IRCCommand* cmd);
void privmsg_cmd(IRCCommand* cmd);
void topic_cmd(IRCCommand* cmd);
void names_cmd(IRCCommand* cmd);
static std::unordered_map<std::string, std::tuple<int, std::function<void(User*, IRCCommand*)>>> commands;
//std::string construct_string(int count, std::string source, std::string command, std::string parameters...);
template<typename... T>
std::string construct_string(std::string source, std::string command, T... parameters);
public:
User(IRCServer* server, UserConnection* conn);
~User();
void process_cmd(IRCCommand* cmd);
void set_nick(std::string nick);
void set_username(std::string username);
void set_realname(std::string realname);
bool has_nick();
bool has_username();
bool has_realname();
std::string get_nick();
std::string get_username();
std::string get_realname();
void send_direct(std::string message);
NameableType what_are_you();
void destroy();
};
#endif

50
userconnection.cpp Normal file
View File

@ -0,0 +1,50 @@
#include "userconnection.h"
#include <iostream>
UserConnection::UserConnection(int fd, IRCServer* server): Connection(fd) {
this->server = server;
this->user = new User(server, this);
}
UserConnection::~UserConnection() {}
void UserConnection::process_cmd(std::string cmd_in) {
if (cmd_in == "")
return;
IRCCommand* cmd = new IRCCommand(cmd_in);
std::cout << cmd_in << std::endl;
if (state == Registered) {
user->process_cmd(cmd);
delete cmd;
return;
}
std::string command = cmd->get_command();
if (command == "NICK" && cmd->has_payload()) {
user->set_nick(cmd->get_payload(0));
}
if (command == "USER" && cmd->has_payload()) {
user->set_username(cmd->get_payload(0));
user->set_realname(cmd->get_payload(3));
}
if (user->has_nick() && user->has_username()) {
state = Registered;
send(":xnoircd 001 " + user->get_nick() + " :Welcome");
send(":xnoircd 002 " + user->get_nick() + " :Welcome");
send(":xnoircd 003 " + user->get_nick() + " :Welcome");
send(":xnoircd 004 " + user->get_nick() + " xnoircd 0 ABCabc ABCabc");
send(":xnoircd 422 " + user->get_nick() + " :No MOTD");
}
delete cmd;
}
void UserConnection::destroy() {
this->user->destroy();
}

29
userconnection.h Normal file
View File

@ -0,0 +1,29 @@
#ifndef USERCONNECTION_H
#define USERCONNECTION_H
#include <string>
#include "connection.h"
#include "ircserver.h"
#include "user.h"
enum UCState {
Initial,
Registered
};
class User;
class UserConnection: public Connection {
private:
User* user;
UCState state = Initial;
IRCServer* server;
public:
UserConnection(int fd, IRCServer* server);
~UserConnection();
void process_cmd(std::string cmd_in);
void destroy();
};
#endif