org.kiwiproject.curator.leader.health.ManagedLeaderLatchHealthCheck Maven / Gradle / Ivy
package org.kiwiproject.curator.leader.health;
import static java.util.stream.Collectors.toList;
import static org.kiwiproject.collect.KiwiLists.isNullOrEmpty;
import static org.kiwiproject.metrics.health.HealthCheckResults.newHealthyResultBuilder;
import static org.kiwiproject.metrics.health.HealthCheckResults.newUnhealthyResultBuilder;
import static org.kiwiproject.metrics.health.HealthStatus.CRITICAL;
import com.codahale.metrics.health.HealthCheck;
import org.apache.curator.framework.recipes.leader.Participant;
import org.kiwiproject.curator.leader.ManagedLeaderLatch;
import java.util.Collection;
import java.util.List;
/**
* Simple check to determine if the leader latch is started (healthy) or otherwise (not healthy).
*
* The check also reports as unhealthy if there is no leader, or if more than one leader is reported by Curator.
*/
public class ManagedLeaderLatchHealthCheck extends HealthCheck {
private static final String LEADER_DETAIL_NAME = "leader";
private static final String PARTICIPANTS_DETAIL_NAME = "participants";
private final ManagedLeaderLatch leaderLatch;
/**
* New health check instance for the given leader latch.
*
* @param leaderLatch the {@link ManagedLeaderLatch} to check
*/
public ManagedLeaderLatchHealthCheck(ManagedLeaderLatch leaderLatch) {
this.leaderLatch = leaderLatch;
}
@Override
protected Result check() {
if (!leaderLatch.isStarted()) {
return newUnhealthyResultBuilder(CRITICAL)
.withMessage("Leader latch state (%s) is not started", leaderLatch.getLatchState())
.withDetail(LEADER_DETAIL_NAME, false)
.withDetail(PARTICIPANTS_DETAIL_NAME, List.of())
.build();
}
var participants = leaderLatch.getParticipants();
var leaderIds = leaderIdsOf(participants);
if (isNullOrEmpty(leaderIds)) {
return newUnhealthyResultBuilder(CRITICAL)
.withMessage("There are NO leaders for latch path %s", leaderLatch.getLatchPath())
.withDetail(LEADER_DETAIL_NAME, false)
.withDetail(PARTICIPANTS_DETAIL_NAME, idsOf(participants))
.build();
} else if (leaderIds.size() > 1) {
return newUnhealthyResultBuilder(CRITICAL)
.withMessage("There is more than one leader for latch path %s. Leader IDs: %s (this latch ID: %s)",
leaderLatch.getLatchPath(), leaderIds, leaderLatch.getId())
.withDetail(LEADER_DETAIL_NAME, leaderLatch.hasLeadership())
.withDetail(PARTICIPANTS_DETAIL_NAME, idsOf(participants))
.build();
}
return newHealthyResultBuilder()
.withMessage("Leader latch is started (has leadership? %s)", leaderLatch.hasLeadership())
.withDetail(LEADER_DETAIL_NAME, leaderLatch.hasLeadership())
.withDetail(PARTICIPANTS_DETAIL_NAME, idsOf(participants))
.build();
}
private static List leaderIdsOf(Collection participants) {
return participants.stream()
.filter(Participant::isLeader)
.map(Participant::getId)
.collect(toList());
}
private static List idsOf(Collection participants) {
return participants.stream()
.map(Participant::getId)
.collect(toList());
}
}