
de.codecentric.boot.admin.server.services.ApplicationRegistry Maven / Gradle / Ivy
/*
* Copyright 2014-2019 the original author or authors.
*
* Licensed 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 de.codecentric.boot.admin.server.services;
import java.time.Instant;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import javax.annotation.Nullable;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.util.function.Tuple2;
import reactor.util.function.Tuples;
import de.codecentric.boot.admin.server.domain.entities.Application;
import de.codecentric.boot.admin.server.domain.entities.Instance;
import de.codecentric.boot.admin.server.domain.values.BuildVersion;
import de.codecentric.boot.admin.server.domain.values.InstanceId;
import de.codecentric.boot.admin.server.domain.values.StatusInfo;
import de.codecentric.boot.admin.server.eventstore.InstanceEventPublisher;
import static de.codecentric.boot.admin.server.domain.values.StatusInfo.STATUS_UNKNOWN;
import static java.util.Comparator.naturalOrder;
import static java.util.stream.Collectors.toList;
import static java.util.stream.Collectors.toMap;
/**
* Registry for all applications that should be managed/administrated by the Spring Boot
* Admin server. Backed by an InstanceRegistry for persistence and an
* InstanceEventPublisher for events
*
* @author Dean de Bree
*/
public class ApplicationRegistry {
private final InstanceRegistry instanceRegistry;
private final InstanceEventPublisher instanceEventPublisher;
public ApplicationRegistry(InstanceRegistry instanceRegistry, InstanceEventPublisher instanceEventPublisher) {
this.instanceRegistry = instanceRegistry;
this.instanceEventPublisher = instanceEventPublisher;
}
/**
* Get a list of all registered applications.
* @return flux of all the applications.
*/
public Flux getApplications() {
return this.instanceRegistry.getInstances().filter(Instance::isRegistered)
.groupBy((instance) -> instance.getRegistration().getName())
.flatMap((grouped) -> toApplication(grouped.key(), grouped), Integer.MAX_VALUE);
}
/**
* Get a specific application instance.
* @param name the name of the application to find.
* @return a Mono with the application or an empty Mono if not found.
*/
public Mono getApplication(String name) {
return this.toApplication(name, this.instanceRegistry.getInstances(name).filter(Instance::isRegistered))
.filter((a) -> !a.getInstances().isEmpty());
}
public Flux getApplicationStream() {
return Flux.from(this.instanceEventPublisher)
.flatMap((event) -> this.instanceRegistry.getInstance(event.getInstance()))
.map(this::getApplicationForInstance).flatMap((group) -> toApplication(group.getT1(), group.getT2()));
}
public Flux deregister(String name) {
return this.instanceRegistry.getInstances(name)
.flatMap((instance) -> this.instanceRegistry.deregister(instance.getId()));
}
protected Tuple2> getApplicationForInstance(Instance instance) {
String name = instance.getRegistration().getName();
return Tuples.of(name, this.instanceRegistry.getInstances(name).filter(Instance::isRegistered));
}
protected Mono toApplication(String name, Flux instances) {
return instances.collectList().map((instanceList) -> {
Tuple2 status = getStatus(instanceList);
return Application.create(name).instances(instanceList).buildVersion(getBuildVersion(instanceList))
.status(status.getT1()).statusTimestamp(status.getT2()).build();
});
}
@Nullable
protected BuildVersion getBuildVersion(List instances) {
List versions = instances.stream().map(Instance::getBuildVersion).filter(Objects::nonNull)
.distinct().sorted().collect(toList());
if (versions.isEmpty()) {
return null;
}
else if (versions.size() == 1) {
return versions.get(0);
}
else {
return BuildVersion.valueOf(versions.get(0) + " ... " + versions.get(versions.size() - 1));
}
}
protected Tuple2 getStatus(List instances) {
// TODO: Correct is just a second readmodel for groups
Map statusWithTime = instances.stream().collect(
toMap((instance) -> instance.getStatusInfo().getStatus(), Instance::getStatusTimestamp, this::getMax));
if (statusWithTime.size() == 1) {
Map.Entry e = statusWithTime.entrySet().iterator().next();
return Tuples.of(e.getKey(), e.getValue());
}
if (statusWithTime.containsKey(StatusInfo.STATUS_UP)) {
Instant oldestNonUp = statusWithTime.entrySet().stream()
.filter((e) -> !StatusInfo.STATUS_UP.equals(e.getKey())).map(Map.Entry::getValue)
.min(naturalOrder()).orElse(Instant.EPOCH);
Instant latest = getMax(oldestNonUp, statusWithTime.getOrDefault(StatusInfo.STATUS_UP, Instant.EPOCH));
return Tuples.of(StatusInfo.STATUS_RESTRICTED, latest);
}
return statusWithTime.entrySet().stream().min(Map.Entry.comparingByKey(StatusInfo.severity()))
.map((e) -> Tuples.of(e.getKey(), e.getValue())).orElse(Tuples.of(STATUS_UNKNOWN, Instant.EPOCH));
}
protected Instant getMax(Instant t1, Instant t2) {
return (t1.compareTo(t2) >= 0) ? t1 : t2;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy