]> jfr.im git - solanum.git/blobdiff - modules/core/m_message.c
Add oper:free_target (#374)
[solanum.git] / modules / core / m_message.c
index 95a68fc76b5eded57416cf2c18ca6a18f0af232d..8825bbba0b198a161f6fbdbe78b43502dee57002 100644 (file)
  *  along with this program; if not, write to the Free Software
  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307
  *  USA
- *
- *  $Id: m_message.c 3173 2007-01-31 23:57:18Z jilles $
  */
 
 #include "stdinc.h"
 #include "client.h"
 #include "ircd.h"
 #include "numeric.h"
-#include "common.h"
 #include "s_conf.h"
 #include "s_serv.h"
 #include "msg.h"
 #include "parse.h"
 #include "modules.h"
 #include "channel.h"
-#include "irc_string.h"
+#include "match.h"
 #include "hash.h"
 #include "class.h"
 #include "msg.h"
 #include "send.h"
 #include "s_newconf.h"
 #include "s_stats.h"
+#include "tgchange.h"
+#include "inline/stringops.h"
+
+static const char message_desc[] =
+       "Provides the PRIVMSG and NOTICE commands to send messages to users and channels";
 
-static int m_message(int, const char *, struct Client *, struct Client *, int, const char **);
-static int m_privmsg(struct Client *, struct Client *, int, const char **);
-static int m_notice(struct Client *, struct Client *, int, const char **);
+static void m_message(enum message_type, struct MsgBuf *, struct Client *, struct Client *, int, const char **);
+static void m_privmsg(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
+static void m_notice(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
+static void m_echo(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
+
+static void echo_msg(struct Client *, struct Client *, enum message_type, const char *);
 
 static void expire_tgchange(void *unused);
 static struct ev_entry *expire_tgchange_event;
 
+static unsigned int CAP_ECHO;
+
 static int
 modinit(void)
 {
@@ -66,17 +73,26 @@ moddeinit(void)
 }
 
 struct Message privmsg_msgtab = {
-       "PRIVMSG", 0, 0, 0, MFLG_SLOW | MFLG_UNREG,
+       "PRIVMSG", 0, 0, 0, 0,
        {mg_unreg, {m_privmsg, 0}, {m_privmsg, 0}, mg_ignore, mg_ignore, {m_privmsg, 0}}
 };
 struct Message notice_msgtab = {
-       "NOTICE", 0, 0, 0, MFLG_SLOW,
+       "NOTICE", 0, 0, 0, 0,
        {mg_unreg, {m_notice, 0}, {m_notice, 0}, {m_notice, 0}, mg_ignore, {m_notice, 0}}
 };
+struct Message echo_msgtab = {
+       "ECHO", 0, 0, 0, 0,
+       {mg_unreg, mg_ignore, {m_echo, 3}, mg_ignore, mg_ignore, mg_ignore}
+};
+
+mapi_clist_av1 message_clist[] = { &privmsg_msgtab, &notice_msgtab, &echo_msgtab, NULL };
 
-mapi_clist_av1 message_clist[] = { &privmsg_msgtab, &notice_msgtab, NULL };
+mapi_cap_list_av2 message_cap_list[] = {
+       { MAPI_CAP_SERVER, "ECHO", NULL, &CAP_ECHO },
+       { 0, NULL, NULL, NULL }
+};
 
-DECLARE_MODULE_AV1(message, modinit, moddeinit, message_clist, NULL, NULL, "$Revision: 3173 $");
+DECLARE_MODULE_AV2(message, modinit, moddeinit, message_clist, NULL, NULL, message_cap_list, NULL, message_desc);
 
 struct entity
 {
@@ -85,38 +101,44 @@ struct entity
        int flags;
 };
 
-static int build_target_list(int p_or_n, const char *command,
+static int build_target_list(enum message_type msgtype,
                             struct Client *client_p,
                             struct Client *source_p, const char *nicks_channels, const char *text);
 
-static int flood_attack_client(int p_or_n, struct Client *source_p, struct Client *target_p);
-static int flood_attack_channel(int p_or_n, struct Client *source_p,
-                               struct Channel *chptr, char *chname);
-static struct Client *find_userhost(const char *, const char *, int *);
+static bool flood_attack_client(enum message_type msgtype, struct Client *source_p, struct Client *target_p);
+
+/* Fifteen seconds should be plenty for a client to reply a ctcp */
+#define LARGE_CTCP_TIME 15
 
 #define ENTITY_NONE    0
 #define ENTITY_CHANNEL 1
-#define ENTITY_CHANOPS_ON_CHANNEL 2
-#define ENTITY_CLIENT  3
+#define ENTITY_CHANNEL_OPMOD 2
+#define ENTITY_CHANOPS_ON_CHANNEL 3
+#define ENTITY_CLIENT  4
 
 static struct entity targets[512];
 static int ntargets = 0;
 
-static int duplicate_ptr(void *);
+static bool duplicate_ptr(void *);
 
-static void msg_channel(int p_or_n, const char *command,
+static void msg_channel(enum message_type msgtype,
                        struct Client *client_p,
                        struct Client *source_p, struct Channel *chptr, const char *text);
 
-static void msg_channel_flags(int p_or_n, const char *command,
+static void msg_channel_opmod(enum message_type msgtype,
+                             struct Client *client_p,
+                             struct Client *source_p, struct Channel *chptr,
+                             const char *text);
+
+static void msg_channel_flags(enum message_type msgtype,
                              struct Client *client_p,
                              struct Client *source_p,
                              struct Channel *chptr, int flags, const char *text);
 
-static void msg_client(int p_or_n, const char *command,
+static void msg_client(enum message_type msgtype,
                       struct Client *source_p, struct Client *target_p, const char *text);
 
-static void handle_special(int p_or_n, const char *command,
+static void handle_special(enum message_type msgtype,
                           struct Client *client_p, struct Client *source_p, const char *nick,
                           const char *text);
 
@@ -138,49 +160,48 @@ static void handle_special(int p_or_n, const char *command,
 ** -db Nov 13, 2000
 **
 */
+const char *cmdname[MESSAGE_TYPE_COUNT] = {
+       [MESSAGE_TYPE_PRIVMSG] = "PRIVMSG",
+       [MESSAGE_TYPE_NOTICE] = "NOTICE",
+};
 
-#define PRIVMSG 0
-#define NOTICE  1
-
-static int
-m_privmsg(struct Client *client_p, struct Client *source_p, int parc, const char *parv[])
+static void
+m_privmsg(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char *parv[])
 {
-       return m_message(PRIVMSG, "PRIVMSG", client_p, source_p, parc, parv);
+       m_message(MESSAGE_TYPE_PRIVMSG, msgbuf_p, client_p, source_p, parc, parv);
 }
 
-static int
-m_notice(struct Client *client_p, struct Client *source_p, int parc, const char *parv[])
+static void
+m_notice(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char *parv[])
 {
-       return m_message(NOTICE, "NOTICE", client_p, source_p, parc, parv);
+       m_message(MESSAGE_TYPE_NOTICE, msgbuf_p, client_p, source_p, parc, parv);
 }
 
 /*
  * inputs      - flag privmsg or notice
- *             - pointer to command "PRIVMSG" or "NOTICE"
  *             - pointer to client_p
  *             - pointer to source_p
  *             - pointer to channel
  */
-static int
-m_message(int p_or_n,
-         const char *command,
+static void
+m_message(enum message_type msgtype, struct MsgBuf *msgbuf_p,
          struct Client *client_p, struct Client *source_p, int parc, const char *parv[])
 {
        int i;
 
        if(parc < 2 || EmptyString(parv[1]))
        {
-               if(p_or_n != NOTICE)
+               if(msgtype != MESSAGE_TYPE_NOTICE)
                        sendto_one(source_p, form_str(ERR_NORECIPIENT), me.name,
-                                  source_p->name, command);
-               return 0;
+                                  source_p->name, cmdname[msgtype]);
+               return;
        }
 
        if(parc < 3 || EmptyString(parv[2]))
        {
-               if(p_or_n != NOTICE)
+               if(msgtype != MESSAGE_TYPE_NOTICE)
                        sendto_one(source_p, form_str(ERR_NOTEXTTOSEND), me.name, source_p->name);
-               return 0;
+               return;
        }
 
        /* Finish the flood grace period if theyre not messaging themselves
@@ -189,9 +210,9 @@ m_message(int p_or_n,
        if(MyClient(source_p) && !IsFloodDone(source_p) && irccmp(source_p->name, parv[1]))
                flood_endgrace(source_p);
 
-       if(build_target_list(p_or_n, command, client_p, source_p, parv[1], parv[2]) < 0)
+       if(build_target_list(msgtype, client_p, source_p, parv[1], parv[2]) < 0)
        {
-               return 0;
+               return;
        }
 
        for(i = 0; i < ntargets; i++)
@@ -199,24 +220,47 @@ m_message(int p_or_n,
                switch (targets[i].type)
                {
                case ENTITY_CHANNEL:
-                       msg_channel(p_or_n, command, client_p, source_p,
+                       msg_channel(msgtype, client_p, source_p,
                                    (struct Channel *) targets[i].ptr, parv[2]);
                        break;
 
+               case ENTITY_CHANNEL_OPMOD:
+                       msg_channel_opmod(msgtype, client_p, source_p,
+                                  (struct Channel *) targets[i].ptr, parv[2]);
+                       break;
+
                case ENTITY_CHANOPS_ON_CHANNEL:
-                       msg_channel_flags(p_or_n, command, client_p, source_p,
+                       msg_channel_flags(msgtype, client_p, source_p,
                                          (struct Channel *) targets[i].ptr,
                                          targets[i].flags, parv[2]);
                        break;
 
                case ENTITY_CLIENT:
-                       msg_client(p_or_n, command, source_p,
+                       msg_client(msgtype, source_p,
                                   (struct Client *) targets[i].ptr, parv[2]);
                        break;
                }
        }
+}
 
-       return 0;
+static void
+m_echo(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p,
+               int parc, const char *parv[])
+{
+       struct Client *target_p = find_person(parv[2]);
+       enum message_type msgtype;
+
+       if (target_p == NULL)
+               return;
+
+       switch (parv[1][0])
+       {
+       case 'P': msgtype = MESSAGE_TYPE_PRIVMSG; break;
+       case 'N': msgtype = MESSAGE_TYPE_NOTICE; break;
+       default: return;
+       }
+
+       echo_msg(source_p, target_p, msgtype, parv[3]);
 }
 
 /*
@@ -234,11 +278,10 @@ m_message(int p_or_n,
  *               all the classic old bizzare oper privmsg tricks
  *               are parsed and sent as is, if prefixed with $
  *               to disambiguate.
- *
  */
 
 static int
-build_target_list(int p_or_n, const char *command, struct Client *client_p,
+build_target_list(enum message_type msgtype, struct Client *client_p,
                  struct Client *source_p, const char *nicks_channels, const char *text)
 {
        int type;
@@ -250,7 +293,7 @@ build_target_list(int p_or_n, const char *command, struct Client *client_p,
 
        ntargets = 0;
 
-       for(nick = strtoken(&p, target_list, ","); nick; nick = strtoken(&p, NULL, ","))
+       for(nick = rb_strtok_r(target_list, ",", &p); nick; nick = rb_strtok_r(NULL, ",", &p))
        {
                char *with_prefix;
                /*
@@ -280,7 +323,7 @@ build_target_list(int p_or_n, const char *command, struct Client *client_p,
                        }
 
                        /* non existant channel */
-                       else if(p_or_n != NOTICE)
+                       else if(msgtype != MESSAGE_TYPE_NOTICE)
                                sendto_one_numeric(source_p, ERR_NOSUCHNICK,
                                                   form_str(ERR_NOSUCHNICK), nick);
 
@@ -332,7 +375,7 @@ build_target_list(int p_or_n, const char *command, struct Client *client_p,
                        if(EmptyString(nick))
                        {
                                sendto_one(source_p, form_str(ERR_NORECIPIENT),
-                                          me.name, source_p->name, command);
+                                          me.name, source_p->name, cmdname[msgtype]);
                                continue;
                        }
 
@@ -349,8 +392,10 @@ build_target_list(int p_or_n, const char *command, struct Client *client_p,
                                if(!IsServer(source_p) && !IsService(source_p) && !is_chanop_voiced(msptr))
                                {
                                        sendto_one(source_p, form_str(ERR_CHANOPRIVSNEEDED),
-                                                  me.name, source_p->name, with_prefix);
-                                       return (-1);
+                                                  get_id(&me, source_p),
+                                                  get_id(source_p, source_p),
+                                                  with_prefix);
+                                       continue;
                                }
 
                                if(!duplicate_ptr(chptr))
@@ -366,7 +411,7 @@ build_target_list(int p_or_n, const char *command, struct Client *client_p,
                                        targets[ntargets++].flags = type;
                                }
                        }
-                       else if(p_or_n != NOTICE)
+                       else if(msgtype != MESSAGE_TYPE_NOTICE)
                        {
                                sendto_one_numeric(source_p, ERR_NOSUCHNICK,
                                                   form_str(ERR_NOSUCHNICK), nick);
@@ -375,14 +420,40 @@ build_target_list(int p_or_n, const char *command, struct Client *client_p,
                        continue;
                }
 
+               if(IsServer(client_p) && *nick == '=' && nick[1] == '#')
+               {
+                       nick++;
+                       if((chptr = find_channel(nick)) != NULL)
+                       {
+                               if(!duplicate_ptr(chptr))
+                               {
+                                       if(ntargets >= ConfigFileEntry.max_targets)
+                                       {
+                                               sendto_one(source_p, form_str(ERR_TOOMANYTARGETS),
+                                                          me.name, source_p->name, nick);
+                                               return (1);
+                                       }
+                                       targets[ntargets].ptr = (void *) chptr;
+                                       targets[ntargets++].type = ENTITY_CHANNEL_OPMOD;
+                               }
+                       }
+
+                       /* non existant channel */
+                       else if(msgtype != MESSAGE_TYPE_NOTICE)
+                               sendto_one_numeric(source_p, ERR_NOSUCHNICK,
+                                                  form_str(ERR_NOSUCHNICK), nick);
+
+                       continue;
+               }
+
                if(strchr(nick, '@') || (IsOper(source_p) && (*nick == '$')))
                {
-                       handle_special(p_or_n, command, client_p, source_p, nick, text);
+                       handle_special(msgtype, client_p, source_p, nick, text);
                        continue;
                }
 
                /* no matching anything found - error if not NOTICE */
-               if(p_or_n != NOTICE)
+               if(msgtype != MESSAGE_TYPE_NOTICE)
                {
                        /* dont give this numeric when source is local,
                         * because its misleading --anfl
@@ -407,25 +478,24 @@ build_target_list(int p_or_n, const char *command, struct Client *client_p,
  * inputs      - pointer to check
  *             - pointer to table of entities
  *             - number of valid entities so far
- * output      - YES if duplicate pointer in table, NO if not.
+ * output      - true if duplicate pointer in table, false if not.
  *               note, this does the canonize using pointers
  * side effects        - NONE
  */
-static int
+static bool
 duplicate_ptr(void *ptr)
 {
        int i;
        for(i = 0; i < ntargets; i++)
                if(targets[i].ptr == ptr)
-                       return YES;
-       return NO;
+                       return true;
+       return false;
 }
 
 /*
  * msg_channel
  *
  * inputs      - flag privmsg or notice
- *             - pointer to command "PRIVMSG" or "NOTICE"
  *             - pointer to client_p
  *             - pointer to source_p
  *             - pointer to channel
@@ -435,60 +505,142 @@ duplicate_ptr(void *ptr)
  * XXX - We need to rework this a bit, it's a tad ugly. --nenolod
  */
 static void
-msg_channel(int p_or_n, const char *command,
+msg_channel(enum message_type msgtype,
            struct Client *client_p, struct Client *source_p, struct Channel *chptr,
            const char *text)
 {
        int result;
-       char text2[BUFSIZE];
+       hook_data_privmsg_channel hdata;
 
        if(MyClient(source_p))
        {
-               /* idle time shouldnt be reset by notices --fl */
-               if(p_or_n != NOTICE)
+               /* idle time shouldn't be reset by notices --fl */
+               if(msgtype != MESSAGE_TYPE_NOTICE)
                        source_p->localClient->last = rb_current_time();
        }
 
-       if(chptr->mode.mode & MODE_NOCOLOR)
+       hdata.msgtype = msgtype;
+       hdata.source_p = source_p;
+       hdata.chptr = chptr;
+       hdata.text = text;
+       hdata.approved = 0;
+
+       call_hook(h_privmsg_channel, &hdata);
+
+       /* memory buffer address may have changed, update pointer */
+       text = hdata.text;
+
+       if (hdata.approved != 0)
+               return;
+
+       /* hook may have reduced the string to nothing. */
+       if (EmptyString(text))
        {
-               strlcpy(text2, text, BUFSIZE);
-               strip_colour(text2);
-               text = text2;
-               if (EmptyString(text))
-               {
-                       /* could be empty after colour stripping and
-                        * that would cause problems later */
-                       if(p_or_n != NOTICE)
-                               sendto_one(source_p, form_str(ERR_NOTEXTTOSEND), me.name, source_p->name);
-                       return;
-               }
+               /* could be empty after colour stripping and
+                * that would cause problems later */
+               if(msgtype != MESSAGE_TYPE_NOTICE)
+                       sendto_one(source_p, form_str(ERR_NOTEXTTOSEND), me.name, source_p->name);
+               return;
        }
 
        /* chanops and voiced can flood their own channel with impunity */
        if((result = can_send(chptr, source_p, NULL)))
        {
+               if(result != CAN_SEND_OPV && MyClient(source_p) &&
+                  !IsOperGeneral(source_p) &&
+                  !add_channel_target(source_p, chptr))
+               {
+                       sendto_one(source_p, form_str(ERR_TARGCHANGE),
+                                  me.name, source_p->name, chptr->chname);
+                       return;
+               }
                if(result == CAN_SEND_OPV ||
-                  !flood_attack_channel(p_or_n, source_p, chptr, chptr->chname))
+                  !flood_attack_channel(msgtype, source_p, chptr, chptr->chname))
                {
                        sendto_channel_flags(client_p, ALL_MEMBERS, source_p, chptr,
-                                            "%s %s :%s", command, chptr->chname, text);
+                                            "%s %s :%s", cmdname[msgtype], chptr->chname, text);
                }
        }
        else if(chptr->mode.mode & MODE_OPMODERATE &&
-                       chptr->mode.mode & MODE_MODERATED &&
-                       IsMember(source_p, chptr))
+                       (!(chptr->mode.mode & MODE_NOPRIVMSGS) ||
+                        IsMember(source_p, chptr)))
        {
-               /* only do +z for +m channels for now, as bans/quiets
-                * aren't tested for remote clients -- jilles */
-               if(!flood_attack_channel(p_or_n, source_p, chptr, chptr->chname))
+               if(MyClient(source_p) && !IsOperGeneral(source_p) &&
+                  !add_channel_target(source_p, chptr))
+               {
+                       sendto_one(source_p, form_str(ERR_TARGCHANGE),
+                                  me.name, source_p->name, chptr->chname);
+                       return;
+               }
+               if(!flood_attack_channel(msgtype, source_p, chptr, chptr->chname))
                {
-                       sendto_channel_flags(client_p, ONLY_CHANOPS, source_p, chptr,
-                                            "%s %s :%s", command, chptr->chname, text);
+                       sendto_channel_opmod(client_p, source_p, chptr,
+                                            cmdname[msgtype], text);
                }
        }
        else
        {
-               if(p_or_n != NOTICE)
+               if(msgtype != MESSAGE_TYPE_NOTICE)
+                       sendto_one_numeric(source_p, ERR_CANNOTSENDTOCHAN,
+                                          form_str(ERR_CANNOTSENDTOCHAN), chptr->chname);
+       }
+}
+/*
+ * msg_channel_opmod
+ *
+ * inputs      - flag privmsg or notice
+ *             - pointer to client_p
+ *             - pointer to source_p
+ *             - pointer to channel
+ * output      - NONE
+ * side effects        - message given channel ops
+ *
+ * XXX - We need to rework this a bit, it's a tad ugly. --nenolod
+ */
+static void
+msg_channel_opmod(enum message_type msgtype,
+                 struct Client *client_p, struct Client *source_p,
+                 struct Channel *chptr, const char *text)
+{
+       hook_data_privmsg_channel hdata;
+
+       hdata.msgtype = msgtype;
+       hdata.source_p = source_p;
+       hdata.chptr = chptr;
+       hdata.text = text;
+       hdata.approved = 0;
+
+       call_hook(h_privmsg_channel, &hdata);
+
+       /* memory buffer address may have changed, update pointer */
+       text = hdata.text;
+
+       if (hdata.approved != 0)
+               return;
+
+       /* hook may have reduced the string to nothing. */
+       if (EmptyString(text))
+       {
+               /* could be empty after colour stripping and
+                * that would cause problems later */
+               if(msgtype != MESSAGE_TYPE_NOTICE)
+                       sendto_one(source_p, form_str(ERR_NOTEXTTOSEND), me.name, source_p->name);
+               return;
+       }
+
+       if(chptr->mode.mode & MODE_OPMODERATE &&
+                       (!(chptr->mode.mode & MODE_NOPRIVMSGS) ||
+                        IsMember(source_p, chptr)))
+       {
+               if(!flood_attack_channel(msgtype, source_p, chptr, chptr->chname))
+               {
+                       sendto_channel_opmod(client_p, source_p, chptr,
+                                            cmdname[msgtype], text);
+               }
+       }
+       else
+       {
+               if(msgtype != MESSAGE_TYPE_NOTICE)
                        sendto_one_numeric(source_p, ERR_CANNOTSENDTOCHAN,
                                           form_str(ERR_CANNOTSENDTOCHAN), chptr->chname);
        }
@@ -497,9 +649,8 @@ msg_channel(int p_or_n, const char *command,
 /*
  * msg_channel_flags
  *
- * inputs      - flag 0 if PRIVMSG 1 if NOTICE. RFC 
+ * inputs      - flag 0 if PRIVMSG 1 if NOTICE. RFC
  *               say NOTICE must not auto reply
- *             - pointer to command, "PRIVMSG" or "NOTICE"
  *             - pointer to client_p
  *             - pointer to source_p
  *             - pointer to channel
@@ -509,11 +660,12 @@ msg_channel(int p_or_n, const char *command,
  * side effects        - message given channel either chanop or voice
  */
 static void
-msg_channel_flags(int p_or_n, const char *command, struct Client *client_p,
+msg_channel_flags(enum message_type msgtype, struct Client *client_p,
                  struct Client *source_p, struct Channel *chptr, int flags, const char *text)
 {
        int type;
        char c;
+       hook_data_privmsg_channel hdata;
 
        if(flags & CHFL_VOICE)
        {
@@ -528,19 +680,38 @@ msg_channel_flags(int p_or_n, const char *command, struct Client *client_p,
 
        if(MyClient(source_p))
        {
-               /* idletime shouldnt be reset by notice --fl */
-               if(p_or_n != NOTICE)
+               /* idle time shouldn't be reset by notices --fl */
+               if(msgtype != MESSAGE_TYPE_NOTICE)
                        source_p->localClient->last = rb_current_time();
        }
 
+       hdata.msgtype = msgtype;
+       hdata.source_p = source_p;
+       hdata.chptr = chptr;
+       hdata.text = text;
+       hdata.approved = 0;
+
+       call_hook(h_privmsg_channel, &hdata);
+
+       /* memory buffer address may have changed, update pointer */
+       text = hdata.text;
+
+       if (hdata.approved != 0)
+               return;
+
+       if (EmptyString(text))
+       {
+               /* could be empty after colour stripping and
+                * that would cause problems later */
+               if(msgtype != MESSAGE_TYPE_NOTICE)
+                       sendto_one(source_p, form_str(ERR_NOTEXTTOSEND), me.name, source_p->name);
+               return;
+       }
+
        sendto_channel_flags(client_p, type, source_p, chptr, "%s %c%s :%s",
-                            command, c, chptr->chname, text);
+                            cmdname[msgtype], c, chptr->chname, text);
 }
 
-#define PREV_FREE_TARGET(x) ((FREE_TARGET(x) == 0) ? 9 : FREE_TARGET(x) - 1)
-#define PREV_TARGET(i) ((i == 0) ? i = 9 : --i)
-#define NEXT_TARGET(i) ((i == 9) ? i = 0 : ++i)
-
 static void
 expire_tgchange(void *unused)
 {
@@ -561,83 +732,38 @@ expire_tgchange(void *unused)
        }
 }
 
-static int
-add_target(struct Client *source_p, struct Client *target_p)
+static void
+echo_msg(struct Client *source_p, struct Client *target_p,
+               enum message_type msgtype, const char *text)
 {
-       int i, j;
-       uint32_t hashv;
-
-       /* can msg themselves or services without using any target slots */
-       if(source_p == target_p || IsService(target_p))
-               return 1;
-
-       /* special condition for those who have had PRIVMSG crippled to allow them
-        * to talk to IRCops still.
-        *
-        * XXX: is this controversial?
-        */
-       if(source_p->localClient->target_last > rb_current_time() && IsOper(target_p))
-               return 1;
-
-       hashv = fnv_hash_upper((const unsigned char *)use_id(target_p), 32);
-
-       if(USED_TARGETS(source_p))
+       if (MyClient(target_p))
        {
-               /* hunt for an existing target */
-               for(i = PREV_FREE_TARGET(source_p), j = USED_TARGETS(source_p);
-                   j; --j, PREV_TARGET(i))
-               {
-                       if(source_p->localClient->targets[i] == hashv)
-                               return 1;
-               }
-
-               /* first message after connect, we may only start clearing
-                * slots after this message --anfl
-                */
-               if(!IsTGChange(source_p))
-               {
-                       SetTGChange(source_p);
-                       source_p->localClient->target_last = rb_current_time();
-               }
-               /* clear as many targets as we can */
-               else if((i = (rb_current_time() - source_p->localClient->target_last) / 60))
-               {
-                       if(i > USED_TARGETS(source_p))
-                               USED_TARGETS(source_p) = 0;
-                       else
-                               USED_TARGETS(source_p) -= i;
+               if (!IsCapable(target_p, CLICAP_ECHO_MESSAGE))
+                       return;
 
-                       source_p->localClient->target_last = rb_current_time();
-               }
-               /* cant clear any, full target list */
-               else if(USED_TARGETS(source_p) == 10)
-               {
-                       ServerStats->is_tgch++;
-                       add_tgchange(source_p->sockhost);
-                       return 0;
-               }
-       }
-       /* no targets in use, reset their target_last so that they cant
-        * abuse a long idle to get targets back more quickly
-        */
-       else
-       {
-               source_p->localClient->target_last = rb_current_time();
-               SetTGChange(source_p);
+               sendto_one(target_p, ":%s!%s@%s %s %s :%s",
+                               target_p->name, target_p->username, target_p->host,
+                               cmdname[msgtype],
+                               source_p->name,
+                               text);
+               return;
        }
 
-       source_p->localClient->targets[FREE_TARGET(source_p)] = hashv;
-       NEXT_TARGET(FREE_TARGET(source_p));
-       ++USED_TARGETS(source_p);
-       return 1;
+       if (!(target_p->from->serv->caps & CAP_ECHO))
+               return;
+
+       sendto_one(target_p, ":%s ECHO %c %s :%s",
+               use_id(source_p),
+               msgtype == MESSAGE_TYPE_PRIVMSG ? 'P' : 'N',
+               use_id(target_p),
+               text);
 }
 
 /*
  * msg_client
  *
- * inputs      - flag 0 if PRIVMSG 1 if NOTICE. RFC 
+ * inputs      - flag 0 if PRIVMSG 1 if NOTICE. RFC
  *               say NOTICE must not auto reply
- *             - pointer to command, "PRIVMSG" or "NOTICE"
  *             - pointer to source_p source (struct Client *)
  *             - pointer to target_p target (struct Client *)
  *             - pointer to text
@@ -645,22 +771,28 @@ add_target(struct Client *source_p, struct Client *target_p)
  * side effects        - message given channel either chanop or voice
  */
 static void
-msg_client(int p_or_n, const char *command,
+msg_client(enum message_type msgtype,
           struct Client *source_p, struct Client *target_p, const char *text)
 {
+       int do_floodcount = 0;
+       hook_data_privmsg_user hdata;
+
        if(MyClient(source_p))
        {
-               /* reset idle time for message only if its not to self 
-                * and its not a notice */
-               if(p_or_n != NOTICE)
+               /* idle time shouldn't be reset by notices --fl */
+               if(msgtype != MESSAGE_TYPE_NOTICE)
                        source_p->localClient->last = rb_current_time();
 
+               /* auto cprivmsg/cnotice */
+               do_floodcount = !IsOperGeneral(source_p) &&
+                       !find_allowing_channel(source_p, target_p);
+
                /* target change stuff, dont limit ctcp replies as that
                 * would allow people to start filling up random users
                 * targets just by ctcping them
                 */
-               if((p_or_n != NOTICE || *text != '\001') &&
-                  ConfigFileEntry.target_change && !IsOper(source_p))
+               if((msgtype != MESSAGE_TYPE_NOTICE || *text != '\001') &&
+                  ConfigFileEntry.target_change && do_floodcount)
                {
                        if(!add_target(source_p, target_p))
                        {
@@ -669,85 +801,58 @@ msg_client(int p_or_n, const char *command,
                                return;
                        }
                }
+
+               if (do_floodcount && msgtype == MESSAGE_TYPE_NOTICE && *text == '\001' &&
+                               target_p->large_ctcp_sent + LARGE_CTCP_TIME >= rb_current_time())
+                       do_floodcount = 0;
+
+               if (do_floodcount &&
+                               flood_attack_client(msgtype, source_p, target_p))
+                       return;
        }
        else if(source_p->from == target_p->from)
        {
-               sendto_realops_snomask(SNO_DEBUG, L_ALL,
+               sendto_realops_snomask(SNO_DEBUG, L_NETWIDE,
                                     "Send message to %s[%s] dropped from %s(Fake Dir)",
                                     target_p->name, target_p->from->name, source_p->name);
                return;
        }
 
-       if(MyConnect(source_p) && (p_or_n != NOTICE) && target_p->user && target_p->user->away)
+       if(MyConnect(source_p) && (msgtype != MESSAGE_TYPE_NOTICE) && target_p->user && target_p->user->away)
                sendto_one_numeric(source_p, RPL_AWAY, form_str(RPL_AWAY),
                                   target_p->name, target_p->user->away);
 
-       if(MyClient(target_p))
-       {
-               /* XXX Controversial? allow opers always to send through a +g */
-               if(!IsServer(source_p) && (IsSetCallerId(target_p) ||
-                                       (IsSetRegOnlyMsg(target_p) && !source_p->user->suser[0])))
-               {
-                       /* Here is the anti-flood bot/spambot code -db */
-                       if(accept_message(source_p, target_p) || IsOper(source_p))
-                       {
-                               sendto_one(target_p, ":%s!%s@%s %s %s :%s",
-                                          source_p->name,
-                                          source_p->username,
-                                          source_p->host, command, target_p->name, text);
-                       }
-                       else if (IsSetRegOnlyMsg(target_p) && !source_p->user->suser[0])
-                       {
-                               if (p_or_n != NOTICE)
-                                       sendto_one_numeric(source_p, ERR_NONONREG,
-                                                       form_str(ERR_NONONREG),
-                                                       target_p->name);
-                               /* Only so opers can watch for floods */
-                               (void) flood_attack_client(p_or_n, source_p, target_p);
-                       }
-                       else
-                       {
-                               /* check for accept, flag recipient incoming message */
-                               if(p_or_n != NOTICE)
-                               {
-                                       sendto_one_numeric(source_p, ERR_TARGUMODEG,
-                                                          form_str(ERR_TARGUMODEG),
-                                                          target_p->name);
-                               }
+       hdata.msgtype = msgtype;
+       hdata.source_p = source_p;
+       hdata.target_p = target_p;
+       hdata.text = text;
+       hdata.approved = 0;
 
-                               if((target_p->localClient->last_caller_id_time +
-                                   ConfigFileEntry.caller_id_wait) < rb_current_time())
-                               {
-                                       if(p_or_n != NOTICE)
-                                               sendto_one_numeric(source_p, RPL_TARGNOTIFY,
-                                                                  form_str(RPL_TARGNOTIFY),
-                                                                  target_p->name);
+       call_hook(h_privmsg_user, &hdata);
 
-                                       sendto_one(target_p, form_str(RPL_UMODEGMSG),
-                                                  me.name, target_p->name, source_p->name,
-                                                  source_p->username, source_p->host);
+       /* buffer location may have changed. */
+       text = hdata.text;
 
-                                       target_p->localClient->last_caller_id_time = rb_current_time();
-                               }
-                               /* Only so opers can watch for floods */
-                               (void) flood_attack_client(p_or_n, source_p, target_p);
-                       }
-               }
-               else
+       if (hdata.approved != 0)
+               return;
+
+       if(MyClient(target_p))
+       {
+               if (EmptyString(text))
                {
-                       /* If the client is remote, we dont perform a special check for
-                        * flooding.. as we wouldnt block their message anyway.. this means
-                        * we dont give warnings.. we then check if theyre opered 
-                        * (to avoid flood warnings), lastly if theyre our client
-                        * and flooding    -- fl */
-                       if(!MyClient(source_p) || IsOper(source_p) ||
-                          !flood_attack_client(p_or_n, source_p, target_p))
-                               sendto_anywhere(target_p, source_p, command, ":%s", text);
+                       /* could be empty after colour stripping and
+                        * that would cause problems later */
+                       if(msgtype != MESSAGE_TYPE_NOTICE)
+                               sendto_one(source_p, form_str(ERR_NOTEXTTOSEND), me.name, source_p->name);
+                       return;
                }
+
+               add_reply_target(target_p, source_p);
+               sendto_anywhere(target_p, source_p, cmdname[msgtype], ":%s", text);
+               echo_msg(target_p, source_p, msgtype, text);
        }
-       else if(!MyClient(source_p) || IsOper(source_p) ||
-               !flood_attack_client(p_or_n, source_p, target_p))
-               sendto_anywhere(target_p, source_p, command, ":%s", text);
+       else
+               sendto_anywhere(target_p, source_p, cmdname[msgtype], ":%s", text);
 
        return;
 }
@@ -755,115 +860,63 @@ msg_client(int p_or_n, const char *command,
 /*
  * flood_attack_client
  * inputs       - flag 0 if PRIVMSG 1 if NOTICE. RFC
- *                say NOTICE must not auto reply
- *              - pointer to source Client 
+ *                says NOTICE must not auto reply
+ *              - pointer to source Client
  *             - pointer to target Client
- * output      - 1 if target is under flood attack
+ * output      - true if target is under flood attack
  * side effects        - check for flood attack on target target_p
  */
-static int
-flood_attack_client(int p_or_n, struct Client *source_p, struct Client *target_p)
+static bool
+flood_attack_client(enum message_type msgtype, struct Client *source_p, struct Client *target_p)
 {
        int delta;
 
-       if(GlobalSetOptions.floodcount && MyConnect(target_p) && IsClient(source_p))
+       /* Services could get many messages legitimately and
+        * can be messaged without rate limiting via aliases
+        * and msg user@server.
+        * -- jilles
+        */
+       if(GlobalSetOptions.floodcount && IsClient(source_p) && source_p != target_p && !IsService(target_p) && !HasPrivilege(target_p, "oper:free_target"))
        {
-               if((target_p->localClient->first_received_message_time + 1) < rb_current_time())
+               if((target_p->first_received_message_time + 1) < rb_current_time())
                {
-                       delta = rb_current_time() - target_p->localClient->first_received_message_time;
-                       target_p->localClient->received_number_of_privmsgs -= delta;
-                       target_p->localClient->first_received_message_time = rb_current_time();
-                       if(target_p->localClient->received_number_of_privmsgs <= 0)
+                       delta = rb_current_time() - target_p->first_received_message_time;
+                       target_p->received_number_of_privmsgs -= delta;
+                       target_p->first_received_message_time = rb_current_time();
+                       if(target_p->received_number_of_privmsgs <= 0)
                        {
-                               target_p->localClient->received_number_of_privmsgs = 0;
-                               target_p->localClient->flood_noticed = 0;
+                               target_p->received_number_of_privmsgs = 0;
+                               target_p->flood_noticed = 0;
                        }
                }
 
-               if((target_p->localClient->received_number_of_privmsgs >=
-                   GlobalSetOptions.floodcount) || target_p->localClient->flood_noticed)
+               if((target_p->received_number_of_privmsgs >=
+                   GlobalSetOptions.floodcount) || target_p->flood_noticed)
                {
-                       if(target_p->localClient->flood_noticed == 0)
+                       if(target_p->flood_noticed == 0)
                        {
                                sendto_realops_snomask(SNO_BOTS, L_NETWIDE,
                                                     "Possible Flooder %s[%s@%s] on %s target: %s",
                                                     source_p->name, source_p->username,
                                                     source_p->orighost,
                                                     source_p->servptr->name, target_p->name);
-                               target_p->localClient->flood_noticed = 1;
+                               target_p->flood_noticed = 1;
                                /* add a bit of penalty */
-                               target_p->localClient->received_number_of_privmsgs += 2;
+                               target_p->received_number_of_privmsgs += 2;
                        }
-                       if(MyClient(source_p) && (p_or_n != NOTICE))
+                       if(MyClient(source_p) && (msgtype != MESSAGE_TYPE_NOTICE))
                                sendto_one(source_p,
                                           ":%s NOTICE %s :*** Message to %s throttled due to flooding",
                                           me.name, source_p->name, target_p->name);
-                       return 1;
+                       return true;
                }
                else
-                       target_p->localClient->received_number_of_privmsgs++;
+                       target_p->received_number_of_privmsgs++;
        }
 
-       return 0;
+       return false;
 }
 
-/*
- * flood_attack_channel
- * inputs       - flag 0 if PRIVMSG 1 if NOTICE. RFC
- *                says NOTICE must not auto reply
- *              - pointer to source Client 
- *             - pointer to target channel
- * output      - 1 if target is under flood attack
- * side effects        - check for flood attack on target chptr
- */
-static int
-flood_attack_channel(int p_or_n, struct Client *source_p, struct Channel *chptr, char *chname)
-{
-       int delta;
-
-       if(GlobalSetOptions.floodcount && MyClient(source_p))
-       {
-               if((chptr->first_received_message_time + 1) < rb_current_time())
-               {
-                       delta = rb_current_time() - chptr->first_received_message_time;
-                       chptr->received_number_of_privmsgs -= delta;
-                       chptr->first_received_message_time = rb_current_time();
-                       if(chptr->received_number_of_privmsgs <= 0)
-                       {
-                               chptr->received_number_of_privmsgs = 0;
-                               chptr->flood_noticed = 0;
-                       }
-               }
-
-               if((chptr->received_number_of_privmsgs >= GlobalSetOptions.floodcount)
-                  || chptr->flood_noticed)
-               {
-                       if(chptr->flood_noticed == 0)
-                       {
-                               sendto_realops_snomask(SNO_BOTS, *chptr->chname == '&' ? L_ALL : L_NETWIDE,
-                                                    "Possible Flooder %s[%s@%s] on %s target: %s",
-                                                    source_p->name, source_p->username,
-                                                    source_p->orighost,
-                                                    source_p->servptr->name, chptr->chname);
-                               chptr->flood_noticed = 1;
-
-                               /* Add a bit of penalty */
-                               chptr->received_number_of_privmsgs += 2;
-                       }
-                       if(MyClient(source_p) && (p_or_n != NOTICE))
-                               sendto_one(source_p,
-                                          ":%s NOTICE %s :*** Message to %s throttled due to flooding",
-                                          me.name, source_p->name, chptr->chname);
-                       return 1;
-               }
-               else
-                       chptr->received_number_of_privmsgs++;
-       }
-
-       return 0;
-}
-
-
 /*
  * handle_special
  *
@@ -881,14 +934,11 @@ flood_attack_channel(int p_or_n, struct Client *source_p, struct Channel *chptr,
  *               This disambiguates the syntax.
  */
 static void
-handle_special(int p_or_n, const char *command, struct Client *client_p,
+handle_special(enum message_type msgtype, struct Client *client_p,
               struct Client *source_p, const char *nick, const char *text)
 {
        struct Client *target_p;
-       char *host;
        char *server;
-       char *s;
-       int count;
 
        /* user[%host]@server addressed?
         * NOTE: users can send to user@server, but not user%host@server
@@ -903,8 +953,6 @@ handle_special(int p_or_n, const char *command, struct Client *client_p,
                        return;
                }
 
-               count = 0;
-
                if(!IsOper(source_p))
                {
                        if(strchr(nick, '%') || (strncmp(nick, "opers", 5) == 0))
@@ -919,43 +967,27 @@ handle_special(int p_or_n, const char *command, struct Client *client_p,
                if(!IsMe(target_p))
                {
                        sendto_one(target_p, ":%s %s %s :%s",
-                                  get_id(source_p, target_p), command, nick, text);
+                                  get_id(source_p, target_p), cmdname[msgtype], nick, text);
                        return;
                }
 
-               *server = '\0';
-
-               if((host = strchr(nick, '%')) != NULL)
-                       *host++ = '\0';
-
                /* Check if someones msg'ing opers@our.server */
-               if(strcmp(nick, "opers") == 0)
+               if(strncmp(nick, "opers@", 6) == 0)
                {
                        sendto_realops_snomask(SNO_GENERAL, L_ALL, "To opers: From: %s: %s",
                                             source_p->name, text);
                        return;
                }
 
-               /*
-                * Look for users which match the destination host
-                * (no host == wildcard) and if one and one only is
-                * found connected to me, deliver message!
+               /* This was not very useful except for bypassing certain
+                * restrictions. Note that we still allow sending to
+                * remote servers this way, for messaging pseudoservers
+                * securely whether they have a service{} block or not.
+                * -- jilles
                 */
-               target_p = find_userhost(nick, host, &count);
-
-               if(target_p != NULL)
-               {
-                       if(server != NULL)
-                               *server = '@';
-                       if(host != NULL)
-                               *--host = '%';
-
-                       if(count == 1)
-                               sendto_anywhere(target_p, source_p, command, ":%s", text);
-                       else
-                               sendto_one(source_p, form_str(ERR_TOOMANYTARGETS),
-                                          get_id(&me, source_p), get_id(source_p, source_p), nick);
-               }
+               sendto_one_numeric(source_p, ERR_NOSUCHNICK,
+                                  form_str(ERR_NOSUCHNICK), nick);
+               return;
        }
 
        /*
@@ -972,7 +1004,7 @@ handle_special(int p_or_n, const char *command, struct Client *client_p,
                {
                        sendto_one(source_p,
                                   ":%s NOTICE %s :The command %s %s is no longer supported, please use $%s",
-                                  me.name, source_p->name, command, nick, nick);
+                                  me.name, source_p->name, cmdname[msgtype], nick, nick);
                        return;
                }
 
@@ -983,61 +1015,20 @@ handle_special(int p_or_n, const char *command, struct Client *client_p,
                        return;
                }
 
-               if((s = strrchr(nick, '.')) == NULL)
-               {
-                       sendto_one_numeric(source_p, ERR_NOTOPLEVEL,
-                                          form_str(ERR_NOTOPLEVEL), nick);
-                       return;
-               }
-               while(*++s)
-                       if(*s == '.' || *s == '*' || *s == '?')
-                               break;
-               if(*s == '*' || *s == '?')
+               if(MyClient(source_p))
                {
-                       sendto_one_numeric(source_p, ERR_WILDTOPLEVEL,
-                                          form_str(ERR_WILDTOPLEVEL), nick);
-                       return;
+                       sendto_realops_snomask(SNO_GENERAL, L_ALL | L_NETWIDE, "%s sent mass-%s to %s: %s",
+                                       get_oper_name(source_p),
+                                       msgtype == MESSAGE_TYPE_PRIVMSG ? "privmsg" : "notice",
+                                       nick, text);
                }
 
                sendto_match_butone(IsServer(client_p) ? client_p : NULL, source_p,
                                    nick + 1,
                                    (*nick == '#') ? MATCH_HOST : MATCH_SERVER,
-                                   "%s $%s :%s", command, nick, text);
+                                   "%s $%s :%s", cmdname[msgtype], nick, text);
+               if (msgtype != MESSAGE_TYPE_NOTICE && *text == '\001')
+                       source_p->large_ctcp_sent = rb_current_time();
                return;
        }
 }
-
-/*
- * find_userhost - find a user@host (server or user).
- * inputs       - user name to look for
- *              - host name to look for
- *             - pointer to count of number of matches found
- * outputs     - pointer to client if found
- *             - count is updated
- * side effects        - none
- *
- */
-static struct Client *
-find_userhost(const char *user, const char *host, int *count)
-{
-       struct Client *c2ptr;
-       struct Client *res = NULL;
-       char *u = LOCAL_COPY(user);
-       rb_dlink_node *ptr;
-       *count = 0;
-       if(collapse(u) != NULL)
-       {
-               RB_DLINK_FOREACH(ptr, global_client_list.head)
-               {
-                       c2ptr = ptr->data;
-                       if(!MyClient(c2ptr))    /* implies mine and an user */
-                               continue;
-                       if((!host || match(host, c2ptr->host)) && irccmp(u, c2ptr->username) == 0)
-                       {
-                               (*count)++;
-                               res = c2ptr;
-                       }
-               }
-       }
-       return (res);
-}