All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.gerrit.server.update.RetryHelper Maven / Gradle / Ivy

There is a newer version: 3.11.0
Show newest version
// Copyright (C) 2017 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package com.google.gerrit.server.update;

import static com.google.common.base.MoreObjects.firstNonNull;
import static java.util.concurrent.TimeUnit.MILLISECONDS;
import static java.util.concurrent.TimeUnit.SECONDS;

import com.github.rholder.retry.Attempt;
import com.github.rholder.retry.RetryException;
import com.github.rholder.retry.RetryListener;
import com.github.rholder.retry.Retryer;
import com.github.rholder.retry.RetryerBuilder;
import com.github.rholder.retry.StopStrategies;
import com.github.rholder.retry.WaitStrategies;
import com.github.rholder.retry.WaitStrategy;
import com.google.auto.value.AutoValue;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Predicate;
import com.google.common.base.Throwables;
import com.google.common.collect.Maps;
import com.google.common.flogger.FluentLogger;
import com.google.gerrit.common.Nullable;
import com.google.gerrit.extensions.restapi.RestApiException;
import com.google.gerrit.metrics.Counter1;
import com.google.gerrit.metrics.Description;
import com.google.gerrit.metrics.Field;
import com.google.gerrit.metrics.Histogram1;
import com.google.gerrit.metrics.MetricMaker;
import com.google.gerrit.server.config.GerritServerConfig;
import com.google.gerrit.server.git.LockFailureException;
import com.google.gerrit.server.notedb.NotesMigration;
import com.google.inject.Inject;
import com.google.inject.Singleton;
import java.time.Duration;
import java.util.Arrays;
import java.util.Map;
import java.util.concurrent.ExecutionException;
import java.util.function.Consumer;
import org.eclipse.jgit.lib.Config;

@Singleton
public class RetryHelper {
  private static final FluentLogger logger = FluentLogger.forEnclosingClass();

  @FunctionalInterface
  public interface ChangeAction {
    T call(BatchUpdate.Factory batchUpdateFactory) throws Exception;
  }

  @FunctionalInterface
  public interface Action {
    T call() throws Exception;
  }

  public enum ActionType {
    ACCOUNT_UPDATE,
    CHANGE_UPDATE,
    GROUP_UPDATE,
    INDEX_QUERY
  }

  /**
   * Options for retrying a single operation.
   *
   * 

This class is similar in function to upstream's {@link RetryerBuilder}, but it exists as its * own class in Gerrit for several reasons: * *

    *
  • Gerrit needs to support defaults for some of the options, such as a default timeout. * {@code RetryerBuilder} doesn't support calling the same setter multiple times, so doing * this with {@code RetryerBuilder} directly would not be easy. *
  • Gerrit explicitly does not want callers to have full control over all possible options, * so this class exposes a curated subset. *
*/ @AutoValue public abstract static class Options { @Nullable abstract RetryListener listener(); @Nullable abstract Duration timeout(); @AutoValue.Builder public abstract static class Builder { public abstract Builder listener(RetryListener listener); public abstract Builder timeout(Duration timeout); public abstract Options build(); } } @VisibleForTesting @Singleton public static class Metrics { final Histogram1 attemptCounts; final Counter1 timeoutCount; @Inject Metrics(MetricMaker metricMaker) { Field view = Field.ofEnum(ActionType.class, "action_type"); attemptCounts = metricMaker.newHistogram( "action/retry_attempt_counts", new Description( "Distribution of number of attempts made by RetryHelper to execute an action" + " (1 == single attempt, no retry)") .setCumulative() .setUnit("attempts"), view); timeoutCount = metricMaker.newCounter( "action/retry_timeout_count", new Description( "Number of action executions of RetryHelper that ultimately timed out") .setCumulative() .setUnit("timeouts"), view); } } public static Options.Builder options() { return new AutoValue_RetryHelper_Options.Builder(); } private static Options defaults() { return options().build(); } private final NotesMigration migration; private final Metrics metrics; private final BatchUpdate.Factory updateFactory; private final Map defaultTimeouts; private final WaitStrategy waitStrategy; @Nullable private final Consumer> overwriteDefaultRetryerStrategySetup; @Inject RetryHelper( @GerritServerConfig Config cfg, Metrics metrics, NotesMigration migration, ReviewDbBatchUpdate.AssistedFactory reviewDbBatchUpdateFactory, NoteDbBatchUpdate.AssistedFactory noteDbBatchUpdateFactory) { this(cfg, metrics, migration, reviewDbBatchUpdateFactory, noteDbBatchUpdateFactory, null); } @VisibleForTesting public RetryHelper( @GerritServerConfig Config cfg, Metrics metrics, NotesMigration migration, ReviewDbBatchUpdate.AssistedFactory reviewDbBatchUpdateFactory, NoteDbBatchUpdate.AssistedFactory noteDbBatchUpdateFactory, @Nullable Consumer> overwriteDefaultRetryerStrategySetup) { this.metrics = metrics; this.migration = migration; this.updateFactory = new BatchUpdate.Factory(migration, reviewDbBatchUpdateFactory, noteDbBatchUpdateFactory); Duration defaultTimeout = Duration.ofMillis( cfg.getTimeUnit("retry", null, "timeout", SECONDS.toMillis(20), MILLISECONDS)); this.defaultTimeouts = Maps.newEnumMap(ActionType.class); Arrays.stream(ActionType.values()) .forEach( at -> defaultTimeouts.put( at, Duration.ofMillis( cfg.getTimeUnit( "retry", at.name(), "timeout", SECONDS.toMillis(defaultTimeout.getSeconds()), MILLISECONDS)))); this.waitStrategy = WaitStrategies.join( WaitStrategies.exponentialWait( cfg.getTimeUnit("retry", null, "maxWait", SECONDS.toMillis(5), MILLISECONDS), MILLISECONDS), WaitStrategies.randomWait(50, MILLISECONDS)); this.overwriteDefaultRetryerStrategySetup = overwriteDefaultRetryerStrategySetup; } public Duration getDefaultTimeout(ActionType actionType) { return defaultTimeouts.get(actionType); } public T execute( ActionType actionType, Action action, Predicate exceptionPredicate) throws Exception { return execute(actionType, action, defaults(), exceptionPredicate); } public T execute( ActionType actionType, Action action, Options opts, Predicate exceptionPredicate) throws Exception { try { return executeWithAttemptAndTimeoutCount(actionType, action, opts, exceptionPredicate); } catch (Throwable t) { Throwables.throwIfUnchecked(t); Throwables.throwIfInstanceOf(t, Exception.class); throw new IllegalStateException(t); } } public T execute(ChangeAction changeAction) throws RestApiException, UpdateException { return execute(changeAction, defaults()); } public T execute(ChangeAction changeAction, Options opts) throws RestApiException, UpdateException { try { if (!migration.disableChangeReviewDb()) { // Either we aren't full-NoteDb, or the underlying ref storage doesn't support atomic // transactions. Either way, retrying a partially-failed operation is not idempotent, so // don't do it automatically. Let the end user decide whether they want to retry. return executeWithTimeoutCount( ActionType.CHANGE_UPDATE, () -> changeAction.call(updateFactory), RetryerBuilder.newBuilder().build()); } return execute( ActionType.CHANGE_UPDATE, () -> changeAction.call(updateFactory), opts, t -> { if (t instanceof UpdateException) { t = t.getCause(); } return t instanceof LockFailureException; }); } catch (Throwable t) { Throwables.throwIfUnchecked(t); Throwables.throwIfInstanceOf(t, UpdateException.class); Throwables.throwIfInstanceOf(t, RestApiException.class); throw new UpdateException(t); } } /** * Executes an action and records the number of attempts and the timeout as metrics. * * @param actionType the type of the action * @param action the action which should be executed and retried on failure * @param opts options for retrying the action on failure * @param exceptionPredicate predicate to control on which exception the action should be retried * @return the result of executing the action * @throws Throwable any error or exception that made the action fail, callers are expected to * catch and inspect this Throwable to decide carefully whether it should be re-thrown */ private T executeWithAttemptAndTimeoutCount( ActionType actionType, Action action, Options opts, Predicate exceptionPredicate) throws Throwable { MetricListener listener = new MetricListener(); try { RetryerBuilder retryerBuilder = createRetryerBuilder(actionType, opts, exceptionPredicate); retryerBuilder.withRetryListener(listener); return executeWithTimeoutCount(actionType, action, retryerBuilder.build()); } finally { if (listener.getAttemptCount() > 1) { logger.atFine().log("%s was attempted %d times", actionType, listener.getAttemptCount()); } metrics.attemptCounts.record(actionType, listener.getAttemptCount()); } } /** * Executes an action and records the timeout as metric. * * @param actionType the type of the action * @param action the action which should be executed and retried on failure * @param retryer the retryer * @return the result of executing the action * @throws Throwable any error or exception that made the action fail, callers are expected to * catch and inspect this Throwable to decide carefully whether it should be re-thrown */ private T executeWithTimeoutCount(ActionType actionType, Action action, Retryer retryer) throws Throwable { try { return retryer.call(action::call); } catch (ExecutionException | RetryException e) { if (e instanceof RetryException) { metrics.timeoutCount.increment(actionType); } if (e.getCause() != null) { throw e.getCause(); } throw e; } } private RetryerBuilder createRetryerBuilder( ActionType actionType, Options opts, Predicate exceptionPredicate) { RetryerBuilder retryerBuilder = RetryerBuilder.newBuilder().retryIfException(exceptionPredicate); if (opts.listener() != null) { retryerBuilder.withRetryListener(opts.listener()); } if (overwriteDefaultRetryerStrategySetup != null) { overwriteDefaultRetryerStrategySetup.accept(retryerBuilder); return retryerBuilder; } return retryerBuilder .withStopStrategy( StopStrategies.stopAfterDelay( firstNonNull(opts.timeout(), getDefaultTimeout(actionType)).toMillis(), MILLISECONDS)) .withWaitStrategy(waitStrategy); } private static class MetricListener implements RetryListener { private long attemptCount; MetricListener() { attemptCount = 1; } @Override public void onRetry(Attempt attempt) { attemptCount = attempt.getAttemptNumber(); } long getAttemptCount() { return attemptCount; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy