2 * oper-override for charybdis.
4 * adds usermode +p and has a timer event that is iterated over to disable
5 * usermode +p after a while...
7 * you need to have oper:override permission on the opers you want to be
8 * able to use this extension.
22 #include "privilege.h"
23 #include "s_newconf.h"
25 static const char override_desc
[] =
26 "Adds user mode +p, an operator-only user mode that grants temporary privileges to override anything";
28 static void check_umode_change(void *data
);
29 static void hack_channel_access(void *data
);
30 static void hack_can_join(void *data
);
31 static void hack_can_kick(void *data
);
32 static void hack_can_send(void *data
);
33 static void handle_client_exit(void *data
);
35 mapi_hfn_list_av1 override_hfnlist
[] = {
36 { "umode_changed", (hookfn
) check_umode_change
},
37 { "get_channel_access", (hookfn
) hack_channel_access
, HOOK_HIGHEST
},
38 { "can_join", (hookfn
) hack_can_join
, HOOK_HIGHEST
},
39 { "can_kick", (hookfn
) hack_can_kick
, HOOK_HIGHEST
},
40 { "can_send", (hookfn
) hack_can_send
, HOOK_HIGHEST
},
41 { "client_exit", (hookfn
) handle_client_exit
},
45 #define CHFL_OVERRIDE 0x0004
46 #define IsOperOverride(x) (HasPrivilege((x), "oper:override"))
48 struct OverrideSession
{
51 struct Client
*client
;
55 rb_dlink_list overriding_opers
= { NULL
, NULL
, 0 };
58 update_session_deadline(struct Client
*source_p
, struct OverrideSession
*session_p
)
60 if (session_p
== NULL
)
64 RB_DLINK_FOREACH(n
, overriding_opers
.head
)
66 struct OverrideSession
*s
= n
->data
;
68 if (s
->client
== source_p
)
76 if (session_p
== NULL
)
78 session_p
= rb_malloc(sizeof(struct OverrideSession
));
79 session_p
->client
= source_p
;
82 session_p
->deadline
= rb_current_time() + 1800;
84 rb_dlinkDelete(&session_p
->node
, &overriding_opers
);
85 rb_dlinkAdd(session_p
, &session_p
->node
, &overriding_opers
);
89 expire_override_deadlines(void *unused
)
91 rb_dlink_node
*n
, *tn
;
93 RB_DLINK_FOREACH_SAFE(n
, tn
, overriding_opers
.head
)
95 struct OverrideSession
*session_p
= n
->data
;
97 if (session_p
->deadline
> rb_current_time())
99 else if (session_p
->deadline
< rb_current_time())
101 const char *parv
[4] = {session_p
->client
->name
, session_p
->client
->name
, "-p", NULL
};
102 user_mode(session_p
->client
, session_p
->client
, 3, parv
);
108 check_umode_change(void *vdata
)
110 hook_data_umode_changed
*data
= (hook_data_umode_changed
*)vdata
;
111 bool changed
= false;
112 struct Client
*source_p
= data
->client
;
114 if (!MyClient(source_p
))
117 if (data
->oldumodes
& UMODE_OPER
&& !IsOper(source_p
))
118 source_p
->umodes
&= ~user_modes
['p'];
120 changed
= ((data
->oldumodes
^ source_p
->umodes
) & user_modes
['p']);
122 if (source_p
->umodes
& user_modes
['p'])
124 if (!IsOperOverride(source_p
))
126 sendto_one_notice(source_p
, ":*** You need oper:override privilege for +p");
127 source_p
->umodes
&= ~user_modes
['p'];
133 update_session_deadline(source_p
, NULL
);
136 else if (changed
&& !(source_p
->umodes
& user_modes
['p']))
138 rb_dlink_node
*n
, *tn
;
140 RB_DLINK_FOREACH_SAFE(n
, tn
, overriding_opers
.head
)
142 struct OverrideSession
*session_p
= n
->data
;
144 if (session_p
->client
!= source_p
)
147 rb_dlinkDelete(n
, &overriding_opers
);
154 hack_channel_access(void *vdata
)
156 hook_data_channel_approval
*data
= (hook_data_channel_approval
*) vdata
;
158 if (data
->dir
== MODE_QUERY
)
161 if (data
->approved
== CHFL_CHANOP
)
164 if (data
->client
->umodes
& user_modes
['p'])
166 update_session_deadline(data
->client
, NULL
);
167 data
->approved
= CHFL_OVERRIDE
;
169 /* we only want to report modehacks, which are always non-NULL */
171 sendto_realops_snomask(SNO_GENERAL
, L_NETWIDE
, "%s is using oper-override on %s (modehacking: %s)",
172 get_oper_name(data
->client
), data
->chptr
->chname
, data
->modestr
);
177 hack_can_join(void *vdata
)
179 hook_data_channel
*data
= (hook_data_channel
*) vdata
;
181 if (data
->approved
== 0)
184 if (data
->client
->umodes
& user_modes
['p'])
186 update_session_deadline(data
->client
, NULL
);
189 sendto_realops_snomask(SNO_GENERAL
, L_NETWIDE
, "%s is using oper-override on %s (banwalking)",
190 get_oper_name(data
->client
), data
->chptr
->chname
);
195 hack_can_kick(void *vdata
)
197 hook_data_channel_approval
*data
= (hook_data_channel_approval
*) vdata
;
200 alevel
= get_channel_access(data
->client
, data
->chptr
, data
->msptr
, data
->dir
, NULL
);
201 if (alevel
!= CHFL_OVERRIDE
)
204 if (data
->client
->umodes
& user_modes
['p'])
206 update_session_deadline(data
->client
, NULL
);
207 sendto_realops_snomask(SNO_GENERAL
, L_NETWIDE
, "%s is using oper-override on %s (KICK %s)",
208 get_oper_name(data
->client
), data
->chptr
->chname
, data
->target
->name
);
213 hack_can_send(void *vdata
)
215 hook_data_channel_approval
*data
= (hook_data_channel_approval
*) vdata
;
217 if (data
->dir
== MODE_QUERY
)
220 if (data
->approved
== CAN_SEND_NONOP
|| data
->approved
== CAN_SEND_OPV
)
223 if (data
->client
->umodes
& user_modes
['p'])
225 data
->approved
= CAN_SEND_NONOP
;
227 if (MyClient(data
->client
))
229 update_session_deadline(data
->client
, NULL
);
230 sendto_realops_snomask(SNO_GENERAL
, L_NETWIDE
, "%s is using oper-override on %s (forcing message)",
231 get_oper_name(data
->client
), data
->chptr
->chname
);
237 handle_client_exit(void *vdata
)
239 hook_data_client_exit
*data
= (hook_data_client_exit
*) vdata
;
240 rb_dlink_node
*n
, *tn
;
241 struct Client
*source_p
= data
->target
;
243 RB_DLINK_FOREACH_SAFE(n
, tn
, overriding_opers
.head
)
245 struct OverrideSession
*session_p
= n
->data
;
247 if (session_p
->client
!= source_p
)
250 rb_dlinkDelete(n
, &overriding_opers
);
255 struct ev_entry
*expire_override_deadlines_ev
= NULL
;
262 /* add the usermode to the available slot */
263 user_modes
['p'] = find_umode_slot();
264 construct_umodebuf();
266 RB_DLINK_FOREACH(ptr
, lclient_list
.head
)
268 struct Client
*client_p
= ptr
->data
;
269 if (IsPerson(client_p
) && (client_p
->umodes
& user_modes
['p']))
270 update_session_deadline(client_p
, NULL
);
273 expire_override_deadlines_ev
= rb_event_add("expire_override_deadlines", expire_override_deadlines
, NULL
, 60);
281 rb_dlink_node
*n
, *tn
;
283 /* disable the umode and remove it from the available list */
285 construct_umodebuf();
287 RB_DLINK_FOREACH_SAFE(n
, tn
, overriding_opers
.head
)
289 rb_dlinkDelete(n
, &overriding_opers
);
293 rb_event_delete(expire_override_deadlines_ev
);
296 DECLARE_MODULE_AV2(override
, _modinit
, _moddeinit
, NULL
, NULL
,
297 override_hfnlist
, NULL
, NULL
, override_desc
);