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

io.micronaut.management.health.aggregator.DefaultHealthAggregator Maven / Gradle / Ivy

The newest version!
/*
 * 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 - 2024 Weber Informatics LLC | Privacy Policy