mirror of
https://github.com/unrealircd/unrealircd.git
synced 2026-07-02 07:06:37 +02:00
b19b70e876
This adds user_can_see_member_fast() which is used in at least 3 places now, more places may follow later. It has extra paramters for membership and membership modes that is very likely already looked up by the caller (or if not, it is worth doing so by the caller). This is work in progress so if everything crashes or people mysteriously seem not present in channels (or the other way around) i would not be surprised :D.
195 lines
4.9 KiB
C
195 lines
4.9 KiB
C
/*
|
|
* IRC - Internet Relay Chat, src/modules/names.c
|
|
* (C) 2006 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 "unrealircd.h"
|
|
|
|
CMD_FUNC(cmd_names);
|
|
|
|
long CAP_MULTI_PREFIX = 0L;
|
|
long CAP_USERHOST_IN_NAMES = 0L;
|
|
|
|
#define MSG_NAMES "NAMES"
|
|
|
|
ModuleHeader MOD_HEADER
|
|
= {
|
|
"names",
|
|
"5.0",
|
|
"command /names",
|
|
"UnrealIRCd Team",
|
|
"unrealircd-6",
|
|
};
|
|
|
|
MOD_INIT()
|
|
{
|
|
ClientCapabilityInfo c;
|
|
memset(&c, 0, sizeof(c));
|
|
c.name = "multi-prefix";
|
|
ClientCapabilityAdd(modinfo->handle, &c, &CAP_MULTI_PREFIX);
|
|
memset(&c, 0, sizeof(c));
|
|
c.name = "userhost-in-names";
|
|
ClientCapabilityAdd(modinfo->handle, &c, &CAP_USERHOST_IN_NAMES);
|
|
|
|
CommandAdd(modinfo->handle, MSG_NAMES, cmd_names, MAXPARA, CMD_USER|CMD_SERVER);
|
|
MARK_AS_OFFICIAL_MODULE(modinfo);
|
|
return MOD_SUCCESS;
|
|
}
|
|
|
|
MOD_LOAD()
|
|
{
|
|
return MOD_SUCCESS;
|
|
}
|
|
|
|
MOD_UNLOAD()
|
|
{
|
|
return MOD_SUCCESS;
|
|
}
|
|
|
|
/************************************************************************
|
|
* cmd_names() - Added by Jto 27 Apr 1989
|
|
* 12 Feb 2000 - geesh, time for a rewrite -lucas
|
|
************************************************************************/
|
|
|
|
/*
|
|
** cmd_names
|
|
** parv[1] = channel
|
|
*/
|
|
#define TRUNCATED_NAMES 64
|
|
CMD_FUNC(cmd_names)
|
|
{
|
|
int multiprefix = (MyConnect(client) && HasCapabilityFast(client, CAP_MULTI_PREFIX));
|
|
int uhnames = (MyConnect(client) && HasCapabilityFast(client, CAP_USERHOST_IN_NAMES)); // cache UHNAMES support
|
|
int bufLen = NICKLEN + (!uhnames ? 0 : (1 + USERLEN + 1 + HOSTLEN));
|
|
int mlen = strlen(me.name) + bufLen + 7;
|
|
Channel *channel;
|
|
Client *acptr;
|
|
Membership *us = NULL;
|
|
Member *cm;
|
|
int idx, flag = 1, spos;
|
|
const char *para = parv[1], *s;
|
|
char nuhBuffer[NICKLEN+USERLEN+HOSTLEN+3];
|
|
char buf[BUFSIZE];
|
|
char can_see_invisible;
|
|
|
|
if (parc < 2 || !MyConnect(client))
|
|
{
|
|
sendnumeric(client, RPL_ENDOFNAMES, "*");
|
|
return;
|
|
}
|
|
|
|
for (s = para; *s; s++)
|
|
{
|
|
if (*s == ',')
|
|
{
|
|
sendnumeric(client, ERR_TOOMANYTARGETS, s+1, 1, "NAMES");
|
|
return;
|
|
}
|
|
}
|
|
|
|
channel = find_channel(para);
|
|
|
|
if (!channel || (!ShowChannel(client, channel) && !ValidatePermissionsForPath("channel:see:names:secret",client,NULL,channel,NULL)))
|
|
{
|
|
sendnumeric(client, RPL_ENDOFNAMES, para);
|
|
return;
|
|
}
|
|
|
|
/* cache whether this user is a member of this channel or not */
|
|
if (IsUser(client))
|
|
us = find_membership_link(client->user->channel, channel);
|
|
|
|
// FIXME: consider rewriting this whole thing to get rid of pointer juggling and stuff.
|
|
|
|
if (PubChannel(channel))
|
|
buf[0] = '=';
|
|
else if (SecretChannel(channel))
|
|
buf[0] = '@';
|
|
else
|
|
buf[0] = '*';
|
|
|
|
idx = 1;
|
|
buf[idx++] = ' ';
|
|
for (s = channel->name; *s; s++)
|
|
buf[idx++] = *s;
|
|
buf[idx++] = ' ';
|
|
buf[idx++] = ':';
|
|
|
|
/* If we go through the following loop and never add anything,
|
|
we need this to be empty, otherwise spurious things from the
|
|
LAST /names call get stuck in there.. - lucas */
|
|
buf[idx] = '\0';
|
|
|
|
spos = idx; /* starting point in buffer for names! */
|
|
|
|
can_see_invisible = ValidatePermissionsForPath("channel:see:names:invisible",client,NULL,channel,NULL);
|
|
|
|
for (cm = channel->members; cm; cm = cm->next)
|
|
{
|
|
acptr = cm->client;
|
|
if (IsInvisible(acptr) && !us && !can_see_invisible)
|
|
continue;
|
|
|
|
if (!user_can_see_member_fast(client, acptr, channel, cm, us ? us->member_modes : NULL))
|
|
continue; /* invisible (eg: due to delayjoin) */
|
|
|
|
if (!multiprefix)
|
|
{
|
|
/* Standard NAMES reply (single character) */
|
|
char c = mode_to_prefix(*cm->member_modes);
|
|
if (c)
|
|
buf[idx++] = c;
|
|
} else {
|
|
/* NAMES reply with all rights included (multi-prefix / NAMESX) */
|
|
strcpy(&buf[idx], modes_to_prefix(cm->member_modes));
|
|
idx += strlen(&buf[idx]);
|
|
}
|
|
|
|
if (!uhnames) {
|
|
s = acptr->name;
|
|
} else {
|
|
strlcpy(nuhBuffer,
|
|
make_nick_user_host(acptr->name, acptr->user->username, GetHost(acptr)),
|
|
bufLen + 1);
|
|
s = nuhBuffer;
|
|
}
|
|
/* 's' is intialized above to point to either acptr->name (normal),
|
|
* or to nuhBuffer (for UHNAMES).
|
|
*/
|
|
for (; *s; s++)
|
|
buf[idx++] = *s;
|
|
if (cm->next)
|
|
buf[idx++] = ' ';
|
|
buf[idx] = '\0';
|
|
flag = 1;
|
|
if (mlen + idx + bufLen + MEMBERMODESLEN >= BUFSIZE - 1)
|
|
{
|
|
sendnumeric(client, RPL_NAMREPLY, buf);
|
|
idx = spos;
|
|
flag = 0;
|
|
}
|
|
}
|
|
|
|
if (flag)
|
|
sendnumeric(client, RPL_NAMREPLY, buf);
|
|
|
|
sendnumeric(client, RPL_ENDOFNAMES, para);
|
|
}
|