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(struct 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(special
.schedule
) {
191 controlreply(np
, "Previous attempt at un/reload still in progress.");
194 special
.modulename
= getsstring(cargv
[0], strlen(cargv
[0]));
195 if(!special
.modulename
) {
196 controlreply(np
, "Unable to copy module name. Seek cow herd to trample on server.");
199 special
.schedule
= scheduleoneshot(time(NULL
) + 1, reloadhandler
, &special
);
200 if(!special
.schedule
) {
201 freesstring(special
.modulename
);
202 special
.modulename
= NULL
;
203 controlreply(np
, "Unable to allocate schedule. Seek cow herd to trample on server.");
206 controlreply(np
, "Special case un/reload in <1 second, no response will be sent, standby. . .");
212 return oldcommand(np
, cargc
, cargv
);
217 int noperserv_rmmod(void *sender
, int cargc
, char **cargv
) {
218 return noperserv_specialmod(sender
, "rmmod", &controlspecialrmmod
, cargc
, cargv
);
221 int noperserv_reload(void *sender
, int cargc
, char **cargv
) {
222 return noperserv_specialmod(sender
, "reload", &controlspecialreloadmod
, cargc
, cargv
);
225 void noperserv_whois_hook(int hooknum
, void *arg
) {
226 controlreply(replynick
, "%s", (char *)arg
);
229 int noperserv_whois(void *sender
, int cargc
, char **cargv
) {
231 nick
*np
= (nick
*)sender
;
232 CommandHandler oldwhois
= noperserv_find_hook("whois");
236 return oldwhois(sender
, cargc
, cargv
);
240 if(cargv
[0][0] != '#') {
241 if(cargv
[0][0] == '*')
244 return oldwhois(sender
, cargc
, cargv
);
248 au
= noperserv_get_autheduser(cargv
[0] + 1);
250 controlreply(np
, "Account not registered.");
254 controlreply(np
, "Account : %s", au
->authname
->content
);
258 registerhook(HOOK_CONTROL_WHOISREPLY
, &noperserv_whois_hook
);
259 noperserv_whois_account_handler(HOOK_CONTROL_WHOISREQUEST_AUTHEDUSER
, (void *)au
);
260 deregisterhook(HOOK_CONTROL_WHOISREPLY
, &noperserv_whois_hook
);
262 controlreply(np
, "Flags : %s", printflags(NOGetAuthLevel(au
), no_userflags
));
267 int noperserv_showcommands(void *sender
, int cargc
, char **cargv
) {
268 nick
*np
= (nick
*)sender
;
269 Command
*cmdlist
[100];
272 n
= getcommandlist(controlcmds
, cmdlist
, 100);
274 controlreply(np
, "The following commands are registered at present:");
277 if(noperserv_policy_command_permitted(cmdlist
[i
]->level
, np
))
278 controlreply(np
, "%s (%s)", cmdlist
[i
]->command
->content
, printflags(cmdlist
[i
]->level
, no_commandflags
));
280 controlreply(np
, "End of list.");
284 void noperserv_whois_handler(int hooknum
, void *arg
) {
286 nick
*np
= (nick
*)arg
;
292 au
= NOGetAuthedUser(np
);
294 snprintf(message
, sizeof(message
), "Flags : %s", printflags(NOGetAuthLevel(au
), no_userflags
));
295 noperserv_whois_account_handler(HOOK_CONTROL_WHOISREQUEST_AUTHEDUSER
, (void *)au
);
297 snprintf(message
, sizeof(message
), "Flags : (user not known)");
298 noperserv_whois_account_handler(HOOK_CONTROL_WHOISREQUEST_AUTHNAME
, (void *)np
->authname
);
300 triggerhook(HOOK_CONTROL_WHOISREPLY
, message
);
305 void noperserv_whois_account_handler(int hooknum
, void *arg
) {
306 int count
= 0, found
= 0;
307 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 */
308 char accountspace
[NICKLEN
+ 3]; /* space, comma, null */
311 nickbuffer
[0] = '\0';
312 if(hooknum
== HOOK_CONTROL_WHOISREQUEST_AUTHEDUSER
) {
313 /* we can just read out the authed user linked list */
314 no_autheduser
*au
= (void *)arg
;
315 no_nicklist
*nl
= au
->nick
;
320 for(;nl
;nl
=nl
->next
) {
321 snprintf(accountspace
, sizeof(accountspace
), "%s%s", count
++?", ":"", nl
->nick
->nick
);
322 strlcat(nickbuffer
, accountspace
, sizeof(nickbuffer
));
324 if(count
>= NO_NICKS_PER_WHOIS_LINE
) {
325 snprintf(message
, sizeof(message
), "Authed : %s", nickbuffer
);
326 triggerhook(HOOK_CONTROL_WHOISREPLY
, message
);
327 nickbuffer
[0] = '\0';
332 /* inefficient way */
333 char *authname
= (char *)arg
;
337 for(;i
<NICKHASHSIZE
;i
++)
338 for(sp
=nicktable
[i
];sp
;sp
=sp
->next
)
339 if(IsAccount(sp
) && !ircd_strcmp(sp
->authname
, authname
)) {
342 snprintf(accountspace
, sizeof(accountspace
), "%s%s", count
++?", ":"", sp
->nick
);
343 strlcat(nickbuffer
, accountspace
, sizeof(nickbuffer
));
345 if(count
>= NO_NICKS_PER_WHOIS_LINE
) {
346 snprintf(message
, sizeof(message
), "Authed : %s", nickbuffer
);
347 triggerhook(HOOK_CONTROL_WHOISREPLY
, message
);
348 nickbuffer
[0] = '\0';
355 snprintf(message
, sizeof(message
), "Authed : (no nicks authed)");
356 triggerhook(HOOK_CONTROL_WHOISREPLY
, message
);
357 } else if(nickbuffer
[0]) {
358 snprintf(message
, sizeof(message
), "Authed : %s", nickbuffer
);
359 triggerhook(HOOK_CONTROL_WHOISREPLY
, message
);
363 /* Obviously pinched from control.c */
364 void noperserv_handle_messages(nick
*target
, int messagetype
, void **args
) {
370 switch(messagetype
) {
371 case LU_PRIVMSG
: /* override these two commands only */
373 /* If it's a message, first arg is nick and second is message */
374 sender
= (nick
*)args
[0];
376 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]);
378 /* Split the line into params */
379 cargc
= splitline((char *)args
[1], cargv
, 50, 0);
381 if(!cargc
) /* Blank line */
384 cmd
= findcommandintree(controlcmds
,cargv
[0],1);
386 controlreply(sender
, "Unknown command.");
390 /* If we were doing "authed user tracking" here we'd put a check in for authlevel */
392 if (!noperserv_policy_command_permitted(cmd
->level
, sender
)) {
393 controlreply(sender
, "Access denied.");
397 /* Check the maxargs */
398 if(cmd
->maxparams
< (cargc
- 1)) {
399 /* We need to do some rejoining */
400 rejoinline(cargv
[cmd
->maxparams
], cargc
- (cmd
->maxparams
));
401 cargc
= (cmd
->maxparams
) + 1;
404 if((cmd
->handler
)((void *)sender
,cargc
-1,&(cargv
[1])) == CMD_USAGE
)
405 controlhelp(sender
, cmd
);
410 oldhandler(target
, messagetype
, args
);
415 void noperserv_reply(nick
*np
, char *format
, ...) {
418 no_autheduser
*au
= NOGetAuthedUser(np
);
420 va_start(va
, format
);
421 vsnprintf(buf
, sizeof(buf
), format
, va
);
424 if(au
&& !(NOGetNoticeLevel(au
) & NL_NOTICES
)) {
425 controlmessage(np
, "%s", buf
);
427 controlnotice(np
, "%s", buf
);
431 int noperserv_help(void *sender
, int cargc
, char **cargv
) {
433 nick
*np
= (nick
*)sender
;
438 cmd
= findcommandintree(controlcmds
, cargv
[0], 1);
440 controlreply(np
, "Unknown command.");
444 if(!noperserv_policy_command_permitted(cmd
->level
, np
)) {
445 controlreply(np
, "Access denied.");
449 controlhelp(np
, cmd
);
453 void noperserv_wall(flag_t permissionlevel
, flag_t noticelevel
, char *format
, ...) {
456 no_autheduser
*au
= authedusers
;
458 char *flags
= printflags(noticelevel
, no_noticeflags
) + 1;
460 va_start(va
, format
);
461 vsnprintf(buf
, sizeof(buf
), format
, va
);
464 Error("noperserv", ERR_INFO
, "$%s$ %s", flags
, buf
);
466 for(;au
;au
=au
->next
) {
467 if((NOGetNoticeLevel(au
) & noticelevel
) && !(NOGetAuthLevel(au
) & __NO_RELAY
)) {
468 for(nl
=au
->nick
;nl
;nl
=nl
->next
)
469 if(noperserv_policy_command_permitted(permissionlevel
, nl
->nick
))
470 controlreply(nl
->nick
, "$%s$ %s", flags
, buf
);