inspircd/src/modules/m_kicknorejoin.cpp

178 lines
4.4 KiB
C++
Raw Normal View History

/*
* InspIRCd -- Internet Relay Chat Daemon
*
2020-01-11 22:02:47 +00:00
* Copyright (C) 2018 A_D <A-UNDERSCORE-D@users.noreply.github.com>
* Copyright (C) 2014 Daniel Vassdal <shutter@canternet.org>
2020-04-24 10:23:47 +01:00
* Copyright (C) 2013, 2017-2020 Sadie Powell <sadie@witchery.services>
2020-01-11 22:02:47 +00:00
* Copyright (C) 2012-2015 Attila Molnar <attilamolnar@hush.com>
* Copyright (C) 2012, 2019 Robby <robby@chatbelgie.be>
* Copyright (C) 2009 Uli Schlachter <psychon@inspircd.org>
* Copyright (C) 2009 Daniel De Graaf <danieldg@inspircd.org>
* Copyright (C) 2007 Dennis Friis <peavey@inspircd.org>
2020-01-11 22:02:47 +00:00
* Copyright (C) 2006, 2010 Craig Edwards <brain@inspircd.org>
2021-03-05 02:04:16 +00:00
* Copyright (C) 2006 Oliver Lupton <om@inspircd.org>
*
* This file is part of InspIRCd. InspIRCd 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, version 2.
*
* 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, see <http://www.gnu.org/licenses/>.
*/
#include "inspircd.h"
#include "modules/invite.h"
enum
{
// From RFC 2182.
ERR_UNAVAILRESOURCE = 437
};
class KickRejoinData
{
struct KickedUser
{
std::string uuid;
time_t expire;
KickedUser(User* user, unsigned int Delay)
: uuid(user->uuid)
, expire(ServerInstance->Time() + Delay)
{
}
};
typedef std::vector<KickedUser> KickedList;
mutable KickedList kicked;
public:
const unsigned int delay;
KickRejoinData(unsigned int Delay) : delay(Delay) { }
bool canjoin(LocalUser* user) const
{
for (KickedList::iterator i = kicked.begin(); i != kicked.end(); )
{
KickedUser& rec = *i;
if (rec.expire > ServerInstance->Time())
{
if (rec.uuid == user->uuid)
return false;
++i;
}
else
{
// Expired record, remove.
stdalgo::vector::swaperase(kicked, i);
if (kicked.empty())
break;
}
}
return true;
}
void add(User* user)
{
// One user can be in the list multiple times if the user gets kicked, force joins
// (skipping OnUserPreJoin) and gets kicked again, but that's okay because canjoin()
// works correctly in this case as well
2021-03-30 19:44:07 +01:00
kicked.emplace_back(user, delay);
}
};
/** Handles channel mode +J
*/
class KickRejoin : public ParamMode<KickRejoin, SimpleExtItem<KickRejoinData> >
{
const unsigned int max = 60;
public:
KickRejoin(Module* Creator)
: ParamMode<KickRejoin, SimpleExtItem<KickRejoinData> >(Creator, "kicknorejoin", 'J')
{
syntax = "<seconds>";
}
ModeAction OnSet(User* source, Channel* channel, std::string& parameter) override
{
unsigned int v = ConvToNum<unsigned int>(parameter);
if (v <= 0)
{
source->WriteNumeric(Numerics::InvalidModeParameter(channel, this, parameter));
return MODEACTION_DENY;
}
if (IS_LOCAL(source) && v > max)
v = max;
ext.Set(channel, v);
return MODEACTION_ALLOW;
}
void SerializeParam(Channel* chan, const KickRejoinData* krd, std::string& out)
{
out.append(ConvToStr(krd->delay));
}
std::string GetModuleSettings() const
{
return ConvToStr(max);
}
};
class ModuleKickNoRejoin : public Module
{
KickRejoin kr;
Invite::API invapi;
public:
ModuleKickNoRejoin()
2020-04-11 14:00:48 +01:00
: Module(VF_VENDOR | VF_COMMON, "Adds channel mode J (kicknorejoin) which prevents users from rejoining after being kicked from a channel.")
, kr(this)
, invapi(this)
{
}
ModResult OnUserPreJoin(LocalUser* user, Channel* chan, const std::string& cname, std::string& privs, const std::string& keygiven) override
{
if (chan)
{
const KickRejoinData* data = kr.ext.Get(chan);
if ((data) && !invapi->IsInvited(user, chan) && (!data->canjoin(user)))
{
user->WriteNumeric(ERR_UNAVAILRESOURCE, chan->name, InspIRCd::Format("You must wait %u seconds after being kicked to rejoin (+J is set)", data->delay));
return MOD_RES_DENY;
}
}
return MOD_RES_PASSTHRU;
}
void OnUserKick(User* source, Membership* memb, const std::string &reason, CUList& excepts) override
{
if ((!IS_LOCAL(memb->user)) || (source == memb->user))
return;
KickRejoinData* data = kr.ext.Get(memb->chan);
if (data)
{
data->add(memb->user);
}
}
void GetLinkData(std::string& data) override
{
data = kr.GetModuleSettings();
}
};
MODULE_INIT(ModuleKickNoRejoin)