-/* +------------------------------------+
- * | Inspire Internet Relay Chat Daemon |
- * +------------------------------------+
+/* +------------------------------------+
+ * | Inspire Internet Relay Chat Daemon |
+ * +------------------------------------+
*
* InspIRCd is copyright (C) 2002-2006 ChatSpike-Dev.
- * E-mail:
- * <brain@chatspike.net>
- * <Craig@chatspike.net>
+ * E-mail:
+ *<brain@chatspike.net>
+ *<Craig@chatspike.net>
*
* Written by Craig Edwards, Craig McLure, and others.
* This program is free but copyrighted software; see
- * the file COPYING for details.
+ *the file COPYING for details.
*
* ---------------------------------------------------
*/
-using namespace std;
-
-#include "inspircd_config.h"
-#include "inspircd_io.h"
-#include <time.h>
-#include <string>
-#ifdef GCC3
-#include <ext/hash_map>
-#else
-#include <hash_map>
-#endif
-#include <map>
-#include <sstream>
-#include <vector>
-#include <deque>
+#include "configreader.h"
#include "users.h"
-#include "ctables.h"
-#include "globals.h"
-#include "modules.h"
-#include "dynamic.h"
-#include "wildcard.h"
#include "message.h"
+#include "modules.h"
#include "commands.h"
-#include "mode.h"
-#include "xline.h"
-#include "inspstring.h"
-#include "dnsqueue.h"
#include "helperfuncs.h"
-#include "hashcomp.h"
-#include "socketengine.h"
-#include "typedefs.h"
-#include "command_parse.h"
-#include "inspircd.h"
-#include "cmd_who.h"
+#include "wildcard.h"
+#include "commands/cmd_who.h"
extern ServerConfig* Config;
-extern InspIRCd* ServerInstance;
-extern int MODCOUNT;
-extern std::vector<Module*> modules;
-extern std::vector<ircd_module*> factory;
-extern time_t TIME;
extern user_hash clientlist;
extern chan_hash chanlist;
extern std::vector<userrec*> all_opers;
-extern std::vector<userrec*> local_users;
-extern userrec* fd_ref_table[MAX_DESCRIPTORS];
-void cmd_who::Handle (char **parameters, int pcnt, userrec *user)
+/* get the last 'visible' chan of a user */
+static char *getlastchanname(userrec *u)
{
- chanrec* Ptr = NULL;
- char tmp[10];
-
- /* theres more to do here, but for now just close the socket */
- if (pcnt == 1)
+ for (std::vector<ucrec*>::const_iterator v = u->chans.begin(); v != u->chans.end(); v++)
+ {
+ ucrec* temp = (ucrec*)*v;
+
+ if (temp->channel)
+ {
+ if (!temp->channel->IsModeSet('s'))
+ return temp->channel->name;
+ }
+ }
+
+ return "*";
+}
+
+bool whomatch(userrec* user, const char* matchtext, bool opt_realname, bool opt_showrealhost)
+{
+ bool realhost = false;
+ bool realname = false;
+
+ if (opt_realname)
+ realname = match(user->fullname, matchtext);
+
+ if (opt_showrealhost)
+ realhost = match(user->host, matchtext);
+
+ return ((realname) || (realhost) || (match(user->dhost, matchtext)) || (match(user->nick, matchtext)) || (match(user->server, matchtext)));
+}
+
+void cmd_who::Handle (const char** parameters, int pcnt, userrec *user)
+{
+ /*
+ * XXX - RFC says:
+ * The <name> passed to WHO is matched against users' host, server, real
+ * name and nickname
+ * Currently, we support WHO #chan, WHO nick, WHO 0, WHO *, and the addition of a 'o' flag, as per RFC.
+ */
+
+ /* WHO options */
+ bool opt_viewopersonly = false;
+ bool opt_showrealhost = false;
+ bool opt_unlimit = false;
+ bool opt_realname = false;
+
+ chanrec *ch = NULL;
+ std::vector<std::string> whoresults;
+ std::string initial = "352 " + std::string(user->nick) + " ";
+
+ const char* matchtext = NULL;
+
+ /* Change '0' into '*' so the wildcard matcher can grok it */
+ matchtext = parameters[0];
+ if (!strcmp(matchtext,"0"))
+ matchtext = "*";
+
+ if (pcnt > 1)
{
- if ((IS_SINGLE(parameters[0],'0')) || (IS_SINGLE(parameters[0],'*')))
+ /* parse flags */
+ const char *iter = parameters[1];
+
+ while (*iter)
{
- if ((user->chans.size()) && (user->chans[0].channel))
+ switch (*iter)
{
- int n_list = 0;
- for (user_hash::const_iterator i = clientlist.begin(); i != clientlist.end(); i++)
- {
- Ptr = i->second->chans[0].channel;
- // suggested by phidjit and FCS
- if ((!common_channels(user,i->second)) && (isnick(i->second->nick)))
- {
- // Bug Fix #29
- *tmp = 0;
- if (*i->second->awaymsg) {
- charlcat(tmp, 'G', 9);
- } else {
- charlcat(tmp, 'H', 9);
- }
- if (*i->second->oper) { charlcat(tmp, '*', 9); }
- WriteServ(user->fd,"352 %s %s %s %s %s %s %s :0 %s",user->nick, Ptr ? Ptr->name : "*", i->second->ident, i->second->dhost, i->second->server, i->second->nick, tmp, i->second->fullname);
- if (n_list++ > Config->MaxWhoResults)
- {
- WriteServ(user->fd,"523 %s WHO :Command aborted: More results than configured limit",user->nick);
- break;
- }
- }
- }
+ case 'o':
+ opt_viewopersonly = true;
+ break;
+ case 'h':
+ if (*user->oper)
+ opt_showrealhost = true;
+ break;
+ case 'u':
+ if (*user->oper)
+ opt_unlimit = true;
+ break;
+ case 'r':
+ opt_realname = true;
+ break;
}
- if (Ptr)
+
+ *iter++;
+ }
+ }
+
+
+ /* who on a channel? */
+ ch = FindChan(matchtext);
+
+ if (ch)
+ {
+ /* who on a channel. */
+ CUList *cu = ch->GetUsers();
+
+ for (CUList::iterator i = cu->begin(); i != cu->end(); i++)
+ {
+ /* opers only, please */
+ if (opt_viewopersonly && !*(i->second)->oper)
+ continue;
+
+ /* XXX - code duplication; this could be more efficient -- w00t */
+ std::string wholine = initial;
+
+ wholine = wholine + getlastchanname(i->second) + " " + i->second->ident + " " + (opt_showrealhost ? i->second->host : i->second->dhost) + " " +
+ i->second->server + " " + i->second->nick + " ";
+
+ /* away? */
+ if (*(i->second)->awaymsg)
{
- WriteServ(user->fd,"315 %s %s :End of /WHO list.",user->nick , parameters[0]);
+ wholine.append("G");
}
else
{
- WriteServ(user->fd,"315 %s %s :End of /WHO list.",user->nick, parameters[0]);
+ wholine.append("H");
+ }
+
+ /* oper? */
+ if (*(i->second)->oper)
+ {
+ wholine.append("*");
}
- return;
+
+ wholine = wholine + cmode(i->second, ch) + " :0 " + i->second->fullname;
+ whoresults.push_back(wholine);
}
- if (parameters[0][0] == '#')
+ }
+ else
+ {
+ /* Match against wildcard of nick, server or host */
+
+ if (opt_viewopersonly)
{
- Ptr = FindChan(parameters[0]);
- if (Ptr)
+ /* Showing only opers */
+ for (std::vector<userrec*>::iterator i = all_opers.begin(); i != all_opers.end(); i++)
{
- int n_list = 0;
- for (user_hash::const_iterator i = clientlist.begin(); i != clientlist.end(); i++)
+ userrec* oper = *i;
+
+ if (whomatch(oper, matchtext, opt_realname, opt_showrealhost))
{
- if ((Ptr->HasUser(i->second)) && (isnick(i->second->nick)))
- {
- // Fix Bug #29 - Part 2..
- *tmp = 0;
- if (*i->second->awaymsg) {
- charlcat(tmp, 'G', 9);
- } else {
- charlcat(tmp, 'H', 9);
- }
- if (*i->second->oper) { charlcat(tmp, '*', 9); }
- strlcat(tmp, cmode(i->second, Ptr),5);
- WriteServ(user->fd,"352 %s %s %s %s %s %s %s :0 %s",user->nick, Ptr->name, i->second->ident, i->second->dhost, i->second->server, i->second->nick, tmp, i->second->fullname);
- n_list++;
- if (n_list > Config->MaxWhoResults)
- {
- WriteServ(user->fd,"523 %s WHO :Command aborted: More results than configured limit",user->nick);
- break;
- }
+ std::string wholine = initial;
+
+ wholine = wholine + getlastchanname(oper) + " " + oper->ident + " " + (opt_showrealhost ? oper->host : oper->dhost) + " " +
+ oper->server + " " + oper->nick + " ";
+ /* away? */
+ if (*oper->awaymsg)
+ {
+ wholine.append("G");
+ }
+ else
+ {
+ wholine.append("H");
}
+
+ /* oper? */
+ if (*oper->oper)
+ {
+ wholine.append("*");
+ }
+
+ wholine = wholine + cmode(oper, ch) + " :0 " + oper->fullname;
+ whoresults.push_back(wholine);
}
- WriteServ(user->fd,"315 %s %s :End of /WHO list.",user->nick, parameters[0]);
- }
- else
- {
- WriteServ(user->fd,"401 %s %s :No such nick/channel",user->nick, parameters[0]);
}
}
else
{
- userrec* u = Find(parameters[0]);
- if (u)
+ for (user_hash::iterator i = clientlist.begin(); i != clientlist.end(); i++)
{
- // Bug Fix #29 -- Part 29..
- *tmp = 0;
- if (*u->awaymsg) {
- charlcat(tmp, 'G' ,9);
- } else {
- charlcat(tmp, 'H' ,9);
+ if (whomatch(i->second, matchtext, opt_realname, opt_showrealhost))
+ {
+ std::string wholine = initial;
+
+ wholine = wholine + getlastchanname(i->second) + " " + i->second->ident + " " + (opt_showrealhost ? i->second->host : i->second->dhost) + " " +
+ i->second->server + " " + i->second->nick + " ";
+
+ /* away? */
+ if (*(i->second)->awaymsg)
+ {
+ wholine.append("G");
+ }
+ else
+ {
+ wholine.append("H");
+ }
+
+ /* oper? */
+ if (*(i->second)->oper)
+ {
+ wholine.append("*");
+ }
+
+ wholine = wholine + cmode(i->second, ch) + " :0 " + i->second->fullname;
+ whoresults.push_back(wholine);
}
- if (*u->oper) { charlcat(tmp, '*' ,9); }
- WriteServ(user->fd,"352 %s %s %s %s %s %s %s :0 %s",user->nick, u->chans.size() && u->chans[0].channel ? u->chans[0].channel->name
- : "*", u->ident, u->dhost, u->server, u->nick, tmp, u->fullname);
}
- WriteServ(user->fd,"315 %s %s :End of /WHO list.",user->nick, parameters[0]);
}
}
- if (pcnt == 2)
+ /* Send the results out */
+ if ((whoresults.size() < (size_t)Config->MaxWhoResults) && (!opt_unlimit))
{
- if ((IS_SINGLE(parameters[0],'0')) || (IS_SINGLE(parameters[0],'*')) && (IS_SINGLE(parameters[1],'o')))
- {
- for (std::vector<userrec*>::iterator i = all_opers.begin(); i != all_opers.end(); i++)
- {
- // If i were a rich man.. I wouldn't need to me making these bugfixes..
- // But i'm a poor bastard with nothing better to do.
- userrec* oper = *i;
- *tmp = 0;
- if (*oper->awaymsg) {
- charlcat(tmp, 'G' ,9);
- } else {
- charlcat(tmp, 'H' ,9);
- }
- WriteServ(user->fd,"352 %s %s %s %s %s %s %s* :0 %s", user->nick, oper->chans.size() && oper->chans[0].channel ? oper->chans[0].channel->name
- : "*", oper->ident, oper->dhost, oper->server, oper->nick, tmp, oper->fullname);
- }
- WriteServ(user->fd,"315 %s %s :End of /WHO list.",user->nick, parameters[0]);
- return;
- }
+ for (std::vector<std::string>::const_iterator n = whoresults.begin(); n != whoresults.end(); n++)
+ WriteServ_NoFormat(user->fd,n->c_str());
+ WriteServ(user->fd,"315 %s %s :End of /WHO list.",user->nick, parameters[0]);
+ }
+ else
+ {
+ /* BZZT! Too many results. */
+ WriteServ(user->fd,"315 %s %s :Too many results",user->nick, parameters[0]);
}
}
-