]> jfr.im git - solanum.git/blame - extensions/helpops.c
chmode: Get elevated access for op-only queries
[solanum.git] / extensions / helpops.c
CommitLineData
4d21f1e8
AC
1/*
2 * Helpops system.
3 * -- kaniini
4 */
5
6#include "stdinc.h"
7#include "modules.h"
8#include "client.h"
9#include "hook.h"
10#include "ircd.h"
11#include "send.h"
12#include "s_conf.h"
13#include "s_user.h"
14#include "s_newconf.h"
15#include "numeric.h"
16
eeabf33a
EM
17static const char helpops_desc[] = "The helpops system as used by freenode";
18
4d21f1e8
AC
19static rb_dlink_list helper_list = { NULL, NULL, 0 };
20static void h_hdl_stats_request(hook_data_int *hdata);
21static void h_hdl_new_remote_user(struct Client *client_p);
22static void h_hdl_client_exit(hook_data_client_exit *hdata);
23static void h_hdl_umode_changed(hook_data_umode_changed *hdata);
24static void h_hdl_whois(hook_data_client *hdata);
4d3f8ead 25static void recurse_client_exit(struct Client *client_p);
b1c32af9
AC
26static void helper_add(struct Client *client_p);
27static void helper_delete(struct Client *client_p);
3c7d6fcc
EM
28static void mo_dehelper(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
29static void me_dehelper(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
30static void do_dehelper(struct Client *source_p, struct Client *target_p);
4d21f1e8
AC
31
32mapi_hfn_list_av1 helpops_hfnlist[] = {
33 { "doing_stats", (hookfn) h_hdl_stats_request },
34 { "new_remote_user", (hookfn) h_hdl_new_remote_user },
35 { "client_exit", (hookfn) h_hdl_client_exit },
36 { "umode_changed", (hookfn) h_hdl_umode_changed },
37 { "doing_whois", (hookfn) h_hdl_whois },
38 { "doing_whois_global", (hookfn) h_hdl_whois },
39 { NULL, NULL }
40};
41
0f8ec938 42#define UMODECHAR_HELPOPS 'h'
4d21f1e8 43
161ac1c8 44struct Message dehelper_msgtab = {
7baa37a9 45 "DEHELPER", 0, 0, 0, 0,
161ac1c8
AC
46 {mg_unreg, mg_not_oper, mg_not_oper, mg_ignore, {me_dehelper, 2}, {mo_dehelper, 2}}
47};
48
49mapi_clist_av1 helpops_clist[] = { &dehelper_msgtab, NULL };
50
3c7d6fcc
EM
51static void
52mo_dehelper(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
161ac1c8
AC
53{
54 struct Client *target_p;
55
4a4b2095 56 if (!HasPrivilege(source_p, "oper:dehelper"))
161ac1c8 57 {
4a4b2095 58 sendto_one(source_p, form_str(ERR_NOPRIVS), me.name, source_p->name, "dehelper");
3c7d6fcc 59 return;
161ac1c8
AC
60 }
61
62 if(!(target_p = find_named_person(parv[1])))
63 {
64 sendto_one_numeric(source_p, ERR_NOSUCHNICK, form_str(ERR_NOSUCHNICK), parv[1]);
3c7d6fcc 65 return;
161ac1c8
AC
66 }
67
68 if(MyClient(target_p))
69 do_dehelper(source_p, target_p);
70 else
71 sendto_one(target_p, ":%s ENCAP %s DEHELPER %s",
72 use_id(source_p), target_p->servptr->name, use_id(target_p));
161ac1c8
AC
73}
74
3c7d6fcc
EM
75static void
76me_dehelper(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
161ac1c8
AC
77{
78 struct Client *target_p = find_person(parv[1]);
79 if(!target_p)
80 {
81 sendto_one_numeric(source_p, ERR_NOSUCHNICK, form_str(ERR_NOSUCHNICK), parv[1]);
3c7d6fcc 82 return;
161ac1c8
AC
83 }
84 if(!MyClient(target_p))
3c7d6fcc 85 return;
161ac1c8
AC
86
87 do_dehelper(source_p, target_p);
161ac1c8
AC
88}
89
3c7d6fcc
EM
90static void
91do_dehelper(struct Client *source_p, struct Client *target_p)
161ac1c8
AC
92{
93 const char *fakeparv[4];
0f8ec938 94 static const char minus_helpops[3] = {'-', UMODECHAR_HELPOPS, '\0'};
161ac1c8 95
6c639159 96 if(!(target_p->umodes & user_modes[UMODECHAR_HELPOPS]))
3c7d6fcc 97 return;
161ac1c8
AC
98
99 sendto_realops_snomask(SNO_GENERAL, L_NETWIDE, "%s is using DEHELPER on %s",
100 source_p->name, target_p->name);
101 sendto_one_notice(target_p, ":*** %s is using DEHELPER on you", source_p->name);
102
103 fakeparv[0] = fakeparv[1] = target_p->name;
0f8ec938 104 fakeparv[2] = minus_helpops;
161ac1c8
AC
105 fakeparv[3] = NULL;
106 user_mode(target_p, target_p, 3, fakeparv);
161ac1c8
AC
107}
108
4d21f1e8
AC
109static int
110_modinit(void)
111{
0c5dd86c
EK
112 rb_dlink_node *ptr;
113
6c639159 114 user_modes[UMODECHAR_HELPOPS] = find_umode_slot();
4d21f1e8
AC
115 construct_umodebuf();
116
0c5dd86c
EK
117 RB_DLINK_FOREACH (ptr, global_client_list.head)
118 {
119 struct Client *client_p = ptr->data;
120 if (IsPerson(client_p) && (client_p->umodes & user_modes[UMODECHAR_HELPOPS]))
121 helper_add(client_p);
122 }
123
4d21f1e8
AC
124 return 0;
125}
126
127static void
128_moddeinit(void)
129{
0c5dd86c
EK
130 rb_dlink_node *n, *tn;
131
6c639159 132 user_modes[UMODECHAR_HELPOPS] = 0;
4d21f1e8 133 construct_umodebuf();
0c5dd86c
EK
134
135 RB_DLINK_FOREACH_SAFE(n, tn, helper_list.head)
136 rb_dlinkDestroy(n, &helper_list);
4d21f1e8
AC
137}
138
139static void
140h_hdl_stats_request(hook_data_int *hdata)
141{
142 struct Client *target_p;
143 rb_dlink_node *helper_ptr;
144 unsigned int count = 0;
145
146 if (hdata->arg2 != 'p')
147 return;
148
149 RB_DLINK_FOREACH (helper_ptr, helper_list.head)
150 {
151 target_p = helper_ptr->data;
152
4d21f1e8
AC
153 if(target_p->user->away)
154 continue;
155
156 count++;
157
158 sendto_one_numeric(hdata->client, RPL_STATSDEBUG,
159 "p :%s (%s@%s)",
160 target_p->name, target_p->username,
161 target_p->host);
162 }
163
164 sendto_one_numeric(hdata->client, RPL_STATSDEBUG,
165 "p :%u staff members", count);
166
167 hdata->result = 1;
168}
169
b1c32af9
AC
170static void
171helper_add(struct Client *client_p)
172{
173 if (rb_dlinkFind(client_p, &helper_list) != NULL)
174 return;
175
176 rb_dlinkAddAlloc(client_p, &helper_list);
177}
178
179static void
180helper_delete(struct Client *client_p)
181{
182 rb_dlinkFindDestroy(client_p, &helper_list);
183}
184
4d21f1e8
AC
185static void
186h_hdl_new_remote_user(struct Client *client_p)
187{
6c639159 188 if (client_p->umodes & user_modes[UMODECHAR_HELPOPS])
b1c32af9 189 helper_add(client_p);
4d21f1e8
AC
190}
191
192static void
4d3f8ead 193recurse_client_exit(struct Client *client_p)
4d21f1e8 194{
4d3f8ead 195 if (IsPerson(client_p))
8093dc5f 196 {
6c639159 197 if (client_p->umodes & user_modes[UMODECHAR_HELPOPS])
b1c32af9 198 helper_delete(client_p);
8093dc5f 199 }
4d3f8ead 200 else if (IsServer(client_p))
8093dc5f
AC
201 {
202 rb_dlink_node *nptr;
203
4d3f8ead
AC
204 RB_DLINK_FOREACH(nptr, client_p->serv->users.head)
205 recurse_client_exit(nptr->data);
8093dc5f 206
4d3f8ead
AC
207 RB_DLINK_FOREACH(nptr, client_p->serv->servers.head)
208 recurse_client_exit(nptr->data);
8093dc5f 209 }
4d21f1e8
AC
210}
211
4d3f8ead
AC
212static void
213h_hdl_client_exit(hook_data_client_exit *hdata)
214{
215 recurse_client_exit(hdata->target);
216}
217
4d21f1e8
AC
218static void
219h_hdl_umode_changed(hook_data_umode_changed *hdata)
220{
221 struct Client *source_p = hdata->client;
222
0f8ec938 223 /* didn't change +h umode, we don't need to do anything */
53390430 224 bool changed = (hdata->oldumodes ^ source_p->umodes) & user_modes[UMODECHAR_HELPOPS];
4d21f1e8 225
6c639159 226 if (source_p->umodes & user_modes[UMODECHAR_HELPOPS])
4d21f1e8
AC
227 {
228 if (MyClient(source_p) && !HasPrivilege(source_p, "usermode:helpops"))
229 {
6c639159 230 source_p->umodes &= ~user_modes[UMODECHAR_HELPOPS];
4d21f1e8 231 sendto_one(source_p, form_str(ERR_NOPRIVS), me.name, source_p->name, "usermode:helpops");
0f8ec938 232 /* they didn't ask for +h so we must be removing it */
53390430
EK
233 if (!changed)
234 helper_delete(source_p);
4d21f1e8
AC
235 return;
236 }
237
53390430
EK
238 if (changed)
239 helper_add(source_p);
4d21f1e8 240 }
53390430
EK
241 else if (changed)
242 {
b1c32af9 243 helper_delete(source_p);
53390430 244 }
4d21f1e8
AC
245}
246
247static void
248h_hdl_whois(hook_data_client *hdata)
249{
250 struct Client *source_p = hdata->client;
251 struct Client *target_p = hdata->target;
252
6c639159 253 if ((target_p->umodes & user_modes[UMODECHAR_HELPOPS]) && EmptyString(target_p->user->away))
4d21f1e8
AC
254 {
255 sendto_one_numeric(source_p, RPL_WHOISHELPOP, form_str(RPL_WHOISHELPOP), target_p->name);
256 }
257}
258
a278a4fc 259DECLARE_MODULE_AV2(helpops, _modinit, _moddeinit, helpops_clist, NULL, helpops_hfnlist, NULL, NULL, helpops_desc);