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

org.graylog2.shared.initializers.PeriodicalsService Maven / Gradle / Ivy

There is a newer version: 6.1.4
Show newest version
/*
 * Copyright (C) 2020 Graylog, Inc.
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the Server Side Public License, version 1,
 * as published by MongoDB, Inc.
 *
 * This program 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
 * Server Side Public License for more details.
 *
 * You should have received a copy of the Server Side Public License
 * along with this program. If not, see
 * .
 */
package org.graylog2.shared.initializers;

import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Sets;
import com.google.common.eventbus.EventBus;
import com.google.common.eventbus.Subscribe;
import com.google.common.util.concurrent.AbstractIdleService;
import org.graylog2.cluster.leader.LeaderChangedEvent;
import org.graylog2.cluster.leader.LeaderElectionService;
import org.graylog2.periodical.Periodicals;
import org.graylog2.plugin.periodical.Periodical;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.inject.Inject;
import javax.inject.Singleton;
import java.util.Collection;
import java.util.HashSet;
import java.util.Set;

@Singleton
public class PeriodicalsService extends AbstractIdleService {
    private static final Logger LOG = LoggerFactory.getLogger(PeriodicalsService.class);

    private final Periodicals periodicals;
    private final Set allPeriodicals;
    private final EventBus eventBus;
    private final LeaderElectionService leaderElectionService;

    private final Set leaderNodePeriodicals = new HashSet<>();
    private final Set anyNodePeriodicals = new HashSet<>();

    @Inject
    public PeriodicalsService(Periodicals periodicals, Set allPeriodicals, EventBus eventBus,
                              LeaderElectionService leaderElectionService) {
        this.periodicals = periodicals;
        this.allPeriodicals = allPeriodicals;
        this.eventBus = eventBus;
        this.leaderElectionService = leaderElectionService;

        allPeriodicals.forEach(p -> {
            if (p.leaderOnly()) {
                leaderNodePeriodicals.add(p);
            } else {
                anyNodePeriodicals.add(p);
            }
        });
    }

    @Override
    protected void startUp() throws Exception {
        eventBus.register(this);

        if (leaderElectionService.isLeader()) {
            LOG.info("Starting {} periodicals ...", allPeriodicals.size());
            startPeriodicals(allPeriodicals);
        } else {
            LOG.info("Starting {} periodicals ...", anyNodePeriodicals.size());
            LOG.info("Delaying start of {} periodicals until this node becomes leader ...", leaderNodePeriodicals.size());
            startPeriodicals(anyNodePeriodicals);
        }
    }

    @Subscribe
    public void leaderChanged(LeaderChangedEvent leaderChangedEvent) {
        if (leaderElectionService.isLeader()) {
            LOG.info("Starting {} periodicals ...", leaderNodePeriodicals.size());
            startPeriodicals(leaderNodePeriodicals);
        } else {
            final Set runningLeaderNodePeriodicals =
                    Sets.intersection(leaderNodePeriodicals, periodicals.getAllRunning());
            LOG.info("Stopping {} periodicals ...", runningLeaderNodePeriodicals);
            stopPeriodicals(runningLeaderNodePeriodicals);
        }
    }

    private synchronized void startPeriodicals(Set periodicalsToStart) {
        final Sets.SetView notYetStartedPeriodicals =
                Sets.difference(periodicalsToStart, ImmutableSet.copyOf(periodicals.getAll()));

        int numOfPeriodicalsToSkip = periodicalsToStart.size() - notYetStartedPeriodicals.size();

        if (numOfPeriodicalsToSkip > 0) {
            LOG.warn("Skipping start of {} periodicals which have already been started.", numOfPeriodicalsToSkip);
        }

        for (Periodical periodical : notYetStartedPeriodicals) {
            try {
                periodical.initialize();

                if (!periodical.startOnThisNode()) {
                    LOG.info("Not starting [{}] periodical. Not configured to run on this node.", periodical.getClass().getCanonicalName());
                    continue;
                }

                // Register and start.
                periodicals.registerAndStart(periodical);
            } catch (Exception e) {
                LOG.error("Could not initialize periodical.", e);
            }
        }
    }

    private synchronized void stopPeriodicals(Collection periodicalsToStop) {
        periodicalsToStop.forEach(periodicals::unregisterAndStop);
    }

    @Override
    protected void shutDown() throws Exception {
        eventBus.unregister(this);
        stopPeriodicals(periodicals.getAllStoppedOnGracefulShutdown());
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy