Retrieving config-related info (version, features) on stats update
This commit is contained in:
parent
00dcff6678
commit
41020e17b6
3 changed files with 90 additions and 82 deletions
|
@ -4,16 +4,13 @@ import com.provectus.kafka.ui.model.ClusterFeature;
|
||||||
import com.provectus.kafka.ui.model.KafkaCluster;
|
import com.provectus.kafka.ui.model.KafkaCluster;
|
||||||
import com.provectus.kafka.ui.service.ReactiveAdminClient.ClusterDescription;
|
import com.provectus.kafka.ui.service.ReactiveAdminClient.ClusterDescription;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collection;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.function.Predicate;
|
import java.util.function.Predicate;
|
||||||
import javax.annotation.Nullable;
|
|
||||||
import lombok.RequiredArgsConstructor;
|
import lombok.RequiredArgsConstructor;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
import org.apache.kafka.common.Node;
|
|
||||||
import org.apache.kafka.common.acl.AclOperation;
|
import org.apache.kafka.common.acl.AclOperation;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
import reactor.core.publisher.Flux;
|
import reactor.core.publisher.Flux;
|
||||||
|
@ -24,11 +21,10 @@ import reactor.core.publisher.Mono;
|
||||||
@Slf4j
|
@Slf4j
|
||||||
public class FeatureService {
|
public class FeatureService {
|
||||||
|
|
||||||
private static final String DELETE_TOPIC_ENABLED_SERVER_PROPERTY = "delete.topic.enable";
|
|
||||||
|
|
||||||
private final AdminClientService adminClientService;
|
private final AdminClientService adminClientService;
|
||||||
|
|
||||||
public Mono<List<ClusterFeature>> getAvailableFeatures(KafkaCluster cluster,
|
public Mono<List<ClusterFeature>> getAvailableFeatures(ReactiveAdminClient adminClient,
|
||||||
|
KafkaCluster cluster,
|
||||||
ClusterDescription clusterDescription) {
|
ClusterDescription clusterDescription) {
|
||||||
List<Mono<ClusterFeature>> features = new ArrayList<>();
|
List<Mono<ClusterFeature>> features = new ArrayList<>();
|
||||||
|
|
||||||
|
@ -46,29 +42,17 @@ public class FeatureService {
|
||||||
features.add(Mono.just(ClusterFeature.SCHEMA_REGISTRY));
|
features.add(Mono.just(ClusterFeature.SCHEMA_REGISTRY));
|
||||||
}
|
}
|
||||||
|
|
||||||
features.add(topicDeletionEnabled(cluster, clusterDescription.getController()));
|
features.add(topicDeletionEnabled(adminClient));
|
||||||
features.add(aclView(cluster));
|
features.add(aclView(cluster));
|
||||||
features.add(aclEdit(clusterDescription));
|
features.add(aclEdit(clusterDescription));
|
||||||
|
|
||||||
return Flux.fromIterable(features).flatMap(m -> m).collectList();
|
return Flux.fromIterable(features).flatMap(m -> m).collectList();
|
||||||
}
|
}
|
||||||
|
|
||||||
private Mono<ClusterFeature> topicDeletionEnabled(KafkaCluster cluster, @Nullable Node controller) {
|
private Mono<ClusterFeature> topicDeletionEnabled(ReactiveAdminClient adminClient) {
|
||||||
if (controller == null) {
|
return adminClient.isTopicDeletionEnabled()
|
||||||
return Mono.just(ClusterFeature.TOPIC_DELETION); // assuming it is enabled by default
|
? Mono.just(ClusterFeature.TOPIC_DELETION)
|
||||||
}
|
: Mono.empty();
|
||||||
return adminClientService.get(cluster)
|
|
||||||
.flatMap(ac -> ac.loadBrokersConfig(List.of(controller.id())))
|
|
||||||
.map(config ->
|
|
||||||
config.values().stream()
|
|
||||||
.flatMap(Collection::stream)
|
|
||||||
.filter(e -> e.name().equals(DELETE_TOPIC_ENABLED_SERVER_PROPERTY))
|
|
||||||
.map(e -> Boolean.parseBoolean(e.value()))
|
|
||||||
.findFirst()
|
|
||||||
.orElse(true))
|
|
||||||
.flatMap(enabled -> enabled
|
|
||||||
? Mono.just(ClusterFeature.TOPIC_DELETION)
|
|
||||||
: Mono.empty());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private Mono<ClusterFeature> aclEdit(ClusterDescription clusterDescription) {
|
private Mono<ClusterFeature> aclEdit(ClusterDescription clusterDescription) {
|
||||||
|
|
|
@ -32,8 +32,9 @@ import java.util.stream.Collectors;
|
||||||
import java.util.stream.Stream;
|
import java.util.stream.Stream;
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
import lombok.AccessLevel;
|
import lombok.AccessLevel;
|
||||||
|
import lombok.AllArgsConstructor;
|
||||||
|
import lombok.Builder;
|
||||||
import lombok.Getter;
|
import lombok.Getter;
|
||||||
import lombok.RequiredArgsConstructor;
|
|
||||||
import lombok.Value;
|
import lombok.Value;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
import org.apache.kafka.clients.admin.AdminClient;
|
import org.apache.kafka.clients.admin.AdminClient;
|
||||||
|
@ -75,7 +76,6 @@ import org.apache.kafka.common.errors.TopicAuthorizationException;
|
||||||
import org.apache.kafka.common.errors.UnknownTopicOrPartitionException;
|
import org.apache.kafka.common.errors.UnknownTopicOrPartitionException;
|
||||||
import org.apache.kafka.common.errors.UnsupportedVersionException;
|
import org.apache.kafka.common.errors.UnsupportedVersionException;
|
||||||
import org.apache.kafka.common.requests.DescribeLogDirsResponse;
|
import org.apache.kafka.common.requests.DescribeLogDirsResponse;
|
||||||
import org.apache.kafka.common.resource.ResourcePattern;
|
|
||||||
import org.apache.kafka.common.resource.ResourcePatternFilter;
|
import org.apache.kafka.common.resource.ResourcePatternFilter;
|
||||||
import reactor.core.publisher.Flux;
|
import reactor.core.publisher.Flux;
|
||||||
import reactor.core.publisher.Mono;
|
import reactor.core.publisher.Mono;
|
||||||
|
@ -85,7 +85,7 @@ import reactor.util.function.Tuples;
|
||||||
|
|
||||||
|
|
||||||
@Slf4j
|
@Slf4j
|
||||||
@RequiredArgsConstructor
|
@AllArgsConstructor
|
||||||
public class ReactiveAdminClient implements Closeable {
|
public class ReactiveAdminClient implements Closeable {
|
||||||
|
|
||||||
public enum SupportedFeature {
|
public enum SupportedFeature {
|
||||||
|
@ -104,7 +104,8 @@ public class ReactiveAdminClient implements Closeable {
|
||||||
this.predicate = (admin, ver) -> Mono.just(ver != null && ver >= fromVersion);
|
this.predicate = (admin, ver) -> Mono.just(ver != null && ver >= fromVersion);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Mono<Set<SupportedFeature>> forVersion(AdminClient ac, @Nullable Float kafkaVersion) {
|
static Mono<Set<SupportedFeature>> forVersion(AdminClient ac, String kafkaVersionStr) {
|
||||||
|
@Nullable Float kafkaVersion = KafkaVersion.parse(kafkaVersionStr).orElse(null);
|
||||||
return Flux.fromArray(SupportedFeature.values())
|
return Flux.fromArray(SupportedFeature.values())
|
||||||
.flatMap(f -> f.predicate.apply(ac, kafkaVersion).map(enabled -> Tuples.of(f, enabled)))
|
.flatMap(f -> f.predicate.apply(ac, kafkaVersion).map(enabled -> Tuples.of(f, enabled)))
|
||||||
.filter(Tuple2::getT2)
|
.filter(Tuple2::getT2)
|
||||||
|
@ -123,19 +124,46 @@ public class ReactiveAdminClient implements Closeable {
|
||||||
Set<AclOperation> authorizedOperations;
|
Set<AclOperation> authorizedOperations;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Mono<ReactiveAdminClient> create(AdminClient adminClient) {
|
@Builder
|
||||||
return getClusterVersion(adminClient)
|
private record ConfigRelatedInfo(String version,
|
||||||
.flatMap(ver ->
|
Set<SupportedFeature> features,
|
||||||
getSupportedUpdateFeaturesForVersion(adminClient, ver)
|
boolean topicDeletionIsAllowed) {
|
||||||
.map(features ->
|
|
||||||
new ReactiveAdminClient(adminClient, ver, features)));
|
private static Mono<ConfigRelatedInfo> extract(AdminClient ac, int controllerId) {
|
||||||
|
return loadBrokersConfig(ac, List.of(controllerId))
|
||||||
|
.map(map -> map.isEmpty() ? List.<ConfigEntry>of() : map.get(controllerId))
|
||||||
|
.flatMap(configs -> {
|
||||||
|
String version = "1.0-UNKNOWN";
|
||||||
|
boolean topicDeletionEnabled = true;
|
||||||
|
for (ConfigEntry entry : configs) {
|
||||||
|
if (entry.name().contains("inter.broker.protocol.version")) {
|
||||||
|
version = entry.value();
|
||||||
|
}
|
||||||
|
if (entry.name().equals("delete.topic.enable")) {
|
||||||
|
topicDeletionEnabled = Boolean.parseBoolean(entry.value());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var builder = ConfigRelatedInfo.builder()
|
||||||
|
.version(version)
|
||||||
|
.topicDeletionIsAllowed(topicDeletionEnabled);
|
||||||
|
return SupportedFeature.forVersion(ac, version)
|
||||||
|
.map(features -> builder.features(features).build());
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Mono<Set<SupportedFeature>> getSupportedUpdateFeaturesForVersion(AdminClient ac, String versionStr) {
|
public static Mono<ReactiveAdminClient> create(AdminClient adminClient) {
|
||||||
@Nullable Float kafkaVersion = KafkaVersion.parse(versionStr).orElse(null);
|
return describeClusterImpl(adminClient, Set.of())
|
||||||
return SupportedFeature.forVersion(ac, kafkaVersion);
|
// choosing node from which we will get configs (starting with controller)
|
||||||
|
.flatMap(descr -> descr.controller != null
|
||||||
|
? Mono.just(descr.controller)
|
||||||
|
: Mono.justOrEmpty(descr.nodes.stream().findFirst())
|
||||||
|
)
|
||||||
|
.flatMap(node -> ConfigRelatedInfo.extract(adminClient, node.id()))
|
||||||
|
.map(info -> new ReactiveAdminClient(adminClient, info));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private static Mono<Boolean> isAuthorizedSecurityEnabled(AdminClient ac, @Nullable Float kafkaVersion) {
|
private static Mono<Boolean> isAuthorizedSecurityEnabled(AdminClient ac, @Nullable Float kafkaVersion) {
|
||||||
return toMono(ac.describeAcls(AclBindingFilter.ANY).values())
|
return toMono(ac.describeAcls(AclBindingFilter.ANY).values())
|
||||||
.thenReturn(true)
|
.thenReturn(true)
|
||||||
|
@ -174,11 +202,10 @@ public class ReactiveAdminClient implements Closeable {
|
||||||
|
|
||||||
@Getter(AccessLevel.PACKAGE) // visible for testing
|
@Getter(AccessLevel.PACKAGE) // visible for testing
|
||||||
private final AdminClient client;
|
private final AdminClient client;
|
||||||
private final String version;
|
private volatile ConfigRelatedInfo configRelatedInfo;
|
||||||
private final Set<SupportedFeature> features;
|
|
||||||
|
|
||||||
public Set<SupportedFeature> getClusterFeatures() {
|
public Set<SupportedFeature> getClusterFeatures() {
|
||||||
return features;
|
return configRelatedInfo.features();
|
||||||
}
|
}
|
||||||
|
|
||||||
public Mono<Set<String>> listTopics(boolean listInternal) {
|
public Mono<Set<String>> listTopics(boolean listInternal) {
|
||||||
|
@ -190,7 +217,20 @@ public class ReactiveAdminClient implements Closeable {
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getVersion() {
|
public String getVersion() {
|
||||||
return version;
|
return configRelatedInfo.version();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isTopicDeletionEnabled() {
|
||||||
|
return configRelatedInfo.topicDeletionIsAllowed();
|
||||||
|
}
|
||||||
|
|
||||||
|
public Mono<Void> updateInternalStats(@Nullable Node controller) {
|
||||||
|
if (controller == null) {
|
||||||
|
return Mono.empty();
|
||||||
|
}
|
||||||
|
return ConfigRelatedInfo.extract(client, controller.id())
|
||||||
|
.doOnNext(info -> this.configRelatedInfo = info)
|
||||||
|
.then();
|
||||||
}
|
}
|
||||||
|
|
||||||
public Mono<Map<String, List<ConfigEntry>>> getTopicsConfig() {
|
public Mono<Map<String, List<ConfigEntry>>> getTopicsConfig() {
|
||||||
|
@ -200,7 +240,7 @@ public class ReactiveAdminClient implements Closeable {
|
||||||
//NOTE: skips not-found topics (for which UnknownTopicOrPartitionException was thrown by AdminClient)
|
//NOTE: skips not-found topics (for which UnknownTopicOrPartitionException was thrown by AdminClient)
|
||||||
//and topics for which DESCRIBE_CONFIGS permission is not set (TopicAuthorizationException was thrown)
|
//and topics for which DESCRIBE_CONFIGS permission is not set (TopicAuthorizationException was thrown)
|
||||||
public Mono<Map<String, List<ConfigEntry>>> getTopicsConfig(Collection<String> topicNames, boolean includeDoc) {
|
public Mono<Map<String, List<ConfigEntry>>> getTopicsConfig(Collection<String> topicNames, boolean includeDoc) {
|
||||||
var includeDocFixed = features.contains(SupportedFeature.CONFIG_DOCUMENTATION_RETRIEVAL) && includeDoc;
|
var includeDocFixed = includeDoc && getClusterFeatures().contains(SupportedFeature.CONFIG_DOCUMENTATION_RETRIEVAL);
|
||||||
// we need to partition calls, because it can lead to AdminClient timeouts in case of large topics count
|
// we need to partition calls, because it can lead to AdminClient timeouts in case of large topics count
|
||||||
return partitionCalls(
|
return partitionCalls(
|
||||||
topicNames,
|
topicNames,
|
||||||
|
@ -349,7 +389,7 @@ public class ReactiveAdminClient implements Closeable {
|
||||||
}
|
}
|
||||||
|
|
||||||
public Mono<ClusterDescription> describeCluster() {
|
public Mono<ClusterDescription> describeCluster() {
|
||||||
return describeClusterImpl(client, features);
|
return describeClusterImpl(client, getClusterFeatures());
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Mono<ClusterDescription> describeClusterImpl(AdminClient client, Set<SupportedFeature> features) {
|
private static Mono<ClusterDescription> describeClusterImpl(AdminClient client, Set<SupportedFeature> features) {
|
||||||
|
@ -371,23 +411,6 @@ public class ReactiveAdminClient implements Closeable {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Mono<String> getClusterVersion(AdminClient client) {
|
|
||||||
return describeClusterImpl(client, Set.of())
|
|
||||||
// choosing node from which we will get configs (starting with controller)
|
|
||||||
.flatMap(descr -> descr.controller != null
|
|
||||||
? Mono.just(descr.controller)
|
|
||||||
: Mono.justOrEmpty(descr.nodes.stream().findFirst())
|
|
||||||
)
|
|
||||||
.flatMap(node -> loadBrokersConfig(client, List.of(node.id())))
|
|
||||||
.flatMap(configs -> configs.values().stream()
|
|
||||||
.flatMap(Collection::stream)
|
|
||||||
.filter(entry -> entry.name().contains("inter.broker.protocol.version"))
|
|
||||||
.findFirst()
|
|
||||||
.map(configEntry -> Mono.just(configEntry.value()))
|
|
||||||
.orElse(Mono.empty()))
|
|
||||||
.switchIfEmpty(Mono.just("1.0-UNKNOWN"));
|
|
||||||
}
|
|
||||||
|
|
||||||
public Mono<Void> deleteConsumerGroups(Collection<String> groupIds) {
|
public Mono<Void> deleteConsumerGroups(Collection<String> groupIds) {
|
||||||
return toMono(client.deleteConsumerGroups(groupIds).all())
|
return toMono(client.deleteConsumerGroups(groupIds).all())
|
||||||
.onErrorResume(GroupIdNotFoundException.class,
|
.onErrorResume(GroupIdNotFoundException.class,
|
||||||
|
@ -421,7 +444,7 @@ public class ReactiveAdminClient implements Closeable {
|
||||||
// NOTE: places whole current topic config with new one. Entries that were present in old config,
|
// NOTE: places whole current topic config with new one. Entries that were present in old config,
|
||||||
// but missed in new will be set to default
|
// but missed in new will be set to default
|
||||||
public Mono<Void> updateTopicConfig(String topicName, Map<String, String> configs) {
|
public Mono<Void> updateTopicConfig(String topicName, Map<String, String> configs) {
|
||||||
if (features.contains(SupportedFeature.INCREMENTAL_ALTER_CONFIGS)) {
|
if (getClusterFeatures().contains(SupportedFeature.INCREMENTAL_ALTER_CONFIGS)) {
|
||||||
return getTopicsConfigImpl(List.of(topicName), false)
|
return getTopicsConfigImpl(List.of(topicName), false)
|
||||||
.map(conf -> conf.getOrDefault(topicName, List.of()))
|
.map(conf -> conf.getOrDefault(topicName, List.of()))
|
||||||
.flatMap(currentConfigs -> incrementalAlterConfig(topicName, currentConfigs, configs));
|
.flatMap(currentConfigs -> incrementalAlterConfig(topicName, currentConfigs, configs));
|
||||||
|
@ -596,17 +619,17 @@ public class ReactiveAdminClient implements Closeable {
|
||||||
}
|
}
|
||||||
|
|
||||||
public Mono<Collection<AclBinding>> listAcls(ResourcePatternFilter filter) {
|
public Mono<Collection<AclBinding>> listAcls(ResourcePatternFilter filter) {
|
||||||
Preconditions.checkArgument(features.contains(SupportedFeature.AUTHORIZED_SECURITY_ENABLED));
|
Preconditions.checkArgument(getClusterFeatures().contains(SupportedFeature.AUTHORIZED_SECURITY_ENABLED));
|
||||||
return toMono(client.describeAcls(new AclBindingFilter(filter, AccessControlEntryFilter.ANY)).values());
|
return toMono(client.describeAcls(new AclBindingFilter(filter, AccessControlEntryFilter.ANY)).values());
|
||||||
}
|
}
|
||||||
|
|
||||||
public Mono<Void> createAcls(Collection<AclBinding> aclBindings) {
|
public Mono<Void> createAcls(Collection<AclBinding> aclBindings) {
|
||||||
Preconditions.checkArgument(features.contains(SupportedFeature.AUTHORIZED_SECURITY_ENABLED));
|
Preconditions.checkArgument(getClusterFeatures().contains(SupportedFeature.AUTHORIZED_SECURITY_ENABLED));
|
||||||
return toMono(client.createAcls(aclBindings).all());
|
return toMono(client.createAcls(aclBindings).all());
|
||||||
}
|
}
|
||||||
|
|
||||||
public Mono<Void> deleteAcls(Collection<AclBinding> aclBindings) {
|
public Mono<Void> deleteAcls(Collection<AclBinding> aclBindings) {
|
||||||
Preconditions.checkArgument(features.contains(SupportedFeature.AUTHORIZED_SECURITY_ENABLED));
|
Preconditions.checkArgument(getClusterFeatures().contains(SupportedFeature.AUTHORIZED_SECURITY_ENABLED));
|
||||||
var filters = aclBindings.stream().map(AclBinding::toFilter).collect(Collectors.toSet());
|
var filters = aclBindings.stream().map(AclBinding::toFilter).collect(Collectors.toSet());
|
||||||
return toMono(client.deleteAcls(filters).all()).then();
|
return toMono(client.deleteAcls(filters).all()).then();
|
||||||
}
|
}
|
||||||
|
|
|
@ -37,25 +37,26 @@ public class StatisticsService {
|
||||||
private Mono<Statistics> getStatistics(KafkaCluster cluster) {
|
private Mono<Statistics> getStatistics(KafkaCluster cluster) {
|
||||||
return adminClientService.get(cluster).flatMap(ac ->
|
return adminClientService.get(cluster).flatMap(ac ->
|
||||||
ac.describeCluster().flatMap(description ->
|
ac.describeCluster().flatMap(description ->
|
||||||
Mono.zip(
|
ac.updateInternalStats(description.getController()).then(
|
||||||
List.of(
|
Mono.zip(
|
||||||
metricsCollector.getBrokerMetrics(cluster, description.getNodes()),
|
List.of(
|
||||||
getLogDirInfo(description, ac),
|
metricsCollector.getBrokerMetrics(cluster, description.getNodes()),
|
||||||
featureService.getAvailableFeatures(cluster, description),
|
getLogDirInfo(description, ac),
|
||||||
loadTopicConfigs(cluster),
|
featureService.getAvailableFeatures(ac, cluster, description),
|
||||||
describeTopics(cluster)),
|
loadTopicConfigs(cluster),
|
||||||
results ->
|
describeTopics(cluster)),
|
||||||
Statistics.builder()
|
results ->
|
||||||
.status(ServerStatusDTO.ONLINE)
|
Statistics.builder()
|
||||||
.clusterDescription(description)
|
.status(ServerStatusDTO.ONLINE)
|
||||||
.version(ac.getVersion())
|
.clusterDescription(description)
|
||||||
.metrics((Metrics) results[0])
|
.version(ac.getVersion())
|
||||||
.logDirInfo((InternalLogDirStats) results[1])
|
.metrics((Metrics) results[0])
|
||||||
.features((List<ClusterFeature>) results[2])
|
.logDirInfo((InternalLogDirStats) results[1])
|
||||||
.topicConfigs((Map<String, List<ConfigEntry>>) results[3])
|
.features((List<ClusterFeature>) results[2])
|
||||||
.topicDescriptions((Map<String, TopicDescription>) results[4])
|
.topicConfigs((Map<String, List<ConfigEntry>>) results[3])
|
||||||
.build()
|
.topicDescriptions((Map<String, TopicDescription>) results[4])
|
||||||
)))
|
.build()
|
||||||
|
))))
|
||||||
.doOnError(e ->
|
.doOnError(e ->
|
||||||
log.error("Failed to collect cluster {} info", cluster.getName(), e))
|
log.error("Failed to collect cluster {} info", cluster.getName(), e))
|
||||||
.onErrorResume(
|
.onErrorResume(
|
||||||
|
|
Loading…
Add table
Reference in a new issue