2002-03-03 18:15:11 +01:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*
|
2002-08-27 01:47:58 +02:00
|
|
|
* $Id: irc-server.c,v 1.14 2002/08/26 23:47:58 alex Exp $
|
2002-03-03 18:15:11 +01:00
|
|
|
*
|
|
|
|
* irc-server.c: IRC-Befehle fuer Server-Links
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2002-03-12 15:37:51 +01:00
|
|
|
#include "portab.h"
|
2002-03-03 18:15:11 +01:00
|
|
|
|
2002-03-12 15:37:51 +01:00
|
|
|
#include "imp.h"
|
2002-03-03 18:15:11 +01:00
|
|
|
#include <assert.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2002-05-27 15:09:26 +02:00
|
|
|
#include "resolve.h"
|
2002-03-03 18:15:11 +01:00
|
|
|
#include "conf.h"
|
2002-05-27 15:09:26 +02:00
|
|
|
#include "conn.h"
|
|
|
|
#include "client.h"
|
|
|
|
#include "channel.h"
|
2002-03-03 18:15:11 +01:00
|
|
|
#include "irc-write.h"
|
|
|
|
#include "log.h"
|
|
|
|
#include "messages.h"
|
2002-05-27 15:09:26 +02:00
|
|
|
#include "parse.h"
|
2002-03-03 18:15:11 +01:00
|
|
|
|
2002-03-12 15:37:51 +01:00
|
|
|
#include "exp.h"
|
2002-03-03 18:15:11 +01:00
|
|
|
#include "irc-server.h"
|
|
|
|
|
|
|
|
|
2002-05-27 15:09:26 +02:00
|
|
|
GLOBAL BOOLEAN
|
|
|
|
IRC_SERVER( CLIENT *Client, REQUEST *Req )
|
2002-03-03 18:15:11 +01:00
|
|
|
{
|
|
|
|
CHAR str[LINE_LEN], *ptr;
|
|
|
|
CLIENT *from, *c, *cl;
|
|
|
|
CL2CHAN *cl2chan;
|
|
|
|
INT max_hops, i;
|
|
|
|
CHANNEL *chan;
|
|
|
|
BOOLEAN ok;
|
|
|
|
|
|
|
|
assert( Client != NULL );
|
|
|
|
assert( Req != NULL );
|
|
|
|
|
|
|
|
/* Fehler liefern, wenn kein lokaler Client */
|
|
|
|
if( Client_Conn( Client ) <= NONE ) return IRC_WriteStrClient( Client, ERR_UNKNOWNCOMMAND_MSG, Client_ID( Client ), Req->command );
|
|
|
|
|
|
|
|
if( Client_Type( Client ) == CLIENT_GOTPASSSERVER )
|
|
|
|
{
|
|
|
|
/* Verbindung soll als Server-Server-Verbindung registriert werden */
|
|
|
|
Log( LOG_DEBUG, "Connection %d: got SERVER command (new server link) ...", Client_Conn( Client ));
|
|
|
|
|
|
|
|
/* Falsche Anzahl Parameter? */
|
|
|
|
if(( Req->argc != 2 ) && ( Req->argc != 3 )) return IRC_WriteStrClient( Client, ERR_NEEDMOREPARAMS_MSG, Client_ID( Client ), Req->command );
|
|
|
|
|
|
|
|
/* Ist dieser Server bei uns konfiguriert? */
|
|
|
|
for( i = 0; i < Conf_Server_Count; i++ ) if( strcasecmp( Req->argv[0], Conf_Server[i].name ) == 0 ) break;
|
|
|
|
if( i >= Conf_Server_Count )
|
|
|
|
{
|
|
|
|
/* Server ist nicht konfiguriert! */
|
|
|
|
Log( LOG_ERR, "Connection %d: Server \"%s\" not configured here!", Client_Conn( Client ), Req->argv[0] );
|
|
|
|
Conn_Close( Client_Conn( Client ), NULL, "Server not configured here", TRUE );
|
|
|
|
return DISCONNECTED;
|
|
|
|
}
|
|
|
|
if( strcmp( Client_Password( Client ), Conf_Server[i].pwd ) != 0 )
|
|
|
|
{
|
|
|
|
/* Falsches Passwort */
|
|
|
|
Log( LOG_ERR, "Connection %d: Bad password for server \"%s\"!", Client_Conn( Client ), Req->argv[0] );
|
|
|
|
Conn_Close( Client_Conn( Client ), NULL, "Bad password", TRUE );
|
|
|
|
return DISCONNECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ist ein Server mit dieser ID bereits registriert? */
|
|
|
|
if( ! Client_CheckID( Client, Req->argv[0] )) return DISCONNECTED;
|
|
|
|
|
|
|
|
/* Server-Strukturen fuellen ;-) */
|
|
|
|
Client_SetID( Client, Req->argv[0] );
|
|
|
|
Client_SetHops( Client, 1 );
|
|
|
|
Client_SetInfo( Client, Req->argv[Req->argc - 1] );
|
|
|
|
|
2002-04-08 03:20:14 +02:00
|
|
|
/* Meldet sich der Server bei uns an (d.h., bauen nicht wir
|
|
|
|
* selber die Verbindung zu einem anderen Server auf)? */
|
|
|
|
if( Client_Token( Client ) != TOKEN_OUTBOUND )
|
2002-03-03 18:15:11 +01:00
|
|
|
{
|
2002-04-08 03:20:14 +02:00
|
|
|
/* Eingehende Verbindung: Unseren SERVER- und PASS-Befehl senden */
|
2002-03-03 18:15:11 +01:00
|
|
|
ok = TRUE;
|
2002-05-30 18:52:20 +02:00
|
|
|
if( ! IRC_WriteStrClient( Client, "PASS %s %s", Conf_Server[i].pwd, PASSSERVERADD )) ok = FALSE;
|
2002-03-03 18:15:11 +01:00
|
|
|
else ok = IRC_WriteStrClient( Client, "SERVER %s 1 :%s", Conf_ServerName, Conf_ServerInfo );
|
|
|
|
if( ! ok )
|
|
|
|
{
|
|
|
|
Conn_Close( Client_Conn( Client ), "Unexpected server behavior!", NULL, FALSE );
|
|
|
|
return DISCONNECTED;
|
|
|
|
}
|
|
|
|
Client_SetIntroducer( Client, Client );
|
|
|
|
Client_SetToken( Client, 1 );
|
|
|
|
}
|
2002-04-08 03:20:14 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Ausgehende verbindung, SERVER und PASS wurden von uns bereits
|
|
|
|
* an die Gegenseite uerbermittelt */
|
|
|
|
Client_SetToken( Client, atoi( Req->argv[1] ));
|
|
|
|
}
|
2002-03-03 18:15:11 +01:00
|
|
|
|
2002-03-27 21:52:58 +01:00
|
|
|
Log( LOG_NOTICE|LOG_snotice, "Server \"%s\" registered (connection %d, 1 hop - direct link).", Client_ID( Client ), Client_Conn( Client ));
|
2002-03-03 18:15:11 +01:00
|
|
|
|
|
|
|
Client_SetType( Client, CLIENT_SERVER );
|
|
|
|
|
|
|
|
/* maximalen Hop Count ermitteln */
|
|
|
|
max_hops = 0;
|
|
|
|
c = Client_First( );
|
|
|
|
while( c )
|
|
|
|
{
|
|
|
|
if( Client_Hops( c ) > max_hops ) max_hops = Client_Hops( c );
|
|
|
|
c = Client_Next( c );
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Alle bisherigen Server dem neuen Server bekannt machen,
|
|
|
|
* die bisherigen Server ueber den neuen informierenn */
|
|
|
|
for( i = 0; i < ( max_hops + 1 ); i++ )
|
|
|
|
{
|
|
|
|
c = Client_First( );
|
|
|
|
while( c )
|
|
|
|
{
|
|
|
|
if(( Client_Type( c ) == CLIENT_SERVER ) && ( c != Client ) && ( c != Client_ThisServer( )) && ( Client_Hops( c ) == i ))
|
|
|
|
{
|
|
|
|
if( Client_Conn( c ) > NONE )
|
|
|
|
{
|
|
|
|
/* Dem gefundenen Server gleich den neuen
|
|
|
|
* Server bekannt machen */
|
|
|
|
if( ! IRC_WriteStrClient( c, "SERVER %s %d %d :%s", Client_ID( Client ), Client_Hops( Client ) + 1, Client_MyToken( Client ), Client_Info( Client ))) return DISCONNECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Den neuen Server ueber den alten informieren */
|
|
|
|
if( ! IRC_WriteStrClientPrefix( Client, Client_Hops( c ) == 1 ? Client_ThisServer( ) : Client_Introducer( c ), "SERVER %s %d %d :%s", Client_ID( c ), Client_Hops( c ) + 1, Client_MyToken( c ), Client_Info( c ))) return DISCONNECTED;
|
|
|
|
}
|
|
|
|
c = Client_Next( c );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* alle User dem neuen Server bekannt machen */
|
|
|
|
c = Client_First( );
|
|
|
|
while( c )
|
|
|
|
{
|
|
|
|
if( Client_Type( c ) == CLIENT_USER )
|
|
|
|
{
|
|
|
|
/* User an neuen Server melden */
|
|
|
|
if( ! IRC_WriteStrClient( Client, "NICK %s %d %s %s %d +%s :%s", Client_ID( c ), Client_Hops( c ) + 1, Client_User( c ), Client_Hostname( c ), Client_MyToken( Client_Introducer( c )), Client_Modes( c ), Client_Info( c ))) return DISCONNECTED;
|
|
|
|
}
|
|
|
|
c = Client_Next( c );
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Channels dem neuen Server bekannt machen */
|
|
|
|
chan = Channel_First( );
|
|
|
|
while( chan )
|
|
|
|
{
|
|
|
|
/* alle Member suchen */
|
|
|
|
cl2chan = Channel_FirstMember( chan );
|
2002-03-06 16:50:14 +01:00
|
|
|
sprintf( str, "NJOIN %s :", Channel_Name( chan ));
|
2002-03-03 18:15:11 +01:00
|
|
|
while( cl2chan )
|
|
|
|
{
|
|
|
|
cl = Channel_GetClient( cl2chan );
|
|
|
|
assert( cl != NULL );
|
|
|
|
|
|
|
|
/* Nick, ggf. mit Modes, anhaengen */
|
|
|
|
if( str[strlen( str ) - 1] != ':' ) strcat( str, "," );
|
|
|
|
if( strchr( Channel_UserModes( chan, cl ), 'v' )) strcat( str, "+" );
|
|
|
|
if( strchr( Channel_UserModes( chan, cl ), 'o' )) strcat( str, "@" );
|
|
|
|
strcat( str, Client_ID( cl ));
|
|
|
|
|
2002-03-06 16:50:14 +01:00
|
|
|
if( strlen( str ) > ( LINE_LEN - CLIENT_NICK_LEN - 8 ))
|
2002-03-03 18:15:11 +01:00
|
|
|
{
|
|
|
|
/* Zeile senden */
|
|
|
|
if( ! IRC_WriteStrClient( Client, str )) return DISCONNECTED;
|
|
|
|
sprintf( str, "NJOIN %s :", Channel_Name( chan ));
|
|
|
|
}
|
|
|
|
|
|
|
|
cl2chan = Channel_NextMember( chan, cl2chan );
|
|
|
|
}
|
|
|
|
|
|
|
|
/* noch Daten da? */
|
|
|
|
if( str[strlen( str ) - 1] != ':')
|
|
|
|
{
|
|
|
|
/* Ja; Also senden ... */
|
|
|
|
if( ! IRC_WriteStrClient( Client, str )) return DISCONNECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* naechsten Channel suchen */
|
|
|
|
chan = Channel_Next( chan );
|
|
|
|
}
|
|
|
|
|
|
|
|
return CONNECTED;
|
|
|
|
}
|
|
|
|
else if( Client_Type( Client ) == CLIENT_SERVER )
|
|
|
|
{
|
|
|
|
/* Neuer Server wird im Netz angekuendigt */
|
|
|
|
|
|
|
|
/* Falsche Anzahl Parameter? */
|
|
|
|
if( Req->argc != 4 ) return IRC_WriteStrClient( Client, ERR_NEEDMOREPARAMS_MSG, Client_ID( Client ), Req->command );
|
|
|
|
|
|
|
|
/* Ist ein Server mit dieser ID bereits registriert? */
|
|
|
|
if( ! Client_CheckID( Client, Req->argv[0] )) return DISCONNECTED;
|
|
|
|
|
|
|
|
/* Ueberfluessige Hostnamen aus Info-Text entfernen */
|
|
|
|
ptr = strchr( Req->argv[3] + 2, '[' );
|
|
|
|
if( ! ptr ) ptr = Req->argv[3];
|
|
|
|
|
2002-03-25 18:08:54 +01:00
|
|
|
from = Client_Search( Req->prefix );
|
2002-03-03 18:15:11 +01:00
|
|
|
if( ! from )
|
|
|
|
{
|
|
|
|
/* Hm, Server, der diesen einfuehrt, ist nicht bekannt!? */
|
|
|
|
Log( LOG_ALERT, "Unknown ID in prefix of SERVER: \"%s\"! (on connection %d)", Req->prefix, Client_Conn( Client ));
|
|
|
|
Conn_Close( Client_Conn( Client ), NULL, "Unknown ID in prefix of SERVER", TRUE );
|
|
|
|
return DISCONNECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Neue Client-Struktur anlegen */
|
|
|
|
c = Client_NewRemoteServer( Client, Req->argv[0], from, atoi( Req->argv[1] ), atoi( Req->argv[2] ), ptr, TRUE );
|
|
|
|
if( ! c )
|
|
|
|
{
|
|
|
|
/* Neue Client-Struktur konnte nicht angelegt werden */
|
|
|
|
Log( LOG_ALERT, "Can't create client structure for server! (on connection %d)", Client_Conn( Client ));
|
|
|
|
Conn_Close( Client_Conn( Client ), NULL, "Can't allocate client structure for remote server", TRUE );
|
|
|
|
return DISCONNECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Log-Meldung zusammenbauen und ausgeben */
|
|
|
|
if(( Client_Hops( c ) > 1 ) && ( Req->prefix[0] )) sprintf( str, "connected to %s, ", Client_ID( from ));
|
|
|
|
else strcpy( str, "" );
|
2002-03-27 21:52:58 +01:00
|
|
|
Log( LOG_NOTICE|LOG_snotice, "Server \"%s\" registered (via %s, %s%d hop%s).", Client_ID( c ), Client_ID( Client ), str, Client_Hops( c ), Client_Hops( c ) > 1 ? "s": "" );
|
2002-03-03 18:15:11 +01:00
|
|
|
|
|
|
|
/* Andere Server informieren */
|
|
|
|
IRC_WriteStrServersPrefix( Client, from, "SERVER %s %d %d :%s", Client_ID( c ), Client_Hops( c ) + 1, Client_MyToken( c ), Client_Info( c ));
|
|
|
|
|
|
|
|
return CONNECTED;
|
|
|
|
}
|
|
|
|
else return IRC_WriteStrClient( Client, ERR_NEEDMOREPARAMS_MSG, Client_ID( Client ), Req->command );
|
|
|
|
} /* IRC_SERVER */
|
|
|
|
|
|
|
|
|
2002-05-27 15:09:26 +02:00
|
|
|
GLOBAL BOOLEAN
|
|
|
|
IRC_NJOIN( CLIENT *Client, REQUEST *Req )
|
2002-03-03 18:15:11 +01:00
|
|
|
{
|
2002-07-25 14:33:19 +02:00
|
|
|
CHAR str[COMMAND_LEN], *channame, *ptr, modes[8];
|
2002-03-03 18:15:11 +01:00
|
|
|
BOOLEAN is_op, is_voiced;
|
|
|
|
CHANNEL *chan;
|
|
|
|
CLIENT *c;
|
|
|
|
|
|
|
|
assert( Client != NULL );
|
|
|
|
assert( Req != NULL );
|
|
|
|
|
|
|
|
if( Client_Type( Client ) != CLIENT_SERVER ) return IRC_WriteStrClient( Client, ERR_NOTREGISTEREDSERVER_MSG, Client_ID( Client ));
|
|
|
|
|
|
|
|
/* Falsche Anzahl Parameter? */
|
|
|
|
if( Req->argc != 2 ) return IRC_WriteStrClient( Client, ERR_NEEDMOREPARAMS_MSG, Client_ID( Client ), Req->command );
|
|
|
|
|
2002-07-25 14:33:19 +02:00
|
|
|
strncpy( str, Req->argv[1], COMMAND_LEN - 1 );
|
|
|
|
str[COMMAND_LEN - 1] = '\0';
|
|
|
|
|
2002-03-03 18:15:11 +01:00
|
|
|
channame = Req->argv[0];
|
2002-07-25 14:33:19 +02:00
|
|
|
ptr = strtok( str, "," );
|
2002-03-03 18:15:11 +01:00
|
|
|
while( ptr )
|
|
|
|
{
|
|
|
|
is_op = is_voiced = FALSE;
|
|
|
|
|
|
|
|
/* Prefixe abschneiden */
|
|
|
|
while(( *ptr == '@' ) || ( *ptr == '+' ))
|
|
|
|
{
|
|
|
|
if( *ptr == '@' ) is_op = TRUE;
|
|
|
|
if( *ptr == '+' ) is_voiced = TRUE;
|
|
|
|
ptr++;
|
|
|
|
}
|
|
|
|
|
2002-03-25 18:08:54 +01:00
|
|
|
c = Client_Search( ptr );
|
2002-03-03 18:15:11 +01:00
|
|
|
if( c )
|
|
|
|
{
|
|
|
|
Channel_Join( c, channame );
|
|
|
|
chan = Channel_Search( channame );
|
|
|
|
assert( chan != NULL );
|
|
|
|
|
|
|
|
if( is_op ) Channel_UserModeAdd( chan, c, 'o' );
|
|
|
|
if( is_voiced ) Channel_UserModeAdd( chan, c, 'v' );
|
|
|
|
|
|
|
|
/* im Channel bekannt machen */
|
|
|
|
IRC_WriteStrChannelPrefix( Client, chan, c, FALSE, "JOIN :%s", channame );
|
|
|
|
|
|
|
|
/* Channel-User-Modes setzen */
|
|
|
|
strcpy( modes, Channel_UserModes( chan, c ));
|
|
|
|
if( modes[0] )
|
|
|
|
{
|
|
|
|
/* Modes im Channel bekannt machen */
|
|
|
|
IRC_WriteStrChannelPrefix( Client, chan, Client, FALSE, "MODE %s +%s %s", channame, modes, Client_ID( c ));
|
|
|
|
}
|
|
|
|
}
|
2002-08-27 01:47:58 +02:00
|
|
|
else Log( LOG_ERR, "Got NJOIN for unknown nick \"%s\" for channel \"%s\"!", ptr, channame );
|
2002-03-03 18:15:11 +01:00
|
|
|
|
|
|
|
/* naechsten Nick suchen */
|
|
|
|
ptr = strtok( NULL, "," );
|
|
|
|
}
|
|
|
|
|
|
|
|
/* an andere Server weiterleiten */
|
|
|
|
IRC_WriteStrServersPrefix( Client, Client_ThisServer( ), "NJOIN %s :%s", Req->argv[0], Req->argv[1] );
|
|
|
|
|
|
|
|
return CONNECTED;
|
|
|
|
} /* IRC_NJOIN */
|
|
|
|
|
|
|
|
|
2002-05-27 15:09:26 +02:00
|
|
|
GLOBAL BOOLEAN
|
|
|
|
IRC_SQUIT( CLIENT *Client, REQUEST *Req )
|
2002-03-03 18:15:11 +01:00
|
|
|
{
|
|
|
|
CLIENT *target;
|
|
|
|
CHAR msg[LINE_LEN + 64];
|
|
|
|
|
|
|
|
assert( Client != NULL );
|
|
|
|
assert( Req != NULL );
|
|
|
|
|
|
|
|
/* SQUIT ist nur fuer Server erlaubt */
|
|
|
|
if( Client_Type( Client ) != CLIENT_SERVER ) return IRC_WriteStrClient( Client, ERR_NOTREGISTERED_MSG, Client_ID( Client ));
|
|
|
|
|
|
|
|
/* Falsche Anzahl Parameter? */
|
|
|
|
if( Req->argc != 2 ) return IRC_WriteStrClient( Client, ERR_NEEDMOREPARAMS_MSG, Client_ID( Client ), Req->command );
|
|
|
|
|
|
|
|
Log( LOG_DEBUG, "Got SQUIT from %s for \"%s\": \"%s\" ...", Client_ID( Client ), Req->argv[0], Req->argv[1] );
|
|
|
|
|
|
|
|
/* SQUIT an alle Server weiterleiten */
|
|
|
|
IRC_WriteStrServers( Client, "SQUIT %s :%s", Req->argv[0], Req->argv[1] );
|
|
|
|
|
2002-03-25 18:08:54 +01:00
|
|
|
target = Client_Search( Req->argv[0] );
|
2002-03-03 18:15:11 +01:00
|
|
|
if( ! target )
|
|
|
|
{
|
2002-03-11 18:33:40 +01:00
|
|
|
/* Den Server kennen wir nicht (mehr), also nichts zu tun. */
|
2002-03-11 23:06:32 +01:00
|
|
|
Log( LOG_WARNING, "Got SQUIT from %s for unknown server \"%s\"!?", Client_ID( Client ), Req->argv[0] );
|
2002-03-03 18:15:11 +01:00
|
|
|
return CONNECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( Req->argv[1][0] )
|
|
|
|
{
|
|
|
|
if( strlen( Req->argv[1] ) > LINE_LEN ) Req->argv[1][LINE_LEN] = '\0';
|
|
|
|
sprintf( msg, "%s (SQUIT from %s).", Req->argv[1], Client_ID( Client ));
|
|
|
|
}
|
|
|
|
else sprintf( msg, "Got SQUIT from %s.", Client_ID( Client ));
|
|
|
|
|
|
|
|
if( Client_Conn( target ) > NONE )
|
|
|
|
{
|
|
|
|
/* dieser Server hat die Connection */
|
|
|
|
if( Req->argv[1][0] ) Conn_Close( Client_Conn( target ), msg, Req->argv[1], TRUE );
|
|
|
|
else Conn_Close( Client_Conn( target ), msg, NULL, TRUE );
|
|
|
|
return DISCONNECTED;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Verbindung hielt anderer Server */
|
2002-03-11 23:04:10 +01:00
|
|
|
Client_Destroy( target, msg, Req->argv[1], FALSE );
|
2002-03-03 18:15:11 +01:00
|
|
|
return CONNECTED;
|
|
|
|
}
|
|
|
|
} /* IRC_SQUIT */
|
|
|
|
|
|
|
|
|
|
|
|
/* -eof- */
|