package org.graylog2.syslog4j.server.impl.net.tcp;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.SocketException;
import java.net.SocketTimeoutException;
import java.util.Iterator;
import javax.net.ServerSocketFactory;
import org.graylog2.syslog4j.SyslogConstants;
import org.graylog2.syslog4j.SyslogRuntimeException;
import org.graylog2.syslog4j.server.SyslogServerEventIF;
import org.graylog2.syslog4j.server.SyslogServerIF;
import org.graylog2.syslog4j.server.impl.AbstractSyslogServer;
import org.graylog2.syslog4j.util.SyslogUtility;
/**
* TCPNetSyslogServer provides a simple threaded TCP/IP server implementation.
* <p/>
* <p>Syslog4j is licensed under the Lesser GNU Public License v2.1. A copy
* of the LGPL license is available in the META-INF folder in all
* distributions of Syslog4j and in the base directory of the "doc" ZIP.</p>
*
* @author <syslog4j@productivity.org>
* @version $Id: TCPNetSyslogServer.java,v 1.23 2010/11/28 22:07:57 cvs Exp $
*/
public class TCPNetSyslogServer extends AbstractSyslogServer {
public static class TCPNetSyslogSocketHandler implements Runnable {
protected SyslogServerIF server = null;
protected Socket socket = null;
protected Sessions sessions = null;
public TCPNetSyslogSocketHandler(Sessions sessions, SyslogServerIF server, Socket socket) {
this.sessions = sessions;
this.server = server;
this.socket = socket;
synchronized (this.sessions) {
this.sessions.addSocket(socket);
}
}
public void run() {
boolean timeout = false;
try {
BufferedReader br = new BufferedReader(new InputStreamReader(this.socket.getInputStream()));
String line = br.readLine();
if (line != null) {
AbstractSyslogServer.handleSessionOpen(this.sessions, this.server, this.socket);
}
while (line != null && line.length() != 0) {
SyslogServerEventIF event = createEvent(this.server.getConfig(), line, this.socket.getInetAddress());
AbstractSyslogServer.handleEvent(this.sessions, this.server, this.socket, event);
line = br.readLine();
}
} catch (SocketTimeoutException ste) {
timeout = true;
} catch (SocketException se) {
AbstractSyslogServer.handleException(this.sessions, this.server, this.socket.getRemoteSocketAddress(), se);
if ("Socket closed".equals(se.getMessage())) {
//
} else {
//
}
} catch (IOException ioe) {
AbstractSyslogServer.handleException(this.sessions, this.server, this.socket.getRemoteSocketAddress(), ioe);
}
try {
AbstractSyslogServer.handleSessionClosed(this.sessions, this.server, this.socket, timeout);
this.sessions.removeSocket(this.socket);
this.socket.close();
} catch (IOException ioe) {
AbstractSyslogServer.handleException(this.sessions, this.server, this.socket.getRemoteSocketAddress(), ioe);
}
}
}
protected ServerSocket serverSocket = null;
protected final Sessions sessions = new Sessions();
protected TCPNetSyslogServerConfigIF tcpNetSyslogServerConfig = null;
public void initialize() throws SyslogRuntimeException {
this.tcpNetSyslogServerConfig = null;
try {
this.tcpNetSyslogServerConfig = (TCPNetSyslogServerConfigIF) this.syslogServerConfig;
} catch (ClassCastException cce) {
throw new SyslogRuntimeException("config must be of type TCPNetSyslogServerConfig");
}
if (this.syslogServerConfig == null) {
throw new SyslogRuntimeException("config cannot be null");
}
if (this.tcpNetSyslogServerConfig.getBacklog() < 1) {
this.tcpNetSyslogServerConfig.setBacklog(SyslogConstants.SERVER_SOCKET_BACKLOG_DEFAULT);
}
}
public Sessions getSessions() {
return this.sessions;
}
public synchronized void shutdown() {
super.shutdown();
try {
if (this.serverSocket != null) {
if (this.syslogServerConfig.getShutdownWait() > 0) {
SyslogUtility.sleep(this.syslogServerConfig.getShutdownWait());
}
this.serverSocket.close();
}
synchronized (this.sessions) {
Iterator i = this.sessions.getSockets();
if (i != null) {
while (i.hasNext()) {
Socket s = (Socket) i.next();
s.close();
}
}
}
} catch (IOException ioe) {
//
}
if (this.thread != null) {
this.thread.interrupt();
this.thread = null;
}
}
protected ServerSocketFactory getServerSocketFactory() throws IOException {
ServerSocketFactory serverSocketFactory = ServerSocketFactory.getDefault();
return serverSocketFactory;
}
protected ServerSocket createServerSocket() throws IOException {
ServerSocket newServerSocket = null;
ServerSocketFactory factory = getServerSocketFactory();
if (this.syslogServerConfig.getHost() != null) {
InetAddress inetAddress = InetAddress.getByName(this.syslogServerConfig.getHost());
newServerSocket = factory.createServerSocket(this.syslogServerConfig.getPort(), this.tcpNetSyslogServerConfig.getBacklog(), inetAddress);
} else {
if (this.tcpNetSyslogServerConfig.getBacklog() < 1) {
newServerSocket = factory.createServerSocket(this.syslogServerConfig.getPort());
} else {
newServerSocket = factory.createServerSocket(this.syslogServerConfig.getPort(), this.tcpNetSyslogServerConfig.getBacklog());
}
}
return newServerSocket;
}
public void run() {
try {
this.serverSocket = createServerSocket();
this.shutdown = false;
} catch (SocketException se) {
throw new SyslogRuntimeException(se);
} catch (IOException ioe) {
throw new SyslogRuntimeException(ioe);
}
handleInitialize(this);
while (!this.shutdown) {
try {
Socket socket = this.serverSocket.accept();
if (this.tcpNetSyslogServerConfig.getTimeout() > 0) {
socket.setSoTimeout(this.tcpNetSyslogServerConfig.getTimeout());
}
if (this.tcpNetSyslogServerConfig.getMaxActiveSockets() > 0 && this.sessions.size() >= this.tcpNetSyslogServerConfig.getMaxActiveSockets()) {
if (this.tcpNetSyslogServerConfig.getMaxActiveSocketsBehavior() == TCPNetSyslogServerConfigIF.MAX_ACTIVE_SOCKETS_BEHAVIOR_REJECT) {
try {
socket.close();
} catch (Exception e) {
//
}
socket = null;
} else if (this.tcpNetSyslogServerConfig.getMaxActiveSocketsBehavior() == TCPNetSyslogServerConfigIF.MAX_ACTIVE_SOCKETS_BEHAVIOR_BLOCK) {
while (!this.shutdown && this.sessions.size() >= this.tcpNetSyslogServerConfig.getMaxActiveSockets() && socket.isConnected() && !socket.isClosed()) {
SyslogUtility.sleep(SyslogConstants.THREAD_LOOP_INTERVAL_DEFAULT);
}
}
}
if (socket != null) {
TCPNetSyslogSocketHandler handler = new TCPNetSyslogSocketHandler(this.sessions, this, socket);
Thread t = new Thread(handler);
t.start();
}
} catch (SocketException se) {
if ("Socket closed".equals(se.getMessage())) {
this.shutdown = true;
} else {
//
}
} catch (IOException ioe) {
//
}
}
handleDestroy(this);
}
}