summaryrefslogtreecommitdiffstats
path: root/src/irc/core/channel-events.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/irc/core/channel-events.c')
-rw-r--r--src/irc/core/channel-events.c402
1 files changed, 402 insertions, 0 deletions
diff --git a/src/irc/core/channel-events.c b/src/irc/core/channel-events.c
new file mode 100644
index 0000000..3cdfe96
--- /dev/null
+++ b/src/irc/core/channel-events.c
@@ -0,0 +1,402 @@
+/*
+ channel-events.c : irssi
+
+ Copyright (C) 1999-2000 Timo Sirainen
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License along
+ with this program; if not, write to the Free Software Foundation, Inc.,
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+*/
+
+#include "module.h"
+#include <irssi/src/core/signals.h>
+#include <irssi/src/core/misc.h>
+#include <irssi/src/irc/core/channel-events.h>
+#include <irssi/src/core/channels-setup.h>
+#include <irssi/src/core/settings.h>
+#include <irssi/src/core/recode.h>
+
+#include <irssi/src/irc/core/irc-servers.h>
+#include <irssi/src/irc/core/irc-channels.h>
+
+static void check_join_failure(IRC_SERVER_REC *server, const char *channel)
+{
+ CHANNEL_REC *chanrec;
+ char *chan2;
+
+ if (channel[0] == '!' && channel[1] == '!')
+ channel++; /* server didn't understand !channels */
+
+ chanrec = channel_find(SERVER(server), channel);
+ if (chanrec == NULL && channel[0] == '!' && strlen(channel) > 6) {
+ /* it probably replied with the full !channel name,
+ find the channel with the short name.. */
+ chan2 = g_strdup_printf("!%s", channel+6);
+ chanrec = channel_find(SERVER(server), chan2);
+ g_free(chan2);
+ }
+
+ if (chanrec != NULL && !chanrec->joined) {
+ chanrec->left = TRUE;
+ channel_destroy(chanrec);
+ }
+}
+
+static void irc_server_event(IRC_SERVER_REC *server, const char *line)
+{
+ char *params, *numeric, *channel;
+
+ /* We'll be checking "4xx <your nick> <channel>" for channels
+ which we haven't joined yet. 4xx are error codes and should
+ indicate that the join failed. */
+ params = event_get_params(line, 3, &numeric, NULL, &channel);
+
+ if (numeric[0] == '4')
+ check_join_failure(server, channel);
+
+ g_free(params);
+}
+
+static void event_no_such_channel(IRC_SERVER_REC *server, const char *data)
+{
+ CHANNEL_REC *chanrec;
+ CHANNEL_SETUP_REC *setup;
+ char *params, *channel;
+
+ params = event_get_params(data, 2, NULL, &channel);
+ chanrec = *channel == '!' && channel[1] != '\0' ?
+ channel_find(SERVER(server), channel) : NULL;
+
+ if (chanrec != NULL) {
+ /* !channel didn't exist, so join failed */
+ setup = channel_setup_find(chanrec->name,
+ chanrec->server->connrec->chatnet);
+ if (setup != NULL && setup->autojoin) {
+ /* it's autojoin channel though, so create it */
+ irc_send_cmdv(server, "JOIN !%s", chanrec->name);
+ g_free(params);
+ return;
+ }
+ }
+
+ check_join_failure(server, channel);
+ g_free(params);
+}
+
+static void event_duplicate_channel(IRC_SERVER_REC *server, const char *data)
+{
+ CHANNEL_REC *chanrec;
+ char *params, *channel, *p;
+
+ g_return_if_fail(data != NULL);
+
+ /* this new addition to ircd breaks completely with older
+ "standards", "nick Duplicate ::!!channel ...." */
+ params = event_get_params(data, 3, NULL, NULL, &channel);
+ p = strchr(channel, ' ');
+ if (p != NULL) *p = '\0';
+
+ if (channel[0] == '!') {
+ chanrec = channel_find(SERVER(server),
+ channel+(channel[1] == '!'));
+ if (chanrec != NULL && !chanrec->names_got) {
+ chanrec->left = TRUE;
+ channel_destroy(chanrec);
+ }
+ }
+
+ g_free(params);
+}
+
+static void channel_change_topic(IRC_SERVER_REC *server, const char *channel,
+ const char *topic, const char *setby,
+ time_t settime)
+{
+ CHANNEL_REC *chanrec;
+ char *recoded = NULL;
+
+ chanrec = channel_find(SERVER(server), channel);
+ if (chanrec == NULL) return;
+ /* the topic may be send out encoded, so we need to
+ recode it back or /topic <tab> will not work properly */
+ recoded = recode_in(SERVER(server), topic, channel);
+ if (topic != NULL) {
+ g_free_not_null(chanrec->topic);
+ chanrec->topic = recoded == NULL ? NULL : g_strdup(recoded);
+ }
+ g_free(recoded);
+
+ g_free_not_null(chanrec->topic_by);
+ chanrec->topic_by = g_strdup(setby);
+
+ if (chanrec->topic_by == NULL) {
+ /* ensure invariant topic_time > 0 <=> topic_by != NULL.
+ this could be triggered by a topic command without sender */
+ chanrec->topic_time = 0;
+ } else {
+ chanrec->topic_time = settime;
+ }
+
+ signal_emit("channel topic changed", 1, chanrec);
+}
+
+static void event_topic_get(IRC_SERVER_REC *server, const char *data)
+{
+ char *params, *channel, *topic;
+
+ g_return_if_fail(data != NULL);
+
+ params = event_get_params(data, 3, NULL, &channel, &topic);
+ channel_change_topic(server, channel, topic, NULL, 0);
+ g_free(params);
+}
+
+static void event_topic(IRC_SERVER_REC *server, const char *data,
+ const char *nick, const char *addr)
+{
+ char *params, *channel, *topic, *mask;
+
+ g_return_if_fail(data != NULL);
+
+ params = event_get_params(data, 2, &channel, &topic);
+ mask = addr == NULL ? g_strdup(nick) :
+ g_strconcat(nick, "!", addr, NULL);
+ channel_change_topic(server, channel, topic, mask, time(NULL));
+ g_free(mask);
+ g_free(params);
+}
+
+static void event_topic_info(IRC_SERVER_REC *server, const char *data)
+{
+ char *params, *channel, *topicby, *topictime;
+ time_t t;
+
+ g_return_if_fail(data != NULL);
+
+ params = event_get_params(data, 4, NULL, &channel,
+ &topicby, &topictime);
+
+ t = (time_t) atol(topictime);
+ channel_change_topic(server, channel, NULL, topicby, t);
+ g_free(params);
+}
+
+/* Find any unjoined channel that matches `channel'. Long channel names are
+ also a bit problematic, so find a channel where start of the name matches. */
+static IRC_CHANNEL_REC *channel_find_unjoined(IRC_SERVER_REC *server,
+ const char *channel)
+{
+ GSList *tmp;
+ int len;
+
+ len = strlen(channel);
+ for (tmp = server->channels; tmp != NULL; tmp = tmp->next) {
+ IRC_CHANNEL_REC *rec = tmp->data;
+
+ if (!IS_IRC_CHANNEL(rec) || rec->joined)
+ continue;
+
+ if (g_ascii_strncasecmp(channel, rec->name, len) == 0 &&
+ (len > 20 || rec->name[len] == '\0'))
+ return rec;
+ }
+
+ return NULL;
+}
+
+static void event_join(IRC_SERVER_REC *server, const char *data, const char *nick, const char *address)
+{
+ char *params, *channel, *tmp, *shortchan;
+ IRC_CHANNEL_REC *chanrec;
+
+ g_return_if_fail(data != NULL);
+
+ if (g_ascii_strcasecmp(nick, server->nick) != 0) {
+ /* someone else joined channel, no need to do anything */
+ return;
+ }
+
+ if (server->userhost == NULL)
+ server->userhost = g_strdup(address);
+
+ params = event_get_params(data, 1, &channel);
+ tmp = strchr(channel, 7); /* ^G does something weird.. */
+ if (tmp != NULL) *tmp = '\0';
+
+ if (*channel != '!' || strlen(channel) < 7)
+ shortchan = NULL;
+ else {
+ /* !channels have 5 chars long identification string before
+ it's name, it's not known when /join is called so rename
+ !channel here to !ABCDEchannel */
+ shortchan = g_strdup_printf("!%s", channel+6);
+ chanrec = channel_find_unjoined(server, shortchan);
+ if (chanrec != NULL) {
+ channel_change_name(CHANNEL(chanrec), channel);
+ g_free(chanrec->name);
+ chanrec->name = g_strdup(channel);
+ } else {
+ /* well, did we join it with full name? if so, and if
+ this was the first short one, change it's name. */
+ chanrec = channel_find_unjoined(server, channel);
+ if (chanrec != NULL &&
+ irc_channel_find(server, shortchan) == NULL) {
+ channel_change_visible_name(CHANNEL(chanrec),
+ shortchan);
+ }
+ }
+ }
+
+ chanrec = irc_channel_find(server, channel);
+ if (chanrec != NULL && chanrec->joined) {
+ /* already joined this channel - probably a broken proxy that
+ forgot to send PART between */
+ chanrec->left = TRUE;
+ channel_destroy(CHANNEL(chanrec));
+ chanrec = NULL;
+ }
+
+ if (chanrec == NULL) {
+ /* look again, because of the channel name cut issues. */
+ chanrec = channel_find_unjoined(server, channel);
+ }
+
+ if (chanrec == NULL) {
+ /* didn't get here with /join command.. */
+ chanrec = irc_channel_create(server, channel, shortchan, TRUE);
+ }
+
+ chanrec->joined = TRUE;
+ if (g_strcmp0(chanrec->name, channel) != 0) {
+ g_free(chanrec->name);
+ chanrec->name = g_strdup(channel);
+ }
+
+ g_free(shortchan);
+ g_free(params);
+}
+
+static void event_part(IRC_SERVER_REC *server, const char *data, const char *nick)
+{
+ char *params, *channel, *reason;
+ CHANNEL_REC *chanrec;
+
+ g_return_if_fail(data != NULL);
+
+ if (g_ascii_strcasecmp(nick, server->nick) != 0) {
+ /* someone else part, no need to do anything here */
+ return;
+ }
+
+ params = event_get_params(data, 2, &channel, &reason);
+
+ chanrec = channel_find(SERVER(server), channel);
+ if (chanrec != NULL && chanrec->joined) {
+ chanrec->left = TRUE;
+ channel_destroy(chanrec);
+ }
+
+ g_free(params);
+}
+
+static void event_kick(IRC_SERVER_REC *server, const char *data)
+{
+ CHANNEL_REC *chanrec;
+ char *params, *channel, *nick, *reason;
+
+ g_return_if_fail(data != NULL);
+
+ params = event_get_params(data, 3, &channel, &nick, &reason);
+
+ if (g_ascii_strcasecmp(nick, server->nick) != 0) {
+ /* someone else was kicked, no need to do anything */
+ g_free(params);
+ return;
+ }
+
+ chanrec = channel_find(SERVER(server), channel);
+ if (chanrec != NULL) {
+ irc_server_purge_output(server, channel);
+ chanrec->kicked = TRUE;
+ channel_destroy(chanrec);
+ }
+
+ g_free(params);
+}
+
+static void event_invite(IRC_SERVER_REC *server, const char *data)
+{
+ char *params, *nick, *channel, *shortchan;
+
+ g_return_if_fail(data != NULL);
+
+ params = event_get_params(data, 2, &nick, &channel);
+
+ if (server->nick_comp_func(nick, server->nick) != 0) {
+ /* someone else was invited, no need to do anything */
+ g_free(params);
+ return;
+ }
+
+ if (irc_channel_find(server, channel) == NULL) {
+ /* check if we're supposed to autojoin this channel */
+ CHANNEL_SETUP_REC *setup;
+
+ setup = channel_setup_find(channel, server->connrec->chatnet);
+ if (setup == NULL && channel[0] == '!' &&
+ strlen(channel) > 6) {
+ shortchan = g_strdup_printf("!%s", channel+6);
+ setup = channel_setup_find(shortchan,
+ server->connrec->chatnet);
+ g_free(shortchan);
+ }
+ if (setup != NULL && setup->autojoin && settings_get_bool("join_auto_chans_on_invite"))
+ server->channels_join(SERVER(server), channel, TRUE);
+ }
+
+ g_free_not_null(server->last_invite);
+ server->last_invite = g_strdup(channel);
+ g_free(params);
+}
+
+void channel_events_init(void)
+{
+ settings_add_bool("misc", "join_auto_chans_on_invite", TRUE);
+
+ signal_add_last("server event", (SIGNAL_FUNC) irc_server_event);
+ signal_add_first("event 403", (SIGNAL_FUNC) event_no_such_channel); /* no such channel */
+ signal_add_first("event 407", (SIGNAL_FUNC) event_duplicate_channel); /* duplicate channel */
+
+ signal_add("event topic", (SIGNAL_FUNC) event_topic);
+ signal_add_first("event join", (SIGNAL_FUNC) event_join);
+ signal_add("event part", (SIGNAL_FUNC) event_part);
+ signal_add("event kick", (SIGNAL_FUNC) event_kick);
+ signal_add("event invite", (SIGNAL_FUNC) event_invite);
+ signal_add("event 332", (SIGNAL_FUNC) event_topic_get);
+ signal_add("event 333", (SIGNAL_FUNC) event_topic_info);
+}
+
+void channel_events_deinit(void)
+{
+ signal_remove("server event", (SIGNAL_FUNC) irc_server_event);
+ signal_remove("event 403", (SIGNAL_FUNC) event_no_such_channel); /* no such channel */
+ signal_remove("event 407", (SIGNAL_FUNC) event_duplicate_channel); /* duplicate channel */
+
+ signal_remove("event topic", (SIGNAL_FUNC) event_topic);
+ signal_remove("event join", (SIGNAL_FUNC) event_join);
+ signal_remove("event part", (SIGNAL_FUNC) event_part);
+ signal_remove("event kick", (SIGNAL_FUNC) event_kick);
+ signal_remove("event invite", (SIGNAL_FUNC) event_invite);
+ signal_remove("event 332", (SIGNAL_FUNC) event_topic_get);
+ signal_remove("event 333", (SIGNAL_FUNC) event_topic_info);
+}