소스 검색

IRCClient: Add support for a bunch of numerics, mostly WHOIS related.

Andreas Kling 6 년 전
부모
커밋
d7659ceebf
2개의 변경된 파일133개의 추가작업 그리고 22개의 파일을 삭제
  1. 121 21
      Applications/IRCClient/IRCClient.cpp
  2. 12 1
      Applications/IRCClient/IRCClient.h

+ 121 - 21
Applications/IRCClient/IRCClient.cpp

@@ -10,10 +10,17 @@
 #include <arpa/inet.h>
 #include <unistd.h>
 #include <stdio.h>
+#include <time.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,
@@ -176,6 +183,11 @@ 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, const String&)
 {
 #ifdef IRC_DEBUG
@@ -197,12 +209,16 @@ void IRCClient::handle(const Message& msg, const String&)
 
     if (is_numeric) {
         switch (numeric) {
-        case RPL_NAMREPLY:
-            handle_namreply(msg);
-            return;
-        case RPL_TOPIC:
-            handle_rpl_topic(msg);
-            return;
+        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);
         }
     }
 
@@ -221,10 +237,14 @@ void IRCClient::handle(const Message& msg, const String&)
     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();
-    }
+    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)
+{
+    m_log->add_message(0, "Server", text);
+    m_server_subwindow->did_add_message();
 }
 
 void IRCClient::send_privmsg(const String& target, const String& text)
@@ -382,20 +402,12 @@ void IRCClient::handle_rpl_topic(const Message& msg)
     // FIXME: Handle RPL_TOPICWHOTIME so we can know who set it and when.
 }
 
-void IRCClient::handle_namreply(const Message& msg)
+void IRCClient::handle_rpl_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& channel = ensure_channel(channel_name);
     auto members = msg.arguments[3].split(' ');
     for (auto& member : members) {
         if (member.is_empty())
@@ -405,8 +417,91 @@ void IRCClient::handle_namreply(const Message& msg)
             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()));
+}
 
-    channel.dump();
+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 %d 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(0, "", String::format("Topic set by %s at %s", nick.characters(), setat.characters()));
 }
 
 void IRCClient::register_subwindow(IRCWindow& subwindow)
@@ -454,4 +549,9 @@ void IRCClient::handle_user_command(const String& input)
             ensure_query(parts[1]);
         return;
     }
+    if (command == "/WHOIS") {
+        if (parts.size() >= 2)
+            send_whois(parts[1]);
+        return;
+    }
 }

+ 12 - 1
Applications/IRCClient/IRCClient.h

@@ -60,6 +60,8 @@ public:
     IRCQuery& ensure_query(const String& name);
     IRCChannel& ensure_channel(const String& name);
 
+    void add_server_message(const String&);
+
     const char* class_name() const override { return "IRCClient"; }
 
 private:
@@ -75,13 +77,22 @@ private:
     void send_nick();
     void send_pong(const String& server);
     void send_privmsg(const String& target, const String&);
+    void send_whois(const String&);
     void process_line(ByteBuffer&&);
     void handle_join(const Message&);
     void handle_part(const Message&);
     void handle_ping(const Message&);
     void handle_topic(const Message&);
     void handle_rpl_topic(const Message&);
-    void handle_namreply(const Message&);
+    void handle_rpl_whoisuser(const Message&);
+    void handle_rpl_whoisserver(const Message&);
+    void handle_rpl_whoisoperator(const Message&);
+    void handle_rpl_whoisidle(const Message&);
+    void handle_rpl_endofwhois(const Message&);
+    void handle_rpl_whoischannels(const Message&);
+    void handle_rpl_topicwhotime(const Message&);
+    void handle_rpl_endofnames(const Message&);
+    void handle_rpl_namreply(const Message&);
     void handle_privmsg(const Message&);
     void handle(const Message&, const String& verbatim);
     void handle_user_command(const String&);