com.yahoo.vespa.hosted.provision.maintenance.LoadBalancerExpirer Maven / Gradle / Ivy
// Copyright 2018 Yahoo Holdings. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
package com.yahoo.vespa.hosted.provision.maintenance;
import com.yahoo.log.LogLevel;
import com.yahoo.vespa.hosted.provision.NodeRepository;
import com.yahoo.vespa.hosted.provision.lb.LoadBalancer;
import com.yahoo.vespa.hosted.provision.lb.LoadBalancer.State;
import com.yahoo.vespa.hosted.provision.lb.LoadBalancerId;
import com.yahoo.vespa.hosted.provision.lb.LoadBalancerService;
import com.yahoo.vespa.hosted.provision.persistence.CuratorDatabaseClient;
import java.time.Duration;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.stream.Collectors;
/**
* Periodically expire load balancers.
*
* Load balancers expire from the following states:
*
* {@link LoadBalancer.State#inactive}: An application is removed and load balancers are deactivated.
* {@link LoadBalancer.State#reserved}: An prepared application is never successfully activated, thus never activating
* any prepared load balancers.
*
* @author mpolden
*/
public class LoadBalancerExpirer extends Maintainer {
private static final Duration reservedExpiry = Duration.ofHours(1);
private static final Duration inactiveExpiry = Duration.ofHours(1);
private final LoadBalancerService service;
private final CuratorDatabaseClient db;
LoadBalancerExpirer(NodeRepository nodeRepository, Duration interval, LoadBalancerService service) {
super(nodeRepository, interval);
this.service = Objects.requireNonNull(service, "service must be non-null");
this.db = nodeRepository.database();
}
@Override
protected void maintain() {
expireReserved();
removeInactive();
}
private void expireReserved() {
try (var lock = db.lockLoadBalancers()) {
var now = nodeRepository().clock().instant();
var expirationTime = now.minus(reservedExpiry);
var expired = nodeRepository().loadBalancers()
.in(State.reserved)
.changedBefore(expirationTime);
expired.forEach(lb -> db.writeLoadBalancer(lb.with(State.inactive, now)));
}
}
private void removeInactive() {
List failed = new ArrayList<>();
Exception lastException = null;
try (var lock = db.lockLoadBalancers()) {
var now = nodeRepository().clock().instant();
var expirationTime = now.minus(inactiveExpiry);
var expired = nodeRepository().loadBalancers()
.in(State.inactive)
.changedBefore(expirationTime);
for (var lb : expired) {
if (hasNodes(lb.id())) { // Defer removal if there are still nodes allocated
continue;
}
try {
service.remove(lb.id().application(), lb.id().cluster());
db.removeLoadBalancer(lb.id());
} catch (Exception e) {
failed.add(lb.id());
lastException = e;
}
}
}
if (!failed.isEmpty()) {
log.log(LogLevel.WARNING, String.format("Failed to remove %d load balancers: %s, retrying in %s",
failed.size(),
failed.stream()
.map(LoadBalancerId::serializedForm)
.collect(Collectors.joining(", ")),
interval()),
lastException);
}
}
private boolean hasNodes(LoadBalancerId loadBalancer) {
return nodeRepository().list().owner(loadBalancer.application()).cluster(loadBalancer.cluster()).size() > 0;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy