2 * m_privs.c: Shows effective operator privileges
4 * Copyright (C) 2008 Jilles Tjoelker
5 * Copyright (C) 2008 charybdis development team
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions are
11 * 1.Redistributions of source code must retain the above copyright notice,
12 * this list of conditions and the following disclaimer.
13 * 2.Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
16 * 3.The name of the author may not be used to endorse or promote products
17 * derived from this software without specific prior written permission.
19 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
20 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
21 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
22 * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
23 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
24 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
25 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
27 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
28 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
40 #include "s_newconf.h"
43 static const char privs_desc
[] = "Provides the PRIVS command to inspect an operator's privileges";
45 static void m_privs(struct MsgBuf
*msgbuf_p
, struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[]);
46 static void me_privs(struct MsgBuf
*msgbuf_p
, struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[]);
47 static void mo_privs(struct MsgBuf
*msgbuf_p
, struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[]);
49 struct Message privs_msgtab
= {
51 {mg_unreg
, {m_privs
, 0}, mg_ignore
, mg_ignore
, {me_privs
, 0}, {mo_privs
, 0}}
54 mapi_clist_av1 privs_clist
[] = {
59 /* XXX this is a copy, not so nice
61 * Sort of... it's int in newconf.c since oper confs don't need 64-bit wide flags.
70 /* there is no such table like this anywhere else */
71 static struct mode_table auth_client_table
[] = {
72 {"resv_exempt", FLAGS_EXEMPTRESV
},
73 {"kline_exempt", FLAGS_EXEMPTKLINE
},
74 {"flood_exempt", FLAGS_EXEMPTFLOOD
},
75 {"spambot_exempt", FLAGS_EXEMPTSPAMBOT
},
76 {"shide_exempt", FLAGS_EXEMPTSHIDE
},
77 {"jupe_exempt", FLAGS_EXEMPTJUPE
},
78 {"extend_chans", FLAGS_EXTENDCHANS
},
82 DECLARE_MODULE_AV2(privs
, NULL
, NULL
, privs_clist
, NULL
, NULL
, NULL
, NULL
, privs_desc
);
84 static void append_priv(struct Client
*source_p
, struct Client
*target_p
, char *buf
, const char *s1
, const char *s2
)
86 /* 510 - ":" - " 270 " - " " - " :* " */
87 size_t sourcelen
= strlen(source_p
->name
);
88 if (sourcelen
< 9) sourcelen
= 9;
89 size_t limit
= 499 - strlen(me
.name
) - sourcelen
- strlen(target_p
->name
);
90 if (strlen(s1
) + strlen(s2
) + strlen(buf
) + 1 > limit
)
92 sendto_one_numeric(source_p
, RPL_PRIVS
, "%s :* %s", target_p
->name
, buf
);
96 rb_strlcat(buf
, " ", BUFSIZE
);
97 rb_strlcat(buf
, s1
, BUFSIZE
);
98 rb_strlcat(buf
, s2
, BUFSIZE
);
101 static void show_privs(struct Client
*source_p
, struct Client
*target_p
)
104 struct mode_table
*p
;
108 if (target_p
->user
->privset
)
109 for (char *s
= target_p
->user
->privset
->privs
; s
!= NULL
; (s
= strchr(s
, ' ')) && s
++)
111 char *c
= strchr(s
, ' ');
113 append_priv(source_p
, target_p
, buf
, s
, "");
117 if (IsOper(target_p
))
119 if (target_p
->user
->opername
)
120 append_priv(source_p
, target_p
, buf
, "operator:", target_p
->user
->opername
);
122 if (target_p
->user
->privset
)
123 append_priv(source_p
, target_p
, buf
, "privset:", target_p
->user
->privset
->name
);
125 p
= &auth_client_table
[0];
126 while (p
->name
!= NULL
)
128 if (target_p
->flags
& p
->mode
)
129 append_priv(source_p
, target_p
, buf
, p
->name
, "");
134 sendto_one_numeric(source_p
, RPL_PRIVS
, "%s :%s", target_p
->name
, buf
);
138 me_privs(struct MsgBuf
*msgbuf_p
, struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
140 struct Client
*target_p
;
142 if (!IsOper(source_p
) || parc
< 2 || EmptyString(parv
[1]))
145 target_p
= find_person(parv
[1]);
147 if (target_p
!= NULL
)
148 show_privs(source_p
, target_p
);
152 mo_privs(struct MsgBuf
*msgbuf_p
, struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
154 struct Client
*target_p
;
155 struct Client
*server_p
;
157 if (parc
< 2 || EmptyString(parv
[1]))
159 server_p
= target_p
= source_p
;
165 server_p
= find_named_client(parv
[1]);
166 target_p
= find_named_person(parv
[2]);
170 server_p
= target_p
= find_named_person(parv
[1]);
172 if (server_p
== NULL
|| target_p
== NULL
)
174 sendto_one_numeric(source_p
, ERR_NOSUCHNICK
,
175 form_str(ERR_NOSUCHNICK
), parv
[1]);
180 if (target_p
!= source_p
&& !HasPrivilege(source_p
, "oper:privs"))
182 sendto_one(source_p
, form_str(ERR_NOPRIVS
),
183 me
.name
, source_p
->name
, "privs");
187 if (!IsServer(server_p
))
188 server_p
= server_p
->servptr
;
191 show_privs(source_p
, target_p
);
193 sendto_one(server_p
, ":%s ENCAP %s PRIVS %s",
194 get_id(source_p
, server_p
),
200 m_privs(struct MsgBuf
*msgbuf_p
, struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
202 if (parc
>= 2 && !EmptyString(parv
[1]) &&
203 irccmp(parv
[1], source_p
->name
)) {
204 sendto_one_numeric(source_p
, ERR_NOPRIVILEGES
,
205 form_str(ERR_NOPRIVILEGES
));
209 show_privs(source_p
, source_p
);