com.yahoo.vespa.hosted.provision.node.History 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.node;
import com.google.common.collect.ImmutableMap;
import com.yahoo.vespa.hosted.provision.Node;
import java.time.Duration;
import java.time.Instant;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Comparator;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
/**
* An immutable record of the last event of each type happening to this node, and a chronological log of the events.
*
* Note that the history cannot be used to find the nodes current state - it will have a record of some
* event happening in the past even if that event is later undone.
*
* @author bratseth
*/
public class History {
private static final int MAX_LOG_SIZE = 10;
private final ImmutableMap events;
private final List log;
private final int maxLogSize;
public History(Collection events, List log) {
this(toImmutableMap(events), log, MAX_LOG_SIZE);
}
History(ImmutableMap events, List log, int maxLogSize) {
this.events = events;
this.log = Objects.requireNonNull(log, "log must be non-null")
.stream()
.sorted(Comparator.comparing(Event::at))
.skip(Math.max(log.size() - maxLogSize, 0))
.toList();
this.maxLogSize = maxLogSize;
}
private static ImmutableMap toImmutableMap(Collection events) {
ImmutableMap.Builder builder = new ImmutableMap.Builder<>();
for (Event event : events)
builder.put(event.type(), event);
return builder.build();
}
/** Returns the age of this node as best as we can determine: The time since the first event registered for it */
public Duration age(Instant now) {
Instant oldestEventTime = events.values().stream().map(Event::at).sorted().findFirst().orElse(now);
return Duration.between(oldestEventTime, now);
}
/** Returns the last event of given type, if it is present in this history */
public Optional event(Event.Type type) { return Optional.ofNullable(events.get(type)); }
/** Returns true if a given event is registered in this history at the given time */
public boolean hasEventAt(Event.Type type, Instant time) {
return event(type)
.map(event -> event.at().equals(time))
.orElse(false);
}
/** Returns true if a given event is registered in this history after the given time */
public boolean hasEventAfter(Event.Type type, Instant time) {
return event(type)
.map(event -> event.at().isAfter(time))
.orElse(false);
}
/** Returns true if a given event is registered in this history before the given time */
public boolean hasEventBefore(Event.Type type, Instant time) {
return event(type)
.map(event -> event.at().isBefore(time))
.orElse(false);
}
/** Returns the instant the services went down, unless the services have been up afterward. */
public Optional downSince() { return instantOf(Event.Type.down, Event.Type.up); }
/** Returns the instant the services went up, unless the services have been down afterward. */
public Optional upSince() { return instantOf(Event.Type.up, Event.Type.down); }
/** Returns true if there is a down event without a later up. */
public boolean isDown() { return downSince().isPresent(); }
/** Returns true if there is an up event without a later down. */
public boolean isUp() { return upSince().isPresent(); }
/** Returns the instant the node suspended, unless the node has been resumed afterward. */
public Optional suspendedSince() { return instantOf(Event.Type.suspended, Event.Type.resumed); }
/** Returns the instant the node was resumed, unless the node has been suspended afterward. */
public Optional resumedSince() { return instantOf(Event.Type.resumed, Event.Type.suspended); }
/** Returns true if there is a suspended event without a later resumed. */
public boolean isSuspended() { return suspendedSince().isPresent(); }
/** Returns true if there is a resumed event without a later suspended. */
public boolean isResumed() { return resumedSince().isPresent(); }
private Optional instantOf(History.Event.Type type, History.Event.Type sentinelType) {
return event(type).map(History.Event::at).filter(instant -> !hasEventAfter(sentinelType, instant));
}
/** Returns the last event of each type in this history */
public Collection events() { return events.values(); }
/**
* Returns the events in this history, in chronological order. Compared to {@link #events()}, this holds all events
* as they occurred, up to log size limit
*/
public List log() { return log; }
/** Returns a copy of this history with the given event added */
public History with(Event event) {
ImmutableMap.Builder builder = builderWithout(event.type());
builder.put(event.type(), event);
List logCopy = new ArrayList<>(log);
logCopy.add(event);
return new History(builder.build(), logCopy, maxLogSize);
}
/** Returns a copy of this history with the given event type removed (or an identical history if it was not
* present) and the log unchanged. */
public History without(Event.Type type) {
return new History(builderWithout(type).build(), log, maxLogSize);
}
private ImmutableMap.Builder builderWithout(Event.Type type) {
ImmutableMap.Builder builder = new ImmutableMap.Builder<>();
for (Event event : events.values())
if (event.type() != type)
builder.put(event.type(), event);
return builder;
}
/** Returns a copy of this history with a record of this state transition added, if applicable */
public History recordStateTransition(Node.State from, Node.State to, Agent agent, Instant at) {
// If the event is a re-reservation, allow the new one to override the older one.
if (from == to && from != Node.State.reserved) return this;
return switch (to) {
case provisioned -> this.with(new Event(Event.Type.provisioned, agent, at));
case deprovisioned -> this.with(new Event(Event.Type.deprovisioned, agent, at));
case ready -> this.withoutApplicationEvents().with(new Event(Event.Type.readied, agent, at));
case active -> this.with(new Event(Event.Type.activated, agent, at));
case inactive -> this.with(new Event(Event.Type.deactivated, agent, at));
case reserved -> this.with(new Event(Event.Type.reserved, agent, at));
case failed -> this.with(new Event(Event.Type.failed, agent, at));
case dirty -> this.with(new Event(Event.Type.deallocated, agent, at));
case parked -> this.with(new Event(Event.Type.parked, agent, at));
case breakfixed -> this.with(new Event(Event.Type.breakfixed, agent, at));
};
}
/**
* Events can be application or node level.
* This returns a copy of this history with all application level events removed and the log unchanged.
*/
private History withoutApplicationEvents() {
return new History(events().stream().filter(e -> ! e.type().isApplicationLevel()).toList(), log);
}
/** Returns the empty history */
public static History empty() { return new History(List.of(), List.of()); }
@Override
public String toString() {
if (events.isEmpty()) return "history: (empty)";
StringBuilder b = new StringBuilder("history: ");
for (Event e : events.values())
b.append(e).append(", ");
b.setLength(b.length() - 2); // remove last comma
return b.toString();
}
/** An event which may happen to a node */
public record Event(Type type, Agent agent, Instant at) {
public enum Type {
// State changes
activated,
breakfixed(false),
deactivated,
deallocated,
deprovisioned(false),
failed(false),
parked,
provisioned(false),
readied,
reserved,
// The node was scheduled for retirement (hard)
wantToRetire(false),
// The node was scheduled for retirement (soft)
preferToRetire(false),
// This node was scheduled for failing
wantToFail,
// The active node was retired
retired,
// The active node went down according to the service monitor
down,
// The active node came up according to the service monitor
up,
// The node has been given permission to suspend by Orchestrator
suspended,
// The node has resumed from suspension by Orchestrator
resumed,
// The node resources/flavor were changed
resized(false),
// The node was rebooted
rebooted(false),
// The node upgraded its OS (implies a reboot)
osUpgraded(false),
// The node verified its firmware (whether this resulted in a reboot depends on the node model)
firmwareVerified(false);
private final boolean applicationLevel;
/** Creates an application level event */
Type() {
this.applicationLevel = true;
}
Type(boolean applicationLevel) {
this.applicationLevel = applicationLevel;
}
/** Returns true if this is an application-level event and false if it's a node-level event */
public boolean isApplicationLevel() { return applicationLevel; }
}
/** Returns the type of event */
public Event.Type type() { return type; }
/** Returns the agent causing this event */
public Agent agent() { return agent; }
/** Returns the instant this even took place */
public Instant at() { return at; }
@Override
public String toString() { return "'" + type + "' event at " + at + " by " + agent; }
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy