
And use these to do the line-by-line reading automagically instead of having that logic in IRCClient. This will definitely come in handy.
457 lines
12 KiB
C++
457 lines
12 KiB
C++
#include "IRCClient.h"
|
|
#include "IRCChannel.h"
|
|
#include "IRCQuery.h"
|
|
#include "IRCLogBuffer.h"
|
|
#include "IRCWindow.h"
|
|
#include "IRCWindowListModel.h"
|
|
#include <LibGUI/GNotifier.h>
|
|
#include <sys/socket.h>
|
|
#include <netinet/in.h>
|
|
#include <arpa/inet.h>
|
|
#include <unistd.h>
|
|
#include <stdio.h>
|
|
|
|
#define IRC_DEBUG
|
|
|
|
enum IRCNumeric {
|
|
RPL_TOPIC = 332,
|
|
RPL_TOPICWHOTIME = 333,
|
|
RPL_NAMREPLY = 353,
|
|
RPL_ENDOFNAMES = 366,
|
|
};
|
|
|
|
IRCClient::IRCClient(const String& address, int port)
|
|
: m_hostname(address)
|
|
, m_port(port)
|
|
, m_nickname("anon")
|
|
, m_log(IRCLogBuffer::create())
|
|
{
|
|
m_socket = new GTCPSocket(this);
|
|
m_client_window_list_model = new IRCWindowListModel(*this);
|
|
}
|
|
|
|
IRCClient::~IRCClient()
|
|
{
|
|
}
|
|
|
|
bool IRCClient::connect()
|
|
{
|
|
if (m_socket->is_connected())
|
|
ASSERT_NOT_REACHED();
|
|
|
|
IPv4Address ipv4_address(127, 0, 0, 1);
|
|
bool success = m_socket->connect(GSocketAddress(ipv4_address), m_port);
|
|
if (!success)
|
|
return false;
|
|
|
|
m_notifier = make<GNotifier>(m_socket->fd(), GNotifier::Read);
|
|
m_notifier->on_ready_to_read = [this] (GNotifier&) { receive_from_server(); };
|
|
|
|
send_user();
|
|
send_nick();
|
|
|
|
if (on_connect)
|
|
on_connect();
|
|
return true;
|
|
}
|
|
|
|
void IRCClient::receive_from_server()
|
|
{
|
|
while (m_socket->can_read_line()) {
|
|
auto line = m_socket->read_line(4096);
|
|
if (line.is_null()) {
|
|
if (!m_socket->is_connected()) {
|
|
printf("IRCClient: Connection closed!\n");
|
|
exit(1);
|
|
}
|
|
ASSERT_NOT_REACHED();
|
|
}
|
|
process_line(move(line));
|
|
}
|
|
}
|
|
|
|
void IRCClient::process_line(ByteBuffer&& line)
|
|
{
|
|
Message msg;
|
|
Vector<char> prefix;
|
|
Vector<char> command;
|
|
Vector<char> current_parameter;
|
|
enum {
|
|
Start,
|
|
InPrefix,
|
|
InCommand,
|
|
InStartOfParameter,
|
|
InParameter,
|
|
InTrailingParameter,
|
|
} state = Start;
|
|
|
|
for (int i = 0; i < line.size(); ++i) {
|
|
char ch = line[i];
|
|
if (ch == '\r')
|
|
continue;
|
|
if (ch == '\n')
|
|
break;
|
|
switch (state) {
|
|
case Start:
|
|
if (ch == ':') {
|
|
state = InPrefix;
|
|
continue;
|
|
}
|
|
state = InCommand;
|
|
[[fallthrough]];
|
|
case InCommand:
|
|
if (ch == ' ') {
|
|
state = InStartOfParameter;
|
|
continue;
|
|
}
|
|
command.append(ch);
|
|
continue;
|
|
case InPrefix:
|
|
if (ch == ' ') {
|
|
state = InCommand;
|
|
continue;
|
|
}
|
|
prefix.append(ch);
|
|
continue;
|
|
case InStartOfParameter:
|
|
if (ch == ':') {
|
|
state = InTrailingParameter;
|
|
continue;
|
|
}
|
|
state = InParameter;
|
|
[[fallthrough]];
|
|
case InParameter:
|
|
if (ch == ' ') {
|
|
if (!current_parameter.is_empty())
|
|
msg.arguments.append(String(current_parameter.data(), current_parameter.size()));
|
|
current_parameter.clear_with_capacity();
|
|
state = InStartOfParameter;
|
|
continue;
|
|
}
|
|
current_parameter.append(ch);
|
|
continue;
|
|
case InTrailingParameter:
|
|
current_parameter.append(ch);
|
|
continue;
|
|
}
|
|
}
|
|
if (!current_parameter.is_empty())
|
|
msg.arguments.append(String(current_parameter.data(), current_parameter.size()));
|
|
msg.prefix = String(prefix.data(), prefix.size());
|
|
msg.command = String(command.data(), command.size());
|
|
handle(msg, String(m_line_buffer.data(), m_line_buffer.size()));
|
|
}
|
|
|
|
void IRCClient::send(const String& text)
|
|
{
|
|
if (!m_socket->send(ByteBuffer::wrap((void*)text.characters(), text.length()))) {
|
|
perror("send");
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
void IRCClient::send_user()
|
|
{
|
|
send(String::format("USER %s 0 * :%s\r\n", m_nickname.characters(), m_nickname.characters()));
|
|
}
|
|
|
|
void IRCClient::send_nick()
|
|
{
|
|
send(String::format("NICK %s\r\n", m_nickname.characters()));
|
|
}
|
|
|
|
void IRCClient::send_pong(const String& server)
|
|
{
|
|
send(String::format("PONG %s\r\n", server.characters()));
|
|
sleep(1);
|
|
}
|
|
|
|
void IRCClient::join_channel(const String& channel_name)
|
|
{
|
|
send(String::format("JOIN %s\r\n", channel_name.characters()));
|
|
}
|
|
|
|
void IRCClient::part_channel(const String& channel_name)
|
|
{
|
|
send(String::format("PART %s\r\n", channel_name.characters()));
|
|
}
|
|
|
|
void IRCClient::handle(const Message& msg, const String&)
|
|
{
|
|
#ifdef IRC_DEBUG
|
|
printf("IRCClient::execute: prefix='%s', command='%s', arguments=%d\n",
|
|
msg.prefix.characters(),
|
|
msg.command.characters(),
|
|
msg.arguments.size()
|
|
);
|
|
|
|
int i = 0;
|
|
for (auto& arg : msg.arguments) {
|
|
printf(" [%d]: %s\n", i, arg.characters());
|
|
++i;
|
|
}
|
|
#endif
|
|
|
|
bool is_numeric;
|
|
int numeric = msg.command.to_uint(is_numeric);
|
|
|
|
if (is_numeric) {
|
|
switch (numeric) {
|
|
case RPL_NAMREPLY:
|
|
handle_namreply(msg);
|
|
return;
|
|
case RPL_TOPIC:
|
|
handle_rpl_topic(msg);
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (msg.command == "PING")
|
|
return handle_ping(msg);
|
|
|
|
if (msg.command == "JOIN")
|
|
return handle_join(msg);
|
|
|
|
if (msg.command == "PART")
|
|
return handle_part(msg);
|
|
|
|
if (msg.command == "TOPIC")
|
|
return handle_topic(msg);
|
|
|
|
if (msg.command == "PRIVMSG")
|
|
return handle_privmsg(msg);
|
|
|
|
if (msg.arguments.size() >= 2) {
|
|
m_log->add_message(0, "Server", String::format("[%s] %s", msg.command.characters(), msg.arguments[1].characters()));
|
|
m_server_subwindow->did_add_message();
|
|
}
|
|
}
|
|
|
|
void IRCClient::send_privmsg(const String& target, const String& text)
|
|
{
|
|
send(String::format("PRIVMSG %s :%s\r\n", target.characters(), text.characters()));
|
|
}
|
|
|
|
void IRCClient::handle_user_input_in_channel(const String& channel_name, const String& input)
|
|
{
|
|
if (input.is_empty())
|
|
return;
|
|
if (input[0] == '/')
|
|
return handle_user_command(input);
|
|
ensure_channel(channel_name).say(input);
|
|
}
|
|
|
|
void IRCClient::handle_user_input_in_query(const String& query_name, const String& input)
|
|
{
|
|
if (input.is_empty())
|
|
return;
|
|
if (input[0] == '/')
|
|
return handle_user_command(input);
|
|
ensure_query(query_name).say(input);
|
|
}
|
|
|
|
void IRCClient::handle_user_input_in_server(const String& input)
|
|
{
|
|
if (input.is_empty())
|
|
return;
|
|
if (input[0] == '/')
|
|
return handle_user_command(input);
|
|
}
|
|
|
|
bool IRCClient::is_nick_prefix(char ch) const
|
|
{
|
|
switch (ch) {
|
|
case '@':
|
|
case '+':
|
|
case '~':
|
|
case '&':
|
|
case '%':
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void IRCClient::handle_privmsg(const Message& msg)
|
|
{
|
|
if (msg.arguments.size() < 2)
|
|
return;
|
|
if (msg.prefix.is_empty())
|
|
return;
|
|
auto parts = msg.prefix.split('!');
|
|
auto sender_nick = parts[0];
|
|
auto target = msg.arguments[0];
|
|
|
|
#ifdef IRC_DEBUG
|
|
printf("handle_privmsg: sender_nick='%s', target='%s'\n", sender_nick.characters(), target.characters());
|
|
#endif
|
|
|
|
if (sender_nick.is_empty())
|
|
return;
|
|
|
|
char sender_prefix = 0;
|
|
if (is_nick_prefix(sender_nick[0])) {
|
|
sender_prefix = sender_nick[0];
|
|
sender_nick = sender_nick.substring(1, sender_nick.length() - 1);
|
|
}
|
|
|
|
{
|
|
auto it = m_channels.find(target);
|
|
if (it != m_channels.end()) {
|
|
(*it).value->add_message(sender_prefix, sender_nick, msg.arguments[1]);
|
|
return;
|
|
}
|
|
}
|
|
auto& query = ensure_query(sender_nick);
|
|
query.add_message(sender_prefix, sender_nick, msg.arguments[1]);
|
|
}
|
|
|
|
IRCQuery& IRCClient::ensure_query(const String& name)
|
|
{
|
|
auto it = m_queries.find(name);
|
|
if (it != m_queries.end())
|
|
return *(*it).value;
|
|
auto query = IRCQuery::create(*this, name);
|
|
auto& query_reference = *query;
|
|
m_queries.set(name, query.copy_ref());
|
|
return query_reference;
|
|
}
|
|
|
|
IRCChannel& IRCClient::ensure_channel(const String& name)
|
|
{
|
|
auto it = m_channels.find(name);
|
|
if (it != m_channels.end())
|
|
return *(*it).value;
|
|
auto channel = IRCChannel::create(*this, name);
|
|
auto& channel_reference = *channel;
|
|
m_channels.set(name, channel.copy_ref());
|
|
return channel_reference;
|
|
}
|
|
|
|
void IRCClient::handle_ping(const Message& msg)
|
|
{
|
|
if (msg.arguments.size() < 0)
|
|
return;
|
|
m_log->add_message(0, "Server", "Ping? Pong!");
|
|
send_pong(msg.arguments[0]);
|
|
}
|
|
|
|
void IRCClient::handle_join(const Message& msg)
|
|
{
|
|
if (msg.arguments.size() != 1)
|
|
return;
|
|
auto prefix_parts = msg.prefix.split('!');
|
|
if (prefix_parts.size() < 1)
|
|
return;
|
|
auto nick = prefix_parts[0];
|
|
auto& channel_name = msg.arguments[0];
|
|
ensure_channel(channel_name).handle_join(nick, msg.prefix);
|
|
}
|
|
|
|
void IRCClient::handle_part(const Message& msg)
|
|
{
|
|
if (msg.arguments.size() != 1)
|
|
return;
|
|
auto prefix_parts = msg.prefix.split('!');
|
|
if (prefix_parts.size() < 1)
|
|
return;
|
|
auto nick = prefix_parts[0];
|
|
auto& channel_name = msg.arguments[0];
|
|
ensure_channel(channel_name).handle_part(nick, msg.prefix);
|
|
}
|
|
|
|
void IRCClient::handle_topic(const Message& msg)
|
|
{
|
|
if (msg.arguments.size() != 2)
|
|
return;
|
|
auto prefix_parts = msg.prefix.split('!');
|
|
if (prefix_parts.size() < 1)
|
|
return;
|
|
auto nick = prefix_parts[0];
|
|
auto& channel_name = msg.arguments[0];
|
|
ensure_channel(channel_name).handle_topic(nick, msg.arguments[1]);
|
|
}
|
|
|
|
void IRCClient::handle_rpl_topic(const Message& msg)
|
|
{
|
|
if (msg.arguments.size() != 3)
|
|
return;
|
|
auto& nick = msg.arguments[0];
|
|
auto& channel_name = msg.arguments[1];
|
|
auto& topic = msg.arguments[2];
|
|
ensure_channel(channel_name).handle_topic(nick, topic);
|
|
// FIXME: Handle RPL_TOPICWHOTIME so we can know who set it and when.
|
|
}
|
|
|
|
void IRCClient::handle_namreply(const Message& msg)
|
|
{
|
|
if (msg.arguments.size() < 4)
|
|
return;
|
|
|
|
auto& channel_name = msg.arguments[2];
|
|
|
|
auto it = m_channels.find(channel_name);
|
|
if (it == m_channels.end()) {
|
|
fprintf(stderr, "Warning: Got RPL_NAMREPLY for untracked channel %s\n", channel_name.characters());
|
|
return;
|
|
}
|
|
auto& channel = *(*it).value;
|
|
|
|
auto members = msg.arguments[3].split(' ');
|
|
for (auto& member : members) {
|
|
if (member.is_empty())
|
|
continue;
|
|
char prefix = 0;
|
|
if (is_nick_prefix(member[0]))
|
|
prefix = member[0];
|
|
channel.add_member(member, prefix);
|
|
}
|
|
|
|
channel.dump();
|
|
}
|
|
|
|
void IRCClient::register_subwindow(IRCWindow& subwindow)
|
|
{
|
|
if (subwindow.type() == IRCWindow::Server) {
|
|
m_server_subwindow = &subwindow;
|
|
subwindow.set_log_buffer(*m_log);
|
|
}
|
|
m_windows.append(&subwindow);
|
|
m_client_window_list_model->update();
|
|
}
|
|
|
|
void IRCClient::unregister_subwindow(IRCWindow& subwindow)
|
|
{
|
|
if (subwindow.type() == IRCWindow::Server) {
|
|
m_server_subwindow = &subwindow;
|
|
}
|
|
for (int i = 0; i < m_windows.size(); ++i) {
|
|
if (m_windows.at(i) == &subwindow) {
|
|
m_windows.remove(i);
|
|
break;
|
|
}
|
|
}
|
|
m_client_window_list_model->update();
|
|
}
|
|
|
|
void IRCClient::handle_user_command(const String& input)
|
|
{
|
|
auto parts = input.split(' ');
|
|
if (parts.is_empty())
|
|
return;
|
|
auto command = parts[0].to_uppercase();
|
|
if (command == "/JOIN") {
|
|
if (parts.size() >= 2)
|
|
join_channel(parts[1]);
|
|
return;
|
|
}
|
|
if (command == "/PART") {
|
|
if (parts.size() >= 2)
|
|
part_channel(parts[1]);
|
|
return;
|
|
}
|
|
if (command == "/QUERY") {
|
|
if (parts.size() >= 2)
|
|
ensure_query(parts[1]);
|
|
return;
|
|
}
|
|
}
|