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

io.gravitee.node.monitoring.DefaultProbeEvaluator Maven / Gradle / Ivy

package io.gravitee.node.monitoring;

import io.gravitee.node.api.healthcheck.Probe;
import io.gravitee.node.api.healthcheck.ProbeEvaluator;
import io.gravitee.node.api.healthcheck.ProbeManager;
import io.gravitee.node.api.healthcheck.Result;
import java.time.Instant;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;
import lombok.RequiredArgsConstructor;

/**
 * @author Jeoffrey HAEYAERT (jeoffrey.haeyaert at graviteesource.com)
 * @author GraviteeSource Team
 */
@RequiredArgsConstructor
public class DefaultProbeEvaluator implements ProbeEvaluator {

    protected static final long SAFEGUARD_DELAY = 5000L;

    private final long cacheDurationMs;
    private final ProbeManager probeManager;
    private final Map lastProbeResults = new ConcurrentHashMap<>();
    private Long lastEvaluation;

    @Override
    public CompletableFuture> evaluate() {
        return evaluate(Set.of());
    }

    @Override
    public CompletableFuture> evaluate(final Set probeIds) {
        final long now = Instant.now().toEpochMilli();
        final long elapsedTime = lastEvaluation != null ? now - lastEvaluation : Long.MAX_VALUE;

        if (elapsedTime < SAFEGUARD_DELAY) {
            // Avoid too much pressure evaluating probes.
            return CompletableFuture.completedFuture(lastProbeResults);
        }

        final List> collect =
            this.probeManager.getProbes()
                .stream()
                .filter(probe -> probeIds == null || probeIds.isEmpty() || probeIds.contains(probe.id()))
                .map(probe -> {
                    if (probe.isCacheable()) {
                        if (elapsedTime < cacheDurationMs && lastProbeResults.containsKey(probe)) {
                            // The probe has been evaluated once and the elapsed time is below the cache limit. Don't re-evaluate.
                            return CompletableFuture.completedFuture(null);
                        }
                    }

                    // Evaluate the probe and update the probe map.
                    return probe
                        .check()
                        .exceptionally(Result::unhealthy)
                        .thenAccept(result -> lastProbeResults.compute(probe, (probe1, result1) -> result))
                        .toCompletableFuture();
                })
                .toList();

        // Ensure all the probes have been resolved and return all the results.
        return CompletableFuture
            .allOf(collect.toArray(new CompletableFuture[0]))
            .thenApply(unused -> lastProbeResults)
            .whenComplete((probeResultMap, throwable) -> {
                if (throwable == null) {
                    lastEvaluation = now;
                }
            });
    }

    public Map getCachedResults() {
        return lastProbeResults;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy