* ngIRCd -- The Next Generation IRC Daemon
* Copyright (c)2001,2002 by Alexander Barton (alex@barton.de)
*
- * Dieses Programm ist freie Software. Sie koennen es unter den Bedingungen
- * der GNU General Public License (GPL), wie von der Free Software Foundation
- * herausgegeben, weitergeben und/oder modifizieren, entweder unter Version 2
- * der Lizenz oder (wenn Sie es wuenschen) jeder spaeteren Version.
- * Naehere Informationen entnehmen Sie bitter der Datei COPYING. Eine Liste
- * der an ngIRCd beteiligten Autoren finden Sie in der Datei AUTHORS.
+ * 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 2 of the License, or
+ * (at your option) any later version.
+ * Please read the file COPYING, README and AUTHORS for more information.
*
- * $Id: client.c,v 1.60 2002/09/03 18:54:31 alex Exp $
- *
- * client.c: Management aller Clients
- *
- * Der Begriff "Client" ist in diesem Fall evtl. etwas verwirrend: Clients sind
- * alle Verbindungen, die im gesamten(!) IRC-Netzwerk bekannt sind. Das sind IRC-
- * Clients (User), andere Server und IRC-Services.
- * Ueber welchen IRC-Server die Verbindung nun tatsaechlich in das Netzwerk her-
- * gestellt wurde, muss der jeweiligen Struktur entnommen werden. Ist es dieser
- * Server gewesen, so existiert eine entsprechende CONNECTION-Struktur.
+ * Client management.
*/
#include "portab.h"
+static char UNUSED id[] = "$Id: client.c,v 1.74 2003/03/31 15:54:21 alex Exp $";
+
#include "imp.h"
#include <assert.h>
#include <unistd.h>
#include <exp.h>
+#define GETID_LEN (CLIENT_NICK_LEN-1) + 1 + (CLIENT_USER_LEN-1) + 1 + (CLIENT_HOST_LEN-1) + 1
+
+
LOCAL CLIENT *This_Server, *My_Clients;
-LOCAL CHAR GetID_Buffer[CLIENT_ID_LEN];
+LOCAL CHAR GetID_Buffer[GETID_LEN];
-LOCAL INT Count PARAMS(( CLIENT_TYPE Type ));
-LOCAL INT MyCount PARAMS(( CLIENT_TYPE Type ));
+LOCAL LONG Count PARAMS(( CLIENT_TYPE Type ));
+LOCAL LONG MyCount PARAMS(( CLIENT_TYPE Type ));
LOCAL CLIENT *New_Client_Struct PARAMS(( VOID ));
LOCAL VOID Generate_MyToken PARAMS(( CLIENT *Client ));
+LOCAL VOID Adjust_Counters PARAMS(( CLIENT *Client ));
+
+#ifndef Client_DestroyNow
+GLOBAL VOID Client_DestroyNow PARAMS((CLIENT *Client ));
+#endif
+
+
+LONG Max_Users = 0, My_Max_Users = 0;
GLOBAL VOID
if( ! This_Server )
{
Log( LOG_EMERG, "Can't allocate client structure for server! Going down." );
- Log( LOG_ALERT, "%s exiting due to fatal errors!", PACKAGE );
+ Log( LOG_ALERT, "%s exiting due to fatal errors!", PACKAGE_NAME );
exit( 1 );
}
gethostname( This_Server->host, CLIENT_HOST_LEN );
h = gethostbyname( This_Server->host );
- if( h ) strcpy( This_Server->host, h->h_name );
+ if( h ) strlcpy( This_Server->host, h->h_name, sizeof( This_Server->host ));
Client_SetID( This_Server, Conf_ServerName );
Client_SetInfo( This_Server, Conf_ServerInfo );
CLIENT *c, *next;
INT cnt;
- if( NGIRCd_Restart ) Client_Destroy( This_Server, "Server going down (restarting).", NULL, FALSE );
+ if( NGIRCd_SignalRestart ) Client_Destroy( This_Server, "Server going down (restarting).", NULL, FALSE );
else Client_Destroy( This_Server, "Server going down.", NULL, FALSE );
cnt = 0;
if( Type == CLIENT_SERVER ) Generate_MyToken( client );
/* ist der User away? */
- if( strchr( client->modes, 'a' )) strcpy( client->away, DEFAULT_AWAY_MSG );
+ if( strchr( client->modes, 'a' )) strlcpy( client->away, DEFAULT_AWAY_MSG, sizeof( client->away ));
/* Verketten */
client->next = (POINTER *)My_Clients;
My_Clients = client;
+ /* Adjust counters */
+ Adjust_Counters( client );
+
return client;
} /* Client_New */
if( ! txt ) txt = "Reason unknown.";
/* Netz-Split-Nachricht vorbereiten (noch nicht optimal) */
- if( Client->type == CLIENT_SERVER ) sprintf( msg, "%s: lost server %s", This_Server->id, Client->id );
+ if( Client->type == CLIENT_SERVER ) snprintf( msg, sizeof( msg ), "%s: lost server %s", This_Server->id, Client->id );
last = NULL;
c = My_Clients;
}
/* andere Server informieren */
- if( ! NGIRCd_Quit )
+ if( ! NGIRCd_SignalQuit )
{
if( FwdMsg ) IRC_WriteStrServersPrefix( Client_NextHop( c ), c, "SQUIT %s :%s", c->id, FwdMsg );
else IRC_WriteStrServersPrefix( Client_NextHop( c ), c, "SQUIT %s :", c->id );
} /* Client_Destroy */
+GLOBAL VOID
+Client_DestroyNow( CLIENT *Client )
+{
+ /* Destroy client structure immediately. This function is only
+ * intended for the connection layer to remove client structures
+ * of connections that can't be established! */
+
+ CLIENT *last, *c;
+
+ assert( Client != NULL );
+
+ last = NULL;
+ c = My_Clients;
+ while( c )
+ {
+ if( c == Client )
+ {
+ /* Wir haben den Client gefunden: entfernen */
+ if( last ) last->next = c->next;
+ else My_Clients = (CLIENT *)c->next;
+ free( c );
+ break;
+ }
+ last = c;
+ c = (CLIENT *)c->next;
+ }
+} /* Client_DestroyNow */
+
+
GLOBAL VOID
Client_SetHostname( CLIENT *Client, CHAR *Hostname )
{
assert( Client != NULL );
assert( Hostname != NULL );
- strncpy( Client->host, Hostname, CLIENT_HOST_LEN - 1 );
- Client->host[CLIENT_HOST_LEN - 1] = '\0';
+ strlcpy( Client->host, Hostname, sizeof( Client->host ));
} /* Client_SetHostname */
assert( Client != NULL );
assert( ID != NULL );
- strncpy( Client->id, ID, CLIENT_ID_LEN - 1 );
- Client->id[CLIENT_ID_LEN - 1] = '\0';
+ strlcpy( Client->id, ID, sizeof( Client->id ));
/* Hash */
Client->hash = Hash( Client->id );
assert( Client != NULL );
assert( User != NULL );
- if( Idented ) strncpy( Client->user, User, CLIENT_USER_LEN - 1 );
+ if( Idented ) strlcpy( Client->user, User, sizeof( Client->user ));
else
{
Client->user[0] = '~';
- strncpy( Client->user + 1, User, CLIENT_USER_LEN - 2 );
+ strlcpy( Client->user + 1, User, sizeof( Client->user ) - 1 );
}
- Client->user[CLIENT_USER_LEN - 1] = '\0';
} /* Client_SetUser */
assert( Client != NULL );
assert( Info != NULL );
- strncpy( Client->info, Info, CLIENT_INFO_LEN - 1 );
- Client->info[CLIENT_INFO_LEN - 1] = '\0';
+ strlcpy( Client->info, Info, sizeof( Client->info ));
} /* Client_SetInfo */
assert( Client != NULL );
assert( Modes != NULL );
- strncpy( Client->modes, Modes, CLIENT_MODE_LEN - 1 );
- Client->modes[CLIENT_MODE_LEN - 1] = '\0';
+ strlcpy( Client->modes, Modes, sizeof( Client->modes ));
} /* Client_SetModes */
assert( Client != NULL );
assert( Flags != NULL );
- strncpy( Client->flags, Flags, CLIENT_FLAGS_LEN - 1 );
- Client->modes[CLIENT_FLAGS_LEN - 1] = '\0';
+ strlcpy( Client->flags, Flags, sizeof( Client->flags ));
} /* Client_SetFlags */
assert( Client != NULL );
assert( Pwd != NULL );
- strncpy( Client->pwd, Pwd, CLIENT_PASS_LEN - 1 );
- Client->pwd[CLIENT_PASS_LEN - 1] = '\0';
+ strlcpy( Client->pwd, Pwd, sizeof( Client->pwd ));
} /* Client_SetPassword */
GLOBAL VOID
Client_SetAway( CLIENT *Client, CHAR *Txt )
{
- /* Von einem Client gelieferte AWAY-Nachricht */
+ /* Set AWAY reason of client */
assert( Client != NULL );
+ assert( Txt != NULL );
- if( Txt )
- {
- /* Client AWAY setzen */
- strncpy( Client->away, Txt, CLIENT_AWAY_LEN - 1 );
- Client->away[CLIENT_AWAY_LEN - 1] = '\0';
- Client_ModeAdd( Client, 'a' );
- Log( LOG_DEBUG, "User \"%s\" is away: %s", Client_Mask( Client ), Txt );
- }
- else
- {
- /* AWAY loeschen */
- Client_ModeDel( Client, 'a' );
- Log( LOG_DEBUG, "User \"%s\" is no longer away.", Client_Mask( Client ));
- }
+ strlcpy( Client->away, Txt, sizeof( Client->away ));
+ Log( LOG_DEBUG, "User \"%s\" is away: %s", Client_Mask( Client ), Txt );
} /* Client_SetAway */
assert( Client != NULL );
Client->type = Type;
if( Type == CLIENT_SERVER ) Generate_MyToken( Client );
+ Adjust_Counters( Client );
} /* Client_SetType */
if( ! strchr( Client->modes, x[0] ))
{
/* Client hat den Mode noch nicht -> setzen */
- strcat( Client->modes, x );
+ strlcat( Client->modes, x, sizeof( Client->modes ));
return TRUE;
}
else return FALSE;
assert( Nick != NULL );
/* Nick kopieren und ggf. Host-Mask abschneiden */
- strncpy( search_id, Nick, CLIENT_ID_LEN - 1 );
- search_id[CLIENT_ID_LEN - 1] = '\0';
+ strlcpy( search_id, Nick, sizeof( search_id ));
ptr = strchr( search_id, '!' );
if( ptr ) *ptr = '\0';
if( Client->type == CLIENT_SERVER ) return Client->id;
- sprintf( GetID_Buffer, "%s!%s@%s", Client->id, Client->user, Client->host );
+ snprintf( GetID_Buffer, GETID_LEN, "%s!%s@%s", Client->id, Client->user, Client->host );
return GetID_Buffer;
} /* Client_Mask */
if( strcasecmp( c->id, ID ) == 0 )
{
/* die Server-ID gibt es bereits */
- sprintf( str, "ID \"%s\" already registered!", ID );
- Log( LOG_ERR, "%s (on connection %d)", str, Client->conn_id );
+ snprintf( str, sizeof( str ), "ID \"%s\" already registered", ID );
+ if( Client->conn_id != c->conn_id ) Log( LOG_ERR, "%s (on connection %d)!", str, c->conn_id );
+ else Log( LOG_ERR, "%s (via network)!", str );
Conn_Close( Client->conn_id, str, str, TRUE );
return FALSE;
}
} /* Client_Next */
-GLOBAL INT
+GLOBAL LONG
Client_UserCount( VOID )
{
return Count( CLIENT_USER );
} /* Client_UserCount */
-GLOBAL INT
+GLOBAL LONG
Client_ServiceCount( VOID )
{
return Count( CLIENT_SERVICE );;
} /* Client_ServiceCount */
-GLOBAL INT
+GLOBAL LONG
Client_ServerCount( VOID )
{
return Count( CLIENT_SERVER );
} /* Client_ServerCount */
-GLOBAL INT
+GLOBAL LONG
Client_MyUserCount( VOID )
{
return MyCount( CLIENT_USER );
} /* Client_MyUserCount */
-GLOBAL INT
+GLOBAL LONG
Client_MyServiceCount( VOID )
{
return MyCount( CLIENT_SERVICE );
} /* Client_MyServiceCount */
-GLOBAL INT
+GLOBAL LONG
Client_MyServerCount( VOID )
{
CLIENT *c;
- INT cnt;
+ LONG cnt;
cnt = 0;
c = My_Clients;
} /* Client_MyServerCount */
-GLOBAL INT
+GLOBAL LONG
Client_OperCount( VOID )
{
CLIENT *c;
- INT cnt;
+ LONG cnt;
cnt = 0;
c = My_Clients;
} /* Client_OperCount */
-GLOBAL INT
+GLOBAL LONG
Client_UnknownCount( VOID )
{
CLIENT *c;
- INT cnt;
+ LONG cnt;
cnt = 0;
c = My_Clients;
} /* Client_UnknownCount */
+GLOBAL LONG
+Client_MaxUserCount( VOID )
+{
+ return Max_Users;
+} /* Client_MaxUserCount */
+
+
+GLOBAL LONG
+Client_MyMaxUserCount( VOID )
+{
+ return My_Max_Users;
+} /* Client_MyMaxUserCount */
+
+
GLOBAL BOOLEAN
Client_IsValidNick( CHAR *Nick )
{
} /* Client_IsValidNick */
-LOCAL INT
+LOCAL LONG
Count( CLIENT_TYPE Type )
{
CLIENT *c;
- INT cnt;
+ LONG cnt;
cnt = 0;
c = My_Clients;
} /* Count */
-LOCAL INT
+LOCAL LONG
MyCount( CLIENT_TYPE Type )
{
CLIENT *c;
- INT cnt;
+ LONG cnt;
cnt = 0;
c = My_Clients;
} /* Generate_MyToken */
+LOCAL VOID
+Adjust_Counters( CLIENT *Client )
+{
+ LONG count;
+
+ assert( Client != NULL );
+
+ if( Client->type != CLIENT_USER ) return;
+
+ if( Client->conn_id != NONE )
+ {
+ /* Local connection */
+ count = Client_MyUserCount( );
+ if( count > My_Max_Users ) My_Max_Users = count;
+ }
+ count = Client_UserCount( );
+ if( count > Max_Users ) Max_Users = count;
+} /* Adjust_Counters */
+
+
/* -eof- */