/* * charybdis: An advanced ircd. * m_list_safelist.c: Version of /list that uses the safelist code. * * Copyright (c) 2006 William Pitcock * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ using namespace ircd; static const char list_desc[] = "Provides the LIST command to clients to view non-hidden channels"; static rb_dlink_list safelisting_clients = { NULL, NULL, 0 }; static struct ev_entry *iterate_clients_ev = NULL; static int _modinit(void); static void _moddeinit(void); static void m_list(struct MsgBuf *, struct Client *, struct Client *, int, const char **); static void mo_list(struct MsgBuf *, struct Client *, struct Client *, int, const char **); static void list_one_channel(struct Client *source_p, chan::chan *chptr, int visible); static void safelist_one_channel(struct Client *source_p, chan::chan *chptr, struct ListClient *params); static void safelist_check_cliexit(hook_data_client_exit * hdata); static void safelist_client_instantiate(struct Client *, struct ListClient *); static void safelist_client_release(struct Client *); static void safelist_iterate_client(struct Client *source_p); static void safelist_iterate_clients(void *unused); static void safelist_channel_named(struct Client *source_p, const char *name, int operspy); struct Message list_msgtab = { "LIST", 0, 0, 0, 0, {mg_unreg, {m_list, 0}, mg_ignore, mg_ignore, mg_ignore, {mo_list, 0}} }; mapi_clist_av1 list_clist[] = { &list_msgtab, NULL }; mapi_hfn_list_av1 list_hfnlist[] = { {"client_exit", (hookfn) safelist_check_cliexit}, {NULL, NULL} }; DECLARE_MODULE_AV2(list, _modinit, _moddeinit, list_clist, NULL, list_hfnlist, NULL, NULL, list_desc); static int _modinit(void) { iterate_clients_ev = rb_event_add("safelist_iterate_clients", safelist_iterate_clients, NULL, 3); /* ELIST=[tokens]: * * M = mask search * N = !mask search * U = user count search (< >) * C = creation time search (C> C<) * T = topic search (T> T<) */ add_isupport("SAFELIST", isupport_string, ""); add_isupport("ELIST", isupport_string, "CTU"); return 0; } static void _moddeinit(void) { rb_event_delete(iterate_clients_ev); delete_isupport("SAFELIST"); delete_isupport("ELIST"); } static void safelist_check_cliexit(hook_data_client_exit * hdata) { /* Cancel the safelist request if we are disconnecting * from the server. That way it doesn't core. :P --nenolod */ if (MyClient(hdata->target) && hdata->target->localClient->safelist_data != NULL) { safelist_client_release(hdata->target); } } /* m_list() * parv[1] = channel * * XXX - With SAFELIST, do we really need to continue pacing? * In theory, the server cannot be lagged by this. --nenolod */ static void m_list(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char *parv[]) { static time_t last_used = 0L; if (source_p->localClient->safelist_data != NULL) { sendto_one_notice(source_p, ":/LIST aborted"); safelist_client_release(source_p); return; } if (parc < 2 || chan::is_name(parv[1])) { /* pace this due to the sheer traffic involved */ if (((last_used + ConfigFileEntry.pace_wait) > rb_current_time())) { sendto_one(source_p, form_str(RPL_LOAD2HI), me.name, source_p->name, "LIST"); sendto_one(source_p, form_str(RPL_LISTEND), me.name, source_p->name); return; } else last_used = rb_current_time(); } mo_list(msgbuf_p, client_p, source_p, parc, parv); } /* mo_list() * parv[1] = channel */ static void mo_list(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char *parv[]) { struct ListClient *params; char *p; char *args = NULL; int i; int operspy = 0; if (source_p->localClient->safelist_data != NULL) { sendto_one_notice(source_p, ":/LIST aborted"); safelist_client_release(source_p); return; } if (parc > 1) { args = LOCAL_COPY(parv[1]); } if (args && *args == '!' && IsOperSpy(source_p)) { args++; report_operspy(source_p, "LIST", args); operspy = 1; } /* Single channel. */ if (args && chan::is_name(args)) { safelist_channel_named(source_p, args, operspy); return; } /* Multiple channels, possibly with parameters. */ params = (ListClient *)rb_malloc(sizeof(struct ListClient)); params->users_min = ConfigChannel.displayed_usercount; params->users_max = INT_MAX; params->operspy = operspy; params->created_min = params->topic_min = params->created_max = params->topic_max = 0; if (args && !EmptyString(args)) { /* Cancel out default minimum. */ params->users_min = 0; for (i = 0; i < 7; i++) { if ((p = strchr(args, ',')) != NULL) *p++ = '\0'; if (*args == '<') { args++; if (rfc1459::is_digit(*args)) { params->users_max = atoi(args); if (params->users_max == 0) params->users_max = INT_MAX; else params->users_max--; } } else if (*args == '>') { args++; if (rfc1459::is_digit(*args)) params->users_min = atoi(args) + 1; else params->users_min = 0; } else if (*args == 'C' || *args == 'c') { args++; if (*args == '>') { /* Creation time earlier than last x minutes. */ args++; if (rfc1459::is_digit(*args)) { params->created_max = rb_current_time() - (60 * atoi(args)); } } else if (*args == '<') { /* Creation time within last x minutes. */ args++; if (rfc1459::is_digit(*args)) { params->created_min = rb_current_time() - (60 * atoi(args)); } } } else if (*args == 'T' || *args == 't') { args++; if (*args == '>') { /* Topic change time earlier than last x minutes. */ args++; if (rfc1459::is_digit(*args)) { params->topic_max = rb_current_time() - (60 * atoi(args)); } } else if (*args == '<') { /* Topic change time within last x minutes. */ args++; if (rfc1459::is_digit(*args)) { params->topic_min = rb_current_time() - (60 * atoi(args)); } } } if (EmptyString(p)) break; else args = p; } } safelist_client_instantiate(source_p, params); } /* * list_one_channel() * * inputs - client pointer, channel pointer, whether normally visible * outputs - none * side effects - a channel is listed */ static void list_one_channel(struct Client *source_p, chan::chan *chptr, int visible) { char topic[TOPICLEN + 1]; if (!chptr->topic.text.empty()) rb_strlcpy(topic, chptr->topic.text.c_str(), sizeof topic); else topic[0] = '\0'; strip_colour(topic); sendto_one(source_p, form_str(RPL_LIST), me.name, source_p->name, visible ? "" : "!", chptr->name.c_str(), rb_dlink_list_length(&chptr->members), topic); } /* * safelist_sendq_exceeded() * * inputs - pointer to client that needs checking * outputs - true if a client has exceeded the reserved * sendq limit, false if not * side effects - none * * When safelisting, we only use half of the SendQ at any * given time. */ static bool safelist_sendq_exceeded(struct Client *client_p) { return rb_linebuf_len(&client_p->localClient->buf_sendq) > (get_sendq(client_p) / 2); } /* * safelist_client_instantiate() * * inputs - pointer to Client to be listed, * pointer to ListClient for params * outputs - none * side effects - the safelist process begins for a * client. * * Please do not ever call this on a non-local client. * If you do, you will get SIGSEGV. */ static void safelist_client_instantiate(struct Client *client_p, struct ListClient *params) { s_assert(MyClient(client_p)); s_assert(params != NULL); client_p->localClient->safelist_data = params; sendto_one(client_p, form_str(RPL_LISTSTART), me.name, client_p->name); /* pop the client onto the queue for processing */ rb_dlinkAddAlloc(client_p, &safelisting_clients); /* give the user some initial data to work with */ safelist_iterate_client(client_p); } /* * safelist_client_release() * * inputs - pointer to Client being listed on * outputs - none * side effects - the client is no longer being * listed */ static void safelist_client_release(struct Client *client_p) { if(!MyClient(client_p)) return; s_assert(MyClient(client_p)); rb_dlinkFindDestroy(client_p, &safelisting_clients); rb_free(client_p->localClient->safelist_data->chname); rb_free(client_p->localClient->safelist_data); client_p->localClient->safelist_data = NULL; sendto_one(client_p, form_str(RPL_LISTEND), me.name, client_p->name); } /* * safelist_channel_named() * * inputs - client pointer, channel name, operspy * outputs - none * side effects - a named channel is listed */ static void safelist_channel_named(struct Client *source_p, const char *name, int operspy) { chan::chan *chptr; char *p; int visible; sendto_one(source_p, form_str(RPL_LISTSTART), me.name, source_p->name); if ((p = (char *)strchr(name, ','))) *p = '\0'; if (*name == '\0') { sendto_one_numeric(source_p, ERR_NOSUCHNICK, form_str(ERR_NOSUCHNICK), name); sendto_one(source_p, form_str(RPL_LISTEND), me.name, source_p->name); return; } chptr = find_channel(name); if (chptr == NULL) { sendto_one_numeric(source_p, ERR_NOSUCHNICK, form_str(ERR_NOSUCHNICK), name); sendto_one(source_p, form_str(RPL_LISTEND), me.name, source_p->name); return; } visible = !secret(chptr) || is_member(chptr, source_p); if (visible || operspy) list_one_channel(source_p, chptr, visible); sendto_one(source_p, form_str(RPL_LISTEND), me.name, source_p->name); return; } /* * safelist_one_channel() * * inputs - client pointer and channel pointer * outputs - none * side effects - a channel is listed if it meets the * requirements */ static void safelist_one_channel(struct Client *source_p, chan::chan *chptr, struct ListClient *params) { int visible; visible = !secret(chptr) || is_member(chptr, source_p); if (!visible && !params->operspy) return; if ((unsigned int)chptr->members.length < params->users_min || (unsigned int)chptr->members.length > params->users_max) return; if (params->topic_min && chptr->topic.time < params->topic_min) return; /* If a topic TS is provided, don't show channels without a topic set. */ if (params->topic_max && (chptr->topic.time > params->topic_max || chptr->topic.time == 0)) return; if (params->created_min && chptr->channelts < params->created_min) return; if (params->created_max && chptr->channelts > params->created_max) return; list_one_channel(source_p, chptr, visible); } /* * safelist_iterate_client() * * inputs - client pointer * outputs - none * side effects - the client's sendq is filled up again */ static void safelist_iterate_client(struct Client *source_p) { rb_radixtree_iteration_state iter; void *elem; RB_RADIXTREE_FOREACH_FROM(elem, &iter, channel_tree, source_p->localClient->safelist_data->chname) { const auto chptr(reinterpret_cast(elem)); if (safelist_sendq_exceeded(source_p->from)) { rb_free(source_p->localClient->safelist_data->chname); source_p->localClient->safelist_data->chname = rb_strdup(chptr->name.c_str()); return; } safelist_one_channel(source_p, chptr, source_p->localClient->safelist_data); } safelist_client_release(source_p); } static void safelist_iterate_clients(void *unused) { rb_dlink_node *n, *n2; RB_DLINK_FOREACH_SAFE(n, n2, safelisting_clients.head) safelist_iterate_client((struct Client *)n->data); }