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
, ...);
53 void handlerehash(int hooknum
, void *arg
);
56 controlcmds
=newcommandtree();
57 controlreply
=&controlmessage
;
58 controlwall
=&controlnoticeopers
;
60 registercontrolhelpcmd("status",NO_DEVELOPER
,1,&controlstatus
,"Usage: status ?level?\nDisplays status information, increasing level gives more verbose information.");
61 registercontrolhelpcmd("whois",NO_OPERED
,1,&controlwhois
,"Usage: whois <nickname|#numeric>\nDisplays lots of information about the specified nickname or numeric.");
62 registercontrolhelpcmd("channel",NO_OPER
,1,&controlchannel
,"Usage: channel <#channel>\nDisplays channel information.");
63 registercontrolhelpcmd("relink",NO_DEVELOPER
,1,&relink
,"Usage: relink\nRelinks service to the network.");
64 registercontrolhelpcmd("die",NO_DEVELOPER
,1,&die
,"Usage: die <reason>\nTerminates the service.");
65 registercontrolhelpcmd("insmod",NO_DEVELOPER
,1,&controlinsmod
,"Usage: insmod <module>\nAdds a module to the running instance.");
66 registercontrolhelpcmd("rmmod",NO_DEVELOPER
,1,&controlrmmod
,"Usage: rmmod <module>\nRemoves a module from the running instance.");
67 registercontrolhelpcmd("lsmod",NO_OPER
,0,&controllsmod
,"Usage: lsmod\nLists currently running modules.");
68 registercontrolhelpcmd("rehash",NO_DEVELOPER
,1,&controlrehash
,"Usage: rehash\nReloads configuration file.");
69 registercontrolhelpcmd("showcommands",NO_ACCOUNT
,0,&controlshowcommands
,"Usage: showcommands\nShows all registered commands.");
70 registercontrolhelpcmd("reload",NO_DEVELOPER
,1,&controlreload
,"Usage: reload <module>\nReloads specified module.");
71 registercontrolhelpcmd("help",NO_ANYONE
,1,&controlhelpcmd
,"Usage: help <command>\nShows help for specified command.");
73 registerhook(HOOK_CORE_REHASH
, &handlerehash
);
74 scheduleoneshot(time(NULL
)+1,&controlconnect
,NULL
);
78 deleteallschedules(&controlconnect
);
80 deregisterlocaluser(mynick
,"Leaving");
83 deregistercontrolcmd("status",&controlstatus
);
84 deregistercontrolcmd("whois",&controlwhois
);
85 deregistercontrolcmd("channel",&controlchannel
);
86 deregistercontrolcmd("relink",&relink
);
87 deregistercontrolcmd("die",&die
);
88 deregistercontrolcmd("insmod",&controlinsmod
);
89 deregistercontrolcmd("rmmod",&controlrmmod
);
90 deregistercontrolcmd("lsmod",&controllsmod
);
91 deregistercontrolcmd("rehash",&controlrehash
);
92 deregistercontrolcmd("showcommands",&controlshowcommands
);
93 deregistercontrolcmd("reload",&controlreload
);
94 deregistercontrolcmd("help",&controlhelpcmd
);
96 destroycommandtree(controlcmds
);
98 deregisterhook(HOOK_CORE_REHASH
, &handlerehash
);
101 void registercontrolhelpcmd(const char *name
, int level
, int maxparams
, CommandHandler handler
, char *help
) {
102 addcommandhelptotree(controlcmds
,name
,level
,maxparams
,handler
,help
);
105 int deregistercontrolcmd(const char *name
, CommandHandler handler
) {
106 return deletecommandfromtree(controlcmds
, name
, handler
);
109 void controlconnect(void *arg
) {
110 sstring
*cnick
, *myident
, *myhost
, *myrealname
, *myauthname
;
113 cnick
=getcopyconfigitem("control","nick","C",NICKLEN
);
114 myident
=getcopyconfigitem("control","ident","control",NICKLEN
);
115 myhost
=getcopyconfigitem("control","hostname",myserver
->content
,HOSTLEN
);
116 myrealname
=getcopyconfigitem("control","realname","NewServ Control Service",REALLEN
);
117 myauthname
=getcopyconfigitem("control","authname","C",ACCOUNTLEN
);
119 mynick
=registerlocaluser(cnick
->content
,myident
->content
,myhost
->content
,myrealname
->content
,myauthname
->content
,UMODE_SERVICE
|UMODE_DEAF
|UMODE_OPER
|UMODE_ACCOUNT
|UMODE_INV
,&handlemessages
);
120 triggerhook(HOOK_CONTROL_REGISTERED
, mynick
);
121 cp
=findchannel("#twilightzone");
123 localcreatechannel(mynick
,"#twilightzone");
125 localjoinchannel(mynick
,cp
);
126 localgetops(mynick
,cp
);
130 freesstring(myident
);
132 freesstring(myrealname
);
133 freesstring(myauthname
);
136 void handlestats(int hooknum
, void *arg
) {
137 controlreply(hooknick
,"%s",(char *)arg
);
140 int controlstatus(void *sender
, int cargc
, char **cargv
) {
141 unsigned long level
=999;
142 hooknick
=(nick
*)sender
;
145 level
=strtoul(cargv
[0],NULL
,10);
148 registerhook(HOOK_CORE_STATSREPLY
,&handlestats
);
150 triggerhook(HOOK_CORE_STATSREQUEST
,(void *)level
);
151 deregisterhook(HOOK_CORE_STATSREPLY
,&handlestats
);
155 int controlrehash(void *sender
, int cargc
, char **cargv
) {
156 nick
*np
=(nick
*)sender
;
158 controlreply(np
,"Rehashing the config file.");
161 triggerhook(HOOK_CORE_REHASH
,(void *)0);
165 void handlewhois(int hooknum
, void *arg
) {
166 controlreply(hooknick
,"%s",(char *)arg
);
169 int controlwhois(void *sender
, int cargc
, char **cargv
) {
178 if (cargv
[0][0]=='#') {
179 if (!(target
=getnickbynumericstr(cargv
[0]+1))) {
180 controlreply(sender
,"Sorry, couldn't find numeric %s",cargv
[0]+1);
184 if ((target
=getnickbynick(cargv
[0]))==NULL
) {
185 controlreply((nick
*)sender
,"Sorry, couldn't find that user");
190 controlreply((nick
*)sender
,"Nick : %s",target
->nick
);
191 controlreply((nick
*)sender
,"Numeric : %s",longtonumeric(target
->numeric
,5));
192 controlreply((nick
*)sender
,"User@Host : %s@%s (%d user(s) on this host)",target
->ident
,target
->host
->name
->content
,target
->host
->clonecount
);
193 if (IsSetHost(target
)) {
194 if (target
->shident
) {
195 controlreply((nick
*)sender
,"Fakehost : %s@%s",target
->shident
->content
, target
->sethost
->content
);
197 controlreply((nick
*)sender
,"Fakehost : %s",target
->sethost
->content
);
200 controlreply((nick
*)sender
,"Timestamp : %lu",target
->timestamp
);
201 controlreply((nick
*)sender
,"IP address: %s",IPtostr(target
->p_ipaddr
));
202 controlreply((nick
*)sender
,"Realname : %s (%d user(s) have this realname)",target
->realname
->name
->content
,target
->realname
->usercount
);
203 if (target
->umodes
) {
204 controlreply((nick
*)sender
,"Umode(s) : %s",printflags(target
->umodes
,umodeflags
));
206 if (IsOper(target
) && target
->opername
)
207 controlreply((nick
*)sender
,"Opered as : %s",target
->opername
->content
);
208 if (IsAccount(target
)) {
209 controlreply((nick
*)sender
,"Account : %s",target
->authname
);
210 if (target
->accountts
)
211 controlreply((nick
*)sender
,"AccountTS : %ld",target
->accountts
);
213 controlreply((nick
*)sender
,"UserID : %ld",target
->auth
->userid
);
214 if (target
->auth
->flags
)
215 controlreply((nick
*)sender
,"AccFlags : %s",printflags(target
->auth
->flags
,accountflags
));
219 hooknick
=(nick
*)sender
;
220 registerhook(HOOK_CONTROL_WHOISREPLY
,&handlewhois
);
221 triggerhook(HOOK_CONTROL_WHOISREQUEST
,target
);
222 deregisterhook(HOOK_CONTROL_WHOISREPLY
,&handlewhois
);
224 if (target
->channels
->cursi
==0) {
225 controlreply((nick
*)sender
,"Channels : none");
226 } else if (target
->channels
->cursi
>50) {
227 controlreply((nick
*)sender
,"Channels : - (total: %d)",target
->channels
->cursi
);
230 channels
=(channel
**)target
->channels
->content
;
231 for (i
=0;i
<=target
->channels
->cursi
;i
++) {
232 if (!((i
==target
->channels
->cursi
) || ((70-strlen(buf
))<channels
[i
]->index
->name
->length
&& strlen(buf
)>0))) {
233 strcat(buf
,channels
[i
]->index
->name
->content
);
236 if (strlen(buf
)==0) {
239 controlreply((nick
*)sender
,"Channels : %s",buf
);
250 int controlinsmod(void *sender
, int cargc
, char **cargv
) {
254 switch(insmod(cargv
[0])) {
256 controlreply((nick
*)sender
,"Unable to load module %s",cargv
[0]);
260 controlreply((nick
*)sender
,"Module %s already loaded, or name not valid",cargv
[0]);
264 controlreply((nick
*)sender
,"Module %s loaded.",cargv
[0]);
268 controlreply((nick
*)sender
,"An unknown error occured.");
273 int controlrmmod(void *sender
, int cargc
, char **cargv
) {
277 switch(rmmod(cargv
[0])) {
279 controlreply((nick
*)sender
,"Module %s is not loaded.",cargv
[0]);
283 controlreply((nick
*)sender
,"Module %s unloaded.",cargv
[0]);
287 controlreply((nick
*)sender
,"An unknown error occured.");
292 int controllsmod(void *sender
, int cargc
, char **cargv
) {
296 if (cargc
< 1) { /* list all loaded modules */
297 const char *ver
, *buildid
;
298 time_t t
, t2
= time(NULL
);
299 ptr
= lsmod(i
, &ver
, &buildid
, &t
);
301 /* 9999d 24h 59m 59s fbf2a4a69ee1-tip */
302 controlreply((nick
*)sender
,"Module Loaded for Version Build id");
303 while (ptr
!= NULL
) {
304 controlreply((nick
*)sender
," %-40s %-17s %-30s %s", ptr
, longtoduration(t2
-t
, 2), ver
?ver
:"", buildid
?buildid
:"");
305 ptr
= lsmod(++i
, &ver
, &buildid
, &t
);
308 ptr
= lsmod(getindex(cargv
[0]), NULL
, NULL
, NULL
);
309 controlreply((nick
*)sender
,"Module \"%s\" %s", cargv
[0], (ptr
? "is loaded." : "is NOT loaded."));
314 int controlreload(void *sender
, int cargc
, char **cargv
) {
318 controlreply((nick
*)sender
,"Imma gonna try and reload %s",cargv
[0]);
320 safereload(cargv
[0]);
325 int relink(void *sender
, int cargc
, char **cargv
) {
327 controlreply((nick
*)sender
,"You must give a reason.");
331 irc_send("%s SQ %s 0 :%s",mynumeric
->content
,myserver
->content
,cargv
[0]);
337 int die(void *sender
, int cargc
, char **cargv
) {
338 if (cargc
<1 || (strlen(cargv
[0]) < 10)) {
339 controlreply((nick
*)sender
,"You must give a reason.");
343 controlwall(NO_OPER
,NL_OPERATIONS
,"DIE from %s: %s",((nick
*)sender
)->nick
, cargv
[0]);
345 newserv_shutdown_pending
=1;
350 int controlchannel(void *sender
, int cargc
, char **cargv
) {
356 int i
,j
, ops
=0, voice
=0;
362 if ((cp
=findchannel(cargv
[0]))==NULL
) {
363 controlreply((nick
*)sender
,"Couldn't find channel: %s",cargv
[0]);
368 sprintf(buf2
,"%d",cp
->limit
);
371 controlreply((nick
*)sender
,"Channel : %s",cp
->index
->name
->content
);
372 strftime(timebuf
, 30, "%d/%m/%y %H:%M", localtime(&(cp
->timestamp
)));
373 controlreply((nick
*)sender
,"C-time : %ld [%s]",cp
->timestamp
,timebuf
);
375 controlreply((nick
*)sender
,"Topic : %s",cp
->topic
->content
);
376 strftime(timebuf
, 30, "%d/%m/%y %H:%M", localtime(&(cp
->topictime
)));
377 controlreply((nick
*)sender
,"T-time : %ld [%s]",cp
->topictime
,timebuf
);
379 controlreply((nick
*)sender
,"Topic : (none)");
381 controlreply((nick
*)sender
,"Mode(s) : %s %s%s%s",printflags(cp
->flags
,cmodeflags
),IsLimit(cp
)?buf2
:"",
382 IsLimit(cp
)?" ":"",IsKey(cp
)?cp
->key
->content
:"");
383 controlreply((nick
*)sender
,"Users : %d (hash size %d, utilisation %.1f%%); %d unique hosts",
384 cp
->users
->totalusers
,cp
->users
->hashsize
,((float)(100*cp
->users
->totalusers
)/cp
->users
->hashsize
),
385 countuniquehosts(cp
));
389 for (j
=0;j
<=cp
->users
->hashsize
;j
++) {
390 if (i
==4 || j
==cp
->users
->hashsize
) {
392 controlreply((nick
*)sender
,"Users : %s",buf
);
396 if (j
==cp
->users
->hashsize
)
399 if (cp
->users
->content
[j
]!=nouser
) {
400 if (cp
->users
->content
[j
]&CUMODE_OP
)
402 else if (cp
->users
->content
[j
]&CUMODE_VOICE
)
404 np
=getnickbynumeric(cp
->users
->content
[j
]);
405 sprintf(&buf
[i
*18],"%c%c%-15s ",cp
->users
->content
[j
]&CUMODE_VOICE
?'+':' ',
406 cp
->users
->content
[j
]&CUMODE_OP
?'@':' ', np
?np
->nick
:"!BUG-NONICK!");
412 controlreply((nick
*)sender
, "Users : Opped: %d, Voiced: %d", ops
,voice
);
413 for (cbp
=cp
->bans
;cbp
;cbp
=cbp
->next
) {
414 controlreply((nick
*)sender
,"Ban : %s",bantostringdebug(cbp
));
419 int controlshowcommands(void *sender
, int cargc
, char **cargv
) {
420 nick
*np
=(nick
*)sender
;
421 Command
*cmdlist
[100];
424 n
=getcommandlist(controlcmds
,cmdlist
,100);
426 controlreply(np
,"The following commands are registered at present:");
429 controlreply(np
,"%s",cmdlist
[i
]->command
->content
);
432 controlreply(np
,"End of list.");
436 void handlemessages(nick
*target
, int messagetype
, void **args
) {
442 switch(messagetype
) {
445 /* If it's a message, first arg is nick and second is message */
446 sender
=(nick
*)args
[0];
448 Error("control",ERR_INFO
,"From: %s!%s@%s: %s",sender
->nick
,sender
->ident
,sender
->host
->name
->content
, (char *)args
[1]);
450 /* Split the line into params */
451 cargc
=splitline((char *)args
[1],cargv
,50,0);
458 cmd
=findcommandintree(controlcmds
,cargv
[0],1);
460 controlreply(sender
,"Unknown command.");
464 if (cmd
->level
>0 && !IsOper(sender
)) {
465 controlreply(sender
,"You need to be opered to use this command.");
469 /* If we were doing "authed user tracking" here we'd put a check in for authlevel */
471 /* Check the maxargs */
472 if (cmd
->maxparams
<(cargc
-1)) {
473 /* We need to do some rejoining */
474 rejoinline(cargv
[cmd
->maxparams
],cargc
-(cmd
->maxparams
));
475 cargc
=(cmd
->maxparams
)+1;
478 if((cmd
->handler
)((void *)sender
,cargc
-1,&(cargv
[1])) == CMD_USAGE
)
479 controlhelp(sender
, cmd
);
483 /* someone killed me? Bastards */
484 scheduleoneshot(time(NULL
)+1,&controlconnect
,NULL
);
493 void controlmessage(nick
*target
, char *message
, ... ) {
501 va_start(va
,message
);
502 vsnprintf(buf
,512,message
,va
);
505 sendmessagetouser(mynick
,target
,"%s",buf
);
508 void controlchanmsg(channel
*cp
, char *message
, ...) {
516 va_start(va
,message
);
517 vsnprintf(buf
,512,message
,va
);
520 sendmessagetochannel(mynick
,cp
,"%s",buf
);
523 void controlnotice(nick
*target
, char *message
, ... ) {
531 va_start(va
,message
);
532 vsnprintf(buf
,512,message
,va
);
535 sendnoticetouser(mynick
,target
,"%s",buf
);
538 void controlspecialrmmod(void *arg
) {
539 struct specialsched
*a
= (struct specialsched
*)arg
;
540 sstring
*froo
= a
->modulename
;
544 rmmod(froo
->content
);
548 void controlspecialreloadmod(void *arg
) {
549 struct specialsched
*a
= (struct specialsched
*)arg
;
550 sstring
*froo
= a
->modulename
;
554 safereload(froo
->content
);
558 void controlhelp(nick
*np
, Command
*cmd
) {
559 sstring
*scp
= cmd
->help
;
561 controlreply(np
, "Sorry, no help for this command.");
563 char *cp
= scp
->content
, *sp
= cp
;
566 if(*cp
== '\0' || *cp
== '\n') {
574 controlreply(np
, "%s", sp
);
587 int controlhelpcmd(void *sender
, int cargc
, char **cargv
) {
589 nick
*np
= (nick
*)sender
;
594 cmd
=findcommandintree(controlcmds
,cargv
[0],1);
596 controlreply(np
,"Unknown command.");
600 controlhelp(np
, cmd
);
604 void controlnoticeopers(flag_t permissionlevel
, flag_t noticelevel
, char *format
, ...) {
610 va_start(va
, format
);
611 vsnprintf(broadcast
, sizeof(broadcast
), format
, va
);
614 for(i
=0;i
<NICKHASHSIZE
;i
++)
615 for(np
=nicktable
[i
];np
;np
=np
->next
)
617 controlnotice(np
, "%s", broadcast
);
620 void controlnswall(int noticelevel
, char *format
, ...) {
624 va_start(va
, format
);
625 vsnprintf(broadcast
, sizeof(broadcast
), format
, va
);
628 controlwall(NO_OPER
, noticelevel
, "%s", broadcast
);
631 void handlerehash(int hooknum
, void *arg
) {
632 long hupped
= (long)arg
;
634 controlwall(NO_OPER
, NL_OPERATIONS
, "SIGHUP received, rehashing...");