X-Git-Url: https://jfr.im/git/solanum.git/blobdiff_plain/0f021a8c1d618a1ca0997bb1b76bf255ba0a653f..d295a3986d7bb97c3b36facb1c1d94cbc35d50a5:/modules/core/m_message.c diff --git a/modules/core/m_message.c b/modules/core/m_message.c index 95a68fc7..8f9c7aa4 100644 --- a/modules/core/m_message.c +++ b/modules/core/m_message.c @@ -20,22 +20,19 @@ * 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" @@ -43,10 +40,18 @@ #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; @@ -66,17 +71,21 @@ 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, ¬ice_msgtab, NULL }; +mapi_clist_av1 message_clist[] = { &privmsg_msgtab, ¬ice_msgtab, &echo_msgtab, NULL }; -DECLARE_MODULE_AV1(message, modinit, moddeinit, message_clist, NULL, NULL, "$Revision: 3173 $"); +DECLARE_MODULE_AV2(message, modinit, moddeinit, message_clist, NULL, NULL, NULL, NULL, message_desc); struct entity { @@ -85,38 +94,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 +153,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 +203,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 +213,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 +271,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 +286,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 +316,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 +368,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 +385,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 +404,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 +413,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 +471,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 +498,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) + 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))) + { + 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_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); + } +} +/* + * 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))) { - /* 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(!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); } @@ -497,9 +642,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 +653,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) { @@ -529,18 +674,37 @@ 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) + 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 +725,35 @@ 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; + 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 +761,29 @@ 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 + /* reset idle time for message only if its not to self * and its not a notice */ - if(p_or_n != NOTICE) + 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,6 +792,14 @@ 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) { @@ -678,76 +809,41 @@ msg_client(int p_or_n, const char *command, 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 +851,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)) { - 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; - } - else - target_p->localClient->received_number_of_privmsgs++; - } - - return 0; -} - -/* - * 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; + return true; } else - chptr->received_number_of_privmsgs++; + target_p->received_number_of_privmsgs++; } - return 0; + return false; } - /* * handle_special * @@ -881,14 +925,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 +944,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 +958,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 +995,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 +1006,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); -}