mirror of
https://github.com/irssi/irssi.git
synced 2024-11-03 04:27:19 -05:00
Ignoring updates. Added support for nickmatch cache. One ignore can't
have both except and normal levels. Nick ignoring checks now with both old and new nicks. git-svn-id: http://svn.irssi.org/repos/irssi/trunk@1155 dbcabf3a-b0e7-0310-adc4-f8d773084564
This commit is contained in:
parent
6c32ffdc4f
commit
56abbcd2e3
@ -58,6 +58,7 @@ void core_init(void)
|
|||||||
signals_init();
|
signals_init();
|
||||||
settings_init();
|
settings_init();
|
||||||
commands_init();
|
commands_init();
|
||||||
|
nickmatch_cache_init();
|
||||||
|
|
||||||
chat_protocols_init();
|
chat_protocols_init();
|
||||||
chatnets_init();
|
chatnets_init();
|
||||||
@ -70,7 +71,6 @@ void core_init(void)
|
|||||||
channels_init();
|
channels_init();
|
||||||
queries_init();
|
queries_init();
|
||||||
nicklist_init();
|
nicklist_init();
|
||||||
nickmatch_cache_init();
|
|
||||||
|
|
||||||
chat_commands_init();
|
chat_commands_init();
|
||||||
settings_check();
|
settings_check();
|
||||||
@ -80,7 +80,6 @@ void core_deinit(void)
|
|||||||
{
|
{
|
||||||
chat_commands_deinit();
|
chat_commands_deinit();
|
||||||
|
|
||||||
nickmatch_cache_deinit();
|
|
||||||
nicklist_deinit();
|
nicklist_deinit();
|
||||||
queries_deinit();
|
queries_deinit();
|
||||||
channels_deinit();
|
channels_deinit();
|
||||||
@ -93,6 +92,7 @@ void core_deinit(void)
|
|||||||
chatnets_deinit();
|
chatnets_deinit();
|
||||||
chat_protocols_deinit();
|
chat_protocols_deinit();
|
||||||
|
|
||||||
|
nickmatch_cache_deinit();
|
||||||
commands_deinit();
|
commands_deinit();
|
||||||
settings_deinit();
|
settings_deinit();
|
||||||
signals_deinit();
|
signals_deinit();
|
||||||
|
@ -29,14 +29,17 @@
|
|||||||
#include "servers.h"
|
#include "servers.h"
|
||||||
#include "channels.h"
|
#include "channels.h"
|
||||||
#include "nicklist.h"
|
#include "nicklist.h"
|
||||||
|
#include "nickmatch-cache.h"
|
||||||
|
|
||||||
#include "ignore.h"
|
#include "ignore.h"
|
||||||
|
|
||||||
GSList *ignores;
|
GSList *ignores;
|
||||||
|
|
||||||
|
static NICKMATCH_REC *nickmatch;
|
||||||
|
|
||||||
/* check if `text' contains ignored nick at the start of the line. */
|
/* check if `text' contains ignored nick at the start of the line. */
|
||||||
static int ignore_check_replies(IGNORE_REC *rec, CHANNEL_REC *channel,
|
static int ignore_check_replies_rec(IGNORE_REC *rec, CHANNEL_REC *channel,
|
||||||
const char *text)
|
const char *text)
|
||||||
{
|
{
|
||||||
GSList *nicks, *tmp;
|
GSList *nicks, *tmp;
|
||||||
|
|
||||||
@ -54,101 +57,159 @@ static int ignore_check_replies(IGNORE_REC *rec, CHANNEL_REC *channel,
|
|||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int ignore_check_replies(CHANNEL_REC *chanrec, const char *text)
|
||||||
|
{
|
||||||
|
GSList *tmp;
|
||||||
|
|
||||||
|
if (text == NULL || chanrec == NULL)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
/* check reply ignores */
|
||||||
|
for (tmp = ignores; tmp != NULL; tmp = tmp->next) {
|
||||||
|
IGNORE_REC *rec = tmp->data;
|
||||||
|
|
||||||
|
if (rec->mask != NULL && rec->replies &&
|
||||||
|
ignore_check_replies_rec(rec, chanrec, text))
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ignore_match_pattern(IGNORE_REC *rec, const char *text)
|
||||||
|
{
|
||||||
|
if (rec->pattern == NULL)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
if (text == NULL)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (rec->regexp) {
|
||||||
|
#ifdef HAVE_REGEX_H
|
||||||
|
return rec->regexp_compiled &&
|
||||||
|
regexec(&rec->preg, text, 0, NULL, 0) == 0;
|
||||||
|
#else
|
||||||
|
return FALSE;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
return rec->fullword ?
|
||||||
|
stristr_full(text, rec->pattern) != NULL :
|
||||||
|
stristr(text, rec->pattern) != NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define ignore_match_level(rec, level) \
|
||||||
|
((level & (rec)->level) != 0)
|
||||||
|
|
||||||
|
#define ignore_match_nickmask(rec, nickmask) \
|
||||||
|
((rec)->mask == NULL || match_wildcards((rec)->mask, nickmask))
|
||||||
|
|
||||||
|
#define ignore_match_server(rec, server) \
|
||||||
|
((rec)->servertag == NULL || \
|
||||||
|
g_strcasecmp((server)->tag, (rec)->servertag) == 0)
|
||||||
|
|
||||||
|
#define ignore_match_channel(rec, channel) \
|
||||||
|
((rec)->channels == NULL || ((channel) != NULL && \
|
||||||
|
strarray_find((rec)->channels, (channel)) != -1))
|
||||||
|
|
||||||
|
static int ignore_check_without_mask(GSList *list, CHANNEL_REC *channel,
|
||||||
|
int level, const char *text)
|
||||||
|
{
|
||||||
|
GSList *tmp;
|
||||||
|
int len, best_mask, best_match, best_patt;
|
||||||
|
|
||||||
|
best_mask = best_patt = 0; best_match = FALSE;
|
||||||
|
for (tmp = list; tmp != NULL; tmp = tmp->next) {
|
||||||
|
IGNORE_REC *rec = tmp->data;
|
||||||
|
|
||||||
|
if (ignore_match_level(rec, level) &&
|
||||||
|
ignore_match_pattern(rec, text)) {
|
||||||
|
len = rec->mask == NULL ? 0 : strlen(rec->mask);
|
||||||
|
if (len > best_mask) {
|
||||||
|
best_mask = len;
|
||||||
|
best_match = !rec->exception;
|
||||||
|
} else if (len == best_mask && rec->pattern != NULL) {
|
||||||
|
len = strlen(rec->pattern);
|
||||||
|
if (len > best_patt) {
|
||||||
|
best_patt = len;
|
||||||
|
best_match = !rec->exception;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (best_match || (level & MSGLEVEL_PUBLIC) == 0)
|
||||||
|
return best_match;
|
||||||
|
|
||||||
|
return ignore_check_replies(channel, text);
|
||||||
|
}
|
||||||
|
|
||||||
int ignore_check(SERVER_REC *server, const char *nick, const char *host,
|
int ignore_check(SERVER_REC *server, const char *nick, const char *host,
|
||||||
const char *channel, const char *text, int level)
|
const char *channel, const char *text, int level)
|
||||||
{
|
{
|
||||||
CHANNEL_REC *chanrec;
|
CHANNEL_REC *chanrec;
|
||||||
GSList *tmp;
|
NICK_REC *nickrec;
|
||||||
int ok, mask_len, patt_len;
|
IGNORE_REC *rec;
|
||||||
int best_mask, best_patt, best_ignore;
|
GSList *tmp, *list;
|
||||||
|
char *nickmask;
|
||||||
|
int len, best_mask, best_match, best_patt;
|
||||||
|
|
||||||
g_return_val_if_fail(server != NULL, 0);
|
g_return_val_if_fail(server != NULL, 0);
|
||||||
|
|
||||||
chanrec = (channel != NULL && server != NULL &&
|
chanrec = (channel != NULL && server != NULL &&
|
||||||
server->ischannel(channel)) ?
|
server->ischannel(channel)) ?
|
||||||
channel_find(server, channel) : NULL;
|
channel_find(server, channel) : NULL;
|
||||||
|
if (chanrec != NULL && nick != NULL &&
|
||||||
|
(nickrec = nicklist_find(chanrec, nick)) != NULL) {
|
||||||
|
/* nick found - check only ignores in nickmatch cache */
|
||||||
|
if (nickrec->host == NULL)
|
||||||
|
nicklist_set_host(chanrec, nickrec, host);
|
||||||
|
|
||||||
best_mask = 0; best_patt = 0; best_ignore = FALSE;
|
list = nickmatch_find(nickmatch, nickrec);
|
||||||
for (tmp = ignores; tmp != NULL; tmp = tmp->next) {
|
return ignore_check_without_mask(list, chanrec, level, text);
|
||||||
IGNORE_REC *rec = tmp->data;
|
|
||||||
|
|
||||||
if ((level & (rec->level|rec->except_level)) == 0)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
/* server */
|
|
||||||
if (rec->servertag != NULL && g_strcasecmp(server->tag, rec->servertag) != 0)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
/* channel list */
|
|
||||||
if (rec->channels != NULL) {
|
|
||||||
if (chanrec == NULL ||
|
|
||||||
strarray_find(rec->channels, channel) == -1)
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* nick mask */
|
|
||||||
mask_len = 0;
|
|
||||||
if (rec->mask != NULL) {
|
|
||||||
if (nick == NULL)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
mask_len = strlen(rec->mask);
|
|
||||||
if (mask_len <= best_mask) continue;
|
|
||||||
|
|
||||||
ok = ((host == NULL || *host == '\0')) ?
|
|
||||||
match_wildcards(rec->mask, nick) :
|
|
||||||
mask_match_address(server, rec->mask, nick, host);
|
|
||||||
if (!ok) {
|
|
||||||
/* nick didn't match, but maybe this is a reply to nick? */
|
|
||||||
if (!rec->replies || chanrec == NULL || text == NULL ||
|
|
||||||
!ignore_check_replies(rec, chanrec, text))
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* pattern */
|
|
||||||
patt_len = 0;
|
|
||||||
if (rec->pattern != NULL) {
|
|
||||||
if (text == NULL)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (!mask_len && !best_mask) {
|
|
||||||
patt_len = strlen(rec->pattern);
|
|
||||||
if (patt_len <= best_patt) continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef HAVE_REGEX_H
|
|
||||||
if (rec->regexp) {
|
|
||||||
ok = !rec->regexp_compiled ? FALSE :
|
|
||||||
regexec(&rec->preg, text, 0, NULL, 0) == 0;
|
|
||||||
} else
|
|
||||||
#endif
|
|
||||||
{
|
|
||||||
ok = rec->fullword ?
|
|
||||||
stristr_full(text, rec->pattern) != NULL :
|
|
||||||
stristr(text, rec->pattern) != NULL;
|
|
||||||
}
|
|
||||||
if (!ok) continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mask_len || best_mask)
|
|
||||||
best_mask = mask_len;
|
|
||||||
else if (patt_len)
|
|
||||||
best_patt = patt_len;
|
|
||||||
|
|
||||||
best_ignore = (rec->level & level) != 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return best_ignore;
|
nickmask = g_strconcat(nick, "!", host, NULL);
|
||||||
|
|
||||||
|
best_mask = best_patt = 0; best_match = FALSE;
|
||||||
|
for (tmp = ignores; tmp != NULL; tmp = tmp->next) {
|
||||||
|
rec = tmp->data;
|
||||||
|
|
||||||
|
if (ignore_match_level(rec, level) &&
|
||||||
|
ignore_match_server(rec, server) &&
|
||||||
|
ignore_match_channel(rec, channel) &&
|
||||||
|
ignore_match_nickmask(rec, nickmask) &&
|
||||||
|
ignore_match_pattern(rec, text)) {
|
||||||
|
len = rec->mask == NULL ? 0 : strlen(rec->mask);
|
||||||
|
if (len > best_mask) {
|
||||||
|
best_mask = len;
|
||||||
|
best_match = !rec->exception;
|
||||||
|
} else if (len == best_mask && rec->pattern != NULL) {
|
||||||
|
len = strlen(rec->pattern);
|
||||||
|
if (len > best_patt) {
|
||||||
|
best_patt = len;
|
||||||
|
best_match = !rec->exception;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
g_free(nickmask);
|
||||||
|
|
||||||
|
if (best_match || (level & MSGLEVEL_PUBLIC) == 0)
|
||||||
|
return best_match;
|
||||||
|
|
||||||
|
return ignore_check_replies(chanrec, text);
|
||||||
}
|
}
|
||||||
|
|
||||||
IGNORE_REC *ignore_find(const char *servertag, const char *mask, char **channels)
|
IGNORE_REC *ignore_find(const char *servertag, const char *mask,
|
||||||
|
char **channels)
|
||||||
{
|
{
|
||||||
GSList *tmp;
|
GSList *tmp;
|
||||||
char **chan;
|
char **chan;
|
||||||
int ignore_servertag;
|
int ignore_servertag;
|
||||||
|
|
||||||
if (mask != NULL && *mask == '\0') mask = NULL;
|
if (mask != NULL && (*mask == '\0' || strcmp(mask, "*") == 0))
|
||||||
|
mask = NULL;
|
||||||
|
|
||||||
ignore_servertag = servertag != NULL && strcmp(servertag, "*") == 0;
|
ignore_servertag = servertag != NULL && strcmp(servertag, "*") == 0;
|
||||||
for (tmp = ignores; tmp != NULL; tmp = tmp->next) {
|
for (tmp = ignores; tmp != NULL; tmp = tmp->next) {
|
||||||
@ -199,10 +260,7 @@ static void ignore_set_config(IGNORE_REC *rec)
|
|||||||
CONFIG_NODE *node;
|
CONFIG_NODE *node;
|
||||||
char *levelstr;
|
char *levelstr;
|
||||||
|
|
||||||
if (rec->level == 0 && rec->except_level == 0)
|
if (rec->level == 0 || rec->time > 0)
|
||||||
return;
|
|
||||||
|
|
||||||
if (rec->time > 0)
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
node = iconfig_node_traverse("(ignores", TRUE);
|
node = iconfig_node_traverse("(ignores", TRUE);
|
||||||
@ -214,12 +272,8 @@ static void ignore_set_config(IGNORE_REC *rec)
|
|||||||
iconfig_node_set_str(node, "level", levelstr);
|
iconfig_node_set_str(node, "level", levelstr);
|
||||||
g_free(levelstr);
|
g_free(levelstr);
|
||||||
}
|
}
|
||||||
if (rec->except_level) {
|
|
||||||
levelstr = bits2level(rec->except_level);
|
|
||||||
iconfig_node_set_str(node, "except_level", levelstr);
|
|
||||||
g_free(levelstr);
|
|
||||||
}
|
|
||||||
iconfig_node_set_str(node, "pattern", rec->pattern);
|
iconfig_node_set_str(node, "pattern", rec->pattern);
|
||||||
|
if (rec->exception) iconfig_node_set_bool(node, "exception", TRUE);
|
||||||
if (rec->regexp) iconfig_node_set_bool(node, "regexp", TRUE);
|
if (rec->regexp) iconfig_node_set_bool(node, "regexp", TRUE);
|
||||||
if (rec->fullword) iconfig_node_set_bool(node, "fullword", TRUE);
|
if (rec->fullword) iconfig_node_set_bool(node, "fullword", TRUE);
|
||||||
if (rec->replies) iconfig_node_set_bool(node, "replies", TRUE);
|
if (rec->replies) iconfig_node_set_bool(node, "replies", TRUE);
|
||||||
@ -285,11 +339,13 @@ static void ignore_destroy(IGNORE_REC *rec)
|
|||||||
g_free_not_null(rec->servertag);
|
g_free_not_null(rec->servertag);
|
||||||
g_free_not_null(rec->pattern);
|
g_free_not_null(rec->pattern);
|
||||||
g_free(rec);
|
g_free(rec);
|
||||||
|
|
||||||
|
nickmatch_rebuild(nickmatch);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ignore_update_rec(IGNORE_REC *rec)
|
void ignore_update_rec(IGNORE_REC *rec)
|
||||||
{
|
{
|
||||||
if (rec->level == 0 && rec->except_level == 0) {
|
if (rec->level == 0) {
|
||||||
/* unignored everything */
|
/* unignored everything */
|
||||||
ignore_remove_config(rec);
|
ignore_remove_config(rec);
|
||||||
ignore_destroy(rec);
|
ignore_destroy(rec);
|
||||||
@ -302,6 +358,7 @@ void ignore_update_rec(IGNORE_REC *rec)
|
|||||||
ignore_set_config(rec);
|
ignore_set_config(rec);
|
||||||
|
|
||||||
signal_emit("ignore changed", 1, rec);
|
signal_emit("ignore changed", 1, rec);
|
||||||
|
nickmatch_rebuild(nickmatch);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -315,7 +372,10 @@ static void read_ignores(void)
|
|||||||
ignore_destroy(ignores->data);
|
ignore_destroy(ignores->data);
|
||||||
|
|
||||||
node = iconfig_node_traverse("ignores", FALSE);
|
node = iconfig_node_traverse("ignores", FALSE);
|
||||||
if (node == NULL) return;
|
if (node == NULL) {
|
||||||
|
nickmatch_rebuild(nickmatch);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
for (tmp = node->value; tmp != NULL; tmp = tmp->next) {
|
for (tmp = node->value; tmp != NULL; tmp = tmp->next) {
|
||||||
node = tmp->data;
|
node = tmp->data;
|
||||||
@ -327,9 +387,19 @@ static void read_ignores(void)
|
|||||||
ignores = g_slist_append(ignores, rec);
|
ignores = g_slist_append(ignores, rec);
|
||||||
|
|
||||||
rec->mask = g_strdup(config_node_get_str(node, "mask", NULL));
|
rec->mask = g_strdup(config_node_get_str(node, "mask", NULL));
|
||||||
|
if (strcmp(rec->mask, "*") == 0) {
|
||||||
|
/* FIXME: remove after .98 */
|
||||||
|
g_free(rec->mask);
|
||||||
|
rec->mask = NULL;
|
||||||
|
}
|
||||||
rec->pattern = g_strdup(config_node_get_str(node, "pattern", NULL));
|
rec->pattern = g_strdup(config_node_get_str(node, "pattern", NULL));
|
||||||
rec->level = level2bits(config_node_get_str(node, "level", ""));
|
rec->level = level2bits(config_node_get_str(node, "level", ""));
|
||||||
rec->except_level = level2bits(config_node_get_str(node, "except_level", ""));
|
rec->exception = config_node_get_bool(node, "exception", FALSE);
|
||||||
|
if (*config_node_get_str(node, "except_level", "") != '\0') {
|
||||||
|
/* FIXME: remove after .98 */
|
||||||
|
rec->level = level2bits(config_node_get_str(node, "except_level", ""));
|
||||||
|
rec->exception = TRUE;
|
||||||
|
}
|
||||||
rec->regexp = config_node_get_bool(node, "regexp", FALSE);
|
rec->regexp = config_node_get_bool(node, "regexp", FALSE);
|
||||||
rec->fullword = config_node_get_bool(node, "fullword", FALSE);
|
rec->fullword = config_node_get_bool(node, "fullword", FALSE);
|
||||||
rec->replies = config_node_get_bool(node, "replies", FALSE);
|
rec->replies = config_node_get_bool(node, "replies", FALSE);
|
||||||
@ -337,11 +407,41 @@ static void read_ignores(void)
|
|||||||
node = config_node_section(node, "channels", -1);
|
node = config_node_section(node, "channels", -1);
|
||||||
if (node != NULL) rec->channels = config_node_get_list(node);
|
if (node != NULL) rec->channels = config_node_get_list(node);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nickmatch_rebuild(nickmatch);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ignore_nick_cache(GHashTable *list, CHANNEL_REC *channel,
|
||||||
|
NICK_REC *nick)
|
||||||
|
{
|
||||||
|
GSList *tmp, *matches;
|
||||||
|
char *nickmask;
|
||||||
|
|
||||||
|
if (nick->host == NULL)
|
||||||
|
return; /* don't check until host is known */
|
||||||
|
|
||||||
|
matches = NULL;
|
||||||
|
nickmask = g_strconcat(nick->nick, "!", nick->host, NULL);
|
||||||
|
for (tmp = ignores; tmp != NULL; tmp = tmp->next) {
|
||||||
|
IGNORE_REC *rec = tmp->data;
|
||||||
|
|
||||||
|
if (ignore_match_nickmask(rec, nickmask) &&
|
||||||
|
ignore_match_server(rec, channel->server) &&
|
||||||
|
ignore_match_channel(rec, channel->name))
|
||||||
|
matches = g_slist_append(matches, rec);
|
||||||
|
}
|
||||||
|
g_free_not_null(nickmask);
|
||||||
|
|
||||||
|
if (matches == NULL)
|
||||||
|
g_hash_table_remove(list, nick);
|
||||||
|
else
|
||||||
|
g_hash_table_insert(list, nick, matches);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ignore_init(void)
|
void ignore_init(void)
|
||||||
{
|
{
|
||||||
ignores = NULL;
|
ignores = NULL;
|
||||||
|
nickmatch = nickmatch_init(ignore_nick_cache);
|
||||||
|
|
||||||
read_ignores();
|
read_ignores();
|
||||||
signal_add("setup reread", (SIGNAL_FUNC) read_ignores);
|
signal_add("setup reread", (SIGNAL_FUNC) read_ignores);
|
||||||
@ -351,6 +451,7 @@ void ignore_deinit(void)
|
|||||||
{
|
{
|
||||||
while (ignores != NULL)
|
while (ignores != NULL)
|
||||||
ignore_destroy(ignores->data);
|
ignore_destroy(ignores->data);
|
||||||
|
nickmatch_deinit(nickmatch);
|
||||||
|
|
||||||
signal_remove("setup reread", (SIGNAL_FUNC) read_ignores);
|
signal_remove("setup reread", (SIGNAL_FUNC) read_ignores);
|
||||||
}
|
}
|
||||||
|
@ -6,17 +6,16 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
|
int level; /* ignore these levels */
|
||||||
char *mask; /* nick mask */
|
char *mask; /* nick mask */
|
||||||
char *servertag; /* this is for autoignoring */
|
char *servertag; /* this is for autoignoring */
|
||||||
char **channels; /* ignore only in these channels */
|
char **channels; /* ignore only in these channels */
|
||||||
char *pattern; /* text body must match this pattern */
|
char *pattern; /* text body must match this pattern */
|
||||||
|
|
||||||
int level; /* ignore these levels */
|
|
||||||
int except_level; /* don't ignore these levels */
|
|
||||||
|
|
||||||
int time; /* time in sec for temp ignores */
|
int time; /* time in sec for temp ignores */
|
||||||
int time_tag;
|
int time_tag;
|
||||||
|
|
||||||
|
unsigned int exception:1; /* *don't* ignore */
|
||||||
unsigned int regexp:1;
|
unsigned int regexp:1;
|
||||||
unsigned int fullword:1;
|
unsigned int fullword:1;
|
||||||
unsigned int replies:1; /* ignore replies to nick in channel */
|
unsigned int replies:1; /* ignore replies to nick in channel */
|
||||||
|
@ -72,7 +72,10 @@ static void sig_message_kick(SERVER_REC *server, const char *channel,
|
|||||||
static void sig_message_nick(SERVER_REC *server, const char *newnick,
|
static void sig_message_nick(SERVER_REC *server, const char *newnick,
|
||||||
const char *oldnick, const char *address)
|
const char *oldnick, const char *address)
|
||||||
{
|
{
|
||||||
if (ignore_check(server, oldnick, address, NULL, NULL, MSGLEVEL_NICKS))
|
if (ignore_check(server, oldnick, address,
|
||||||
|
NULL, NULL, MSGLEVEL_NICKS) ||
|
||||||
|
ignore_check(server, newnick, address,
|
||||||
|
NULL, NULL, MSGLEVEL_NICKS))
|
||||||
signal_stop();
|
signal_stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,83 +36,27 @@ static char *ignore_get_key(IGNORE_REC *rec)
|
|||||||
char *chans, *ret;
|
char *chans, *ret;
|
||||||
|
|
||||||
if (rec->channels == NULL)
|
if (rec->channels == NULL)
|
||||||
return rec->mask != NULL ? g_strdup(rec->mask) : NULL;
|
return g_strdup(rec->mask == NULL ? "*" : rec->mask);
|
||||||
|
|
||||||
chans = g_strjoinv(",", rec->channels);
|
chans = g_strjoinv(",", rec->channels);
|
||||||
if (rec->mask == NULL) return chans;
|
if (rec->mask == NULL) return chans;
|
||||||
|
|
||||||
ret = g_strdup_printf("%s %s", rec->mask, chans);
|
ret = g_strdup_printf("%s %s", rec->mask == NULL ?
|
||||||
|
"*" : rec->mask, chans);
|
||||||
g_free(chans);
|
g_free(chans);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static char *ignore_get_levels(int level, int xlevel)
|
|
||||||
{
|
|
||||||
GString *str;
|
|
||||||
char *levelstr, *p, *ret;
|
|
||||||
|
|
||||||
str = g_string_new(NULL);
|
|
||||||
if (level != 0) {
|
|
||||||
levelstr = bits2level(level);
|
|
||||||
g_string_append(str, levelstr);
|
|
||||||
g_free(levelstr);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (xlevel != 0) {
|
|
||||||
if (str->len > 0) g_string_append_c(str, ' ');
|
|
||||||
|
|
||||||
levelstr = bits2level(xlevel);
|
|
||||||
for (p = levelstr; *p != '\0'; p++) {
|
|
||||||
if (!isspace(*p) && (p == levelstr || isspace(p[-1])))
|
|
||||||
g_string_append_c(str, '^');
|
|
||||||
g_string_append_c(str, *p);
|
|
||||||
}
|
|
||||||
g_free(levelstr);
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = str->str;
|
|
||||||
g_string_free(str, FALSE);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* msgs ^notices : level=msgs, xlevel=notices */
|
|
||||||
static void ignore_split_levels(const char *levels, int *level, int *xlevel)
|
|
||||||
{
|
|
||||||
GString *slevel, *sxlevel;
|
|
||||||
char **levellist, **tmp;
|
|
||||||
|
|
||||||
if (*levels == '\0') return;
|
|
||||||
|
|
||||||
slevel = g_string_new(NULL);
|
|
||||||
sxlevel = g_string_new(NULL);
|
|
||||||
|
|
||||||
levellist = g_strsplit(levels, " ", -1);
|
|
||||||
for (tmp = levellist; *tmp != NULL; tmp++) {
|
|
||||||
if (**tmp == '^')
|
|
||||||
g_string_sprintfa(sxlevel, "%s ", (*tmp)+1);
|
|
||||||
else if (**tmp == '-' && (*tmp)[1] == '^')
|
|
||||||
g_string_sprintfa(sxlevel, "-%s ", (*tmp)+2);
|
|
||||||
else
|
|
||||||
g_string_sprintfa(slevel, "%s ", *tmp);
|
|
||||||
}
|
|
||||||
g_strfreev(levellist);
|
|
||||||
|
|
||||||
*level = combine_level(*level, slevel->str);
|
|
||||||
*xlevel = combine_level(*xlevel, sxlevel->str);
|
|
||||||
|
|
||||||
g_string_free(slevel, TRUE);
|
|
||||||
g_string_free(sxlevel, TRUE);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void ignore_print(int index, IGNORE_REC *rec)
|
static void ignore_print(int index, IGNORE_REC *rec)
|
||||||
{
|
{
|
||||||
GString *options;
|
GString *options;
|
||||||
char *key, *levels;
|
char *key, *levels;
|
||||||
|
|
||||||
key = ignore_get_key(rec);
|
key = ignore_get_key(rec);
|
||||||
levels = ignore_get_levels(rec->level, rec->except_level);
|
levels = bits2level(rec->level);
|
||||||
|
|
||||||
options = g_string_new(NULL);
|
options = g_string_new(NULL);
|
||||||
|
if (rec->exception) g_string_sprintfa(options, "-except ");
|
||||||
if (rec->regexp) g_string_sprintfa(options, "-regexp ");
|
if (rec->regexp) g_string_sprintfa(options, "-regexp ");
|
||||||
if (rec->fullword) g_string_sprintfa(options, "-word ");
|
if (rec->fullword) g_string_sprintfa(options, "-word ");
|
||||||
if (rec->replies) g_string_sprintfa(options, "-replies ");
|
if (rec->replies) g_string_sprintfa(options, "-replies ");
|
||||||
@ -189,34 +133,30 @@ static void cmd_ignore(const char *data)
|
|||||||
if (rec == NULL) {
|
if (rec == NULL) {
|
||||||
rec = g_new0(IGNORE_REC, 1);
|
rec = g_new0(IGNORE_REC, 1);
|
||||||
|
|
||||||
rec->mask = (mask != NULL && *mask != '\0') ?
|
rec->mask = mask == NULL || *mask == '\0' ||
|
||||||
g_strdup(mask) : NULL;
|
strcmp(mask, "*") == 0 ? NULL : g_strdup(mask);
|
||||||
rec->channels = channels;
|
rec->channels = channels;
|
||||||
} else {
|
} else {
|
||||||
g_free_and_null(rec->pattern);
|
g_free_and_null(rec->pattern);
|
||||||
g_strfreev(channels);
|
g_strfreev(channels);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (g_hash_table_lookup(optlist, "except") != NULL) {
|
rec->level = combine_level(rec->level, levels);
|
||||||
rec->except_level = combine_level(rec->except_level, levels);
|
|
||||||
} else {
|
|
||||||
ignore_split_levels(levels, &rec->level, &rec->except_level);
|
|
||||||
}
|
|
||||||
|
|
||||||
rec->pattern = (patternarg == NULL || *patternarg == '\0') ?
|
rec->pattern = (patternarg == NULL || *patternarg == '\0') ?
|
||||||
NULL : g_strdup(patternarg);
|
NULL : g_strdup(patternarg);
|
||||||
|
rec->exception = g_hash_table_lookup(optlist, "except") != NULL;
|
||||||
rec->regexp = g_hash_table_lookup(optlist, "regexp") != NULL;
|
rec->regexp = g_hash_table_lookup(optlist, "regexp") != NULL;
|
||||||
rec->fullword = g_hash_table_lookup(optlist, "word") != NULL;
|
rec->fullword = g_hash_table_lookup(optlist, "word") != NULL;
|
||||||
rec->replies = g_hash_table_lookup(optlist, "replies") != NULL;
|
rec->replies = g_hash_table_lookup(optlist, "replies") != NULL;
|
||||||
timestr = g_hash_table_lookup(optlist, "time");
|
timestr = g_hash_table_lookup(optlist, "time");
|
||||||
rec->time = timestr == NULL ? 0 : atoi(timestr);
|
rec->time = timestr == NULL ? 0 : atoi(timestr);
|
||||||
|
|
||||||
if (rec->level == 0 && rec->except_level == 0) {
|
if (rec->level == 0) {
|
||||||
printformat(NULL, NULL, MSGLEVEL_CLIENTNOTICE, TXT_UNIGNORED,
|
printformat(NULL, NULL, MSGLEVEL_CLIENTNOTICE, TXT_UNIGNORED,
|
||||||
rec->mask == NULL ? "" : rec->mask);
|
rec->mask == NULL ? "*" : rec->mask);
|
||||||
} else {
|
} else {
|
||||||
key = ignore_get_key(rec);
|
key = ignore_get_key(rec);
|
||||||
levels = ignore_get_levels(rec->level, rec->except_level);
|
levels = bits2level(rec->level);
|
||||||
printformat(NULL, NULL, MSGLEVEL_CLIENTNOTICE, TXT_IGNORED, key, levels);
|
printformat(NULL, NULL, MSGLEVEL_CLIENTNOTICE, TXT_IGNORED, key, levels);
|
||||||
g_free(key);
|
g_free(key);
|
||||||
g_free(levels);
|
g_free(levels);
|
||||||
@ -242,7 +182,6 @@ static void fe_unignore(IGNORE_REC *rec)
|
|||||||
g_free(key);
|
g_free(key);
|
||||||
|
|
||||||
rec->level = 0;
|
rec->level = 0;
|
||||||
rec->except_level = 0;
|
|
||||||
ignore_update_rec(rec);
|
ignore_update_rec(rec);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -401,7 +401,7 @@ static void print_nick_change(SERVER_REC *server, const char *newnick,
|
|||||||
WINDOW_REC *window =
|
WINDOW_REC *window =
|
||||||
window_item_window((WI_ITEM_REC *) channel);
|
window_item_window((WI_ITEM_REC *) channel);
|
||||||
|
|
||||||
if (nicklist_find(channel, oldnick) == NULL ||
|
if (nicklist_find(channel, newnick) == NULL ||
|
||||||
g_slist_find(windows, window) != NULL)
|
g_slist_find(windows, window) != NULL)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
@ -333,7 +333,7 @@ static void sig_connected(IRC_SERVER_REC *server)
|
|||||||
|
|
||||||
void irc_nicklist_init(void)
|
void irc_nicklist_init(void)
|
||||||
{
|
{
|
||||||
signal_add("event nick", (SIGNAL_FUNC) event_nick);
|
signal_add_first("event nick", (SIGNAL_FUNC) event_nick);
|
||||||
signal_add_first("event 352", (SIGNAL_FUNC) event_who);
|
signal_add_first("event 352", (SIGNAL_FUNC) event_who);
|
||||||
signal_add("silent event who", (SIGNAL_FUNC) event_who);
|
signal_add("silent event who", (SIGNAL_FUNC) event_who);
|
||||||
signal_add("silent event whois", (SIGNAL_FUNC) event_whois);
|
signal_add("silent event whois", (SIGNAL_FUNC) event_whois);
|
||||||
|
@ -327,8 +327,8 @@ void perl_ignore_fill_hash(HV *hv, IGNORE_REC *ignore)
|
|||||||
hv_store(hv, "pattern", 7, new_pv(ignore->pattern), 0);
|
hv_store(hv, "pattern", 7, new_pv(ignore->pattern), 0);
|
||||||
|
|
||||||
hv_store(hv, "level", 5, newSViv(ignore->level), 0);
|
hv_store(hv, "level", 5, newSViv(ignore->level), 0);
|
||||||
hv_store(hv, "except_level", 12, newSViv(ignore->except_level), 0);
|
|
||||||
|
|
||||||
|
hv_store(hv, "exception", 6, newSViv(ignore->exception), 0);
|
||||||
hv_store(hv, "regexp", 6, newSViv(ignore->regexp), 0);
|
hv_store(hv, "regexp", 6, newSViv(ignore->regexp), 0);
|
||||||
hv_store(hv, "fullword", 8, newSViv(ignore->fullword), 0);
|
hv_store(hv, "fullword", 8, newSViv(ignore->fullword), 0);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user