de.siegmar.logbackgelf.GelfTcpAppender Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of logback-gelf Show documentation
Show all versions of logback-gelf Show documentation
Logback appender for sending GELF messages with zero additional dependencies.
The newest version!
/*
* Logback GELF - zero dependencies Logback GELF appender library.
* Copyright (C) 2016 Oliver Siegmar
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
package de.siegmar.logbackgelf;
import javax.net.SocketFactory;
import de.siegmar.logbackgelf.pool.SimpleObjectPool;
public class GelfTcpAppender extends AbstractGelfAppender {
private static final int DEFAULT_CONNECT_TIMEOUT = 15_000;
private static final int DEFAULT_SOCKET_TIMEOUT = 5_000;
private static final int DEFAULT_RECONNECT_INTERVAL = 60;
private static final int DEFAULT_MAX_RETRIES = 2;
private static final int DEFAULT_RETRY_DELAY = 3_000;
private static final int DEFAULT_POOL_SIZE = 2;
private static final int DEFAULT_POOL_MAX_WAIT_TIME = 5_000;
private static final int DEFAULT_POOL_MAX_IDLE_TIME = -1;
/**
* Maximum time (in milliseconds) to wait for establishing a connection. A value of 0 disables
* the connect timeout. Default: {@value DEFAULT_CONNECT_TIMEOUT} milliseconds.
*/
private int connectTimeout = DEFAULT_CONNECT_TIMEOUT;
/**
* Maximum time (in milliseconds) to block when reading a socket. A value of 0 disables the socket timeout.
* Default: {@value DEFAULT_SOCKET_TIMEOUT} milliseconds.
*/
private int socketTimeout = DEFAULT_SOCKET_TIMEOUT;
/**
* Time interval (in seconds) after an existing connection is closed and re-opened.
* A value of -1 disables automatic reconnects. Default: {@value DEFAULT_RECONNECT_INTERVAL} seconds.
*/
private int reconnectInterval = DEFAULT_RECONNECT_INTERVAL;
/**
* Number of retries. A value of 0 disables retry attempts. Default: {@value DEFAULT_MAX_RETRIES}.
*/
private int maxRetries = DEFAULT_MAX_RETRIES;
/**
* Time (in milliseconds) between retry attempts. Ignored if maxRetries is 0.
* Default: {@value DEFAULT_RETRY_DELAY} milliseconds.
*/
private int retryDelay = DEFAULT_RETRY_DELAY;
/**
* Number of concurrent tcp connections (minimum 1). Default: {@value DEFAULT_POOL_SIZE}.
*/
private int poolSize = DEFAULT_POOL_SIZE;
/**
* Maximum amount of time (in milliseconds) to wait for a connection to become
* available from the pool. A value of -1 disables the timeout.
* Default: {@value DEFAULT_POOL_MAX_WAIT_TIME} milliseconds.
*/
private int poolMaxWaitTime = DEFAULT_POOL_MAX_WAIT_TIME;
/**
* Maximum amount of time (in seconds) that a pooled connection can be idle before it is
* considered 'stale' and will not be reused. A value of -1 disables the max idle time feature.
* Default: {@value DEFAULT_POOL_MAX_IDLE_TIME}.
*/
private int poolMaxIdleTime = DEFAULT_POOL_MAX_IDLE_TIME;
private SimpleObjectPool connectionPool;
public int getConnectTimeout() {
return connectTimeout;
}
public void setConnectTimeout(final int connectTimeout) {
this.connectTimeout = connectTimeout;
}
public int getSocketTimeout() {
return socketTimeout;
}
public void setSocketTimeout(final int socketTimeout) {
this.socketTimeout = socketTimeout;
}
public int getReconnectInterval() {
return reconnectInterval;
}
public void setReconnectInterval(final int reconnectInterval) {
this.reconnectInterval = reconnectInterval;
}
public int getMaxRetries() {
return maxRetries;
}
public void setMaxRetries(final int maxRetries) {
this.maxRetries = maxRetries;
}
public int getRetryDelay() {
return retryDelay;
}
public void setRetryDelay(final int retryDelay) {
this.retryDelay = retryDelay;
}
public int getPoolSize() {
return poolSize;
}
public void setPoolSize(final int poolSize) {
this.poolSize = poolSize;
}
public long getPoolMaxWaitTime() {
return poolMaxWaitTime;
}
public void setPoolMaxWaitTime(final int poolMaxWaitTime) {
this.poolMaxWaitTime = poolMaxWaitTime;
}
public int getPoolMaxIdleTime() {
return poolMaxIdleTime;
}
public void setPoolMaxIdleTime(final int poolMaxIdleTime) {
this.poolMaxIdleTime = poolMaxIdleTime;
}
@Override
protected void startAppender() {
final AddressResolver addressResolver = new AddressResolver(getGraylogHost());
connectionPool = new SimpleObjectPool<>(() -> new TcpConnection(initSocketFactory(),
addressResolver, getGraylogPort(), connectTimeout, socketTimeout),
poolSize, poolMaxWaitTime, reconnectInterval, poolMaxIdleTime);
}
protected SocketFactory initSocketFactory() {
return SocketFactory.getDefault();
}
@SuppressWarnings("checkstyle:IllegalCatch")
@Override
protected void appendMessage(final byte[] messageToSend) {
try {
RetryUtil.retry(() -> sendMessage(messageToSend), this::isStarted, maxRetries, retryDelay);
} catch (final Exception e) {
addError(String.format("Error sending message via tcp://%s:%s",
getGraylogHost(), getGraylogPort()), e);
}
}
/**
* Send message to socket's output stream.
*
* @param messageToSend message to send.
*/
private Void sendMessage(final byte[] messageToSend) throws Exception {
connectionPool.execute(tcpConnection -> tcpConnection.write(messageToSend));
return null;
}
@Override
protected void close() {
connectionPool.close();
}
}