From 70254a5553b97109df8417c2a6fcb71d55ab371d Mon Sep 17 00:00:00 2001 From: Scott Perry Date: Mon, 12 May 2008 12:59:55 +0200 Subject: [PATCH] This patch implements a (maybe) compliant WHOWAS command. It is hard to test this in the test suite because we 1) shouldn't rely on previous tests populating WHOWAS and 2) don't connect a user for more than 30 seconds. Also makes WHOWAS return ERR_NONICKNAMEGIVEN_MSG as implied by RFC. --- src/ngircd/irc-info.c | 149 +++++++++++++++++++++++------------------- src/ngircd/messages.h | 1 + 2 files changed, 81 insertions(+), 69 deletions(-) diff --git a/src/ngircd/irc-info.c b/src/ngircd/irc-info.c index 86e1ace2..43f2c1b6 100644 --- a/src/ngircd/irc-info.c +++ b/src/ngircd/irc-info.c @@ -994,6 +994,22 @@ IRC_WHOIS( CLIENT *Client, REQUEST *Req ) } /* 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. * This function implements the IRC command "WHOWHAS". It handles local @@ -1004,40 +1020,41 @@ IRC_WHOWAS( CLIENT *Client, REQUEST *Req ) { CLIENT *target, *prefix; WHOWAS *whowas; - int max, last, count, i; - char t_str[60]; - + char tok_buf[COMMAND_LEN]; + int max, last, count, i, nc; + const char *nick; + assert( Client != NULL ); assert( Req != NULL ); /* Wrong number of parameters? */ - if(( Req->argc < 1 ) || ( Req->argc > 3 )) - return IRC_WriteStrClient( Client, ERR_NEEDMOREPARAMS_MSG, - Client_ID( Client ), Req->command ); + if (Req->argc > 3) + return IRC_WriteStrClient(Client, ERR_NEEDMOREPARAMS_MSG, + Client_ID(Client), Req->command); + if (Req->argc < 1) + return IRC_WriteStrClient(Client, ERR_NONICKNAMEGIVEN_MSG, Client_ID(Client)); - /* Search taget */ - if( Req->argc == 3 ) - target = Client_Search( Req->argv[2] ); + /* Search target */ + if (Req->argc == 3) + target = Client_Search(Req->argv[2]); else - target = Client_ThisServer( ); + target = Client_ThisServer(); /* Get prefix */ - if( Client_Type( Client ) == CLIENT_SERVER ) - prefix = Client_Search( Req->prefix ); + if (Client_Type(Client) == CLIENT_SERVER) + prefix = Client_Search(Req->prefix); else prefix = Client; - if( ! prefix ) - return IRC_WriteStrClient( Client, ERR_NOSUCHNICK_MSG, - Client_ID( Client ), Req->prefix ); + if (!prefix) + return IRC_WriteStrClient(Client, ERR_NOSUCHNICK_MSG, + Client_ID(Client), Req->prefix); /* Forward to other server? */ - if( target != Client_ThisServer( )) - { - if(( ! target ) || ( Client_Type( target ) != CLIENT_SERVER )) - return IRC_WriteStrClient( prefix, ERR_NOSUCHSERVER_MSG, - Client_ID( prefix ), - Req->argv[2] ); + if (target != Client_ThisServer()) { + if (!target || (Client_Type(target) != CLIENT_SERVER)) + return IRC_WriteStrClient(prefix, ERR_NOSUCHSERVER_MSG, + Client_ID(prefix), Req->argv[2]); /* Forward */ IRC_WriteStrClientPrefix( target, prefix, "WHOWAS %s %s %s", @@ -1045,58 +1062,52 @@ IRC_WHOWAS( CLIENT *Client, REQUEST *Req ) Req->argv[2] ); return CONNECTED; } - + whowas = Client_GetWhowas( ); last = Client_GetLastWhowasIndex( ); - if( last < 0 ) last = 0; - - if( Req->argc > 1 ) - { - max = atoi( Req->argv[1] ); - if( max < 1 ) max = MAX_WHOWAS; + if (last < 0) + last = 0; + + max = DEFAULT_WHOWAS; + if (Req->argc > 1) { + max = atoi(Req->argv[1]); + if (max < 1) + max = MAX_WHOWAS; } - else - max = DEFAULT_WHOWAS; - - i = last; - count = 0; - do - { - /* Used entry? */ - if( whowas[i].time > 0 && - strcasecmp( Req->argv[0], whowas[i].id ) == 0 ) - { - (void)strftime( t_str, sizeof(t_str), - "%a %b %d %H:%M:%S %Y", - localtime( &whowas[i].time )); - - if( ! IRC_WriteStrClient( prefix, RPL_WHOWASUSER_MSG, - Client_ID( prefix ), - whowas[i].id, - whowas[i].user, - whowas[i].host, - whowas[i].info )) - return DISCONNECTED; - - if( ! IRC_WriteStrClient( prefix, RPL_WHOISSERVER_MSG, - Client_ID( prefix ), - whowas[i].id, - whowas[i].server, t_str )) - return DISCONNECTED; - - count++; - if( count >= max ) break; - } - - /* previos entry */ - i--; - /* "underflow", wrap around */ - if( i < 0 ) i = MAX_WHOWAS - 1; - } while( i != last ); - - return IRC_WriteStrClient( prefix, RPL_ENDOFWHOWAS_MSG, - Client_ID( prefix ), Req->argv[0] ); + /* + * Break up the nick argument into a list of nicks, if applicable + * Can't modify Req->argv[0] because we need it for RPL_ENDOFWHOWAS_MSG. + */ + strlcpy(tok_buf, Req->argv[0], sizeof(tok_buf)); + nick = strtok(tok_buf, ","); + + for (i=last, count=0; nick != NULL ; nick = strtok(NULL, ",")) { + nc = 0; + do { + /* Used entry? */ + if (whowas[i].time > 0 && strcasecmp(nick, whowas[i].id) == 0) { + if (!WHOWAS_EntryWrite(prefix, &whowas[i])) + return DISCONNECTED; + nc++; + count++; + } + /* previous entry */ + i--; + + /* "underflow", wrap around */ + if (i < 0) + i = MAX_WHOWAS - 1; + + if (nc && count >= max) + break; + } while (i != last); + + if (nc == 0 && !IRC_WriteStrClient(prefix, ERR_WASNOSUCHNICK_MSG, + Client_ID(prefix), nick)) + return DISCONNECTED; + } + return IRC_WriteStrClient(prefix, RPL_ENDOFWHOWAS_MSG, Client_ID(prefix), Req->argv[0]); } /* IRC_WHOWAS */ diff --git a/src/ngircd/messages.h b/src/ngircd/messages.h index bde1c095..36d6f48e 100644 --- a/src/ngircd/messages.h +++ b/src/ngircd/messages.h @@ -99,6 +99,7 @@ #define ERR_NOTEXTTOSEND_MSG "412 %s :No text to send" #define ERR_UNKNOWNCOMMAND_MSG "421 %s %s :Unknown command" #define ERR_NOMOTD_MSG "422 %s :MOTD file is missing" +#define ERR_NONICKNAMEGIVEN_MSG "431 %s :No nickname given" #define ERR_ERRONEUSNICKNAME_MSG "432 %s %s :Erroneous nickname" #define ERR_NICKNAMEINUSE_MSG "433 %s %s :Nickname already in use" #define ERR_USERNOTINCHANNEL_MSG "441 %s %s %s :They aren't on that channel" -- 2.40.0