]> jfr.im git - irc/quakenet/newserv.git/blobdiff - noperserv/noperserv_commands.c
CHANSERV: batcher activate account should use ciphertext of password for hmac
[irc/quakenet/newserv.git] / noperserv / noperserv_commands.c
index 0f0dc038c4e5956aa82a17ec0ef16c2e4459ff7a..5c14dc494430fa95bb242906477153c13a555bf1 100644 (file)
-#include <stdio.h>\r
-#include <stdlib.h>\r
-#include <stdarg.h>\r
-#include <string.h>\r
-#include <strings.h>\r
-#include <time.h>\r
-\r
-#include "../control/control.h"\r
-#include "../nick/nick.h"\r
-#include "../lib/irc_string.h"\r
-#include "../lib/strlfunc.h"\r
-#include "../localuser/localuserchannel.h"\r
-#include "../geoip/geoip.h"\r
-\r
-int controlkill(void *sender, int cargc, char **cargv);\r
-int controlopchan(void *sender, int cargc, char **cargv);\r
-int controlkick(void *sender, int cargc, char **cargv);\r
-int controlspewchan(void *sender, int cargc, char **cargv);\r
-int controlspew(void *sender, int cargc, char **cargv);\r
-int controlresync(void *sender, int cargc, char **cargv);\r
-int controlbroadcast(void *sender, int cargc, char **cargv);\r
-int controlobroadcast(void *sender, int cargc, char **cargv);\r
-int controlmbroadcast(void *sender, int cargc, char **cargv);\r
-int controlsbroadcast(void *sender, int cargc, char **cargv);\r
-int controlcbroadcast(void *sender, int cargc, char **cargv);\r
-\r
-void _init() {\r
-  registercontrolhelpcmd("kill", NO_OPER, 2, &controlkill, "Usage: kill <nick> ?reason?\nKill specified user with given reason (or 'Killed').");\r
-  registercontrolhelpcmd("kick", NO_OPER, 3, &controlkick, "Usage: kick <channel> <user> ?reason?\nKick a user from the given channel, with given reason (or 'Kicked').");\r
-\r
-  registercontrolhelpcmd("spewchan", NO_OPER, 1, &controlspewchan, "Usage: spewchan <pattern>\nShows all channels which are matched by the given pattern");\r
-\r
-/*  registercontrolhelpcmd("spew", NO_OPER, 1, &controlspew, "Usage: spewchan <pattern>\nShows all userss which are matched by the given pattern"); */\r
-\r
-  registercontrolhelpcmd("resync", NO_OPER, 1, &controlresync, "Usage: resync <channel>\nResyncs a desynched channel");\r
-\r
-  registercontrolhelpcmd("broadcast", NO_OPER, 1, &controlbroadcast, "Usage: broadcast <text>\nSends a message to all users.");\r
-  registercontrolhelpcmd("obroadcast", NO_OPER, 1, &controlobroadcast, "Usage: obroadcast <text>\nSends a message to all IRC operators.");\r
-  registercontrolhelpcmd("mbroadcast", NO_OPER, 2, &controlmbroadcast, "Usage: mbroadcast <mask> <text>\nSends a message to all users matching the mask");\r
-  registercontrolhelpcmd("sbroadcast", NO_OPER, 2, &controlsbroadcast, "Usage: sbroadcast <mask> <text>\nSends a message to all users on specific server(s).");\r
-  registercontrolhelpcmd("cbroadcast", NO_OPER, 2, &controlcbroadcast, "Usage: sbroadcast <2 letter country> <text>\nSends a message to all users in the specified country (GeoIP must be loaded).");\r
-}\r
-\r
-void _fini() {\r
-  deregistercontrolcmd("obroadcast", controlobroadcast);\r
-  deregistercontrolcmd("sbroadcast", controlsbroadcast);\r
-  deregistercontrolcmd("mbroadcast", controlmbroadcast);\r
-  deregistercontrolcmd("broadcast", controlbroadcast);\r
-\r
-  deregistercontrolcmd("resync", controlresync);\r
-/*   deregistercontrolcmd("spew", controlspew); */\r
-  deregistercontrolcmd("spewchan", controlspewchan);\r
-\r
-  deregistercontrolcmd("kill", controlkill); \r
-  deregistercontrolcmd("kick", controlkick);\r
-}\r
-\r
-int controlkick(void *sender, int cargc, char **cargv) {\r
-  nick *np=(nick *)sender;\r
-  channel *cp;\r
-  nick *target;\r
-\r
-  if (cargc<2)\r
-    return CMD_USAGE;\r
-\r
-  if ((cp=findchannel(cargv[0]))==NULL) {\r
-    controlreply(np,"Couldn't find that channel.");\r
-    return CMD_ERROR;\r
-  }\r
-\r
-  if ((target=getnickbynick(cargv[1]))==NULL) {\r
-    controlreply(np,"Sorry, couldn't find that user");\r
-    return CMD_ERROR;\r
-  }\r
-\r
-  controlwall(NO_OPER, NL_KICKS, "%s/%s sent KICK for %s!%s@%s on %s (%s)", np->nick, np->authname, target->nick, target->ident, target->host->name->content,cp->index->name->content, (cargc>2)?cargv[2]:"Kicked");\r
-  localkickuser(NULL, cp, target, (cargc>2)?cargv[2]:"Kicked");\r
-  controlreply(sender, "KICK sent.");\r
-\r
-  return CMD_OK;\r
-}\r
-\r
-/* NO checking to see if it's us or anyone important */\r
-int controlkill(void *sender, int cargc, char **cargv) {\r
-  nick *target;\r
-  nick *np = (nick *)sender;\r
\r
-  if (cargc<1)\r
-    return CMD_USAGE;  \r
-\r
-  if ((target=getnickbynick(cargv[0]))==NULL) {\r
-    controlreply(np,"Sorry, couldn't find that user.");\r
-    return CMD_ERROR;\r
-  }\r
-\r
-  controlwall(NO_OPER, NL_KILLS, "%s/%s sent KILL for %s!%s@%s (%s)", np->nick, np->authname, target->nick, target->ident, target->host->name->content, (cargc>1)?cargv[1]:"Killed");\r
-  killuser(NULL, target, (cargc>1)?cargv[1]:"Killed");\r
-  controlreply(np, "KILL sent.");\r
-\r
-  return CMD_OK;\r
-}\r
-\r
-int controlresync(void *sender, int cargc, char **cargv) {\r
-  nick *np = (nick*)sender;\r
-  channel *cp;\r
-  modechanges changes;\r
-  int a;\r
-\r
-  if (cargc < 1)\r
-    return CMD_USAGE;\r
-\r
-  cp = findchannel(cargv[0]);\r
-\r
-  if (cp == NULL) {\r
-    controlreply(np, "No such channel.");\r
-\r
-    return CMD_ERROR;\r
-  }\r
-\r
-  controlwall(NO_OPER, NL_KICKS, "%s/%s RESYNC'ed %s", np->nick, np->authname, cp->index->name->content);\r
-\r
-  irc_send("%s CM %s o", mynumeric->content, cp->index->name->content);\r
-\r
-  localsetmodeinit(&changes, cp, mynick);\r
-\r
-  for(a=0;a<cp->users->hashsize;a++) {\r
-    if (cp->users->content[a] != nouser) {\r
-      np = getnickbynumeric(cp->users->content[a]);\r
-\r
-      /* make newserv believe that this user is not opped */\r
-      if (cp->users->content[a] & CUMODE_OP)\r
-        cp->users->content[a] &= ~CUMODE_OP;\r
-      else if (!IsService(np)) /* if the user wasn't opped before and is not a service we don't care about him */\r
-        continue;\r
-\r
-      /* now reop him */\r
-      localdosetmode_nick(&changes, np, MC_OP);\r
-    }\r
-  }\r
-\r
-  localsetmodeflush(&changes, 1);\r
-\r
-  controlreply(np, "Done.");\r
-\r
-  return CMD_OK;\r
-}\r
-\r
-int controlspew(void *sender, int cargc, char **cargv) {\r
-  nick *np = (nick *)sender;\r
-\r
-  return CMD_OK;\r
-}\r
-\r
-int controlspewchan(void *sender, int cargc, char **cargv) {\r
-  nick *np = (nick*)sender;\r
-  nick *np2;\r
-  int i, a, ccount = 0, morecservices, ucount, cscount;\r
-  chanindex *cip;\r
-  channel *cp;\r
-  char cservices[300];\r
-  char strcscount[40];\r
-\r
-  if (cargc < 1)\r
-    return CMD_USAGE;\r
-\r
-  for (i=0; i<CHANNELHASHSIZE; i++) {\r
-    for (cip=chantable[i]; cip; cip=cip->next) {\r
-      if ((cp = cip->channel) != NULL && match2strings(cargv[0], cip->name->content)) {\r
-        /* find channel services */\r
-        cservices[0] = '\0';\r
-        cscount = ucount = morecservices = 0;\r
-\r
-        for (a=0;a<cp->users->hashsize;a++) {\r
-          if (cp->users->content[a] != nouser) {\r
-            np2 = getnickbynumeric(cp->users->content[a]);\r
-\r
-            if (IsService(np2)) {\r
-              cscount++;\r
-\r
-              if (strlen(cservices) < 100) {\r
-                if (cservices[0])\r
-                  strlcat(cservices, ", ", sizeof(cservices));\r
-  \r
-                strlcat(cservices, np2->nick, sizeof(cservices));\r
-              } else {\r
-                morecservices++;\r
-              }\r
-            }\r
-            \r
-            ucount++;\r
-          }\r
-        }\r
-\r
-        if (morecservices)\r
-          snprintf(cservices + strlen(cservices), sizeof(cservices), " and %d more", morecservices);\r
-\r
-        snprintf(strcscount, sizeof(strcscount), "%d", cscount);\r
-\r
-        /* TODO: print this channel ;; @slug -- WTF? */\r
-        controlreply(np, "%-30s %5ld %-8s%s%s%-11s%s%s%s",\r
-              cip->name->content,\r
-              ucount,\r
-              ucount != 1 ? "users" : "user",\r
-              *cservices ? "- found " : "",\r
-              *cservices ? strcscount : "",\r
-              *cservices ? (strchr(cservices, ',') ? " chanservs" : " chanserv") : "",\r
-              *cservices ? "(" : "",\r
-              *cservices ? cservices : "",\r
-              *cservices ? ")" : "");\r
-\r
-        ccount++;\r
-        \r
-        if (ccount > 1000)\r
-          break; /* Don't ever list more than 1000 channels */\r
-      }\r
-    }\r
-  }\r
-\r
-  if (ccount > 1000)\r
-    controlreply(np, "More than 1000 matching channels were found. Please use a more specific pattern.");\r
-  else\r
-    controlreply(np, "Found %d channels matching specified pattern.", ccount);\r
-  \r
-  return CMD_OK;\r
-}\r
-\r
-int controlbroadcast(void *sender, int cargc, char **cargv) {\r
-  nick *np = (nick*)sender;\r
-\r
-  if (cargc<1)\r
-    return CMD_USAGE;\r
-\r
-  controlwall(NO_OPER, NL_BROADCASTS, "%s/%s sent a broadcast: %s", np->nick, np->authname, cargv[0]);\r
-\r
-  irc_send("%s O $* :(Broadcast) %s", longtonumeric(mynick->numeric,5), cargv[0]);\r
-\r
-  controlreply(np, "Message broadcasted.");\r
-\r
-  return CMD_OK;\r
-}\r
-\r
-int controlmbroadcast(void *sender, int cargc, char **cargv) {\r
-  nick *np = (nick*)sender;\r
-\r
-  if (cargc<2)\r
-    return CMD_USAGE;\r
-\r
-  controlwall(NO_OPER, NL_BROADCASTS, "%s/%s sent an mbroadcast to %s: %s", np->nick, np->authname, cargv[0], cargv[1]);\r
-\r
-  irc_send("%s O $@%s :(mBroadcast) %s", longtonumeric(mynick->numeric,5), cargv[0], cargv[1]);\r
-\r
-  controlreply(np, "Message mbroadcasted.");\r
-\r
-  return CMD_OK;\r
-}\r
-\r
-int controlsbroadcast(void *sender, int cargc, char **cargv) {\r
-  nick *np = (nick *)sender;\r
-\r
-  if(cargc<2)\r
-    return CMD_USAGE;\r
-\r
-  controlwall(NO_OPER, NL_BROADCASTS, "%s/%s sent an sbroadcast to %s: %s", np->nick, np->authname, cargv[0], cargv[1]);\r
-\r
-  irc_send("%s O $%s :(sBroadcast) %s", longtonumeric(mynick->numeric,5), cargv[0], cargv[1]);\r
-\r
-  controlreply(np, "Message sbroadcasted.");\r
-\r
-  return CMD_OK;\r
-}\r
-\r
-int controlobroadcast(void *sender, int cargc, char **cargv) {\r
-  nick *np = (nick *)sender, *nip;\r
-  int i;\r
-\r
-  if(cargc<1)\r
-    return CMD_USAGE;\r
-\r
-  controlwall(NO_OPER, NL_BROADCASTS, "%s/%s sent an obroadcast: %s", np->nick, np->authname, cargv[0]);\r
-\r
-  for(i=0;i<NICKHASHSIZE;i++)\r
-    for(nip=nicktable[i];nip;nip=nip->next)\r
-      if(nip && IsOper(nip))\r
-        controlnotice(nip, "(oBroadcast) %s", cargv[0]);\r
-\r
-  controlreply(np, "Message obroadcasted.");\r
-\r
-  return CMD_OK;\r
-}\r
-\r
-const char GeoIP_country_code[247][3] = { "--","AP","EU","AD","AE","AF","AG","AI","AL","AM","AN","AO","AQ","AR","AS","AT","AU","AW","AZ","BA","BB","BD","BE","BF","BG","BH","BI","BJ","BM","BN","BO","BR","BS","BT","BV","BW","BY","BZ","CA","CC","CD","CF","CG","CH","CI","CK","CL","CM","CN","CO","CR","CU","CV","CX","CY","CZ","DE","DJ","DK","DM","DO","DZ","EC","EE","EG","EH","ER","ES","ET","FI","FJ","FK","FM","FO","FR","FX","GA","GB","GD","GE","GF","GH","GI","GL","GM","GN","GP","GQ","GR","GS","GT","GU","GW","GY","HK","HM","HN","HR","HT","HU","ID","IE","IL","IN","IO","IQ","IR","IS","IT","JM","JO","JP","KE","KG","KH","KI","KM","KN","KP","KR","KW","KY","KZ","LA","LB","LC","LI","LK","LR","LS","LT","LU","LV","LY","MA","MC","MD","MG","MH","MK","ML","MM","MN","MO","MP","MQ","MR","MS","MT","MU","MV","MW","MX","MY","MZ","NA","NC","NE","NF","NG","NI","NL","NO","NP","NR","NU","NZ","OM","PA","PE","PF","PG","PH","PK","PL","PM","PN","PR","PS","PT","PW","PY","QA","RE","RO","RU","RW","SA","SB","SC","SD","SE","SG","SH","SI","SJ","SK","SL","SM","SN","SO","SR","ST","SV","SY","SZ","TC","TD","TF","TG","TH","TJ","TK","TM","TN","TO","TP","TR","TT","TV","TW","TZ","UA","UG","UM","US","UY","UZ","VA","VC","VE","VG","VI","VN","VU","WF","WS","YE","YT","YU","ZA","ZM","ZR","ZW","A1","A2","O1"};\r
-\r
-int controlcbroadcast(void *sender, int cargc, char **cargv) {\r
-  nick *np = (nick *)sender, *nip;\r
-  int i, ext, target;\r
-\r
-  if(cargc < 2)\r
-    return CMD_USAGE;\r
-\r
-  ext = findnickext("geoip");\r
-  if(ext == -1) {\r
-    controlreply(np, "Geoip module not loaded.");\r
-    return CMD_ERROR;\r
-  }\r
-\r
-  target = COUNTRY_MIN - 1;\r
-  for(i=COUNTRY_MIN;i<COUNTRY_MAX;i++) {\r
-    if(!strcasecmp(cargv[0], GeoIP_country_code[i])) {\r
-      target = i;\r
-      break;\r
-    }\r
-  }\r
-\r
-  if(target == -1) {\r
-    controlreply(np, "Invalid country.");\r
-    return CMD_ERROR;\r
-  }\r
-\r
-  controlwall(NO_OPER, NL_BROADCASTS, "%s/%s sent a cbroadcast %s: %s", np->nick, np->authname, cargv[0], cargv[1]);\r
-\r
-  for(i=0;i<NICKHASHSIZE;i++)\r
-    for(nip=nicktable[i];nip;nip=nip->next)\r
-      if(nip && ((int)((long)nip->exts[ext]) == target))\r
-        controlnotice(nip, "(cBroadcast) %s", cargv[1]);\r
-\r
-  controlreply(np, "Message cbroadcasted.");\r
-\r
-  return CMD_OK;\r
-}\r
+#include <stdio.h>
+#include <stdlib.h>
+#include <stdarg.h>
+#include <string.h>
+#include <strings.h>
+#include <time.h>
+
+#include "../control/control.h"
+#include "../nick/nick.h"
+#include "../newsearch/newsearch.h"
+#include "../lib/irc_string.h"
+#include "../lib/strlfunc.h"
+#include "../localuser/localuserchannel.h"
+#include "../lib/version.h"
+#include "../core/modules.h"
+
+MODULE_VERSION("");
+
+int controlkill(void *sender, int cargc, char **cargv);
+int controlkick(void *sender, int cargc, char **cargv);
+int controlspewchan(void *sender, int cargc, char **cargv);
+int controlspew(void *sender, int cargc, char **cargv);
+int controlcompare(void *sender, int cargc, char **cargv);
+int controlbroadcast(void *sender, int cargc, char **cargv);
+int controlobroadcast(void *sender, int cargc, char **cargv);
+int controlmbroadcast(void *sender, int cargc, char **cargv);
+int controlsbroadcast(void *sender, int cargc, char **cargv);
+int controlcbroadcast(void *sender, int cargc, char **cargv);
+
+void _init() {
+  registercontrolhelpcmd("kill", NO_OPER, 2, &controlkill, "Usage: kill <nick> ?reason?\nKill specified user with given reason (or 'Killed').");
+  registercontrolhelpcmd("kick", NO_OPER, 3, &controlkick, "Usage: kick <channel> <user> ?reason?\nKick a user from the given channel, with given reason (or 'Kicked').");
+
+  registercontrolhelpcmd("spewchan", NO_OPER, 1, &controlspewchan, "Usage: spewchan <pattern>\nShows all channels which are matched by the given pattern");
+  registercontrolhelpcmd("spew", NO_OPER, 1, &controlspew, "Usage: spew <pattern>\nShows all userss which are matched by the given pattern");
+  registercontrolhelpcmd("compare", NO_OPER, 255, &controlcompare, "Usage: compare <nick1> <nick2> ... <nickn>\nShows channels shared by supplied nicknames\nUsage: compare <channel1> <channel2> ... <channeln>\nShares nicks that share the supplied channels");
+
+  registercontrolhelpcmd("broadcast", NO_OPER, 1, &controlbroadcast, "Usage: broadcast <text>\nSends a message to all users.");
+  registercontrolhelpcmd("obroadcast", NO_OPER, 1, &controlobroadcast, "Usage: obroadcast <text>\nSends a message to all IRC operators.");
+  registercontrolhelpcmd("mbroadcast", NO_OPER, 2, &controlmbroadcast, "Usage: mbroadcast <mask> <text>\nSends a message to all users matching the mask");
+  registercontrolhelpcmd("sbroadcast", NO_OPER, 2, &controlsbroadcast, "Usage: sbroadcast <mask> <text>\nSends a message to all users on specific server(s).");
+  registercontrolhelpcmd("cbroadcast", NO_OPER, 2, &controlcbroadcast, "Usage: cbroadcast <2 letter country> <text>\nSends a message to all users in the specified country (GeoIP must be loaded).");
+}
+
+void _fini() {
+  deregistercontrolcmd("obroadcast", controlobroadcast);
+  deregistercontrolcmd("sbroadcast", controlsbroadcast);
+  deregistercontrolcmd("mbroadcast", controlmbroadcast);
+  deregistercontrolcmd("broadcast", controlbroadcast);
+
+  deregistercontrolcmd("spew", controlspew);
+  deregistercontrolcmd("spewchan", controlspewchan);
+
+  deregistercontrolcmd("kill", controlkill);
+  deregistercontrolcmd("kick", controlkick);
+
+  deregistercontrolcmd("compare", controlcompare);
+}
+
+int controlkick(void *sender, int cargc, char **cargv) {
+  nick *np=(nick *)sender;
+  channel *cp;
+  nick *target;
+
+  if (cargc<2)
+    return CMD_USAGE;
+
+  if ((cp=findchannel(cargv[0]))==NULL) {
+    controlreply(np,"Couldn't find that channel.");
+    return CMD_ERROR;
+  }
+
+  if ((target=getnickbynick(cargv[1]))==NULL) {
+    controlreply(np,"Sorry, couldn't find that user");
+    return CMD_ERROR;
+  }
+
+  if(IsService(target) && SIsService(&serverlist[homeserver(target->numeric)])) {
+    controlreply(np,"Sorry, cannot kick network services.");
+    return CMD_ERROR;
+  }
+
+  controlwall(NO_OPER, NL_KICKKILLS, "%s/%s sent KICK for %s!%s@%s on %s (%s)", np->nick, np->authname, target->nick, target->ident, target->host->name->content,cp->index->name->content, (cargc>2)?cargv[2]:"Kicked");
+  localkickuser(NULL, cp, target, (cargc>2)?cargv[2]:"Kicked");
+  controlreply(sender, "KICK sent.");
+
+  return CMD_OK;
+}
+
+int controlkill(void *sender, int cargc, char **cargv) {
+  nick *target;
+  nick *np = (nick *)sender;
+
+  if (cargc<1)
+    return CMD_USAGE;
+
+  if ((target=getnickbynick(cargv[0]))==NULL) {
+    controlreply(np,"Sorry, couldn't find that user.");
+    return CMD_ERROR;
+  }
+
+  if(IsService(target) && SIsService(&serverlist[homeserver(target->numeric)])) {
+    controlreply(np,"Sorry, cannot kill network services.");
+    controlwall(NO_OPER, NL_KICKKILLS, "%s/%s attempted to KILL service %s!%s@%s (%s)", np->nick, np->authname, target->nick, target->ident, target->host->name->content, (cargc>1)?cargv[1]:"Killed");
+    return CMD_ERROR;
+  }
+
+  controlwall(NO_OPER, NL_KICKKILLS, "%s/%s sent KILL for %s!%s@%s (%s)", np->nick, np->authname, target->nick, target->ident, target->host->name->content, (cargc>1)?cargv[1]:"Killed");
+  killuser(NULL, target, "%s", (cargc>1)?cargv[1]:"Killed");
+  controlreply(np, "KILL sent.");
+
+  return CMD_OK;
+}
+
+int controlspew(void *sender, int cargc, char **cargv) {
+  searchASTExpr tree;
+
+  if(cargc < 1)
+    return CMD_USAGE;
+
+  tree = NSASTNode(match_parse, NSASTNode(hostmask_parse), NSASTLiteral(cargv[0]));
+  return ast_nicksearch(&tree, controlreply, sender, controlnswall, printnick, NULL, NULL, 500);
+}
+
+int controlspewchan(void *sender, int cargc, char **cargv) {
+  searchASTExpr tree;
+
+  if(cargc < 1)
+    return CMD_USAGE;
+
+  tree = NSASTNode(match_parse, NSASTNode(name_parse), NSASTLiteral(cargv[0]));
+  return ast_chansearch(&tree, controlreply, sender, controlnswall, printchannel, NULL, NULL, 500);
+}
+
+int controlcompare(void *sender, int cargc, char **cargv) {
+  searchASTExpr nodes[cargc], literals[cargc], tree;
+  int i;
+  void *displayfn;
+  ASTFunc execfn;
+  parseFunc parsefn;
+  
+  if(cargc < 2)
+    return CMD_USAGE;
+
+  if(cargv[0][0] == '#') {
+    execfn = (ASTFunc)ast_nicksearch;
+    displayfn = printnick;
+    parsefn = channel_parse;
+  } else {
+    execfn = (ASTFunc)ast_chansearch;
+    displayfn = printchannel;
+    parsefn = nick_parse;
+  }
+    
+  for(i=0;i<cargc;i++) {
+    literals[i] = NSASTLiteral(cargv[i]);
+    nodes[i] = NSASTManualNode(parsefn, 1, &literals[i]);
+  }
+  tree = NSASTManualNode(and_parse, cargc, nodes);
+  
+  return (execfn)(&tree, controlreply, sender, controlnswall, displayfn, NULL, NULL, 500);
+}
+
+int controlbroadcast(void *sender, int cargc, char **cargv) {
+  nick *np = (nick*)sender;
+
+  if (cargc<1)
+    return CMD_USAGE;
+
+  controlwall(NO_OPER, NL_BROADCASTS, "%s/%s sent a broadcast: %s", np->nick, np->authname, cargv[0]);
+
+  irc_send("%s O $* :(Broadcast) %s", longtonumeric(mynick->numeric,5), cargv[0]);
+
+  controlreply(np, "Message broadcasted.");
+
+  return CMD_OK;
+}
+
+int controlmbroadcast(void *sender, int cargc, char **cargv) {
+  nick *np = (nick*)sender;
+
+  if (cargc<2)
+    return CMD_USAGE;
+
+  controlwall(NO_OPER, NL_BROADCASTS, "%s/%s sent an mbroadcast to %s: %s", np->nick, np->authname, cargv[0], cargv[1]);
+
+  irc_send("%s O $@%s :(mBroadcast) %s", longtonumeric(mynick->numeric,5), cargv[0], cargv[1]);
+
+  controlreply(np, "Message mbroadcasted.");
+
+  return CMD_OK;
+}
+
+int controlsbroadcast(void *sender, int cargc, char **cargv) {
+  nick *np = (nick *)sender;
+
+  if(cargc<2)
+    return CMD_USAGE;
+
+  controlwall(NO_OPER, NL_BROADCASTS, "%s/%s sent an sbroadcast to %s: %s", np->nick, np->authname, cargv[0], cargv[1]);
+
+  irc_send("%s O $%s :(sBroadcast) %s", longtonumeric(mynick->numeric,5), cargv[0], cargv[1]);
+
+  controlreply(np, "Message sbroadcasted.");
+
+  return CMD_OK;
+}
+
+int controlobroadcast(void *sender, int cargc, char **cargv) {
+  nick *np = (nick *)sender;
+  searchASTExpr tree;
+  int ret;
+  char message[512];
+  
+  if(cargc<1)
+    return CMD_USAGE;
+
+  snprintf(message, sizeof(message), "(oBroadcast) %s", cargv[0]);
+
+  tree = NSASTNode(and_parse, NSASTNode(modes_parse, NSASTLiteral("+o")), NSASTNode(notice_parse, NSASTLiteral(message)));
+  if((ret=ast_nicksearch(&tree, controlreply, sender, controlnswall, NULL, NULL, NULL, -1)) == CMD_OK) {
+    controlwall(NO_OPER, NL_BROADCASTS, "%s/%s sent an obroadcast: %s", np->nick, np->authname, cargv[0]);
+    controlreply(np, "Message obroadcasted.");
+  } else {
+    controlreply(np, "An error occured.");
+  }
+
+  return ret;
+}
+
+int controlcbroadcast(void *sender, int cargc, char **cargv) {
+  nick *np = (nick *)sender;
+  int ret;
+  searchASTExpr tree;
+  char message[512];
+  
+  if(cargc < 2)
+    return CMD_USAGE;
+
+  snprintf(message, sizeof(message), "(cBroadcast) %s", cargv[1]);
+  tree = NSASTNode(and_parse, NSASTNode(country_parse, NSASTLiteral(cargv[0])), NSASTNode(notice_parse, NSASTLiteral(message)));
+  if((ret=ast_nicksearch(&tree, controlreply, sender, controlnswall, NULL, NULL, NULL, -1)) == CMD_OK) {
+    controlwall(NO_OPER, NL_BROADCASTS, "%s/%s sent a cbroadcast %s: %s", np->nick, np->authname, cargv[0], cargv[1]);
+    controlreply(np, "Message cbroadcasted.");
+  } else {
+    controlreply(np, "An error occured.");
+  }
+
+  return ret;
+}