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

io.hekate.cluster.seed.SeedNodeProviderGroup Maven / Gradle / Ivy

/*
 * Copyright 2022 The Hekate Project
 *
 * The Hekate Project licenses this file to you 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.
 */

package io.hekate.cluster.seed;

import io.hekate.cluster.ClusterService;
import io.hekate.core.HekateException;
import io.hekate.core.internal.util.ArgAssert;
import io.hekate.core.internal.util.ConfigCheck;
import io.hekate.core.jmx.JmxSupport;
import io.hekate.core.report.ConfigReportSupport;
import io.hekate.core.report.ConfigReporter;
import io.hekate.util.format.ToString;
import java.net.InetSocketAddress;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.CopyOnWriteArrayList;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import static io.hekate.core.internal.util.StreamUtils.nullSafe;
import static java.util.Collections.unmodifiableList;
import static java.util.stream.Collectors.toList;

/**
 * Group of {@link SeedNodeProvider}s.
 *
 * 

* This class provides support for registering multiple {@link SeedNodeProvider}s simultaneously to the {@link ClusterService} and act as a * single provider. *

* *

* It is possible to specify an error handling policy via the {@link SeedNodeProviderGroupConfig#setPolicy(SeedNodeProviderGroupPolicy)} * method. For the list of available policies please see the documentation of {@link SeedNodeProviderGroupPolicy} enum's values. *

* *

* Below is the configuration example: * ${source: cluster/seed/SeedNodeProviderGroupJavadocTest.java#configuration} *

* * @see SeedNodeProviderGroupPolicy * @see SeedNodeProviderGroupConfig */ public class SeedNodeProviderGroup implements SeedNodeProvider, JmxSupport>, ConfigReportSupport { /** See {@link #withPolicy(String, List, SeedNodeProviderTask)}. */ private interface SeedNodeProviderTask { void execute(SeedNodeProvider provider) throws HekateException; } private static final Logger log = LoggerFactory.getLogger(SeedNodeProviderGroup.class); /** Error handling policy (see {@link #withPolicy(String, List, SeedNodeProviderTask)}. */ private final SeedNodeProviderGroupPolicy policy; /** All configured providers. */ private final List allProviders; /** Providers that were successfully started in {@link #startDiscovery(String, InetSocketAddress)}. */ private final List liveProviders = new CopyOnWriteArrayList<>(); /** * Constructs a new instance. * * @param cfg Configuration. */ public SeedNodeProviderGroup(SeedNodeProviderGroupConfig cfg) { ArgAssert.notNull(cfg, "Configuration"); this.policy = cfg.getPolicy(); this.allProviders = unmodifiableList(nullSafe(cfg.getProviders()).collect(toList())); ConfigCheck check = ConfigCheck.get(SeedNodeProviderGroupConfig.class); check.notNull(policy, "policy"); check.isFalse(allProviders.isEmpty(), "providers can't be empty."); } @Override public void report(ConfigReporter report) { report.section("group", r -> { r.value("policy", policy); r.section("providers", pr -> allProviders.forEach(p -> pr.value("provider", p) ) ); }); } /** * Returns an immutable list of all providers that are configured for this group. * * @return Immutable list of all providers of this group. * * @see SeedNodeProviderGroupConfig#setProviders(List) */ public List allProviders() { return allProviders; } /** * Returns a copy of a list of all providers that were successfully started. * * @return Copy of a list of all providers that were successfully started. * * @see SeedNodeProvider#startDiscovery(String, InetSocketAddress) */ public List liveProviders() { return new ArrayList<>(liveProviders); } /** * Searches for a seed node provider of the specified type. * * @param type Type of a seed node provider. * @param Type of a seed node provider. * * @return Seed node provider. */ public Optional findProvider(Class type) { return allProviders.stream() .filter(it -> type.isAssignableFrom(it.getClass())) .map(type::cast) .findFirst(); } /** * Returns the error handling policy of this group. * * @return Error handling policy of this group. * * @see SeedNodeProviderGroupConfig#setPolicy(SeedNodeProviderGroupPolicy) */ public SeedNodeProviderGroupPolicy policy() { return policy; } @Override public long cleanupInterval() { return allProviders.stream() .mapToLong(SeedNodeProvider::cleanupInterval) .filter(i -> i > 0) .min() .orElse(0); } @Override public List findSeedNodes(String namespace) throws HekateException { List seedNodes = new ArrayList<>(); withPolicy("find seed nodes", liveProviders, provider -> seedNodes.addAll(provider.findSeedNodes(namespace)) ); return seedNodes; } @Override public void startDiscovery(String namespace, InetSocketAddress node) throws HekateException { try { withPolicy("start discovery", allProviders, provider -> { try { provider.startDiscovery(namespace, node); // Add to live provider only if started successfully. liveProviders.add(provider); } catch (Throwable e) { // Cleanup this provider if it failed to start. failSafeStop(provider, namespace, node); throw e; } }); } catch (Throwable e) { // Cleanup partially started providers. stopDiscovery(namespace, node); throw e; } } @Override public void suspendDiscovery() throws HekateException { liveProviders.forEach(provider -> { try { provider.suspendDiscovery(); } catch (Throwable e) { if (log.isWarnEnabled()) { log.warn("Failed to suspend discovery [provider={}]", provider, e); } } }); } @Override public void stopDiscovery(String namespace, InetSocketAddress node) throws HekateException { try { for (SeedNodeProvider provider : liveProviders) { failSafeStop(provider, namespace, node); } } finally { // Cleanup live providers. liveProviders.clear(); } } @Override public void registerRemote(String namespace, InetSocketAddress node) throws HekateException { withPolicy("register a remote seed node", liveProviders, provider -> provider.registerRemote(namespace, node) ); } @Override public void unregisterRemote(String namespace, InetSocketAddress node) throws HekateException { withPolicy("unregister a remote seed node", liveProviders, provider -> provider.unregisterRemote(namespace, node) ); } @Override public Collection jmx() { return allProviders; } private void withPolicy(String taskName, List providers, SeedNodeProviderTask task) throws HekateException { if (!providers.isEmpty()) { int success = 0; for (SeedNodeProvider provider : providers) { try { task.execute(provider); success++; } catch (Exception e) { if (policy == SeedNodeProviderGroupPolicy.FAIL_ON_FIRST_ERROR) { throw e; } else { if (log.isWarnEnabled()) { log.warn("Failed to {}.", taskName, e); } } } } if (success == 0) { throw new HekateException("All seed node providers failed to " + taskName + '.'); } } } private void failSafeStop(SeedNodeProvider provider, String cluster, InetSocketAddress node) { try { provider.stopDiscovery(cluster, node); } catch (Throwable t) { if (log.isWarnEnabled()) { log.warn("Failed to stop discovery [provider={}]", provider, t); } } } @Override public String toString() { return ToString.format(this); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy