2016-01-12 06:52:04 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2006 Jilles Tjoelker
|
|
|
|
* Copyright (C) 2006 Stephen Bennett <spb@gentoo.org>
|
|
|
|
*/
|
|
|
|
|
2016-08-13 05:05:54 +02:00
|
|
|
using namespace ircd;
|
|
|
|
|
2016-03-09 08:29:41 +01:00
|
|
|
static const char grant_desc[] =
|
|
|
|
"Provides the grant facility for giving other users specific privilege sets";
|
|
|
|
|
2016-08-23 02:37:07 +02:00
|
|
|
static void mo_grant(struct MsgBuf *msgbuf_p, client::client &client, client::client &source, int parc, const char *parv[]);
|
|
|
|
static void me_grant(struct MsgBuf *msgbuf_p, client::client &client, client::client &source, int parc, const char *parv[]);
|
2016-01-12 06:52:04 +01:00
|
|
|
|
2016-08-23 02:37:07 +02:00
|
|
|
static void do_grant(client::client &source, client::client *target_p, const char *new_privset);
|
2016-01-12 06:52:04 +01:00
|
|
|
|
|
|
|
struct Message grant_msgtab = {
|
2016-03-07 08:59:08 +01:00
|
|
|
"GRANT", 0, 0, 0, 0,
|
|
|
|
{ mg_ignore, mg_not_oper, mg_ignore, mg_ignore, {me_grant, 3}, {mo_grant, 3}}
|
2016-01-12 06:52:04 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
mapi_clist_av1 grant_clist[] = { &grant_msgtab, NULL };
|
|
|
|
|
2016-03-07 08:59:08 +01:00
|
|
|
DECLARE_MODULE_AV2(grant, NULL, NULL, grant_clist, NULL, NULL, NULL, NULL, grant_desc);
|
2016-01-12 06:52:04 +01:00
|
|
|
|
2016-03-09 08:37:03 +01:00
|
|
|
static void
|
2016-08-23 02:37:07 +02:00
|
|
|
mo_grant(struct MsgBuf *msgbuf_p, client::client &client, client::client &source, int parc, const char *parv[])
|
2016-01-12 06:52:04 +01:00
|
|
|
{
|
2016-08-22 03:57:43 +02:00
|
|
|
client::client *target_p;
|
2016-01-12 06:52:04 +01:00
|
|
|
|
2016-08-23 02:37:07 +02:00
|
|
|
if(!HasPrivilege(&source, "oper:grant"))
|
2016-01-12 06:52:04 +01:00
|
|
|
{
|
2016-08-23 02:37:07 +02:00
|
|
|
sendto_one(&source, form_str(ERR_NOPRIVS), me.name, source.name, "grant");
|
2016-03-09 08:37:03 +01:00
|
|
|
return;
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
|
2016-08-22 03:57:43 +02:00
|
|
|
target_p = client::find_named_person(parv[1]);
|
2016-01-12 06:52:04 +01:00
|
|
|
if (target_p == NULL)
|
|
|
|
{
|
2016-08-23 02:37:07 +02:00
|
|
|
sendto_one_numeric(&source, ERR_NOSUCHNICK,
|
2016-01-12 06:52:04 +01:00
|
|
|
form_str(ERR_NOSUCHNICK), parv[1]);
|
2016-03-09 08:37:03 +01:00
|
|
|
return;
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
|
2016-08-23 04:33:36 +02:00
|
|
|
if (my(*target_p))
|
2016-01-12 06:52:04 +01:00
|
|
|
{
|
2016-08-23 02:37:07 +02:00
|
|
|
do_grant(source, target_p, parv[2]);
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sendto_one(target_p, ":%s ENCAP %s GRANT %s %s",
|
2016-08-23 02:37:07 +02:00
|
|
|
get_id(&source, target_p), target_p->servptr->name,
|
2016-01-12 06:52:04 +01:00
|
|
|
get_id(target_p, target_p), parv[2]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-09 08:37:03 +01:00
|
|
|
static void
|
2016-08-23 02:37:07 +02:00
|
|
|
me_grant(struct MsgBuf *msgbuf_p, client::client &client, client::client &source, int parc, const char *parv[])
|
2016-01-12 06:52:04 +01:00
|
|
|
{
|
2016-08-22 03:57:43 +02:00
|
|
|
client::client *target_p;
|
2016-01-12 06:52:04 +01:00
|
|
|
|
2016-08-22 03:57:43 +02:00
|
|
|
target_p = client::find_person(parv[1]);
|
2016-01-12 06:52:04 +01:00
|
|
|
if (target_p == NULL)
|
|
|
|
{
|
2016-08-23 02:37:07 +02:00
|
|
|
sendto_one_numeric(&source, ERR_NOSUCHNICK,
|
2016-01-12 06:52:04 +01:00
|
|
|
form_str(ERR_NOSUCHNICK), parv[1]);
|
2016-03-09 08:37:03 +01:00
|
|
|
return;
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
|
2016-08-23 02:37:07 +02:00
|
|
|
if(!find_shared_conf(source.username, source.host,
|
|
|
|
source.servptr->name, SHARED_GRANT))
|
2016-01-12 06:52:04 +01:00
|
|
|
{
|
2016-08-23 02:37:07 +02:00
|
|
|
sendto_one(&source, ":%s NOTICE %s :You don't have an appropriate shared"
|
|
|
|
"block to grant privilege on this server.", me.name, source.name);
|
2016-03-09 08:37:03 +01:00
|
|
|
return;
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
|
2016-08-23 02:37:07 +02:00
|
|
|
do_grant(source, target_p, parv[2]);
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-03-09 08:37:03 +01:00
|
|
|
static void
|
2016-08-23 02:37:07 +02:00
|
|
|
do_grant(client::client &source, client::client *target_p, const char *new_privset)
|
2016-01-12 06:52:04 +01:00
|
|
|
{
|
|
|
|
int dooper = 0, dodeoper = 0;
|
|
|
|
struct PrivilegeSet *privset = 0;
|
2016-03-09 08:37:03 +01:00
|
|
|
const char *modeparv[4];
|
2016-01-12 06:52:04 +01:00
|
|
|
|
|
|
|
if (!strcmp(new_privset, "deoper"))
|
|
|
|
{
|
2016-08-24 00:25:09 +02:00
|
|
|
if (!is(*target_p, umode::OPER))
|
2016-01-12 06:52:04 +01:00
|
|
|
{
|
2016-08-23 02:37:07 +02:00
|
|
|
sendto_one_notice(&source, ":You can't deoper someone who isn't an oper.");
|
2016-03-09 08:37:03 +01:00
|
|
|
return;
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
new_privset = "default";
|
|
|
|
dodeoper = 1;
|
|
|
|
|
2016-08-23 02:37:07 +02:00
|
|
|
sendto_one_notice(target_p, ":%s is deopering you.", source.name);
|
|
|
|
sendto_realops_snomask(SNO_GENERAL, L_NETWIDE, "%s is deopering %s.", get_oper_name(&source), target_p->name);
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!(privset = privilegeset_get(new_privset)))
|
|
|
|
{
|
2016-08-23 02:37:07 +02:00
|
|
|
sendto_one_notice(&source, ":There is no privilege set named '%s'.", new_privset);
|
2016-03-09 08:37:03 +01:00
|
|
|
return;
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (privset == target_p->localClient->privset)
|
|
|
|
{
|
2016-08-23 02:37:07 +02:00
|
|
|
sendto_one_notice(&source, ":%s already has privilege set %s.", target_p->name, target_p->localClient->privset->name);
|
2016-03-09 08:37:03 +01:00
|
|
|
return;
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!dodeoper)
|
|
|
|
{
|
2016-08-24 00:25:09 +02:00
|
|
|
if (!is(*target_p, umode::OPER))
|
2016-01-12 06:52:04 +01:00
|
|
|
{
|
2016-08-23 02:37:07 +02:00
|
|
|
sendto_one_notice(target_p, ":%s is opering you with privilege set %s", source.name, privset->name);
|
|
|
|
sendto_realops_snomask(SNO_GENERAL, L_NETWIDE, "%s is opering %s with privilege set %s", get_oper_name(&source), target_p->name, privset->name);
|
2016-01-12 06:52:04 +01:00
|
|
|
dooper = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-08-23 02:37:07 +02:00
|
|
|
sendto_one_notice(target_p, ":%s is changing your privilege set to %s", source.name, privset->name);
|
|
|
|
sendto_realops_snomask(SNO_GENERAL, L_NETWIDE, "%s is changing the privilege set of %s to %s", get_oper_name(&source), target_p->name, privset->name);
|
2016-01-12 06:52:04 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dodeoper)
|
|
|
|
{
|
|
|
|
const char *modeparv[4];
|
|
|
|
modeparv[0] = modeparv[1] = target_p->name;
|
|
|
|
modeparv[2] = "-o";
|
|
|
|
modeparv[3] = NULL;
|
|
|
|
user_mode(target_p, target_p, 3, modeparv);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dooper)
|
|
|
|
{
|
|
|
|
struct oper_conf oper;
|
|
|
|
oper.name = "<grant>";
|
|
|
|
oper.umodes = 0;
|
|
|
|
oper.snomask = 0;
|
|
|
|
oper.privset = privset;
|
|
|
|
|
|
|
|
oper_up(target_p, &oper);
|
|
|
|
}
|
|
|
|
|
|
|
|
target_p->localClient->privset = privset;
|
|
|
|
modeparv[0] = modeparv[1] = target_p->name;
|
|
|
|
modeparv[2] = "+";
|
|
|
|
modeparv[3] = NULL;
|
|
|
|
user_mode(target_p, target_p, 3, modeparv);
|
|
|
|
}
|