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

org.jmxtrans.agent.JmxTransExporter Maven / Gradle / Ivy

/*
 * 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;

import java.lang.management.ManagementFactory;
import java.math.BigInteger;
import java.sql.Timestamp;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.logging.Level;

import javax.management.MBeanServer;

import org.jmxtrans.agent.util.GcdCalculator;
import org.jmxtrans.agent.util.logging.Logger;

/**
 * @author Cyrille Le Clerc
 */
public class JmxTransExporter {
    private final Logger logger = Logger.getLogger(getClass().getName());
    private ThreadFactory threadFactory = new ThreadFactory() {
        final AtomicInteger counter = new AtomicInteger();

        @Override
        public Thread newThread(Runnable r) {
            Thread thread = Executors.defaultThreadFactory().newThread(r);
            thread.setDaemon(true);
            thread.setName("jmxtrans-agent-" + counter.incrementAndGet());
            return thread;
        }
    };
    private ScheduledExecutorService scheduledExecutorService;
    private MBeanServer mbeanServer = ManagementFactory.getPlatformMBeanServer();
    private ScheduledFuture scheduledFuture;
    private JmxTransConfigurationLoader configLoader;
    private volatile JmxTransExporterConfiguration config;
    private volatile List collectors;
    private volatile long runIntervalMillis;

    public JmxTransExporter(JmxTransConfigurationLoader configLoader) {
        this.configLoader = configLoader;
        loadNewConfiguration();
    }

    private void loadNewConfiguration() {
        this.config = configLoader.loadConfiguration();
        logger.finest("Configuration loaded: " + config);
        this.collectors = createTimeTrackingCollectors();
        this.runIntervalMillis = calculateRunIntervalMillis();
    }


    private List createTimeTrackingCollectors() {
        List newCollectors = new ArrayList<>();
        for (Query q : config.getQueries()) {
            TimeTrackingCollector timeTrackingCollector = createTimeTrackingCollector(q, q.getCollectIntervalOverrideOrNull());
            newCollectors.add(timeTrackingCollector);
        }
        for (Invocation i : config.getInvocations()) {
            TimeTrackingCollector timeTrackingCollector = createTimeTrackingCollector(i, i.getCollectIntervalOverrideOrNull());
            newCollectors.add(timeTrackingCollector);
        }
        return newCollectors;
    }

    private TimeTrackingCollector createTimeTrackingCollector(Collector collector, Integer collectIntervalOverride) {
        int actualCollectInterval = collectIntervalOverride != null ? collectIntervalOverride : config.getCollectInterval();
        TimeTrackingCollector timeTrackingCollector = new TimeTrackingCollector(collector, TimeUnit.MILLISECONDS.convert(actualCollectInterval, config.getCollectIntervalTimeUnit()));
        return timeTrackingCollector;
    }

    private long calculateRunIntervalMillis() {
        // Use the greatest common divisor of the collect intervals to get period in which we need to collect
        if (collectors.isEmpty()) {
            // Ensure that we trigger runs even if there are no queries so that config refresh works.
            return config.getCollectInterval();
        }
        List collectIntervals = new ArrayList<>();
        for (TimeTrackingCollector c : collectors) {
            collectIntervals.add(c.getCollectIntervalMillis());
        }
        return GcdCalculator.gcd(collectIntervals);
    }

    public void start() {
        if (logger.isLoggable(Level.FINER)) {
            logger.fine("starting " + this.toString() + " ...");
        } else {
            logger.fine("starting " + getClass().getName() + " ...");
        }

        if (scheduledExecutorService != null || scheduledFuture != null)
            throw new IllegalArgumentException("Exporter is already started: scheduledExecutorService=" + scheduledExecutorService + ", scheduledFuture=" + scheduledFuture);

        scheduledExecutorService = Executors.newScheduledThreadPool(1, threadFactory);

        if (config.getResultNameStrategy() == null)
            throw new IllegalStateException("resultNameStrategy is not defined, jmxTransExporter is not properly initialised");

        scheduledFuture = scheduledExecutorService.scheduleWithFixedDelay(new Runnable() {
            @Override
            public void run() {
                collectAndExport();
            }
        }, runIntervalMillis / 2, runIntervalMillis, TimeUnit.MILLISECONDS);

        if (config.getConfigReloadInterval() >= 0) {
            Runnable runnable = new Runnable() {
                private final Logger logger = Logger.getLogger(JmxTransExporter.class.getName() + ".reloader");
                private long lastModified = configLoader.lastModified();

                @Override
                public void run() {
                    long newLastModified = configLoader.lastModified();
                    if (newLastModified == 0L) {
                        if (logger.isLoggable(Level.FINER))
                            logger.finer("Don't reload lastModified=" + lastModified + " / " + new Timestamp(lastModified) +
                                    ", newLastModified=" + newLastModified);
                        // ignore new config not found
                    } else if (newLastModified > lastModified) {
                        logger.info("jmxtrans-agent configuration has changed. Reload " + configLoader);
                        if (logger.isLoggable(Level.FINER))
                            logger.finer("Reload lastModified=" + lastModified + " / " + new Timestamp(lastModified) +
                                ", newLastModified=" + newLastModified + " / " + new Timestamp(newLastModified));

                        lastModified = newLastModified;
                        stop();
                        loadNewConfiguration();
                        start();
                    } else {
                        if (logger.isLoggable(Level.FINER))
                            logger.finer("Don't reload lastModified=" + lastModified + " / " + new Timestamp(lastModified) +
                                    ", newLastModified=" + newLastModified + " / " + new Timestamp(newLastModified));

                        // ignore, config not modified
                    }

                }
            };
            int configReloadIntervalInSecs = Math.max(config.getConfigReloadInterval(), 5);
            if (logger.isLoggable(Level.INFO))
                logger.info("Configuration reload interval: " + configReloadIntervalInSecs + "secs");
            scheduledExecutorService.scheduleWithFixedDelay(runnable, 0, configReloadIntervalInSecs, TimeUnit.SECONDS);
        }

        logger.fine(getClass().getName() + " started");
    }

    public void stop() {
        // cancel jobs
        if (scheduledFuture != null) {
            scheduledFuture.cancel(true);
            scheduledFuture = null;
        }
        scheduledExecutorService.shutdown();

        // one last export
        collectAndExport();

        // wait for stop
        try {
            scheduledExecutorService.awaitTermination(runIntervalMillis, TimeUnit.MILLISECONDS);

        } catch (InterruptedException e) {
            throw new IllegalStateException(e);
        }
        scheduledExecutorService = null;

        config.getOutputWriter().preDestroy();

        logger.info(getClass().getName() + " stopped.");

    }

    protected void collectAndExport() {
        OutputWriter outputWriter = config.getOutputWriter();
        try {
            outputWriter.preCollect();
            for (TimeTrackingCollector collector : collectors) {
                try {
                    collector.collectIfEnoughTimeHasPassed(mbeanServer, outputWriter);
                } catch (Exception e) {
                    logger.log(Level.WARNING, "Ignore exception collecting with collector " + collector, e);
                }
            }
            outputWriter.postCollect();
        } catch (Exception e) {
            logger.log(Level.WARNING, "Ignore exception flushing metrics ", e);
        }
    }

    @Override
    public String toString() {
        return "JmxTransExporter{" +
                ", configuration=" + config +
                '}';
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy