pissircd/src/modules/quit.c

179 lines
4.3 KiB
C

/*
* Unreal Internet Relay Chat Daemon, src/modules/quit.c
* (C) 2000-2001 Carsten V. Munk and the UnrealIRCd Team
* Moved to modules by Fish (Justin Hammond)
*
* 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 1, 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., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include "unrealircd.h"
CMD_FUNC(cmd_quit);
#define MSG_QUIT "QUIT" /* QUIT */
ModuleHeader MOD_HEADER
= {
"quit", /* Name of module */
"5.0", /* Version */
"command /quit", /* Short description of module */
"UnrealIRCd Team",
"unrealircd-6",
};
/* This is called on module init, before Server Ready */
MOD_INIT()
{
CommandAdd(modinfo->handle, MSG_QUIT, cmd_quit, 1, CMD_UNREGISTERED|CMD_USER|CMD_VIRUS);
MARK_AS_OFFICIAL_MODULE(modinfo);
return MOD_SUCCESS;
}
/* Is first run when server is 100% ready */
MOD_LOAD()
{
return MOD_SUCCESS;
}
/* Called when module is unloaded */
MOD_UNLOAD()
{
return MOD_SUCCESS;
}
/*
** cmd_quit
** parv[1] = comment
*/
CMD_FUNC(cmd_quit)
{
const char *comment = (parc > 1 && parv[1]) ? parv[1] : client->name;
char commentbuf[MAXQUITLEN + 1];
char commentbuf2[MAXQUITLEN + 1];
if (parc > 1 && parv[1])
{
strlncpy(commentbuf, parv[1], sizeof(commentbuf), iConf.quit_length);
comment = commentbuf;
} else {
comment = client->name;
}
if (MyUser(client))
{
int n;
Hook *tmphook;
const char *str;
if ((str = get_setting_for_user_string(client, SET_STATIC_QUIT)))
{
exit_client(client, recv_mtags, str);
return;
}
if (IsVirus(client))
{
exit_client(client, recv_mtags, "Client exited");
return;
}
if (match_spamfilter(client, comment, SPAMF_QUIT, "QUIT", NULL, 0, NULL))
{
comment = client->name;
if (IsDead(client))
return;
}
if (!ValidatePermissionsForPath("immune:anti-spam-quit-message-time",client,NULL,NULL,NULL) && ANTI_SPAM_QUIT_MSG_TIME)
{
if (client->local->creationtime+ANTI_SPAM_QUIT_MSG_TIME > TStime())
comment = client->name;
}
if (iConf.part_instead_of_quit_on_comment_change && MyUser(client))
{
Membership *lp, *lp_next;
const char *newcomment;
Channel *channel;
for (lp = client->user->channel; lp; lp = lp_next)
{
channel = lp->channel;
newcomment = comment;
lp_next = lp->next;
for (tmphook = Hooks[HOOKTYPE_PRE_LOCAL_QUIT_CHAN]; tmphook; tmphook = tmphook->next)
{
newcomment = (*(tmphook->func.stringfunc))(client, channel, comment);
if (!newcomment)
break;
}
if (newcomment && is_banned(client, channel, BANCHK_LEAVE_MSG, &newcomment, NULL))
newcomment = NULL;
/* Comment changed? Then PART the user before we do the QUIT. */
if (comment != newcomment)
{
const char *parx[4];
char tmp[512];
int ret;
parx[0] = NULL;
parx[1] = channel->name;
if (newcomment)
{
strlcpy(tmp, newcomment, sizeof(tmp));
parx[2] = tmp;
parx[3] = NULL;
} else {
parx[2] = NULL;
}
do_cmd(client, recv_mtags, "PART", newcomment ? 3 : 2, parx);
/* This would be unusual, but possible (somewhere in the future perhaps): */
if (IsDead(client))
return;
}
}
}
for (tmphook = Hooks[HOOKTYPE_PRE_LOCAL_QUIT]; tmphook; tmphook = tmphook->next)
{
comment = (*(tmphook->func.stringfunc))(client, comment);
if (!comment)
{
comment = client->name;
break;
}
}
if (PREFIX_QUIT)
snprintf(commentbuf2, sizeof(commentbuf2), "%s: %s", PREFIX_QUIT, comment);
else
strlcpy(commentbuf2, comment, sizeof(commentbuf2));
exit_client(client, recv_mtags, commentbuf2);
}
else
{
/* Remote quits and non-person quits always use their original comment.
* Also pass recv_mtags so to keep the msgid and such.
*/
exit_client(client, recv_mtags, comment);
}
}