1
0
mirror of https://github.com/unrealircd/unrealircd.git synced 2026-07-04 02:23:13 +02:00
Files
unrealircd/src/modules/chanmodes/secureonly.c
T
Bram Matthys 10faaf91ba Bump module header version to "unrealircd-5", add author field
and remove old dependency field (never used, was always NULL,
broken since 3.2.x)
I'll add some constraints later on things like names and versions.
IOTW: more changes to follow, don't mass update your own mods yet.
2019-08-26 20:29:47 +02:00

199 lines
5.3 KiB
C

/*
* Only allow secure users to join UnrealIRCd Module (Channel Mode +z)
* (C) Copyright 2014 Travis McArthur (Heero) and the UnrealIRCd team
*
* 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"
ModuleHeader MOD_HEADER(secureonly)
= {
"chanmodes/secureonly",
"4.2",
"Channel Mode +z",
"UnrealIRCd Team",
"unrealircd-5",
};
Cmode_t EXTCMODE_SECUREONLY;
#define IsSecureOnly(chptr) (chptr->mode.extmode & EXTCMODE_SECUREONLY)
int secureonly_check_join(aClient *sptr, aChannel *chptr, char *key, char *parv[]);
void secureonly_channel_sync (aChannel* chptr, int merge, int removetheirs, int nomode);
int secureonly_send_channel(aClient *acptr, aChannel* chptr);
int secureonly_check_secure(aChannel* chptr);
int secureonly_check_sajoin(aClient *acptr, aChannel* chptr, aClient *sptr);
int secureonly_specialcheck(aClient *sptr, aChannel *chptr, char *parv[]);
MOD_TEST(secureonly)
{
return MOD_SUCCESS;
}
MOD_INIT(secureonly)
{
CmodeInfo req;
memset(&req, 0, sizeof(req));
req.paracount = 0;
req.flag = 'z';
req.is_ok = extcmode_default_requirechop;
CmodeAdd(modinfo->handle, req, &EXTCMODE_SECUREONLY);
HookAdd(modinfo->handle, HOOKTYPE_PRE_LOCAL_JOIN, 0, secureonly_specialcheck);
HookAdd(modinfo->handle, HOOKTYPE_CAN_JOIN, 0, secureonly_check_join);
HookAddVoid(modinfo->handle, HOOKTYPE_CHANNEL_SYNCED, 0, secureonly_channel_sync);
HookAdd(modinfo->handle, HOOKTYPE_IS_CHANNEL_SECURE, 0, secureonly_check_secure);
HookAdd(modinfo->handle, HOOKTYPE_SEND_CHANNEL, 0, secureonly_send_channel);
HookAdd(modinfo->handle, HOOKTYPE_CAN_SAJOIN, 0, secureonly_check_sajoin);
MARK_AS_OFFICIAL_MODULE(modinfo);
return MOD_SUCCESS;
}
MOD_LOAD(secureonly)
{
return MOD_SUCCESS;
}
MOD_UNLOAD(secureonly)
{
return MOD_SUCCESS;
}
/** Kicks all insecure users on a +z channel */
static void secureonly_kick_insecure_users(aChannel *chptr)
{
Member *member, *mb2;
aClient *cptr;
int i = 0;
Hook *h;
char *comment = "Insecure user not allowed on secure channel (+z)";
if (!IsSecureOnly(chptr))
return;
for (member = chptr->members; member; member = mb2)
{
mb2 = member->next;
cptr = member->cptr;
if (MyClient(cptr) && !IsSecureConnect(cptr) && !IsULine(cptr))
{
int prefix = 0;
MessageTag *mtags = NULL;
RunHook5(HOOKTYPE_LOCAL_KICK, &me, &me, cptr, chptr, comment);
if (invisible_user_in_channel(cptr, chptr))
{
/* Send only to chanops */
prefix = CHFL_HALFOP|CHFL_CHANOP|CHFL_CHANOWNER|CHFL_CHANADMIN;
}
new_message(&me, NULL, &mtags);
sendto_channel(chptr, &me, cptr,
prefix, 0,
SEND_LOCAL, mtags,
":%s KICK %s %s :%s",
me.name, chptr->chname, cptr->name, comment);
sendto_prefix_one(cptr, &me, mtags, ":%s KICK %s %s :%s", me.name, chptr->chname, cptr->name, comment);
sendto_server(&me, 0, 0, mtags, ":%s KICK %s %s :%s", me.name, chptr->chname, cptr->name, comment);
free_message_tags(mtags);
remove_user_from_channel(cptr, chptr);
}
}
}
int secureonly_check_join(aClient *sptr, aChannel *chptr, char *key, char *parv[])
{
Link *lp;
if (IsSecureOnly(chptr) && !(sptr->umodes & UMODE_SECURE))
{
if (ValidatePermissionsForPath("channel:override:secureonly",sptr,NULL,chptr,NULL))
{
/* if the channel is +z we still allow an ircop to bypass it
* if they are invited.
*/
for (lp = sptr->user->invited; lp; lp = lp->next)
if (lp->value.chptr == chptr)
return HOOK_CONTINUE;
}
return (ERR_SECUREONLYCHAN);
}
return 0;
}
int secureonly_check_secure(aChannel *chptr)
{
if (IsSecureOnly(chptr))
{
return 1;
}
return 0;
}
void secureonly_channel_sync(aChannel *chptr, int merge, int removetheirs, int nomode)
{
if ((!merge && !removetheirs && !nomode) || (merge && !nomode))
{
secureonly_kick_insecure_users(chptr);
}
}
int secureonly_send_channel(aClient *acptr, aChannel *chptr)
{
if (IsSecureOnly(chptr))
if (!IsSecure(acptr))
return HOOK_DENY;
return HOOK_CONTINUE;
}
int secureonly_check_sajoin(aClient *acptr, aChannel *chptr, aClient *sptr)
{
if (IsSecureOnly(chptr) && !IsSecure(acptr))
{
sendnotice(sptr, "You cannot SAJOIN %s to %s because the channel is +z and the user is not connected via SSL/TLS",
acptr->name, chptr->chname);
return HOOK_DENY;
}
return HOOK_CONTINUE;
}
/* Special check for +z in set::modes-on-join. Needs to be done early.
* Perhaps one day this will be properly handled in the core so this can be removed.
*/
int secureonly_specialcheck(aClient *sptr, aChannel *chptr, char *parv[])
{
if ((chptr->users == 0) && (iConf.modes_on_join.extmodes & EXTCMODE_SECUREONLY) && !IsSecure(sptr) && !IsOper(sptr))
{
sendnumeric(sptr, ERR_SECUREONLYCHAN, chptr->chname);
return HOOK_DENY;
}
return HOOK_CONTINUE;
}