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

io.kroxylicious.proxy.config.ServiceBasedPluginFactoryRegistry Maven / Gradle / Ivy

The newest version!
/*
 * Copyright Kroxylicious Authors.
 *
 * Licensed under the Apache Software License version 2.0, available at http://www.apache.org/licenses/LICENSE-2.0
 */

package io.kroxylicious.proxy.config;

import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Objects;
import java.util.ServiceLoader;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import io.kroxylicious.proxy.plugin.Plugin;
import io.kroxylicious.proxy.plugin.UnknownPluginInstanceException;

import edu.umd.cs.findbugs.annotations.NonNull;

/**
 * A {@link PluginFactoryRegistry} that is implemented using {@link ServiceLoader} discovery.
 */
public class ServiceBasedPluginFactoryRegistry implements PluginFactoryRegistry {

    private static final Logger LOGGER = LoggerFactory.getLogger(ServiceBasedPluginFactoryRegistry.class);

    public record ProviderAndConfigType(@NonNull ServiceLoader.Provider provider,
                                        @NonNull Class config) {
        public ProviderAndConfigType {
            Objects.requireNonNull(provider);
            Objects.requireNonNull(config);
        }
    }

    private final Map, Map> pluginInterfaceToNameToProvider = new ConcurrentHashMap<>();

    @NonNull
    Map load(@NonNull Class pluginInterface) {
        Objects.requireNonNull(pluginInterface);
        return pluginInterfaceToNameToProvider.computeIfAbsent(pluginInterface,
                i -> loadProviders(pluginInterface));
    }

    private static Map loadProviders(Class pluginInterface) {
        HashMap> nameToProviders = new HashMap<>();
        ServiceLoader load = ServiceLoader.load(pluginInterface);
        load.stream().forEach(provider -> {
            Class providerType = provider.type();
            Plugin annotation = providerType.getAnnotation(Plugin.class);
            if (annotation == null) {
                LOGGER.warn("Failed to find a @PluginConfigType on provider {} of service {}", providerType, pluginInterface);
            }
            else {
                ProviderAndConfigType providerAndConfigType = new ProviderAndConfigType(provider, annotation.configType());
                Stream.of(providerType.getName(), providerType.getSimpleName()).forEach(name2 -> nameToProviders.compute(name2, (k2, v) -> {
                    if (v == null) {
                        v = new HashSet<>();
                    }
                    v.add(providerAndConfigType);
                    return v;
                }));
            }
        });
        var bySingleton = nameToProviders.entrySet().stream().collect(
                Collectors.partitioningBy(e -> e.getValue().size() == 1));
        if (LOGGER.isWarnEnabled()) {
            for (Map.Entry> ambiguousInstanceNameToProviders : bySingleton.get(false)) {
                LOGGER.warn("'{}' would be an ambiguous reference to a {} provider. "
                        + "It could refer to any of {}"
                        + " so to avoid ambiguous behaviour those fully qualified names must be used",
                        ambiguousInstanceNameToProviders.getKey(),
                        pluginInterface.getSimpleName(),
                        ambiguousInstanceNameToProviders.getValue().stream().map(p -> p.provider().type().getName()).collect(Collectors.joining(", ")));
            }
        }
        return bySingleton.get(true).stream().collect(Collectors.toMap(
                Map.Entry::getKey,
                e -> e.getValue().iterator().next()));
    }

    @Override
    public 

@NonNull PluginFactory

pluginFactory(@NonNull Class

pluginClass) { var nameToProvider = load(pluginClass); return new PluginFactory<>() { @Override public @NonNull P pluginInstance(@NonNull String instanceName) { if (Objects.requireNonNull(instanceName).isEmpty()) { throw new IllegalArgumentException(); } var provider = nameToProvider.get(instanceName); if (provider != null) { Class type = provider.provider().type(); if (type.isAnnotationPresent(Deprecated.class)) { LOGGER.warn("{} plugin with id {} is deprecated", pluginClass.getName(), instanceName); } return pluginClass.cast(provider.provider().get()); } throw unknownPluginInstanceException(instanceName); } private UnknownPluginInstanceException unknownPluginInstanceException(String name) { return new UnknownPluginInstanceException("Unknown " + pluginClass.getName() + " plugin instance for name '" + name + "'. " + "Known plugin instances are " + nameToProvider.keySet() + ". " + "Plugins must be loadable by java.util.ServiceLoader and annotated with @" + Plugin.class.getSimpleName() + "."); } @NonNull @Override public Class configType(@NonNull String instanceName) { var providerAndConfigType = nameToProvider.get(instanceName); if (providerAndConfigType != null) { return providerAndConfigType.config(); } throw unknownPluginInstanceException(instanceName); } }; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy