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

org.appenders.log4j2.elasticsearch.ahc.discovery.AHCServiceDiscovery Maven / Gradle / Ivy

Go to download

Log4j2 Appender plugin pushing logs in batches to Elasticsearch (2.x/5.x/6.x/7.x/8.x) clusters

The newest version!
package org.appenders.log4j2.elasticsearch.ahc.discovery;

/*-
 * #%L
 * log4j2-elasticsearch
 * %%
 * Copyright (C) 2022 Rafal Foltynski
 * %%
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *      http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * #L%
 */

import org.appenders.log4j2.elasticsearch.ClientProvider;
import org.appenders.log4j2.elasticsearch.LifeCycle;
import org.appenders.log4j2.elasticsearch.metrics.Measured;
import org.appenders.log4j2.elasticsearch.metrics.MetricsRegistry;

import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

import static org.appenders.core.logging.InternalLogging.getLogger;

/**
 * Notifies configured listeners of changes in the address list.
 *
 * @param  client type
 */
public class AHCServiceDiscovery implements ServiceDiscovery, LifeCycle, Measured {

    private static final String NAME = AHCServiceDiscovery.class.getSimpleName();
    private volatile State state = State.STOPPED;

    private final ClientProvider clientProvider;
    private final ServiceDiscoveryRequest serviceDiscoveryRequest;
    private final long refreshInterval;

    private final List listeners = new ArrayList<>();
    private final Map cache = new ConcurrentHashMap<>();

    private final ScheduledExecutorService executor =
            Executors.newSingleThreadScheduledExecutor(runnable -> {
                Thread thread = new Thread(runnable);
                thread.setName("ServiceDiscovery");
                thread.setDaemon(true);
                return thread;
            });

    /**
     * @param clientProvider client to use
     * @param serviceDiscoveryRequest client-specific service discovery request
     * @param refreshInterval millis after previous request was completed
     */
    public AHCServiceDiscovery(
            final ClientProvider clientProvider,
            final ServiceDiscoveryRequest serviceDiscoveryRequest,
            final long refreshInterval) {
        this.refreshInterval = refreshInterval;
        this.clientProvider = clientProvider;
        this.serviceDiscoveryRequest = serviceDiscoveryRequest;
    }

    @Override
    public void addListener(final ServerInfoListener listener) {
        this.cache.clear();
        this.listeners.add(listener);
    }

    public void refresh() {

        if (!isStarted()) {
            throw new IllegalStateException(NAME + " not started");
        }

        getLogger().debug("{} : Refreshing address list", NAME);

        serviceDiscoveryRequest.execute(clientProvider.createClient(), new ServiceDiscoveryCallback());

    }

    private void processResult(final List addresses) {

        if (addresses.isEmpty()) {
            return;
        }

        final int previousCacheSize = cache.size();

        final List lastResult = new ArrayList<>();
        for (String address : addresses) {
            lastResult.add(cachedResult(address));
        }

        if (addresses.size() != cache.size()) {
            removeStaleEntries(lastResult);
        } else if (previousCacheSize == cache.size()) {
            // no changes
            lastResult.clear();
            return;
        }

        for (ServerInfoListener listener : listeners) {
            listener.onServerInfo(new ArrayList<>(lastResult));
        }

        lastResult.clear();

    }

    private ServerInfo cachedResult(final String address) {

        if (!cache.containsKey(address)) {
            final ServerInfo serverInfo = new ServerInfo(address);
            cache.put(address, serverInfo);
            getLogger().info("{}: New address found: {}", NAME, address);
        }

        return cache.get(address);

    }

    private void removeStaleEntries(final List lastResult) {
        final Collection cachedValues = cache.values();
        cachedValues.retainAll(lastResult);
    }

    @Override
    public void register(final MetricsRegistry registry) {
        Measured.of(clientProvider).register(registry);
    }

    @Override
    public void deregister() {
        Measured.of(clientProvider).deregister();
    }

    class ServiceDiscoveryCallback implements org.appenders.log4j2.elasticsearch.ahc.discovery.ServiceDiscoveryCallback> {

        @Override
        public void onSuccess(final List result) {
            processResult(result);
        }

        @Override
        public void onFailure(final Exception e) {
            getLogger().error(NAME + ": Unable to refresh addresses: " + e.getMessage(), e);
        }

    }
    class RefreshServerList extends Thread {

        @Override
        public void run() {
            try {
                refresh();
            } catch (Exception e) {
                getLogger().error(NAME + ": Unable to refresh addresses: " + e.getMessage(), e);
            }
        }

    }

    @Override
    public void start() {

        if (isStarted()) {
            return;
        }

        // Shared client will cause cycles. State must be set here
        state = State.STARTED;

        LifeCycle.of(clientProvider).start();

        getLogger().debug("{}: Starting executor", NAME);

        scheduleRefreshTask();

        getLogger().debug("{}: Started", NAME);

    }

    /* visible for testing */
    void scheduleRefreshTask() {
        this.executor.scheduleWithFixedDelay(
                new RefreshServerList(),
                0,
                refreshInterval,
                TimeUnit.MILLISECONDS
        );
    }

    @Override
    public void stop() {

        if (isStopped()) {
            return;
        }

        // Shared client will cause cycles. State must be set here
        state = State.STOPPED;

        getLogger().debug("{}: Shutting down executor", NAME);
        executor.shutdown();

        deregister();
        LifeCycle.of(clientProvider).stop();

        listeners.clear();

        getLogger().debug("{}: Stopped", NAME);

    }

    @Override
    public boolean isStarted() {
        return state == State.STARTED;
    }

    @Override
    public boolean isStopped() {
        return state == State.STOPPED;
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy