/* * irc-protocol.c - implementation of IRC protocol (RFCs 1459/2810/2811/2812/2813) * * Copyright (C) 2003-2021 Sébastien Helleu * Copyright (C) 2006 Emmanuel Bouthenot * Copyright (C) 2014 Shawn Smith * * This file is part of WeeChat, the extensible chat client. * * WeeChat 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 3 of the License, or * (at your option) any later version. * * WeeChat 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 WeeChat. If not, see . */ /* this define is needed for strptime() (not on OpenBSD/Sun) */ #if !defined(__OpenBSD__) && !defined(__sun) #define _XOPEN_SOURCE 700 #endif #ifndef __USE_XOPEN #define __USE_XOPEN #endif #include #include #include #include #include #include #include "../weechat-plugin.h" #include "irc.h" #include "irc-protocol.h" #include "irc-bar-item.h" #include "irc-buffer.h" #include "irc-channel.h" #include "irc-color.h" #include "irc-command.h" #include "irc-config.h" #include "irc-ctcp.h" #include "irc-ignore.h" #include "irc-message.h" #include "irc-mode.h" #include "irc-modelist.h" #include "irc-msgbuffer.h" #include "irc-nick.h" #include "irc-notify.h" #include "irc-sasl.h" #include "irc-server.h" #include "irc-tag.h" #include "irc-typing.h" /* * Checks if a command is numeric. * * Returns: * 1: all chars are numeric * 0: command has other chars (not numeric) */ int irc_protocol_is_numeric_command (const char *command) { if (!command || !command[0]) return 0; while (command && command[0]) { if (!isdigit ((unsigned char)command[0])) return 0; command++; } return 1; } /* * Gets log level for IRC command. */ int irc_protocol_log_level_for_command (const char *command) { if (!command || !command[0]) return 0; if ((strcmp (command, "privmsg") == 0) || (strcmp (command, "notice") == 0)) return 1; if (strcmp (command, "nick") == 0) return 2; if ((strcmp (command, "join") == 0) || (strcmp (command, "part") == 0) || (strcmp (command, "quit") == 0) || (strcmp (command, "nick_back") == 0)) return 4; return 3; } /* * Builds tags list with IRC command and optional tags and nick. */ const char * irc_protocol_tags (const char *command, const char *tags, const char *nick, const char *address) { static char string[1024]; int log_level; char str_log_level[32]; str_log_level[0] = '\0'; if (!command && !tags && !nick) return NULL; if (command && command[0]) { log_level = irc_protocol_log_level_for_command (command); if (log_level > 0) { snprintf (str_log_level, sizeof (str_log_level), ",log%d", log_level); } } snprintf (string, sizeof (string), "%s%s%s%s%s%s%s%s%s", (command && command[0]) ? "irc_" : "", (command && command[0]) ? command : "", (tags && tags[0]) ? "," : "", (tags && tags[0]) ? tags : "", (nick && nick[0]) ? ",nick_" : "", (nick && nick[0]) ? nick : "", (address && address[0]) ? ",host_" : "", (address && address[0]) ? address : "", str_log_level); return string; } /* * Builds a string with nick and optional address. * * If server_message is 1, the nick is colored according to option * irc.look.color_nicks_in_server_messages. * * Argument nickname is mandatory, address can be NULL. * If nickname and address are NULL, an empty string is returned. */ const char * irc_protocol_nick_address (struct t_irc_server *server, int server_message, struct t_irc_nick *nick, const char *nickname, const char *address) { static char string[1024]; string[0] = '\0'; if (nickname && address && (strcmp (nickname, address) != 0)) { /* display nick and address if they are different */ snprintf (string, sizeof (string), "%s%s %s(%s%s%s)%s", irc_nick_color_for_msg (server, server_message, nick, nickname), nickname, IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_HOST, address, IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET); } else if (nickname) { /* display only nick if no address or if nick == address */ snprintf (string, sizeof (string), "%s%s%s", irc_nick_color_for_msg (server, server_message, nick, nickname), nickname, IRC_COLOR_RESET); } return string; } /* * Parses date/time received in a "time" tag. * * Returns value of time (timestamp), 0 if error. */ time_t irc_protocol_parse_time (const char *time) { time_t time_value, time_msg, time_gm, time_local; struct tm tm_date, tm_date_gm, tm_date_local; long value; char *time2, *pos, *error; if (!time || !time[0]) return 0; time_value = 0; if (strchr (time, '-')) { /* date is with ISO 8601 format: "2012-11-24T07:41:02.018Z" */ /* initialize structure, because strptime does not do it */ memset (&tm_date, 0, sizeof (struct tm)); if (strptime (time, "%Y-%m-%dT%H:%M:%S", &tm_date)) { if (tm_date.tm_year > 0) { time_msg = mktime (&tm_date); gmtime_r (&time_msg, &tm_date_gm); localtime_r (&time_msg, &tm_date_local); time_gm = mktime (&tm_date_gm); time_local = mktime (&tm_date_local); time_value = mktime (&tm_date_local) + (time_local - time_gm); } } } else { /* date is with timestamp format: "1353403519.478" */ time2 = strdup (time); if (time2) { pos = strchr (time2, '.'); if (pos) pos[0] = '\0'; pos = strchr (time2, ','); if (pos) pos[0] = '\0'; value = strtol (time2, &error, 10); if (error && !error[0] && (value >= 0)) time_value = (int)value; free (time2); } } return time_value; } /* * Builds a string with concatenation of IRC command parameters, from * arg_start to arg_end. * * Note: result must be free after use. */ char * irc_protocol_string_params (const char **params, int arg_start, int arg_end) { int i, length; char *result; if (!params || (arg_start < 0) || (arg_end < arg_start)) return strdup (""); length = 0; for (i = 0; params[i]; i++) { if (i > arg_end) break; if (i >= arg_start) length += strlen (params[i]) + 1; } if (length == 0) return strdup (""); result = malloc (length + 1); if (!result) return NULL; result[0] = '\0'; for (i = arg_start; params[i]; i++) { if (i > arg_end) break; strcat (result, params[i]); if ((i + 1 <= arg_end) && params[i + 1]) strcat (result, " "); } return result; } /* * Prints a FAIL/WARN/NOTE command. * * Called by callbacks for commands: FAIL, WARN, NOTE. * * Commands looks like: * FAIL * NEED_REGISTRATION :You need to be registered to continue * FAIL ACC REG_INVALID_CALLBACK REGISTER :Email address is not valid * FAIL BOX BOXES_INVALID STACK CLOCKWISE :Given boxes are not supported * WARN REHASH CERTS_EXPIRED :Certificate [xxx] has expired * NOTE * OPER_MESSAGE :The message */ void irc_protocol_print_error_warning_msg (struct t_irc_server *server, time_t date, const char *command, const char *prefix, const char *label, const char **params, int num_params) { const char *ptr_command; char *str_context; ptr_command = (strcmp (params[0], "*") != 0) ? params[0] : NULL; str_context = (num_params > 2) ? irc_protocol_string_params (params, 1, num_params - 2) : NULL; weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, NULL, NULL, NULL), "%s%s%s%s%s%s[%s%s%s]%s %s", (prefix) ? prefix : "", (label) ? label : "", (label) ? " " : "", (ptr_command) ? ptr_command : "", (ptr_command) ? " " : "", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (str_context) ? str_context : "", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, params[num_params - 1]); if (str_context) free (str_context); } /* * Callback for the IRC command "ACCOUNT": account info about a nick * (with capability "account-notify"). * * Command looks like: * ACCOUNT * * ACCOUNT :accountname */ IRC_PROTOCOL_CALLBACK(account) { struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; struct t_irc_channel_speaking *ptr_nick_speaking; const char *pos_account; char str_account[512]; int cap_account_notify, local_account, smart_filter; IRC_PROTOCOL_MIN_PARAMS(1); local_account = (irc_server_strcasecmp (server, nick, server->nick) == 0); pos_account = params[0]; if (strcmp (pos_account, "*") == 0) pos_account = NULL; str_account[0] = '\0'; if (pos_account) { snprintf (str_account, sizeof (str_account), "%s%s", irc_nick_color_for_msg (server, 1, NULL, pos_account), pos_account); } cap_account_notify = weechat_hashtable_has_key (server->cap_list, "account-notify"); for (ptr_channel = server->channels; ptr_channel; ptr_channel = ptr_channel->next_channel) { ptr_nick = irc_nick_search (server, ptr_channel, nick); if (ptr_nick) { if (!ignored && weechat_config_boolean (irc_config_look_display_account_message)) { ptr_nick_speaking = ((weechat_config_boolean (irc_config_look_smart_filter)) && (weechat_config_boolean (irc_config_look_smart_filter_account))) ? irc_channel_nick_speaking_time_search (server, ptr_channel, nick, 1) : NULL; smart_filter = (!local_account && weechat_config_boolean (irc_config_look_smart_filter) && weechat_config_boolean (irc_config_look_smart_filter_account) && !ptr_nick_speaking); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags ( command, smart_filter ? "irc_smart_filter" : NULL, nick, address), (pos_account) ? _("%s%s%s%s has identified as %s") : _("%s%s%s%s has unidentified"), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_MESSAGE_ACCOUNT, (pos_account) ? str_account : NULL); } if (ptr_nick->account) free (ptr_nick->account); ptr_nick->account = (cap_account_notify && pos_account) ? strdup (pos_account) : NULL; } } return WEECHAT_RC_OK; } /* * Callback for the IRC command "AUTHENTICATE". * * Command looks like: * AUTHENTICATE + * AUTHENTICATE QQDaUzXAmVffxuzFy77XWBGwABBQAgdinelBrKZaR3wE7nsIETuTVY= */ IRC_PROTOCOL_CALLBACK(authenticate) { int sasl_mechanism; char *sasl_username, *sasl_password, *sasl_key, *answer; char *sasl_error; IRC_PROTOCOL_MIN_PARAMS(1); if (!irc_server_sasl_enabled (server)) return WEECHAT_RC_OK; irc_server_sasl_get_creds (server, &sasl_username, &sasl_password, &sasl_key); sasl_mechanism = IRC_SERVER_OPTION_INTEGER( server, IRC_SERVER_OPTION_SASL_MECHANISM); answer = NULL; sasl_error = NULL; switch (sasl_mechanism) { case IRC_SASL_MECHANISM_PLAIN: answer = irc_sasl_mechanism_plain (sasl_username, sasl_password); break; case IRC_SASL_MECHANISM_SCRAM_SHA_1: answer = irc_sasl_mechanism_scram ( server, "sha1", params[0], sasl_username, sasl_password, &sasl_error); break; case IRC_SASL_MECHANISM_SCRAM_SHA_256: answer = irc_sasl_mechanism_scram ( server, "sha256", params[0], sasl_username, sasl_password, &sasl_error); break; case IRC_SASL_MECHANISM_SCRAM_SHA_512: answer = irc_sasl_mechanism_scram ( server, "sha512", params[0], sasl_username, sasl_password, &sasl_error); break; case IRC_SASL_MECHANISM_ECDSA_NIST256P_CHALLENGE: answer = irc_sasl_mechanism_ecdsa_nist256p_challenge ( server, params[0], sasl_username, sasl_key, &sasl_error); break; case IRC_SASL_MECHANISM_EXTERNAL: answer = strdup ("+"); break; } if (answer) { if (sasl_error) { weechat_printf (server->buffer, _("%s%s: SASL error: %s"), weechat_prefix ("error"), IRC_PLUGIN_NAME, sasl_error); } irc_server_sendf (server, 0, NULL, "AUTHENTICATE %s", answer); free (answer); } else { weechat_printf (server->buffer, _("%s%s: SASL error: %s"), weechat_prefix ("error"), IRC_PLUGIN_NAME, (sasl_error) ? sasl_error : _("internal error")); irc_server_sendf (server, 0, NULL, "CAP END"); } if (sasl_username) free (sasl_username); if (sasl_password) free (sasl_password); if (sasl_key) free (sasl_key); if (sasl_error) free (sasl_error); return WEECHAT_RC_OK; } /* * Callback for the IRC command "AWAY": away info about a nick (with capability * "away-notify"). * * Command looks like: * AWAY * AWAY :I am away */ IRC_PROTOCOL_CALLBACK(away) { struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; IRC_PROTOCOL_MIN_PARAMS(0); IRC_PROTOCOL_CHECK_NICK; for (ptr_channel = server->channels; ptr_channel; ptr_channel = ptr_channel->next_channel) { ptr_nick = irc_nick_search (server, ptr_channel, nick); if (ptr_nick) { irc_nick_set_away (server, ptr_channel, ptr_nick, (num_params > 0)); } } return WEECHAT_RC_OK; } /* * Callback for IRC server capabilities string hashtable map. */ void irc_protocol_cap_print_cb (void *data, struct t_hashtable *hashtable, const char *key, const char *value) { char **str_caps; /* make C compiler happy */ (void) hashtable; str_caps = (char **)data; if (*str_caps[0]) weechat_string_dyn_concat (str_caps, " ", -1); weechat_string_dyn_concat (str_caps, key, -1); if (value) { weechat_string_dyn_concat (str_caps, "=", -1); weechat_string_dyn_concat (str_caps, value, -1); } } /* * Get capabilities to enable on the server (server option "capabilities" with * "sasl" if requested, "*" is replaced by all supported capabilities). * * Note: result must be freed after use. */ char * irc_protocol_cap_to_enable (const char *capabilities, int sasl_requested) { char **str_caps, **caps, *supported_caps; int i, num_caps; str_caps = weechat_string_dyn_alloc (128); if (!str_caps) return NULL; if (capabilities && capabilities[0]) { caps = weechat_string_split ( capabilities, ",", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &num_caps); if (caps) { for (i = 0; i < num_caps; i++) { if (strcmp (caps[i], "*") == 0) { supported_caps = weechat_string_replace ( IRC_COMMAND_CAP_SUPPORTED_COMPLETION, "|", ","); if (supported_caps) { if (*str_caps[0]) weechat_string_dyn_concat (str_caps, ",", -1); weechat_string_dyn_concat (str_caps, supported_caps, -1); free (supported_caps); } } else { if (*str_caps[0]) weechat_string_dyn_concat (str_caps, ",", -1); weechat_string_dyn_concat (str_caps, caps[i], -1); } } weechat_string_free_split (caps); } } if (sasl_requested) { if (*str_caps[0]) weechat_string_dyn_concat (str_caps, ",", -1); weechat_string_dyn_concat (str_caps, "sasl", -1); } return weechat_string_dyn_free (str_caps, 0); } /* * Synchronizes requested capabilities for IRC server. */ void irc_protocol_cap_sync (struct t_irc_server *server, int sasl) { char **caps_server, *caps_to_enable, **list_caps_to_enable, **cap_req; const char *ptr_cap_option; int sasl_requested, sasl_to_do, sasl_fail; int i, num_caps_server; sasl_requested = (sasl) ? irc_server_sasl_enabled (server) : 0; sasl_to_do = 0; ptr_cap_option = IRC_SERVER_OPTION_STRING( server, IRC_SERVER_OPTION_CAPABILITIES); cap_req = weechat_string_dyn_alloc (128); caps_to_enable = irc_protocol_cap_to_enable (ptr_cap_option, sasl_requested); list_caps_to_enable = weechat_string_split ( caps_to_enable, ",", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, NULL); if (list_caps_to_enable) { caps_server = weechat_string_split ( weechat_hashtable_get_string (server->cap_ls, "keys"), ",", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &num_caps_server); if (caps_server) { for (i = 0; i < num_caps_server; i++) { if (!weechat_hashtable_has_key (server->cap_list, caps_server[i]) && weechat_string_match_list (caps_server[i], (const char **)list_caps_to_enable, 0)) { if (sasl && strcmp (caps_server[i], "sasl") == 0) sasl_to_do = 1; if (*cap_req[0]) weechat_string_dyn_concat (cap_req, " ", -1); weechat_string_dyn_concat (cap_req, caps_server[i], -1); } } weechat_string_free_split (caps_server); } if (*cap_req[0]) { weechat_printf ( server->buffer, _("%s%s: client capability, requesting: %s"), weechat_prefix ("network"), IRC_PLUGIN_NAME, *cap_req); irc_server_sendf (server, 0, NULL, "CAP REQ :%s", *cap_req); } if (sasl) { if (!sasl_to_do) irc_server_sendf (server, 0, NULL, "CAP END"); if (sasl_requested && !sasl_to_do) { weechat_printf ( server->buffer, _("%s%s: client capability: SASL not supported"), weechat_prefix ("network"), IRC_PLUGIN_NAME); if (weechat_config_boolean (irc_config_network_sasl_fail_unavailable)) { /* same handling as for sasl_end_fail */ sasl_fail = IRC_SERVER_OPTION_INTEGER(server, IRC_SERVER_OPTION_SASL_FAIL); if ((sasl_fail == IRC_SERVER_SASL_FAIL_RECONNECT) || (sasl_fail == IRC_SERVER_SASL_FAIL_DISCONNECT)) { irc_server_disconnect ( server, 0, (sasl_fail == IRC_SERVER_SASL_FAIL_RECONNECT) ? 1 : 0); } } } } weechat_string_free_split (list_caps_to_enable); } if (caps_to_enable) free (caps_to_enable); if (cap_req) weechat_string_dyn_free (cap_req, 1); } /* * Callback for the IRC command "CAP": client capability. * * Command looks like: * CAP * LS :identify-msg multi-prefix sasl * CAP * ACK :sasl * CAP * NAK :sasl */ IRC_PROTOCOL_CALLBACK(cap) { char **caps_supported, **caps_added, **caps_removed; char **caps_enabled, *pos_value, *str_name, **str_caps; char str_msg_auth[512], **str_caps_enabled, **str_caps_disabled, *str_params; int arg_caps, num_caps_supported, num_caps_added, num_caps_removed; int num_caps_enabled, sasl_to_do, sasl_mechanism; int i, j, timeout, last_reply; IRC_PROTOCOL_MIN_PARAMS(2); if (strcmp (params[1], "LS") == 0) { if (num_params < 3) return WEECHAT_RC_OK; if ((num_params > 3) && (strcmp (params[2], "*") == 0)) { arg_caps = 3; last_reply = 0; } else { arg_caps = 2; last_reply = 1; } if (!server->checking_cap_ls) { weechat_hashtable_remove_all (server->cap_ls); server->checking_cap_ls = 1; } if (last_reply) server->checking_cap_ls = 0; for (i = arg_caps; i < num_params; i++) { caps_supported = weechat_string_split ( params[i], " ", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &num_caps_supported); if (!caps_supported) continue; for (j = 0; j < num_caps_supported; j++) { pos_value = strstr (caps_supported[j], "="); if (pos_value) { str_name = strndup (caps_supported[j], pos_value - caps_supported[j]); if (str_name) { weechat_hashtable_set (server->cap_ls, str_name, pos_value + 1); free (str_name); } } else { weechat_hashtable_set (server->cap_ls, caps_supported[j], NULL); } } weechat_string_free_split (caps_supported); } if (last_reply) { str_caps = weechat_string_dyn_alloc (128); weechat_hashtable_map_string (server->cap_ls, irc_protocol_cap_print_cb, str_caps); weechat_printf_date_tags ( server->buffer, date, NULL, _("%s%s: client capability, server supports: %s"), weechat_prefix ("network"), IRC_PLUGIN_NAME, *str_caps); weechat_string_dyn_free (str_caps, 1); } /* auto-enable capabilities only when connecting to server */ if (last_reply && !server->is_connected) irc_protocol_cap_sync (server, 1); } else if (strcmp (params[1], "LIST") == 0) { if (num_params < 3) return WEECHAT_RC_OK; if ((num_params > 3) && (strcmp (params[2], "*") == 0)) { arg_caps = 3; last_reply = 0; } else { arg_caps = 2; last_reply = 1; } if (!server->checking_cap_list) { weechat_hashtable_remove_all (server->cap_list); server->checking_cap_list = 1; } if (last_reply) server->checking_cap_list = 0; for (i = arg_caps; i < num_params; i++) { caps_enabled = weechat_string_split ( params[i], " ", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &num_caps_enabled); if (!caps_enabled) continue; for (j = 0; j < num_caps_enabled; j++) { pos_value = strstr (caps_enabled[j], "="); if (pos_value) { str_name = strndup (caps_enabled[j], pos_value - caps_enabled[j]); if (str_name) { weechat_hashtable_set (server->cap_list, str_name, pos_value + 1); free (str_name); } } else { weechat_hashtable_set (server->cap_list, caps_enabled[j], NULL); } } weechat_string_free_split (caps_enabled); } if (last_reply) { str_caps = weechat_string_dyn_alloc (128); weechat_hashtable_map_string (server->cap_list, irc_protocol_cap_print_cb, str_caps); weechat_printf_date_tags ( server->buffer, date, NULL, _("%s%s: client capability, currently enabled: %s"), weechat_prefix ("network"), IRC_PLUGIN_NAME, *str_caps); weechat_string_dyn_free (str_caps, 1); } } else if (strcmp (params[1], "ACK") == 0) { if (num_params < 3) return WEECHAT_RC_OK; sasl_to_do = 0; str_caps_enabled = weechat_string_dyn_alloc (128); str_caps_disabled = weechat_string_dyn_alloc (128); for (i = 2; i < num_params; i++) { caps_supported = weechat_string_split ( params[i], " ", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &num_caps_supported); if (!caps_supported) continue; for (j = 0; j < num_caps_supported; j++) { if (caps_supported[j][0] == '-') { if (*str_caps_disabled[0]) weechat_string_dyn_concat (str_caps_disabled, " ", -1); weechat_string_dyn_concat (str_caps_disabled, caps_supported[j] + 1, -1); weechat_hashtable_remove (server->cap_list, caps_supported[j] + 1); } else { if (*str_caps_enabled[0]) weechat_string_dyn_concat (str_caps_enabled, " ", -1); weechat_string_dyn_concat (str_caps_enabled, caps_supported[j], -1); weechat_hashtable_set (server->cap_list, caps_supported[j], NULL); if (strcmp (caps_supported[j], "sasl") == 0) sasl_to_do = 1; } } weechat_string_free_split (caps_supported); } if (*str_caps_enabled[0] && *str_caps_disabled[0]) { weechat_printf_date_tags ( server->buffer, date, NULL, _("%s%s: client capability, enabled: %s, disabled: %s"), weechat_prefix ("network"), IRC_PLUGIN_NAME, *str_caps_enabled, *str_caps_disabled); } else if (*str_caps_enabled[0]) { weechat_printf_date_tags ( server->buffer, date, NULL, _("%s%s: client capability, enabled: %s"), weechat_prefix ("network"), IRC_PLUGIN_NAME, *str_caps_enabled); } else if (*str_caps_disabled[0]) { weechat_printf_date_tags ( server->buffer, date, NULL, _("%s%s: client capability, disabled: %s"), weechat_prefix ("network"), IRC_PLUGIN_NAME, *str_caps_disabled); } weechat_string_dyn_free (str_caps_enabled, 1); weechat_string_dyn_free (str_caps_disabled, 1); if (sasl_to_do) { sasl_mechanism = IRC_SERVER_OPTION_INTEGER( server, IRC_SERVER_OPTION_SASL_MECHANISM); if ((sasl_mechanism >= 0) && (sasl_mechanism < IRC_NUM_SASL_MECHANISMS)) { snprintf (str_msg_auth, sizeof (str_msg_auth), "AUTHENTICATE %s", irc_sasl_mechanism_string[sasl_mechanism]); weechat_string_toupper (str_msg_auth); irc_server_sendf (server, 0, NULL, str_msg_auth); if (server->hook_timer_sasl) weechat_unhook (server->hook_timer_sasl); timeout = IRC_SERVER_OPTION_INTEGER( server, IRC_SERVER_OPTION_SASL_TIMEOUT); server->hook_timer_sasl = weechat_hook_timer ( timeout * 1000, 0, 1, &irc_server_timer_sasl_cb, server, NULL); } } } else if (strcmp (params[1], "NAK") == 0) { if (num_params < 3) return WEECHAT_RC_OK; str_params = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( server->buffer, date, NULL, _("%s%s: client capability, refused: %s"), weechat_prefix ("error"), IRC_PLUGIN_NAME, str_params); if (str_params) free (str_params); if (!server->is_connected) irc_server_sendf (server, 0, NULL, "CAP END"); } else if (strcmp (params[1], "NEW") == 0) { if (num_params < 3) return WEECHAT_RC_OK; str_params = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( server->buffer, date, NULL, _("%s%s: client capability, now available: %s"), weechat_prefix ("network"), IRC_PLUGIN_NAME, str_params); if (str_params) free (str_params); for (i = 2; i < num_params; i++) { caps_added = weechat_string_split ( params[i], " ", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &num_caps_added); if (!caps_added) continue; for (j = 0; j < num_caps_added; j++) { pos_value = strstr (caps_added[j], "="); if (pos_value) { str_name = strndup (caps_added[j], pos_value - caps_added[j]); if (str_name) { weechat_hashtable_set (server->cap_ls, str_name, pos_value + 1); free (str_name); } } else { weechat_hashtable_set (server->cap_ls, caps_added[j], NULL); } } weechat_string_free_split (caps_added); } /* TODO: SASL Reauthentication */ irc_protocol_cap_sync (server, 0); } else if (strcmp (params[1], "DEL") == 0) { if (num_params < 3) return WEECHAT_RC_OK; str_params = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( server->buffer, date, NULL, _("%s%s: client capability, removed: %s"), weechat_prefix ("network"), IRC_PLUGIN_NAME, str_params); if (str_params) free (str_params); for (i = 2; i < num_params; i++) { caps_removed = weechat_string_split ( params[i], " ", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &num_caps_removed); if (!caps_removed) continue; for (j = 0; j < num_caps_removed; j++) { weechat_hashtable_remove (server->cap_ls, caps_removed[j]); weechat_hashtable_remove (server->cap_list, caps_removed[j]); } weechat_string_free_split (caps_removed); } } return WEECHAT_RC_OK; } /* * Callback for the IRC command "CHGHOST": user/host change of a nick (with * capability "chghost"): * https://ircv3.net/specs/extensions/chghost-3.2.html * * Command looks like: * CHGHOST user new.host.goes.here * CHGHOST newuser host * CHGHOST newuser new.host.goes.here * CHGHOST newuser :new.host.goes.here */ IRC_PROTOCOL_CALLBACK(chghost) { int length, local_chghost, smart_filter; char *str_host; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; struct t_irc_channel_speaking *ptr_nick_speaking; IRC_PROTOCOL_MIN_PARAMS(2); IRC_PROTOCOL_CHECK_NICK; local_chghost = (irc_server_strcasecmp (server, nick, server->nick) == 0); length = strlen (params[0]) + 1 + strlen (params[1]) + 1; str_host = malloc (length); if (!str_host) { weechat_printf ( server->buffer, _("%s%s: not enough memory for \"%s\" command"), weechat_prefix ("error"), IRC_PLUGIN_NAME, "chghost"); return WEECHAT_RC_OK; } snprintf (str_host, length, "%s@%s", params[0], params[1]); if (local_chghost) irc_server_set_host (server, str_host); for (ptr_channel = server->channels; ptr_channel; ptr_channel = ptr_channel->next_channel) { ptr_nick = irc_nick_search (server, ptr_channel, nick); if (ptr_nick) { if (!ignored) { ptr_nick_speaking = ((weechat_config_boolean (irc_config_look_smart_filter)) && (weechat_config_boolean (irc_config_look_smart_filter_chghost))) ? irc_channel_nick_speaking_time_search (server, ptr_channel, nick, 1) : NULL; smart_filter = (!local_chghost && weechat_config_boolean (irc_config_look_smart_filter) && weechat_config_boolean (irc_config_look_smart_filter_chghost) && !ptr_nick_speaking); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags ( command, smart_filter ? "irc_smart_filter" : NULL, nick, address), _("%s%s%s%s (%s%s%s)%s has changed host to %s%s"), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_HOST, (address) ? address : "", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_MESSAGE_CHGHOST, IRC_COLOR_CHAT_HOST, str_host); } irc_nick_set_host (ptr_nick, str_host); } } free (str_host); return WEECHAT_RC_OK; } /* * Callback for the IRC command "ERROR". * * Command looks like: * ERROR :Closing Link: irc.server.org (Bad Password) */ IRC_PROTOCOL_CALLBACK(error) { char *str_error; IRC_PROTOCOL_MIN_PARAMS(1); str_error = irc_protocol_string_params (params, 0, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, NULL, NULL, NULL), "%s%s", weechat_prefix ("error"), str_error); if (str_error && (strncmp (str_error, "Closing Link", 12) == 0)) irc_server_disconnect (server, !server->is_connected, 1); if (str_error) free (str_error); return WEECHAT_RC_OK; } /* * Callback for an IRC error command (used by many error messages, but not for * command "ERROR"). * * Command looks like: * 404 nick #channel :Cannot send to channel */ IRC_PROTOCOL_CALLBACK(generic_error) { int arg_error; char *str_error; const char *pos_chan_nick; struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_PARAMS(2); arg_error = (irc_server_strcasecmp (server, params[0], server->nick) == 0) ? 1 : 0; pos_chan_nick = NULL; if (params[arg_error + 1] && irc_channel_is_channel (server, params[arg_error])) { pos_chan_nick = params[arg_error]; arg_error++; } ptr_channel = NULL; if (pos_chan_nick) ptr_channel = irc_channel_search (server, pos_chan_nick); ptr_buffer = (ptr_channel) ? ptr_channel->buffer : server->buffer; str_error = irc_protocol_string_params (params, arg_error, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, ((strcmp (command, "401") == 0) || (strcmp (command, "402") == 0)) ? "whois" : NULL, ptr_buffer), date, irc_protocol_tags (command, NULL, NULL, NULL), "%s%s%s%s%s%s", weechat_prefix ("network"), (ptr_channel && pos_chan_nick && (irc_server_strcasecmp (server, pos_chan_nick, ptr_channel->name) == 0)) ? IRC_COLOR_CHAT_CHANNEL : "", (pos_chan_nick) ? pos_chan_nick : "", IRC_COLOR_RESET, (pos_chan_nick) ? ": " : "", str_error); if (str_error) free (str_error); return WEECHAT_RC_OK; } /* * Callback for the IRC command "FAIL". * * Command looks like: * FAIL * NEED_REGISTRATION :You need to be registered to continue * FAIL ACC REG_INVALID_CALLBACK REGISTER :Email address is not valid * FAIL BOX BOXES_INVALID STACK CLOCKWISE :Given boxes are not supported */ IRC_PROTOCOL_CALLBACK(fail) { IRC_PROTOCOL_MIN_PARAMS(2); irc_protocol_print_error_warning_msg ( server, date, command, weechat_prefix ("error"), _("Failure:"), params, num_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "INVITE". * * Command looks like: * INVITE mynick :#channel * * With invite-notify capability, the whole message looks like: * : INVITE * :ChanServ!ChanServ@example.com INVITE Attila #channel * * For more information, see: * https://ircv3.net/specs/extensions/invite-notify-3.2.html */ IRC_PROTOCOL_CALLBACK(invite) { IRC_PROTOCOL_MIN_PARAMS(2); IRC_PROTOCOL_CHECK_NICK; if (ignored) return WEECHAT_RC_OK; if (irc_server_strcasecmp (server, params[0], server->nick) == 0) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, nick, command, NULL, NULL), date, irc_protocol_tags (command, "notify_highlight", nick, address), _("%sYou have been invited to %s%s%s by %s%s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_RESET, irc_nick_color_for_msg (server, 1, NULL, nick), nick, IRC_COLOR_RESET); } else { /* CAP invite-notify */ /* imitate numeric 341 output */ weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, nick, command, NULL, NULL), date, irc_protocol_tags (command, NULL, nick, address), _("%s%s%s%s has invited %s%s%s to %s%s%s"), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, NULL, nick), nick, IRC_COLOR_RESET, irc_nick_color_for_msg (server, 1, NULL, params[0]), params[0], IRC_COLOR_RESET, IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_RESET); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "JOIN". * * Command looks like: * JOIN #channel * JOIN :#channel * * With extended-join capability: * JOIN #channel * :real name * JOIN #channel account :real name */ IRC_PROTOCOL_CALLBACK(join) { struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; struct t_irc_channel_speaking *ptr_nick_speaking; const char *pos_account, *pos_realname; char str_account[512], str_realname[512]; int local_join, display_host, smart_filter; IRC_PROTOCOL_MIN_PARAMS(1); IRC_PROTOCOL_CHECK_NICK; local_join = (irc_server_strcasecmp (server, nick, server->nick) == 0); pos_account = ((num_params > 1) && (strcmp (params[1], "*") != 0)) ? params[1] : NULL; pos_realname = (num_params > 2) ? params[2] : NULL; str_account[0] = '\0'; if (pos_account && weechat_config_boolean (irc_config_look_display_extended_join)) { snprintf (str_account, sizeof (str_account), "%s [%s%s%s]", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_HOST, pos_account, IRC_COLOR_CHAT_DELIMITERS); } str_realname[0] = '\0'; if (pos_realname && weechat_config_boolean (irc_config_look_display_extended_join)) { snprintf (str_realname, sizeof (str_realname), "%s (%s%s%s)", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_HOST, pos_realname, IRC_COLOR_CHAT_DELIMITERS); } ptr_channel = irc_channel_search (server, params[0]); if (ptr_channel) { ptr_channel->part = 0; } else { /* * if someone else joins and channel is not opened, then just * ignore it (we should receive our self join first) */ if (!local_join) return WEECHAT_RC_OK; ptr_channel = irc_channel_new (server, IRC_CHANNEL_TYPE_CHANNEL, params[0], 1, 1); if (!ptr_channel) { weechat_printf (server->buffer, _("%s%s: cannot create new channel \"%s\""), weechat_prefix ("error"), IRC_PLUGIN_NAME, params[0]); return WEECHAT_RC_ERROR; } } /* * local join? clear nicklist to be sure it is empty (when using znc, after * reconnection to network, we receive a JOIN for channel with existing * nicks in irc plugin, so we need to clear the nicklist now) */ if (local_join) irc_nick_free_all (server, ptr_channel); /* reset some variables if joining new channel */ if (!ptr_channel->nicks) { irc_channel_set_topic (ptr_channel, NULL); if (ptr_channel->modes) { free (ptr_channel->modes); ptr_channel->modes = NULL; } ptr_channel->limit = 0; weechat_hashtable_remove_all (ptr_channel->join_msg_received); ptr_channel->checking_whox = 0; } /* add nick in channel */ ptr_nick = irc_nick_new (server, ptr_channel, nick, address, NULL, 0, (pos_account) ? pos_account : NULL, (pos_realname) ? pos_realname : NULL); /* rename the nick if it was in list with a different case */ irc_channel_nick_speaking_rename_if_present (server, ptr_channel, nick); if (!ignored) { ptr_nick_speaking = ((weechat_config_boolean (irc_config_look_smart_filter)) && (weechat_config_boolean (irc_config_look_smart_filter_join))) ? irc_channel_nick_speaking_time_search (server, ptr_channel, nick, 1) : NULL; display_host = (local_join) ? weechat_config_boolean (irc_config_look_display_host_join_local) : weechat_config_boolean (irc_config_look_display_host_join); /* * "smart" filter the join message if it's not a join from myself, if * smart filtering is enabled, and if nick was not speaking in channel */ smart_filter = (!local_join && weechat_config_boolean (irc_config_look_smart_filter) && weechat_config_boolean (irc_config_look_smart_filter_join) && !ptr_nick_speaking); /* display the join */ weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, smart_filter ? "irc_smart_filter" : NULL, nick, address), _("%s%s%s%s%s%s%s%s%s%s%s%s has joined %s%s%s"), weechat_prefix ("join"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, str_account, str_realname, IRC_COLOR_CHAT_DELIMITERS, (display_host) ? " (" : "", IRC_COLOR_CHAT_HOST, (display_host) ? address : "", IRC_COLOR_CHAT_DELIMITERS, (display_host) ? ")" : "", IRC_COLOR_MESSAGE_JOIN, IRC_COLOR_CHAT_CHANNEL, params[0], IRC_COLOR_MESSAGE_JOIN); /* * if join is smart filtered, save the nick in hashtable, and if nick * is speaking shortly after the join, it will be unmasked * (option irc.look.smart_filter_join_unmask) */ if (smart_filter) { irc_channel_join_smart_filtered_add (ptr_channel, nick, time (NULL)); } /* display message in private if private has flag "has_quit_server" */ if (!local_join) { irc_channel_display_nick_back_in_pv (server, ptr_nick, nick); irc_channel_set_topic_private_buffers (server, ptr_nick, nick, address); } } if (local_join) { irc_server_set_host (server, address); irc_bar_item_update_channel (); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "KICK". * * Command looks like: * KICK #channel nick :kick reason */ IRC_PROTOCOL_CALLBACK(kick) { const char *pos_comment, *ptr_autorejoin; int rejoin; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick, *ptr_nick_kicked; IRC_PROTOCOL_MIN_PARAMS(2); IRC_PROTOCOL_CHECK_NICK; pos_comment = (num_params > 2) ? params[2] : NULL; ptr_channel = irc_channel_search (server, params[0]); if (!ptr_channel) return WEECHAT_RC_OK; ptr_nick = irc_nick_search (server, ptr_channel, nick); ptr_nick_kicked = irc_nick_search (server, ptr_channel, params[1]); if (pos_comment) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, NULL, NULL, address), _("%s%s%s%s has kicked %s%s%s %s(%s%s%s)"), weechat_prefix ("quit"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_MESSAGE_KICK, irc_nick_color_for_msg (server, 1, ptr_nick_kicked, params[1]), params[1], IRC_COLOR_MESSAGE_KICK, IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_REASON_KICK, pos_comment, IRC_COLOR_CHAT_DELIMITERS); } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, NULL, NULL, address), _("%s%s%s%s has kicked %s%s%s"), weechat_prefix ("quit"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_MESSAGE_KICK, irc_nick_color_for_msg (server, 1, ptr_nick_kicked, params[1]), params[1], IRC_COLOR_MESSAGE_KICK); } if (irc_server_strcasecmp (server, params[1], server->nick) == 0) { /* * my nick was kicked => free all nicks, channel is not active any * more */ irc_nick_free_all (server, ptr_channel); irc_channel_modelist_set_state (ptr_channel, IRC_MODELIST_STATE_MODIFIED); /* read option "autorejoin" in server */ rejoin = IRC_SERVER_OPTION_BOOLEAN(server, IRC_SERVER_OPTION_AUTOREJOIN); /* * if buffer has a local variable "autorejoin", use it * (it has higher priority than server option */ ptr_autorejoin = weechat_buffer_get_string (ptr_channel->buffer, "localvar_autorejoin"); if (ptr_autorejoin) rejoin = weechat_config_string_to_boolean (ptr_autorejoin); if (rejoin) { if (IRC_SERVER_OPTION_INTEGER(server, IRC_SERVER_OPTION_AUTOREJOIN_DELAY) == 0) { /* immediately rejoin if delay is 0 */ irc_channel_rejoin (server, ptr_channel); } else { /* rejoin channel later, according to delay */ ptr_channel->hook_autorejoin = weechat_hook_timer ( IRC_SERVER_OPTION_INTEGER(server, IRC_SERVER_OPTION_AUTOREJOIN_DELAY) * 1000, 0, 1, &irc_channel_autorejoin_cb, ptr_channel, NULL); } } irc_bar_item_update_channel (); } else { /* * someone was kicked from channel (but not me) => remove only this * nick */ if (ptr_nick_kicked) irc_nick_free (server, ptr_channel, ptr_nick_kicked); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "KILL". * * Command looks like: * KILL nick :kill reason */ IRC_PROTOCOL_CALLBACK(kill) { const char *pos_comment; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick, *ptr_nick_killed; IRC_PROTOCOL_MIN_PARAMS(1); IRC_PROTOCOL_CHECK_NICK; pos_comment = (num_params > 1) ? params[1] : NULL; for (ptr_channel = server->channels; ptr_channel; ptr_channel = ptr_channel->next_channel) { ptr_nick = irc_nick_search (server, ptr_channel, nick); ptr_nick_killed = irc_nick_search (server, ptr_channel, params[0]); if (pos_comment) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, NULL, NULL, address), _("%s%sYou were killed by %s%s%s %s(%s%s%s)"), weechat_prefix ("quit"), IRC_COLOR_MESSAGE_KICK, irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_MESSAGE_KICK, IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_REASON_KICK, pos_comment, IRC_COLOR_CHAT_DELIMITERS); } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, NULL, NULL, address), _("%s%sYou were killed by %s%s%s"), weechat_prefix ("quit"), IRC_COLOR_MESSAGE_KICK, irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_MESSAGE_KICK); } if (irc_server_strcasecmp (server, params[0], server->nick) == 0) { /* * my nick was killed => free all nicks, channel is not active any * more */ irc_nick_free_all (server, ptr_channel); irc_channel_modelist_set_state (ptr_channel, IRC_MODELIST_STATE_MODIFIED); irc_bar_item_update_channel (); } else { /* * someone was killed on channel (but not me) => remove only this * nick */ if (ptr_nick_killed) irc_nick_free (server, ptr_channel, ptr_nick_killed); } } return WEECHAT_RC_OK; } /* * Callback for the IRC command "MODE". * * Command looks like: * MODE #test +nt * MODE #test +o nick * MODE #test +o :nick */ IRC_PROTOCOL_CALLBACK(mode) { char *msg_modes_args, *modes_args; int smart_filter, local_mode; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_PARAMS(2); IRC_PROTOCOL_CHECK_NICK; msg_modes_args = (num_params > 2) ? irc_protocol_string_params (params, 2, num_params - 1) : NULL; if (irc_channel_is_channel (server, params[0])) { smart_filter = 0; ptr_channel = irc_channel_search (server, params[0]); if (ptr_channel) { smart_filter = irc_mode_channel_set (server, ptr_channel, host, params[1], msg_modes_args); } local_mode = (irc_server_strcasecmp (server, nick, server->nick) == 0); ptr_nick = irc_nick_search (server, ptr_channel, nick); ptr_buffer = (ptr_channel) ? ptr_channel->buffer : server->buffer; modes_args = irc_mode_get_arguments (msg_modes_args); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, ptr_buffer), date, irc_protocol_tags (command, (smart_filter && !local_mode) ? "irc_smart_filter" : NULL, NULL, address), _("%sMode %s%s %s[%s%s%s%s%s]%s by %s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, (ptr_channel) ? ptr_channel->name : params[0], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, params[1], (modes_args && modes_args[0]) ? " " : "", (modes_args && modes_args[0]) ? modes_args : "", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick); if (modes_args) free (modes_args); } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, NULL, NULL, address), _("%sUser mode %s[%s%s%s]%s by %s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, irc_nick_color_for_msg (server, 1, NULL, nick), nick); irc_mode_user_set (server, params[1], 0); } if (msg_modes_args) free (msg_modes_args); return WEECHAT_RC_OK; } /* * Callback for the IRC command "NICK". * * Command looks like: * NICK :newnick */ IRC_PROTOCOL_CALLBACK(nick) { struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick, *ptr_nick_found; char *old_color, str_tags[512]; const char *buffer_name; int local_nick, smart_filter; struct t_irc_channel_speaking *ptr_nick_speaking; IRC_PROTOCOL_MIN_PARAMS(1); IRC_PROTOCOL_CHECK_NICK; if (!params[0][0]) return WEECHAT_RC_OK; local_nick = (irc_server_strcasecmp (server, nick, server->nick) == 0) ? 1 : 0; if (local_nick) { irc_server_set_nick (server, params[0]); irc_server_set_host (server, address); } ptr_nick_found = NULL; /* first display message in server buffer if it's local nick */ if (local_nick) { /* temporary disable hotlist */ weechat_buffer_set (NULL, "hotlist", "-"); snprintf (str_tags, sizeof (str_tags), "irc_nick1_%s,irc_nick2_%s", nick, params[0]); weechat_printf_date_tags ( server->buffer, date, irc_protocol_tags (command, str_tags, NULL, address), _("%sYou are now known as %s%s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_NICK_SELF, params[0], IRC_COLOR_RESET); /* enable hotlist */ weechat_buffer_set (NULL, "hotlist", "+"); } for (ptr_channel = server->channels; ptr_channel; ptr_channel = ptr_channel->next_channel) { switch (ptr_channel->type) { case IRC_CHANNEL_TYPE_PRIVATE: /* rename private window if this is with "old nick" */ if ((irc_server_strcasecmp (server, ptr_channel->name, nick) == 0) && !irc_channel_search (server, params[0])) { free (ptr_channel->name); ptr_channel->name = strdup (params[0]); if (ptr_channel->pv_remote_nick_color) { free (ptr_channel->pv_remote_nick_color); ptr_channel->pv_remote_nick_color = NULL; } buffer_name = irc_buffer_build_name (server->name, ptr_channel->name); weechat_buffer_set (ptr_channel->buffer, "name", buffer_name); weechat_buffer_set (ptr_channel->buffer, "short_name", ptr_channel->name); weechat_buffer_set (ptr_channel->buffer, "localvar_set_channel", ptr_channel->name); } break; case IRC_CHANNEL_TYPE_CHANNEL: /* rename nick in nicklist if found */ ptr_nick = irc_nick_search (server, ptr_channel, nick); if (ptr_nick) { ptr_nick_found = ptr_nick; /* set host in nick if needed */ irc_nick_set_host (ptr_nick, address); /* change nick and display message on channel */ old_color = strdup (ptr_nick->color); irc_nick_change (server, ptr_channel, ptr_nick, params[0]); if (local_nick) { /* temporary disable hotlist */ weechat_buffer_set (NULL, "hotlist", "-"); snprintf (str_tags, sizeof (str_tags), "irc_nick1_%s,irc_nick2_%s", nick, params[0]); weechat_printf_date_tags (ptr_channel->buffer, date, irc_protocol_tags (command, str_tags, NULL, address), _("%sYou are now known as " "%s%s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_NICK_SELF, params[0], IRC_COLOR_RESET); /* enable hotlist */ weechat_buffer_set (NULL, "hotlist", "+"); } else { if (!irc_ignore_check (server, ptr_channel->name, nick, host)) { ptr_nick_speaking = ((weechat_config_boolean (irc_config_look_smart_filter)) && (weechat_config_boolean (irc_config_look_smart_filter_nick))) ? irc_channel_nick_speaking_time_search (server, ptr_channel, nick, 1) : NULL; smart_filter = (weechat_config_boolean (irc_config_look_smart_filter) && weechat_config_boolean (irc_config_look_smart_filter_nick) && !ptr_nick_speaking); snprintf (str_tags, sizeof (str_tags), "%sirc_nick1_%s,irc_nick2_%s", (smart_filter) ? "irc_smart_filter," : "", nick, params[0]); weechat_printf_date_tags ( ptr_channel->buffer, date, irc_protocol_tags (command, str_tags, NULL, address), _("%s%s%s%s is now known as %s%s%s"), weechat_prefix ("network"), weechat_config_boolean (irc_config_look_color_nicks_in_server_messages) ? old_color : IRC_COLOR_CHAT_NICK, nick, IRC_COLOR_RESET, irc_nick_color_for_msg (server, 1, ptr_nick, params[0]), params[0], IRC_COLOR_RESET); } irc_channel_nick_speaking_rename (ptr_channel, nick, params[0]); irc_channel_nick_speaking_time_rename (server, ptr_channel, nick, params[0]); irc_channel_join_smart_filtered_rename (ptr_channel, nick, params[0]); } if (old_color) free (old_color); } break; } } if (!local_nick) { irc_channel_display_nick_back_in_pv (server, ptr_nick_found, params[0]); irc_channel_set_topic_private_buffers (server, ptr_nick_found, params[0], address); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "NOTE". * * Command looks like: * NOTE * OPER_MESSAGE :The message */ IRC_PROTOCOL_CALLBACK(note) { IRC_PROTOCOL_MIN_PARAMS(2); irc_protocol_print_error_warning_msg ( server, date, command, weechat_prefix ("network"), _("Note:"), params, num_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "NOTICE". * * Command looks like: * NOTICE mynick :notice text * NOTICE #channel :notice text * NOTICE @#channel :notice text for channel ops */ IRC_PROTOCOL_CALLBACK(notice) { char *notice_args, *pos, end_char, *channel, status_notice[2]; const char *pos_target, *pos_args, *nick_address; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; int notify_private, is_channel, is_channel_orig; struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_PARAMS(2); if (ignored) return WEECHAT_RC_OK; status_notice[0] = '\0'; status_notice[1] = '\0'; notice_args = irc_protocol_string_params (params, 1, num_params - 1); if (!notice_args) return WEECHAT_RC_ERROR; pos_args = notice_args; pos_target = params[0]; is_channel = irc_channel_is_channel (server, pos_target + 1); if (is_channel && irc_server_prefix_char_statusmsg (server, pos_target[0])) { status_notice[0] = pos_target[0]; pos_target++; } if (nick && (pos_args[0] == '\01')) { irc_ctcp_display_reply_from_nick (server, date, command, nick, address, pos_args); } else { channel = NULL; is_channel = irc_channel_is_channel (server, pos_target); is_channel_orig = is_channel; if (is_channel) { channel = strdup (pos_target); } else if (weechat_config_boolean (irc_config_look_notice_welcome_redirect)) { end_char = ' '; switch (pos_args[0]) { case '[': end_char = ']'; break; case '(': end_char = ')'; break; case '{': end_char = '}'; break; case '<': end_char = '>'; break; } if (end_char != ' ') { pos = strchr (pos_args, end_char); if (pos && (pos > pos_args + 1)) { channel = weechat_strndup (pos_args + 1, pos - pos_args - 1); if (channel && irc_channel_search (server, channel)) { is_channel = 1; pos_args = pos + 1; while (pos_args[0] == ' ') { pos_args++; } } } } } if (is_channel) { /* notice for channel */ ptr_channel = irc_channel_search (server, channel); /* * unmask a smart filtered join if it is in hashtable * "join_smart_filtered" of channel */ if (ptr_channel) irc_channel_join_smart_filtered_unmask (ptr_channel, nick); if (ptr_channel && weechat_config_boolean (irc_config_look_typing_status_nicks)) { irc_typing_channel_set_nick (ptr_channel, nick, IRC_CHANNEL_TYPING_STATE_OFF); } ptr_nick = irc_nick_search (server, ptr_channel, nick); weechat_printf_date_tags ( (ptr_channel) ? ptr_channel->buffer : server->buffer, date, irc_protocol_tags (command, (is_channel_orig) ? "notify_message" : weechat_config_string (irc_config_look_notice_welcome_tags), nick, address), "%s%s%s%s%s%s%s(%s%s%s)%s: %s", weechat_prefix ("network"), IRC_COLOR_NOTICE, (is_channel_orig) ? "" : "Pv", /* TRANSLATORS: "Notice" is command name in IRC protocol (translation is frequently the same word) */ _("Notice"), (status_notice[0]) ? ":" : "", status_notice, IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 0, ptr_nick, nick), (nick && nick[0]) ? nick : "?", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, pos_args); } else { /* notice for user */ notify_private = 0; if (server->is_connected && nick && (weechat_strcasecmp (nick, "nickserv") != 0) && (weechat_strcasecmp (nick, "chanserv") != 0) && (weechat_strcasecmp (nick, "memoserv") != 0)) { /* * add tag "notify_private" only if: * - server is connected (message 001 already received) * and: * - notice is from a non-empty nick different from * nickserv/chanserv/memoserv */ notify_private = 1; } ptr_channel = NULL; if (nick && weechat_config_integer (irc_config_look_notice_as_pv) != IRC_CONFIG_LOOK_NOTICE_AS_PV_NEVER) { ptr_channel = irc_channel_search (server, nick); if (!ptr_channel && weechat_config_integer (irc_config_look_notice_as_pv) == IRC_CONFIG_LOOK_NOTICE_AS_PV_ALWAYS) { ptr_channel = irc_channel_new (server, IRC_CHANNEL_TYPE_PRIVATE, nick, 0, 0); if (!ptr_channel) { weechat_printf (server->buffer, _("%s%s: cannot create new " "private buffer \"%s\""), weechat_prefix ("error"), IRC_PLUGIN_NAME, nick); } } } if (ptr_channel) { if (weechat_config_boolean (irc_config_look_typing_status_nicks)) { irc_typing_channel_set_nick (ptr_channel, nick, IRC_CHANNEL_TYPING_STATE_OFF); } if (!ptr_channel->topic) irc_channel_set_topic (ptr_channel, address); weechat_printf_date_tags ( ptr_channel->buffer, date, irc_protocol_tags (command, "notify_private", nick, address), "%s%s%s%s: %s", weechat_prefix ("network"), irc_nick_color_for_msg (server, 0, NULL, nick), nick, IRC_COLOR_RESET, pos_args); if ((ptr_channel->type == IRC_CHANNEL_TYPE_PRIVATE) && ptr_channel->has_quit_server) { ptr_channel->has_quit_server = 0; } } else { ptr_buffer = irc_msgbuffer_get_target_buffer (server, nick, command, NULL, NULL); /* * if notice is sent from myself (for example another WeeChat * via relay), then display message of outgoing notice */ if (nick && (irc_server_strcasecmp (server, server->nick, nick) == 0)) { weechat_printf_date_tags ( ptr_buffer, date, irc_protocol_tags (command, (notify_private) ? "notify_private" : NULL, server->nick, address), "%s%s%s%s -> %s%s%s: %s", weechat_prefix ("network"), IRC_COLOR_NOTICE, /* TRANSLATORS: "Notice" is command name in IRC protocol (translation is frequently the same word) */ _("Notice"), IRC_COLOR_RESET, irc_nick_color_for_msg (server, 0, NULL, pos_target), pos_target, IRC_COLOR_RESET, pos_args); } else { nick_address = irc_protocol_nick_address (server, 0, NULL, nick, address); weechat_printf_date_tags ( ptr_buffer, date, irc_protocol_tags (command, (notify_private) ? "notify_private" : NULL, nick, address), "%s%s%s%s", weechat_prefix ("network"), nick_address, (nick_address[0]) ? ": " : "", pos_args); } } } if (channel) free (channel); } free (notice_args); return WEECHAT_RC_OK; } /* * Callback for the IRC command "PART". * * Command looks like: * PART #channel :part message * * On undernet server, it can be: * PART :#channel * PART #channel :part message */ IRC_PROTOCOL_CALLBACK(part) { char *str_comment, *join_string; int join_length, local_part, display_host; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; struct t_irc_channel_speaking *ptr_nick_speaking; IRC_PROTOCOL_MIN_PARAMS(1); IRC_PROTOCOL_CHECK_NICK; ptr_channel = irc_channel_search (server, params[0]); if (!ptr_channel) return WEECHAT_RC_OK; str_comment = (num_params > 1) ? irc_protocol_string_params (params, 1, num_params - 1) : NULL; ptr_nick = irc_nick_search (server, ptr_channel, nick); local_part = (irc_server_strcasecmp (server, nick, server->nick) == 0); /* display part message */ if (!ignored) { ptr_nick_speaking = NULL; if (ptr_channel->type == IRC_CHANNEL_TYPE_CHANNEL) { ptr_nick_speaking = ((weechat_config_boolean (irc_config_look_smart_filter)) && (weechat_config_boolean (irc_config_look_smart_filter_quit))) ? irc_channel_nick_speaking_time_search (server, ptr_channel, nick, 1) : NULL; } display_host = weechat_config_boolean (irc_config_look_display_host_quit); if (str_comment && str_comment[0]) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, (local_part || (ptr_channel->type != IRC_CHANNEL_TYPE_CHANNEL) || !weechat_config_boolean (irc_config_look_smart_filter) || !weechat_config_boolean (irc_config_look_smart_filter_quit) || ptr_nick_speaking) ? NULL : "irc_smart_filter", nick, address), _("%s%s%s%s%s%s%s%s%s%s has left %s%s%s %s(%s%s%s)"), weechat_prefix ("quit"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_CHAT_DELIMITERS, (display_host) ? " (" : "", IRC_COLOR_CHAT_HOST, (display_host) ? address : "", IRC_COLOR_CHAT_DELIMITERS, (display_host) ? ")" : "", IRC_COLOR_MESSAGE_QUIT, IRC_COLOR_CHAT_CHANNEL, ptr_channel->name, IRC_COLOR_MESSAGE_QUIT, IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_REASON_QUIT, str_comment, IRC_COLOR_CHAT_DELIMITERS); } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, (local_part || (ptr_channel->type != IRC_CHANNEL_TYPE_CHANNEL) || !weechat_config_boolean (irc_config_look_smart_filter) || !weechat_config_boolean (irc_config_look_smart_filter_quit) || ptr_nick_speaking) ? NULL : "irc_smart_filter", nick, address), _("%s%s%s%s%s%s%s%s%s%s has left %s%s%s"), weechat_prefix ("quit"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_CHAT_DELIMITERS, (display_host) ? " (" : "", IRC_COLOR_CHAT_HOST, (display_host) ? address : "", IRC_COLOR_CHAT_DELIMITERS, (display_host) ? ")" : "", IRC_COLOR_MESSAGE_QUIT, IRC_COLOR_CHAT_CHANNEL, ptr_channel->name, IRC_COLOR_MESSAGE_QUIT); } } /* part request was issued by local client ? */ if (local_part) { if (weechat_config_boolean (irc_config_look_typing_status_nicks)) irc_typing_channel_reset (ptr_channel); irc_nick_free_all (server, ptr_channel); irc_channel_modelist_set_state (ptr_channel, IRC_MODELIST_STATE_MODIFIED); /* cycling ? => rejoin channel immediately */ if (ptr_channel->cycle) { ptr_channel->cycle = 0; if (ptr_channel->key) { join_length = strlen (ptr_channel->name) + 1 + strlen (ptr_channel->key) + 1; join_string = malloc (join_length); if (join_string) { snprintf (join_string, join_length, "%s %s", ptr_channel->name, ptr_channel->key); irc_command_join_server (server, join_string, 1, 1); free (join_string); } else irc_command_join_server (server, ptr_channel->name, 1, 1); } else irc_command_join_server (server, ptr_channel->name, 1, 1); } else { if (weechat_config_boolean (irc_config_look_part_closes_buffer)) weechat_buffer_close (ptr_channel->buffer); else ptr_channel->part = 1; } irc_bar_item_update_channel (); } else { /* part from another user */ if (weechat_config_boolean (irc_config_look_typing_status_nicks)) { irc_typing_channel_set_nick (ptr_channel, nick, IRC_CHANNEL_TYPING_STATE_OFF); } if (ptr_nick) { irc_channel_join_smart_filtered_remove (ptr_channel, ptr_nick->name); irc_nick_free (server, ptr_channel, ptr_nick); } } if (str_comment) free (str_comment); return WEECHAT_RC_OK; } /* * Callback for the IRC command "PING". * * Command looks like: * PING :arguments */ IRC_PROTOCOL_CALLBACK(ping) { char *str_params; IRC_PROTOCOL_MIN_PARAMS(1); str_params = irc_protocol_string_params (params, 0, num_params - 1); irc_server_sendf (server, 0, NULL, "PONG :%s", str_params); if (str_params) free (str_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "PONG". * * Command looks like: * PONG server :arguments */ IRC_PROTOCOL_CALLBACK(pong) { struct timeval tv; char *str_params; IRC_PROTOCOL_MIN_PARAMS(0); if (server->lag_check_time.tv_sec != 0) { /* calculate lag (time diff with lag check) */ gettimeofday (&tv, NULL); server->lag = (int)(weechat_util_timeval_diff (&(server->lag_check_time), &tv) / 1000); /* schedule next lag check */ server->lag_check_time.tv_sec = 0; server->lag_check_time.tv_usec = 0; server->lag_next_check = time (NULL) + weechat_config_integer (irc_config_network_lag_check); /* refresh lag bar item if needed */ if (server->lag != server->lag_displayed) { server->lag_displayed = server->lag; irc_server_set_lag (server); } } else { str_params = (num_params > 1) ? irc_protocol_string_params (params, 1, num_params - 1) : NULL; weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, NULL, NULL, NULL), "PONG%s%s", (str_params) ? ": " : "", (str_params) ? str_params : ""); if (str_params) free (str_params); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "PRIVMSG". * * Command looks like: * PRIVMSG #channel :message for channel here * PRIVMSG @#channel :message for channel ops here * PRIVMSG mynick :message for private here * PRIVMSG #channel :\01ACTION is testing action\01 * PRIVMSG mynick :\01ACTION is testing action\01 * PRIVMSG #channel :\01VERSION\01 * PRIVMSG mynick :\01VERSION\01 * PRIVMSG mynick :\01DCC SEND file.txt 1488915698 50612 128\01 */ IRC_PROTOCOL_CALLBACK(privmsg) { char *msg_args, str_tags[1024], *str_color, status_msg[2], *color; const char *pos_target, *remote_nick, *pv_tags; int is_channel, nick_is_me; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; IRC_PROTOCOL_MIN_PARAMS(2); IRC_PROTOCOL_CHECK_NICK; if (ignored) return WEECHAT_RC_OK; msg_args = irc_protocol_string_params (params, 1, num_params - 1); if (!msg_args) return WEECHAT_RC_ERROR; status_msg[0] = '\0'; status_msg[1] = '\0'; pos_target = params[0]; is_channel = irc_channel_is_channel (server, pos_target); if (!is_channel) { if (irc_channel_is_channel (server, pos_target + 1) && irc_server_prefix_char_statusmsg (server, pos_target[0])) { is_channel = 1; status_msg[0] = pos_target[0]; pos_target++; } } /* receiver is a channel ? */ if (is_channel) { ptr_channel = irc_channel_search (server, pos_target); if (ptr_channel) { /* * unmask a smart filtered join if it is in hashtable * "join_smart_filtered" of channel */ irc_channel_join_smart_filtered_unmask (ptr_channel, nick); /* CTCP to channel */ if (msg_args[0] == '\01') { irc_ctcp_recv (server, date, command, ptr_channel, address, nick, NULL, msg_args, irc_message); goto end; } /* other message */ if (weechat_config_boolean (irc_config_look_typing_status_nicks)) { irc_typing_channel_set_nick (ptr_channel, nick, IRC_CHANNEL_TYPING_STATE_OFF); } ptr_nick = irc_nick_search (server, ptr_channel, nick); if (ptr_nick) irc_nick_set_host (ptr_nick, address); if (status_msg[0]) { /* message to channel ops/voiced (to "@#channel" or "+#channel") */ weechat_printf_date_tags ( ptr_channel->buffer, date, irc_protocol_tags (command, "notify_message", nick, address), "%s%s%s%s%s(%s%s%s)%s: %s", weechat_prefix ("network"), "Msg", (status_msg[0]) ? ":" : "", status_msg, IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 0, ptr_nick, nick), (nick && nick[0]) ? nick : "?", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, msg_args); } else { /* standard message (to "#channel") */ color = irc_nick_find_color_name ((ptr_nick) ? ptr_nick->name : nick); str_color = irc_color_for_tags (color); if (color) free (color); snprintf (str_tags, sizeof (str_tags), "notify_message,prefix_nick_%s", (str_color) ? str_color : "default"); if (str_color) free (str_color); weechat_printf_date_tags ( ptr_channel->buffer, date, irc_protocol_tags (command, str_tags, nick, address), "%s%s", irc_nick_as_prefix (server, ptr_nick, (ptr_nick) ? NULL : nick, NULL), msg_args); } irc_channel_nick_speaking_add ( ptr_channel, nick, weechat_string_has_highlight (msg_args, server->nick)); irc_channel_nick_speaking_time_remove_old (ptr_channel); irc_channel_nick_speaking_time_add (server, ptr_channel, nick, time (NULL)); } } else { nick_is_me = (irc_server_strcasecmp (server, server->nick, nick) == 0); remote_nick = (nick_is_me) ? pos_target : nick; /* CTCP to user */ if (msg_args[0] == '\01') { irc_ctcp_recv (server, date, command, NULL, address, nick, remote_nick, msg_args, irc_message); goto end; } /* private message received => display it */ ptr_channel = irc_channel_search (server, remote_nick); if (!ptr_channel) { ptr_channel = irc_channel_new (server, IRC_CHANNEL_TYPE_PRIVATE, remote_nick, 0, 0); if (!ptr_channel) { weechat_printf (server->buffer, _("%s%s: cannot create new " "private buffer \"%s\""), weechat_prefix ("error"), IRC_PLUGIN_NAME, remote_nick); goto end; } } if (weechat_config_boolean (irc_config_look_typing_status_nicks)) { irc_typing_channel_set_nick (ptr_channel, nick, IRC_CHANNEL_TYPING_STATE_OFF); } irc_channel_set_topic (ptr_channel, address); if (nick_is_me) { str_color = irc_color_for_tags ( weechat_config_color ( weechat_config_get ("weechat.color.chat_nick_self"))); } else { if (weechat_config_boolean (irc_config_look_color_pv_nick_like_channel)) { color = irc_nick_find_color_name (nick); str_color = irc_color_for_tags (color); if (color) free (color); } else { str_color = irc_color_for_tags ( weechat_config_color ( weechat_config_get ("weechat.color.chat_nick_other"))); } } if (nick_is_me) { snprintf (str_tags, sizeof (str_tags), "self_msg,notify_none,no_highlight,prefix_nick_%s", (str_color) ? str_color : "default"); } else { pv_tags = weechat_config_string (irc_config_look_pv_tags); snprintf (str_tags, sizeof (str_tags), "%s%sprefix_nick_%s", (pv_tags && pv_tags[0]) ? pv_tags : "", (pv_tags && pv_tags[0]) ? "," : "", (str_color) ? str_color : "default"); } if (str_color) free (str_color); weechat_printf_date_tags ( ptr_channel->buffer, date, irc_protocol_tags (command, str_tags, nick, address), "%s%s", irc_nick_as_prefix ( server, NULL, nick, (nick_is_me) ? IRC_COLOR_CHAT_NICK_SELF : irc_nick_color_for_pv (ptr_channel, nick)), msg_args); if (ptr_channel->has_quit_server) ptr_channel->has_quit_server = 0; (void) weechat_hook_signal_send ("irc_pv", WEECHAT_HOOK_SIGNAL_STRING, (char *)irc_message); } end: if (msg_args) free (msg_args); return WEECHAT_RC_OK; } /* * Callback for the IRC command "QUIT". * * Command looks like: * QUIT :quit message */ IRC_PROTOCOL_CALLBACK(quit) { char *str_quit_msg; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; struct t_irc_channel_speaking *ptr_nick_speaking; int local_quit, display_host; IRC_PROTOCOL_MIN_PARAMS(0); IRC_PROTOCOL_CHECK_NICK; str_quit_msg = (num_params > 0) ? irc_protocol_string_params (params, 0, num_params - 1) : NULL; for (ptr_channel = server->channels; ptr_channel; ptr_channel = ptr_channel->next_channel) { if (weechat_config_boolean (irc_config_look_typing_status_nicks)) { irc_typing_channel_set_nick (ptr_channel, nick, IRC_CHANNEL_TYPING_STATE_OFF); } if (ptr_channel->type == IRC_CHANNEL_TYPE_PRIVATE) ptr_nick = NULL; else ptr_nick = irc_nick_search (server, ptr_channel, nick); if (ptr_nick || (irc_server_strcasecmp (server, ptr_channel->name, nick) == 0)) { local_quit = (irc_server_strcasecmp (server, nick, server->nick) == 0); if (!irc_ignore_check (server, ptr_channel->name, nick, host)) { /* display quit message */ ptr_nick_speaking = NULL; if (ptr_channel->type == IRC_CHANNEL_TYPE_CHANNEL) { ptr_nick_speaking = ((weechat_config_boolean (irc_config_look_smart_filter)) && (weechat_config_boolean (irc_config_look_smart_filter_quit))) ? irc_channel_nick_speaking_time_search (server, ptr_channel, nick, 1) : NULL; } if (ptr_channel->type == IRC_CHANNEL_TYPE_PRIVATE) { ptr_channel->has_quit_server = 1; } display_host = weechat_config_boolean (irc_config_look_display_host_quit); if (str_quit_msg) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags ( command, (local_quit || (ptr_channel->type != IRC_CHANNEL_TYPE_CHANNEL) || !weechat_config_boolean (irc_config_look_smart_filter) || !weechat_config_boolean (irc_config_look_smart_filter_quit) || ptr_nick_speaking) ? NULL : "irc_smart_filter", nick, address), _("%s%s%s%s%s%s%s%s%s%s has quit %s(%s%s%s)"), weechat_prefix ("quit"), (ptr_channel->type == IRC_CHANNEL_TYPE_PRIVATE) ? irc_nick_color_for_pv (ptr_channel, nick) : irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_CHAT_DELIMITERS, (display_host) ? " (" : "", IRC_COLOR_CHAT_HOST, (display_host) ? address : "", IRC_COLOR_CHAT_DELIMITERS, (display_host) ? ")" : "", IRC_COLOR_MESSAGE_QUIT, IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_REASON_QUIT, str_quit_msg, IRC_COLOR_CHAT_DELIMITERS); } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags ( command, (local_quit || (ptr_channel->type != IRC_CHANNEL_TYPE_CHANNEL) || !weechat_config_boolean (irc_config_look_smart_filter) || !weechat_config_boolean (irc_config_look_smart_filter_quit) || ptr_nick_speaking) ? NULL : "irc_smart_filter", nick, address), _("%s%s%s%s%s%s%s%s%s%s has quit"), weechat_prefix ("quit"), (ptr_channel->type == IRC_CHANNEL_TYPE_PRIVATE) ? irc_nick_color_for_pv (ptr_channel, nick) : irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_CHAT_DELIMITERS, (display_host) ? " (" : "", IRC_COLOR_CHAT_HOST, (display_host) ? address : "", IRC_COLOR_CHAT_DELIMITERS, (display_host) ? ")" : "", IRC_COLOR_MESSAGE_QUIT); } } if (!local_quit && ptr_nick) { irc_channel_join_smart_filtered_remove (ptr_channel, ptr_nick->name); } if (ptr_nick) irc_nick_free (server, ptr_channel, ptr_nick); } } if (str_quit_msg) free (str_quit_msg); return WEECHAT_RC_OK; } /* * Callback for the IRC command "SETNAME": set real name * (received when capability "setname" is enabled). * * Command looks like: * SETNAME :the realname */ IRC_PROTOCOL_CALLBACK(setname) { int local_setname; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; char *str_realname, *realname_color; IRC_PROTOCOL_MIN_PARAMS(1); IRC_PROTOCOL_CHECK_NICK; local_setname = (irc_server_strcasecmp (server, nick, server->nick) == 0); str_realname = irc_protocol_string_params (params, 0, num_params - 1); if (!str_realname) return WEECHAT_RC_ERROR; if (weechat_hashtable_has_key (server->cap_list, "setname")) { for (ptr_channel = server->channels; ptr_channel; ptr_channel = ptr_channel->next_channel) { ptr_nick = irc_nick_search (server, ptr_channel, nick); if (ptr_nick) { if (ptr_nick->realname) free (ptr_nick->realname); ptr_nick->realname = strdup (str_realname); } } } if (!ignored) { realname_color = irc_color_decode ( str_realname, weechat_config_boolean (irc_config_network_colors_receive)); if (local_setname) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, NULL, NULL, NULL), _("%sYour real name has been set to \"%s\""), weechat_prefix ("network"), (realname_color) ? realname_color : ""); } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, NULL, NULL, NULL), _("%sReal name of %s%s%s has been set to \"%s\""), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, NULL, nick), nick, IRC_COLOR_RESET, (realname_color) ? realname_color : ""); } if (realname_color) free (realname_color); } free (str_realname); return WEECHAT_RC_OK; } /* * Callback for the IRC command "TAGMSG": message with tags but no text content * (received when capability "message-tags" is enabled). * * Whole message looks like: * @msgid=6gqz7dxd22v7r3x9pvu;+typing=active :nick!user@host TAGMSG #channel * @msgid=6gqz7dxd22v7r3x9pvu;+typing=active :nick!user@host TAGMSG :#channel */ IRC_PROTOCOL_CALLBACK(tagmsg) { struct t_irc_channel *ptr_channel; const char *ptr_typing_value; int state; IRC_PROTOCOL_MIN_PARAMS(1); if (ignored) return WEECHAT_RC_OK; if (!tags) return WEECHAT_RC_OK; ptr_channel = NULL; if (irc_channel_is_channel (server, params[0])) ptr_channel = irc_channel_search (server, params[0]); else if (irc_server_strcasecmp (server, params[0], server->nick) == 0) ptr_channel = irc_channel_search (server, nick); if (!ptr_channel) return WEECHAT_RC_OK; if (weechat_config_boolean (irc_config_look_typing_status_nicks)) { ptr_typing_value = weechat_hashtable_get (tags, "+typing"); if (ptr_typing_value && ptr_typing_value[0]) { if (strcmp (ptr_typing_value, "active") == 0) state = IRC_CHANNEL_TYPING_STATE_ACTIVE; else if (strcmp (ptr_typing_value, "paused") == 0) state = IRC_CHANNEL_TYPING_STATE_PAUSED; else state = IRC_CHANNEL_TYPING_STATE_OFF; irc_typing_channel_set_nick (ptr_channel, nick, state); } } return WEECHAT_RC_OK; } /* * Callback for an IRC command with mode and reason (numeric). */ IRC_PROTOCOL_CALLBACK(server_mode_reason) { char *str_params; const char *pos_mode; int arg_text; IRC_PROTOCOL_MIN_PARAMS(1); /* skip nickname if at beginning of server message */ if (irc_server_strcasecmp (server, server->nick, params[0]) == 0) { if (num_params < 2) return WEECHAT_RC_OK; pos_mode = params[1]; arg_text = 2; } else { pos_mode = params[0]; arg_text = 1; } str_params = irc_protocol_string_params (params, arg_text, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s%s%s", weechat_prefix ("network"), pos_mode, (str_params && str_params[0]) ? ": " : "", (str_params && str_params[0]) ? str_params : ""); if (str_params) free (str_params); return WEECHAT_RC_OK; } /* * Callback for a numeric IRC command. */ IRC_PROTOCOL_CALLBACK(numeric) { int arg_text; char *str_params; IRC_PROTOCOL_MIN_PARAMS(1); arg_text = (irc_server_strcasecmp (server, server->nick, params[0]) == 0) ? 1 : 0; str_params = irc_protocol_string_params (params, arg_text, num_params - 1); if (str_params && str_params[0]) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s", weechat_prefix ("network"), str_params); } if (str_params) free (str_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "TOPIC". * * Command looks like: * TOPIC #channel :new topic for channel */ IRC_PROTOCOL_CALLBACK(topic) { char *str_topic, *old_topic_color, *topic_color; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_PARAMS(1); IRC_PROTOCOL_CHECK_NICK; if (!irc_channel_is_channel (server, params[0])) { weechat_printf (server->buffer, _("%s%s: \"%s\" command received without channel"), weechat_prefix ("error"), IRC_PLUGIN_NAME, "topic"); return WEECHAT_RC_OK; } str_topic = (num_params > 1) ? irc_protocol_string_params (params, 1, num_params - 1) : NULL; ptr_channel = irc_channel_search (server, params[0]); ptr_nick = irc_nick_search (server, ptr_channel, nick); ptr_buffer = (ptr_channel) ? ptr_channel->buffer : server->buffer; /* * unmask a smart filtered join if it is in hashtable * "join_smart_filtered" of channel */ if (ptr_channel) irc_channel_join_smart_filtered_unmask (ptr_channel, nick); if (str_topic) { topic_color = irc_color_decode ( str_topic, weechat_config_boolean (irc_config_network_colors_receive)); if (weechat_config_boolean (irc_config_look_display_old_topic) && ptr_channel && ptr_channel->topic && ptr_channel->topic[0]) { old_topic_color = irc_color_decode ( ptr_channel->topic, weechat_config_boolean (irc_config_network_colors_receive)); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_buffer), date, irc_protocol_tags (command, NULL, NULL, address), _("%s%s%s%s has changed topic for %s%s%s from \"%s%s%s\" to " "\"%s%s%s\""), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_RESET, IRC_COLOR_CHAT_CHANNEL, params[0], IRC_COLOR_RESET, IRC_COLOR_TOPIC_OLD, (old_topic_color) ? old_topic_color : ptr_channel->topic, IRC_COLOR_RESET, IRC_COLOR_TOPIC_NEW, (topic_color) ? topic_color : str_topic, IRC_COLOR_RESET); if (old_topic_color) free (old_topic_color); } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_buffer), date, irc_protocol_tags (command, NULL, NULL, address), _("%s%s%s%s has changed topic for %s%s%s to \"%s%s%s\""), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_RESET, IRC_COLOR_CHAT_CHANNEL, params[0], IRC_COLOR_RESET, IRC_COLOR_TOPIC_NEW, (topic_color) ? topic_color : str_topic, IRC_COLOR_RESET); } if (topic_color) free (topic_color); } else { if (weechat_config_boolean (irc_config_look_display_old_topic) && ptr_channel && ptr_channel->topic && ptr_channel->topic[0]) { old_topic_color = irc_color_decode ( ptr_channel->topic, weechat_config_boolean (irc_config_network_colors_receive)); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_buffer), date, irc_protocol_tags (command, NULL, NULL, address), _("%s%s%s%s has unset topic for %s%s%s (old topic: " "\"%s%s%s\")"), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_RESET, IRC_COLOR_CHAT_CHANNEL, params[0], IRC_COLOR_RESET, IRC_COLOR_TOPIC_OLD, (old_topic_color) ? old_topic_color : ptr_channel->topic, IRC_COLOR_RESET); if (old_topic_color) free (old_topic_color); } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_buffer), date, irc_protocol_tags (command, NULL, NULL, address), _("%s%s%s%s has unset topic for %s%s%s"), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, ptr_nick, nick), nick, IRC_COLOR_RESET, IRC_COLOR_CHAT_CHANNEL, params[0], IRC_COLOR_RESET); } } if (ptr_channel) irc_channel_set_topic (ptr_channel, str_topic); if (str_topic) free (str_topic); return WEECHAT_RC_OK; } /* * Callback for the IRC command "WALLOPS". * * Command looks like: * WALLOPS :message from admin */ IRC_PROTOCOL_CALLBACK(wallops) { const char *nick_address; char *str_message; IRC_PROTOCOL_MIN_PARAMS(1); if (ignored) return WEECHAT_RC_OK; nick_address = irc_protocol_nick_address (server, 0, NULL, nick, address); str_message = irc_protocol_string_params (params, 0, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, nick, command, NULL, NULL), date, irc_protocol_tags (command, "notify_private", nick, address), _("%sWallops from %s: %s"), weechat_prefix ("network"), (nick_address[0]) ? nick_address : "?", str_message); if (str_message) free (str_message); return WEECHAT_RC_OK; } /* * Callback for the IRC command "WARN". * * Command looks like: * WARN REHASH CERTS_EXPIRED :Certificate [xxx] has expired */ IRC_PROTOCOL_CALLBACK(warn) { IRC_PROTOCOL_MIN_PARAMS(2); irc_protocol_print_error_warning_msg ( server, date, command, weechat_prefix ("error"), _("Warning:"), params, num_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "001": connected to IRC server. * * Command looks like: * 001 mynick :Welcome to the dancer-ircd Network */ IRC_PROTOCOL_CALLBACK(001) { char **commands, **ptr_command, *command2, *command3, *slash_command; char *away_msg, *usermode; const char *ptr_server_command; int length; IRC_PROTOCOL_MIN_PARAMS(1); if (irc_server_strcasecmp (server, server->nick, params[0]) != 0) irc_server_set_nick (server, params[0]); irc_protocol_cb_numeric (server, date, irc_message, tags, nick, address, host, command, ignored, argc, argv, argv_eol, params, num_params); /* connection to IRC server is OK! */ server->is_connected = 1; server->reconnect_delay = 0; server->monitor_time = time (NULL) + 5; irc_server_set_tls_version (server); if (server->hook_timer_connection) { weechat_unhook (server->hook_timer_connection); server->hook_timer_connection = NULL; } server->lag_next_check = time (NULL) + weechat_config_integer (irc_config_network_lag_check); irc_server_set_buffer_title (server); /* set away message if user was away (before disconnection for example) */ if (server->away_message && server->away_message[0]) { away_msg = strdup (server->away_message); if (away_msg) { irc_command_away_server (server, away_msg, 0); free (away_msg); } } /* send signal "irc_server_connected" with server name */ (void) weechat_hook_signal_send ("irc_server_connected", WEECHAT_HOOK_SIGNAL_STRING, server->name); /* set usermode when connected */ usermode = irc_server_eval_expression ( server, IRC_SERVER_OPTION_STRING(server, IRC_SERVER_OPTION_USERMODE)); if (usermode && usermode[0]) { irc_server_sendf (server, IRC_SERVER_SEND_OUTQ_PRIO_HIGH, NULL, "MODE %s %s", server->nick, usermode); } if (usermode) free (usermode); /* execute command when connected */ ptr_server_command = IRC_SERVER_OPTION_STRING(server, IRC_SERVER_OPTION_COMMAND); if (ptr_server_command && ptr_server_command[0]) { /* split command on ';' which can be escaped with '\;' */ commands = weechat_string_split_command (ptr_server_command, ';'); if (commands) { for (ptr_command = commands; *ptr_command; ptr_command++) { command2 = irc_server_eval_expression (server, *ptr_command); if (command2) { command3 = irc_message_replace_vars (server, NULL, command2); if (command3) { if (weechat_string_is_command_char (command3)) { weechat_command (server->buffer, command3); } else { length = 1 + strlen(command3) + 1; slash_command = malloc (length); if (slash_command) { snprintf (slash_command, length, "/%s", command3); weechat_command (server->buffer, slash_command); free (slash_command); } } free (command3); } free (command2); } } weechat_string_free_split_command (commands); } if (IRC_SERVER_OPTION_INTEGER(server, IRC_SERVER_OPTION_COMMAND_DELAY) > 0) server->command_time = time (NULL) + 1; else irc_server_autojoin_channels (server); } else { irc_server_autojoin_channels (server); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "005": some infos from server. * * Command looks like: * 005 mynick MODES=4 CHANLIMIT=#:20 NICKLEN=16 USERLEN=10 HOSTLEN=63 * TOPICLEN=450 KICKLEN=450 CHANNELLEN=30 KEYLEN=23 CHANTYPES=# * PREFIX=(ov)@+ CASEMAPPING=ascii CAPAB IRCD=dancer * :are available on this server */ IRC_PROTOCOL_CALLBACK(005) { char *str_info, *error, *isupport2, *pos_start; int i, arg_last, length_isupport, length, casemapping, utf8mapping; long value; IRC_PROTOCOL_MIN_PARAMS(2); irc_protocol_cb_numeric (server, date, irc_message, tags, nick, address, host, command, ignored, argc, argv, argv_eol, params, num_params); arg_last = (strstr (irc_message, " :")) ? num_params - 2 : num_params - 1; for (i = 1; i <= arg_last; i++) { if (strncmp (params[i], "PREFIX=", 7) == 0) { /* save prefix */ irc_server_set_prefix_modes_chars (server, params[i] + 7); } else if (strncmp (params[i], "NICKLEN=", 8) == 0) { /* save max nick length */ error = NULL; value = strtol (params[i] + 8, &error, 10); if (error && !error[0] && (value > 0)) server->nick_max_length = (int)value; } else if (strncmp (params[i], "USERLEN=", 8) == 0) { /* save max user length */ error = NULL; value = strtol (params[i] + 8, &error, 10); if (error && !error[0] && (value > 0)) server->user_max_length = (int)value; } else if (strncmp (params[i], "HOSTLEN=", 8) == 0) { /* save max host length */ error = NULL; value = strtol (params[i] + 8, &error, 10); if (error && !error[0] && (value > 0)) server->host_max_length = (int)value; } else if (strncmp (params[i], "CASEMAPPING=", 12) == 0) { /* save casemapping */ casemapping = irc_server_search_casemapping (params[i] + 12); if (casemapping >= 0) server->casemapping = casemapping; } else if (strncmp (params[i], "UTF8MAPPING=", 12) == 0) { /* save utf8mapping */ utf8mapping = irc_server_search_utf8mapping (params[i] + 12); if (utf8mapping >= 0) server->utf8mapping = utf8mapping; } else if (strncmp (params[i], "CHANTYPES=", 10) == 0) { /* save chantypes */ if (server->chantypes) free (server->chantypes); server->chantypes = strdup (params[i] + 10); } else if (strncmp (params[i], "CHANMODES=", 10) == 0) { /* save chanmodes */ if (server->chanmodes) free (server->chanmodes); server->chanmodes = strdup (params[i] + 10); } else if (strncmp (params[i], "MONITOR=", 8) == 0) { /* save monitor (limit) */ error = NULL; value = strtol (params[i] + 8, &error, 10); if (error && !error[0] && (value > 0)) server->monitor = (int)value; } else if (strncmp (params[i], "CLIENTTAGDENY=", 14) == 0) { /* save client tag deny */ irc_server_set_clienttagdeny (server, params[i] + 14); } } /* save whole message (concatenate to existing isupport, if any) */ str_info = irc_protocol_string_params (params, 1, arg_last); if (str_info && str_info[0]) { pos_start = NULL; length = strlen (str_info); if (server->isupport) { length_isupport = strlen (server->isupport); isupport2 = realloc (server->isupport, length_isupport + /* existing */ 1 + /* space */ length + /* new */ 1); if (isupport2) { server->isupport = isupport2; pos_start = server->isupport + length_isupport; } } else { server->isupport = malloc (1 + length + 1); if (server->isupport) pos_start = server->isupport; } if (pos_start) { pos_start[0] = ' '; memcpy (pos_start + 1, str_info, length); pos_start[length + 1] = '\0'; } } if (str_info) free (str_info); return WEECHAT_RC_OK; } /* * Callback for the IRC command "008": server notice mask. * * Command looks like: * 008 nick +Zbfkrsuy :Server notice mask */ IRC_PROTOCOL_CALLBACK(008) { char *str_params; IRC_PROTOCOL_MIN_PARAMS(2); str_params = irc_protocol_string_params (params, 1, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, params[0], command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, address), _("%sServer notice mask for %s%s%s: %s"), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, NULL, params[0]), params[0], IRC_COLOR_RESET, str_params); if (str_params) free (str_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "221": user mode string. * * Command looks like: * 221 nick :+s */ IRC_PROTOCOL_CALLBACK(221) { char *str_modes; IRC_PROTOCOL_MIN_PARAMS(2); str_modes = irc_protocol_string_params (params, 1, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, params[0], command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, address), _("%sUser mode for %s%s%s is %s[%s%s%s]"), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, NULL, params[0]), params[0], IRC_COLOR_RESET, IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_modes, IRC_COLOR_CHAT_DELIMITERS); if (irc_server_strcasecmp (server, params[0], server->nick) == 0) irc_mode_user_set (server, str_modes, 1); if (str_modes) free (str_modes); return WEECHAT_RC_OK; } /* * Callback for the IRC command "301": away message. * * Command is received when we are talking to a user in private and that remote * user is away (we receive away message). * * Command looks like: * 301 mynick nick :away message for nick */ IRC_PROTOCOL_CALLBACK(301) { char *str_away_msg; struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_PARAMS(1); if (num_params < 3) return WEECHAT_RC_OK; str_away_msg = irc_protocol_string_params (params, 2, num_params - 1); if (!str_away_msg) return WEECHAT_RC_ERROR; /* look for private buffer to display message */ ptr_channel = irc_channel_search (server, params[1]); if (!weechat_config_boolean (irc_config_look_display_pv_away_once) || !ptr_channel || !(ptr_channel->away_message) || (strcmp (ptr_channel->away_message, str_away_msg) != 0)) { ptr_buffer = (ptr_channel) ? ptr_channel->buffer : server->buffer; weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, params[1], command, "whois", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, address), _("%s%s[%s%s%s]%s is away: %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_away_msg); if (ptr_channel) { if (ptr_channel->away_message) free (ptr_channel->away_message); ptr_channel->away_message = strdup (str_away_msg); } } free (str_away_msg); return WEECHAT_RC_OK; } /* * Callback for the IRC command "303": ison. * * Command looks like: * 303 mynick :nick1 nick2 */ IRC_PROTOCOL_CALLBACK(303) { char *str_nicks; IRC_PROTOCOL_MIN_PARAMS(2); str_nicks = irc_protocol_string_params (params, 1, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%sUsers online: %s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_NICK, str_nicks); if (str_nicks) free (str_nicks); return WEECHAT_RC_OK; } /* * Callback for the IRC command "305": unaway. * * Command looks like: * 305 mynick :Does this mean you're really back? */ IRC_PROTOCOL_CALLBACK(305) { char *str_away_msg; IRC_PROTOCOL_MIN_PARAMS(1); if (num_params > 1) { str_away_msg = irc_protocol_string_params (params, 1, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "unaway", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s", weechat_prefix ("network"), str_away_msg); if (str_away_msg) free (str_away_msg); } server->is_away = 0; server->away_time = 0; weechat_bar_item_update ("away"); return WEECHAT_RC_OK; } /* * Callback for the IRC command "306": now away. * * Command looks like: * 306 mynick :We'll miss you */ IRC_PROTOCOL_CALLBACK(306) { char *str_away_msg; IRC_PROTOCOL_MIN_PARAMS(1); if (num_params > 1) { str_away_msg = irc_protocol_string_params (params, 1, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "away", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s", weechat_prefix ("network"), str_away_msg); if (str_away_msg) free (str_away_msg); } server->is_away = 1; server->away_time = time (NULL); weechat_bar_item_update ("away"); return WEECHAT_RC_OK; } /* * Callback for the whois commands with nick and message. * * Command looks like: * 319 flashy FlashCode :some text here */ IRC_PROTOCOL_CALLBACK(whois_nick_msg) { char *str_params; IRC_PROTOCOL_MIN_PARAMS(3); str_params = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, params[1], command, "whois", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s] %s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_params); if (str_params) free (str_params); return WEECHAT_RC_OK; } /* * Callback for the whowas commands with nick and message. * * Command looks like: * 369 flashy FlashCode :some text here */ IRC_PROTOCOL_CALLBACK(whowas_nick_msg) { char *str_params; IRC_PROTOCOL_MIN_PARAMS(3); str_params = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, params[1], command, "whowas", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s] %s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_params); if (str_params) free (str_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "311": whois, user. * * Command looks like: * 311 mynick nick user host * :realname here */ IRC_PROTOCOL_CALLBACK(311) { char *str_realname; IRC_PROTOCOL_MIN_PARAMS(6); str_realname = irc_protocol_string_params (params, 5, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, params[1], command, "whois", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s] (%s%s@%s%s)%s: %s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_HOST, params[2], params[3], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_realname); if (str_realname) free (str_realname); return WEECHAT_RC_OK; } /* * Callback for the IRC command "312": whois, server. * * Command looks like: * 312 mynick nick tungsten.libera.chat :Umeå, SE */ IRC_PROTOCOL_CALLBACK(312) { char *str_server; IRC_PROTOCOL_MIN_PARAMS(4); str_server = irc_protocol_string_params (params, 3, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, params[1], command, "whois", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s] %s%s %s(%s%s%s)", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, params[2], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_server, IRC_COLOR_CHAT_DELIMITERS); if (str_server) free (str_server); return WEECHAT_RC_OK; } /* * Callback for the IRC command "314": whowas. * * Command looks like: * 314 mynick nick user host * :realname here */ IRC_PROTOCOL_CALLBACK(314) { char *str_realname; IRC_PROTOCOL_MIN_PARAMS(6); str_realname = irc_protocol_string_params (params, 5, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, params[1], command, "whowas", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s] (%s%s@%s%s)%s was %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_HOST, params[2], params[3], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_realname); if (str_realname) free (str_realname); return WEECHAT_RC_OK; } /* * Callback for the IRC command "315": end of /who. * * Command looks like: * 315 mynick #channel :End of /WHO list. */ IRC_PROTOCOL_CALLBACK(315) { char *str_text; struct t_irc_channel *ptr_channel; IRC_PROTOCOL_MIN_PARAMS(3); ptr_channel = irc_channel_search (server, params[1]); if (ptr_channel && (ptr_channel->checking_whox > 0)) { ptr_channel->checking_whox--; } else { str_text = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "who", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s]%s %s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_text); if (str_text) free (str_text); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "317": whois, idle. * * Command looks like: * 317 mynick nick 122877 1205327880 :seconds idle, signon time */ IRC_PROTOCOL_CALLBACK(317) { int idle_time, day, hour, min, sec; time_t datetime; struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_PARAMS(4); idle_time = atoi (params[2]); day = idle_time / (60 * 60 * 24); hour = (idle_time % (60 * 60 * 24)) / (60 * 60); min = ((idle_time % (60 * 60 * 24)) % (60 * 60)) / 60; sec = ((idle_time % (60 * 60 * 24)) % (60 * 60)) % 60; datetime = (time_t)(atol (params[3])); ptr_buffer = irc_msgbuffer_get_target_buffer (server, params[1], command, "whois", NULL); if (day > 0) { weechat_printf_date_tags ( ptr_buffer, date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s]%s idle: %s%d %s%s, %s%02d %s%s %s%02d %s%s %s%02d " "%s%s, signon at: %s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, IRC_COLOR_CHAT_CHANNEL, day, IRC_COLOR_RESET, NG_("day", "days", day), IRC_COLOR_CHAT_CHANNEL, hour, IRC_COLOR_RESET, NG_("hour", "hours", hour), IRC_COLOR_CHAT_CHANNEL, min, IRC_COLOR_RESET, NG_("minute", "minutes", min), IRC_COLOR_CHAT_CHANNEL, sec, IRC_COLOR_RESET, NG_("second", "seconds", sec), IRC_COLOR_CHAT_CHANNEL, weechat_util_get_time_string (&datetime)); } else { weechat_printf_date_tags ( ptr_buffer, date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s]%s idle: %s%02d %s%s %s%02d %s%s %s%02d %s%s, " "signon at: %s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, IRC_COLOR_CHAT_CHANNEL, hour, IRC_COLOR_RESET, NG_("hour", "hours", hour), IRC_COLOR_CHAT_CHANNEL, min, IRC_COLOR_RESET, NG_("minute", "minutes", min), IRC_COLOR_CHAT_CHANNEL, sec, IRC_COLOR_RESET, NG_("second", "seconds", sec), IRC_COLOR_CHAT_CHANNEL, weechat_util_get_time_string (&datetime)); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "321": /list start. * * Command looks like: * 321 mynick Channel :Users Name */ IRC_PROTOCOL_CALLBACK(321) { char *str_params; IRC_PROTOCOL_MIN_PARAMS(2); str_params = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "list", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s%s%s", weechat_prefix ("network"), params[1], (str_params && str_params[0]) ? " " : "", (str_params && str_params[0]) ? str_params : ""); if (str_params) free (str_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "322": channel for /list. * * Command looks like: * 322 mynick #channel 3 :topic of channel */ IRC_PROTOCOL_CALLBACK(322) { char *str_topic; IRC_PROTOCOL_MIN_PARAMS(3); if (!server->cmd_list_regexp || (regexec (server->cmd_list_regexp, params[1], 0, NULL, 0) == 0)) { str_topic = irc_protocol_string_params (params, 3, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "list", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s%s%s(%s%s%s)%s%s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, params[2], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (str_topic && str_topic[0]) ? ": " : "", (str_topic && str_topic[0]) ? str_topic : ""); if (str_topic) free (str_topic); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "323": end of /list. * * Command looks like: * 323 mynick :End of /LIST */ IRC_PROTOCOL_CALLBACK(323) { char *str_params; IRC_PROTOCOL_MIN_PARAMS(1); str_params = irc_protocol_string_params (params, 1, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, "list", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s", weechat_prefix ("network"), str_params); if (str_params) free (str_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "324": channel mode. * * Command looks like: * 324 mynick #channel +nt */ IRC_PROTOCOL_CALLBACK(324) { char *str_modes, *str_modes_args; struct t_irc_channel *ptr_channel; IRC_PROTOCOL_MIN_PARAMS(2); str_modes = (num_params > 2) ? irc_protocol_string_params (params, 2, num_params - 1) : NULL; str_modes_args = (num_params > 3) ? irc_protocol_string_params (params, 3, num_params - 1) : NULL; ptr_channel = irc_channel_search (server, params[1]); if (ptr_channel) { irc_channel_set_modes (ptr_channel, str_modes); if (num_params > 2) { (void) irc_mode_channel_set (server, ptr_channel, host, str_modes, str_modes_args); } } if (!ptr_channel || (weechat_hashtable_has_key (ptr_channel->join_msg_received, command) || weechat_hashtable_has_key (irc_config_hashtable_display_join_message, command))) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, (ptr_channel) ? ptr_channel->buffer : NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, address), _("%sMode %s%s %s[%s%s%s]"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (str_modes) ? str_modes : "", IRC_COLOR_CHAT_DELIMITERS); } if (ptr_channel) weechat_hashtable_set (ptr_channel->join_msg_received, command, "1"); if (str_modes) free (str_modes); if (str_modes_args) free (str_modes_args); return WEECHAT_RC_OK; } /* * Callback for the IRC command "327": whois, host. * * Command looks like: * 327 mynick nick host ip :real hostname/ip */ IRC_PROTOCOL_CALLBACK(327) { char *str_realname; struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_PARAMS(4); str_realname = (num_params > 4) ? irc_protocol_string_params (params, 4, num_params - 1) : NULL; ptr_buffer = irc_msgbuffer_get_target_buffer (server, params[1], command, "whois", NULL); if (str_realname && str_realname[0]) { weechat_printf_date_tags ( ptr_buffer, date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s] %s%s %s %s(%s%s%s)", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_HOST, params[2], params[3], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_realname, IRC_COLOR_CHAT_DELIMITERS); } else { weechat_printf_date_tags ( ptr_buffer, date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s] %s%s %s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_HOST, params[2], params[3]); } if (str_realname) free (str_realname); return WEECHAT_RC_OK; } /* * Callback for the IRC command "328": channel URL. * * Command looks like: * 328 mynick #channel :https://example.com/ */ IRC_PROTOCOL_CALLBACK(328) { char *str_url; struct t_irc_channel *ptr_channel; IRC_PROTOCOL_MIN_PARAMS(3); ptr_channel = irc_channel_search (server, params[1]); if (ptr_channel) { str_url = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%sURL for %s%s%s: %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_RESET, str_url); if (str_url) free (str_url); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "329": channel creation date. * * Command looks like: * 329 mynick #channel 1205327894 */ IRC_PROTOCOL_CALLBACK(329) { struct t_irc_channel *ptr_channel; time_t datetime; IRC_PROTOCOL_MIN_PARAMS(3); ptr_channel = irc_channel_search (server, params[1]); datetime = (time_t)(atol (params[2])); if (ptr_channel) { if (weechat_hashtable_has_key (ptr_channel->join_msg_received, command) || weechat_hashtable_has_key (irc_config_hashtable_display_join_message, command)) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), /* TRANSLATORS: "%s" after "created on" is a date */ _("%sChannel created on %s"), weechat_prefix ("network"), weechat_util_get_time_string (&datetime)); } } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), /* TRANSLATORS: "%s" after "created on" is a date */ _("%sChannel %s%s%s created on %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_RESET, weechat_util_get_time_string (&datetime)); } if (ptr_channel) weechat_hashtable_set (ptr_channel->join_msg_received, command, "1"); return WEECHAT_RC_OK; } /* * Callback for the IRC commands "330" (whois, is logged in as), and "343" * (whois, is opered as). * * Commands look like: * 330 mynick nick1 nick2 :is logged in as * 330 mynick #channel https://example.com/ * 343 mynick nick1 nick2 :is opered as */ IRC_PROTOCOL_CALLBACK(330_343) { char *str_text; struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_PARAMS(3); if (num_params >= 4) { str_text = irc_protocol_string_params (params, 3, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, params[1], command, "whois", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s] %s%s %s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_text, irc_nick_color_for_msg (server, 1, NULL, params[2]), params[2]); if (str_text) free (str_text); } else { ptr_channel = (irc_channel_is_channel (server, params[1])) ? irc_channel_search (server, params[1]) : NULL; ptr_buffer = (ptr_channel) ? ptr_channel->buffer : server->buffer; str_text = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, params[1], command, "whois", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s] %s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_text); if (str_text) free (str_text); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "331": no topic for channel. * * Command looks like: * 331 mynick #channel :There isn't a topic. */ IRC_PROTOCOL_CALLBACK(331) { struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_PARAMS(2); ptr_channel = irc_channel_search (server, params[1]); ptr_buffer = (ptr_channel) ? ptr_channel->buffer : server->buffer; weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, params[1], command, NULL, ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%sNo topic set for channel %s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1]); return WEECHAT_RC_OK; } /* * Callback for the IRC command "332": topic of channel. * * Command looks like: * 332 mynick #channel :topic of channel */ IRC_PROTOCOL_CALLBACK(332) { char *str_topic, *topic_no_color, *topic_color; struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_PARAMS(2); str_topic = (num_params >= 3) ? irc_protocol_string_params (params, 2, num_params - 1) : NULL; ptr_channel = irc_channel_search (server, params[1]); if (ptr_channel && ptr_channel->nicks) { if (str_topic) { topic_no_color = (weechat_config_boolean (irc_config_network_colors_receive)) ? NULL : irc_color_decode (str_topic, 0); irc_channel_set_topic (ptr_channel, (topic_no_color) ? topic_no_color : str_topic); if (topic_no_color) free (topic_no_color); } ptr_buffer = ptr_channel->buffer; } else ptr_buffer = server->buffer; topic_color = NULL; if (str_topic) { topic_color = irc_color_decode ( str_topic, (weechat_config_boolean (irc_config_network_colors_receive)) ? 1 : 0); } if (!ptr_channel || (weechat_hashtable_has_key (ptr_channel->join_msg_received, command)) || weechat_hashtable_has_key (irc_config_hashtable_display_join_message, command)) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%sTopic for %s%s%s is \"%s%s%s\""), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_RESET, IRC_COLOR_TOPIC_CURRENT, (topic_color) ? topic_color : ((str_topic) ? str_topic : ""), IRC_COLOR_RESET); } if (topic_color) free (topic_color); if (ptr_channel) weechat_hashtable_set (ptr_channel->join_msg_received, command, "1"); if (str_topic) free (str_topic); return WEECHAT_RC_OK; } /* * Callback for the IRC command "333": infos about topic (nick / date). * * Command looks like: * 333 mynick #channel nick!user@host 1205428096 * 333 mynick #channel 1205428096 */ IRC_PROTOCOL_CALLBACK(333) { struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; time_t datetime; const char *topic_nick, *topic_address; IRC_PROTOCOL_MIN_PARAMS(3); topic_nick = (num_params > 3) ? irc_message_get_nick_from_host (params[2]) : NULL; topic_address = (num_params > 3) ? irc_message_get_address_from_host (params[2]) : NULL; if (topic_nick && topic_address && strcmp (topic_nick, topic_address) == 0) topic_address = NULL; ptr_channel = irc_channel_search (server, params[1]); ptr_nick = (ptr_channel) ? irc_nick_search (server, ptr_channel, topic_nick) : NULL; datetime = (num_params > 3) ? (time_t)(atol (params[3])) : (time_t)(atol (params[2])); if (!topic_nick && (datetime == 0)) return WEECHAT_RC_OK; if (ptr_channel && ptr_channel->nicks) { if (weechat_hashtable_has_key (ptr_channel->join_msg_received, command) || weechat_hashtable_has_key (irc_config_hashtable_display_join_message, command)) { if (topic_nick) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), /* TRANSLATORS: "%s" after "on" is a date */ _("%sTopic set by %s%s%s%s%s%s%s%s%s on %s"), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, ptr_nick, topic_nick), topic_nick, IRC_COLOR_CHAT_DELIMITERS, (topic_address && topic_address[0]) ? " (" : "", IRC_COLOR_CHAT_HOST, (topic_address) ? topic_address : "", IRC_COLOR_CHAT_DELIMITERS, (topic_address && topic_address[0]) ? ")" : "", IRC_COLOR_RESET, weechat_util_get_time_string (&datetime)); } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, ptr_channel->buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), /* TRANSLATORS: "%s" after "on" is a date */ _("%sTopic set on %s"), weechat_prefix ("network"), weechat_util_get_time_string (&datetime)); } } } else { if (topic_nick) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), /* TRANSLATORS: "%s" after "on" is a date */ _("%sTopic for %s%s%s set by %s%s%s%s%s%s%s%s%s on %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_RESET, irc_nick_color_for_msg (server, 1, ptr_nick, topic_nick), topic_nick, IRC_COLOR_CHAT_DELIMITERS, (topic_address && topic_address[0]) ? " (" : "", IRC_COLOR_CHAT_HOST, (topic_address) ? topic_address : "", IRC_COLOR_CHAT_DELIMITERS, (topic_address && topic_address[0]) ? ")" : "", IRC_COLOR_RESET, weechat_util_get_time_string (&datetime)); } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), /* TRANSLATORS: "%s" after "on" is a date */ _("%sTopic for %s%s%s set on %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_RESET, weechat_util_get_time_string (&datetime)); } } if (ptr_channel) weechat_hashtable_set (ptr_channel->join_msg_received, command, "1"); return WEECHAT_RC_OK; } /* * Callback for the IRC command "338": whois, host. * * Command looks like: * 338 mynick nick host :actually using host */ IRC_PROTOCOL_CALLBACK(338) { char *str_text; IRC_PROTOCOL_MIN_PARAMS(4); str_text = irc_protocol_string_params (params, 3, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, params[1], command, "whois", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s]%s %s %s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, str_text, IRC_COLOR_CHAT_HOST, params[2]); if (str_text) free (str_text); return WEECHAT_RC_OK; } /* * Callback for the IRC command "341": inviting. * * Command looks like: * 341 mynick nick #channel * 341 mynick nick :#channel */ IRC_PROTOCOL_CALLBACK(341) { IRC_PROTOCOL_MIN_PARAMS(3); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, params[0], command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", params[0], address), _("%s%s%s%s has invited %s%s%s to %s%s%s"), weechat_prefix ("network"), irc_nick_color_for_msg (server, 1, NULL, params[0]), params[0], IRC_COLOR_RESET, irc_nick_color_for_msg (server, 1, NULL, params[1]), params[1], IRC_COLOR_RESET, IRC_COLOR_CHAT_CHANNEL, params[2], IRC_COLOR_RESET); return WEECHAT_RC_OK; } /* * Callback for the IRC command "344": channel reop. * * Command looks like: * 344 mynick #channel nick!user@host */ IRC_PROTOCOL_CALLBACK(344) { char *str_host; IRC_PROTOCOL_MIN_PARAMS(3); str_host = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, "reop", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%sChannel reop %s%s%s: %s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_RESET, IRC_COLOR_CHAT_HOST, str_host); if (str_host) free (str_host); return WEECHAT_RC_OK; } /* * Callback for the IRC command "345": end of channel reop. * * Command looks like: * 345 mynick #channel :End of Channel Reop List */ IRC_PROTOCOL_CALLBACK(345) { char *str_params; IRC_PROTOCOL_MIN_PARAMS(3); str_params = irc_protocol_string_params (params, 2, num_params - 1); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, "reop", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s%s%s: %s", weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_RESET, str_params); if (str_params) free (str_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "346": channel invite list. * * Command looks like: * 346 mynick #channel invitemask nick!user@host 1205590879 * 346 mynick #channel invitemask */ IRC_PROTOCOL_CALLBACK(346) { struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; struct t_irc_modelist *ptr_modelist; time_t datetime; const char *nick_address; char str_number[64]; IRC_PROTOCOL_MIN_PARAMS(3); ptr_channel = irc_channel_search (server, params[1]); ptr_buffer = (ptr_channel && ptr_channel->nicks) ? ptr_channel->buffer : server->buffer; ptr_modelist = irc_modelist_search (ptr_channel, 'I'); if (ptr_modelist) { /* start receiving new list */ if (ptr_modelist->state != IRC_MODELIST_STATE_RECEIVING) { irc_modelist_item_free_all (ptr_modelist); ptr_modelist->state = IRC_MODELIST_STATE_RECEIVING; } snprintf (str_number, sizeof (str_number), "%s[%s%d%s] ", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, ((ptr_modelist->last_item) ? ptr_modelist->last_item->number + 1 : 0) + 1, IRC_COLOR_CHAT_DELIMITERS); } else str_number[0] = '\0'; if (num_params >= 4) { nick_address = irc_protocol_nick_address ( server, 1, NULL, irc_message_get_nick_from_host (params[3]), irc_message_get_address_from_host (params[3])); if (num_params >= 5) { datetime = (time_t)(atol (params[4])); if (ptr_modelist) irc_modelist_item_new (ptr_modelist, params[2], params[3], datetime); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "invitelist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), /* TRANSLATORS: "%s" after "on" is a date */ _("%s%s[%s%s%s] %s%s%s%s invited by %s on %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_CHAT_HOST, params[2], IRC_COLOR_RESET, (nick_address[0]) ? nick_address : "?", weechat_util_get_time_string (&datetime)); } else { if (ptr_modelist) irc_modelist_item_new (ptr_modelist, params[2], params[3], 0); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "invitelist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s] %s%s%s%s invited by %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_CHAT_HOST, params[2], IRC_COLOR_RESET, (nick_address[0]) ? nick_address : "?"); } } else { if (ptr_modelist) irc_modelist_item_new (ptr_modelist, params[2], NULL, 0); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "invitelist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s] %s%s%s%s invited"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_CHAT_HOST, params[2], IRC_COLOR_RESET); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "347": end of channel invite list. * * Command looks like: * 347 mynick #channel :End of Channel Invite List */ IRC_PROTOCOL_CALLBACK(347) { char *str_params; struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; struct t_irc_modelist *ptr_modelist; IRC_PROTOCOL_MIN_PARAMS(2); str_params = (num_params > 2) ? irc_protocol_string_params (params, 2, num_params - 1) : NULL; ptr_channel = irc_channel_search (server, params[1]); ptr_buffer = (ptr_channel && ptr_channel->nicks) ? ptr_channel->buffer : server->buffer; ptr_modelist = irc_modelist_search (ptr_channel, 'I'); if (ptr_modelist) { if (ptr_modelist->state != IRC_MODELIST_STATE_RECEIVING) { /* * remove all items if no invite was received before * the end of invite list */ irc_modelist_item_free_all (ptr_modelist); } ptr_modelist->state = IRC_MODELIST_STATE_RECEIVED; } weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "invitelist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s]%s%s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, params[1], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (str_params) ? " " : "", (str_params) ? str_params : ""); if (str_params) free (str_params); return WEECHAT_RC_OK; } /* * Callback for the IRC command "348": channel exception list. * * Command looks like: * :server 348 mynick #channel nick1!user1@host1 nick2!user2@host2 1205585109 * (nick2 is nick who set exception on nick1) */ IRC_PROTOCOL_CALLBACK(348) { struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; struct t_irc_modelist *ptr_modelist; time_t datetime; const char *nick_address; char str_number[64]; IRC_PROTOCOL_MIN_ARGS(5); ptr_channel = irc_channel_search (server, argv[3]); ptr_buffer = (ptr_channel && ptr_channel->nicks) ? ptr_channel->buffer : server->buffer; ptr_modelist = irc_modelist_search (ptr_channel, 'e'); if (ptr_modelist) { /* start receiving new list */ if (ptr_modelist->state != IRC_MODELIST_STATE_RECEIVING) { irc_modelist_item_free_all (ptr_modelist); ptr_modelist->state = IRC_MODELIST_STATE_RECEIVING; } snprintf (str_number, sizeof (str_number), " %s[%s%d%s]", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, ((ptr_modelist->last_item) ? ptr_modelist->last_item->number + 1 : 0) + 1, IRC_COLOR_CHAT_DELIMITERS); } else str_number[0] = '\0'; if (argc >= 6) { nick_address = irc_protocol_nick_address ( server, 1, NULL, irc_message_get_nick_from_host (argv[5]), irc_message_get_address_from_host (argv[5])); if (argc >= 7) { datetime = (time_t)(atol ((argv[6][0] == ':') ? argv[6] + 1 : argv[6])); if (ptr_modelist) irc_modelist_item_new (ptr_modelist, argv[4], argv[5], datetime); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "exceptionlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), /* TRANSLATORS: "%s" after "on" is a date */ _("%s%s[%s%s%s]%s%s exception %s%s%s by %s on %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_RESET, IRC_COLOR_CHAT_HOST, argv[4], IRC_COLOR_RESET, (nick_address[0]) ? nick_address : "?", weechat_util_get_time_string (&datetime)); } else { if (ptr_modelist) irc_modelist_item_new (ptr_modelist, argv[4], argv[5], 0); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "exceptionlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s]%s%s exception %s%s%s by %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_RESET, IRC_COLOR_CHAT_HOST, argv[4], IRC_COLOR_RESET, (nick_address[0]) ? nick_address : "?"); } } else { if (ptr_modelist) irc_modelist_item_new (ptr_modelist, argv[4], NULL, 0); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "exceptionlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s]%s%s exception %s%s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_RESET, IRC_COLOR_CHAT_HOST, argv[4]); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "349": end of channel exception list. * * Command looks like: * :server 349 mynick #channel :End of Channel Exception List */ IRC_PROTOCOL_CALLBACK(349) { char *pos_args; struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; struct t_irc_modelist *ptr_modelist; IRC_PROTOCOL_MIN_ARGS(4); pos_args = (argc > 4) ? ((argv_eol[4][0] == ':') ? argv_eol[4] + 1 : argv_eol[4]) : NULL; ptr_channel = irc_channel_search (server, argv[3]); ptr_buffer = (ptr_channel && ptr_channel->nicks) ? ptr_channel->buffer : server->buffer; ptr_modelist = irc_modelist_search (ptr_channel, 'e'); if (ptr_modelist) { if (ptr_modelist->state != IRC_MODELIST_STATE_RECEIVING) { /* * remove all items if no exception was received before * the end of exception list */ irc_modelist_item_free_all (ptr_modelist); } ptr_modelist->state = IRC_MODELIST_STATE_RECEIVED; } weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "exceptionlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s]%s%s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (pos_args) ? " " : "", (pos_args) ? pos_args : ""); return WEECHAT_RC_OK; } /* * Callback for the IRC command "351": server version. * * Command looks like: * :server 351 mynick dancer-ircd-1.0.36(2006/07/23_13:11:50). server :iMZ dncrTS/v4 */ IRC_PROTOCOL_CALLBACK(351) { struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_ARGS(5); ptr_buffer = irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL); if (argc > 5) { weechat_printf_date_tags ( ptr_buffer, date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s %s (%s)", weechat_prefix ("network"), argv[3], argv[4], (argv_eol[5][0] == ':') ? argv_eol[5] + 1 : argv_eol[5]); } else { weechat_printf_date_tags ( ptr_buffer, date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s %s", weechat_prefix ("network"), argv[3], argv[4]); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "352": who. * * Command looks like: * :server 352 mynick #channel user host server nick (*) (H/G) :0 flashcode */ IRC_PROTOCOL_CALLBACK(352) { char *pos_attr, *pos_hopcount, *pos_realname, *str_host; int arg_start, length; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; IRC_PROTOCOL_MIN_ARGS(5); /* silently ignore malformed 352 message (missing infos) */ if (argc < 8) return WEECHAT_RC_OK; pos_attr = NULL; pos_hopcount = NULL; pos_realname = NULL; if (argc > 8) { arg_start = ((argc > 9) && (strcmp (argv[8], "*") == 0)) ? 9 : 8; if (argv[arg_start][0] == ':') { pos_attr = NULL; pos_hopcount = (argc > arg_start) ? argv[arg_start] + 1 : NULL; pos_realname = (argc > arg_start + 1) ? argv_eol[arg_start + 1] : NULL; } else { pos_attr = argv[arg_start]; pos_hopcount = (argc > arg_start + 1) ? argv[arg_start + 1] + 1 : NULL; pos_realname = (argc > arg_start + 2) ? argv_eol[arg_start + 2] : NULL; } } ptr_channel = irc_channel_search (server, argv[3]); ptr_nick = (ptr_channel) ? irc_nick_search (server, ptr_channel, argv[7]) : NULL; /* update host in nick */ if (ptr_nick) { length = strlen (argv[4]) + 1 + strlen (argv[5]) + 1; str_host = malloc (length); if (str_host) { snprintf (str_host, length, "%s@%s", argv[4], argv[5]); irc_nick_set_host (ptr_nick, str_host); free (str_host); } } /* update away flag in nick */ if (ptr_channel && ptr_nick && pos_attr) { irc_nick_set_away (server, ptr_channel, ptr_nick, (pos_attr[0] == 'G') ? 1 : 0); } /* update realname in nick */ if (ptr_channel && ptr_nick && pos_realname) { if (ptr_nick->realname) free (ptr_nick->realname); ptr_nick->realname = (pos_realname) ? strdup (pos_realname) : NULL; } /* display output of who (manual who from user) */ if (!ptr_channel || (ptr_channel->checking_whox <= 0)) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "who", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s] %s%s %s(%s%s@%s%s)%s %s%s%s%s%s(%s%s%s)", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, argv[7]), argv[7], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_HOST, argv[4], argv[5], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (pos_attr) ? pos_attr : "", (pos_attr) ? " " : "", (pos_hopcount) ? pos_hopcount : "", (pos_hopcount) ? " " : "", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (pos_realname) ? pos_realname : "", IRC_COLOR_CHAT_DELIMITERS); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "353": list of users on a channel. * * Command looks like: * :server 353 mynick = #channel :mynick nick1 @nick2 +nick3 */ IRC_PROTOCOL_CALLBACK(353) { char *pos_channel, *pos_nick, *pos_nick_orig, *pos_host, *nickname; char *prefixes, *str_nicks, *color; int args, i, length; struct t_irc_channel *ptr_channel; IRC_PROTOCOL_MIN_ARGS(5); if (irc_channel_is_channel (server, argv[3])) { pos_channel = argv[3]; args = 4; } else { pos_channel = argv[4]; args = 5; } IRC_PROTOCOL_MIN_ARGS(args + 1); ptr_channel = irc_channel_search (server, pos_channel); str_nicks = NULL; /* * for a channel without buffer, prepare a string that will be built * with nicks and colors (argc - args is the number of nicks) */ if (!ptr_channel) { /* * prefix color (16) + nick color (16) + reset color (16) = 48 bytes * added for each nick */ length = strlen (argv_eol[args]) + ((argc - args) * (16 + 16 + 16)) + 1; str_nicks = malloc (length); if (str_nicks) str_nicks[0] = '\0'; } for (i = args; i < argc; i++) { pos_nick = (argv[i][0] == ':') ? argv[i] + 1 : argv[i]; pos_nick_orig = pos_nick; /* skip and save prefix(es) */ while (pos_nick[0] && (irc_server_get_prefix_char_index (server, pos_nick[0]) >= 0)) { pos_nick++; } prefixes = (pos_nick > pos_nick_orig) ? weechat_strndup (pos_nick_orig, pos_nick - pos_nick_orig) : NULL; /* extract nick from host */ pos_host = strchr (pos_nick, '!'); if (pos_host) { nickname = weechat_strndup (pos_nick, pos_host - pos_nick); pos_host++; } else nickname = strdup (pos_nick); /* add or update nick on channel */ if (nickname) { if (ptr_channel && ptr_channel->nicks) { if (!irc_nick_new (server, ptr_channel, nickname, pos_host, prefixes, 0, NULL, NULL)) { weechat_printf ( server->buffer, _("%s%s: cannot create nick \"%s\" for channel \"%s\""), weechat_prefix ("error"), IRC_PLUGIN_NAME, nickname, ptr_channel->name); } } else if (!ptr_channel && str_nicks) { if (str_nicks[0]) { strcat (str_nicks, IRC_COLOR_RESET); strcat (str_nicks, " "); } if (prefixes) { strcat (str_nicks, weechat_color ( irc_nick_get_prefix_color_name (server, prefixes[0]))); strcat (str_nicks, prefixes); } if (weechat_config_boolean (irc_config_look_color_nicks_in_names)) { if (irc_server_strcasecmp (server, nickname, server->nick) == 0) strcat (str_nicks, IRC_COLOR_CHAT_NICK_SELF); else { color = irc_nick_find_color (nickname); strcat (str_nicks, color); if (color) free (color); } } else strcat (str_nicks, IRC_COLOR_RESET); strcat (str_nicks, nickname); } free (nickname); } if (prefixes) free (prefixes); } if (!ptr_channel) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "names", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%sNicks %s%s%s: %s[%s%s%s]"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, pos_channel, IRC_COLOR_RESET, IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (str_nicks) ? str_nicks : "", IRC_COLOR_CHAT_DELIMITERS); } if (str_nicks) free (str_nicks); return WEECHAT_RC_OK; } /* * Callback for the IRC command "354": WHOX output * * Command looks like: * :server 354 mynick #channel user host server nick status hopcount account :GECOS Information */ IRC_PROTOCOL_CALLBACK(354) { char *pos_attr, *pos_hopcount, *pos_account, *pos_realname, *str_host; int length; struct t_irc_channel *ptr_channel; struct t_irc_nick *ptr_nick; IRC_PROTOCOL_MIN_ARGS(4); ptr_channel = irc_channel_search (server, argv[3]); /* * if there are less than 11 arguments, we are unable to parse the message, * some infos are missing but we don't know which ones; in this case we * just display the message as-is */ if (argc < 11) { if (!ptr_channel || (ptr_channel->checking_whox <= 0)) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "who", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s]%s%s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (argc > 4) ? " " : "", (argc > 4) ? argv_eol[4] : ""); } return WEECHAT_RC_OK; } ptr_nick = (ptr_channel) ? irc_nick_search (server, ptr_channel, argv[7]) : NULL; pos_attr = argv[8]; pos_hopcount = argv[9]; pos_account = (strcmp (argv[10], "0") != 0) ? argv[10] : NULL; pos_realname = (argc > 11) ? ((argv_eol[11][0] == ':') ? argv_eol[11] + 1 : argv_eol[11]) : NULL; /* update host in nick */ if (ptr_nick) { length = strlen (argv[4]) + 1 + strlen (argv[5]) + 1; str_host = malloc (length); if (str_host) { snprintf (str_host, length, "%s@%s", argv[4], argv[5]); irc_nick_set_host (ptr_nick, str_host); free (str_host); } } /* update away flag in nick */ if (ptr_channel && ptr_nick) { irc_nick_set_away (server, ptr_channel, ptr_nick, (pos_attr && (pos_attr[0] == 'G')) ? 1 : 0); } /* update account flag in nick */ if (ptr_nick) { if (ptr_nick->account) free (ptr_nick->account); if (ptr_channel && pos_account && weechat_hashtable_has_key (server->cap_list, "account-notify")) { ptr_nick->account = strdup (pos_account); } else { ptr_nick->account = NULL; } } /* update realname in nick */ if (ptr_nick) { if (ptr_nick->realname) free (ptr_nick->realname); ptr_nick->realname = (ptr_channel && pos_realname) ? strdup (pos_realname) : NULL; } /* display output of who (manual who from user) */ if (!ptr_channel || (ptr_channel->checking_whox <= 0)) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "who", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s] %s%s %s%s%s%s%s%s(%s%s@%s%s)%s %s%s%s%s%s(%s%s%s)", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, irc_nick_color_for_msg (server, 1, NULL, argv[7]), argv[7], IRC_COLOR_CHAT_DELIMITERS, (pos_account) ? "[" : "", (pos_account) ? IRC_COLOR_CHAT_HOST : "", (pos_account) ? pos_account : "", (pos_account) ? IRC_COLOR_CHAT_DELIMITERS : "", (pos_account) ? "] " : "", IRC_COLOR_CHAT_HOST, argv[4], argv[5], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (pos_attr) ? pos_attr : "", (pos_attr) ? " " : "", (pos_hopcount) ? pos_hopcount : "", (pos_hopcount) ? " " : "", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (pos_realname) ? pos_realname : "", IRC_COLOR_CHAT_DELIMITERS); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "366": end of /names list. * * Command looks like: * :server 366 mynick #channel :End of /NAMES list. */ IRC_PROTOCOL_CALLBACK(366) { struct t_irc_channel *ptr_channel; struct t_infolist *infolist; struct t_config_option *ptr_option; int num_nicks, num_op, num_halfop, num_voice, num_normal, length, i; char *string, str_nicks_count[2048], *color; const char *prefix, *prefix_color, *nickname; IRC_PROTOCOL_MIN_ARGS(5); ptr_channel = irc_channel_search (server, argv[3]); if (ptr_channel && ptr_channel->nicks) { /* display users on channel */ if (weechat_hashtable_has_key (ptr_channel->join_msg_received, "353") || weechat_hashtable_has_key (irc_config_hashtable_display_join_message, "353")) { infolist = weechat_infolist_get ("nicklist", ptr_channel->buffer, NULL); if (infolist) { length = 0; while (weechat_infolist_next (infolist)) { if (strcmp (weechat_infolist_string (infolist, "type"), "nick") == 0) { ptr_option = weechat_config_get (weechat_infolist_string (infolist, "prefix_color")); length += ((ptr_option) ? strlen (weechat_color (weechat_config_string (ptr_option))) : 0) + strlen (weechat_infolist_string (infolist, "prefix")) + 16 + /* nick color */ strlen (weechat_infolist_string (infolist, "name")) + 16 + /* reset color */ 1; /* space */ } } if (length > 0) { string = malloc (length); if (string) { string[0] = '\0'; i = 0; while (weechat_infolist_next (infolist)) { if (strcmp (weechat_infolist_string (infolist, "type"), "nick") == 0) { if (i > 0) { strcat (string, IRC_COLOR_RESET); strcat (string, " "); } prefix = weechat_infolist_string (infolist, "prefix"); if (prefix[0] && (prefix[0] != ' ')) { prefix_color = weechat_infolist_string (infolist, "prefix_color"); if (strchr (prefix_color, '.')) { ptr_option = weechat_config_get (weechat_infolist_string (infolist, "prefix_color")); if (ptr_option) strcat (string, weechat_color (weechat_config_string (ptr_option))); } else { strcat (string, weechat_color (prefix_color)); } strcat (string, prefix); } nickname = weechat_infolist_string (infolist, "name"); if (weechat_config_boolean (irc_config_look_color_nicks_in_names)) { if (irc_server_strcasecmp (server, nickname, server->nick) == 0) strcat (string, IRC_COLOR_CHAT_NICK_SELF); else { color = irc_nick_find_color (nickname); strcat (string, color); if (color) free (color); } } else strcat (string, IRC_COLOR_RESET); strcat (string, nickname); i++; } } weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "names", ptr_channel->buffer), date, irc_protocol_tags ( command, "irc_numeric", NULL, NULL), _("%sNicks %s%s%s: %s[%s%s]"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, ptr_channel->name, IRC_COLOR_RESET, IRC_COLOR_CHAT_DELIMITERS, string, IRC_COLOR_CHAT_DELIMITERS); free (string); } } weechat_infolist_free (infolist); } } /* display number of nicks, ops, halfops & voices on the channel */ if (weechat_hashtable_has_key (ptr_channel->join_msg_received, "366") || weechat_hashtable_has_key (irc_config_hashtable_display_join_message, "366")) { irc_nick_count (server, ptr_channel, &num_nicks, &num_op, &num_halfop, &num_voice, &num_normal); str_nicks_count[0] = '\0'; if (irc_server_get_prefix_mode_index (server, 'o') >= 0) { length = strlen (str_nicks_count); snprintf (str_nicks_count + length, sizeof (str_nicks_count) - length, "%s%s%d%s %s", (str_nicks_count[0]) ? ", " : "", IRC_COLOR_CHAT_CHANNEL, num_op, IRC_COLOR_RESET, NG_("op", "ops", num_op)); } if (irc_server_get_prefix_mode_index (server, 'h') >= 0) { length = strlen (str_nicks_count); snprintf (str_nicks_count + length, sizeof (str_nicks_count) - length, "%s%s%d%s %s", (str_nicks_count[0]) ? ", " : "", IRC_COLOR_CHAT_CHANNEL, num_halfop, IRC_COLOR_RESET, NG_("halfop", "halfops", num_halfop)); } if (irc_server_get_prefix_mode_index (server, 'v') >= 0) { length = strlen (str_nicks_count); snprintf (str_nicks_count + length, sizeof (str_nicks_count) - length, "%s%s%d%s %s", (str_nicks_count[0]) ? ", " : "", IRC_COLOR_CHAT_CHANNEL, num_voice, IRC_COLOR_RESET, NG_("voice", "voices", num_voice)); } length = strlen (str_nicks_count); snprintf (str_nicks_count + length, sizeof (str_nicks_count) - length, /* TRANSLATORS: number of "normal" nicks on a channel (ie no op/voice), for example: "56 normals" */ NG_("%s%s%d%s normal", "%s%s%d%s normals", num_normal), (str_nicks_count[0]) ? ", " : "", IRC_COLOR_CHAT_CHANNEL, num_normal, IRC_COLOR_RESET); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "names", ptr_channel->buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%sChannel %s%s%s: %s%d%s %s %s(%s%s)"), weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, ptr_channel->name, IRC_COLOR_RESET, IRC_COLOR_CHAT_CHANNEL, num_nicks, IRC_COLOR_RESET, NG_("nick", "nicks", num_nicks), IRC_COLOR_CHAT_DELIMITERS, str_nicks_count, IRC_COLOR_CHAT_DELIMITERS); } if (!weechat_hashtable_has_key (ptr_channel->join_msg_received, command)) { irc_command_mode_server (server, "MODE", ptr_channel, NULL, IRC_SERVER_SEND_OUTQ_PRIO_LOW); irc_channel_check_whox (server, ptr_channel); } } else { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "names", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s%s%s: %s", weechat_prefix ("network"), IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_RESET, (argv[4][0] == ':') ? argv_eol[4] + 1 : argv_eol[4]); } if (ptr_channel) { weechat_hashtable_set (ptr_channel->join_msg_received, "353", "1"); weechat_hashtable_set (ptr_channel->join_msg_received, "366", "1"); } weechat_bar_item_update ("input_prompt"); return WEECHAT_RC_OK; } /* * Callback for the IRC command "367": banlist. * * Command looks like: * :server 367 mynick #channel banmask nick!user@host 1205590879 */ IRC_PROTOCOL_CALLBACK(367) { struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; struct t_irc_modelist *ptr_modelist; time_t datetime; const char *nick_address; char str_number[64]; IRC_PROTOCOL_MIN_ARGS(5); ptr_channel = irc_channel_search (server, argv[3]); ptr_buffer = (ptr_channel && ptr_channel->nicks) ? ptr_channel->buffer : server->buffer; ptr_modelist = irc_modelist_search (ptr_channel, 'b'); if (ptr_modelist) { /* start receiving new list */ if (ptr_modelist->state != IRC_MODELIST_STATE_RECEIVING) { irc_modelist_item_free_all (ptr_modelist); ptr_modelist->state = IRC_MODELIST_STATE_RECEIVING; } snprintf (str_number, sizeof (str_number), "%s[%s%d%s] ", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, ((ptr_modelist->last_item) ? ptr_modelist->last_item->number + 1 : 0) + 1, IRC_COLOR_CHAT_DELIMITERS); } else str_number[0] = '\0'; if (argc >= 6) { nick_address = irc_protocol_nick_address ( server, 1, NULL, irc_message_get_nick_from_host (argv[5]), irc_message_get_address_from_host (argv[5])); if (argc >= 7) { datetime = (time_t)(atol ((argv[6][0] == ':') ? argv[6] + 1 : argv[6])); if (ptr_modelist) irc_modelist_item_new (ptr_modelist, argv[4], argv[5], datetime); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "banlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), /* TRANSLATORS: "%s" after "on" is a date */ _("%s%s[%s%s%s] %s%s%s%s banned by %s on %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_CHAT_HOST, argv[4], IRC_COLOR_RESET, (nick_address[0]) ? nick_address : "?", weechat_util_get_time_string (&datetime)); } else { if (ptr_modelist) irc_modelist_item_new (ptr_modelist, argv[4], argv[5], 0); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "banlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s] %s%s%s%s banned by %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_CHAT_HOST, argv[4], IRC_COLOR_RESET, (nick_address[0]) ? nick_address : "?"); } } else { if (ptr_modelist) irc_modelist_item_new (ptr_modelist, argv[4], NULL, 0); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "banlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s] %s%s%s%s banned"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_CHAT_HOST, argv[4], IRC_COLOR_RESET); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "368": end of banlist. * * Command looks like: * :server 368 mynick #channel :End of Channel Ban List */ IRC_PROTOCOL_CALLBACK(368) { char *pos_args; struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; struct t_irc_modelist *ptr_modelist; IRC_PROTOCOL_MIN_ARGS(4); pos_args = (argc > 4) ? ((argv_eol[4][0] == ':') ? argv_eol[4] + 1 : argv_eol[4]) : NULL; ptr_channel = irc_channel_search (server, argv[3]); ptr_buffer = (ptr_channel && ptr_channel->nicks) ? ptr_channel->buffer : server->buffer; ptr_modelist = irc_modelist_search (ptr_channel, 'b'); if (ptr_modelist) { if (ptr_modelist->state != IRC_MODELIST_STATE_RECEIVING) { /* * remove all items if no ban was received before * the end of ban list */ irc_modelist_item_free_all (ptr_modelist); } ptr_modelist->state = IRC_MODELIST_STATE_RECEIVED; } weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "banlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s]%s%s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (pos_args) ? " " : "", (pos_args) ? pos_args : ""); return WEECHAT_RC_OK; } /* * Callback for the IRC command "432": erroneous nickname. * * Command looks like: * :server 432 * mynick :Erroneous Nickname */ IRC_PROTOCOL_CALLBACK(432) { const char *alternate_nick; struct t_gui_buffer *ptr_buffer; irc_protocol_cb_generic_error (server, date, irc_message, tags, nick, address, host, command, ignored, argc, argv, argv_eol, params, num_params); if (!server->is_connected) { ptr_buffer = irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL); alternate_nick = irc_server_get_alternate_nick (server); if (!alternate_nick) { weechat_printf_date_tags ( ptr_buffer, date, NULL, _("%s%s: all declared nicknames are already in use or " "invalid, closing connection with server"), weechat_prefix ("error"), IRC_PLUGIN_NAME); irc_server_disconnect (server, 0, 1); return WEECHAT_RC_OK; } weechat_printf_date_tags ( ptr_buffer, date, NULL, _("%s%s: nickname \"%s\" is invalid, trying nickname \"%s\""), weechat_prefix ("error"), IRC_PLUGIN_NAME, server->nick, alternate_nick); irc_server_set_nick (server, alternate_nick); irc_server_sendf ( server, 0, NULL, "NICK %s%s", (server->nick && strchr (server->nick, ':')) ? ":" : "", server->nick); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "433": nickname already in use. * * Command looks like: * :server 433 * mynick :Nickname is already in use. */ IRC_PROTOCOL_CALLBACK(433) { const char *alternate_nick; struct t_gui_buffer *ptr_buffer; if (!server->is_connected) { ptr_buffer = irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL); alternate_nick = irc_server_get_alternate_nick (server); if (!alternate_nick) { weechat_printf_date_tags ( ptr_buffer, date, NULL, _("%s%s: all declared nicknames are already in use, closing " "connection with server"), weechat_prefix ("error"), IRC_PLUGIN_NAME); irc_server_disconnect (server, 0, 1); return WEECHAT_RC_OK; } weechat_printf_date_tags ( ptr_buffer, date, NULL, _("%s%s: nickname \"%s\" is already in use, trying nickname " "\"%s\""), weechat_prefix ("network"), IRC_PLUGIN_NAME, server->nick, alternate_nick); irc_server_set_nick (server, alternate_nick); irc_server_sendf ( server, 0, NULL, "NICK %s%s", (server->nick && strchr (server->nick, ':')) ? ":" : "", server->nick); } else { return irc_protocol_cb_generic_error (server, date, irc_message, tags, nick, address, host, command, ignored, argc, argv, argv_eol, params, num_params); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "437": nick/channel temporarily unavailable. * * Command looks like: * :server 437 * mynick :Nick/channel is temporarily unavailable */ IRC_PROTOCOL_CALLBACK(437) { const char *alternate_nick; struct t_gui_buffer *ptr_buffer; irc_protocol_cb_generic_error (server, date, irc_message, tags, nick, address, host, command, ignored, argc, argv, argv_eol, params, num_params); if (!server->is_connected) { if ((argc >= 4) && (irc_server_strcasecmp (server, server->nick, argv[3]) == 0)) { ptr_buffer = irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL); alternate_nick = irc_server_get_alternate_nick (server); if (!alternate_nick) { weechat_printf_date_tags ( ptr_buffer, date, NULL, _("%s%s: all declared nicknames are already in use or " "invalid, closing connection with server"), weechat_prefix ("error"), IRC_PLUGIN_NAME); irc_server_disconnect (server, 0, 1); return WEECHAT_RC_OK; } weechat_printf_date_tags ( ptr_buffer, date, NULL, _("%s%s: nickname \"%s\" is unavailable, trying nickname " "\"%s\""), weechat_prefix ("error"), IRC_PLUGIN_NAME, server->nick, alternate_nick); irc_server_set_nick (server, alternate_nick); irc_server_sendf ( server, 0, NULL, "NICK %s%s", (server->nick && strchr (server->nick, ':')) ? ":" : "", server->nick); } } return WEECHAT_RC_OK; } /* * Callback for the IRC command "438": not authorized to change nickname. * * Command looks like: * :server 438 mynick newnick :Nick change too fast. Please wait 30 seconds. */ IRC_PROTOCOL_CALLBACK(438) { struct t_gui_buffer *ptr_buffer; IRC_PROTOCOL_MIN_ARGS(4); ptr_buffer = irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL); if (argc >= 5) { weechat_printf_date_tags ( ptr_buffer, date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s (%s => %s)", weechat_prefix ("network"), (argv_eol[4][0] == ':') ? argv_eol[4] + 1 : argv_eol[4], argv[2], argv[3]); } else { weechat_printf_date_tags ( ptr_buffer, date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s %s", weechat_prefix ("network"), argv[2], argv[3]); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "470": forwarding to another channel. * * Command looks like: * :server 470 mynick #channel ##channel :Forwarding to another channel */ IRC_PROTOCOL_CALLBACK(470) { struct t_gui_buffer *ptr_buffer; struct t_gui_lines *own_lines; const char *buffer_name, *short_name, *localvar_channel; char *old_channel_lower, *new_channel_lower; int lines_count; irc_protocol_cb_generic_error (server, date, irc_message, tags, nick, address, host, command, ignored, argc, argv, argv_eol, params, num_params); if ((argc >= 5) && !irc_channel_search (server, argv[3])) { ptr_buffer = irc_channel_search_buffer (server, IRC_CHANNEL_TYPE_CHANNEL, argv[3]); if (ptr_buffer) { short_name = weechat_buffer_get_string (ptr_buffer, "short_name"); localvar_channel = weechat_buffer_get_string (ptr_buffer, "localvar_channel"); if (!short_name || (localvar_channel && (strcmp (localvar_channel, short_name) == 0))) { /* * update the short_name only if it was not changed by the * user */ weechat_buffer_set (ptr_buffer, "short_name", argv[4]); } buffer_name = irc_buffer_build_name (server->name, argv[4]); weechat_buffer_set (ptr_buffer, "name", buffer_name); weechat_buffer_set (ptr_buffer, "localvar_set_channel", argv[4]); /* * check if logger backlog should be displayed for the new channel * name: it is displayed only if the buffer is currently completely * empty (no messages at all) */ lines_count = 0; own_lines = weechat_hdata_pointer (weechat_hdata_get ("buffer"), ptr_buffer, "own_lines"); if (own_lines) { lines_count = weechat_hdata_integer ( weechat_hdata_get ("lines"), own_lines, "lines_count"); } if (lines_count == 0) { (void) weechat_hook_signal_send ("logger_backlog", WEECHAT_HOOK_SIGNAL_POINTER, ptr_buffer); } } old_channel_lower = strdup (argv[3]); if (old_channel_lower) { weechat_string_tolower (old_channel_lower); new_channel_lower = strdup (argv[4]); if (new_channel_lower) { weechat_string_tolower (new_channel_lower); if (weechat_hashtable_has_key (server->join_manual, old_channel_lower)) { weechat_hashtable_set (server->join_manual, new_channel_lower, weechat_hashtable_get ( server->join_manual, old_channel_lower)); weechat_hashtable_remove (server->join_manual, old_channel_lower); } if (weechat_hashtable_has_key (server->join_noswitch, old_channel_lower)) { weechat_hashtable_set (server->join_noswitch, new_channel_lower, weechat_hashtable_get ( server->join_noswitch, old_channel_lower)); weechat_hashtable_remove (server->join_noswitch, old_channel_lower); } free (new_channel_lower); } free (old_channel_lower); } } return WEECHAT_RC_OK; } /* * Callback for the IRC command "728": quietlist. * * Command looks like: * :server 728 mynick #channel mode quietmask nick!user@host 1351350090 */ IRC_PROTOCOL_CALLBACK(728) { struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; struct t_irc_modelist *ptr_modelist; time_t datetime; const char *nick_address; char str_number[64]; IRC_PROTOCOL_MIN_ARGS(6); ptr_channel = irc_channel_search (server, argv[3]); ptr_buffer = (ptr_channel && ptr_channel->nicks) ? ptr_channel->buffer : server->buffer; ptr_modelist = irc_modelist_search (ptr_channel, argv[4][0]); if (ptr_modelist) { /* start receiving new list */ if (ptr_modelist->state != IRC_MODELIST_STATE_RECEIVING) { irc_modelist_item_free_all (ptr_modelist); ptr_modelist->state = IRC_MODELIST_STATE_RECEIVING; } snprintf (str_number, sizeof (str_number), "%s[%s%d%s] ", IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, ((ptr_modelist->last_item) ? ptr_modelist->last_item->number + 1 : 0) + 1, IRC_COLOR_CHAT_DELIMITERS); } else str_number[0] = '\0'; if (argc >= 7) { nick_address = irc_protocol_nick_address ( server, 1, NULL, irc_message_get_nick_from_host (argv[6]), irc_message_get_address_from_host (argv[6])); if (argc >= 8) { datetime = (time_t)(atol ((argv[7][0] == ':') ? argv[7] + 1 : argv[7])); if (ptr_modelist) irc_modelist_item_new (ptr_modelist, argv[5], argv[6], datetime); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "quietlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), /* TRANSLATORS: "%s" after "on" is a date */ _("%s%s[%s%s%s] %s%s%s%s quieted by %s on %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_CHAT_HOST, argv[5], IRC_COLOR_RESET, (nick_address[0]) ? nick_address : "?", weechat_util_get_time_string (&datetime)); } else { if (ptr_modelist) irc_modelist_item_new (ptr_modelist, argv[5], argv[6], 0); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "quietlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s] %s%s%s%s quieted by %s"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_CHAT_HOST, argv[5], IRC_COLOR_RESET, (nick_address[0]) ? nick_address : "?"); } } else { if (ptr_modelist) irc_modelist_item_new (ptr_modelist, argv[5], NULL, 0); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "quietlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), _("%s%s[%s%s%s] %s%s%s%s quieted"), weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, str_number, IRC_COLOR_CHAT_HOST, argv[5], IRC_COLOR_RESET); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "729": end of quietlist. * * Command looks like: * :server 729 mynick #channel mode :End of Channel Quiet List */ IRC_PROTOCOL_CALLBACK(729) { char *pos_args; struct t_irc_channel *ptr_channel; struct t_gui_buffer *ptr_buffer; struct t_irc_modelist *ptr_modelist; IRC_PROTOCOL_MIN_ARGS(5); pos_args = (argc > 5) ? ((argv_eol[5][0] == ':') ? argv_eol[5] + 1 : argv_eol[5]) : NULL; ptr_channel = irc_channel_search (server, argv[3]); ptr_buffer = (ptr_channel && ptr_channel->nicks) ? ptr_channel->buffer : server->buffer; ptr_modelist = irc_modelist_search (ptr_channel, argv[4][0]); if (ptr_modelist) { if (ptr_modelist->state != IRC_MODELIST_STATE_RECEIVING) { /* * remove all items if no quiet was received before * the end of quiet list */ irc_modelist_item_free_all (ptr_modelist); } ptr_modelist->state = IRC_MODELIST_STATE_RECEIVED; } weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "quietlist", ptr_buffer), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s[%s%s%s]%s%s%s", weechat_prefix ("network"), IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_CHANNEL, argv[3], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_RESET, (pos_args) ? " " : "", (pos_args) ? pos_args : ""); return WEECHAT_RC_OK; } /* * Callback for the IRC command "730": monitored nicks are online * (RPL_MONONLINE). * * Command looks like: * :server 730 mynick :nick1!user1@host1,nick2!user2@host2 */ IRC_PROTOCOL_CALLBACK(730) { struct t_irc_notify *ptr_notify; const char *monitor_nick, *monitor_host; char **nicks; int i, num_nicks; IRC_PROTOCOL_MIN_ARGS(4); nicks = weechat_string_split ((argv_eol[3][0] == ':') ? argv_eol[3] + 1 : argv_eol[3], ",", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &num_nicks); if (nicks) { for (i = 0; i < num_nicks; i++) { monitor_nick = irc_message_get_nick_from_host (nicks[i]); monitor_host = strchr (nicks[i], '!'); if (monitor_host) monitor_host++; ptr_notify = irc_notify_search (server, monitor_nick); if (ptr_notify) irc_notify_set_is_on_server (ptr_notify, monitor_host, 1); } weechat_string_free_split (nicks); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "731": monitored nicks are offline * (RPL_MONOFFLINE). * * Command looks like: * :server 731 mynick :nick1!user1@host1,nick2!user2@host2 */ IRC_PROTOCOL_CALLBACK(731) { struct t_irc_notify *ptr_notify; const char *monitor_nick, *monitor_host; char **nicks; int i, num_nicks; IRC_PROTOCOL_MIN_ARGS(4); nicks = weechat_string_split ((argv_eol[3][0] == ':') ? argv_eol[3] + 1 : argv_eol[3], ",", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &num_nicks); if (nicks) { for (i = 0; i < num_nicks; i++) { monitor_nick = irc_message_get_nick_from_host (nicks[i]); monitor_host = strchr (nicks[i], '!'); if (monitor_host) monitor_host++; ptr_notify = irc_notify_search (server, monitor_nick); if (ptr_notify) irc_notify_set_is_on_server (ptr_notify, monitor_host, 0); } weechat_string_free_split (nicks); } return WEECHAT_RC_OK; } /* * Callback for the IRC command "732": list of monitored nicks (RPL_MONLIST). * * Command looks like: * :server 732 mynick :nick1!user1@host1,nick2!user2@host2 */ IRC_PROTOCOL_CALLBACK(732) { char *pos_args; IRC_PROTOCOL_MIN_ARGS(3); pos_args = (argc > 3) ? ((argv_eol[3][0] == ':') ? argv_eol[3] + 1 : argv_eol[3]) : NULL; weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "monitor", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s", weechat_prefix ("network"), (pos_args && pos_args[0]) ? pos_args : ""); return WEECHAT_RC_OK; } /* * Callback for the IRC command "733": end of a monitor list (RPL_ENDOFMONLIST). * * Command looks like: * :server 733 mynick :End of MONITOR list */ IRC_PROTOCOL_CALLBACK(733) { char *pos_args; IRC_PROTOCOL_MIN_ARGS(3); pos_args = (argc > 3) ? ((argv_eol[3][0] == ':') ? argv_eol[3] + 1 : argv_eol[3]) : NULL; weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "monitor", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s", weechat_prefix ("network"), (pos_args && pos_args[0]) ? pos_args : ""); return WEECHAT_RC_OK; } /* * Callback for the IRC command "734": monitor list is full (ERR_MONLISTFULL) * * Command looks like: * :server 734 mynick limit nick1,nick2 :Monitor list is full. */ IRC_PROTOCOL_CALLBACK(734) { char *pos_args; IRC_PROTOCOL_MIN_ARGS(5); pos_args = (argc > 5) ? ((argv_eol[5][0] == ':') ? argv_eol[5] + 1 : argv_eol[5]) : NULL; weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer ( server, NULL, command, "monitor", NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s (%s)", weechat_prefix ("error"), (pos_args && pos_args[0]) ? pos_args : "", argv[3]); return WEECHAT_RC_OK; } /* * Callback for the IRC command "900": logged in as (SASL). * * Command looks like: * :server 900 mynick nick!user@host mynick :You are now logged in as mynick */ IRC_PROTOCOL_CALLBACK(900) { IRC_PROTOCOL_MIN_ARGS(6); weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, argv[3], command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s %s(%s%s%s)", weechat_prefix ("network"), (argv_eol[5][0] == ':') ? argv_eol[5] + 1 : argv_eol[5], IRC_COLOR_CHAT_DELIMITERS, IRC_COLOR_CHAT_HOST, argv[3], IRC_COLOR_CHAT_DELIMITERS); irc_server_free_sasl_data (server); return WEECHAT_RC_OK; } /* * Callback for the IRC command "901": you are now logged in. * * Command looks like: * :server 901 mynick nick user host :You are now logged in. (id nick, username user, hostname host) */ IRC_PROTOCOL_CALLBACK(901) { IRC_PROTOCOL_MIN_ARGS(6); if (argc >= 7) { weechat_printf_date_tags ( irc_msgbuffer_get_target_buffer (server, NULL, command, NULL, NULL), date, irc_protocol_tags (command, "irc_numeric", NULL, NULL), "%s%s", weechat_prefix ("network"), (argv_eol[6][0] == ':') ? argv_eol[6] + 1 : argv_eol[6]); } else { irc_protocol_cb_numeric (server, date, irc_message, tags, nick, address, host, command, ignored, argc, argv, argv_eol, params, num_params); } return WEECHAT_RC_OK; } /* * Callback for the IRC commands "903" and "907" (SASL OK). * * Commands look like: * :server 903 nick :SASL authentication successful */ IRC_PROTOCOL_CALLBACK(sasl_end_ok) { if (server->hook_timer_sasl) { weechat_unhook (server->hook_timer_sasl); server->hook_timer_sasl = NULL; } irc_protocol_cb_numeric (server, date, irc_message, tags, nick, address, host, command, ignored, argc, argv, argv_eol, params, num_params); if (!server->is_connected) irc_server_sendf (server, 0, NULL, "CAP END"); irc_server_free_sasl_data (server); return WEECHAT_RC_OK; } /* * Callback for the IRC commands "902", "904", "905", "906" (SASL failed). * * Commands look like: * :server 904 nick :SASL authentication failed */ IRC_PROTOCOL_CALLBACK(sasl_end_fail) { int sasl_fail; if (server->hook_timer_sasl) { weechat_unhook (server->hook_timer_sasl); server->hook_timer_sasl = NULL; } irc_protocol_cb_numeric (server, date, irc_message, tags, nick, address, host, command, ignored, argc, argv, argv_eol, params, num_params); sasl_fail = IRC_SERVER_OPTION_INTEGER(server, IRC_SERVER_OPTION_SASL_FAIL); if (!server->is_connected && ((sasl_fail == IRC_SERVER_SASL_FAIL_RECONNECT) || (sasl_fail == IRC_SERVER_SASL_FAIL_DISCONNECT))) { irc_server_disconnect ( server, 0, (sasl_fail == IRC_SERVER_SASL_FAIL_RECONNECT) ? 1 : 0); return WEECHAT_RC_OK; } if (!server->is_connected) irc_server_sendf (server, 0, NULL, "CAP END"); irc_server_free_sasl_data (server); return WEECHAT_RC_OK; } /* * Executes action when an IRC command is received. * * Argument "irc_message" is the full message without optional tags. */ void irc_protocol_recv_command (struct t_irc_server *server, const char *irc_message, const char *msg_command, const char *msg_channel) { int i, cmd_found, return_code, argc, decode_color, keep_trailing_spaces; int message_ignored, flags, num_params; char *message_colors_decoded, *pos_space, *tags, **params; struct t_irc_channel *ptr_channel; t_irc_recv_func *cmd_recv_func; const char *cmd_name, *ptr_msg_after_tags; time_t date; const char *nick1, *address1, *host1; char *nick, *address, *address_color, *host, *host_no_color, *host_color; char **argv, **argv_eol; struct t_hashtable *hash_tags; struct t_irc_protocol_msg irc_protocol_messages[] = { /* format: "command", decode_color, keep_trailing_spaces, func_cb */ IRCB(account, 1, 0, account), /* account (cap account-notify) */ IRCB(authenticate, 1, 0, authenticate), /* authenticate */ IRCB(away, 1, 0, away), /* away (cap away-notify) */ IRCB(cap, 1, 0, cap), /* client capability */ IRCB(chghost, 1, 0, chghost), /* user/host change (cap chghost) */ IRCB(error, 1, 0, error), /* error received from server */ IRCB(fail, 1, 0, fail), /* error received from server */ IRCB(invite, 1, 0, invite), /* invite a nick on a channel */ IRCB(join, 1, 0, join), /* join a channel */ IRCB(kick, 1, 1, kick), /* kick a user */ IRCB(kill, 1, 1, kill), /* close client-server connection */ IRCB(mode, 1, 0, mode), /* change channel or user mode */ IRCB(nick, 1, 0, nick), /* change current nickname */ IRCB(note, 1, 0, note), /* note received from server */ IRCB(notice, 1, 1, notice), /* send notice message to user */ IRCB(part, 1, 1, part), /* leave a channel */ IRCB(ping, 1, 0, ping), /* ping server */ IRCB(pong, 1, 0, pong), /* answer to a ping message */ IRCB(privmsg, 1, 1, privmsg), /* message received */ IRCB(quit, 1, 1, quit), /* close all connections and quit */ IRCB(setname, 0, 1, setname), /* set realname */ IRCB(tagmsg, 0, 1, tagmsg), /* tag message */ IRCB(topic, 0, 1, topic), /* get/set channel topic */ IRCB(wallops, 1, 1, wallops), /* wallops */ IRCB(warn, 1, 0, warn), /* warning received from server */ IRCB(001, 1, 0, 001), /* a server message */ IRCB(005, 1, 0, 005), /* a server message */ IRCB(008, 1, 0, 008), /* server notice mask */ IRCB(221, 1, 0, 221), /* user mode string */ IRCB(223, 1, 0, whois_nick_msg), /* whois (charset is) */ IRCB(264, 1, 0, whois_nick_msg), /* whois (encrypted connection) */ IRCB(275, 1, 0, whois_nick_msg), /* whois (secure connection) */ IRCB(276, 1, 0, whois_nick_msg), /* whois (client cert. fingerprint)*/ IRCB(301, 1, 1, 301), /* away message */ IRCB(303, 1, 0, 303), /* ison */ IRCB(305, 1, 0, 305), /* unaway */ IRCB(306, 1, 0, 306), /* now away */ IRCB(307, 1, 0, whois_nick_msg), /* whois (registered nick) */ IRCB(310, 1, 0, whois_nick_msg), /* whois (help mode) */ IRCB(311, 1, 0, 311), /* whois (user) */ IRCB(312, 1, 0, 312), /* whois (server) */ IRCB(313, 1, 0, whois_nick_msg), /* whois (operator) */ IRCB(314, 1, 0, 314), /* whowas */ IRCB(315, 1, 0, 315), /* end of /who list */ IRCB(317, 1, 0, 317), /* whois (idle) */ IRCB(318, 1, 0, whois_nick_msg), /* whois (end) */ IRCB(319, 1, 0, whois_nick_msg), /* whois (channels) */ IRCB(320, 1, 0, whois_nick_msg), /* whois (identified user) */ IRCB(321, 1, 0, 321), /* /list start */ IRCB(322, 1, 0, 322), /* channel (for /list) */ IRCB(323, 1, 0, 323), /* end of /list */ IRCB(324, 1, 0, 324), /* channel mode */ IRCB(326, 1, 0, whois_nick_msg), /* whois (has oper privs) */ IRCB(327, 1, 0, 327), /* whois (host) */ IRCB(328, 1, 0, 328), /* channel URL */ IRCB(329, 1, 0, 329), /* channel creation date */ IRCB(330, 1, 0, 330_343), /* is logged in as */ IRCB(331, 1, 0, 331), /* no topic for channel */ IRCB(332, 0, 1, 332), /* topic of channel */ IRCB(333, 1, 0, 333), /* topic info (nick/date) */ IRCB(335, 1, 0, whois_nick_msg), /* is a bot on */ IRCB(338, 1, 0, 338), /* whois (host) */ IRCB(341, 1, 0, 341), /* inviting */ IRCB(343, 1, 0, 330_343), /* is opered as */ IRCB(344, 1, 0, 344), /* channel reop */ IRCB(345, 1, 0, 345), /* end of channel reop list */ IRCB(346, 1, 0, 346), /* invite list */ IRCB(347, 1, 0, 347), /* end of invite list */ IRCB(348, 1, 0, 348), /* channel exception list */ IRCB(349, 1, 0, 349), /* end of channel exception list */ IRCB(351, 1, 0, 351), /* server version */ IRCB(352, 1, 0, 352), /* who */ IRCB(353, 1, 0, 353), /* list of nicks on channel */ IRCB(354, 1, 0, 354), /* whox */ IRCB(366, 1, 0, 366), /* end of /names list */ IRCB(367, 1, 0, 367), /* banlist */ IRCB(368, 1, 0, 368), /* end of banlist */ IRCB(369, 1, 0, whowas_nick_msg), /* whowas (end) */ IRCB(378, 1, 0, whois_nick_msg), /* whois (connecting from) */ IRCB(379, 1, 0, whois_nick_msg), /* whois (using modes) */ IRCB(401, 1, 0, generic_error), /* no such nick/channel */ IRCB(402, 1, 0, generic_error), /* no such server */ IRCB(403, 1, 0, generic_error), /* no such channel */ IRCB(404, 1, 0, generic_error), /* cannot send to channel */ IRCB(405, 1, 0, generic_error), /* too many channels */ IRCB(406, 1, 0, generic_error), /* was no such nick */ IRCB(407, 1, 0, generic_error), /* was no such nick */ IRCB(409, 1, 0, generic_error), /* no origin */ IRCB(410, 1, 0, generic_error), /* no services */ IRCB(411, 1, 0, generic_error), /* no recipient */ IRCB(412, 1, 0, generic_error), /* no text to send */ IRCB(413, 1, 0, generic_error), /* no toplevel */ IRCB(414, 1, 0, generic_error), /* wilcard in toplevel domain */ IRCB(421, 1, 0, generic_error), /* unknown command */ IRCB(422, 1, 0, generic_error), /* MOTD is missing */ IRCB(423, 1, 0, generic_error), /* no administrative info */ IRCB(424, 1, 0, generic_error), /* file error */ IRCB(431, 1, 0, generic_error), /* no nickname given */ IRCB(432, 1, 0, 432), /* erroneous nickname */ IRCB(433, 1, 0, 433), /* nickname already in use */ IRCB(436, 1, 0, generic_error), /* nickname collision */ IRCB(437, 1, 0, 437), /* nick/channel unavailable */ IRCB(438, 1, 0, 438), /* not auth. to change nickname */ IRCB(441, 1, 0, generic_error), /* user not in channel */ IRCB(442, 1, 0, generic_error), /* not on channel */ IRCB(443, 1, 0, generic_error), /* user already on channel */ IRCB(444, 1, 0, generic_error), /* user not logged in */ IRCB(445, 1, 0, generic_error), /* summon has been disabled */ IRCB(446, 1, 0, generic_error), /* users has been disabled */ IRCB(451, 1, 0, generic_error), /* you are not registered */ IRCB(461, 1, 0, generic_error), /* not enough parameters */ IRCB(462, 1, 0, generic_error), /* you may not register */ IRCB(463, 1, 0, generic_error), /* host not privileged */ IRCB(464, 1, 0, generic_error), /* password incorrect */ IRCB(465, 1, 0, generic_error), /* banned from this server */ IRCB(467, 1, 0, generic_error), /* channel key already set */ IRCB(470, 1, 0, 470), /* forwarding to another channel */ IRCB(471, 1, 0, generic_error), /* channel is already full */ IRCB(472, 1, 0, generic_error), /* unknown mode char to me */ IRCB(473, 1, 0, generic_error), /* cannot join (invite only) */ IRCB(474, 1, 0, generic_error), /* cannot join (banned) */ IRCB(475, 1, 0, generic_error), /* cannot join (bad key) */ IRCB(476, 1, 0, generic_error), /* bad channel mask */ IRCB(477, 1, 0, generic_error), /* channel doesn't support modes */ IRCB(481, 1, 0, generic_error), /* you're not an IRC operator */ IRCB(482, 1, 0, generic_error), /* you're not channel operator */ IRCB(483, 1, 0, generic_error), /* you can't kill a server! */ IRCB(484, 1, 0, generic_error), /* your connection is restricted! */ IRCB(485, 1, 0, generic_error), /* user immune from kick/deop */ IRCB(487, 1, 0, generic_error), /* network split */ IRCB(491, 1, 0, generic_error), /* no O-lines for your host */ IRCB(501, 1, 0, generic_error), /* unknown mode flag */ IRCB(502, 1, 0, generic_error), /* can't chg mode for other users */ IRCB(671, 1, 0, whois_nick_msg), /* whois (secure connection) */ IRCB(728, 1, 0, 728), /* quietlist */ IRCB(729, 1, 0, 729), /* end of quietlist */ IRCB(730, 1, 0, 730), /* monitored nicks online */ IRCB(731, 1, 0, 731), /* monitored nicks offline */ IRCB(732, 1, 0, 732), /* list of monitored nicks */ IRCB(733, 1, 0, 733), /* end of monitor list */ IRCB(734, 1, 0, 734), /* monitor list is full */ IRCB(900, 1, 0, 900), /* logged in as (SASL) */ IRCB(901, 1, 0, 901), /* you are now logged in */ IRCB(902, 1, 0, sasl_end_fail), /* SASL auth failed (acc. locked) */ IRCB(903, 1, 0, sasl_end_ok), /* SASL auth successful */ IRCB(904, 1, 0, sasl_end_fail), /* SASL auth failed */ IRCB(905, 1, 0, sasl_end_fail), /* SASL message too long */ IRCB(906, 1, 0, sasl_end_fail), /* SASL authentication aborted */ IRCB(907, 1, 0, sasl_end_ok), /* already completed SASL auth */ IRCB(936, 1, 0, generic_error), /* censored word */ IRCB(973, 1, 0, server_mode_reason), /* whois (secure conn.) */ IRCB(974, 1, 0, server_mode_reason), /* whois (secure conn.) */ IRCB(975, 1, 0, server_mode_reason), /* whois (secure conn.) */ { NULL, 0, 0, NULL }, }; if (!msg_command) return; message_colors_decoded = NULL; argv = NULL; argv_eol = NULL; date = 0; hash_tags = NULL; ptr_msg_after_tags = irc_message; /* get tags as hashtable */ if (irc_message && (irc_message[0] == '@')) { pos_space = strchr (irc_message, ' '); if (pos_space) { tags = weechat_strndup (irc_message + 1, pos_space - (irc_message + 1)); if (tags) { hash_tags = weechat_hashtable_new (32, WEECHAT_HASHTABLE_STRING, WEECHAT_HASHTABLE_STRING, NULL, NULL); if (hash_tags) { irc_tag_parse (tags, hash_tags, NULL); date = irc_protocol_parse_time ( weechat_hashtable_get (hash_tags, "time")); } free (tags); } ptr_msg_after_tags = pos_space; while (ptr_msg_after_tags[0] == ' ') { ptr_msg_after_tags++; } } else ptr_msg_after_tags = NULL; } /* get nick/host/address from IRC message */ nick1 = NULL; address1 = NULL; host1 = NULL; if (ptr_msg_after_tags && (ptr_msg_after_tags[0] == ':')) { nick1 = irc_message_get_nick_from_host (ptr_msg_after_tags); address1 = irc_message_get_address_from_host (ptr_msg_after_tags); host1 = ptr_msg_after_tags + 1; } nick = (nick1) ? strdup (nick1) : NULL; address = (address1) ? strdup (address1) : NULL; address_color = (address) ? irc_color_decode ( address, weechat_config_boolean (irc_config_network_colors_receive)) : NULL; host = (host1) ? strdup (host1) : NULL; if (host) { pos_space = strchr (host, ' '); if (pos_space) pos_space[0] = '\0'; } host_no_color = (host) ? irc_color_decode (host, 0) : NULL; host_color = (host) ? irc_color_decode ( host, weechat_config_boolean (irc_config_network_colors_receive)) : NULL; /* check if message is ignored or not */ ptr_channel = NULL; if (msg_channel) ptr_channel = irc_channel_search (server, msg_channel); message_ignored = irc_ignore_check ( server, (ptr_channel) ? ptr_channel->name : msg_channel, nick, host_no_color); /* send signal with received command, even if command is ignored */ return_code = irc_server_send_signal (server, "irc_raw_in", msg_command, irc_message, NULL); if (return_code == WEECHAT_RC_OK_EAT) goto end; /* send signal with received command, only if message is not ignored */ if (!message_ignored) { return_code = irc_server_send_signal (server, "irc_in", msg_command, irc_message, NULL); if (return_code == WEECHAT_RC_OK_EAT) goto end; } /* look for IRC command */ cmd_found = -1; for (i = 0; irc_protocol_messages[i].name; i++) { if (weechat_strcasecmp (irc_protocol_messages[i].name, msg_command) == 0) { cmd_found = i; break; } } /* command not found */ if (cmd_found < 0) { /* for numeric commands, we use default recv function */ if (irc_protocol_is_numeric_command (msg_command)) { cmd_name = msg_command; decode_color = 1; keep_trailing_spaces = 0; cmd_recv_func = irc_protocol_cb_numeric; } else { weechat_printf (server->buffer, _("%s%s: command \"%s\" not found: \"%s\""), weechat_prefix ("error"), IRC_PLUGIN_NAME, msg_command, irc_message); goto end; } } else { cmd_name = irc_protocol_messages[cmd_found].name; decode_color = irc_protocol_messages[cmd_found].decode_color; keep_trailing_spaces = irc_protocol_messages[cmd_found].keep_trailing_spaces; cmd_recv_func = irc_protocol_messages[cmd_found].recv_function; } if (cmd_recv_func != NULL) { if (ptr_msg_after_tags) { if (decode_color) { message_colors_decoded = irc_color_decode ( ptr_msg_after_tags, weechat_config_boolean (irc_config_network_colors_receive)); } else { message_colors_decoded = strdup (ptr_msg_after_tags); } } else { message_colors_decoded = NULL; } argv = weechat_string_split (message_colors_decoded, " ", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &argc); flags = WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS | WEECHAT_STRING_SPLIT_KEEP_EOL; if (!keep_trailing_spaces) flags |= WEECHAT_STRING_SPLIT_STRIP_RIGHT; argv_eol = weechat_string_split (message_colors_decoded, " ", NULL, flags, 0, NULL); irc_message_parse (server, message_colors_decoded, NULL, /* tags */ NULL, /* message_without_tags */ NULL, /* nick */ NULL, /* user */ NULL, /* host */ NULL, /* command */ NULL, /* channel */ NULL, /* arguments */ NULL, /* text */ ¶ms, &num_params, NULL, /* pos_command */ NULL, /* pos_arguments */ NULL, /* pos_channel */ NULL); /* pos_text */ return_code = (int) (cmd_recv_func) (server, date, message_colors_decoded, hash_tags, nick, address_color, host_color, cmd_name, message_ignored, argc, argv, argv_eol, (const char **)params, num_params); if (return_code == WEECHAT_RC_ERROR) { weechat_printf (server->buffer, _("%s%s: failed to parse command \"%s\" (please " "report to developers): \"%s\""), weechat_prefix ("error"), IRC_PLUGIN_NAME, cmd_name, irc_message); } /* send signal with received command (if message is not ignored) */ if (!message_ignored) { (void) irc_server_send_signal (server, "irc_in2", msg_command, irc_message, NULL); } } /* send signal with received command, even if command is ignored */ (void) irc_server_send_signal (server, "irc_raw_in2", msg_command, irc_message, NULL); end: if (nick) free (nick); if (address) free (address); if (address_color) free (address_color); if (host) free (host); if (host_no_color) free (host_no_color); if (host_color) free (host_color); if (message_colors_decoded) free (message_colors_decoded); if (argv) weechat_string_free_split (argv); if (argv_eol) weechat_string_free_split (argv_eol); if (hash_tags) weechat_hashtable_free (hash_tags); }