diff options
Diffstat (limited to 'modules/m_accept.c')
-rw-r--r-- | modules/m_accept.c | 221 |
1 files changed, 221 insertions, 0 deletions
diff --git a/modules/m_accept.c b/modules/m_accept.c new file mode 100644 index 0000000..3fff6c1 --- /dev/null +++ b/modules/m_accept.c @@ -0,0 +1,221 @@ +/* + * ircd-hybrid: an advanced Internet Relay Chat Daemon(ircd). + * + * Copyright (C) 2002 by the past and present ircd coders, and others. + * + * 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 2 of the License, 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 + * USA + */ + +/*! \file m_accept.c + * \brief Includes required functions for processing the ACCEPT command. + * \version $Id$ + */ + +#include "stdinc.h" +#include "client.h" +#include "irc_string.h" +#include "sprintf_irc.h" +#include "ircd.h" +#include "list.h" +#include "numeric.h" +#include "conf.h" +#include "s_serv.h" +#include "send.h" +#include "parse.h" +#include "modules.h" + + +/*! \brief Creates and sends a list of nick!user\@host masks a Client + * has on its acceptlist. + * + * \param source_p The actual Client the list will be sent to. + */ +static void +list_accepts(struct Client *source_p) +{ + int len = 0; + char nicks[IRCD_BUFSIZE] = { '\0' }; + char *t = nicks; + const dlink_node *ptr = NULL; + + len = strlen(me.name) + strlen(source_p->name) + 12; + + DLINK_FOREACH(ptr, source_p->localClient->acceptlist.head) + { + const struct split_nuh_item *accept_p = ptr->data; + size_t masklen = strlen(accept_p->nickptr) + + strlen(accept_p->userptr) + + strlen(accept_p->hostptr) + 2 /* !@ */ ; + + if ((t - nicks) + masklen + len > IRCD_BUFSIZE) + { + *(t - 1) = '\0'; + sendto_one(source_p, form_str(RPL_ACCEPTLIST), + me.name, source_p->name, nicks); + t = nicks; + } + + t += ircsprintf(t, "%s!%s@%s ", + accept_p->nickptr, + accept_p->userptr, accept_p->hostptr); + } + + if (nicks[0] != '\0') + { + *(t - 1) = '\0'; + sendto_one(source_p, form_str(RPL_ACCEPTLIST), + me.name, source_p->name, nicks); + } + + sendto_one(source_p, form_str(RPL_ENDOFACCEPT), + me.name, source_p->name); +} + +/*! \brief Allocates and adds a split_nuh_item holding a nick!user\@host + * mask to a Client's acceptlist. + * + * \param nuh A split_nuh_item already prepared with required masks. + * \param source_p The actual Client the new accept is added to. + */ +static void +add_accept(const struct split_nuh_item *nuh, struct Client *source_p) +{ + struct split_nuh_item *accept_p = MyMalloc(sizeof(*accept_p)); + + DupString(accept_p->nickptr, nuh->nickptr); + DupString(accept_p->userptr, nuh->userptr); + DupString(accept_p->hostptr, nuh->hostptr); + + dlinkAdd(accept_p, &accept_p->node, &source_p->localClient->acceptlist); + + list_accepts(source_p); +} + +/*! \brief ACCEPT command handler + * + * \param client_p Pointer to allocated Client struct with physical connection + * to this server, i.e. with an open socket connected. + * \param source_p Pointer to allocated Client struct from which the message + * originally comes from. This can be a local or remote client. + * \param parc Integer holding the number of supplied arguments. + * \param parv Argument vector where parv[0] .. parv[parc-1] are non-NULL + * pointers. + * \note Valid arguments for this command are: + * - parv[0] = sender prefix + * - parv[1] = list of masks to be accepted or removed (optional) + */ +static void +m_accept(struct Client *client_p, struct Client *source_p, + int parc, char *parv[]) +{ + char *mask = NULL; + char *p = NULL; + char nick[NICKLEN + 1]; + char user[USERLEN + 1]; + char host[HOSTLEN + 1]; + struct split_nuh_item nuh; + struct split_nuh_item *accept_p = NULL; + + if (EmptyString(parv[1]) || !irccmp(parv[1], "*")) + { + list_accepts(source_p); + return; + } + + for (mask = strtoken(&p, parv[1], ","); mask != NULL; + mask = strtoken(&p, NULL, ",")) + { + if (*mask == '-' && *++mask != '\0') + { + nuh.nuhmask = mask; + nuh.nickptr = nick; + nuh.userptr = user; + nuh.hostptr = host; + + nuh.nicksize = sizeof(nick); + nuh.usersize = sizeof(user); + nuh.hostsize = sizeof(host); + + split_nuh(&nuh); + + if ((accept_p = find_accept(nick, user, host, source_p, 0)) == NULL) + { + sendto_one(source_p, form_str(ERR_ACCEPTNOT), + me.name, source_p->name, nick, user, host); + continue; + } + + del_accept(accept_p, source_p); + } + else if (*mask != '\0') + { + if (dlink_list_length(&source_p->localClient->acceptlist) >= + ConfigFileEntry.max_accept) + { + sendto_one(source_p, form_str(ERR_ACCEPTFULL), + me.name, source_p->name); + return; + } + + nuh.nuhmask = mask; + nuh.nickptr = nick; + nuh.userptr = user; + nuh.hostptr = host; + + nuh.nicksize = sizeof(nick); + nuh.usersize = sizeof(user); + nuh.hostsize = sizeof(host); + + split_nuh(&nuh); + + if ((accept_p = find_accept(nick, user, host, source_p, 0)) != NULL) + { + sendto_one(source_p, form_str(ERR_ACCEPTEXIST), + me.name, source_p->name, nick, user, host); + continue; + } + + add_accept(&nuh, source_p); + } + } +} + +static struct Message accept_msgtab = { + "ACCEPT", 0, 0, 0, MAXPARA, MFLG_SLOW, 0, + { m_unregistered, m_accept, m_ignore, m_ignore, m_accept, m_ignore } +}; + +static void +module_init(void) +{ + mod_add_cmd(&accept_msgtab); +} + +static void +module_exit(void) +{ + mod_del_cmd(&accept_msgtab); +} + +struct module module_entry = { + .node = { NULL, NULL, NULL }, + .name = NULL, + .version = "$Revision$", + .handle = NULL, + .modinit = module_init, + .modexit = module_exit, + .flags = 0 +}; |