2 * This is the first client module for newserv :)
4 * A very simple bot which should give people some ideas for how to
5 * implement stuff on this thing
8 #include "../irc/irc_config.h"
9 #include "../parser/parser.h"
10 #include "../localuser/localuser.h"
11 #include "../localuser/localuserchannel.h"
12 #include "../nick/nick.h"
13 #include "../lib/sstring.h"
14 #include "../core/config.h"
15 #include "../irc/irc.h"
16 #include "../lib/splitline.h"
17 #include "../channel/channel.h"
18 #include "../lib/flags.h"
19 #include "../core/schedule.h"
20 #include "../lib/base64.h"
21 #include "../core/modules.h"
22 #include "../lib/version.h"
23 #include "../lib/irc_string.h"
36 CommandTree
*controlcmds
;
37 ControlMsg controlreply
;
38 ControlWall controlwall
;
40 void handlemessages(nick
*target
, int messagetype
, void **args
);
41 int controlstatus(void *sender
, int cargc
, char **cargv
);
42 void controlconnect(void *arg
);
43 int controlwhois(void *sender
, int cargc
, char **cargv
);
44 int controlchannel(void *sender
, int cargc
, char **cargv
);
45 int relink(void *sender
, int cargc
, char **cargv
);
46 int die(void *sender
, int cargc
, char **cargv
);
47 int controlinsmod(void *sender
, int cargc
, char **cargv
);
48 int controllsmod(void *sender
, int cargc
, char **cargv
);
49 int controlrehash(void *sender
, int cargc
, char **cargv
);
50 int controlreload(void *sender
, int cargc
, char **cargv
);
51 int controlhelpcmd(void *sender
, int cargc
, char **cargv
);
52 void controlnoticeopers(flag_t permissionlevel
, flag_t noticelevel
, char *format
, ...) __attribute__ ((format (printf
, 3, 4)));
53 void controlnoticeopers(flag_t permissionlevel
, flag_t noticelevel
, char *format
, ...);
54 void handlesignal(int hooknum
, void *arg
);
57 controlcmds
=newcommandtree();
58 controlreply
=&controlmessage
;
59 controlwall
=&controlnoticeopers
;
61 registercontrolhelpcmd("status",NO_DEVELOPER
,1,&controlstatus
,"Usage: status ?level?\nDisplays status information, increasing level gives more verbose information.");
62 registercontrolhelpcmd("whois",NO_OPERED
,1,&controlwhois
,"Usage: whois <nickname|#numeric>\nDisplays lots of information about the specified nickname or numeric.");
63 registercontrolhelpcmd("channel",NO_OPER
,1,&controlchannel
,"Usage: channel <#channel>\nDisplays channel information.");
64 registercontrolhelpcmd("relink",NO_DEVELOPER
,1,&relink
,"Usage: relink\nRelinks service to the network.");
65 registercontrolhelpcmd("die",NO_DEVELOPER
,1,&die
,"Usage: die <reason>\nTerminates the service.");
66 registercontrolhelpcmd("insmod",NO_DEVELOPER
,1,&controlinsmod
,"Usage: insmod <module>\nAdds a module to the running instance.");
67 registercontrolhelpcmd("rmmod",NO_DEVELOPER
,1,&controlrmmod
,"Usage: rmmod <module>\nRemoves a module from the running instance.");
68 registercontrolhelpcmd("lsmod",NO_OPER
,0,&controllsmod
,"Usage: lsmod\nLists currently running modules.");
69 registercontrolhelpcmd("rehash",NO_DEVELOPER
,1,&controlrehash
,"Usage: rehash\nReloads configuration file.");
70 registercontrolhelpcmd("showcommands",NO_ACCOUNT
,0,&controlshowcommands
,"Usage: showcommands\nShows all registered commands.");
71 registercontrolhelpcmd("reload",NO_DEVELOPER
,1,&controlreload
,"Usage: reload <module>\nReloads specified module.");
72 registercontrolhelpcmd("help",NO_ANYONE
,1,&controlhelpcmd
,"Usage: help <command>\nShows help for specified command.");
74 registerhook(HOOK_CORE_REHASH
, &handlesignal
);
75 registerhook(HOOK_CORE_SIGINT
, &handlesignal
);
76 scheduleoneshot(time(NULL
)+1,&controlconnect
,NULL
);
80 deleteallschedules(&controlconnect
);
82 deregisterlocaluser(mynick
,"Leaving");
85 deregistercontrolcmd("status",&controlstatus
);
86 deregistercontrolcmd("whois",&controlwhois
);
87 deregistercontrolcmd("channel",&controlchannel
);
88 deregistercontrolcmd("relink",&relink
);
89 deregistercontrolcmd("die",&die
);
90 deregistercontrolcmd("insmod",&controlinsmod
);
91 deregistercontrolcmd("rmmod",&controlrmmod
);
92 deregistercontrolcmd("lsmod",&controllsmod
);
93 deregistercontrolcmd("rehash",&controlrehash
);
94 deregistercontrolcmd("showcommands",&controlshowcommands
);
95 deregistercontrolcmd("reload",&controlreload
);
96 deregistercontrolcmd("help",&controlhelpcmd
);
98 destroycommandtree(controlcmds
);
100 deregisterhook(HOOK_CORE_REHASH
, &handlesignal
);
101 deregisterhook(HOOK_CORE_SIGINT
, &handlesignal
);
104 void registercontrolhelpcmd(const char *name
, int level
, int maxparams
, CommandHandler handler
, char *help
) {
105 addcommandhelptotree(controlcmds
,name
,level
,maxparams
,handler
,help
);
108 int deregistercontrolcmd(const char *name
, CommandHandler handler
) {
109 return deletecommandfromtree(controlcmds
, name
, handler
);
112 void controlconnect(void *arg
) {
113 sstring
*cnick
, *myident
, *myhost
, *myrealname
, *myauthname
;
116 cnick
=getcopyconfigitem("control","nick","C",NICKLEN
);
117 myident
=getcopyconfigitem("control","ident","control",NICKLEN
);
118 myhost
=getcopyconfigitem("control","hostname",myserver
->content
,HOSTLEN
);
119 myrealname
=getcopyconfigitem("control","realname","NewServ Control Service",REALLEN
);
120 myauthname
=getcopyconfigitem("control","authname","C",ACCOUNTLEN
);
122 mynick
=registerlocaluser(cnick
->content
,myident
->content
,myhost
->content
,myrealname
->content
,myauthname
->content
,UMODE_SERVICE
|UMODE_DEAF
|UMODE_OPER
|UMODE_ACCOUNT
|UMODE_INV
,&handlemessages
);
123 triggerhook(HOOK_CONTROL_REGISTERED
, mynick
);
124 cp
=findchannel("#twilightzone");
126 localcreatechannel(mynick
,"#twilightzone");
128 localjoinchannel(mynick
,cp
);
129 localgetops(mynick
,cp
);
133 freesstring(myident
);
135 freesstring(myrealname
);
136 freesstring(myauthname
);
139 void handlestats(int hooknum
, void *arg
) {
140 controlreply(hooknick
,"%s",(char *)arg
);
143 int controlstatus(void *sender
, int cargc
, char **cargv
) {
144 unsigned long level
=999;
145 hooknick
=(nick
*)sender
;
148 level
=strtoul(cargv
[0],NULL
,10);
151 registerhook(HOOK_CORE_STATSREPLY
,&handlestats
);
153 triggerhook(HOOK_CORE_STATSREQUEST
,(void *)level
);
154 deregisterhook(HOOK_CORE_STATSREPLY
,&handlestats
);
158 int controlrehash(void *sender
, int cargc
, char **cargv
) {
159 nick
*np
=(nick
*)sender
;
161 controlreply(np
,"Rehashing the config file.");
164 triggerhook(HOOK_CORE_REHASH
,(void *)0);
168 void handlewhois(int hooknum
, void *arg
) {
169 controlreply(hooknick
,"%s",(char *)arg
);
172 int controlwhois(void *sender
, int cargc
, char **cargv
) {
181 if (cargv
[0][0]=='#') {
182 if (!(target
=getnickbynumericstr(cargv
[0]+1))) {
183 controlreply(sender
,"Sorry, couldn't find numeric %s",cargv
[0]+1);
187 if ((target
=getnickbynick(cargv
[0]))==NULL
) {
188 controlreply((nick
*)sender
,"Sorry, couldn't find that user");
193 controlreply((nick
*)sender
,"Nick : %s",target
->nick
);
194 controlreply((nick
*)sender
,"Numeric : %s",longtonumeric(target
->numeric
,5));
195 controlreply((nick
*)sender
,"User@Host : %s@%s (%d user(s) on this host)",target
->ident
,target
->host
->name
->content
,target
->host
->clonecount
);
196 if (IsSetHost(target
)) {
197 if (target
->shident
) {
198 controlreply((nick
*)sender
,"Fakehost : %s@%s",target
->shident
->content
, target
->sethost
->content
);
200 controlreply((nick
*)sender
,"Fakehost : %s",target
->sethost
->content
);
203 controlreply((nick
*)sender
,"Timestamp : %lu",target
->timestamp
);
204 controlreply((nick
*)sender
,"IP address: %s",IPtostr(target
->p_ipaddr
));
205 controlreply((nick
*)sender
,"Realname : %s (%d user(s) have this realname)",target
->realname
->name
->content
,target
->realname
->usercount
);
206 if (target
->umodes
) {
207 controlreply((nick
*)sender
,"Umode(s) : %s",printflags(target
->umodes
,umodeflags
));
209 if (IsOper(target
) && target
->opername
)
210 controlreply((nick
*)sender
,"Opered as : %s",target
->opername
->content
);
211 if (IsAccount(target
)) {
212 controlreply((nick
*)sender
,"Account : %s",target
->authname
);
213 if (target
->accountts
)
214 controlreply((nick
*)sender
,"AccountTS : %ld",target
->accountts
);
216 controlreply((nick
*)sender
,"UserID : %ld",target
->auth
->userid
);
217 if (target
->auth
->flags
)
218 controlreply((nick
*)sender
,"AccFlags : %s",printflags(target
->auth
->flags
,accountflags
));
222 hooknick
=(nick
*)sender
;
223 registerhook(HOOK_CONTROL_WHOISREPLY
,&handlewhois
);
224 triggerhook(HOOK_CONTROL_WHOISREQUEST
,target
);
225 deregisterhook(HOOK_CONTROL_WHOISREPLY
,&handlewhois
);
227 if (target
->channels
->cursi
==0) {
228 controlreply((nick
*)sender
,"Channels : none");
229 } else if (target
->channels
->cursi
>50) {
230 controlreply((nick
*)sender
,"Channels : - (total: %d)",target
->channels
->cursi
);
233 channels
=(channel
**)target
->channels
->content
;
234 for (i
=0;i
<=target
->channels
->cursi
;i
++) {
235 if (!((i
==target
->channels
->cursi
) || ((70-strlen(buf
))<channels
[i
]->index
->name
->length
&& strlen(buf
)>0))) {
236 strcat(buf
,channels
[i
]->index
->name
->content
);
239 if (strlen(buf
)==0) {
242 controlreply((nick
*)sender
,"Channels : %s",buf
);
253 int controlinsmod(void *sender
, int cargc
, char **cargv
) {
257 switch(insmod(cargv
[0])) {
259 controlreply((nick
*)sender
,"Unable to load module %s",cargv
[0]);
263 controlreply((nick
*)sender
,"Module %s already loaded, or name not valid",cargv
[0]);
267 controlreply((nick
*)sender
,"Module %s loaded.",cargv
[0]);
271 controlreply((nick
*)sender
,"An unknown error occured.");
276 int controlrmmod(void *sender
, int cargc
, char **cargv
) {
280 switch(rmmod(cargv
[0])) {
282 controlreply((nick
*)sender
,"Module %s is not loaded.",cargv
[0]);
286 controlreply((nick
*)sender
,"Module %s unloaded.",cargv
[0]);
290 controlreply((nick
*)sender
,"An unknown error occured.");
295 int controllsmod(void *sender
, int cargc
, char **cargv
) {
299 if (cargc
< 1) { /* list all loaded modules */
300 const char *ver
, *buildid
;
301 time_t t
, t2
= time(NULL
);
302 ptr
= lsmod(i
, &ver
, &buildid
, &t
);
304 /* 9999d 24h 59m 59s fbf2a4a69ee1-tip */
305 controlreply((nick
*)sender
,"Module Loaded for Version Build id");
306 while (ptr
!= NULL
) {
307 controlreply((nick
*)sender
," %-40s %-17s %-30s %s", ptr
, longtoduration(t2
-t
, 2), ver
?ver
:"", buildid
?buildid
:"");
308 ptr
= lsmod(++i
, &ver
, &buildid
, &t
);
311 ptr
= lsmod(getindex(cargv
[0]), NULL
, NULL
, NULL
);
312 controlreply((nick
*)sender
,"Module \"%s\" %s", cargv
[0], (ptr
? "is loaded." : "is NOT loaded."));
317 int controlreload(void *sender
, int cargc
, char **cargv
) {
321 controlreply((nick
*)sender
,"Imma gonna try and reload %s",cargv
[0]);
323 safereload(cargv
[0]);
328 int relink(void *sender
, int cargc
, char **cargv
) {
330 controlreply((nick
*)sender
,"You must give a reason.");
334 irc_send("%s SQ %s 0 :%s",mynumeric
->content
,myserver
->content
,cargv
[0]);
340 int die(void *sender
, int cargc
, char **cargv
) {
341 if (cargc
<1 || (strlen(cargv
[0]) < 10)) {
342 controlreply((nick
*)sender
,"You must give a reason.");
346 controlwall(NO_OPER
,NL_OPERATIONS
,"DIE from %s: %s",((nick
*)sender
)->nick
, cargv
[0]);
348 newserv_shutdown_pending
=1;
353 int controlchannel(void *sender
, int cargc
, char **cargv
) {
359 int i
,j
, ops
=0, voice
=0;
365 if ((cp
=findchannel(cargv
[0]))==NULL
) {
366 controlreply((nick
*)sender
,"Couldn't find channel: %s",cargv
[0]);
371 sprintf(buf2
,"%d",cp
->limit
);
374 controlreply((nick
*)sender
,"Channel : %s",cp
->index
->name
->content
);
375 strftime(timebuf
, 30, "%d/%m/%y %H:%M", localtime(&(cp
->timestamp
)));
376 controlreply((nick
*)sender
,"C-time : %ld [%s]",cp
->timestamp
,timebuf
);
378 controlreply((nick
*)sender
,"Topic : %s",cp
->topic
->content
);
379 strftime(timebuf
, 30, "%d/%m/%y %H:%M", localtime(&(cp
->topictime
)));
380 controlreply((nick
*)sender
,"T-time : %ld [%s]",cp
->topictime
,timebuf
);
382 controlreply((nick
*)sender
,"Topic : (none)");
384 controlreply((nick
*)sender
,"Mode(s) : %s %s%s%s",printflags(cp
->flags
,cmodeflags
),IsLimit(cp
)?buf2
:"",
385 IsLimit(cp
)?" ":"",IsKey(cp
)?cp
->key
->content
:"");
386 controlreply((nick
*)sender
,"Users : %d (hash size %d, utilisation %.1f%%); %d unique hosts",
387 cp
->users
->totalusers
,cp
->users
->hashsize
,((float)(100*cp
->users
->totalusers
)/cp
->users
->hashsize
),
388 countuniquehosts(cp
));
392 for (j
=0;j
<=cp
->users
->hashsize
;j
++) {
393 if (i
==4 || j
==cp
->users
->hashsize
) {
395 controlreply((nick
*)sender
,"Users : %s",buf
);
399 if (j
==cp
->users
->hashsize
)
402 if (cp
->users
->content
[j
]!=nouser
) {
403 if (cp
->users
->content
[j
]&CUMODE_OP
)
405 else if (cp
->users
->content
[j
]&CUMODE_VOICE
)
407 np
=getnickbynumeric(cp
->users
->content
[j
]);
408 sprintf(&buf
[i
*18],"%c%c%-15s ",cp
->users
->content
[j
]&CUMODE_VOICE
?'+':' ',
409 cp
->users
->content
[j
]&CUMODE_OP
?'@':' ', np
?np
->nick
:"!BUG-NONICK!");
415 controlreply((nick
*)sender
, "Users : Opped: %d, Voiced: %d", ops
,voice
);
416 for (cbp
=cp
->bans
;cbp
;cbp
=cbp
->next
) {
417 controlreply((nick
*)sender
,"Ban : %s",bantostringdebug(cbp
));
422 int controlshowcommands(void *sender
, int cargc
, char **cargv
) {
423 nick
*np
=(nick
*)sender
;
424 Command
*cmdlist
[100];
427 n
=getcommandlist(controlcmds
,cmdlist
,100);
429 controlreply(np
,"The following commands are registered at present:");
432 controlreply(np
,"%s",cmdlist
[i
]->command
->content
);
435 controlreply(np
,"End of list.");
439 void handlemessages(nick
*target
, int messagetype
, void **args
) {
445 switch(messagetype
) {
448 /* If it's a message, first arg is nick and second is message */
449 sender
=(nick
*)args
[0];
451 Error("control",ERR_INFO
,"From: %s!%s@%s: %s",sender
->nick
,sender
->ident
,sender
->host
->name
->content
, (char *)args
[1]);
453 /* Split the line into params */
454 cargc
=splitline((char *)args
[1],cargv
,50,0);
461 cmd
=findcommandintree(controlcmds
,cargv
[0],1);
463 controlreply(sender
,"Unknown command.");
467 if (cmd
->level
>0 && !IsOper(sender
)) {
468 controlreply(sender
,"You need to be opered to use this command.");
472 /* If we were doing "authed user tracking" here we'd put a check in for authlevel */
474 /* Check the maxargs */
475 if (cmd
->maxparams
<(cargc
-1)) {
476 /* We need to do some rejoining */
477 rejoinline(cargv
[cmd
->maxparams
],cargc
-(cmd
->maxparams
));
478 cargc
=(cmd
->maxparams
)+1;
481 if((cmd
->handler
)((void *)sender
,cargc
-1,&(cargv
[1])) == CMD_USAGE
)
482 controlhelp(sender
, cmd
);
486 /* someone killed me? Bastards */
487 scheduleoneshot(time(NULL
)+1,&controlconnect
,NULL
);
489 triggerhook(HOOK_CONTROL_REGISTERED
, NULL
);
497 void controlmessage(nick
*target
, char *message
, ... ) {
505 va_start(va
,message
);
506 vsnprintf(buf
,512,message
,va
);
509 sendmessagetouser(mynick
,target
,"%s",buf
);
512 void controlchanmsg(channel
*cp
, char *message
, ...) {
520 va_start(va
,message
);
521 vsnprintf(buf
,512,message
,va
);
524 sendmessagetochannel(mynick
,cp
,"%s",buf
);
527 void controlnotice(nick
*target
, char *message
, ... ) {
535 va_start(va
,message
);
536 vsnprintf(buf
,512,message
,va
);
539 sendnoticetouser(mynick
,target
,"%s",buf
);
542 void controlspecialrmmod(void *arg
) {
543 struct specialsched
*a
= (struct specialsched
*)arg
;
544 sstring
*froo
= a
->modulename
;
548 rmmod(froo
->content
);
552 void controlspecialreloadmod(void *arg
) {
553 struct specialsched
*a
= (struct specialsched
*)arg
;
554 sstring
*froo
= a
->modulename
;
558 safereload(froo
->content
);
562 void controlhelp(nick
*np
, Command
*cmd
) {
563 sstring
*scp
= cmd
->help
;
565 controlreply(np
, "Sorry, no help for this command.");
567 char *cp
= scp
->content
, *sp
= cp
;
570 if(*cp
== '\0' || *cp
== '\n') {
578 controlreply(np
, "%s", sp
);
591 int controlhelpcmd(void *sender
, int cargc
, char **cargv
) {
593 nick
*np
= (nick
*)sender
;
598 cmd
=findcommandintree(controlcmds
,cargv
[0],1);
600 controlreply(np
,"Unknown command.");
604 controlhelp(np
, cmd
);
608 void controlnoticeopers(flag_t permissionlevel
, flag_t noticelevel
, char *format
, ...) {
614 va_start(va
, format
);
615 vsnprintf(broadcast
, sizeof(broadcast
), format
, va
);
618 for(i
=0;i
<NICKHASHSIZE
;i
++)
619 for(np
=nicktable
[i
];np
;np
=np
->next
)
621 controlnotice(np
, "%s", broadcast
);
624 void controlnswall(int noticelevel
, char *format
, ...) {
628 va_start(va
, format
);
629 vsnprintf(broadcast
, sizeof(broadcast
), format
, va
);
632 controlwall(NO_OPER
, noticelevel
, "%s", broadcast
);
635 void handlesignal(int hooknum
, void *arg
) {
636 char *signal
, *action
;
638 if(hooknum
== HOOK_CORE_SIGINT
) {
640 action
= "terminating";
642 long hupped
= (long)arg
;
647 action
= "rehashing";
650 controlwall(NO_OPER
, NL_OPERATIONS
, "SIG%s received, %s...", signal
, action
);