io.kroxylicious.test.tester.DefaultKroxyliciousTester Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kroxylicious-integration-test-support Show documentation
Show all versions of kroxylicious-integration-test-support Show documentation
Support code to simplify writing integration tests for filters
/*
* 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.test.tester;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.IntStream;
import org.apache.kafka.clients.CommonClientConfigs;
import org.apache.kafka.clients.admin.Admin;
import org.apache.kafka.clients.admin.CreateTopicsResult;
import org.apache.kafka.clients.admin.NewTopic;
import org.apache.kafka.clients.consumer.Consumer;
import org.apache.kafka.clients.producer.Producer;
import org.apache.kafka.common.config.SslConfigs;
import org.apache.kafka.common.security.auth.SecurityProtocol;
import org.apache.kafka.common.serialization.Serde;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import io.kroxylicious.proxy.KafkaProxy;
import io.kroxylicious.proxy.config.Configuration;
import io.kroxylicious.proxy.config.ConfigurationBuilder;
import io.kroxylicious.proxy.config.ServiceBasedPluginFactoryRegistry;
import io.kroxylicious.proxy.config.VirtualCluster;
import io.kroxylicious.proxy.config.tls.Tls;
import io.kroxylicious.test.client.KafkaClient;
import edu.umd.cs.findbugs.annotations.NonNull;
import edu.umd.cs.findbugs.annotations.Nullable;
import info.schnatterer.mobynamesgenerator.MobyNamesGenerator;
public class DefaultKroxyliciousTester implements KroxyliciousTester {
private AutoCloseable proxy;
@SuppressWarnings("OptionalUsedAsFieldOrParameterType")
private final Optional trustStoreConfiguration;
private final Configuration kroxyliciousConfig;
private final Map clients;
private final Map> topicsPerVirtualCluster;
private final ClientFactory clientFactory;
private static final Logger LOGGER = LoggerFactory.getLogger(DefaultKroxyliciousTester.class);
DefaultKroxyliciousTester(ConfigurationBuilder configurationBuilder, Function kroxyliciousFactory, ClientFactory clientFactory,
@Nullable KroxyliciousTesterBuilder.TrustStoreConfiguration trustStoreConfiguration) {
this.kroxyliciousConfig = configurationBuilder.build();
this.proxy = kroxyliciousFactory.apply(kroxyliciousConfig);
this.trustStoreConfiguration = Optional.ofNullable(trustStoreConfiguration);
this.clients = new ConcurrentHashMap<>();
this.clientFactory = clientFactory;
topicsPerVirtualCluster = new ConcurrentHashMap<>();
}
private KroxyliciousClients clients() {
return clients(onlyVirtualCluster());
}
private String onlyVirtualCluster() {
int numVirtualClusters = kroxyliciousConfig.virtualClusters().size();
if (numVirtualClusters == 1) {
return kroxyliciousConfig.virtualClusters().keySet().stream().findFirst().orElseThrow();
}
else {
throw new AmbiguousVirtualClusterException(
"no default virtual cluster determined because there were multiple or no virtual clusters in kroxylicious configuration");
}
}
private KroxyliciousClients clients(String virtualCluster) {
return clients.computeIfAbsent(virtualCluster,
k -> clientFactory.build(virtualCluster, buildDefaultClientConfiguration(virtualCluster)));
}
@NonNull
private Map buildDefaultClientConfiguration(String virtualCluster) {
Map defaultClientConfig = new HashMap<>();
defaultClientConfig.put(CommonClientConfigs.BOOTSTRAP_SERVERS_CONFIG, getBootstrapAddress(virtualCluster));
configureClientTls(virtualCluster, defaultClientConfig);
return defaultClientConfig;
}
@Override
@NonNull
public String getBootstrapAddress() {
return getBootstrapAddress(onlyVirtualCluster());
}
@Override
@NonNull
public String getBootstrapAddress(String virtualCluster) {
return KroxyliciousConfigUtils.bootstrapServersFor(virtualCluster, kroxyliciousConfig);
}
private void configureClientTls(String virtualCluster, Map defaultClientConfig) {
final VirtualCluster definedCluster = kroxyliciousConfig.virtualClusters().get(virtualCluster);
if (definedCluster != null) {
final Optional tls = definedCluster.tls();
if (tls.isPresent()) {
defaultClientConfig.put(CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, SecurityProtocol.SSL.name);
if (trustStoreConfiguration.isPresent()) {
final KroxyliciousTesterBuilder.TrustStoreConfiguration storeConfiguration = trustStoreConfiguration.get();
defaultClientConfig.put(SslConfigs.SSL_TRUSTSTORE_LOCATION_CONFIG, storeConfiguration.trustStoreLocation());
defaultClientConfig.put(SslConfigs.SSL_TRUSTSTORE_PASSWORD_CONFIG, storeConfiguration.trustStorePassword());
}
}
}
}
@Override
public Admin admin(Map additionalConfig) {
return clients().admin(additionalConfig);
}
@Override
public Admin admin() {
return clients().admin();
}
@Override
public Producer producer(Map additionalConfig) {
return clients().producer(additionalConfig);
}
@Override
public Producer producer() {
return clients().producer();
}
@Override
public Producer producer(Serde keySerde, Serde valueSerde, Map additionalConfig) {
return clients().producer(keySerde, valueSerde, additionalConfig);
}
@Override
public Consumer consumer(Map additionalConfig) {
return clients().consumer(additionalConfig);
}
@Override
public Consumer consumer() {
return clients().consumer();
}
@Override
public Consumer consumer(Serde keySerde, Serde valueSerde, Map additionalConfig) {
return clients().consumer(keySerde, valueSerde, additionalConfig);
}
@Override
public KafkaClient simpleTestClient() {
return clients().simpleTestClient();
}
@Override
public KafkaClient simpleTestClient(String virtualCluster) {
return clients(virtualCluster).simpleTestClient();
}
@Override
public Admin admin(String virtualCluster, Map additionalConfig) {
return clients(virtualCluster).admin(additionalConfig);
}
@Override
public Admin admin(String virtualCluster) {
return clients(virtualCluster).admin();
}
@Override
public Producer producer(String virtualCluster, Map additionalConfig) {
return clients(virtualCluster).producer(additionalConfig);
}
@Override
public Producer producer(String virtualCluster) {
return clients(virtualCluster).producer();
}
@Override
public Producer producer(String virtualCluster, Serde keySerde, Serde valueSerde, Map additionalConfig) {
return clients(virtualCluster).producer(keySerde, valueSerde, additionalConfig);
}
@Override
public Consumer consumer(String virtualCluster, Map additionalConfig) {
return clients(virtualCluster).consumer(additionalConfig);
}
@Override
public Consumer consumer(String virtualCluster) {
return clients(virtualCluster).consumer();
}
@Override
public Consumer consumer(String virtualCluster, Serde keySerde, Serde valueSerde, Map additionalConfig) {
return clients(virtualCluster).consumer(keySerde, valueSerde, additionalConfig);
}
public void restartProxy() {
try {
proxy.close();
proxy = spawnProxy(kroxyliciousConfig);
}
catch (Exception e) {
throw new IllegalStateException(e);
}
}
@Override
public void close() {
try {
List exceptions = new ArrayList<>();
for (KroxyliciousClients c : clients.values()) {
closeClient(c).ifPresent(exceptions::add);
}
proxy.close();
if (!exceptions.isEmpty()) {
// if we encountered any exceptions while closing, log them all and then throw whichever one came first.
exceptions.forEach(e -> LOGGER.error(e.getMessage(), e));
throw exceptions.get(0);
}
}
catch (Exception e) {
throw new IllegalStateException(e);
}
}
private static Optional closeClient(KroxyliciousClients c) {
try {
c.close();
return Optional.empty();
}
catch (Exception e) {
return Optional.of(e);
}
}
static KafkaProxy spawnProxy(Configuration config) {
KafkaProxy kafkaProxy = new KafkaProxy(new ServiceBasedPluginFactoryRegistry(), config);
try {
kafkaProxy.startup();
}
catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
return kafkaProxy;
}
@Override
public Set createTopics(String virtualCluster, int numberOfTopics) {
try (Admin admin = clients(virtualCluster).admin()) {
final List newTopics = IntStream.range(0, numberOfTopics).mapToObj(ignored -> {
final String topicName = MobyNamesGenerator.getRandomName();
return new NewTopic(topicName, (short) 1, (short) 1);
}).toList();
final CreateTopicsResult createTopicsResult = admin.createTopics(newTopics);
createTopicsResult.all().get(30, TimeUnit.SECONDS);
// TODO should this be driven by result of the createTopic call?
final Set topicNames = newTopics.stream().map(NewTopic::name).collect(Collectors.toSet());
topicsForVirtualCluster(virtualCluster).addAll(topicNames);
return topicNames;
}
catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new IllegalStateException("Failed to create topics on " + virtualCluster, e);
}
catch (TimeoutException e) {
throw new IllegalStateException("Timed out creating topics on " + virtualCluster, e);
}
catch (ExecutionException e) {
throw new IllegalStateException("Failed to create topics on " + virtualCluster, e.getCause());
}
}
@Override
public void deleteTopics(String virtualCluster) {
try (Admin admin = clients(virtualCluster).admin()) {
final Set topics = topicsForVirtualCluster(virtualCluster);
if (!topics.isEmpty()) {
admin.deleteTopics(topics)
.all()
.get(30, TimeUnit.SECONDS);
}
}
catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new IllegalStateException("Failed to delete topics on " + virtualCluster, e);
}
catch (ExecutionException e) {
throw new IllegalStateException("Failed to delete topics on " + virtualCluster, e.getCause());
}
catch (TimeoutException e) {
throw new IllegalStateException("Timed out deleting topics on " + virtualCluster, e);
}
}
private Set topicsForVirtualCluster(String clusterName) {
return topicsPerVirtualCluster.computeIfAbsent(clusterName, key -> ConcurrentHashMap.newKeySet());
}
@FunctionalInterface
interface ClientFactory {
KroxyliciousClients build(String clusterName, Map defaultClientConfig);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy