/* * IRC - Internet Relay Chat, ircd/m_silence.c * Copyright (C) 1990 Jarkko Oikarinen and * University of Oulu, Computing Center * * See file AUTHORS in IRC package for additional names of * the programmers. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 1, or (at your option) * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ /** @file * @brief Handlers for SILENCE command. * @version $Id: m_silence.c 1937 2010-01-07 03:06:13Z entrope $ */ #include "config.h" #include "channel.h" #include "client.h" #include "hash.h" #include "ircd.h" #include "ircd_features.h" #include "ircd_log.h" #include "ircd_reply.h" #include "ircd_snprintf.h" #include "ircd_string.h" #include "list.h" #include "msg.h" #include "numeric.h" #include "numnicks.h" #include "s_user.h" #include "send.h" #include "struct.h" /* #include -- Now using assert in ircd_log.h */ #include #include /** Attempt to apply a SILENCE update to a user. * * Silences are propagated lazily between servers to save on bandwidth * and remote memory. Any removal and any silence exception must be * propagated until a server has not seen the mask being removed or * has no positive silences for the user. * * @param[in] sptr Client to update. * @param[in] mask Single silence mask to apply, optionally preceded by '+' or '-' and maybe '~'. * @return The new ban entry on success, NULL on failure. */ static struct Ban * apply_silence(struct Client *sptr, char *mask) { struct Ban *sile; int flags; int res; char orig_mask[NICKLEN+USERLEN+HOSTLEN+3]; assert(mask && mask[0]); /* Check for add or remove. */ if (mask[0] == '-') { flags = BAN_DEL; mask++; } else if (mask[0] == '+') { flags = BAN_ADD; mask++; } else flags = BAN_ADD; /* Check for being an exception. */ if (mask[0] == '~') { flags |= BAN_EXCEPTION; mask++; } /* Make the silence and set additional flags. */ ircd_strncpy(orig_mask, mask, sizeof(orig_mask)); sile = make_ban(pretty_mask(mask)); sile->flags |= flags; /* If they're a local user trying to ban too broad a mask, forbid it. */ if (MyUser(sptr) && (sile->flags & BAN_IPMASK) && sile->addrbits > 0 && sile->addrbits < (irc_in_addr_is_ipv4(&sile->address) ? 112 : 32)) { send_reply(sptr, ERR_MASKTOOWIDE, orig_mask); free_ban(sile); return NULL; } /* Apply it to the silence list. */ res = apply_ban(&cli_user(sptr)->silence, sile, 1); return res ? NULL : sile; } /** Apply and send silence updates for a user. * @param[in] sptr Client whose silence list has been updated. * @param[in] silences Comma-separated list of silence updates. * @param[in] dest Direction to send updates in (NULL for broadcast). */ static void forward_silences(struct Client *sptr, char *silences, struct Client *dest) { struct Ban *accepted[MAXPARA], *sile, **plast; char *cp, *p, buf[BUFSIZE]; size_t ac_count, buf_used, slen, ii; /* Split the list of silences and try to apply each one in turn. */ for (cp = ircd_strtok(&p, silences, ","), ac_count = 0; cp && (ac_count < MAXPARA); cp = ircd_strtok(&p, 0, ",")) { if ((sile = apply_silence(sptr, cp))) accepted[ac_count++] = sile; } if (MyUser(sptr)) { size_t siles, maxsiles, totlength, maxlength, jj; /* Check that silence count and total length are permitted. */ maxsiles = feature_int(FEAT_MAXSILES); maxlength = maxsiles * feature_int(FEAT_AVBANLEN); siles = totlength = 0; /* Count number of current silences and their total length. */ plast = &cli_user(sptr)->silence; for (sile = cli_user(sptr)->silence; sile; sile = sile->next) { if (sile->flags & (BAN_OVERLAPPED | BAN_ADD | BAN_DEL)) continue; siles++; totlength += strlen(sile->banstr); plast = &sile->next; } for (ii = jj = 0; ii < ac_count; ++ii) { sile = accepted[ii]; /* If the update is being added, and we would exceed the maximum * count or length, drop the update. */ if (!(sile->flags & (BAN_OVERLAPPED | BAN_DEL))) { slen = strlen(sile->banstr); if ((siles >= maxsiles) || (totlength + slen >= maxlength)) { *plast = NULL; if (MyUser(sptr)) send_reply(sptr, ERR_SILELISTFULL, accepted[ii]->banstr); free_ban(accepted[ii]); continue; } /* Update counts. */ siles++; totlength += slen; plast = &sile->next; } /* Store the update. */ accepted[jj++] = sile; } /* Write back the number of accepted updates. */ ac_count = jj; /* Send the silence update list, including overlapped silences (to * make it easier on clients). */ buf_used = 0; for (sile = cli_user(sptr)->silence; sile; sile = sile->next) { char ch; if (sile->flags & (BAN_OVERLAPPED | BAN_DEL)) ch = '-'; else if (sile->flags & BAN_ADD) ch = '+'; else continue; slen = strlen(sile->banstr); if (buf_used + slen + 4 > 400) { buf[buf_used] = '\0'; sendcmdto_one(sptr, CMD_SILENCE, sptr, "%s", buf); buf_used = 0; } if (buf_used) buf[buf_used++] = ','; buf[buf_used++] = ch; if (sile->flags & BAN_EXCEPTION) buf[buf_used++] = '~'; memcpy(buf + buf_used, sile->banstr, slen); buf_used += slen; } if (buf_used > 0) { buf[buf_used] = '\0'; sendcmdto_one(sptr, CMD_SILENCE, sptr, "%s", buf); buf_used = 0; } } /* Forward any silence removals or exceptions updates to other * servers if the user has positive silences. */ if (!dest || !MyUser(dest)) { for (ii = buf_used = 0; ii < ac_count; ++ii) { char ch; sile = accepted[ii]; if (sile->flags & BAN_OVERLAPPED) continue; else if (sile->flags & BAN_DEL) ch = '-'; else if (sile->flags & BAN_ADD) { if (!(sile->flags & BAN_EXCEPTION) && !feature_bool(FEAT_SILENCE_CHANMSGS)) continue; ch = '+'; } else continue; slen = strlen(sile->banstr); if (buf_used + slen + 4 > 400) { buf[buf_used] = '\0'; if (dest) sendcmdto_one(sptr, CMD_SILENCE, dest, "%C %s", dest, buf); else sendcmdto_serv_butone(sptr, CMD_SILENCE, sptr, "* %s", buf); buf_used = 0; } if (buf_used) buf[buf_used++] = ','; buf[buf_used++] = ch; if (sile->flags & BAN_EXCEPTION) buf[buf_used++] = '~'; memcpy(buf + buf_used, sile->banstr, slen); buf_used += slen; } if (buf_used > 0) { buf[buf_used] = '\0'; if (dest) sendcmdto_one(sptr, CMD_SILENCE, dest, "%C %s", dest, buf); else sendcmdto_serv_butone(sptr, CMD_SILENCE, sptr, "* %s", buf); buf_used = 0; } } /* Remove overlapped and deleted silences from the user's silence * list. Clear BAN_ADD since we're walking the list anyway. */ for (plast = &cli_user(sptr)->silence; (sile = *plast) != NULL; ) { if (sile->flags & (BAN_OVERLAPPED | BAN_DEL)) { *plast = sile->next; free_ban(sile); } else { sile->flags &= ~BAN_ADD; *plast = sile; plast = &sile->next; } } /* Free any silence-deleting updates. */ for (ii = 0; ii < ac_count; ++ii) { if ((accepted[ii]->flags & (BAN_ADD | BAN_DEL)) == BAN_DEL) { free_ban(accepted[ii]); } } } /** Handle a SILENCE command from a local user. * See @ref m_functions for general discussion of parameters. * * \a parv[1] may be any of the following: * \li Omitted or empty, to view your own silence list. * \li Nickname of a user, to view that user's silence list. * \li A comma-separated list of silence updates * * @param[in] cptr Client that sent us the message. * @param[in] sptr Original source of message. * @param[in] parc Number of arguments. * @param[in] parv Argument vector. */ int m_silence(struct Client* cptr, struct Client* sptr, int parc, char* parv[]) { struct Client *acptr; struct Ban *sile; assert(0 != cptr); assert(cptr == sptr); /* See if the user is requesting a silence list. */ acptr = sptr; if (parc < 2 || EmptyString(parv[1]) || (acptr = FindUser(parv[1]))) { if (cli_user(acptr) && ((acptr == sptr) || IsChannelService(acptr))) { for (sile = cli_user(acptr)->silence; sile; sile = sile->next) { send_reply(sptr, RPL_SILELIST, cli_name(acptr), (sile->flags & BAN_EXCEPTION ? "~" : ""), sile->banstr); } } send_reply(sptr, RPL_ENDOFSILELIST, cli_name(acptr)); return 0; } /* The user must be attempting to update their list. */ forward_silences(sptr, parv[1], NULL); return 0; } /** Handle a SILENCE command from a server. * See @ref m_functions for general discussion of parameters. * * \a parv[1] may be one of the following: * \li "*" to indicate a broadcast update (removing a SILENCE) * \li A client numnick that should be specifically SILENCEd. * * \a parv[2] is a comma-separated list of silence updates. * * @param[in] cptr Client that sent us the message. * @param[in] sptr Original source of message. * @param[in] parc Number of arguments. * @param[in] parv Argument vector. */ int ms_silence(struct Client* cptr, struct Client* sptr, int parc, char* parv[]) { if (parc < 3 || EmptyString(parv[2])) return need_more_params(sptr, "SILENCE"); if (IsServer(sptr)) { struct Client *acptr = findNUser(parv[1]); /* User not found to silently return. */ if (!acptr) return 0; if (MyUser(acptr)) forward_silences(acptr, parv[2], NULL); else sendcmdto_one(sptr, CMD_SILENCE, acptr, "%C %s", acptr, parv[2]); return 0; } /* Figure out which silences can be forwarded. */ forward_silences(sptr, parv[2], findNUser(parv[1])); return 0; (void)cptr; }