blob: a43d7d9c22ef46fb1f5c969ff2eaee22dbf56f57 [file] [log] [blame]
// Copyright (C) 2013 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.googlesource.gerrit.plugins.replication;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.Multimap;
import com.google.common.flogger.FluentLogger;
import com.google.gerrit.common.FileUtil;
import com.google.gerrit.extensions.annotations.PluginData;
import com.google.gerrit.extensions.annotations.PluginName;
import com.google.gerrit.reviewdb.client.Project;
import com.google.gerrit.server.config.SitePaths;
import com.google.gerrit.server.git.WorkQueue;
import com.google.inject.Inject;
import com.google.inject.Provider;
import com.google.inject.Singleton;
import java.io.IOException;
import java.nio.file.Path;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import org.eclipse.jgit.errors.ConfigInvalidException;
import org.eclipse.jgit.transport.URIish;
@Singleton
public class AutoReloadConfigDecorator implements ReplicationConfig {
private static final FluentLogger logger = FluentLogger.forEnclosingClass();
private static final long RELOAD_DELAY = 120;
private static final long RELOAD_INTERVAL = 60;
private volatile ReplicationFileBasedConfig currentConfig;
private long currentConfigTs;
private long lastFailedConfigTs;
private final SitePaths site;
private final Destination.Factory destinationFactory;
private final Path pluginDataDir;
// Use Provider<> instead of injecting the ReplicationQueue because of circular dependency with
// ReplicationConfig
private final Provider<ReplicationQueue> replicationQueue;
private final ScheduledExecutorService autoReloadExecutor;
private ScheduledFuture<?> autoReloadRunnable;
private volatile boolean shuttingDown;
@Inject
public AutoReloadConfigDecorator(
SitePaths site,
Destination.Factory destinationFactory,
Provider<ReplicationQueue> replicationQueue,
@PluginData Path pluginDataDir,
@PluginName String pluginName,
WorkQueue workQueue)
throws ConfigInvalidException, IOException {
this.site = site;
this.destinationFactory = destinationFactory;
this.pluginDataDir = pluginDataDir;
this.currentConfig = loadConfig();
this.currentConfigTs = getLastModified(currentConfig);
this.replicationQueue = replicationQueue;
this.autoReloadExecutor = workQueue.createQueue(1, pluginName + "_auto-reload-config");
}
private static long getLastModified(ReplicationFileBasedConfig cfg) {
return FileUtil.lastModified(cfg.getCfgPath());
}
private ReplicationFileBasedConfig loadConfig() throws ConfigInvalidException, IOException {
return new ReplicationFileBasedConfig(site, destinationFactory, pluginDataDir);
}
private synchronized boolean isAutoReload() {
return currentConfig.getConfig().getBoolean("gerrit", "autoReload", false);
}
@Override
public List<Destination> getDestinations(URIish uri, Project.NameKey project, String ref) {
reloadIfNeeded();
return currentConfig.getDestinations(uri, project, ref);
}
@Override
public synchronized List<Destination> getDestinations(FilterType filterType) {
return currentConfig.getDestinations(filterType);
}
@Override
public synchronized Multimap<Destination, URIish> getURIs(
Optional<String> remoteName, Project.NameKey projectName, FilterType filterType) {
return currentConfig.getURIs(remoteName, projectName, filterType);
}
private synchronized void reloadIfNeeded() {
reload(false);
}
@VisibleForTesting
public void forceReload() {
reload(true);
}
private void reload(boolean force) {
if (force || isAutoReload()) {
ReplicationQueue queue = replicationQueue.get();
long lastModified = getLastModified(currentConfig);
try {
if (force
|| (!shuttingDown
&& lastModified > currentConfigTs
&& lastModified > lastFailedConfigTs
&& queue.isRunning()
&& !queue.isReplaying())) {
queue.stop();
currentConfig = loadConfig();
currentConfigTs = lastModified;
lastFailedConfigTs = 0;
logger.atInfo().log(
"Configuration reloaded: %d destinations",
currentConfig.getDestinations(FilterType.ALL).size());
}
} catch (Exception e) {
logger.atSevere().withCause(e).log(
"Cannot reload replication configuration: keeping existing settings");
lastFailedConfigTs = lastModified;
return;
} finally {
queue.start();
}
}
}
@Override
public synchronized boolean isReplicateAllOnPluginStart() {
return currentConfig.isReplicateAllOnPluginStart();
}
@Override
public synchronized boolean isDefaultForceUpdate() {
return currentConfig.isDefaultForceUpdate();
}
@Override
public synchronized int getMaxRefsToLog() {
return currentConfig.getMaxRefsToLog();
}
@Override
public synchronized boolean isEmpty() {
return currentConfig.isEmpty();
}
@Override
public Path getEventsDirectory() {
return currentConfig.getEventsDirectory();
}
/* shutdown() cannot be set as a synchronized method because
* it may need to wait for pending events to complete;
* e.g. when enabling the drain of replication events before
* shutdown.
*
* As a rule of thumb for synchronized methods, because they
* implicitly define a critical section and associated lock,
* they should never hold waiting for another resource, otherwise
* the risk of deadlock is very high.
*
* See more background about deadlocks, what they are and how to
* prevent them at: https://en.wikipedia.org/wiki/Deadlock
*/
@Override
public int shutdown() {
this.shuttingDown = true;
if (autoReloadRunnable != null) {
autoReloadRunnable.cancel(false);
autoReloadRunnable = null;
}
return currentConfig.shutdown();
}
@Override
public synchronized void startup(WorkQueue workQueue) {
shuttingDown = false;
currentConfig.startup(workQueue);
autoReloadRunnable =
autoReloadExecutor.scheduleAtFixedRate(
this::reloadIfNeeded, RELOAD_DELAY, RELOAD_INTERVAL, TimeUnit.SECONDS);
}
@Override
public synchronized int getSshConnectionTimeout() {
return currentConfig.getSshConnectionTimeout();
}
@Override
public synchronized int getSshCommandTimeout() {
return currentConfig.getSshCommandTimeout();
}
}