com.yahoo.vespa.hosted.provision.testutils.MockDeployer Maven / Gradle / Ivy
// Copyright Vespa.ai. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
package com.yahoo.vespa.hosted.provision.testutils;
import com.yahoo.component.Version;
import com.yahoo.component.annotation.Inject;
import com.yahoo.config.provision.ActivationContext;
import com.yahoo.config.provision.ApplicationId;
import com.yahoo.config.provision.ApplicationTransaction;
import com.yahoo.config.provision.Capacity;
import com.yahoo.config.provision.ClusterSpec;
import com.yahoo.config.provision.Deployer;
import com.yahoo.config.provision.Deployment;
import com.yahoo.config.provision.HostFilter;
import com.yahoo.config.provision.HostSpec;
import com.yahoo.transaction.NestedTransaction;
import com.yahoo.vespa.hosted.provision.Node;
import com.yahoo.vespa.hosted.provision.NodeMutex;
import com.yahoo.vespa.hosted.provision.NodeRepository;
import com.yahoo.vespa.hosted.provision.provisioning.NodeRepositoryProvisioner;
import com.yahoo.vespa.service.duper.InfraApplication;
import java.time.Clock;
import java.time.Duration;
import java.time.Instant;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.locks.ReentrantLock;
import java.util.stream.Collectors;
/**
* @author bratseth
*/
public class MockDeployer implements Deployer {
// For actual deploy mode
private final NodeRepositoryProvisioner provisioner;
private final Map applications;
// For mock deploy anything, changing wantToRetire to retired only
private final NodeRepository nodeRepository;
/** The number of activations done to this, which is also the config generation */
public int activations = 0;
private final Map lastPrepareTimes = new HashMap<>();
private final Map lastActivationTimes = new HashMap<>();
private final Clock clock;
private final ReentrantLock lock = new ReentrantLock();
private boolean failActivate = false;
private boolean bootstrapping = true;
private Instant readiedReindexingAt = null;
/** Create a mock deployer which returns empty on every deploy request. */
@Inject
@SuppressWarnings("unused")
public MockDeployer() {
this(null, Clock.systemUTC(), List.of());
}
/**
* Create a mock deployer which returns a deployment on every request,
* and fulfills it by not actually deploying but only changing any wantToRetire nodes
* for the application to retired.
*/
public MockDeployer(NodeRepository nodeRepository) {
this.provisioner = null;
this.applications = Map.of();
this.nodeRepository = nodeRepository;
this.clock = nodeRepository.clock();
}
/**
* Create a mock deployer which contains a substitute for an application repository, filled to
* be able to call provision with the right parameters.
*/
public MockDeployer(NodeRepositoryProvisioner provisioner,
Clock clock,
List applications) {
this.provisioner = provisioner;
this.applications = applications.stream().collect(Collectors.toMap(ApplicationContext::id, c -> c));
this.nodeRepository = null;
this.clock = clock;
}
public ReentrantLock lock() { return lock; }
public void setFailActivate(boolean failActivate) { this.failActivate = failActivate; }
public void setBootstrapping(boolean bootstrapping) { this.bootstrapping = bootstrapping; }
public void setReadiedReindexingAt(Instant readiedReindexingAt) { this.readiedReindexingAt = readiedReindexingAt; }
@Override
public Optional deployFromLocalActive(ApplicationId id, boolean bootstrap) {
return deployFromLocalActive(id, Duration.ofSeconds(60));
}
@Override
public Optional deployFromLocalActive(ApplicationId id, Duration timeout) {
try {
lock.lockInterruptibly();
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
try {
if (provisioner != null)
return Optional.ofNullable(applications.get(id))
.map(application -> new MockDeployment(provisioner, application));
else
return Optional.of(new RetiringOnlyMockDeployment(nodeRepository, id));
}
finally {
lock.unlock();
}
}
@Override
public Optional deployFromLocalActive(ApplicationId application, Duration timeout, boolean bootstrap) {
return deployFromLocalActive(application, timeout);
}
@Override
public Optional activationTime(ApplicationId application) {
return Optional.ofNullable(lastActivationTimes.get(application));
}
@Override
public Optional deployTime(ApplicationId application) {
return Optional.ofNullable(lastPrepareTimes.get(application));
}
@Override
public boolean readiedReindexingAfter(ApplicationId application, Instant instant) {
return readiedReindexingAt != null && readiedReindexingAt.isAfter(instant);
}
@Override
public boolean bootstrapping() {
return bootstrapping;
}
@Override
public Duration serverDeployTimeout() { return Duration.ofSeconds(60); }
public void removeApplication(ApplicationId applicationId) {
new MockDeployment(provisioner, new ApplicationContext(applicationId, List.of())).activate();
applications.remove(applicationId);
lastPrepareTimes.remove(applicationId);
lastActivationTimes.remove(applicationId);
}
public class MockDeployment implements Deployment {
private final NodeRepositoryProvisioner provisioner;
private final ApplicationContext application;
/** The list of hosts prepared in this. Only set after prepare is called (and a provisioner is assigned) */
private List preparedHosts = null;
private MockDeployment(NodeRepositoryProvisioner provisioner, ApplicationContext application) {
this.provisioner = provisioner;
this.application = application;
}
@Override
public void prepare() {
preparedHosts = application.prepare(provisioner);
lastPrepareTimes.put(application.id, clock.instant());
}
@Override
public long activate() {
if (preparedHosts == null)
prepare();
if (failActivate)
throw new IllegalStateException("failActivate is true");
activations++;
try (var lock = provisioner.lock(application.id)) {
try (NestedTransaction t = new NestedTransaction()) {
provisioner.activate(preparedHosts, new ActivationContext(activations), new ApplicationTransaction(lock, t));
t.commit();
lastActivationTimes.put(application.id, clock.instant());
}
}
return activations;
}
@Override
public void restart(HostFilter filter) {}
}
public class RetiringOnlyMockDeployment implements Deployment {
private final NodeRepository nodeRepository;
private final ApplicationId applicationId;
private RetiringOnlyMockDeployment(NodeRepository nodeRepository, ApplicationId applicationId) {
this.nodeRepository = nodeRepository;
this.applicationId = applicationId;
}
@Override
public void prepare() {
lastPrepareTimes.put(applicationId, clock.instant());
}
@Override
public long activate() {
lastActivationTimes.put(applicationId, clock.instant());
for (Node node : nodeRepository.nodes().list().owner(applicationId).state(Node.State.active).retiring()) {
try (NodeMutex lock = nodeRepository.nodes().lockAndGetRequired(node)) {
nodeRepository.nodes().write(lock.node().retire(nodeRepository.clock().instant()), lock);
}
}
return activations++;
}
@Override
public void restart(HostFilter filter) {}
}
/** An application context which substitutes for an application repository */
public static class ApplicationContext {
private final ApplicationId id;
private final List clusterContexts;
public ApplicationContext(ApplicationId id, List clusterContexts) {
this.id = id;
this.clusterContexts = clusterContexts;
}
public ApplicationContext(ApplicationId id, ClusterSpec cluster, Capacity capacity) {
this(id, List.of(new ClusterContext(id, cluster, capacity)));
}
public ApplicationContext(InfraApplication application, Version version) {
this(application.getApplicationId(), List.of(new ClusterContext(
application.getApplicationId(), application.getClusterSpecWithVersion(version), application.getCapacity())));
}
public ApplicationId id() { return id; }
/** Returns list of cluster specs of this application. */
public List clusterContexts() { return clusterContexts; }
private List prepare(NodeRepositoryProvisioner provisioner) {
return clusterContexts.stream()
.map(clusterContext -> clusterContext.prepare(provisioner))
.flatMap(List::stream)
.toList();
}
}
public static class ClusterContext {
private final ApplicationId id;
private final ClusterSpec cluster;
private final Capacity capacity;
public ClusterContext(ApplicationId id, ClusterSpec cluster, Capacity capacity) {
this.id = id;
this.cluster = cluster;
this.capacity = capacity;
}
public ApplicationId id() { return id; }
public ClusterSpec cluster() { return cluster; }
private List prepare(NodeRepositoryProvisioner provisioner) {
return provisioner.prepare(id, cluster, capacity, new InMemoryProvisionLogger());
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy