io.micronaut.management.health.aggregator.DefaultHealthAggregator Maven / Gradle / Ivy
/*
* Copyright 2017-2020 original 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
*
* https://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.micronaut.management.health.aggregator;
import io.micronaut.context.annotation.Requires;
import io.micronaut.context.env.Environment;
import io.micronaut.core.util.CollectionUtils;
import io.micronaut.health.HealthStatus;
import io.micronaut.management.endpoint.health.HealthEndpoint;
import io.micronaut.management.endpoint.health.HealthLevelOfDetail;
import io.micronaut.management.health.indicator.HealthIndicator;
import io.micronaut.management.health.indicator.HealthResult;
import io.micronaut.runtime.ApplicationConfiguration;
import jakarta.inject.Singleton;
import org.reactivestreams.Publisher;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;
/**
* Default implementation of {@link HealthAggregator} that creates a {status: , description: (optional) , details: }
* response. The top level object represents the most severe status found in the provided health results, or
* {@link HealthStatus#UNKNOWN} if none found. All registered indicators have their own
* {status: , description: (optional) , details: } object, keyed by the name of the {@link HealthResult} defined inside
* the details of the top level object.
*
* Example:
* [status: "UP, details: [diskSpace: [status: UP, details: [:]], cpuUsage: ...]]
*
* @author James Kleeh
* @author Graeme Rocher
* @since 1.0
*/
@Singleton
@Requires(beans = HealthEndpoint.class)
public class DefaultHealthAggregator implements HealthAggregator {
private static final Logger LOG = LoggerFactory.getLogger(DefaultHealthAggregator.class);
private final ApplicationConfiguration applicationConfiguration;
/**
* Default constructor.
*
* @param applicationConfiguration The application configuration.
*/
public DefaultHealthAggregator(ApplicationConfiguration applicationConfiguration) {
this.applicationConfiguration = applicationConfiguration;
}
@Override
public Publisher aggregate(HealthIndicator[] indicators, HealthLevelOfDetail healthLevelOfDetail) {
Flux results = aggregateResults(indicators);
Mono result = results.collectList().map(list -> {
HealthStatus overallStatus = calculateOverallStatus(list);
return buildResult(overallStatus, aggregateDetails(list), healthLevelOfDetail);
});
return result.flux();
}
@Override
public Publisher aggregate(String name, Publisher results) {
Mono result = Flux.from(results).collectList().map(list -> {
HealthStatus overallStatus = calculateOverallStatus(list);
Object details = aggregateDetails(list);
return HealthResult.builder(name, overallStatus).details(details).build();
});
return result.flux();
}
/**
* @param results A list of {@link HealthResult}
* @return The calculated overall health status
*/
protected HealthStatus calculateOverallStatus(List results) {
return results.stream()
.map(HealthResult::getStatus)
.sorted()
.distinct()
.reduce((a, b) -> b)
.orElse(HealthStatus.UNKNOWN);
}
/**
* @param indicators An array of {@link HealthIndicator}
* @return The aggregated results from all health indicators
*/
protected Flux aggregateResults(HealthIndicator[] indicators) {
return Flux.merge(
Arrays.stream(indicators)
.map(HealthIndicator::getResult)
.collect(Collectors.toList())
);
}
/**
* @param results A list of health results
* @return The aggregated details for the results
*/
protected Object aggregateDetails(List results) {
Map aggregatedDetails = CollectionUtils.newHashMap(results.size());
results.forEach(r -> {
var name = r.getName();
var details = r.getDetails();
var status = r.getStatus();
aggregatedDetails.put(name, buildResult(status, details, HealthLevelOfDetail.STATUS_DESCRIPTION_DETAILS));
if (LOG.isTraceEnabled()) {
LOG.trace("Health result for {}: status {}, details {}", name, status, details != null ? details : "{}");
} else if (LOG.isDebugEnabled()) {
LOG.debug("Health result for {}: status {}", name, status);
}
});
return aggregatedDetails;
}
/**
* @param status A {@link HealthStatus}
* @param details The health status details
* @param healthLevelOfDetail The {@link HealthLevelOfDetail}
* @return A {@link Map} with the results from the health status
*/
@SuppressWarnings("MagicNumber")
protected HealthResult buildResult(HealthStatus status, Object details, HealthLevelOfDetail healthLevelOfDetail) {
if (healthLevelOfDetail == HealthLevelOfDetail.STATUS) {
return HealthResult.builder(null, status).build();
}
return HealthResult.builder(
applicationConfiguration.getName().orElse(Environment.DEFAULT_NAME),
status
).details(details).build();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy