/* X-Chat
* Copyright (C) 1998 Peter Zelezny.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
*/
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include "hexchat.h"
#include "hexchatc.h"
#include "modes.h"
#include "server.h"
#include "text.h"
#include "fe.h"
#include "util.h"
#include "inbound.h"
#ifdef HAVE_STRINGS_H
#include <strings.h>
#endif
#include <glib/gprintf.h>
typedef struct
{
server *serv;
char *op;
char *deop;
char *voice;
char *devoice;
} mode_run;
static int is_prefix_char (server * serv, char c);
static void record_chan_mode (session *sess, char sign, char mode, char *arg);
static char *mode_cat (char *str, char *addition);
static void handle_single_mode (mode_run *mr, char sign, char mode, char *nick,
char *chan, char *arg, int quiet, int is_324,
const message_tags_data *tags_data);
static int mode_has_arg (server *serv, char sign, char mode);
static void mode_print_grouped (session *sess, char *nick, mode_run *mr,
const message_tags_data *tags_data);
static int mode_chanmode_type (server * serv, char mode);
/* word[] - list of nicks.
wpos - index into word[]. Where nicks really start.
end - index into word[]. Last entry plus one.
sign - a char, e.g. '+' or '-'
mode - a mode, e.g. 'o' or 'v' */
void
send_channel_modes (session *sess, char *tbuf, char *word[], int wpos,
int end, char sign, char mode, int modes_per_line)
{
int usable_modes, orig_len, len, wlen, i, max;
server *serv = sess->server;
/* sanity check. IRC RFC says three per line. */
if (serv->modes_per_line < 3)
serv->modes_per_line = 3;
if (modes_per_line < 1)
modes_per_line = serv->modes_per_line;
/* RFC max, minus length of "MODE %s " and "\r\n" and 1 +/- sign */
/* 512 - 6 - 2 - 1 - strlen(chan) */
max = 503 - strlen (sess->channel);
while (wpos < end)
{
tbuf[0] = '\0';
orig_len = len = 0;
/* we'll need this many modechars too */
len += modes_per_line;
/* how many can we fit? */
for (i = 0; i < modes_per_line; i++)
{
/* no more nicks left? */
if (wpos + i >= end)
break;
wlen = strlen (word[wpos + i]) + 1;
if (wlen + len > max)
break;
len += wlen; /* length of our whole string so far */
}
if (i < 1)
return;
usable_modes = i; /* this is how many we'll send on this line */
/* add the +/-modemodemodemode */
len = orig_len;
tbuf[len] = sign;
len++;
for (i = 0; i < usable_modes; i++)
{
tbuf[len] = mode;
len++;
}
tbuf[len] = 0; /* null terminate for the strcat() to work */
/* add all the nicknames */
for (i = 0; i < usable_modes; i++)
{
strcat (tbuf, " ");
strcat (tbuf, word[wpos + i]);
}
serv->p_mode (serv, sess->channel, tbuf);
wpos += usable_modes;
}
}
/* does 'chan' have a valid prefix? e.g. # or & */
int
is_channel (server * serv, char *chan)
{
if (strchr (serv->chantypes, chan[0]))
return 1;
return 0;
}
/* is the given char a valid nick mode char? e.g. @ or + */
static int
is_prefix_char (server * serv, char c)
{
int pos = 0;
char *np = serv->nick_prefixes;
while (np[0])
{
if (np[0] == c)
return pos;
pos++;
np++;
}
if (serv->bad_prefix)
{
if (strchr (serv->bad_nick_prefixes, c))
/* valid prefix char, but mode unknown */
return -2;
}
return -1;
}
/* returns '@' for ops etc... */
char
get_nick_prefix (server * serv, unsigned int access)
{
int pos;
char c;
for (pos = 0; pos < USERACCESS_SIZE; pos++)
{
c = serv->nick_prefixes[pos];
if (c == 0)
break;
if (access & (1 << pos))
return c;
}
return 0;
}
/* returns the access bitfield for a nickname. E.g.
@nick would return 000010 in binary
%nick would return 000100 in binary
+nick would return 001000 in binary */
unsigned int
nick_access (server * serv, char *nick, int *modechars)
{
int i;
unsigned int access = 0;
char *orig = nick;
while (*nick)
{
i = is_prefix_char (serv, *nick);
if (i == -1)
break;
/* -2 == valid prefix char, but mode unknown */
if (i != -2)
access |= (1 << i);
nick++;
}
*modechars = nick - orig;
return access;
}
/* returns the access number for a particular mode. e.g.
mode 'a' returns 0
mode 'o' returns 1
mode 'h' returns 2
mode 'v' returns 3
Also puts the nick-prefix-char in 'prefix' */
int
mode_access (server * serv, char mode, char *prefix)
{
int pos = 0;
while (serv->nick_modes[pos])
{
if (serv->nick_modes[pos] == mode)
{
*prefix = serv->nick_prefixes[pos];
return pos;
}
pos++;
}
*prefix = 0;
return -1;
}
static void
record_chan_mode (session *sess, char sign, char mode, char *arg)
{
/* Somebody needed to acutally update sess->current_modes, needed to
play nice with bouncers, and less mode calls. Also keeps modes up
to date for scripts */
server *serv = sess->server;
GString *current = g_string_new(sess->current_modes);
gint mode_pos = -1;
gchar *current_char = current->str;
gint modes_length;
gint argument_num = 0;
gint argument_offset = 0;
gint argument_length = 0;
int i = 0;
gchar *arguments_start;
/* find out if the mode currently exists */
arguments_start = g_strstr_len(current->str , -1, " ");
if (arguments_start) {
modes_length = arguments_start - current->str;
}
else {
modes_length = current->len;
/* set this to the end of the modes */
arguments_start = current->str + current->len;
}
while (mode_pos == -1 && i < modes_length)
{
if (*current_char == mode)
{
mode_pos = i;
}
else
{
i++;
current_char++;
}
}
/* if the mode currently exists and has an arg, need to know where
* (including leading space) */
if (mode_pos != -1 && mode_has_arg(serv, '+', mode))
{
current_char = current->str;
i = 0;
while (i <= mode_pos)
{
if (mode_has_arg(serv, '+', *current_char))
argument_num++;
current_char++;
i++;
}
/* check through arguments for where to start */
current_char = arguments_start;
i = 0;
while (i < argument_num && *current_char != '\0')
{
if (*current_char == ' ')
i++;
if (i != argument_num)
current_char++;
}
argument_offset = current_char - current->str;
/* how long the existing argument is for this key
* important for malloc and strncpy */
if (i == argument_num)
{
argument_length++;
current_char++;
while (*current_char != '\0' && *current_char != ' ')
{
argument_length++;
current_char++;
}
}
}
/* two cases, adding and removing a mode, handled differently */
if (sign == '+')
{
if (mode_pos != -1)
{
/* if it already exists, only need to do something (change)
* if there should be a param */
if (mode_has_arg(serv, sign, mode))
{
/* leave the old space there */
current = g_string_erase(current, argument_offset+1, argument_length-1);
current = g_string_insert(current, argument_offset+1, arg);
g_free(sess->current_modes);
sess->current_modes = g_string_free(current, FALSE);
}
}
/* mode wasn't there before */
else
{
/* insert the new mode character */
current = g_string_insert_c(current, modes_length, mode);
/* add the argument, with space if there is one */
if (mode_has_arg(serv, sign, mode))
{
current = g_string_append_c(current, ' ');
current = g_string_append(current, arg);
}
g_free(sess->current_modes);
sess->current_modes = g_string_free(current, FALSE);
}
}
else if (sign == '-' && mode_pos != -1)
{
/* remove the argument first if it has one*/
if (mode_has_arg(serv, '+', mode))
current = g_string_erase(current, argument_offset, argument_length);
/* remove the mode character */
current = g_string_erase(current, mode_pos, 1);
g_free(sess->current_modes);
sess->current_modes = g_string_free(current, FALSE);
}
}
static char *
mode_cat (char *str, char *addition)
{
int len;
if (str)
{
len = strlen (str) + strlen (addition) + 2;
str = g_realloc (str, len);
strcat (str, " ");
strcat (str, addition);
}
else
{
str = g_strdup (addition);
}
return str;
}
/* handle one mode, e.g.
handle_single_mode (mr,'+','b',"elite","#warez","banneduser",) */
static void
handle_single_mode (mode_run *mr, char sign, char mode, char *nick,
char *chan, char *arg, int quiet, int is_324,
const message_tags_data *tags_data)
{
session *sess;
server *serv = mr->serv;
char outbuf[4];
char *cm = serv->chanmodes;
gboolean supportsq = FALSE;
outbuf[0] = sign;
outbuf[1] = 0;
outbuf[2] = mode;
outbuf[3] = 0;
sess = find_channel (serv, chan);
if (!sess || !is_channel (serv, chan))
{
/* got modes for a chan we're not in! probably nickmode +isw etc */
sess = serv->front_session;
goto genmode;
}
/* is this a nick mode? */
if (strchr (serv->nick_modes, mode))
{
/* update the user in the userlist */
userlist_update_mode (sess, /*nickname */ arg, mode, sign);
} else
{
if (!is_324 && !sess->ignore_mode && mode_chanmode_type(serv, mode) >= 1)
record_chan_mode (sess, sign, mode, arg);
}
/* Is q a chanmode on this server? */
if (cm)
while (*cm)
{
if (*cm == ',')
break;
if (*cm == 'q')
supportsq = TRUE;
cm++;
}
switch (sign)
{
case '+':
switch (mode)
{
case 'k':
safe_strcpy (sess->channelkey, arg, sizeof (sess->channelkey));
fe_update_channel_key (sess);
fe_update_mode_buttons (sess, mode, sign);
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANSETKEY, sess, nick, arg, NULL,
NULL, 0, tags_data->timestamp);
return;
case 'l':
sess->limit = atoi (arg);
fe_update_channel_limit (sess);
fe_update_mode_buttons (sess, mode, sign);
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANSETLIMIT, sess, nick, arg, NULL,
NULL, 0, tags_data->timestamp);
return;
case 'o':
if (!quiet)
mr->op = mode_cat (mr->op, arg);
return;
case 'h':
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANHOP, sess, nick, arg, NULL, NULL,
0, tags_data->timestamp);
return;
case 'v':
if (!quiet)
mr->voice = mode_cat (mr->voice, arg);
return;
case 'b':
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANBAN, sess, nick, arg, NULL, NULL,
0, tags_data->timestamp);
return;
case 'e':
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANEXEMPT, sess, nick, arg, NULL,
NULL, 0, tags_data->timestamp);
return;
case 'I':
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANINVITE, sess, nick, arg, NULL, NULL,
0, tags_data->timestamp);
return;
case 'q':
if (!supportsq)
break; /* +q is owner on this server */
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANQUIET, sess, nick, arg, NULL, NULL, 0,
tags_data->timestamp);
return;
}
break;
case '-':
switch (mode)
{
case 'k':
sess->channelkey[0] = 0;
fe_update_channel_key (sess);
fe_update_mode_buttons (sess, mode, sign);
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANRMKEY, sess, nick, NULL, NULL,
NULL, 0, tags_data->timestamp);
return;
case 'l':
sess->limit = 0;
fe_update_channel_limit (sess);
fe_update_mode_buttons (sess, mode, sign);
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANRMLIMIT, sess, nick, NULL, NULL,
NULL, 0, tags_data->timestamp);
return;
case 'o':
if (!quiet)
mr->deop = mode_cat (mr->deop, arg);
return;
case 'h':
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANDEHOP, sess, nick, arg, NULL,
NULL, 0, tags_data->timestamp);
return;
case 'v':
if (!quiet)
mr->devoice = mode_cat (mr->devoice, arg);
return;
case 'b':
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANUNBAN, sess, nick, arg, NULL, NULL,
0, tags_data->timestamp);
return;
case 'e':
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANRMEXEMPT, sess, nick, arg, NULL,
NULL, 0, tags_data->timestamp);
return;
case 'I':
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANRMINVITE, sess, nick, arg, NULL,
NULL, 0, tags_data->timestamp);
return;
case 'q':
if (!supportsq)
break; /* -q is owner on this server */
if (!quiet)
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANUNQUIET, sess, nick, arg, NULL,
NULL, 0, tags_data->timestamp);
return;
}
}
fe_update_mode_buttons (sess, mode, sign);
genmode:
/* Received umode +e. If we're waiting to send JOIN then send now! */
if (mode == 'e' && sign == '+' && !serv->p_cmp (chan, serv->nick))
inbound_identified (serv);
if (!quiet)
{
if (*arg)
{
char *buf = g_strdup_printf ("%s %s", chan, arg);
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANMODEGEN, sess, nick, outbuf,
outbuf + 2, buf, 0, tags_data->timestamp);
g_free (buf);
}
else
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANMODEGEN, sess, nick, outbuf,
outbuf + 2, chan, 0, tags_data->timestamp);
}
}
/* does this mode have an arg? like +b +l +o */
static int
mode_has_arg (server * serv, char sign, char mode)
{
int type;
/* if it's a nickmode, it must have an arg */
if (strchr (serv->nick_modes, mode))
return 1;
type = mode_chanmode_type (serv, mode);
switch (type)
{
case 0: /* type A */
case 1: /* type B */
return 1;
case 2: /* type C */
if (sign == '+')
return 1;
case 3: /* type D */
return 0;
default:
return 0;
}
}
/* what type of chanmode is it? -1 for not in chanmode */
static int
mode_chanmode_type (server * serv, char mode)
{
/* see what numeric 005 CHANMODES=xxx said */
char *cm = serv->chanmodes;
int type = 0;
int found = 0;
while (*cm && !found)
{
if (*cm == ',')
{
type++;
} else if (*cm == mode)
{
found = 1;
}
cm++;
}
if (found)
return type;
/* not found? -1 */
else
return -1;
}
static void
mode_print_grouped (session *sess, char *nick, mode_run *mr,
const message_tags_data *tags_data)
{
/* print all the grouped Op/Deops */
if (mr->op)
{
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANOP, sess, nick, mr->op, NULL, NULL, 0,
tags_data->timestamp);
g_free(mr->op);
mr->op = NULL;
}
if (mr->deop)
{
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANDEOP, sess, nick, mr->deop, NULL, NULL,
0, tags_data->timestamp);
g_free(mr->deop);
mr->deop = NULL;
}
if (mr->voice)
{
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANVOICE, sess, nick, mr->voice, NULL, NULL,
0, tags_data->timestamp);
g_free(mr->voice);
mr->voice = NULL;
}
if (mr->devoice)
{
EMIT_SIGNAL_TIMESTAMP (XP_TE_CHANDEVOICE, sess, nick, mr->devoice, NULL,
NULL, 0, tags_data->timestamp);
g_free(mr->devoice);
mr->devoice = NULL;
}
}
/* handle a MODE or numeric 324 from server */
void
handle_mode (server * serv, char *word[], char *word_eol[],
char *nick, int numeric_324, const message_tags_data *tags_data)
{
session *sess;
char *chan;
char *modes;
char *argstr;
char sign;
int len;
size_t arg;
size_t i, num_args;
int num_modes;
size_t offset = 3;
int all_modes_have_args = FALSE;
int using_front_tab = FALSE;
mode_run mr;
mr.serv = serv;
mr.op = mr.deop = mr.voice = mr.devoice = NULL;
/* numeric 324 has everything 1 word later (as opposed to MODE) */
if (numeric_324)
offset++;
chan = word[offset];
modes = word[offset + 1];
if (*modes == ':')
modes++;
if (*modes == 0)
return; /* beyondirc's blank modes */
sess = find_channel (serv, chan);
if (!sess)
{
sess = serv->front_session;
using_front_tab = TRUE;
}
/* remove trailing space */
len = strlen (word_eol[offset]) - 1;
if (word_eol[offset][len] == ' ')
word_eol[offset][len] = 0;
if (prefs.hex_irc_raw_modes && !numeric_324)
EMIT_SIGNAL_TIMESTAMP (XP_TE_RAWMODES, sess, nick, word_eol[offset], 0, 0, 0,
tags_data->timestamp);
if (numeric_324 && !using_front_tab)
{
g_free (sess->current_modes);
sess->current_modes = g_strdup (word_eol[offset+1]);
}
sign = *modes;
modes++;
arg = 1;
/* count the number of arguments (e.g. after the -o+v) */
num_args = 0;
i = 1;
while ((i + offset + 1) < PDIWORDS)
{
i++;
if (!(*word[i + offset]))
break;
num_args++;
if (word[i + offset][0] == ':')
break;
}
/* count the number of modes (without the -/+ chars */
num_modes = 0;
i = 0;
while (i < strlen (modes))
{
if (modes[i] != '+' && modes[i] != '-')
num_modes++;
i++;
}
if (num_args == num_modes)
all_modes_have_args = TRUE;
while (*modes)
{
switch (*modes)
{
case '-':
case '+':
/* print all the grouped Op/Deops */
mode_print_grouped (sess, nick, &mr, tags_data);
sign = *modes;
break;
default:
argstr = "";
if ((all_modes_have_args || mode_has_arg (serv, sign, *modes)) && arg < (num_args + 1))
{
arg++;
argstr = STRIP_COLON(word, word_eol, arg+offset);
}
handle_single_mode (&mr, sign, *modes, nick, chan,
argstr, numeric_324 || prefs.hex_irc_raw_modes,
numeric_324, tags_data);
}
modes++;
}
/* update the title at the end, now that the mode update is internal now */
if (!using_front_tab)
fe_set_title (sess);
/* print all the grouped Op/Deops */
mode_print_grouped (sess, nick, &mr, tags_data);
}
static char
hex_to_chr(char chr)
{
return g_ascii_isdigit (chr) ? chr - '0' : g_ascii_tolower (chr) - 'a' + 10;
}
static void
parse_005_token (const char *token, char **name, char **value, gboolean *adding)
{
char *toksplit, *valuecurr;
size_t idx;
if (token[0] == '-')
{
*adding = FALSE;
token++;
} else
{
*adding = TRUE;
}
toksplit = strchr (token, '=');
if (toksplit && *toksplit++)
{
/* The token has a value; parse any escape codes. */
*name = g_strndup (token, toksplit - token - 1);
*value = g_malloc (strlen (toksplit) + 1);
valuecurr = *value;
while (*toksplit)
{
if (toksplit[0] == '\\')
{
/** If it's a malformed escape then just skip it. */
if (toksplit[1] == 'x' && g_ascii_isxdigit (toksplit[2]) && g_ascii_isxdigit (toksplit[3]))
*valuecurr++ = hex_to_chr (toksplit[2]) << 4 | hex_to_chr (toksplit[3]);
for (idx = 0; idx < 4; ++idx)
{
/* We need to do this to avoid jumping past the end of the array. */
if (*toksplit)
toksplit++;
}
} else
{
/** Non-escape characters can be copied as is. */
*valuecurr++ = *toksplit++;
}
}
*valuecurr++ = 0;
} else
{
/* The token has no value; store a dummy value instead. */
*name = g_strdup (token);
*value = g_strdup ("");
}
}
/* handle the 005 numeric */
void
inbound_005 (server * serv, char *word[], const message_tags_data *tags_data)
{
int w;
char *pre;
char *tokname, *tokvalue;
gboolean tokadding;
w = 4; /* start at the 4th word */
while (w < PDIWORDS && *word[w])
{
if (word[w][0] == ':')
break; // :are supported by this server
parse_005_token(word[w], &tokname, &tokvalue, &tokadding);
if (g_strcmp0 (tokname, "MODES") == 0)
{
serv->modes_per_line = atoi (tokvalue);
} else if (g_strcmp0 (tokname, "CHANTYPES") == 0)
{
g_free (serv->chantypes);
serv->chantypes = g_strdup (tokvalue);
} else if (g_strcmp0 (tokname, "CHANMODES") == 0)
{
g_free (serv->chanmodes);
serv->chanmodes = g_strdup (tokvalue);
} else if (g_strcmp0 (tokname, "PREFIX") == 0)
{
pre = strchr (tokvalue, ')');
if (pre)
{
pre[0] = 0; /* NULL out the ')' */
g_free (serv->nick_prefixes);
g_free (serv->nick_modes);
serv->nick_prefixes = g_strdup (pre + 1);
serv->nick_modes = g_strdup (tokvalue);
} else
{
/* bad! some ircds don't give us the modes. */
/* in this case, we use it only to strip /NAMES */
serv->bad_prefix = TRUE;
g_free (serv->bad_nick_prefixes);
serv->bad_nick_prefixes = g_strdup (tokvalue);
}
} else if (g_strcmp0 (tokname, "WATCH") == 0)
{
serv->supports_watch = tokadding;
} else if (g_strcmp0 (tokname, "MONITOR") == 0)
{
serv->supports_monitor = tokadding;
} else if (g_strcmp0 (tokname, "NETWORK") == 0)
{
if (serv->server_session->type == SESS_SERVER && strlen (tokvalue))
{
safe_strcpy (serv->server_session->channel, tokvalue, CHANLEN);
fe_set_channel (serv->server_session);
}
} else if (g_strcmp0 (tokname, "CASEMAPPING") == 0)
{
if (g_strcmp0 (tokvalue, "ascii") == 0)
serv->p_cmp = (void *)g_ascii_strcasecmp;
} else if (g_strcmp0 (tokname, "CHARSET") == 0)
{
if (g_ascii_strcasecmp (tokvalue, "UTF-8") == 0)
{
server_set_encoding (serv, "UTF-8");
}
} else if (g_strcmp0 (tokname, "NAMESX") == 0)
{
/* 12345678901234567 */
tcp_send_len (serv, "PROTOCTL NAMESX\r\n", 17);
} else if (g_strcmp0 (tokname, "WHOX") == 0)
{
serv->have_whox = tokadding;
} else if (g_strcmp0 (tokname, "EXCEPTS") == 0)
{
serv->have_except = tokadding;
} else if (g_strcmp0 (tokname, "INVEX") == 0)
{
/* supports mode letter +I, default channel invite */
serv->have_invite = tokadding;
} else if (g_strcmp0 (tokname, "ELIST") == 0)
{
/* supports LIST >< min/max user counts? */
if (strchr (tokvalue, 'U') || strchr (tokvalue, 'u'))
serv->use_listargs = TRUE;
}
g_free (tokname);
g_free (tokvalue);
w++;
}
}