mirror of
https://github.com/osmarks/ngircd.git
synced 2024-12-12 09:50:29 +00:00
irc-info: move static functions at the top of the file
This commit is contained in:
parent
0ad0fe207a
commit
5facf5c15e
@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* ngIRCd -- The Next Generation IRC Daemon
|
* ngIRCd -- The Next Generation IRC Daemon
|
||||||
* Copyright (c)2001-2012 Alexander Barton (alex@barton.de) and Contributors.
|
* Copyright (c)2001-2013 Alexander Barton (alex@barton.de) and Contributors.
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -44,6 +44,441 @@
|
|||||||
#include "exp.h"
|
#include "exp.h"
|
||||||
#include "irc-info.h"
|
#include "irc-info.h"
|
||||||
|
|
||||||
|
/* Local functions */
|
||||||
|
|
||||||
|
static unsigned int
|
||||||
|
t_diff(time_t *t, const time_t d)
|
||||||
|
{
|
||||||
|
time_t diff, remain;
|
||||||
|
|
||||||
|
diff = *t / d;
|
||||||
|
remain = diff * d;
|
||||||
|
*t -= remain;
|
||||||
|
|
||||||
|
return (unsigned int)diff;
|
||||||
|
}
|
||||||
|
|
||||||
|
static unsigned int
|
||||||
|
uptime_days(time_t *now)
|
||||||
|
{
|
||||||
|
return t_diff(now, 60 * 60 * 24);
|
||||||
|
}
|
||||||
|
|
||||||
|
static unsigned int
|
||||||
|
uptime_hrs(time_t *now)
|
||||||
|
{
|
||||||
|
return t_diff(now, 60 * 60);
|
||||||
|
}
|
||||||
|
|
||||||
|
static unsigned int
|
||||||
|
uptime_mins(time_t *now)
|
||||||
|
{
|
||||||
|
return t_diff(now, 60);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool
|
||||||
|
write_whoreply(CLIENT *Client, CLIENT *c, const char *channelname, const char *flags)
|
||||||
|
{
|
||||||
|
return IRC_WriteStrClient(Client, RPL_WHOREPLY_MSG, Client_ID(Client),
|
||||||
|
channelname, Client_User(c),
|
||||||
|
Client_HostnameDisplayed(c),
|
||||||
|
Client_ID(Client_Introducer(c)), Client_ID(c),
|
||||||
|
flags, Client_Hops(c), Client_Info(c));
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *
|
||||||
|
who_flags_status(const char *client_modes)
|
||||||
|
{
|
||||||
|
if (strchr(client_modes, 'a'))
|
||||||
|
return "G"; /* away */
|
||||||
|
return "H";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return channel user mode prefix(es).
|
||||||
|
*
|
||||||
|
* @param Client The client requesting the mode prefixes.
|
||||||
|
* @param chan_user_modes String with channel user modes.
|
||||||
|
* @param str String buffer to which the prefix(es) will be appended.
|
||||||
|
* @param len Size of "str" buffer.
|
||||||
|
* @return Pointer to "str".
|
||||||
|
*/
|
||||||
|
static char *
|
||||||
|
who_flags_qualifier(CLIENT *Client, const char *chan_user_modes,
|
||||||
|
char *str, size_t len)
|
||||||
|
{
|
||||||
|
assert(Client != NULL);
|
||||||
|
|
||||||
|
if (Client_Cap(Client) & CLIENT_CAP_MULTI_PREFIX) {
|
||||||
|
if (strchr(chan_user_modes, 'q'))
|
||||||
|
strlcat(str, "~", len);
|
||||||
|
if (strchr(chan_user_modes, 'a'))
|
||||||
|
strlcat(str, "&", len);
|
||||||
|
if (strchr(chan_user_modes, 'o'))
|
||||||
|
strlcat(str, "@", len);
|
||||||
|
if (strchr(chan_user_modes, 'h'))
|
||||||
|
strlcat(str, "%", len);
|
||||||
|
if (strchr(chan_user_modes, 'v'))
|
||||||
|
strlcat(str, "+", len);
|
||||||
|
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strchr(chan_user_modes, 'q'))
|
||||||
|
strlcat(str, "~", len);
|
||||||
|
else if (strchr(chan_user_modes, 'a'))
|
||||||
|
strlcat(str, "&", len);
|
||||||
|
else if (strchr(chan_user_modes, 'o'))
|
||||||
|
strlcat(str, "@", len);
|
||||||
|
else if (strchr(chan_user_modes, 'h'))
|
||||||
|
strlcat(str, "%", len);
|
||||||
|
else if (strchr(chan_user_modes, 'v'))
|
||||||
|
strlcat(str, "+", len);
|
||||||
|
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send WHO reply for a "channel target" ("WHO #channel").
|
||||||
|
*
|
||||||
|
* @param Client Client requesting the information.
|
||||||
|
* @param Chan Channel being requested.
|
||||||
|
* @param OnlyOps Only display IRC operators.
|
||||||
|
* @return CONNECTED or DISCONNECTED.
|
||||||
|
*/
|
||||||
|
static bool
|
||||||
|
IRC_WHO_Channel(CLIENT *Client, CHANNEL *Chan, bool OnlyOps)
|
||||||
|
{
|
||||||
|
bool is_visible, is_member, is_ircop;
|
||||||
|
CL2CHAN *cl2chan;
|
||||||
|
const char *client_modes;
|
||||||
|
char flags[10];
|
||||||
|
CLIENT *c;
|
||||||
|
int count = 0;
|
||||||
|
|
||||||
|
assert( Client != NULL );
|
||||||
|
assert( Chan != NULL );
|
||||||
|
|
||||||
|
IRC_SetPenalty(Client, 1);
|
||||||
|
|
||||||
|
is_member = Channel_IsMemberOf(Chan, Client);
|
||||||
|
|
||||||
|
/* Secret channel? */
|
||||||
|
if (!is_member && strchr(Channel_Modes(Chan), 's'))
|
||||||
|
return IRC_WriteStrClient(Client, RPL_ENDOFWHO_MSG,
|
||||||
|
Client_ID(Client), Channel_Name(Chan));
|
||||||
|
|
||||||
|
cl2chan = Channel_FirstMember(Chan);
|
||||||
|
for (; cl2chan ; cl2chan = Channel_NextMember(Chan, cl2chan)) {
|
||||||
|
c = Channel_GetClient(cl2chan);
|
||||||
|
|
||||||
|
client_modes = Client_Modes(c);
|
||||||
|
is_ircop = strchr(client_modes, 'o') != NULL;
|
||||||
|
if (OnlyOps && !is_ircop)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
is_visible = strchr(client_modes, 'i') == NULL;
|
||||||
|
if (is_member || is_visible) {
|
||||||
|
strlcpy(flags, who_flags_status(client_modes),
|
||||||
|
sizeof(flags));
|
||||||
|
if (is_ircop)
|
||||||
|
strlcat(flags, "*", sizeof(flags));
|
||||||
|
|
||||||
|
who_flags_qualifier(Client, Channel_UserModes(Chan, c),
|
||||||
|
flags, sizeof(flags));
|
||||||
|
|
||||||
|
if (!write_whoreply(Client, c, Channel_Name(Chan),
|
||||||
|
flags))
|
||||||
|
return DISCONNECTED;
|
||||||
|
count++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If there are a lot of clients, augment penalty a bit */
|
||||||
|
if (count > MAX_RPL_WHO)
|
||||||
|
IRC_SetPenalty(Client, 1);
|
||||||
|
|
||||||
|
return IRC_WriteStrClient(Client, RPL_ENDOFWHO_MSG, Client_ID(Client),
|
||||||
|
Channel_Name(Chan));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send WHO reply for a "mask target" ("WHO m*sk").
|
||||||
|
*
|
||||||
|
* @param Client Client requesting the information.
|
||||||
|
* @param Mask Mask being requested or NULL for "all" clients.
|
||||||
|
* @param OnlyOps Only display IRC operators.
|
||||||
|
* @return CONNECTED or DISCONNECTED.
|
||||||
|
*/
|
||||||
|
static bool
|
||||||
|
IRC_WHO_Mask(CLIENT *Client, char *Mask, bool OnlyOps)
|
||||||
|
{
|
||||||
|
CLIENT *c;
|
||||||
|
CL2CHAN *cl2chan;
|
||||||
|
CHANNEL *chan;
|
||||||
|
bool client_match, is_visible;
|
||||||
|
char flags[4];
|
||||||
|
int count = 0;
|
||||||
|
|
||||||
|
assert (Client != NULL);
|
||||||
|
|
||||||
|
if (Mask)
|
||||||
|
ngt_LowerStr(Mask);
|
||||||
|
|
||||||
|
IRC_SetPenalty(Client, 3);
|
||||||
|
for (c = Client_First(); c != NULL; c = Client_Next(c)) {
|
||||||
|
if (Client_Type(c) != CLIENT_USER)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (OnlyOps && !Client_HasMode(c, 'o'))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (Mask) {
|
||||||
|
/* Match pattern against user host/server/name/nick */
|
||||||
|
client_match = MatchCaseInsensitive(Mask,
|
||||||
|
Client_Hostname(c));
|
||||||
|
if (!client_match)
|
||||||
|
client_match = MatchCaseInsensitive(Mask,
|
||||||
|
Client_ID(Client_Introducer(c)));
|
||||||
|
if (!client_match)
|
||||||
|
client_match = MatchCaseInsensitive(Mask,
|
||||||
|
Client_Info(c));
|
||||||
|
if (!client_match)
|
||||||
|
client_match = MatchCaseInsensitive(Mask,
|
||||||
|
Client_ID(c));
|
||||||
|
if (!client_match)
|
||||||
|
continue; /* no match: skip this client */
|
||||||
|
}
|
||||||
|
|
||||||
|
is_visible = !Client_HasMode(c, 'i');
|
||||||
|
|
||||||
|
/* Target client is invisible, but mask matches exactly? */
|
||||||
|
if (!is_visible && Mask && strcasecmp(Client_ID(c), Mask) == 0)
|
||||||
|
is_visible = true;
|
||||||
|
|
||||||
|
/* Target still invisible, but are both on the same channel? */
|
||||||
|
if (!is_visible) {
|
||||||
|
cl2chan = Channel_FirstChannelOf(Client);
|
||||||
|
while (cl2chan && !is_visible) {
|
||||||
|
chan = Channel_GetChannel(cl2chan);
|
||||||
|
if (Channel_IsMemberOf(chan, c))
|
||||||
|
is_visible = true;
|
||||||
|
cl2chan = Channel_NextChannelOf(Client, cl2chan);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!is_visible) /* target user is not visible */
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (IRC_CheckListTooBig(Client, count, MAX_RPL_WHO, "WHO"))
|
||||||
|
break;
|
||||||
|
|
||||||
|
strlcpy(flags, who_flags_status(Client_Modes(c)), sizeof(flags));
|
||||||
|
if (strchr(Client_Modes(c), 'o'))
|
||||||
|
strlcat(flags, "*", sizeof(flags));
|
||||||
|
|
||||||
|
if (!write_whoreply(Client, c, "*", flags))
|
||||||
|
return DISCONNECTED;
|
||||||
|
count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return IRC_WriteStrClient(Client, RPL_ENDOFWHO_MSG, Client_ID(Client),
|
||||||
|
Mask ? Mask : "*");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generate WHOIS reply of one actual client.
|
||||||
|
*
|
||||||
|
* @param Client The client from which this command has been received.
|
||||||
|
* @param from The client requesting the information ("originator").
|
||||||
|
* @param c The client of which information should be returned.
|
||||||
|
* @return CONNECTED or DISCONNECTED.
|
||||||
|
*/
|
||||||
|
static bool
|
||||||
|
IRC_WHOIS_SendReply(CLIENT *Client, CLIENT *from, CLIENT *c)
|
||||||
|
{
|
||||||
|
char str[LINE_LEN + 1];
|
||||||
|
CL2CHAN *cl2chan;
|
||||||
|
CHANNEL *chan;
|
||||||
|
|
||||||
|
assert(Client != NULL);
|
||||||
|
assert(from != NULL);
|
||||||
|
assert(c != NULL);
|
||||||
|
|
||||||
|
/* Nick, user, hostname and client info */
|
||||||
|
if (!IRC_WriteStrClient(from, RPL_WHOISUSER_MSG, Client_ID(from),
|
||||||
|
Client_ID(c), Client_User(c),
|
||||||
|
Client_HostnameDisplayed(c), Client_Info(c)))
|
||||||
|
return DISCONNECTED;
|
||||||
|
|
||||||
|
/* Server */
|
||||||
|
if (!IRC_WriteStrClient(from, RPL_WHOISSERVER_MSG, Client_ID(from),
|
||||||
|
Client_ID(c), Client_ID(Client_Introducer(c)),
|
||||||
|
Client_Info(Client_Introducer(c))))
|
||||||
|
return DISCONNECTED;
|
||||||
|
|
||||||
|
/* Channels */
|
||||||
|
snprintf(str, sizeof(str), RPL_WHOISCHANNELS_MSG,
|
||||||
|
Client_ID(from), Client_ID(c));
|
||||||
|
cl2chan = Channel_FirstChannelOf(c);
|
||||||
|
while (cl2chan) {
|
||||||
|
chan = Channel_GetChannel(cl2chan);
|
||||||
|
assert(chan != NULL);
|
||||||
|
|
||||||
|
/* next */
|
||||||
|
cl2chan = Channel_NextChannelOf(c, cl2chan);
|
||||||
|
|
||||||
|
/* Secret channel? */
|
||||||
|
if (strchr(Channel_Modes(chan), 's')
|
||||||
|
&& !Channel_IsMemberOf(chan, Client))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* Local channel and request is not from a user? */
|
||||||
|
if (Client_Type(Client) == CLIENT_SERVER
|
||||||
|
&& Channel_IsLocal(chan))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* Concatenate channel names */
|
||||||
|
if (str[strlen(str) - 1] != ':')
|
||||||
|
strlcat(str, " ", sizeof(str));
|
||||||
|
|
||||||
|
who_flags_qualifier(Client, Channel_UserModes(chan, c),
|
||||||
|
str, sizeof(str));
|
||||||
|
strlcat(str, Channel_Name(chan), sizeof(str));
|
||||||
|
|
||||||
|
if (strlen(str) > (LINE_LEN - CHANNEL_NAME_LEN - 4)) {
|
||||||
|
/* Line becomes too long: send it! */
|
||||||
|
if (!IRC_WriteStrClient(Client, "%s", str))
|
||||||
|
return DISCONNECTED;
|
||||||
|
snprintf(str, sizeof(str), RPL_WHOISCHANNELS_MSG,
|
||||||
|
Client_ID(from), Client_ID(c));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(str[strlen(str) - 1] != ':') {
|
||||||
|
/* There is data left to send: */
|
||||||
|
if (!IRC_WriteStrClient(Client, "%s", str))
|
||||||
|
return DISCONNECTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* IRC-Operator? */
|
||||||
|
if (Client_HasMode(c, 'o') &&
|
||||||
|
!IRC_WriteStrClient(from, RPL_WHOISOPERATOR_MSG,
|
||||||
|
Client_ID(from), Client_ID(c)))
|
||||||
|
return DISCONNECTED;
|
||||||
|
|
||||||
|
/* IRC-Bot? */
|
||||||
|
if (Client_HasMode(c, 'B') &&
|
||||||
|
!IRC_WriteStrClient(from, RPL_WHOISBOT_MSG,
|
||||||
|
Client_ID(from), Client_ID(c)))
|
||||||
|
return DISCONNECTED;
|
||||||
|
|
||||||
|
/* Connected using SSL? */
|
||||||
|
if (Conn_UsesSSL(Client_Conn(c)) &&
|
||||||
|
!IRC_WriteStrClient(from, RPL_WHOISSSL_MSG, Client_ID(from),
|
||||||
|
Client_ID(c)))
|
||||||
|
return DISCONNECTED;
|
||||||
|
|
||||||
|
/* Registered nickname? */
|
||||||
|
if (Client_HasMode(c, 'R') &&
|
||||||
|
!IRC_WriteStrClient(from, RPL_WHOISREGNICK_MSG,
|
||||||
|
Client_ID(from), Client_ID(c)))
|
||||||
|
return DISCONNECTED;
|
||||||
|
|
||||||
|
/* Local client and requester is the user itself or an IRC Op? */
|
||||||
|
if (Client_Conn(c) > NONE &&
|
||||||
|
(from == c || (!Conf_MorePrivacy && Client_HasMode(from, 'o')))) {
|
||||||
|
/* Client hostname */
|
||||||
|
if (!IRC_WriteStrClient(from, RPL_WHOISHOST_MSG,
|
||||||
|
Client_ID(from), Client_ID(c), Client_Hostname(c),
|
||||||
|
Conn_GetIPAInfo(Client_Conn(c))))
|
||||||
|
return DISCONNECTED;
|
||||||
|
/* Client modes */
|
||||||
|
if (!IRC_WriteStrClient(from, RPL_WHOISMODES_MSG,
|
||||||
|
Client_ID(from), Client_ID(c), Client_Modes(c)))
|
||||||
|
return DISCONNECTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Idle and signon time (local clients only!) */
|
||||||
|
if (!Conf_MorePrivacy && Client_Conn(c) > NONE &&
|
||||||
|
!IRC_WriteStrClient(from, RPL_WHOISIDLE_MSG,
|
||||||
|
Client_ID(from), Client_ID(c),
|
||||||
|
(unsigned long)Conn_GetIdle(Client_Conn(c)),
|
||||||
|
(unsigned long)Conn_GetSignon(Client_Conn(c))))
|
||||||
|
return DISCONNECTED;
|
||||||
|
|
||||||
|
/* Away? */
|
||||||
|
if (Client_HasMode(c, 'a') &&
|
||||||
|
!IRC_WriteStrClient(from, RPL_AWAY_MSG,
|
||||||
|
Client_ID(from), Client_ID(c), Client_Away(c)))
|
||||||
|
return DISCONNECTED;
|
||||||
|
|
||||||
|
return CONNECTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool
|
||||||
|
WHOWAS_EntryWrite(CLIENT *prefix, WHOWAS *entry)
|
||||||
|
{
|
||||||
|
char t_str[60];
|
||||||
|
|
||||||
|
(void)strftime(t_str, sizeof(t_str), "%a %b %d %H:%M:%S %Y",
|
||||||
|
localtime(&entry->time));
|
||||||
|
|
||||||
|
if (!IRC_WriteStrClient(prefix, RPL_WHOWASUSER_MSG, Client_ID(prefix),
|
||||||
|
entry->id, entry->user, entry->host, entry->info))
|
||||||
|
return DISCONNECTED;
|
||||||
|
|
||||||
|
return IRC_WriteStrClient(prefix, RPL_WHOISSERVER_MSG, Client_ID(prefix),
|
||||||
|
entry->id, entry->server, t_str);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool
|
||||||
|
Show_MOTD_Start(CLIENT *Client)
|
||||||
|
{
|
||||||
|
return IRC_WriteStrClient(Client, RPL_MOTDSTART_MSG,
|
||||||
|
Client_ID( Client ), Client_ID( Client_ThisServer( )));
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool
|
||||||
|
Show_MOTD_Sendline(CLIENT *Client, const char *msg)
|
||||||
|
{
|
||||||
|
return IRC_WriteStrClient(Client, RPL_MOTD_MSG, Client_ID( Client ), msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool
|
||||||
|
Show_MOTD_End(CLIENT *Client)
|
||||||
|
{
|
||||||
|
if (!IRC_WriteStrClient(Client, RPL_ENDOFMOTD_MSG, Client_ID(Client)))
|
||||||
|
return DISCONNECTED;
|
||||||
|
|
||||||
|
if (*Conf_CloakHost)
|
||||||
|
return IRC_WriteStrClient(Client, RPL_HOSTHIDDEN_MSG,
|
||||||
|
Client_ID(Client),
|
||||||
|
Client_Hostname(Client));
|
||||||
|
|
||||||
|
return CONNECTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef SSL_SUPPORT
|
||||||
|
static bool Show_MOTD_SSLInfo(CLIENT *Client)
|
||||||
|
{
|
||||||
|
bool ret = true;
|
||||||
|
char buf[COMMAND_LEN] = "Connected using Cipher ";
|
||||||
|
|
||||||
|
if (!Conn_GetCipherInfo(Client_Conn(Client), buf + 23, sizeof buf - 23))
|
||||||
|
return true;
|
||||||
|
|
||||||
|
if (!Show_MOTD_Sendline(Client, buf))
|
||||||
|
ret = false;
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
static inline bool
|
||||||
|
Show_MOTD_SSLInfo(UNUSED CLIENT *c)
|
||||||
|
{ return true; }
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Global functions */
|
||||||
|
|
||||||
GLOBAL bool
|
GLOBAL bool
|
||||||
IRC_ADMIN(CLIENT *Client, REQUEST *Req )
|
IRC_ADMIN(CLIENT *Client, REQUEST *Req )
|
||||||
@ -467,40 +902,6 @@ IRC_NAMES( CLIENT *Client, REQUEST *Req )
|
|||||||
} /* IRC_NAMES */
|
} /* IRC_NAMES */
|
||||||
|
|
||||||
|
|
||||||
static unsigned int
|
|
||||||
t_diff(time_t *t, const time_t d)
|
|
||||||
{
|
|
||||||
time_t diff, remain;
|
|
||||||
|
|
||||||
diff = *t / d;
|
|
||||||
remain = diff * d;
|
|
||||||
*t -= remain;
|
|
||||||
|
|
||||||
return (unsigned int)diff;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static unsigned int
|
|
||||||
uptime_days(time_t *now)
|
|
||||||
{
|
|
||||||
return t_diff(now, 60 * 60 * 24);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static unsigned int
|
|
||||||
uptime_hrs(time_t *now)
|
|
||||||
{
|
|
||||||
return t_diff(now, 60 * 60);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static unsigned int
|
|
||||||
uptime_mins(time_t *now)
|
|
||||||
{
|
|
||||||
return t_diff(now, 60);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handler for the IRC command "STATS".
|
* Handler for the IRC command "STATS".
|
||||||
* See RFC 2812 section 3.4.4.
|
* See RFC 2812 section 3.4.4.
|
||||||
@ -787,221 +1188,6 @@ IRC_VERSION( CLIENT *Client, REQUEST *Req )
|
|||||||
} /* IRC_VERSION */
|
} /* IRC_VERSION */
|
||||||
|
|
||||||
|
|
||||||
static bool
|
|
||||||
write_whoreply(CLIENT *Client, CLIENT *c, const char *channelname, const char *flags)
|
|
||||||
{
|
|
||||||
return IRC_WriteStrClient(Client, RPL_WHOREPLY_MSG, Client_ID(Client),
|
|
||||||
channelname, Client_User(c),
|
|
||||||
Client_HostnameDisplayed(c),
|
|
||||||
Client_ID(Client_Introducer(c)), Client_ID(c),
|
|
||||||
flags, Client_Hops(c), Client_Info(c));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static const char *
|
|
||||||
who_flags_status(const char *client_modes)
|
|
||||||
{
|
|
||||||
if (strchr(client_modes, 'a'))
|
|
||||||
return "G"; /* away */
|
|
||||||
return "H";
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return channel user mode prefix(es).
|
|
||||||
*
|
|
||||||
* @param Client The client requesting the mode prefixes.
|
|
||||||
* @param chan_user_modes String with channel user modes.
|
|
||||||
* @param str String buffer to which the prefix(es) will be appended.
|
|
||||||
* @param len Size of "str" buffer.
|
|
||||||
* @return Pointer to "str".
|
|
||||||
*/
|
|
||||||
static char *
|
|
||||||
who_flags_qualifier(CLIENT *Client, const char *chan_user_modes,
|
|
||||||
char *str, size_t len)
|
|
||||||
{
|
|
||||||
assert(Client != NULL);
|
|
||||||
|
|
||||||
if (Client_Cap(Client) & CLIENT_CAP_MULTI_PREFIX) {
|
|
||||||
if (strchr(chan_user_modes, 'q'))
|
|
||||||
strlcat(str, "~", len);
|
|
||||||
if (strchr(chan_user_modes, 'a'))
|
|
||||||
strlcat(str, "&", len);
|
|
||||||
if (strchr(chan_user_modes, 'o'))
|
|
||||||
strlcat(str, "@", len);
|
|
||||||
if (strchr(chan_user_modes, 'h'))
|
|
||||||
strlcat(str, "%", len);
|
|
||||||
if (strchr(chan_user_modes, 'v'))
|
|
||||||
strlcat(str, "+", len);
|
|
||||||
|
|
||||||
return str;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (strchr(chan_user_modes, 'q'))
|
|
||||||
strlcat(str, "~", len);
|
|
||||||
else if (strchr(chan_user_modes, 'a'))
|
|
||||||
strlcat(str, "&", len);
|
|
||||||
else if (strchr(chan_user_modes, 'o'))
|
|
||||||
strlcat(str, "@", len);
|
|
||||||
else if (strchr(chan_user_modes, 'h'))
|
|
||||||
strlcat(str, "%", len);
|
|
||||||
else if (strchr(chan_user_modes, 'v'))
|
|
||||||
strlcat(str, "+", len);
|
|
||||||
|
|
||||||
return str;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send WHO reply for a "channel target" ("WHO #channel").
|
|
||||||
*
|
|
||||||
* @param Client Client requesting the information.
|
|
||||||
* @param Chan Channel being requested.
|
|
||||||
* @param OnlyOps Only display IRC operators.
|
|
||||||
* @return CONNECTED or DISCONNECTED.
|
|
||||||
*/
|
|
||||||
static bool
|
|
||||||
IRC_WHO_Channel(CLIENT *Client, CHANNEL *Chan, bool OnlyOps)
|
|
||||||
{
|
|
||||||
bool is_visible, is_member, is_ircop;
|
|
||||||
CL2CHAN *cl2chan;
|
|
||||||
const char *client_modes;
|
|
||||||
char flags[10];
|
|
||||||
CLIENT *c;
|
|
||||||
int count = 0;
|
|
||||||
|
|
||||||
assert( Client != NULL );
|
|
||||||
assert( Chan != NULL );
|
|
||||||
|
|
||||||
IRC_SetPenalty(Client, 1);
|
|
||||||
|
|
||||||
is_member = Channel_IsMemberOf(Chan, Client);
|
|
||||||
|
|
||||||
/* Secret channel? */
|
|
||||||
if (!is_member && strchr(Channel_Modes(Chan), 's'))
|
|
||||||
return IRC_WriteStrClient(Client, RPL_ENDOFWHO_MSG,
|
|
||||||
Client_ID(Client), Channel_Name(Chan));
|
|
||||||
|
|
||||||
cl2chan = Channel_FirstMember(Chan);
|
|
||||||
for (; cl2chan ; cl2chan = Channel_NextMember(Chan, cl2chan)) {
|
|
||||||
c = Channel_GetClient(cl2chan);
|
|
||||||
|
|
||||||
client_modes = Client_Modes(c);
|
|
||||||
is_ircop = strchr(client_modes, 'o') != NULL;
|
|
||||||
if (OnlyOps && !is_ircop)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
is_visible = strchr(client_modes, 'i') == NULL;
|
|
||||||
if (is_member || is_visible) {
|
|
||||||
strlcpy(flags, who_flags_status(client_modes),
|
|
||||||
sizeof(flags));
|
|
||||||
if (is_ircop)
|
|
||||||
strlcat(flags, "*", sizeof(flags));
|
|
||||||
|
|
||||||
who_flags_qualifier(Client, Channel_UserModes(Chan, c),
|
|
||||||
flags, sizeof(flags));
|
|
||||||
|
|
||||||
if (!write_whoreply(Client, c, Channel_Name(Chan),
|
|
||||||
flags))
|
|
||||||
return DISCONNECTED;
|
|
||||||
count++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* If there are a lot of clients, augment penalty a bit */
|
|
||||||
if (count > MAX_RPL_WHO)
|
|
||||||
IRC_SetPenalty(Client, 1);
|
|
||||||
|
|
||||||
return IRC_WriteStrClient(Client, RPL_ENDOFWHO_MSG, Client_ID(Client),
|
|
||||||
Channel_Name(Chan));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send WHO reply for a "mask target" ("WHO m*sk").
|
|
||||||
*
|
|
||||||
* @param Client Client requesting the information.
|
|
||||||
* @param Mask Mask being requested or NULL for "all" clients.
|
|
||||||
* @param OnlyOps Only display IRC operators.
|
|
||||||
* @return CONNECTED or DISCONNECTED.
|
|
||||||
*/
|
|
||||||
static bool
|
|
||||||
IRC_WHO_Mask(CLIENT *Client, char *Mask, bool OnlyOps)
|
|
||||||
{
|
|
||||||
CLIENT *c;
|
|
||||||
CL2CHAN *cl2chan;
|
|
||||||
CHANNEL *chan;
|
|
||||||
bool client_match, is_visible;
|
|
||||||
char flags[4];
|
|
||||||
int count = 0;
|
|
||||||
|
|
||||||
assert (Client != NULL);
|
|
||||||
|
|
||||||
if (Mask)
|
|
||||||
ngt_LowerStr(Mask);
|
|
||||||
|
|
||||||
IRC_SetPenalty(Client, 3);
|
|
||||||
for (c = Client_First(); c != NULL; c = Client_Next(c)) {
|
|
||||||
if (Client_Type(c) != CLIENT_USER)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (OnlyOps && !Client_HasMode(c, 'o'))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (Mask) {
|
|
||||||
/* Match pattern against user host/server/name/nick */
|
|
||||||
client_match = MatchCaseInsensitive(Mask,
|
|
||||||
Client_Hostname(c));
|
|
||||||
if (!client_match)
|
|
||||||
client_match = MatchCaseInsensitive(Mask,
|
|
||||||
Client_ID(Client_Introducer(c)));
|
|
||||||
if (!client_match)
|
|
||||||
client_match = MatchCaseInsensitive(Mask,
|
|
||||||
Client_Info(c));
|
|
||||||
if (!client_match)
|
|
||||||
client_match = MatchCaseInsensitive(Mask,
|
|
||||||
Client_ID(c));
|
|
||||||
if (!client_match)
|
|
||||||
continue; /* no match: skip this client */
|
|
||||||
}
|
|
||||||
|
|
||||||
is_visible = !Client_HasMode(c, 'i');
|
|
||||||
|
|
||||||
/* Target client is invisible, but mask matches exactly? */
|
|
||||||
if (!is_visible && Mask && strcasecmp(Client_ID(c), Mask) == 0)
|
|
||||||
is_visible = true;
|
|
||||||
|
|
||||||
/* Target still invisible, but are both on the same channel? */
|
|
||||||
if (!is_visible) {
|
|
||||||
cl2chan = Channel_FirstChannelOf(Client);
|
|
||||||
while (cl2chan && !is_visible) {
|
|
||||||
chan = Channel_GetChannel(cl2chan);
|
|
||||||
if (Channel_IsMemberOf(chan, c))
|
|
||||||
is_visible = true;
|
|
||||||
cl2chan = Channel_NextChannelOf(Client, cl2chan);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!is_visible) /* target user is not visible */
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (IRC_CheckListTooBig(Client, count, MAX_RPL_WHO, "WHO"))
|
|
||||||
break;
|
|
||||||
|
|
||||||
strlcpy(flags, who_flags_status(Client_Modes(c)), sizeof(flags));
|
|
||||||
if (strchr(Client_Modes(c), 'o'))
|
|
||||||
strlcat(flags, "*", sizeof(flags));
|
|
||||||
|
|
||||||
if (!write_whoreply(Client, c, "*", flags))
|
|
||||||
return DISCONNECTED;
|
|
||||||
count++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return IRC_WriteStrClient(Client, RPL_ENDOFWHO_MSG, Client_ID(Client),
|
|
||||||
Mask ? Mask : "*");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handler for the IRC "WHO" command.
|
* Handler for the IRC "WHO" command.
|
||||||
*
|
*
|
||||||
@ -1059,136 +1245,6 @@ IRC_WHO(CLIENT *Client, REQUEST *Req)
|
|||||||
} /* IRC_WHO */
|
} /* IRC_WHO */
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate WHOIS reply of one actual client.
|
|
||||||
*
|
|
||||||
* @param Client The client from which this command has been received.
|
|
||||||
* @param from The client requesting the information ("originator").
|
|
||||||
* @param c The client of which information should be returned.
|
|
||||||
* @return CONNECTED or DISCONNECTED.
|
|
||||||
*/
|
|
||||||
static bool
|
|
||||||
IRC_WHOIS_SendReply(CLIENT *Client, CLIENT *from, CLIENT *c)
|
|
||||||
{
|
|
||||||
char str[LINE_LEN + 1];
|
|
||||||
CL2CHAN *cl2chan;
|
|
||||||
CHANNEL *chan;
|
|
||||||
|
|
||||||
assert(Client != NULL);
|
|
||||||
assert(from != NULL);
|
|
||||||
assert(c != NULL);
|
|
||||||
|
|
||||||
/* Nick, user, hostname and client info */
|
|
||||||
if (!IRC_WriteStrClient(from, RPL_WHOISUSER_MSG, Client_ID(from),
|
|
||||||
Client_ID(c), Client_User(c),
|
|
||||||
Client_HostnameDisplayed(c), Client_Info(c)))
|
|
||||||
return DISCONNECTED;
|
|
||||||
|
|
||||||
/* Server */
|
|
||||||
if (!IRC_WriteStrClient(from, RPL_WHOISSERVER_MSG, Client_ID(from),
|
|
||||||
Client_ID(c), Client_ID(Client_Introducer(c)),
|
|
||||||
Client_Info(Client_Introducer(c))))
|
|
||||||
return DISCONNECTED;
|
|
||||||
|
|
||||||
/* Channels */
|
|
||||||
snprintf(str, sizeof(str), RPL_WHOISCHANNELS_MSG,
|
|
||||||
Client_ID(from), Client_ID(c));
|
|
||||||
cl2chan = Channel_FirstChannelOf(c);
|
|
||||||
while (cl2chan) {
|
|
||||||
chan = Channel_GetChannel(cl2chan);
|
|
||||||
assert(chan != NULL);
|
|
||||||
|
|
||||||
/* next */
|
|
||||||
cl2chan = Channel_NextChannelOf(c, cl2chan);
|
|
||||||
|
|
||||||
/* Secret channel? */
|
|
||||||
if (strchr(Channel_Modes(chan), 's')
|
|
||||||
&& !Channel_IsMemberOf(chan, Client))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
/* Local channel and request is not from a user? */
|
|
||||||
if (Client_Type(Client) == CLIENT_SERVER
|
|
||||||
&& Channel_IsLocal(chan))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
/* Concatenate channel names */
|
|
||||||
if (str[strlen(str) - 1] != ':')
|
|
||||||
strlcat(str, " ", sizeof(str));
|
|
||||||
|
|
||||||
who_flags_qualifier(Client, Channel_UserModes(chan, c),
|
|
||||||
str, sizeof(str));
|
|
||||||
strlcat(str, Channel_Name(chan), sizeof(str));
|
|
||||||
|
|
||||||
if (strlen(str) > (LINE_LEN - CHANNEL_NAME_LEN - 4)) {
|
|
||||||
/* Line becomes too long: send it! */
|
|
||||||
if (!IRC_WriteStrClient(Client, "%s", str))
|
|
||||||
return DISCONNECTED;
|
|
||||||
snprintf(str, sizeof(str), RPL_WHOISCHANNELS_MSG,
|
|
||||||
Client_ID(from), Client_ID(c));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if(str[strlen(str) - 1] != ':') {
|
|
||||||
/* There is data left to send: */
|
|
||||||
if (!IRC_WriteStrClient(Client, "%s", str))
|
|
||||||
return DISCONNECTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* IRC-Operator? */
|
|
||||||
if (Client_HasMode(c, 'o') &&
|
|
||||||
!IRC_WriteStrClient(from, RPL_WHOISOPERATOR_MSG,
|
|
||||||
Client_ID(from), Client_ID(c)))
|
|
||||||
return DISCONNECTED;
|
|
||||||
|
|
||||||
/* IRC-Bot? */
|
|
||||||
if (Client_HasMode(c, 'B') &&
|
|
||||||
!IRC_WriteStrClient(from, RPL_WHOISBOT_MSG,
|
|
||||||
Client_ID(from), Client_ID(c)))
|
|
||||||
return DISCONNECTED;
|
|
||||||
|
|
||||||
/* Connected using SSL? */
|
|
||||||
if (Conn_UsesSSL(Client_Conn(c)) &&
|
|
||||||
!IRC_WriteStrClient(from, RPL_WHOISSSL_MSG, Client_ID(from),
|
|
||||||
Client_ID(c)))
|
|
||||||
return DISCONNECTED;
|
|
||||||
|
|
||||||
/* Registered nickname? */
|
|
||||||
if (Client_HasMode(c, 'R') &&
|
|
||||||
!IRC_WriteStrClient(from, RPL_WHOISREGNICK_MSG,
|
|
||||||
Client_ID(from), Client_ID(c)))
|
|
||||||
return DISCONNECTED;
|
|
||||||
|
|
||||||
/* Local client and requester is the user itself or an IRC Op? */
|
|
||||||
if (Client_Conn(c) > NONE &&
|
|
||||||
(from == c || (!Conf_MorePrivacy && Client_HasMode(from, 'o')))) {
|
|
||||||
/* Client hostname */
|
|
||||||
if (!IRC_WriteStrClient(from, RPL_WHOISHOST_MSG,
|
|
||||||
Client_ID(from), Client_ID(c), Client_Hostname(c),
|
|
||||||
Conn_GetIPAInfo(Client_Conn(c))))
|
|
||||||
return DISCONNECTED;
|
|
||||||
/* Client modes */
|
|
||||||
if (!IRC_WriteStrClient(from, RPL_WHOISMODES_MSG,
|
|
||||||
Client_ID(from), Client_ID(c), Client_Modes(c)))
|
|
||||||
return DISCONNECTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Idle and signon time (local clients only!) */
|
|
||||||
if (!Conf_MorePrivacy && Client_Conn(c) > NONE &&
|
|
||||||
!IRC_WriteStrClient(from, RPL_WHOISIDLE_MSG,
|
|
||||||
Client_ID(from), Client_ID(c),
|
|
||||||
(unsigned long)Conn_GetIdle(Client_Conn(c)),
|
|
||||||
(unsigned long)Conn_GetSignon(Client_Conn(c))))
|
|
||||||
return DISCONNECTED;
|
|
||||||
|
|
||||||
/* Away? */
|
|
||||||
if (Client_HasMode(c, 'a') &&
|
|
||||||
!IRC_WriteStrClient(from, RPL_AWAY_MSG,
|
|
||||||
Client_ID(from), Client_ID(c), Client_Away(c)))
|
|
||||||
return DISCONNECTED;
|
|
||||||
|
|
||||||
return CONNECTED;
|
|
||||||
} /* IRC_WHOIS_SendReply */
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handler for the IRC "WHOIS" command.
|
* Handler for the IRC "WHOIS" command.
|
||||||
*
|
*
|
||||||
@ -1309,22 +1365,6 @@ IRC_WHOIS( CLIENT *Client, REQUEST *Req )
|
|||||||
} /* IRC_WHOIS */
|
} /* IRC_WHOIS */
|
||||||
|
|
||||||
|
|
||||||
static bool
|
|
||||||
WHOWAS_EntryWrite(CLIENT *prefix, WHOWAS *entry)
|
|
||||||
{
|
|
||||||
char t_str[60];
|
|
||||||
|
|
||||||
(void)strftime(t_str, sizeof(t_str), "%a %b %d %H:%M:%S %Y",
|
|
||||||
localtime(&entry->time));
|
|
||||||
|
|
||||||
if (!IRC_WriteStrClient(prefix, RPL_WHOWASUSER_MSG, Client_ID(prefix),
|
|
||||||
entry->id, entry->user, entry->host, entry->info))
|
|
||||||
return DISCONNECTED;
|
|
||||||
|
|
||||||
return IRC_WriteStrClient(prefix, RPL_WHOISSERVER_MSG, Client_ID(prefix),
|
|
||||||
entry->id, entry->server, t_str);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* IRC "WHOWAS" function.
|
* IRC "WHOWAS" function.
|
||||||
* This function implements the IRC command "WHOWHAS". It handles local
|
* This function implements the IRC command "WHOWHAS". It handles local
|
||||||
@ -1503,53 +1543,6 @@ IRC_Send_LUSERS(CLIENT *Client)
|
|||||||
} /* IRC_Send_LUSERS */
|
} /* IRC_Send_LUSERS */
|
||||||
|
|
||||||
|
|
||||||
static bool
|
|
||||||
Show_MOTD_Start(CLIENT *Client)
|
|
||||||
{
|
|
||||||
return IRC_WriteStrClient(Client, RPL_MOTDSTART_MSG,
|
|
||||||
Client_ID( Client ), Client_ID( Client_ThisServer( )));
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool
|
|
||||||
Show_MOTD_Sendline(CLIENT *Client, const char *msg)
|
|
||||||
{
|
|
||||||
return IRC_WriteStrClient(Client, RPL_MOTD_MSG, Client_ID( Client ), msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool
|
|
||||||
Show_MOTD_End(CLIENT *Client)
|
|
||||||
{
|
|
||||||
if (!IRC_WriteStrClient(Client, RPL_ENDOFMOTD_MSG, Client_ID(Client)))
|
|
||||||
return DISCONNECTED;
|
|
||||||
|
|
||||||
if (*Conf_CloakHost)
|
|
||||||
return IRC_WriteStrClient(Client, RPL_HOSTHIDDEN_MSG,
|
|
||||||
Client_ID(Client),
|
|
||||||
Client_Hostname(Client));
|
|
||||||
|
|
||||||
return CONNECTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef SSL_SUPPORT
|
|
||||||
static bool Show_MOTD_SSLInfo(CLIENT *Client)
|
|
||||||
{
|
|
||||||
bool ret = true;
|
|
||||||
char buf[COMMAND_LEN] = "Connected using Cipher ";
|
|
||||||
|
|
||||||
if (!Conn_GetCipherInfo(Client_Conn(Client), buf + 23, sizeof buf - 23))
|
|
||||||
return true;
|
|
||||||
|
|
||||||
if (!Show_MOTD_Sendline(Client, buf))
|
|
||||||
ret = false;
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
static inline bool
|
|
||||||
Show_MOTD_SSLInfo(UNUSED CLIENT *c)
|
|
||||||
{ return true; }
|
|
||||||
#endif
|
|
||||||
|
|
||||||
GLOBAL bool
|
GLOBAL bool
|
||||||
IRC_Show_MOTD( CLIENT *Client )
|
IRC_Show_MOTD( CLIENT *Client )
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user