serenity/Applications/IRCClient/IRCClient.cpp
Andreas Kling 4ea229accd LibCore: Convert CTCPServer to ObjectPtr
Also get rid of the custom CNotifier::create() in favor of construct().
2019-09-21 15:25:08 +02:00

747 lines
20 KiB
C++

#include "IRCClient.h"
#include "IRCAppWindow.h"
#include "IRCChannel.h"
#include "IRCLogBuffer.h"
#include "IRCQuery.h"
#include "IRCWindow.h"
#include "IRCWindowListModel.h"
#include <AK/StringBuilder.h>
#include <LibCore/CNotifier.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <stdio.h>
#include <sys/socket.h>
#include <time.h>
#include <unistd.h>
#define IRC_DEBUG
enum IRCNumeric {
RPL_WHOISUSER = 311,
RPL_WHOISSERVER = 312,
RPL_WHOISOPERATOR = 313,
RPL_WHOISIDLE = 317,
RPL_ENDOFWHOIS = 318,
RPL_WHOISCHANNELS = 319,
RPL_TOPIC = 332,
RPL_TOPICWHOTIME = 333,
RPL_NAMREPLY = 353,
RPL_ENDOFNAMES = 366,
};
IRCClient::IRCClient()
: m_nickname("seren1ty")
, m_client_window_list_model(IRCWindowListModel::create(*this))
, m_log(IRCLogBuffer::create())
, m_config(CConfigFile::get_for_app("IRCClient"))
{
m_socket = CTCPSocket::construct(this);
m_nickname = m_config->read_entry("User", "Nickname", "seren1ty");
m_hostname = m_config->read_entry("Connection", "Server", "");
m_port = m_config->read_num_entry("Connection", "Port", 6667);
}
IRCClient::~IRCClient()
{
}
void IRCClient::set_server(const String& hostname, int port)
{
m_hostname = hostname;
m_port = port;
m_config->write_entry("Connection", "Server", hostname);
m_config->write_num_entry("Connection", "Port", port);
m_config->sync();
}
void IRCClient::on_socket_connected()
{
m_notifier = CNotifier::construct(m_socket->fd(), CNotifier::Read);
m_notifier->on_ready_to_read = [this] { receive_from_server(); };
send_user();
send_nick();
auto channel_str = m_config->read_entry("Connection", "AutoJoinChannels", "#test");
dbgprintf("IRCClient: Channels to autojoin: %s\n", channel_str.characters());
auto channels = channel_str.split(',');
for (auto& channel : channels) {
join_channel(channel);
dbgprintf("IRCClient: Auto joining channel: %s\n", channel.characters());
}
}
bool IRCClient::connect()
{
if (m_socket->is_connected())
ASSERT_NOT_REACHED();
m_socket->on_connected = [this] { on_socket_connected(); };
bool success = m_socket->connect(m_hostname, m_port);
if (!success)
return false;
return true;
}
void IRCClient::receive_from_server()
{
while (m_socket->can_read_line()) {
auto line = m_socket->read_line(PAGE_SIZE);
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, 32> prefix;
Vector<char, 32> command;
Vector<char, 256> 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::copy(current_parameter));
msg.prefix = String::copy(prefix);
msg.command = String::copy(command);
handle(msg);
}
void IRCClient::send(const String& text)
{
if (!m_socket->send(ByteBuffer::wrap(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::send_whois(const String& nick)
{
send(String::format("WHOIS %s\r\n", nick.characters()));
}
void IRCClient::handle(const Message& msg)
{
#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_WHOISCHANNELS:
return handle_rpl_whoischannels(msg);
case RPL_ENDOFWHOIS:
return handle_rpl_endofwhois(msg);
case RPL_WHOISOPERATOR:
return handle_rpl_whoisoperator(msg);
case RPL_WHOISSERVER:
return handle_rpl_whoisserver(msg);
case RPL_WHOISUSER:
return handle_rpl_whoisuser(msg);
case RPL_WHOISIDLE:
return handle_rpl_whoisidle(msg);
case RPL_TOPICWHOTIME:
return handle_rpl_topicwhotime(msg);
case RPL_TOPIC:
return handle_rpl_topic(msg);
case RPL_NAMREPLY:
return handle_rpl_namreply(msg);
case RPL_ENDOFNAMES:
return handle_rpl_endofnames(msg);
}
}
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_or_notice(msg, PrivmsgOrNotice::Privmsg);
if (msg.command == "NOTICE")
return handle_privmsg_or_notice(msg, PrivmsgOrNotice::Notice);
if (msg.command == "NICK")
return handle_nick(msg);
if (msg.arguments.size() >= 2)
add_server_message(String::format("[%s] %s", msg.command.characters(), msg.arguments[1].characters()));
}
void IRCClient::add_server_message(const String& text, Color color)
{
m_log->add_message(0, "", text, color);
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::send_notice(const String& target, const String& text)
{
send(String::format("NOTICE %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;
}
static bool has_ctcp_payload(const StringView& string)
{
return string.length() >= 2 && string[0] == 0x01 && string[string.length() - 1] == 0x01;
}
void IRCClient::handle_privmsg_or_notice(const Message& msg, PrivmsgOrNotice type)
{
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];
bool is_ctcp = has_ctcp_payload(msg.arguments[1]);
#ifdef IRC_DEBUG
printf("handle_privmsg_or_notice: type='%s'%s, sender_nick='%s', target='%s'\n",
type == PrivmsgOrNotice::Privmsg ? "privmsg" : "notice",
is_ctcp ? " (ctcp)" : "",
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);
}
String message_text = msg.arguments[1];
auto message_color = Color::Black;
if (is_ctcp) {
auto ctcp_payload = msg.arguments[1].substring_view(1, msg.arguments[1].length() - 2);
if (type == PrivmsgOrNotice::Privmsg)
handle_ctcp_request(sender_nick, ctcp_payload);
else
handle_ctcp_response(sender_nick, ctcp_payload);
StringBuilder builder;
builder.append("(CTCP) ");
builder.append(ctcp_payload);
message_text = builder.to_string();
message_color = Color::Blue;
}
{
auto it = m_channels.find(target);
if (it != m_channels.end()) {
(*it).value->add_message(sender_prefix, sender_nick, message_text, message_color);
return;
}
}
// For NOTICE or CTCP messages, only put them in query if one already exists.
// Otherwise, put them in the server window. This seems to match other clients.
IRCQuery* query = nullptr;
if (is_ctcp || type == PrivmsgOrNotice::Notice) {
query = query_with_name(sender_nick);
} else {
query = &ensure_query(sender_nick);
}
if (query)
query->add_message(sender_prefix, sender_nick, message_text, message_color);
else {
add_server_message(String::format("<%s> %s", sender_nick.characters(), message_text.characters()), message_color);
}
}
IRCQuery* IRCClient::query_with_name(const String& name)
{
return m_queries.get(name).value_or(nullptr);
}
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);
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);
return channel_reference;
}
void IRCClient::handle_ping(const Message& msg)
{
if (msg.arguments.size() < 0)
return;
m_log->add_message(0, "", "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_nick(const Message& msg)
{
auto prefix_parts = msg.prefix.split('!');
if (prefix_parts.size() < 1)
return;
auto old_nick = prefix_parts[0];
if (msg.arguments.size() != 1)
return;
auto& new_nick = msg.arguments[0];
if (old_nick == m_nickname)
m_nickname = new_nick;
add_server_message(String::format("~ %s changed nickname to %s", old_nick.characters(), new_nick.characters()));
if (on_nickname_changed)
on_nickname_changed(new_nick);
for (auto& it : m_channels) {
it.value->notify_nick_changed(old_nick, new_nick);
}
}
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& channel_name = msg.arguments[1];
auto& topic = msg.arguments[2];
ensure_channel(channel_name).handle_topic({}, topic);
// FIXME: Handle RPL_TOPICWHOTIME so we can know who set it and when.
}
void IRCClient::handle_rpl_namreply(const Message& msg)
{
if (msg.arguments.size() < 4)
return;
auto& channel_name = msg.arguments[2];
auto& channel = ensure_channel(channel_name);
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);
}
}
void IRCClient::handle_rpl_endofnames(const Message&)
{
}
void IRCClient::handle_rpl_endofwhois(const Message&)
{
add_server_message("// End of WHOIS");
}
void IRCClient::handle_rpl_whoisoperator(const Message& msg)
{
if (msg.arguments.size() < 2)
return;
auto& nick = msg.arguments[1];
add_server_message(String::format("* %s is an IRC operator", nick.characters()));
}
void IRCClient::handle_rpl_whoisserver(const Message& msg)
{
if (msg.arguments.size() < 3)
return;
auto& nick = msg.arguments[1];
auto& server = msg.arguments[2];
add_server_message(String::format("* %s is using server %s", nick.characters(), server.characters()));
}
void IRCClient::handle_rpl_whoisuser(const Message& msg)
{
if (msg.arguments.size() < 6)
return;
auto& nick = msg.arguments[1];
auto& username = msg.arguments[2];
auto& host = msg.arguments[3];
auto& asterisk = msg.arguments[4];
auto& realname = msg.arguments[5];
(void)asterisk;
add_server_message(String::format("* %s is %s@%s, real name: %s",
nick.characters(),
username.characters(),
host.characters(),
realname.characters()));
}
void IRCClient::handle_rpl_whoisidle(const Message& msg)
{
if (msg.arguments.size() < 3)
return;
auto& nick = msg.arguments[1];
auto& secs = msg.arguments[2];
add_server_message(String::format("* %s is %s seconds idle", nick.characters(), secs.characters()));
}
void IRCClient::handle_rpl_whoischannels(const Message& msg)
{
if (msg.arguments.size() < 3)
return;
auto& nick = msg.arguments[1];
auto& channel_list = msg.arguments[2];
add_server_message(String::format("* %s is in channels %s", nick.characters(), channel_list.characters()));
}
void IRCClient::handle_rpl_topicwhotime(const Message& msg)
{
if (msg.arguments.size() < 4)
return;
auto& channel_name = msg.arguments[1];
auto& nick = msg.arguments[2];
auto setat = msg.arguments[3];
bool ok;
time_t setat_time = setat.to_uint(ok);
if (ok) {
auto* tm = localtime(&setat_time);
setat = String::format("%4u-%02u-%02u %02u:%02u:%02u",
tm->tm_year + 1900,
tm->tm_mon + 1,
tm->tm_mday,
tm->tm_hour,
tm->tm_min,
tm->tm_sec);
}
ensure_channel(channel_name).add_message(String::format("*** (set by %s at %s)", nick.characters(), setat.characters()), Color::Blue);
}
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_view(' ');
if (parts.is_empty())
return;
auto command = String(parts[0]).to_uppercase();
if (command == "/NICK") {
if (parts.size() >= 2)
change_nick(parts[1]);
return;
}
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) {
auto& query = ensure_query(parts[1]);
IRCAppWindow::the().set_active_window(query.window());
}
return;
}
if (command == "/MSG") {
if (parts.size() < 3)
return;
auto nick = parts[1];
auto& query = ensure_query(nick);
IRCAppWindow::the().set_active_window(query.window());
query.say(input.view().substring_view_starting_after_substring(nick));
return;
}
if (command == "/WHOIS") {
if (parts.size() >= 2)
send_whois(parts[1]);
return;
}
}
void IRCClient::change_nick(const String& nick)
{
send(String::format("NICK %s\r\n", nick.characters()));
}
void IRCClient::handle_whois_action(const String& nick)
{
send_whois(nick);
}
void IRCClient::handle_open_query_action(const String& nick)
{
ensure_query(nick);
}
void IRCClient::handle_change_nick_action(const String& nick)
{
change_nick(nick);
}
void IRCClient::handle_close_query_action(const String& nick)
{
m_queries.remove(nick);
m_client_window_list_model->update();
}
void IRCClient::handle_join_action(const String& channel)
{
join_channel(channel);
}
void IRCClient::handle_part_action(const String& channel)
{
part_channel(channel);
}
void IRCClient::did_part_from_channel(Badge<IRCChannel>, IRCChannel& channel)
{
if (on_part_from_channel)
on_part_from_channel(channel);
}
void IRCClient::send_ctcp_response(const StringView& peer, const StringView& payload)
{
StringBuilder builder;
builder.append(0x01);
builder.append(payload);
builder.append(0x01);
auto message = builder.to_string();
send_notice(peer, message);
}
void IRCClient::handle_ctcp_request(const StringView& peer, const StringView& payload)
{
dbg() << "handle_ctcp_request: " << payload;
if (payload == "VERSION") {
send_ctcp_response(peer, "VERSION IRC Client [x86] / Serenity OS");
return;
}
if (payload.starts_with("PING")) {
send_ctcp_response(peer, payload);
return;
}
}
void IRCClient::handle_ctcp_response(const StringView& peer, const StringView& payload)
{
dbg() << "handle_ctcp_response(" << peer << "): " << payload;
}