diff options
Diffstat (limited to 'src/plugins')
-rw-r--r-- | src/plugins/irc/irc-command.c | 111 | ||||
-rw-r--r-- | src/plugins/irc/irc-info.c | 42 | ||||
-rw-r--r-- | src/plugins/irc/irc-input.c | 57 | ||||
-rw-r--r-- | src/plugins/irc/irc-message.c | 520 | ||||
-rw-r--r-- | src/plugins/irc/irc-message.h | 2 | ||||
-rw-r--r-- | src/plugins/irc/irc-protocol.c | 22 | ||||
-rw-r--r-- | src/plugins/irc/irc-server.c | 88 | ||||
-rw-r--r-- | src/plugins/irc/irc-server.h | 12 | ||||
-rw-r--r-- | src/plugins/relay/relay-client-irc.c | 151 |
9 files changed, 841 insertions, 164 deletions
diff --git a/src/plugins/irc/irc-command.c b/src/plugins/irc/irc-command.c index 5346df5ed..7495262de 100644 --- a/src/plugins/irc/irc-command.c +++ b/src/plugins/irc/irc-command.c @@ -334,20 +334,16 @@ irc_command_allserv (void *data, struct t_gui_buffer *buffer, int argc, } /* - * irc_command_me_channel: send a ctcp action to a channel + * irc_command_me_channel_display: display a ctcp action on channel */ void -irc_command_me_channel (struct t_irc_server *server, - struct t_irc_channel *channel, - const char *arguments) +irc_command_me_channel_display (struct t_irc_server *server, + struct t_irc_channel *channel, + const char *arguments) { char *string; - irc_server_sendf (server, IRC_SERVER_SEND_OUTQ_PRIO_HIGH, NULL, - "PRIVMSG %s :\01ACTION %s\01", - channel->name, - (arguments && arguments[0]) ? arguments : ""); string = (arguments && arguments[0]) ? irc_color_decode (arguments, weechat_config_boolean (irc_config_network_colors_receive)) : NULL; @@ -366,6 +362,42 @@ irc_command_me_channel (struct t_irc_server *server, } /* + * irc_command_me_channel: send a ctcp action to a channel + */ + +void +irc_command_me_channel (struct t_irc_server *server, + struct t_irc_channel *channel, + const char *arguments) +{ + struct t_hashtable *hashtable; + int number; + char hash_key[32]; + const char *str_args; + + hashtable = irc_server_sendf (server, + IRC_SERVER_SEND_OUTQ_PRIO_HIGH | IRC_SERVER_SEND_RETURN_HASHTABLE, + NULL, + "PRIVMSG %s :\01ACTION %s\01", + channel->name, + (arguments && arguments[0]) ? arguments : ""); + if (hashtable) + { + number = 1; + while (1) + { + snprintf (hash_key, sizeof (hash_key), "args%d", number); + str_args = weechat_hashtable_get (hashtable, hash_key); + if (!str_args) + break; + irc_command_me_channel_display (server, channel, str_args); + number++; + } + weechat_hashtable_free (hashtable); + } +} + +/* * irc_command_me_all_channels: send a ctcp action to all channels of a server */ @@ -1737,7 +1769,7 @@ irc_command_ison (void *data, struct t_gui_buffer *buffer, int argc, if (argc > 1) { irc_server_sendf (ptr_server, IRC_SERVER_SEND_OUTQ_PRIO_HIGH, NULL, - "ISON %s", argv_eol[1]); + "ISON :%s", argv_eol[1]); } else { @@ -2682,9 +2714,10 @@ int irc_command_notice (void *data, struct t_gui_buffer *buffer, int argc, char **argv, char **argv_eol) { - char *string; - int arg_nick, arg_text; + char *string, hash_key[32], *str_args; + int arg_nick, arg_text, number; struct t_irc_channel *ptr_channel; + struct t_hashtable *hashtable; IRC_BUFFER_GET_SERVER(buffer); @@ -2703,28 +2736,42 @@ irc_command_notice (void *data, struct t_gui_buffer *buffer, int argc, } IRC_COMMAND_CHECK_SERVER("notice", 1); - - string = irc_color_decode (argv_eol[arg_text], - weechat_config_boolean (irc_config_network_colors_receive)); ptr_channel = irc_channel_search (ptr_server, argv[arg_nick]); - weechat_printf_tags ((ptr_channel) ? ptr_channel->buffer : ptr_server->buffer, - "notify_none,no_highlight", - "%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_CHAT, - (irc_channel_is_channel (argv[arg_nick])) ? - IRC_COLOR_CHAT_CHANNEL : IRC_COLOR_CHAT_NICK, - argv[arg_nick], - IRC_COLOR_CHAT, - (string) ? string : argv_eol[arg_text]); - if (string) - free (string); - irc_server_sendf (ptr_server, IRC_SERVER_SEND_OUTQ_PRIO_HIGH, NULL, - "NOTICE %s :%s", - argv[arg_nick], argv_eol[arg_text]); + hashtable = irc_server_sendf (ptr_server, + IRC_SERVER_SEND_OUTQ_PRIO_HIGH | IRC_SERVER_SEND_RETURN_HASHTABLE, + NULL, + "NOTICE %s :%s", + argv[arg_nick], argv_eol[arg_text]); + if (hashtable) + { + number = 1; + while (1) + { + snprintf (hash_key, sizeof (hash_key), "args%d", number); + str_args = weechat_hashtable_get (hashtable, hash_key); + if (!str_args) + break; + string = irc_color_decode (str_args, + weechat_config_boolean (irc_config_network_colors_receive)); + weechat_printf_tags ((ptr_channel) ? ptr_channel->buffer : ptr_server->buffer, + "notify_none,no_highlight", + "%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_CHAT, + (irc_channel_is_channel (argv[arg_nick])) ? + IRC_COLOR_CHAT_CHANNEL : IRC_COLOR_CHAT_NICK, + argv[arg_nick], + IRC_COLOR_CHAT, + (string) ? string : str_args); + if (string) + free (string); + number++; + } + weechat_hashtable_free (hashtable); + } } else { diff --git a/src/plugins/irc/irc-info.c b/src/plugins/irc/irc-info.c index a482be480..0cad62129 100644 --- a/src/plugins/irc/irc-info.c +++ b/src/plugins/irc/irc-info.c @@ -240,23 +240,34 @@ struct t_hashtable * irc_info_get_info_hashtable_cb (void *data, const char *info_name, struct t_hashtable *hashtable) { - const char *message; + const char *server, *message; + struct t_irc_server *ptr_server; struct t_hashtable *value; /* make C compiler happy */ (void) data; - if (weechat_strcasecmp (info_name, "irc_parse_message") == 0) + if (!hashtable) + return NULL; + + if (weechat_strcasecmp (info_name, "irc_message_parse") == 0) { - if (hashtable) + message = weechat_hashtable_get (hashtable, "message"); + if (message) { - message = (const char *)weechat_hashtable_get (hashtable, - "message"); - if (message) - { - value = irc_message_parse_to_hashtable (message); - return value; - } + value = irc_message_parse_to_hashtable (message); + return value; + } + } + else if (weechat_strcasecmp (info_name, "irc_message_split") == 0) + { + server = weechat_hashtable_get (hashtable, "server"); + ptr_server = (server) ? irc_server_search (server) : NULL; + message = weechat_hashtable_get (hashtable, "message"); + if (message) + { + value = irc_message_split (ptr_server, message); + return value; } } @@ -592,7 +603,7 @@ irc_info_init () &irc_info_get_info_cb, NULL); /* info_hashtable hooks */ - weechat_hook_info_hashtable ("irc_parse_message", + weechat_hook_info_hashtable ("irc_message_parse", N_("parse an IRC message"), N_("\"message\": IRC message"), /* TRANSLATORS: please do not translate key names (enclosed by quotes) */ @@ -600,6 +611,15 @@ irc_info_init () "\"command\": command, \"channel\": channel, " "\"arguments\": arguments (includes channel)"), &irc_info_get_info_hashtable_cb, NULL); + weechat_hook_info_hashtable ("irc_message_split", + N_("split an IRC message (to fit in 512 bytes)"), + N_("\"message\": IRC message, \"server\": server " + "name (optional)"), + /* TRANSLATORS: please do not translate key names (enclosed by quotes) */ + N_("\"msg1\" ... \"msgN\": messages to send " + "(without final \"\\r\\n\"), " + "\"args1\" ... \"argsN\": arguments of messages"), + &irc_info_get_info_hashtable_cb, NULL); /* infolist hooks */ weechat_hook_infolist ("irc_server", diff --git a/src/plugins/irc/irc-input.c b/src/plugins/irc/irc-input.c index cce50d1f7..0a1b7b5ec 100644 --- a/src/plugins/irc/irc-input.c +++ b/src/plugins/irc/irc-input.c @@ -22,6 +22,7 @@ */ #include <stdlib.h> +#include <stdio.h> #include <string.h> #include "../weechat-plugin.h" @@ -85,8 +86,9 @@ void irc_input_send_user_message (struct t_gui_buffer *buffer, int flags, const char *tags, char *message) { - int max_length; - char *pos, *pos_max, *last_space, *pos_next, *next, saved_char; + int number; + char hash_key[32], *str_args; + struct t_hashtable *hashtable; IRC_BUFFER_GET_SERVER_CHANNEL(buffer); @@ -100,45 +102,24 @@ irc_input_send_user_message (struct t_gui_buffer *buffer, int flags, weechat_prefix ("error"), IRC_PLUGIN_NAME); return; } - - next = NULL; - last_space = NULL; - saved_char = '\0'; - - max_length = 512 - 16 - 65 - 10 - strlen (ptr_server->nick) - - strlen (ptr_channel->name); - - if (max_length > 0) + hashtable = irc_server_sendf (ptr_server, + flags | IRC_SERVER_SEND_RETURN_HASHTABLE, + tags, + "PRIVMSG %s :%s", + ptr_channel->name, message); + if (hashtable) { - if ((int)strlen (message) > max_length) + number = 1; + while (1) { - pos = message; - pos_max = message + max_length; - while (pos[0]) - { - if (pos[0] == ' ') - last_space = pos; - pos_next = weechat_utf8_next_char (pos); - if (pos_next > pos_max) - break; - pos = pos_next; - } - if (last_space && (last_space < pos)) - pos = last_space + 1; - saved_char = pos[0]; - pos[0] = '\0'; - next = pos; + snprintf (hash_key, sizeof (hash_key), "args%d", number); + str_args = weechat_hashtable_get (hashtable, hash_key); + if (!str_args) + break; + irc_input_user_message_display (buffer, str_args); + number++; } - } - - irc_server_sendf (ptr_server, flags, tags, - "PRIVMSG %s :%s", ptr_channel->name, message); - irc_input_user_message_display (buffer, message); - - if (next) - { - next[0] = saved_char; - irc_input_send_user_message (buffer, flags, tags, next); + weechat_hashtable_free (hashtable); } } diff --git a/src/plugins/irc/irc-message.c b/src/plugins/irc/irc-message.c index c4314e0f6..28c965642 100644 --- a/src/plugins/irc/irc-message.c +++ b/src/plugins/irc/irc-message.c @@ -338,3 +338,523 @@ irc_message_replace_vars (struct t_irc_server *server, /* return result */ return res; } + +/* + * irc_message_split_add: add a message + arguments in hashtable + */ + +void +irc_message_split_add (struct t_hashtable *hashtable, int number, + const char *message, const char *arguments) +{ + char key[32]; + + if (message) + { + snprintf (key, sizeof (key), "msg%d", number); + weechat_hashtable_set (hashtable, key, message); + if (weechat_irc_plugin->debug >= 2) + { + weechat_printf (NULL, + "irc_message_split_add >> %s='%s' (%d bytes)", + key, message, + strlen (message)); + } + } + if (arguments) + { + snprintf (key, sizeof (key), "args%d", number); + weechat_hashtable_set (hashtable, key, arguments); + if (weechat_irc_plugin->debug >= 2) + { + weechat_printf (NULL, + "irc_message_split_add >> %s='%s'", + key, arguments); + } + } +} + +/* + * irc_message_split_string: split "arguments" using delimiter and max length + * messages added to hashtable are: + * host + command + target + XXX + suffix + * (where XXX is part of "arguments") + * return 1 if split ok, 0 if error + */ + +int +irc_message_split_string (struct t_hashtable *hashtable, + const char *host, + const char *command, + const char *target, + const char *prefix, + const char *arguments, + const char *suffix, + const char delimiter, + int max_length_host) +{ + const char *pos, *pos_max, *pos_next, *pos_last_delim; + char message[1024], *dup_arguments; + int max_length, number; + + /* + * Examples of arguments for this function: + * + * message..: :nick!user@host.com PRIVMSG #channel :Hello world! + * arguments: + * host : ":nick!user@host.com" + * command : "PRIVMSG" + * target : "#channel" + * prefix : ":" + * arguments: "Hello world!" + * suffix : "" + * + * message..: :nick!user@host.com PRIVMSG #channel :\01ACTION is eating\01 + * arguments: + * host : ":nick!user@host.com" + * command : "PRIVMSG" + * target : "#channel" + * prefix : ":\01ACTION " + * arguments: "is eating" + * suffix : "\01" + */ + + max_length = 510; + if (max_length_host >= 0) + max_length -= max_length_host; + else + max_length -= (host) ? strlen (host) + 1 : 0; + max_length -= strlen (command) + 1; + if (target) + max_length -= strlen (target); + if (prefix) + max_length -= strlen (prefix); + if (suffix) + max_length -= strlen (suffix); + + if (max_length < 2) + return 0; + + /* debug message */ + if (weechat_irc_plugin->debug >= 2) + { + weechat_printf (NULL, + "irc_message_split_string: host='%s', command='%s', " + "target='%s', prefix='%s', arguments='%s', " + "suffix='%s', max_length=%d", + host, command, target, prefix, arguments, suffix, + max_length); + } + + number = 1; + + if (!arguments || !arguments[0]) + { + snprintf (message, sizeof (message), "%s%s%s %s%s%s%s", + (host) ? host : "", + (host) ? " " : "", + command, + (target) ? target : "", + (target && target[0]) ? " " : "", + (prefix) ? prefix : "", + (suffix) ? suffix : ""); + irc_message_split_add (hashtable, 1, message, ""); + return 1; + } + + while (arguments && arguments[0]) + { + pos = arguments; + pos_max = pos + max_length; + pos_last_delim = NULL; + while (pos && pos[0]) + { + if (pos[0] == delimiter) + pos_last_delim = pos; + pos_next = weechat_utf8_next_char (pos); + if (pos_next > pos_max) + break; + pos = pos_next; + } + if (pos[0] && pos_last_delim) + pos = pos_last_delim; + dup_arguments = weechat_strndup (arguments, pos - arguments); + if (dup_arguments) + { + snprintf (message, sizeof (message), "%s%s%s %s%s%s%s%s", + (host) ? host : "", + (host) ? " " : "", + command, + (target) ? target : "", + (target && target[0]) ? " " : "", + (prefix) ? prefix : "", + dup_arguments, + (suffix) ? suffix : ""); + irc_message_split_add (hashtable, number, message, dup_arguments); + number++; + free (dup_arguments); + } + arguments = (pos == pos_last_delim) ? pos + 1 : pos; + } + + return 1; +} + +/* + * irc_message_split_join: split a JOIN message, taking care of keeping + * channel keys with channel names + * return 1 if split ok, 0 if error + */ + +int +irc_message_split_join (struct t_hashtable *hashtable, + const char *host, const char *arguments) +{ + int number, channels_count, keys_count, length, length_no_channel; + int length_to_add, index_channel; + char **channels, **keys, *pos, *str; + char msg_to_send[2048], keys_to_add[2048]; + + number = 1; + + channels = NULL; + channels_count = 0; + keys = NULL; + keys_count = 0; + + pos = strchr (arguments, ' '); + if (pos) + { + str = weechat_strndup (arguments, pos - arguments); + if (!str) + return 0; + channels = weechat_string_split (str, ",", 0, 0, &channels_count); + free (str); + while (pos[0] == ' ') + { + pos++; + } + if (pos[0]) + keys = weechat_string_split (pos, ",", 0, 0, &keys_count); + } + else + { + channels = weechat_string_split (arguments, ",", 0, 0, &channels_count); + } + + snprintf (msg_to_send, sizeof (msg_to_send), "%s%sJOIN", + (host) ? host : "", + (host) ? " " : ""); + length = strlen (msg_to_send); + length_no_channel = length; + keys_to_add[0] = '\0'; + index_channel = 0; + while (index_channel < channels_count) + { + length_to_add = 1 + strlen (channels[index_channel]); + if (index_channel < keys_count) + length_to_add += 1 + strlen (keys[index_channel]); + if ((length + length_to_add < 510) || (length == length_no_channel)) + { + if (length + length_to_add < (int)sizeof (msg_to_send)) + { + strcat (msg_to_send, (length == length_no_channel) ? " " : ","); + strcat (msg_to_send, channels[index_channel]); + } + if (index_channel < keys_count) + { + if (strlen (keys_to_add) + 1 + + strlen (keys[index_channel]) < (int)sizeof (keys_to_add)) + { + strcat (keys_to_add, (keys_to_add[0]) ? "," : " "); + strcat (keys_to_add, keys[index_channel]); + } + } + length += length_to_add; + index_channel++; + } + else + { + strcat (msg_to_send, keys_to_add); + irc_message_split_add (hashtable, number, + msg_to_send, + msg_to_send + length_no_channel + 1); + number++; + snprintf (msg_to_send, sizeof (msg_to_send), "%s%sJOIN", + (host) ? host : "", + (host) ? " " : ""); + length = strlen (msg_to_send); + keys_to_add[0] = '\0'; + } + } + + if (length > length_no_channel) + { + strcat (msg_to_send, keys_to_add); + irc_message_split_add (hashtable, number, + msg_to_send, + msg_to_send + length_no_channel + 1); + } + + if (channels) + weechat_string_free_split (channels); + if (keys) + weechat_string_free_split (keys); + + return 1; +} + +/* + * irc_message_split_privmsg: split a PRIVMSG message, taking care of keeping + * the '\01' char used in CTCP messages + * return 1 if split ok, 0 if error + */ + +int +irc_message_split_privmsg (struct t_hashtable *hashtable, + char *host, char *command, char *target, + char *arguments, int max_length_host) +{ + char prefix[512], suffix[2], *pos, saved_char; + int length, rc; + + /* + * message sent looks like: + * PRIVMSG #channel :hello world! + * + * when IRC server sends message to other people, message looks like: + * :nick!user@host.com PRIVMSG #channel :hello world! + */ + + /* for CTCP, target2 will be '\01xxxx' and suffix '\01' */ + prefix[0] = '\0'; + suffix[0] = '\0'; + length = strlen (arguments); + if ((arguments[0] == '\01') + && (arguments[length - 1] == '\01')) + { + pos = strchr (arguments, ' '); + if (pos) + { + pos++; + saved_char = pos[0]; + pos[0] = '\0'; + snprintf (prefix, sizeof (prefix), ":%s", arguments); + pos[0] = saved_char; + arguments[length - 1] = '\0'; + arguments = pos; + suffix[0] = '\01'; + suffix[1] = '\0'; + } + } + if (!prefix[0]) + strcpy (prefix, ":"); + + rc = irc_message_split_string (hashtable, host, command, target, + prefix, arguments, suffix, + ' ', max_length_host); + + return rc; +} + +/* + * irc_message_split_005: split a 005 message (isupport) + * return 1 if split ok, 0 if error + */ + +int +irc_message_split_005 (struct t_hashtable *hashtable, + char *host, char *command, char *target, char *arguments) +{ + char *pos, suffix[512]; + + /* + * 005 message looks like: + * :server 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 + */ + + /* search suffix */ + suffix[0] = '\0'; + pos = strstr (arguments, " :"); + if (pos) + { + snprintf (suffix, sizeof (suffix), "%s", pos); + pos[0] = '\0'; + } + + return irc_message_split_string (hashtable, host, command, target, + NULL, arguments, suffix, ' ', -1); +} + +/* + * irc_message_split: split an IRC message about to be sent to IRC server + * The maximum length of an IRC message is 510 bytes for + * user data + final "\r\n", so full size is 512 bytes. + * The split takes care about type of message to do a split + * at best place in message. + * The hashtable returned contains keys "msg1", "msg2", ..., + * "msgN" with split of message (these messages do not + * include the final "\r\n"). + * Hashtable contains "args1", "args2", ..., "argsN" with + * split of arguments only (no host/command here). + * Each message in hashtable has command and arguments, and + * then is ready to be sent to IRC server. + */ + +struct t_hashtable * +irc_message_split (struct t_irc_server *server, const char *message) +{ + struct t_hashtable *hashtable; + char **argv, **argv_eol, *host, *command, *arguments, target[512]; + int split_ok, argc, index_args, max_length_nick, max_length_host; + + split_ok = 0; + host = NULL; + command = NULL; + arguments = NULL; + index_args = 0; + + /* debug message */ + if (weechat_irc_plugin->debug >= 2) + weechat_printf (NULL, "irc_message_split: message='%s'", message); + + hashtable = weechat_hashtable_new (8, + WEECHAT_HASHTABLE_STRING, + WEECHAT_HASHTABLE_STRING, + NULL, + NULL); + if (!hashtable) + return NULL; + + if (!message || !message[0]) + goto end; + + argv = weechat_string_split (message, " ", 0, 0, &argc); + argv_eol = weechat_string_split (message, " ", 1, 0, NULL); + + if (argc < 2) + goto end; + + if (argv[0][0] == ':') + { + if (argc < 3) + goto end; + host = argv[0]; + command = argv[1]; + arguments = argv_eol[2]; + index_args = 2; + } + else + { + command = argv[0]; + arguments = argv_eol[1]; + index_args = 1; + } + + max_length_nick = (server && (server->nick_max_length > 0)) ? + server->nick_max_length : 16; + max_length_host = 1 + /* ":" */ + max_length_nick + /* nick */ + 1 + /* "!" */ + 63 + /* host */ + 1; /* " " */ + + if ((weechat_strcasecmp (command, "ison") == 0) + || (weechat_strcasecmp (command, "wallops") == 0)) + { + split_ok = irc_message_split_string (hashtable, host, command, + NULL, ":", + (argv_eol[index_args][0] == ':') ? + argv_eol[index_args] + 1 : argv_eol[index_args], + NULL, ' ', max_length_host); + } + else if (weechat_strcasecmp (command, "join") == 0) + { + /* split join (if it's more than 510 bytes) */ + if (strlen (message) > 510) + split_ok = irc_message_split_join (hashtable, host, arguments); + } + else if (weechat_strcasecmp (command, "notice") == 0) + { + if (index_args + 1 <= argc - 1) + { + split_ok = irc_message_split_string (hashtable, host, command, + argv[index_args], ":", + (argv_eol[index_args + 1][0] == ':') ? + argv_eol[index_args + 1] + 1 : argv_eol[index_args + 1], + NULL, ' ', max_length_host); + } + } + else if (weechat_strcasecmp (command, "privmsg") == 0) + { + /* split privmsg */ + if (index_args + 1 <= argc - 1) + { + split_ok = irc_message_split_privmsg (hashtable, host, command, + argv[index_args], + (argv_eol[index_args + 1][0] == ':') ? + argv_eol[index_args + 1] + 1 : argv_eol[index_args + 1], + max_length_host); + } + } + else if (weechat_strcasecmp (command, "005") == 0) + { + /* split 005 (isupport) */ + if (index_args + 1 <= argc - 1) + { + split_ok = irc_message_split_005 (hashtable, host, command, + argv[index_args], + (argv_eol[index_args + 1][0] == ':') ? + argv_eol[index_args + 1] + 1 : argv_eol[index_args + 1]); + } + } + else if (weechat_strcasecmp (command, "353") == 0) + { + /* + * split 353 (list of users on channel): + * :server 353 mynick = #channel :mynick nick1 @nick2 +nick3 + */ + if (index_args + 2 <= argc - 1) + { + if (irc_channel_is_channel (argv[index_args + 1])) + { + snprintf (target, sizeof (target), "%s %s", + argv[index_args], argv[index_args + 1]); + split_ok = irc_message_split_string (hashtable, host, command, + target, ":", + (argv_eol[index_args + 2][0] == ':') ? + argv_eol[index_args + 2] + 1 : argv_eol[index_args + 2], + NULL, ' ', -1); + } + else + { + if (index_args + 3 <= argc - 1) + { + snprintf (target, sizeof (target), "%s %s %s", + argv[index_args], argv[index_args + 1], + argv[index_args + 2]); + split_ok = irc_message_split_string (hashtable, host, command, + target, ":", + (argv_eol[index_args + 3][0] == ':') ? + argv_eol[index_args + 3] + 1 : argv_eol[index_args + 3], + NULL, ' ', -1); + } + } + } + } + +end: + if (!split_ok + || (weechat_hashtable_get_integer (hashtable, "items_count") == 0)) + { + irc_message_split_add (hashtable, 1, message, arguments); + } + + weechat_string_free_split (argv); + weechat_string_free_split (argv_eol); + + return hashtable; +} diff --git a/src/plugins/irc/irc-message.h b/src/plugins/irc/irc-message.h index e68d994f8..25eb4c4d8 100644 --- a/src/plugins/irc/irc-message.h +++ b/src/plugins/irc/irc-message.h @@ -32,5 +32,7 @@ extern const char *irc_message_get_address_from_host (const char *host); extern char *irc_message_replace_vars (struct t_irc_server *server, struct t_irc_channel *channel, const char *string); +extern struct t_hashtable *irc_message_split (struct t_irc_server *server, + const char *message); #endif /* __WEECHAT_IRC_MESSAGE_H */ diff --git a/src/plugins/irc/irc-protocol.c b/src/plugins/irc/irc-protocol.c index 9512c1dcf..17dbd7b77 100644 --- a/src/plugins/irc/irc-protocol.c +++ b/src/plugins/irc/irc-protocol.c @@ -1908,9 +1908,9 @@ IRC_PROTOCOL_CALLBACK(001) IRC_PROTOCOL_CALLBACK(005) { - char *pos, *pos2, *pos_start; - int length_isupport, length; - + char *pos, *pos2, *pos_start, *error; + int length_isupport, length, nick_max_length; + /* * 005 message looks like: * :server 005 mynick MODES=4 CHANLIMIT=#:20 NICKLEN=16 USERLEN=10 @@ -1938,6 +1938,22 @@ IRC_PROTOCOL_CALLBACK(005) pos2[0] = ' '; } + /* save max nick length */ + pos = strstr (argv_eol[3], "NICKLEN="); + if (pos) + { + pos += 8; + pos2 = strchr (pos, ' '); + if (pos2) + pos2[0] = '\0'; + error = NULL; + nick_max_length = (int)strtol (pos, &error, 10); + if (error && !error[0] && (nick_max_length > 0)) + server->nick_max_length = nick_max_length; + if (pos2) + pos2[0] = ' '; + } + /* save whole message (concatenate to existing isupport, if any) */ pos_start = NULL; pos = strstr (argv_eol[3], " :"); diff --git a/src/plugins/irc/irc-server.c b/src/plugins/irc/irc-server.c index 2de04d13d..fed0b122d 100644 --- a/src/plugins/irc/irc-server.c +++ b/src/plugins/irc/irc-server.c @@ -633,6 +633,7 @@ irc_server_alloc (const char *name) new_server->isupport = NULL; new_server->prefix_modes = NULL; new_server->prefix_chars = NULL; + new_server->nick_max_length = 0; new_server->reconnect_delay = 0; new_server->reconnect_start = 0; new_server->command_time = 0; @@ -1738,32 +1739,85 @@ irc_server_send_one_msg (struct t_irc_server *server, int flags, /* * irc_server_sendf: send formatted data to IRC server * many messages may be sent, separated by '\n' + * if flags contains "IRC_SERVER_SEND_RETURN_HASHTABLE", then + * hashtable with split of message is returned + * (see function irc_message_split() in irc-message.c) + * note: hashtable must be freed after use */ -void +struct t_hashtable * irc_server_sendf (struct t_irc_server *server, int flags, const char *tags, const char *format, ...) { - va_list args; - static char buffer[4096]; - char **items; - int i, items_count; + char **items, hash_key[32]; + const char *str_message, *str_args; + int i, items_count, number, ret_number, rc; + struct t_hashtable *hashtable, *ret_hashtable; if (!server) - return; + return NULL; - va_start (args, format); - vsnprintf (buffer, sizeof (buffer) - 1, format, args); - va_end (args); - - items = weechat_string_split (buffer, "\n", 0, 0, &items_count); + weechat_va_format (format); + if (!vbuffer) + return NULL; + + ret_hashtable = NULL; + ret_number = 1; + if (flags & IRC_SERVER_SEND_RETURN_HASHTABLE) + { + ret_hashtable = weechat_hashtable_new (8, + WEECHAT_HASHTABLE_STRING, + WEECHAT_HASHTABLE_STRING, + NULL, + NULL); + } + + rc = 1; + items = weechat_string_split (vbuffer, "\n", 0, 0, &items_count); for (i = 0; i < items_count; i++) { - if (!irc_server_send_one_msg (server, flags, items[i], tags)) - break; + /* split message if needed (max is 512 bytes including final "\r\n") */ + hashtable = irc_message_split (server, items[i]); + if (hashtable) + { + number = 1; + while (1) + { + snprintf (hash_key, sizeof (hash_key), "msg%d", number); + str_message = weechat_hashtable_get (hashtable, hash_key); + if (!str_message) + break; + snprintf (hash_key, sizeof (hash_key), "args%d", number); + str_args = weechat_hashtable_get (hashtable, hash_key); + + rc = irc_server_send_one_msg (server, flags, str_message, tags); + if (!rc) + break; + + if (ret_hashtable) + { + snprintf (hash_key, sizeof (hash_key), "msg%d", ret_number); + weechat_hashtable_set (ret_hashtable, hash_key, str_message); + if (str_args) + { + snprintf (hash_key, sizeof (hash_key), "args%d", ret_number); + weechat_hashtable_set (ret_hashtable, hash_key, str_args); + } + ret_number++; + } + number++; + } + weechat_hashtable_free (hashtable); + if (!rc) + break; + } } if (items) weechat_string_free_split (items); + + free (vbuffer); + + return ret_hashtable; } /* @@ -3550,14 +3604,14 @@ irc_server_autojoin_channels (struct t_irc_server *server) if (ptr_channel->key) { irc_server_sendf (server, - IRC_SERVER_SEND_OUTQ_PRIO_LOW, NULL, + IRC_SERVER_SEND_OUTQ_PRIO_HIGH, NULL, "JOIN %s %s", ptr_channel->name, ptr_channel->key); } else { irc_server_sendf (server, - IRC_SERVER_SEND_OUTQ_PRIO_LOW, NULL, + IRC_SERVER_SEND_OUTQ_PRIO_HIGH, NULL, "JOIN %s", ptr_channel->name); } @@ -3995,6 +4049,7 @@ irc_server_hdata_server_cb (void *data, const char *hdata_name) WEECHAT_HDATA_VAR(struct t_irc_server, isupport, STRING, NULL); WEECHAT_HDATA_VAR(struct t_irc_server, prefix_modes, STRING, NULL); WEECHAT_HDATA_VAR(struct t_irc_server, prefix_chars, STRING, NULL); + WEECHAT_HDATA_VAR(struct t_irc_server, nick_max_length, INTEGER, NULL); WEECHAT_HDATA_VAR(struct t_irc_server, reconnect_delay, INTEGER, NULL); WEECHAT_HDATA_VAR(struct t_irc_server, reconnect_start, TIME, NULL); WEECHAT_HDATA_VAR(struct t_irc_server, command_time, TIME, NULL); @@ -4180,6 +4235,8 @@ irc_server_add_to_infolist (struct t_infolist *infolist, return 0; if (!weechat_infolist_new_var_string (ptr_item, "prefix_chars", server->prefix_chars)) return 0; + if (!weechat_infolist_new_var_integer (ptr_item, "nick_max_length", server->nick_max_length)) + return 0; if (!weechat_infolist_new_var_integer (ptr_item, "reconnect_delay", server->reconnect_delay)) return 0; if (!weechat_infolist_new_var_time (ptr_item, "reconnect_start", server->reconnect_start)) @@ -4481,6 +4538,7 @@ irc_server_print_log () weechat_log_printf (" isupport . . . . . . : '%s'", ptr_server->isupport); weechat_log_printf (" prefix_modes . . . . : '%s'", ptr_server->prefix_modes); weechat_log_printf (" prefix_chars . . . . : '%s'", ptr_server->prefix_chars); + weechat_log_printf (" nick_max_length. . . : %d", ptr_server->nick_max_length); weechat_log_printf (" reconnect_delay. . . : %d", ptr_server->reconnect_delay); weechat_log_printf (" reconnect_start. . . : %ld", ptr_server->reconnect_start); weechat_log_printf (" command_time . . . . : %ld", ptr_server->command_time); diff --git a/src/plugins/irc/irc-server.h b/src/plugins/irc/irc-server.h index ae2c53677..01ec048bd 100644 --- a/src/plugins/irc/irc-server.h +++ b/src/plugins/irc/irc-server.h @@ -98,8 +98,9 @@ enum t_irc_server_option #define IRC_SERVER_NUM_OUTQUEUES_PRIO 2 /* flags for irc_server_sendf() */ -#define IRC_SERVER_SEND_OUTQ_PRIO_HIGH 1 -#define IRC_SERVER_SEND_OUTQ_PRIO_LOW 2 +#define IRC_SERVER_SEND_OUTQ_PRIO_HIGH 1 +#define IRC_SERVER_SEND_OUTQ_PRIO_LOW 2 +#define IRC_SERVER_SEND_RETURN_HASHTABLE 4 /* output queue of messages to server (for sending slowly to server) */ @@ -154,6 +155,7 @@ struct t_irc_server char *isupport; /* copy of message 005 (ISUPPORT) */ char *prefix_modes; /* prefix modes from msg 005 (eg "aohv") */ char *prefix_chars; /* prefix chars from msg 005 (eg "&@%+") */ + int nick_max_length; /* max lenth of nick (from msg 005) */ int reconnect_delay; /* current reconnect delay (growing) */ time_t reconnect_start; /* this time + delay = reconnect time */ time_t command_time; /* this time + command_delay = time to */ @@ -240,8 +242,10 @@ extern void irc_server_send_signal (struct t_irc_server *server, const char *full_message, const char *tags); extern void irc_server_set_send_default_tags (const char *tags); -extern void irc_server_sendf (struct t_irc_server *server, int flags, - const char *tags, const char *format, ...); +extern struct t_hashtable *irc_server_sendf (struct t_irc_server *server, + int flags, + const char *tags, + const char *format, ...); extern struct t_irc_server *irc_server_search (const char *server_name); extern void irc_server_set_buffer_title (struct t_irc_server *server); extern struct t_gui_buffer *irc_server_create_buffer (struct t_irc_server *server); diff --git a/src/plugins/relay/relay-client-irc.c b/src/plugins/relay/relay-client-irc.c index 6f898f270..2c7eda9b1 100644 --- a/src/plugins/relay/relay-client-irc.c +++ b/src/plugins/relay/relay-client-irc.c @@ -91,11 +91,11 @@ relay_client_irc_command_ignored (const char *irc_command) } /* - * relay_client_irc_parse_message: parse IRC message + * relay_client_irc_message_parse: parse IRC message */ struct t_hashtable * -relay_client_irc_parse_message (const char *message) +relay_client_irc_message_parse (const char *message) { struct t_hashtable *hash_msg, *hash_parsed; @@ -115,7 +115,7 @@ relay_client_irc_parse_message (const char *message) goto end; } weechat_hashtable_set (hash_msg, "message", message); - hash_parsed = weechat_info_get_hashtable ("irc_parse_message", + hash_parsed = weechat_info_get_hashtable ("irc_message_parse", hash_msg); if (!hash_parsed) { @@ -139,55 +139,77 @@ end: int relay_client_irc_sendf (struct t_relay_client *client, const char *format, ...) { - va_list args; - static char buffer[4096]; - int length, num_sent; - char *pos; + int length, num_sent, total_sent, number; + char *pos, hash_key[32], *message; + const char *str_message; + struct t_hashtable *hashtable_in, *hashtable_out; if (!client) return 0; - va_start (args, format); - vsnprintf (buffer, sizeof (buffer) - 3, format, args); - va_end (args); - - if (weechat_relay_plugin->debug >= 2) - { - weechat_printf (NULL, "%s: send: %s", - RELAY_PLUGIN_NAME, buffer); - } + weechat_va_format (format); + if (!vbuffer) + return 0; - length = strlen (buffer); + total_sent = 0; - pos = strchr (buffer, '\r'); + pos = strchr (vbuffer, '\r'); if (pos) pos[0] = '\0'; - - relay_raw_print (client, 1, buffer); - + pos = strchr (vbuffer, '\n'); if (pos) - pos[0] = '\r'; - else + pos[0] = '\0'; + + hashtable_in = weechat_hashtable_new (8, + WEECHAT_HASHTABLE_STRING, + WEECHAT_HASHTABLE_STRING, + NULL, + NULL); + if (hashtable_in) { - buffer[length] = '\r'; - buffer[length + 1] = '\n'; - buffer[length + 2] = '\0'; - length += 2; + weechat_hashtable_set (hashtable_in, "server", client->protocol_args); + weechat_hashtable_set (hashtable_in, "message", vbuffer); + hashtable_out = weechat_info_get_hashtable ("irc_message_split", + hashtable_in); + if (hashtable_out) + { + number = 1; + while (1) + { + snprintf (hash_key, sizeof (hash_key), "msg%d", number); + str_message = weechat_hashtable_get (hashtable_out, hash_key); + if (!str_message) + break; + relay_raw_print (client, 1, str_message); + length = strlen (str_message) + 16 + 1; + message = malloc (length); + if (message) + { + snprintf (message, length, "%s\r\n", str_message); + num_sent = send (client->sock, message, strlen (message), 0); + if (num_sent >= 0) + total_sent += num_sent; + else + { + weechat_printf (NULL, + _("%s%s: error sending data to client: %s"), + weechat_prefix ("error"), RELAY_PLUGIN_NAME, + strerror (errno)); + } + free (message); + } + number++; + } + weechat_hashtable_free (hashtable_out); + } + weechat_hashtable_free (hashtable_in); } - num_sent = send (client->sock, buffer, length, 0); + client->bytes_sent += total_sent; - if (num_sent >= 0) - client->bytes_sent += num_sent; - else - { - weechat_printf (NULL, - _("%s%s: error sending data to client: %s"), - weechat_prefix ("error"), RELAY_PLUGIN_NAME, - strerror (errno)); - } + free (vbuffer); - return num_sent; + return total_sent; } /* @@ -220,7 +242,7 @@ relay_client_irc_signal_irc_in2_cb (void *data, const char *signal, ptr_msg); } - hash_parsed = relay_client_irc_parse_message (ptr_msg); + hash_parsed = relay_client_irc_message_parse (ptr_msg); if (hash_parsed) { irc_nick = weechat_hashtable_get (hash_parsed, "nick"); @@ -360,7 +382,7 @@ relay_client_irc_signal_irc_outtags_cb (void *data, const char *signal, if (relay_client_irc_tag_relay_client_id (tags) == client->id) goto end; - hash_parsed = relay_client_irc_parse_message (ptr_message); + hash_parsed = relay_client_irc_message_parse (ptr_message); if (hash_parsed) { irc_command = weechat_hashtable_get (hash_parsed, "command"); @@ -588,33 +610,41 @@ relay_client_irc_input_send (struct t_relay_client *client, int flags, const char *format, ...) { - va_list args; - static char buffer[4096]; - int length; + char buf_beginning[1024], *buf; + int length_beginning, length_vbuffer; + + weechat_va_format (format); + if (!vbuffer) + return; - snprintf (buffer, sizeof (buffer), + snprintf (buf_beginning, sizeof (buf_beginning), "%s;%s;%d;relay_client_%d;", client->protocol_args, (irc_channel) ? irc_channel : "", flags, client->id); - - length = strlen (buffer); - - va_start (args, format); - vsnprintf (buffer + length, sizeof (buffer) - 1 - length, format, args); - va_end (args); - - if (weechat_relay_plugin->debug >= 2) + + length_beginning = strlen (buf_beginning); + length_vbuffer = strlen (vbuffer); + buf = malloc (length_beginning + length_vbuffer + 1); + if (buf) { - weechat_printf (NULL, - "%s: irc_input_send: \"%s\"", - RELAY_PLUGIN_NAME, buffer); + memcpy (buf, buf_beginning, length_beginning); + memcpy (buf + length_beginning, vbuffer, length_vbuffer); + buf[length_beginning + length_vbuffer] = '\0'; + if (weechat_relay_plugin->debug >= 2) + { + weechat_printf (NULL, + "%s: irc_input_send: \"%s\"", + RELAY_PLUGIN_NAME, buf); + } + + weechat_hook_signal_send ("irc_input_send", + WEECHAT_HOOK_SIGNAL_STRING, + buf); + free (buf); } - - weechat_hook_signal_send ("irc_input_send", - WEECHAT_HOOK_SIGNAL_STRING, - buffer); + free (vbuffer); } /* @@ -690,7 +720,7 @@ relay_client_irc_recv_one_msg (struct t_relay_client *client, char *data) relay_raw_print (client, 0, data); /* parse IRC message */ - hash_parsed = relay_client_irc_parse_message (data); + hash_parsed = relay_client_irc_message_parse (data); if (!hash_parsed) goto end; irc_command = weechat_hashtable_get (hash_parsed, "command"); @@ -826,7 +856,6 @@ relay_client_irc_recv_one_msg (struct t_relay_client *client, char *data) { isupport++; } - /* TODO: split this message into many messages */ relay_client_irc_sendf (client, ":%s 005 %s %s :are supported " "by this server", |