ai.vespa.reindexing.Reindexer Maven / Gradle / Ivy
// Copyright Vespa.ai. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
package ai.vespa.reindexing;
import ai.vespa.reindexing.Reindexing.Status;
import ai.vespa.reindexing.Reindexing.Trigger;
import ai.vespa.reindexing.ReindexingCurator.ReindexingLockException;
import com.yahoo.document.DocumentType;
import com.yahoo.document.select.parser.ParseException;
import com.yahoo.documentapi.DocumentAccess;
import com.yahoo.documentapi.ProgressToken;
import com.yahoo.documentapi.VisitorControlHandler;
import com.yahoo.documentapi.VisitorControlHandler.CompletionCode;
import com.yahoo.documentapi.VisitorParameters;
import com.yahoo.documentapi.messagebus.protocol.DocumentProtocol;
import com.yahoo.jdisc.Metric;
import com.yahoo.messagebus.DynamicThrottlePolicy;
import com.yahoo.vespa.curator.Lock;
import java.time.Clock;
import java.time.Duration;
import java.time.Instant;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.Phaser;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.Function;
import java.util.logging.Logger;
import static java.util.Comparator.comparingDouble;
import static java.util.Objects.requireNonNull;
import static java.util.logging.Level.FINE;
import static java.util.logging.Level.INFO;
import static java.util.logging.Level.WARNING;
/**
* Progresses reindexing efforts by creating visitor sessions against its own content cluster,
* which send documents straight to storage — via indexing if the document type has "index" mode.
* The {@link #reindex} method blocks until shutdown is called, or until no more reindexing is left to do.
*
* @author jonmv
*/
class Reindexer {
private static final Logger log = Logger.getLogger(Reindexer.class.getName());
static final Duration failureGrace = Duration.ofMinutes(10);
static final Duration PROGRESS_TOKEN_STORE_INTERVAL = Duration.ofSeconds(60);
private final Cluster cluster;
private final List ready;
private final ReindexingCurator database;
private final Function visitorSessions;
private final ReindexingMetrics metrics;
private final Clock clock;
private final Phaser phaser = new Phaser(2); // Reindexer and visitor.
Reindexer(Cluster cluster, List ready, ReindexingCurator database,
DocumentAccess access, Metric metric, Clock clock) {
this(cluster,
ready,
database,
parameters -> {
try {
return access.createVisitorSession(parameters)::destroy;
}
catch (ParseException e) {
throw new IllegalStateException(e);
}
},
metric,
clock
);
}
Reindexer(Cluster cluster, List ready, ReindexingCurator database,
Function visitorSessions, Metric metric, Clock clock) {
for (Trigger trigger : ready)
cluster.bucketSpaceOf(trigger.type()); // Verifies this is known.
this.cluster = cluster;
this.ready = ready.stream() // Iterate through document types in consistent order.
.sorted(comparingDouble(Trigger::speed).reversed().thenComparing(Trigger::readyAt).thenComparing(Trigger::type))
.toList();
this.database = database;
this.visitorSessions = visitorSessions;
this.metrics = new ReindexingMetrics(metric, cluster.name);
this.clock = clock;
database.initializeIfEmpty(cluster.name, ready, clock.instant());
}
/** Lets the reindexer abort any ongoing visit session, wait for it to complete normally, then exit. */
void shutdown() {
phaser.forceTermination(); // All parties waiting on this phaser are immediately allowed to proceed.
}
/** Starts and tracks reprocessing of ready document types until done, or interrupted. */
void reindex() throws ReindexingLockException {
if (phaser.isTerminated())
throw new IllegalStateException("Already shut down");
// Keep metrics in sync across cluster controller containers.
AtomicReference reindexing = new AtomicReference<>(database.readReindexing(cluster.name()));
metrics.dump(reindexing.get());
try (Lock lock = database.lockReindexing(cluster.name())) {
reindexing.set(updateWithReady(ready, database.readReindexing(cluster.name()), clock.instant()));
database.writeReindexing(reindexing.get(), cluster.name());
metrics.dump(reindexing.get());
// We consider only document types for which we have config.
for (Trigger trigger : ready) {
if (trigger.readyAt().isAfter(clock.instant()))
log.log(INFO, "Received config for reindexing which is ready in the future — will process later " +
"(" + trigger.readyAt() + " is after " + clock.instant() + ")");
else if (trigger.speed() > 0)
progress(trigger.type(), trigger.speed(), reindexing, new AtomicReference<>(reindexing.get().status().get(trigger.type())));
if (phaser.isTerminated())
break;
}
}
}
static Reindexing updateWithReady(List ready, Reindexing reindexing, Instant now) {
for (Trigger trigger : ready) { // We update only for document types for which we have config.
if ( ! trigger.readyAt().isAfter(now)) {
Status status = reindexing.status().get(trigger.type());
if (status == null || status.startedAt().isBefore(trigger.readyAt()))
status = Status.ready(now);
reindexing = reindexing.with(trigger.type(), status);
}
}
return reindexing;
}
@SuppressWarnings("fallthrough") // (ノಠ ∩ಠ)ノ彡( \o°o)\
private void progress(DocumentType type, double speed, AtomicReference reindexing, AtomicReference status) {
switch (status.get().state()) {
default:
log.log(WARNING, "Unknown reindexing state '" + status.get().state() + "'—not continuing reindexing of " + type);
case SUCCESSFUL: // Intentional fallthrough — all three are done states.
return;
case RUNNING:
log.log(WARNING, "Unexpected state 'RUNNING' of reindexing of " + type);
break;
case FAILED:
if (clock.instant().isBefore(status.get().endedAt().get().plus(failureGrace)))
return;
case READY:
status.updateAndGet(Status::running);
}
// Visit buckets until they're all done, or until we are shut down.
AtomicReference progressLastStored = new AtomicReference<>(clock.instant());
VisitorControlHandler control = new VisitorControlHandler() {
@Override
public void onProgress(ProgressToken token) {
super.onProgress(token);
status.updateAndGet(value -> value.progressed(token));
if (progressLastStored.get().isBefore(clock.instant().minus(PROGRESS_TOKEN_STORE_INTERVAL))) {
progressLastStored.set(clock.instant());
database.writeReindexing(reindexing.updateAndGet(value -> value.with(type, status.get())), cluster.name());
metrics.dump(reindexing.get());
}
}
@Override
public void onDone(CompletionCode code, String message) {
super.onDone(code, message);
phaser.arriveAndAwaitAdvance(); // Synchronize with the reindexer control thread.
}
};
VisitorParameters parameters = createParameters(type, speed, status.get().progress().orElse(null));
parameters.setControlHandler(control);
Runnable sessionShutdown = visitorSessions.apply(parameters); // Also starts the visitor session.
log.log(FINE, () -> "Running reindexing of " + type);
// Wait until done; or until termination is forced, in which case we shut down the visitor session immediately.
phaser.arriveAndAwaitAdvance(); // Synchronize with visitor completion.
sessionShutdown.run(); // Shutdown aborts the session unless already complete, then waits for it to terminate normally.
// Only as a last resort will we be interrupted here, and the wait for outstanding replies terminate.
CompletionCode result = control.getResult() != null ? control.getResult().getCode() : CompletionCode.ABORTED;
switch (result) {
default:
log.log(WARNING, "Unexpected visitor result '" + control.getResult().getCode() + "'");
case FAILURE: // Intentional fallthrough — this is an error.
log.log(WARNING, "Visiting failed: " + control.getResult().getMessage());
status.updateAndGet(value -> value.failed(clock.instant(), control.getResult().getMessage()));
break;
case ABORTED:
log.log(FINE, () -> "Halting reindexing of " + type + " due to shutdown — will continue later");
status.updateAndGet(Status::halted);
break;
case SUCCESS:
log.log(INFO, "Completed reindexing of " + type + " after " + Duration.between(status.get().startedAt(), clock.instant()));
status.updateAndGet(value -> value.successful(clock.instant()));
}
database.writeReindexing(reindexing.updateAndGet(value -> value.with(type, status.get())), cluster.name());
metrics.dump(reindexing.get());
}
VisitorParameters createParameters(DocumentType type, double speed, ProgressToken progress) {
VisitorParameters parameters = new VisitorParameters(type.getName());
parameters.setThrottlePolicy(new DynamicThrottlePolicy().setWindowSizeIncrement(speed)
.setWindowSizeDecrementFactor(3)
.setResizeRate(5)
.setMaxWindowSize(128)
.setMinWindowSize(3 + (int) (5 * speed)));
parameters.setRemoteDataHandler(cluster.name());
parameters.setMaxPending(8);
parameters.setResumeToken(progress);
parameters.setFieldSet(type.getName() + ":[document]");
parameters.setPriority(DocumentProtocol.Priority.NORMAL_3);
parameters.setRoute(cluster.route());
parameters.setBucketSpace(cluster.bucketSpaceOf(type));
parameters.setMaxBucketsPerVisitor(1);
parameters.setVisitorLibrary("ReindexingVisitor");
return parameters;
}
static class Cluster {
private final String name;
private final Map documentBuckets;
Cluster(String name, Map documentBuckets) {
this.name = requireNonNull(name);
this.documentBuckets = Map.copyOf(documentBuckets);
}
String name() {
return name;
}
String route() {
return "[Content:cluster=" + name + "]";
}
String bucketSpaceOf(DocumentType documentType) {
return requireNonNull(documentBuckets.get(documentType), "Unknown bucket space for " + documentType);
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Cluster cluster = (Cluster) o;
return name.equals(cluster.name) &&
documentBuckets.equals(cluster.documentBuckets);
}
@Override
public int hashCode() {
return Objects.hash(name, documentBuckets);
}
@Override
public String toString() {
return "Cluster{" +
"name='" + name + '\'' +
", documentBuckets=" + documentBuckets +
'}';
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy