📄 m_list.c
字号:
/* * IRC - Internet Relay Chat, src/modules/m_list.c * (C) 2004 The UnrealIRCd Team * * 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. */#include "config.h"#include "struct.h"#include "common.h"#include "sys.h"#include "numeric.h"#include "msg.h"#include "proto.h"#include "channel.h"#include <time.h>#include <sys/stat.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#ifdef _WIN32#include <io.h>#endif#include <fcntl.h>#include "h.h"#ifdef STRIPBADWORDS#include "badwords.h"#endif#ifdef _WIN32#include "version.h"#endifDLLFUNC int m_list(aClient *cptr, aClient *sptr, int parc, char *parv[]);void _send_list(aClient *cptr, int numsend);#define MSG_LIST "LIST" #define TOK_LIST "(" ModuleHeader MOD_HEADER(m_list) = { "m_list", "$Id: m_list.c,v 1.1.6.8 2006/12/22 21:10:33 syzop Exp $", "command /list", "3.2-b8-1", NULL };DLLFUNC int MOD_TEST(m_list)(ModuleInfo *modinfo){ MARK_AS_OFFICIAL_MODULE(modinfo); EfunctionAddVoid(modinfo->handle, EFUNC_SEND_LIST, _send_list); return MOD_SUCCESS;}DLLFUNC int MOD_INIT(m_list)(ModuleInfo *modinfo){ add_Command(MSG_LIST, TOK_LIST, m_list, MAXPARA); MARK_AS_OFFICIAL_MODULE(modinfo); return MOD_SUCCESS;}DLLFUNC int MOD_LOAD(m_list)(int module_load){ return MOD_SUCCESS;}DLLFUNC int MOD_UNLOAD(m_list)(int module_unload){ if (del_Command(MSG_LIST, TOK_LIST, m_list) < 0) { sendto_realops("Failed to delete commands when unloading %s", MOD_HEADER(m_list).name); } return MOD_SUCCESS;}/* Originally from bahamut, modified a bit for Unreal by codemastr * also Opers can now see +s channels and can still use /list while * HTM is active -- codemastr *//* * * m_list * parv[0] = sender prefix * parv[1] = channel */DLLFUNC CMD_FUNC(m_list){ aChannel *chptr; TS currenttime = TStime(); char *name, *p = NULL; LOpts *lopt = NULL; Link *lp; int usermax, usermin, error = 0, doall = 0; TS chantimemin, chantimemax; TS topictimemin, topictimemax; Link *yeslist = NULL, *nolist = NULL; static char *usage[] = { " Usage: /LIST <options>", "", "If you don't include any options, the default is to send you the", "entire unfiltered list of channels. Below are the options you can", "use, and what channels LIST will return when you use them.", ">number List channels with more than <number> people.", "<number List channels with less than <number> people.", "C>number List channels created between now and <number> minutes ago.", "C<number List channels created earlier than <number> minutes ago.", "T>number List channels whose topics are older than <number> minutes", " (Ie, they have not changed in the last <number> minutes.", "T<number List channels whose topics are not older than <number> minutes.", "*mask* List channels that match *mask*", "!*mask* List channels that do not match *mask*", NULL }; /* Some starting san checks -- No interserver lists allowed. */ if (cptr != sptr || !sptr->user) return 0; /* If a /list is in progress, then another one will cancel it */ if ((lopt = sptr->user->lopt) != NULL) { sendto_one(sptr, rpl_str(RPL_LISTEND), me.name, parv[0]); free_str_list(sptr->user->lopt->yeslist); free_str_list(sptr->user->lopt->nolist); MyFree(sptr->user->lopt); sptr->user->lopt = NULL; return 0; } /* if HTM, drop this too */#ifndef NO_FDLIST if (lifesux && !IsOper(cptr)) { sendto_one(sptr, err_str(ERR_HTMDISABLED), me.name, sptr->name, "/List"); /* We need this for mIRC compatibility -- codemastr */ sendto_one(sptr, rpl_str(RPL_LISTEND), me.name, parv[0]); return 0; }#endif if (parc < 2 || BadPtr(parv[1])) { sendto_one(sptr, rpl_str(RPL_LISTSTART), me.name, parv[0]); lopt = sptr->user->lopt = (LOpts *) MyMalloc(sizeof(LOpts)); memset(lopt, '\0', sizeof(LOpts)); lopt->showall = 1; if (DBufLength(&cptr->sendQ) < 2048) send_list(cptr, 64); return 0; } if ((parc == 2) && (parv[1][0] == '?') && (parv[1][1] == '\0')) { char **ptr = usage; for (; *ptr; ptr++) sendto_one(sptr, rpl_str(RPL_LISTSYNTAX), me.name, cptr->name, *ptr); return 0; } sendto_one(sptr, rpl_str(RPL_LISTSTART), me.name, parv[0]); chantimemax = topictimemax = currenttime + 86400; chantimemin = topictimemin = 0; usermin = 1; /* Minimum of 1 */ usermax = -1; /* No maximum */ for (name = strtoken(&p, parv[1], ","); name && !error; name = strtoken(&p, (char *)NULL, ",")) { switch (*name) { case '<': usermax = atoi(name + 1) - 1; doall = 1; break; case '>': usermin = atoi(name + 1) + 1; doall = 1; break; case 'C': case 'c': /* Channel TS time -- creation time? */ ++name; switch (*name++) { case '<': chantimemax = currenttime - 60 * atoi(name); doall = 1; break; case '>': chantimemin = currenttime - 60 * atoi(name); doall = 1; break; default: sendto_one(sptr, err_str(ERR_LISTSYNTAX), me.name, cptr->name); error = 1; } break;#ifdef LIST_USE_T case 'T': case 't': ++name; switch (*name++) { case '<': topictimemax = currenttime - 60 * atoi(name); doall = 1; break; case '>': topictimemin = currenttime - 60 * atoi(name); doall = 1; break; default: sendto_one(sptr, err_str(ERR_LISTSYNTAX), me.name, cptr->name, "Bad list syntax, type /list ?"); error = 1; } break;#endif default: /* A channel, possibly with wildcards. * Thought for the future: Consider turning wildcard * processing on the fly. * new syntax: !channelmask will tell ircd to ignore * any channels matching that mask, and then * channelmask will tell ircd to send us a list of * channels only masking channelmask. Note: Specifying * a channel without wildcards will return that * channel even if any of the !channelmask masks * matches it. */ if (*name == '!') { doall = 1; lp = make_link(); lp->next = nolist; nolist = lp; DupString(lp->value.cp, name + 1); } else if (strchr(name, '*') || strchr(name, '?')) { doall = 1; lp = make_link(); lp->next = yeslist; yeslist = lp; DupString(lp->value.cp, name); } else /* Just a normal channel */ { chptr = find_channel(name, NullChn); if (chptr && (ShowChannel(sptr, chptr) || OPCanSeeSecret(sptr))) {#ifdef LIST_SHOW_MODES modebuf[0] = '['; channel_modes(sptr, &modebuf[1], parabuf, chptr); if (modebuf[2] == '\0') modebuf[0] = '\0'; else strlcat(modebuf, "]", sizeof modebuf);#endif sendto_one(sptr, rpl_str(RPL_LIST), me.name, parv[0], name, chptr->users,#ifdef LIST_SHOW_MODES modebuf,#endif (chptr->topic ? chptr->topic : ""));} } } /* switch */ } /* while */ if (doall) { lopt = sptr->user->lopt = (LOpts *) MyMalloc(sizeof(LOpts)); memset(lopt, '\0', sizeof(LOpts)); lopt->usermin = usermin; lopt->usermax = usermax; lopt->topictimemax = topictimemax; lopt->topictimemin = topictimemin; lopt->chantimemax = chantimemax; lopt->chantimemin = chantimemin; lopt->nolist = nolist; lopt->yeslist = yeslist; if (DBufLength(&cptr->sendQ) < 2048) send_list(cptr, 64); return 0; } sendto_one(sptr, rpl_str(RPL_LISTEND), me.name, parv[0]); return 0;}/* * The function which sends the actual channel list back to the user. * Operates by stepping through the hashtable, sending the entries back if * they match the criteria. * cptr = Local client to send the output back to. * numsend = Number (roughly) of lines to send back. Once this number has * been exceeded, send_list will finish with the current hash bucket, * and record that number as the number to start next time send_list * is called for this user. So, this function will almost always send * back more lines than specified by numsend (though not by much, * assuming CH_MAX is was well picked). So be conservative in your choice * of numsend. -Rak *//* Taken from bahamut, modified for Unreal by codemastr */void _send_list(aClient *cptr, int numsend){ aChannel *chptr; LOpts *lopt = cptr->user->lopt; unsigned int hashnum; /* Begin of /list? then send official channels. */ if ((lopt->starthash == 0) && conf_offchans) { ConfigItem_offchans *x; for (x = conf_offchans; x; x = (ConfigItem_offchans *)x->next) { if (find_channel(x->chname, (aChannel *)NULL)) continue; /* exists, >0 users.. will be sent later */ sendto_one(cptr, rpl_str(RPL_LIST), me.name, cptr->name, x->chname, 0,#ifdef LIST_SHOW_MODES "",#endif x->topic ? x->topic : ""); } } for (hashnum = lopt->starthash; hashnum < CH_MAX; hashnum++) { if (numsend > 0) for (chptr = (aChannel *)hash_get_chan_bucket(hashnum); chptr; chptr = chptr->hnextch) { if (SecretChannel(chptr) && !IsMember(cptr, chptr) && !OPCanSeeSecret(cptr)) continue; /* Much more readable like this -- codemastr */ if ((!lopt->showall)) { /* User count must be in range */ if ((chptr->users < lopt->usermin) || ((lopt->usermax >= 0) && (chptr->users > lopt->usermax))) continue; /* Creation time must be in range */ if ((chptr->creationtime && (chptr->creationtime < lopt->chantimemin)) || (chptr->creationtime > lopt->chantimemax)) continue; /* Topic time must be in range */ if ((chptr->topic_time < lopt->topictimemin) || (chptr->topic_time > lopt->topictimemax)) continue; /* Must not be on nolist (if it exists) */ if (lopt->nolist && find_str_match_link(lopt->nolist, chptr->chname)) continue; /* Must be on yeslist (if it exists) */ if (lopt->yeslist && !find_str_match_link(lopt->yeslist, chptr->chname)) continue; }#ifdef LIST_SHOW_MODES modebuf[0] = '['; channel_modes(cptr, &modebuf[1], parabuf, chptr); if (modebuf[2] == '\0') modebuf[0] = '\0'; else strlcat(modebuf, "]", sizeof modebuf);#endif if (!OPCanSeeSecret(cptr)) sendto_one(cptr, rpl_str(RPL_LIST), me.name, cptr->name, ShowChannel(cptr, chptr) ? chptr->chname : "*", chptr->users,#ifdef LIST_SHOW_MODES ShowChannel(cptr, chptr) ? modebuf : "",#endif ShowChannel(cptr, chptr) ? (chptr->topic ? chptr->topic : "") : ""); else sendto_one(cptr, rpl_str(RPL_LIST), me.name, cptr->name, chptr->chname, chptr->users,#ifdef LIST_SHOW_MODES modebuf,#endif (chptr->topic ? chptr->topic : "")); numsend--; } else break; } /* All done */ if (hashnum == CH_MAX) { sendto_one(cptr, rpl_str(RPL_LISTEND), me.name, cptr->name); free_str_list(cptr->user->lopt->yeslist); free_str_list(cptr->user->lopt->nolist); MyFree(cptr->user->lopt); cptr->user->lopt = NULL; return; } /* * We've exceeded the limit on the number of channels to send back * at once. */ lopt->starthash = hashnum; return;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -