+LOCAL VOID
+Handle_GLOBAL( INT Line, CHAR *Var, CHAR *Arg )
+{
+ struct passwd *pwd;
+ struct group *grp;
+ CHAR *ptr;
+ LONG port;
+
+ assert( Line > 0 );
+ assert( Var != NULL );
+ assert( Arg != NULL );
+
+ if( strcasecmp( Var, "Name" ) == 0 )
+ {
+ /* Der Server-Name */
+ strncpy( Conf_ServerName, Arg, CLIENT_ID_LEN - 1 );
+ Conf_ServerName[CLIENT_ID_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "Info" ) == 0 )
+ {
+ /* Server-Info-Text */
+ strncpy( Conf_ServerInfo, Arg, CLIENT_INFO_LEN - 1 );
+ Conf_ServerInfo[CLIENT_INFO_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "Password" ) == 0 )
+ {
+ /* Server-Passwort */
+ strncpy( Conf_ServerPwd, Arg, CLIENT_PASS_LEN - 1 );
+ Conf_ServerPwd[CLIENT_PASS_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "AdminInfo1" ) == 0 )
+ {
+ /* Server-Info-Text */
+ strncpy( Conf_ServerAdmin1, Arg, CLIENT_INFO_LEN - 1 );
+ Conf_ServerAdmin1[CLIENT_INFO_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "AdminInfo2" ) == 0 )
+ {
+ /* Server-Info-Text */
+ strncpy( Conf_ServerAdmin2, Arg, CLIENT_INFO_LEN - 1 );
+ Conf_ServerAdmin2[CLIENT_INFO_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "AdminEMail" ) == 0 )
+ {
+ /* Server-Info-Text */
+ strncpy( Conf_ServerAdminMail, Arg, CLIENT_INFO_LEN - 1 );
+ Conf_ServerAdminMail[CLIENT_INFO_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "Ports" ) == 0 )
+ {
+ /* Ports, durch "," getrennt, auf denen der Server
+ * Verbindungen annehmen soll */
+ ptr = strtok( Arg, "," );
+ while( ptr )
+ {
+ ngt_TrimStr( ptr );
+ port = atol( ptr );
+ if( Conf_ListenPorts_Count + 1 > MAX_LISTEN_PORTS ) Config_Error( LOG_ERR, "Too many listen ports configured. Port %ld ignored.", port );
+ else
+ {
+ if( port > 0 && port < 0xFFFF ) Conf_ListenPorts[Conf_ListenPorts_Count++] = (UINT)port;
+ else Config_Error( LOG_ERR, "%s, line %d (section \"Global\"): Illegal port number %ld!", NGIRCd_ConfFile, Line, port );
+ }
+ ptr = strtok( NULL, "," );
+ }
+ return;
+ }
+ if( strcasecmp( Var, "MotdFile" ) == 0 )
+ {
+ /* Datei mit der "message of the day" (MOTD) */
+ strncpy( Conf_MotdFile, Arg, FNAME_LEN - 1 );
+ Conf_MotdFile[FNAME_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "ServerUID" ) == 0 )
+ {
+ /* UID, mit der der Daemon laufen soll */
+ pwd = getpwnam( Arg );
+ if( pwd ) Conf_UID = pwd->pw_uid;
+ else Conf_UID = (UINT)atoi( Arg );
+ return;
+ }
+ if( strcasecmp( Var, "ServerGID" ) == 0 )
+ {
+ /* GID, mit der der Daemon laufen soll */
+ grp = getgrnam( Arg );
+ if( grp ) Conf_GID = grp->gr_gid;
+ else Conf_GID = (UINT)atoi( Arg );
+ return;
+ }
+ if( strcasecmp( Var, "PingTimeout" ) == 0 )
+ {
+ /* PING-Timeout */
+ Conf_PingTimeout = atoi( Arg );
+ if(( Conf_PingTimeout ) < 5 ) Conf_PingTimeout = 5;
+ return;
+ }
+ if( strcasecmp( Var, "PongTimeout" ) == 0 )
+ {
+ /* PONG-Timeout */
+ Conf_PongTimeout = atoi( Arg );
+ if(( Conf_PongTimeout ) < 5 ) Conf_PongTimeout = 5;
+ return;
+ }
+ if( strcasecmp( Var, "ConnectRetry" ) == 0 )
+ {
+ /* Sekunden zwischen Verbindungsversuchen zu anderen Servern */
+ Conf_ConnectRetry = atoi( Arg );
+ if(( Conf_ConnectRetry ) < 5 ) Conf_ConnectRetry = 5;
+ return;
+ }
+ if( strcasecmp( Var, "OperCanUseMode" ) == 0 )
+ {
+ /* Koennen IRC-Operatoren immer MODE benutzen? */
+ if( strcasecmp( Arg, "yes" ) == 0 ) Conf_OperCanMode = TRUE;
+ else if( strcasecmp( Arg, "true" ) == 0 ) Conf_OperCanMode = TRUE;
+ else if( atoi( Arg ) != 0 ) Conf_OperCanMode = TRUE;
+ else Conf_OperCanMode = FALSE;
+ return;
+ }
+ if( strcasecmp( Var, "MaxConnections" ) == 0 )
+ {
+ /* Maximale Anzahl von Verbindungen. Werte <= 0 stehen
+ * fuer "kein Limit". */
+ Conf_MaxConnections = atol( Arg );
+ return;
+ }
+
+ Config_Error( LOG_ERR, "%s, line %d (section \"Global\"): Unknown variable \"%s\"!", NGIRCd_ConfFile, Line, Var );
+} /* Handle_GLOBAL */
+
+
+LOCAL VOID
+Handle_OPERATOR( INT Line, CHAR *Var, CHAR *Arg )
+{
+ assert( Line > 0 );
+ assert( Var != NULL );
+ assert( Arg != NULL );
+ assert( Conf_Oper_Count > 0 );
+
+ if( strcasecmp( Var, "Name" ) == 0 )
+ {
+ /* Name des IRC Operator */
+ strncpy( Conf_Oper[Conf_Oper_Count - 1].name, Arg, CLIENT_PASS_LEN - 1 );
+ Conf_Oper[Conf_Oper_Count - 1].name[CLIENT_PASS_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "Password" ) == 0 )
+ {
+ /* Passwort des IRC Operator */
+ strncpy( Conf_Oper[Conf_Oper_Count - 1].pwd, Arg, CLIENT_PASS_LEN - 1 );
+ Conf_Oper[Conf_Oper_Count - 1].pwd[CLIENT_PASS_LEN - 1] = '\0';
+ return;
+ }
+
+ Config_Error( LOG_ERR, "%s, line %d (section \"Operator\"): Unknown variable \"%s\"!", NGIRCd_ConfFile, Line, Var );
+} /* Handle_OPERATOR */
+
+
+LOCAL VOID
+Handle_SERVER( INT Line, CHAR *Var, CHAR *Arg )
+{
+ LONG port;
+
+ assert( Line > 0 );
+ assert( Var != NULL );
+ assert( Arg != NULL );
+
+ if( strcasecmp( Var, "Host" ) == 0 )
+ {
+ /* Hostname des Servers */
+ strncpy( Conf_Server[Conf_Server_Count - 1].host, Arg, HOST_LEN - 1 );
+ Conf_Server[Conf_Server_Count - 1].host[HOST_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "Name" ) == 0 )
+ {
+ /* Name des Servers ("Nick") */
+ strncpy( Conf_Server[Conf_Server_Count - 1].name, Arg, CLIENT_ID_LEN - 1 );
+ Conf_Server[Conf_Server_Count - 1].name[CLIENT_ID_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "MyPassword" ) == 0 )
+ {
+ /* Passwort dieses Servers, welches empfangen werden muss */
+ strncpy( Conf_Server[Conf_Server_Count - 1].pwd_in, Arg, CLIENT_PASS_LEN - 1 );
+ Conf_Server[Conf_Server_Count - 1].pwd_in[CLIENT_PASS_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "PeerPassword" ) == 0 )
+ {
+ /* Passwort des anderen Servers, welches gesendet werden muss */
+ strncpy( Conf_Server[Conf_Server_Count - 1].pwd_out, Arg, CLIENT_PASS_LEN - 1 );
+ Conf_Server[Conf_Server_Count - 1].pwd_out[CLIENT_PASS_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "Port" ) == 0 )
+ {
+ /* Port, zu dem Verbunden werden soll */
+ port = atol( Arg );
+ if( port > 0 && port < 0xFFFF ) Conf_Server[Conf_Server_Count - 1].port = (INT)port;
+ else Config_Error( LOG_ERR, "%s, line %d (section \"Server\"): Illegal port number %ld!", NGIRCd_ConfFile, Line, port );
+ return;
+ }
+ if( strcasecmp( Var, "Group" ) == 0 )
+ {
+ /* Server-Gruppe */
+ Conf_Server[Conf_Server_Count - 1].group = atoi( Arg );
+ return;
+ }
+
+ Config_Error( LOG_ERR, "%s, line %d (section \"Server\"): Unknown variable \"%s\"!", NGIRCd_ConfFile, Line, Var );
+} /* Handle_SERVER */
+
+
+LOCAL VOID
+Handle_CHANNEL( INT Line, CHAR *Var, CHAR *Arg )
+{
+ assert( Line > 0 );
+ assert( Var != NULL );
+ assert( Arg != NULL );
+
+ if( strcasecmp( Var, "Name" ) == 0 )
+ {
+ /* Hostname des Servers */
+ strncpy( Conf_Channel[Conf_Channel_Count - 1].name, Arg, CHANNEL_NAME_LEN - 1 );
+ Conf_Channel[Conf_Channel_Count - 1].name[CHANNEL_NAME_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "Modes" ) == 0 )
+ {
+ /* Name des Servers ("Nick") */
+ strncpy( Conf_Channel[Conf_Channel_Count - 1].modes, Arg, CHANNEL_MODE_LEN - 1 );
+ Conf_Channel[Conf_Channel_Count - 1].modes[CHANNEL_MODE_LEN - 1] = '\0';
+ return;
+ }
+ if( strcasecmp( Var, "Topic" ) == 0 )
+ {
+ /* Passwort des Servers */
+ strncpy( Conf_Channel[Conf_Channel_Count - 1].topic, Arg, CHANNEL_TOPIC_LEN - 1 );
+ Conf_Channel[Conf_Channel_Count - 1].topic[CHANNEL_TOPIC_LEN - 1] = '\0';
+ return;
+ }
+
+ Config_Error( LOG_ERR, "%s, line %d (section \"Channel\"): Unknown variable \"%s\"!", NGIRCd_ConfFile, Line, Var );
+} /* Handle_CHANNEL */
+
+
+LOCAL VOID
+Validate_Config( VOID )
+{
+ /* Konfiguration ueberpruefen */
+
+ if( ! Conf_ServerName[0] )
+ {
+ /* Kein Servername konfiguriert */
+ Config_Error( LOG_ALERT, "No server name configured in \"%s\" ('ServerName')!", NGIRCd_ConfFile );
+ Config_Error( LOG_ALERT, "%s exiting due to fatal errors!", PACKAGE );
+ exit( 1 );
+ }
+
+#ifdef STRICT_RFC
+ if( ! Conf_ServerAdminMail[0] )
+ {
+ /* Keine Server-Information konfiguriert */
+ Config_Error( LOG_ALERT, "No administrator email address configured in \"%s\" ('AdminEMail')!", NGIRCd_ConfFile );
+ Config_Error( LOG_ALERT, "%s exiting due to fatal errors!", PACKAGE );
+ exit( 1 );
+ }
+#endif
+
+ if( ! Conf_ServerAdmin1[0] && ! Conf_ServerAdmin2[0] && ! Conf_ServerAdminMail[0] )
+ {
+ /* Keine Server-Information konfiguriert */
+ Log( LOG_WARNING, "No administrative information configured but required by RFC!" );
+ }
+} /* Validate_Config */
+
+
+#ifdef PROTOTYPES
+LOCAL VOID Config_Error( CONST INT Level, CONST CHAR *Format, ... )
+#else
+LOCAL VOID Config_Error( Level, Format, va_alist )
+CONST INT Level;
+CONST CHAR *Format;
+va_dcl
+#endif
+{
+ /* Fehler! Auf Console und/oder ins Log schreiben */
+
+ CHAR msg[MAX_LOG_MSG_LEN];
+ va_list ap;
+
+ assert( Format != NULL );
+
+ /* String mit variablen Argumenten zusammenbauen ... */
+#ifdef PROTOTYPES
+ va_start( ap, Format );
+#else
+ va_start( ap );
+#endif
+ vsnprintf( msg, MAX_LOG_MSG_LEN, Format, ap );
+ va_end( ap );
+
+ /* Im "normalen Betrieb" soll der Log-Mechanismus des ngIRCd verwendet
+ * werden, beim Testen der Konfiguration jedoch nicht, hier sollen alle
+ * Meldungen direkt auf die Konsole ausgegeben werden: */
+ if( Use_Log ) Log( Level, "%s", msg );
+ else puts( msg );
+} /* Config_Error */
+
+