1 #include "../control/control.h"
2 #include "../localuser/localuser.h"
3 #include "../core/schedule.h"
4 #include "../core/modules.h"
5 #include "../lib/splitline.h"
6 #include "../lib/flags.h"
7 #include "../lib/irc_string.h"
8 #include "../lib/strlfunc.h"
10 #include "noperserv.h"
11 #include "noperserv_db.h"
12 #include "noperserv_hooks.h"
13 #include "noperserv_policy.h"
24 struct storedhook
*next
;
27 struct storedhook
*storedhooks
= NULL
;
30 nick
*replynick
= NULL
;
32 UserMessageHandler oldhandler
;
36 void noperserv_trap_registration(int hooknum
, void *arg
);
37 int noperserv_showcommands(void *sender
, int cargc
, char **cargv
);
38 int noperserv_rmmod(void *sender
, int cargc
, char **cargv
);
39 int noperserv_reload(void *sender
, int cargc
, char **cargv
);
40 int noperserv_whois(void *sender
, int cargc
, char **cargv
);
41 int noperserv_help(void *sender
, int cargc
, char **cargv
);
42 void noperserv_whois_handler(int hooknum
, void *arg
);
43 void noperserv_whois_account_handler(int hooknum
, void *arg
);
44 void noperserv_handle_messages(nick
*target
, int messagetype
, void **args
);
45 void noperserv_reply(nick
*np
, char *format
, ...);
46 void noperserv_wall(flag_t permissionlevel
, flag_t noticelevel
, char *format
, ...);
48 struct specialsched special
;
50 #define HOOK_CONTROL_WHOISREQUEST_AUTHNAME -1
51 #define HOOK_CONTROL_WHOISREQUEST_AUTHEDUSER -2
53 void noperserv_setup_hooks(void) {
54 oldreply
= controlreply
;
55 controlreply
= &noperserv_reply
;
57 oldwall
= controlwall
;
58 controlwall
= &noperserv_wall
;
60 memset(&special
, 0, sizeof(specialsched
));
63 registerhook(HOOK_CONTROL_REGISTERED
, &noperserv_trap_registration
);
65 noperserv_trap_registration(0, (void *)mynick
);
68 registerhook(HOOK_CONTROL_WHOISREQUEST
, &noperserv_whois_handler
);
71 int noperserv_hook_command(char *command
, CommandHandler newcommand
, char *newhelp
) {
72 struct storedhook
*newhook
;
73 Command
*fetchcommand
= findcommandintree(controlcmds
, command
, 1);
78 newhook
= (struct storedhook
*)malloc(sizeof(struct storedhook
));
82 newhook
->name
= getsstring(command
, strlen(command
));
88 newhook
->old
= fetchcommand
->handler
;
90 int len
= strlen(newhelp
) + 1;
91 newhook
->newhelp
= (char *)malloc(len
);
92 if(!newhook
->newhelp
) {
93 freesstring(newhook
->name
);
96 strlcpy(newhook
->newhelp
, newhelp
, len
);
97 newhook
->oldhelp
= fetchcommand
->help
;
98 fetchcommand
->help
= newhook
->newhelp
;
100 newhook
->newhelp
= NULL
;
103 newhook
->next
= storedhooks
;
104 storedhooks
= newhook
;
106 fetchcommand
->handler
= newcommand
;
111 void noperserv_unhook_all_commands(void) {
112 struct storedhook
*nh
, *ch
= storedhooks
;
113 Command
*fetchcommand
;
116 if(ch
->old
&& (fetchcommand
= findcommandintree(controlcmds
, ch
->name
->content
, 1))) {
117 fetchcommand
->handler
= ch
->old
;
119 fetchcommand
->help
= ch
->oldhelp
;
124 freesstring(ch
->name
);
130 void noperserv_cleanup_hooks(void) {
131 deregisterhook(HOOK_CONTROL_WHOISREQUEST
, &noperserv_whois_handler
);
132 deregisterhook(HOOK_CONTROL_REGISTERED
, &noperserv_trap_registration
);
135 noperserv_unhook_all_commands();
140 hooklocaluserhandler(mynick
, oldhandler
);
142 controlwall
= oldwall
;
143 controlreply
= oldreply
;
146 void noperserv_trap_registration(int hooknum
, void *arg
) {
147 oldhandler
= hooklocaluserhandler((nick
*)arg
, &noperserv_handle_messages
);
153 noperserv_hook_command("rmmod", &noperserv_rmmod
, NULL
);
154 noperserv_hook_command("reload", &noperserv_reload
, NULL
);
155 noperserv_hook_command("showcommands", &noperserv_showcommands
, NULL
);
156 noperserv_hook_command("whois", &noperserv_whois
, "Usage: whois <nickname|#authname|*numeric>\nDisplays lots of information about the specified nickname, auth name or numeric.");
157 noperserv_hook_command("help", &noperserv_help
, NULL
);
161 CommandHandler
noperserv_find_hook(char *command
) {
162 struct storedhook
*hh
= storedhooks
;
164 if(!ircd_strcmp(hh
->name
->content
, command
))
170 int noperserv_modules_loaded(char *mask
) {
174 for(i
=0,ptr
=lsmod(i
);ptr
;ptr
=lsmod(++i
))
175 if(match2strings(mask
, ptr
))
181 int noperserv_specialmod(nick
*np
, char *command
, ScheduleCallback reloadhandler
, int cargc
, char **cargv
) {
182 CommandHandler oldcommand
= noperserv_find_hook(command
);
185 return oldcommand(np
, cargc
, cargv
);
189 if(!strcmp(cargv
[0], "noperserv")) {
190 if(noperserv_modules_loaded("noperserv_*")) {
191 controlreply(np
, "NOT UNLOADING. Unload all dependencies first.");
194 if(special
.schedule
) {
195 controlreply(np
, "Previous attempt at un/reload still in progress.");
198 special
.modulename
= getsstring(cargv
[0], strlen(cargv
[0]));
199 if(!special
.modulename
) {
200 controlreply(np
, "Unable to copy module name. Seek cow herd to trample on server.");
203 special
.schedule
= scheduleoneshot(time(NULL
) + 1, reloadhandler
, &special
);
204 if(!special
.schedule
) {
205 freesstring(special
.modulename
);
206 special
.modulename
= NULL
;
207 controlreply(np
, "Unable to allocate schedule. Seek cow herd to trample on server.");
210 controlreply(np
, "Special case un/reload in <1 second, no response will be sent, standby. . .");
216 return oldcommand(np
, cargc
, cargv
);
221 int noperserv_rmmod(void *sender
, int cargc
, char **cargv
) {
222 return noperserv_specialmod(sender
, "rmmod", &controlspecialrmmod
, cargc
, cargv
);
225 int noperserv_reload(void *sender
, int cargc
, char **cargv
) {
226 return noperserv_specialmod(sender
, "reload", &controlspecialreloadmod
, cargc
, cargv
);
229 void noperserv_whois_hook(int hooknum
, void *arg
) {
230 controlreply(replynick
, "%s", (char *)arg
);
233 int noperserv_whois(void *sender
, int cargc
, char **cargv
) {
235 nick
*np
= (nick
*)sender
;
236 CommandHandler oldwhois
= noperserv_find_hook("whois");
240 return oldwhois(sender
, cargc
, cargv
);
244 if(cargv
[0][0] != '#') {
245 if(cargv
[0][0] == '*')
248 return oldwhois(sender
, cargc
, cargv
);
252 au
= noperserv_get_autheduser(cargv
[0] + 1);
254 controlreply(np
, "Account not registered.");
258 controlreply(np
, "Account : %s", au
->authname
->content
);
262 registerhook(HOOK_CONTROL_WHOISREPLY
, &noperserv_whois_hook
);
263 noperserv_whois_account_handler(HOOK_CONTROL_WHOISREQUEST_AUTHEDUSER
, (void *)au
);
264 deregisterhook(HOOK_CONTROL_WHOISREPLY
, &noperserv_whois_hook
);
266 controlreply(np
, "Flags : %s", printflags(NOGetAuthLevel(au
), no_userflags
));
271 int noperserv_showcommands(void *sender
, int cargc
, char **cargv
) {
272 nick
*np
= (nick
*)sender
;
273 Command
*cmdlist
[100];
276 n
= getcommandlist(controlcmds
, cmdlist
, 100);
278 controlreply(np
, "The following commands are registered at present:");
281 if(noperserv_policy_command_permitted(cmdlist
[i
]->level
, np
))
282 controlreply(np
, "%s (%s)", cmdlist
[i
]->command
->content
, printflags(cmdlist
[i
]->level
, no_commandflags
));
284 controlreply(np
, "End of list.");
288 void noperserv_whois_handler(int hooknum
, void *arg
) {
290 nick
*np
= (nick
*)arg
;
296 au
= NOGetAuthedUser(np
);
298 snprintf(message
, sizeof(message
), "Flags : %s", printflags(NOGetAuthLevel(au
), no_userflags
));
299 noperserv_whois_account_handler(HOOK_CONTROL_WHOISREQUEST_AUTHEDUSER
, (void *)au
);
301 snprintf(message
, sizeof(message
), "Flags : (user not known)");
302 noperserv_whois_account_handler(HOOK_CONTROL_WHOISREQUEST_AUTHNAME
, (void *)np
->authname
);
304 triggerhook(HOOK_CONTROL_WHOISREPLY
, message
);
309 void noperserv_whois_account_handler(int hooknum
, void *arg
) {
310 int count
= 0, found
= 0;
311 char nickbuffer
[(NICKLEN
+ 2) * NO_NICKS_PER_WHOIS_LINE
- 1]; /* since we don't need space or comma for the first item we're fine NULL wise */
312 char accountspace
[NICKLEN
+ 3]; /* space, comma, null */
315 nickbuffer
[0] = '\0';
316 if(hooknum
== HOOK_CONTROL_WHOISREQUEST_AUTHEDUSER
) {
317 /* we can just read out the authed user linked list */
318 no_autheduser
*au
= (void *)arg
;
319 no_nicklist
*nl
= au
->nick
;
324 for(;nl
;nl
=nl
->next
) {
325 snprintf(accountspace
, sizeof(accountspace
), "%s%s", count
++?", ":"", nl
->nick
->nick
);
326 strlcat(nickbuffer
, accountspace
, sizeof(nickbuffer
));
328 if(count
>= NO_NICKS_PER_WHOIS_LINE
) {
329 snprintf(message
, sizeof(message
), "Authed : %s", nickbuffer
);
330 triggerhook(HOOK_CONTROL_WHOISREPLY
, message
);
331 nickbuffer
[0] = '\0';
336 /* inefficient way */
337 char *authname
= (char *)arg
;
341 for(;i
<NICKHASHSIZE
;i
++)
342 for(sp
=nicktable
[i
];sp
;sp
=sp
->next
)
343 if(IsAccount(sp
) && !ircd_strcmp(sp
->authname
, authname
)) {
346 snprintf(accountspace
, sizeof(accountspace
), "%s%s", count
++?", ":"", sp
->nick
);
347 strlcat(nickbuffer
, accountspace
, sizeof(nickbuffer
));
349 if(count
>= NO_NICKS_PER_WHOIS_LINE
) {
350 snprintf(message
, sizeof(message
), "Authed : %s", nickbuffer
);
351 triggerhook(HOOK_CONTROL_WHOISREPLY
, message
);
352 nickbuffer
[0] = '\0';
359 snprintf(message
, sizeof(message
), "Authed : (no nicks authed)");
360 triggerhook(HOOK_CONTROL_WHOISREPLY
, message
);
361 } else if(nickbuffer
[0]) {
362 snprintf(message
, sizeof(message
), "Authed : %s", nickbuffer
);
363 triggerhook(HOOK_CONTROL_WHOISREPLY
, message
);
367 /* Obviously pinched from control.c */
368 void noperserv_handle_messages(nick
*target
, int messagetype
, void **args
) {
374 switch(messagetype
) {
375 case LU_PRIVMSG
: /* override these two commands only */
377 /* If it's a message, first arg is nick and second is message */
378 sender
= (nick
*)args
[0];
380 controlwall(NO_DEVELOPER
, NL_ALL_COMMANDS
, "From: %s!%s@%s%s%s: %s", sender
->nick
, sender
->ident
, sender
->host
->name
->content
, IsAccount(sender
)?"/":"", IsAccount(sender
)?sender
->authname
:"", (char *)args
[1]);
382 /* Split the line into params */
383 cargc
= splitline((char *)args
[1], cargv
, 50, 0);
385 if(!cargc
) /* Blank line */
388 cmd
= findcommandintree(controlcmds
,cargv
[0],1);
390 controlreply(sender
, "Unknown command.");
394 /* If we were doing "authed user tracking" here we'd put a check in for authlevel */
396 if (!noperserv_policy_command_permitted(cmd
->level
, sender
)) {
397 controlreply(sender
, "Access denied.");
401 /* Check the maxargs */
402 if(cmd
->maxparams
< (cargc
- 1)) {
403 /* We need to do some rejoining */
404 rejoinline(cargv
[cmd
->maxparams
], cargc
- (cmd
->maxparams
));
405 cargc
= (cmd
->maxparams
) + 1;
408 if((cmd
->handler
)((void *)sender
,cargc
-1,&(cargv
[1])) == CMD_USAGE
)
409 controlhelp(sender
, cmd
);
414 oldhandler(target
, messagetype
, args
);
419 void noperserv_reply(nick
*np
, char *format
, ...) {
422 no_autheduser
*au
= NOGetAuthedUser(np
);
424 va_start(va
, format
);
425 vsnprintf(buf
, sizeof(buf
), format
, va
);
428 if(au
&& !(NOGetNoticeLevel(au
) & NL_NOTICES
)) {
429 controlmessage(np
, "%s", buf
);
431 controlnotice(np
, "%s", buf
);
435 int noperserv_help(void *sender
, int cargc
, char **cargv
) {
437 nick
*np
= (nick
*)sender
;
442 cmd
= findcommandintree(controlcmds
, cargv
[0], 1);
444 controlreply(np
, "Unknown command.");
448 if(!noperserv_policy_command_permitted(cmd
->level
, np
)) {
449 controlreply(np
, "Access denied.");
453 controlhelp(np
, cmd
);
457 void noperserv_wall(flag_t permissionlevel
, flag_t noticelevel
, char *format
, ...) {
460 no_autheduser
*au
= authedusers
;
462 char *flags
= printflags(noticelevel
, no_noticeflags
) + 1;
464 va_start(va
, format
);
465 vsnprintf(buf
, sizeof(buf
), format
, va
);
468 Error("noperserv", ERR_INFO
, "$%s$ %s", flags
, buf
);
470 for(;au
;au
=au
->next
) {
471 if(NOGetNoticeLevel(au
) & noticelevel
) {
472 for(nl
=au
->nick
;nl
;nl
=nl
->next
)
473 if(noperserv_policy_command_permitted(permissionlevel
, nl
->nick
))
474 controlreply(nl
->nick
, "$%s$ %s", flags
, buf
);