mirror of
https://github.com/irssi/irssi.git
synced 2024-12-04 14:46:39 -05:00
6d6b02d10e
of them had ops, so the message always went to status/active window. git-svn-id: http://svn.irssi.org/repos/irssi/trunk@2557 dbcabf3a-b0e7-0310-adc4-f8d773084564
433 lines
11 KiB
C
433 lines
11 KiB
C
/*
|
|
fe-netjoin.c : irssi
|
|
|
|
Copyright (C) 2000 Timo Sirainen
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "module.h"
|
|
#include "module-formats.h"
|
|
#include "signals.h"
|
|
#include "levels.h"
|
|
#include "misc.h"
|
|
#include "settings.h"
|
|
|
|
#include "irc-servers.h"
|
|
#include "modes.h"
|
|
#include "ignore.h"
|
|
#include "netsplit.h"
|
|
|
|
#include "printtext.h"
|
|
|
|
#define NETJOIN_WAIT_TIME 5 /* how many seconds to wait for the netsplitted JOIN messages to stop */
|
|
#define NETJOIN_MAX_WAIT 30 /* how many seconds to wait for nick to join to the rest of the channels she was before the netsplit */
|
|
|
|
typedef struct {
|
|
char *nick;
|
|
GSList *old_channels;
|
|
GSList *now_channels;
|
|
} NETJOIN_REC;
|
|
|
|
typedef struct {
|
|
IRC_SERVER_REC *server;
|
|
time_t last_netjoin;
|
|
|
|
GSList *netjoins;
|
|
} NETJOIN_SERVER_REC;
|
|
|
|
typedef struct {
|
|
int count;
|
|
GString *nicks;
|
|
} TEMP_PRINT_REC;
|
|
|
|
static int join_tag;
|
|
static int netjoin_max_nicks, hide_netsplit_quits;
|
|
static int printing_joins;
|
|
static GSList *joinservers;
|
|
|
|
static NETJOIN_SERVER_REC *netjoin_find_server(IRC_SERVER_REC *server)
|
|
{
|
|
GSList *tmp;
|
|
|
|
g_return_val_if_fail(server != NULL, NULL);
|
|
|
|
for (tmp = joinservers; tmp != NULL; tmp = tmp->next) {
|
|
NETJOIN_SERVER_REC *rec = tmp->data;
|
|
|
|
if (rec->server == server)
|
|
return rec;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static NETJOIN_REC *netjoin_add(IRC_SERVER_REC *server, const char *nick,
|
|
GSList *channels)
|
|
{
|
|
NETJOIN_REC *rec;
|
|
NETJOIN_SERVER_REC *srec;
|
|
|
|
g_return_val_if_fail(server != NULL, NULL);
|
|
g_return_val_if_fail(nick != NULL, NULL);
|
|
|
|
rec = g_new0(NETJOIN_REC, 1);
|
|
rec->nick = g_strdup(nick);
|
|
while (channels != NULL) {
|
|
NETSPLIT_CHAN_REC *channel = channels->data;
|
|
|
|
rec->old_channels = g_slist_append(rec->old_channels,
|
|
g_strdup(channel->name));
|
|
channels = channels->next;
|
|
}
|
|
|
|
srec = netjoin_find_server(server);
|
|
if (srec == NULL) {
|
|
srec = g_new0(NETJOIN_SERVER_REC, 1);
|
|
srec->server = server;
|
|
joinservers = g_slist_append(joinservers, srec);
|
|
}
|
|
|
|
srec->last_netjoin = time(NULL);
|
|
srec->netjoins = g_slist_append(srec->netjoins, rec);
|
|
return rec;
|
|
}
|
|
|
|
static NETJOIN_REC *netjoin_find(IRC_SERVER_REC *server, const char *nick)
|
|
{
|
|
NETJOIN_SERVER_REC *srec;
|
|
GSList *tmp;
|
|
|
|
g_return_val_if_fail(server != NULL, NULL);
|
|
g_return_val_if_fail(nick != NULL, NULL);
|
|
|
|
srec = netjoin_find_server(server);
|
|
if (srec == NULL) return NULL;
|
|
|
|
for (tmp = srec->netjoins; tmp != NULL; tmp = tmp->next) {
|
|
NETJOIN_REC *rec = tmp->data;
|
|
|
|
if (g_strcasecmp(rec->nick, nick) == 0)
|
|
return rec;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void netjoin_remove(NETJOIN_SERVER_REC *server, NETJOIN_REC *rec)
|
|
{
|
|
server->netjoins = g_slist_remove(server->netjoins, rec);
|
|
|
|
g_slist_foreach(rec->old_channels, (GFunc) g_free, NULL);
|
|
g_slist_foreach(rec->now_channels, (GFunc) g_free, NULL);
|
|
g_slist_free(rec->old_channels);
|
|
g_slist_free(rec->now_channels);
|
|
|
|
g_free(rec->nick);
|
|
g_free(rec);
|
|
}
|
|
|
|
static void netjoin_server_remove(NETJOIN_SERVER_REC *server)
|
|
{
|
|
joinservers = g_slist_remove(joinservers, server);
|
|
|
|
while (server->netjoins != NULL)
|
|
netjoin_remove(server, server->netjoins->data);
|
|
g_free(server);
|
|
}
|
|
|
|
static void print_channel_netjoins(char *channel, TEMP_PRINT_REC *rec,
|
|
NETJOIN_SERVER_REC *server)
|
|
{
|
|
if (rec->nicks->len > 0)
|
|
g_string_truncate(rec->nicks, rec->nicks->len-2);
|
|
|
|
printformat(server->server, channel, MSGLEVEL_JOINS,
|
|
rec->count > netjoin_max_nicks ?
|
|
IRCTXT_NETSPLIT_JOIN_MORE : IRCTXT_NETSPLIT_JOIN,
|
|
rec->nicks->str, rec->count-netjoin_max_nicks);
|
|
|
|
g_string_free(rec->nicks, TRUE);
|
|
g_free(rec);
|
|
g_free(channel);
|
|
}
|
|
|
|
static void print_netjoins(NETJOIN_SERVER_REC *server)
|
|
{
|
|
TEMP_PRINT_REC *temp;
|
|
GHashTable *channels;
|
|
GSList *tmp, *next, *old;
|
|
|
|
g_return_if_fail(server != NULL);
|
|
|
|
printing_joins = TRUE;
|
|
|
|
/* save nicks to string, clear now_channels and remove the same
|
|
channels from old_channels list */
|
|
channels = g_hash_table_new((GHashFunc) g_istr_hash,
|
|
(GCompareFunc) g_istr_equal);
|
|
for (tmp = server->netjoins; tmp != NULL; tmp = next) {
|
|
NETJOIN_REC *rec = tmp->data;
|
|
|
|
next = tmp->next;
|
|
while (rec->now_channels != NULL) {
|
|
char *channel = rec->now_channels->data;
|
|
char *realchannel = channel +
|
|
(isnickflag(*channel) && ischannel(channel[1]));
|
|
|
|
temp = g_hash_table_lookup(channels, realchannel);
|
|
if (temp == NULL) {
|
|
temp = g_new0(TEMP_PRINT_REC, 1);
|
|
temp->nicks = g_string_new(NULL);
|
|
g_hash_table_insert(channels,
|
|
g_strdup(realchannel),
|
|
temp);
|
|
}
|
|
|
|
temp->count++;
|
|
if (temp->count <= netjoin_max_nicks) {
|
|
if (isnickflag(*channel))
|
|
g_string_append_c(temp->nicks,
|
|
*channel);
|
|
g_string_sprintfa(temp->nicks, "%s, ",
|
|
rec->nick);
|
|
}
|
|
|
|
/* remove the channel from old_channels too */
|
|
old = gslist_find_icase_string(rec->old_channels,
|
|
realchannel);
|
|
if (old != NULL) {
|
|
g_free(old->data);
|
|
rec->old_channels =
|
|
g_slist_remove(rec->old_channels,
|
|
old->data);
|
|
}
|
|
|
|
g_free(channel);
|
|
rec->now_channels =
|
|
g_slist_remove(rec->now_channels, channel);
|
|
}
|
|
|
|
if (rec->old_channels == NULL)
|
|
netjoin_remove(server, rec);
|
|
}
|
|
|
|
g_hash_table_foreach(channels, (GHFunc) print_channel_netjoins,
|
|
server);
|
|
g_hash_table_destroy(channels);
|
|
|
|
if (server->netjoins == NULL)
|
|
netjoin_server_remove(server);
|
|
|
|
printing_joins = FALSE;
|
|
}
|
|
|
|
/* something is going to be printed to screen, print our current netsplit
|
|
message before it. */
|
|
static void sig_print_starting(void)
|
|
{
|
|
GSList *tmp, *next;
|
|
|
|
if (printing_joins)
|
|
return;
|
|
|
|
for (tmp = joinservers; tmp != NULL; tmp = next) {
|
|
NETJOIN_SERVER_REC *server = tmp->data;
|
|
|
|
next = tmp->next;
|
|
if (server->netjoins != NULL)
|
|
print_netjoins(server);
|
|
}
|
|
}
|
|
|
|
static int sig_check_netjoins(void)
|
|
{
|
|
GSList *tmp, *next;
|
|
int diff;
|
|
|
|
for (tmp = joinservers; tmp != NULL; tmp = next) {
|
|
NETJOIN_SERVER_REC *server = tmp->data;
|
|
|
|
next = tmp->next;
|
|
diff = time(NULL)-server->last_netjoin;
|
|
if (diff <= NETJOIN_WAIT_TIME) {
|
|
/* wait for more JOINs */
|
|
continue;
|
|
}
|
|
|
|
if (server->netjoins != NULL)
|
|
print_netjoins(server);
|
|
else if (diff >= NETJOIN_MAX_WAIT) {
|
|
/* waited long enough, remove the netjoin */
|
|
netjoin_server_remove(server);
|
|
}
|
|
}
|
|
|
|
if (joinservers == NULL) {
|
|
g_source_remove(join_tag);
|
|
signal_remove("print starting", (SIGNAL_FUNC) sig_print_starting);
|
|
join_tag = -1;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static void msg_quit(IRC_SERVER_REC *server, const char *nick,
|
|
const char *address, const char *reason)
|
|
{
|
|
if (IS_IRC_SERVER(server) && quitmsg_is_split(reason))
|
|
signal_stop();
|
|
}
|
|
|
|
static void msg_join(IRC_SERVER_REC *server, const char *channel,
|
|
const char *nick, const char *address)
|
|
{
|
|
NETSPLIT_REC *split;
|
|
NETJOIN_REC *netjoin;
|
|
|
|
if (!IS_IRC_SERVER(server))
|
|
return;
|
|
|
|
if (ignore_check(SERVER(server), nick, address,
|
|
channel, NULL, MSGLEVEL_JOINS))
|
|
return;
|
|
|
|
split = netsplit_find(server, nick, address);
|
|
netjoin = netjoin_find(server, nick);
|
|
if (split == NULL && netjoin == NULL)
|
|
return;
|
|
|
|
if (join_tag == -1) {
|
|
join_tag = g_timeout_add(1000, (GSourceFunc)
|
|
sig_check_netjoins, NULL);
|
|
signal_add("print starting", (SIGNAL_FUNC) sig_print_starting);
|
|
}
|
|
|
|
if (netjoin == NULL)
|
|
netjoin = netjoin_add(server, nick, split->channels);
|
|
|
|
netjoin->now_channels = g_slist_append(netjoin->now_channels,
|
|
g_strdup(channel));
|
|
signal_stop();
|
|
}
|
|
|
|
static int netjoin_set_nickmode(NETJOIN_REC *rec, const char *channel,
|
|
char mode)
|
|
{
|
|
GSList *pos;
|
|
|
|
pos = gslist_find_icase_string(rec->now_channels, channel);
|
|
if (pos == NULL)
|
|
return FALSE;
|
|
|
|
g_free(pos->data);
|
|
pos->data = g_strdup_printf("%c%s", mode, channel);
|
|
return TRUE;
|
|
}
|
|
|
|
#define isnickmode(c) \
|
|
((c) == 'o' || (c) == 'v' || (c) == 'h')
|
|
#define nickmodechar(c) \
|
|
((c) == 'o' ? '@' : ((c) == 'v' ? '+' : ((c) == 'h' ? '%' : '\0')))
|
|
|
|
static void msg_mode(IRC_SERVER_REC *server, const char *channel,
|
|
const char *sender, const char *addr, const char *data)
|
|
{
|
|
NETJOIN_REC *rec;
|
|
char *params, *mode, *nicks;
|
|
char **nicklist, **nick, type, modechr;
|
|
int show;
|
|
|
|
g_return_if_fail(data != NULL);
|
|
if (!ischannel(*channel) || addr != NULL)
|
|
return;
|
|
|
|
params = event_get_params(data, 2 | PARAM_FLAG_GETREST,
|
|
&mode, &nicks);
|
|
|
|
/* parse server mode changes - hide operator status changes and
|
|
show them in the netjoin message instead as @ before the nick */
|
|
nick = nicklist = g_strsplit(nicks, " ", -1);
|
|
|
|
type = '+'; show = FALSE;
|
|
for (; *mode != '\0'; mode++) {
|
|
if (*mode == '+' || *mode == '-') {
|
|
type = *mode;
|
|
continue;
|
|
}
|
|
|
|
if (*nick != NULL && isnickmode(*mode)) {
|
|
/* give/remove ops */
|
|
rec = netjoin_find(server, *nick);
|
|
modechr = nickmodechar(*mode);
|
|
if (rec == NULL || type != '+' || modechr == '\0' ||
|
|
!netjoin_set_nickmode(rec, channel, modechr))
|
|
show = TRUE;
|
|
nick++;
|
|
} else {
|
|
if (HAS_MODE_ARG(type, *mode) && *nick != NULL)
|
|
nick++;
|
|
show = TRUE;
|
|
}
|
|
}
|
|
|
|
if (!show) signal_stop();
|
|
|
|
g_strfreev(nicklist);
|
|
g_free(params);
|
|
}
|
|
|
|
static void read_settings(void)
|
|
{
|
|
int old_hide;
|
|
|
|
old_hide = hide_netsplit_quits;
|
|
hide_netsplit_quits = settings_get_bool("hide_netsplit_quits");
|
|
netjoin_max_nicks = settings_get_int("netjoin_max_nicks");
|
|
|
|
if (old_hide && !hide_netsplit_quits) {
|
|
signal_remove("message quit", (SIGNAL_FUNC) msg_quit);
|
|
signal_remove("message join", (SIGNAL_FUNC) msg_join);
|
|
signal_remove("message mode", (SIGNAL_FUNC) msg_mode);
|
|
} else if (!old_hide && hide_netsplit_quits) {
|
|
signal_add("message quit", (SIGNAL_FUNC) msg_quit);
|
|
signal_add("message join", (SIGNAL_FUNC) msg_join);
|
|
signal_add("message mode", (SIGNAL_FUNC) msg_mode);
|
|
}
|
|
}
|
|
|
|
void fe_netjoin_init(void)
|
|
{
|
|
settings_add_bool("misc", "hide_netsplit_quits", TRUE);
|
|
settings_add_int("misc", "netjoin_max_nicks", 10);
|
|
|
|
join_tag = -1;
|
|
printing_joins = FALSE;
|
|
|
|
read_settings();
|
|
signal_add("setup changed", (SIGNAL_FUNC) read_settings);
|
|
}
|
|
|
|
void fe_netjoin_deinit(void)
|
|
{
|
|
while (joinservers != NULL)
|
|
netjoin_server_remove(joinservers->data);
|
|
if (join_tag != -1) {
|
|
g_source_remove(join_tag);
|
|
signal_remove("print starting", (SIGNAL_FUNC) sig_print_starting);
|
|
}
|
|
|
|
signal_remove("setup changed", (SIGNAL_FUNC) read_settings);
|
|
}
|