All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jmxtrans.agent.zabbix.ZabbixTcpOutputWriter Maven / Gradle / Ivy

There is a newer version: 1.2.11
Show newest version
/*
 * Copyright (c) 2010-2013 the original author or authors
 *
 * Permission is hereby granted, free of charge, to any person obtaining
 * a copy of this software and associated documentation files (the
 * "Software"), to deal in the Software without restriction, including
 * without limitation the rights to use, copy, modify, merge, publish,
 * distribute, sublicense, and/or sell copies of the Software, and to
 * permit persons to whom the Software is furnished to do so, subject to
 * the following conditions:
 *
 * The above copyright notice and this permission notice shall be
 * included in all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
 * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
 * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 *
 */
package org.jmxtrans.agent.zabbix;

import static org.jmxtrans.agent.util.ConfigurationUtils.getInt;
import static org.jmxtrans.agent.util.ConfigurationUtils.getString;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.net.ConnectException;
import java.net.InetSocketAddress;
import java.net.Socket;
import java.nio.charset.StandardCharsets;
import java.util.Map;
import java.util.concurrent.TimeUnit;
import java.util.logging.Level;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;

import org.jmxtrans.agent.AbstractOutputWriter;
import org.jmxtrans.agent.OutputWriter;
import org.jmxtrans.agent.util.io.IoUtils;
import org.jmxtrans.agent.util.net.HostAndPort;

/**
 * @author Steve McDuff
 */
public class ZabbixTcpOutputWriter extends AbstractOutputWriter implements OutputWriter
{

    private static final int ZABBIX_HEADER_LENGTH = 13;
    public final static String SETTING_SOCKET_CONNECT_TIMEOUT_IN_MILLIS = "socket.connectTimeoutInMillis";
    public final static int SETTING_SOCKET_CONNECT_TIMEOUT_IN_MILLIS_DEFAULT_VALUE = 500;

    protected HostAndPort zabbixServerHostAndPort;
    private Socket socket;
    private ByteArrayOutputStream out = new ByteArrayOutputStream();
    private byte[] readBuffer = new byte[10000];
    private int socketConnectTimeoutInMillis = SETTING_SOCKET_CONNECT_TIMEOUT_IN_MILLIS_DEFAULT_VALUE;
    private ZabbixMetricMessageBuilder messageBuilder;

    private int printMetricCount = 0;
    private int metricBatchSize;
    
    private boolean failedConnection = false;

    @Override
    public void postConstruct(Map settings)
    {
        super.postConstruct(settings);

        zabbixServerHostAndPort = new HostAndPort(getString(settings, ZabbixOutputWriterCommonSettings.SETTING_HOST),
            getInt(settings, ZabbixOutputWriterCommonSettings.SETTING_PORT,
                ZabbixOutputWriterCommonSettings.SETTING_PORT_DEFAULT_VALUE));
        messageBuilder = new ZabbixMetricMessageBuilder(
            ZabbixOutputWriterCommonSettings.getConfiguredHostName(settings));
        socketConnectTimeoutInMillis = getInt(settings, SETTING_SOCKET_CONNECT_TIMEOUT_IN_MILLIS,
            SETTING_SOCKET_CONNECT_TIMEOUT_IN_MILLIS_DEFAULT_VALUE);

        metricBatchSize = ZabbixOutputWriterCommonSettings.getMetricBatchSize(settings);

        logger.log(getInfoLevel(),
            "ZabbixTcpOutputWriter is configured with " + zabbixServerHostAndPort + ", serverName=" +
                messageBuilder.getHostName() + ", socketConnectTimeoutInMillis=" + socketConnectTimeoutInMillis +
                ", metricBatchSize=" + metricBatchSize);
    }

    @Override
    public void writeInvocationResult(@Nonnull String invocationName, @Nullable Object value) throws IOException
    {
        writeQueryResult(invocationName, null, value);
    }

    private byte[] messageHeader = "{\"request\":\"sender data\",\"data\":[".getBytes(StandardCharsets.UTF_8);
    private byte[] comma = ",".getBytes(StandardCharsets.UTF_8);
    private byte[] messageFooter = "]}".getBytes(StandardCharsets.UTF_8);
    

    @Override
    public void writeQueryResult(@Nonnull String metricName, @Nullable String type, @Nullable Object value)
        throws IOException
    {
        String msg = messageBuilder.buildMessage(metricName, value,
            TimeUnit.SECONDS.convert(System.currentTimeMillis(), TimeUnit.MILLISECONDS));

        try
        {
            if (logger.isLoggable(getTraceLevel()))
            {
                logger.log(getTraceLevel(), "Send '" + msg + "' to " + zabbixServerHostAndPort);
            }

            if (printMetricCount == 0)
            {
                // print the header
                writeMessage(messageHeader);
            }
            else
            {
                writeMessage(comma);
            }

            printMetricCount += 1;
            writeMessage(msg);

            if (printMetricCount >= metricBatchSize)
            {
                if (logger.isLoggable(Level.FINE))
                {
                    logger.fine(
                        "Reached batch size maximum of " + metricBatchSize + " .Forcing message output to Zabbix.");
                }
                postCollect();
            }

        }
        catch (IOException e)
        {
            logger.log(Level.WARNING,
                "Exception sending '" + msg + "' of size : " + out.size() + "bytes to " + zabbixServerHostAndPort, e);

            releaseZabbixConnection();
            throw e;
        }
    }

    private void writeMessage(String msg) throws IOException
    {
        if (logger.isLoggable(Level.FINEST))
        {
            logger.finest("Print : " + msg);
        }

        byte[] bytes = msg.getBytes(StandardCharsets.UTF_8);
        writeMessage(bytes);
    }

    protected void writeMessage(byte[] data) throws IOException
    {
        out.write(data);
    }

    private void writeZabbixHeader(int length) throws IOException
    {
        byte[] zabbixHeader = new byte[]
        { 'Z', 'B', 'X', 'D', '\1', (byte) (length & 0xFF), (byte) ((length >> 8) & 0x00FF),
            (byte) ((length >> 16) & 0x0000FF), (byte) ((length >> 24) & 0x000000FF), '\0', '\0', '\0', '\0' };

        socket.getOutputStream().write(zabbixHeader);
    }

    private void releaseZabbixConnection()
    {
        IoUtils.closeQuietly(out);
        IoUtils.closeQuietly(socket);
    }

    private void ensureZabbixConnection() throws IOException
    {
        boolean socketIsValid;
        try
        {
            socketIsValid = socket != null && socket.isConnected() && socket.isBound() && !socket.isClosed() &&
                !socket.isInputShutdown() && !socket.isOutputShutdown();
        }
        catch (Exception e)
        {
            socketIsValid = false;
        }
        if (!socketIsValid)
        {
            long start = System.currentTimeMillis();
            try
            {
                socket = new Socket();
                socket.setKeepAlive(true);
                socket.connect(
                    new InetSocketAddress(zabbixServerHostAndPort.getHost(), zabbixServerHostAndPort.getPort()),
                    socketConnectTimeoutInMillis);
            }
            catch (IOException e)
            {
                ConnectException ce = new ConnectException("Exception connecting to " + zabbixServerHostAndPort);
                ce.initCause(e);
                throw ce;
            }
            long end = System.currentTimeMillis();

            if (logger.isLoggable(Level.FINE))
            {
                logger.fine("Connect time : " + (end - start));
            }
        }
    }

    @Override
    public void postCollect() throws IOException
    {
        if (printMetricCount == 0)
        {
            // nothing to print
            return;
        }

        try
        {
            writeMessage(messageFooter);

            byte[] byteArray = out.toByteArray();
            out.reset();

            if (logger.isLoggable(Level.FINEST))
            {
                String msg = new String(byteArray, StandardCharsets.UTF_8);
                logger.finest("message : " + msg);
            }

            ensureZabbixConnection();

            writeZabbixHeader(byteArray.length);
            socket.getOutputStream().write(byteArray);

            printMetricCount = 0;

            drainInputStream();

            // Zabbix seems to drop connections on every message.
            // Forcing a drop prevents timeout exceptions for subsequent messages.
            releaseZabbixConnection();
            failedConnection = false;
        }
        catch (IOException e)
        {
            // log subsequent failures at a lower log level.
            if( failedConnection ) {
                logger.log(Level.FINE, "Exception flushing the stream to " + zabbixServerHostAndPort, e);
            }
            else {
                logger.log(Level.WARNING, "Exception flushing the stream to " + zabbixServerHostAndPort, e);
            }
            releaseZabbixConnection();
            throw e;
        }
    }

    private void drainInputStream() throws IOException
    {
        try
        {
            int readSize = socket.getInputStream().read(readBuffer);
            if (logger.isLoggable(Level.FINE))
            {

                String message = extractZabbixResponseString(readSize, readBuffer);
                logger.log(Level.FINE, message);
            }
        }
        catch (Exception ex)
        {
            logger.log(Level.WARNING, "Failed to read from scoket " + zabbixServerHostAndPort, ex);
        }
    }

    private String extractZabbixResponseString(int readSize, byte[] buffer)
    {
        String msg = "";
        if (readSize > ZABBIX_HEADER_LENGTH)
        {
            // skip the zabbix header
            msg = new String(buffer, ZABBIX_HEADER_LENGTH, readSize - ZABBIX_HEADER_LENGTH, StandardCharsets.UTF_8);
        }
        return msg;
    }

    @Override
    public String toString()
    {
        return "ZabbixTcpOutputWriter{" + ", " + zabbixServerHostAndPort + ", serverName='" +
            messageBuilder.getHostName() + '\'' + '}';
    }

    @Override
    public void preDestroy()
    {
        super.preDestroy();
        releaseZabbixConnection();
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy