Merge branch 'stable-2.14'
* stable-2.14: Index: Extract common parts into AbstractVersionManager ES: Adds configuration for the JEST client Change-Id: Idcd918450d6ab51fb71f68d949f77e35494043ee
This commit is contained in:
@@ -2566,7 +2566,7 @@ values are:
|
|||||||
A link:http://lucene.apache.org/[Lucene] index is used.
|
A link:http://lucene.apache.org/[Lucene] index is used.
|
||||||
+
|
+
|
||||||
+
|
+
|
||||||
* `ELASTICSEARCH`
|
* `ELASTICSEARCH` look into link:#elasticsearch[Elasticsearch section]
|
||||||
+
|
+
|
||||||
An link:http://www.elasticsearch.org/[Elasticsearch] index is used.
|
An link:http://www.elasticsearch.org/[Elasticsearch] index is used.
|
||||||
|
|
||||||
@@ -2706,21 +2706,16 @@ Sample Lucene index configuration:
|
|||||||
maxBufferedDocs = 500
|
maxBufferedDocs = 500
|
||||||
----
|
----
|
||||||
|
|
||||||
|
[[elasticsearch]]
|
||||||
==== Elasticsearch configuration
|
=== Section elasticsearch
|
||||||
|
|
||||||
WARNING: The Elasticsearch support is incomplete. Online reindexing
|
WARNING: The Elasticsearch support is incomplete. Online reindexing
|
||||||
is not implemented yet.
|
is still considered as beta.
|
||||||
|
|
||||||
Open and closed changes are indexed in a single index, separated
|
Open and closed changes are indexed in a single index, separated
|
||||||
into types 'open_changes' and 'closed_changes' respectively.
|
into types 'open_changes' and 'closed_changes' respectively.
|
||||||
|
|
||||||
The following settings are only used when the index type is
|
[[elasticsearch.prefix]]elasticsearch.prefix::
|
||||||
`ELASTICSEARCH`. To configure Elasticsearch servers look into
|
|
||||||
link:#elasticsearch[Elasticsearch section]
|
|
||||||
|
|
||||||
|
|
||||||
[[index.prefix]]index.prefix::
|
|
||||||
+
|
+
|
||||||
This setting can be used to prefix index names to allow multiple Gerrit
|
This setting can be used to prefix index names to allow multiple Gerrit
|
||||||
instances in a single Elasticsearch cluster. Prefix 'gerrit1_' would result in a
|
instances in a single Elasticsearch cluster. Prefix 'gerrit1_' would result in a
|
||||||
@@ -2728,13 +2723,57 @@ change index named 'gerrit1_changes_0001'.
|
|||||||
+
|
+
|
||||||
Not set by default.
|
Not set by default.
|
||||||
|
|
||||||
[[elasticsearch]]
|
[[elasticsearch.username]]elasticsearch.username::
|
||||||
=== Section elasticsearch
|
+
|
||||||
|
Username used to connect to Elasticsearch.
|
||||||
|
+
|
||||||
|
Not set by default.
|
||||||
|
|
||||||
WARNING: The Elasticsearch support is incomplete. Online reindexing
|
[[elasticsearch.password]]elasticsearch.password::
|
||||||
is not implemented yet.
|
+
|
||||||
|
Password used to connect to Elasticsearch.
|
||||||
|
+
|
||||||
|
Not set by default.
|
||||||
|
|
||||||
Each section correspond to the one Elasticsearch server.
|
[[elasticsearch.requestCompression]]elasticsearch.requestCompression::
|
||||||
|
+
|
||||||
|
Enable request compression.
|
||||||
|
+
|
||||||
|
Defaults to `false`.
|
||||||
|
|
||||||
|
[[elasticsearch.connectionTimeout]]elasticsearch.connectionTimeout::
|
||||||
|
+
|
||||||
|
How long should Gerrit waits for connection.
|
||||||
|
+
|
||||||
|
The value is in the usual time-unit format like "1 m", "5 m".
|
||||||
|
+
|
||||||
|
Defaults to `5 m`
|
||||||
|
|
||||||
|
[[elasticsearch.maxConnectionIdleTime]]elasticsearch.maxConnectionIdleTime::
|
||||||
|
+
|
||||||
|
How long connection can stay in idle.
|
||||||
|
+
|
||||||
|
The value is in the usual time-unit format like "1 m", "5 m".
|
||||||
|
+
|
||||||
|
Defaults to `5 m`
|
||||||
|
|
||||||
|
[[elasticsearch.maxTotalConnection]]elasticsearch.maxTotalConnection::
|
||||||
|
+
|
||||||
|
How many connections can be spawn simultaneously.
|
||||||
|
+
|
||||||
|
Defaults to `1`
|
||||||
|
|
||||||
|
[[elasticsearch.maxReadTimeout]]elasticsearch.maxReadTimeout::
|
||||||
|
+
|
||||||
|
Timeout for the read operation.
|
||||||
|
+
|
||||||
|
The value is in the usual time-unit format like "1 m", "5 m".
|
||||||
|
+
|
||||||
|
Defaults to `5 m`
|
||||||
|
|
||||||
|
===== Elasticsearch server(s) configuration
|
||||||
|
|
||||||
|
Each section correspond to one Elasticsearch server.
|
||||||
|
|
||||||
[[elasticsearch.name.protocol]]elasticsearch.name.protocol::
|
[[elasticsearch.name.protocol]]elasticsearch.name.protocol::
|
||||||
+
|
+
|
||||||
|
|||||||
@@ -83,7 +83,7 @@ abstract class AbstractElasticIndex<K, V> implements Index<K, V> {
|
|||||||
this.indexName =
|
this.indexName =
|
||||||
String.format(
|
String.format(
|
||||||
"%s%s%04d",
|
"%s%s%04d",
|
||||||
Strings.nullToEmpty(cfg.getString("index", null, "prefix")),
|
Strings.nullToEmpty(cfg.getString("elasticsearch", null, "prefix")),
|
||||||
indexName,
|
indexName,
|
||||||
schema.getVersion());
|
schema.getVersion());
|
||||||
this.client = clientBuilder.build();
|
this.client = clientBuilder.build();
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ import java.util.ArrayList;
|
|||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
import org.eclipse.jgit.lib.Config;
|
import org.eclipse.jgit.lib.Config;
|
||||||
|
|
||||||
@Singleton
|
@Singleton
|
||||||
@@ -33,9 +34,29 @@ class ElasticConfiguration {
|
|||||||
private static final String DEFAULT_PROTOCOL = "http";
|
private static final String DEFAULT_PROTOCOL = "http";
|
||||||
|
|
||||||
final List<String> urls;
|
final List<String> urls;
|
||||||
|
final String username;
|
||||||
|
final String password;
|
||||||
|
final boolean requestCompression;
|
||||||
|
final long connectionTimeout;
|
||||||
|
final long maxConnectionIdleTime;
|
||||||
|
final TimeUnit maxConnectionIdleUnit = TimeUnit.MILLISECONDS;
|
||||||
|
final int maxTotalConnection;
|
||||||
|
final int readTimeout;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
ElasticConfiguration(@GerritServerConfig Config cfg) {
|
ElasticConfiguration(@GerritServerConfig Config cfg) {
|
||||||
|
this.username = cfg.getString("elasticsearch", null, "username");
|
||||||
|
this.password = cfg.getString("elasticsearch", null, "password");
|
||||||
|
this.requestCompression = cfg.getBoolean("elasticsearch", null, "requestCompression", false);
|
||||||
|
this.connectionTimeout =
|
||||||
|
cfg.getTimeUnit("elasticsearch", null, "connectionTimeout", 3000, TimeUnit.MILLISECONDS);
|
||||||
|
this.maxConnectionIdleTime =
|
||||||
|
cfg.getTimeUnit(
|
||||||
|
"elasticsearch", null, "maxConnectionIdleTime", 3000, TimeUnit.MILLISECONDS);
|
||||||
|
this.maxTotalConnection = cfg.getInt("elasticsearch", null, "maxTotalConnection", 1);
|
||||||
|
this.readTimeout =
|
||||||
|
(int) cfg.getTimeUnit("elasticsearch", null, "readTimeout", 3000, TimeUnit.MICROSECONDS);
|
||||||
|
|
||||||
Set<String> subsections = cfg.getSubsections("elasticsearch");
|
Set<String> subsections = cfg.getSubsections("elasticsearch");
|
||||||
if (subsections.isEmpty()) {
|
if (subsections.isEmpty()) {
|
||||||
this.urls = Arrays.asList(buildUrl(DEFAULT_PROTOCOL, DEFAULT_HOST, DEFAULT_PORT));
|
this.urls = Arrays.asList(buildUrl(DEFAULT_PROTOCOL, DEFAULT_HOST, DEFAULT_PORT));
|
||||||
|
|||||||
@@ -14,59 +14,31 @@
|
|||||||
|
|
||||||
package com.google.gerrit.elasticsearch;
|
package com.google.gerrit.elasticsearch;
|
||||||
|
|
||||||
import static com.google.common.base.Preconditions.checkArgument;
|
|
||||||
|
|
||||||
import com.google.common.base.MoreObjects;
|
import com.google.common.base.MoreObjects;
|
||||||
import com.google.common.collect.Lists;
|
|
||||||
import com.google.common.collect.Maps;
|
|
||||||
import com.google.common.primitives.Ints;
|
import com.google.common.primitives.Ints;
|
||||||
import com.google.gerrit.extensions.events.LifecycleListener;
|
import com.google.gerrit.extensions.events.LifecycleListener;
|
||||||
import com.google.gerrit.server.config.GerritServerConfig;
|
import com.google.gerrit.server.config.GerritServerConfig;
|
||||||
import com.google.gerrit.server.config.SitePaths;
|
import com.google.gerrit.server.config.SitePaths;
|
||||||
|
import com.google.gerrit.server.index.AbstractVersionManager;
|
||||||
|
import com.google.gerrit.server.index.GerritIndexStatus;
|
||||||
import com.google.gerrit.server.index.Index;
|
import com.google.gerrit.server.index.Index;
|
||||||
import com.google.gerrit.server.index.IndexCollection;
|
|
||||||
import com.google.gerrit.server.index.IndexDefinition;
|
import com.google.gerrit.server.index.IndexDefinition;
|
||||||
import com.google.gerrit.server.index.IndexDefinition.IndexFactory;
|
|
||||||
import com.google.gerrit.server.index.IndexUtils;
|
|
||||||
import com.google.gerrit.server.index.OnlineReindexer;
|
|
||||||
import com.google.gerrit.server.index.ReindexerAlreadyRunningException;
|
|
||||||
import com.google.gerrit.server.index.Schema;
|
import com.google.gerrit.server.index.Schema;
|
||||||
import com.google.inject.Inject;
|
import com.google.inject.Inject;
|
||||||
import com.google.inject.ProvisionException;
|
|
||||||
import com.google.inject.Singleton;
|
import com.google.inject.Singleton;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.TreeMap;
|
import java.util.TreeMap;
|
||||||
import org.eclipse.jgit.lib.Config;
|
import org.eclipse.jgit.lib.Config;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
@Singleton
|
@Singleton
|
||||||
public class ElasticVersionManager implements LifecycleListener {
|
public class ElasticVersionManager extends AbstractVersionManager implements LifecycleListener {
|
||||||
private static final Logger log = LoggerFactory.getLogger(ElasticVersionManager.class);
|
private static final Logger log = LoggerFactory.getLogger(ElasticVersionManager.class);
|
||||||
|
|
||||||
private static class Version<V> {
|
|
||||||
private final Schema<V> schema;
|
|
||||||
private final int version;
|
|
||||||
private final boolean ready;
|
|
||||||
|
|
||||||
private Version(Schema<V> schema, int version, boolean ready) {
|
|
||||||
checkArgument(schema == null || schema.getVersion() == version);
|
|
||||||
this.schema = schema;
|
|
||||||
this.version = version;
|
|
||||||
this.ready = ready;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private final Map<String, IndexDefinition<?, ?, ?>> defs;
|
|
||||||
private final Map<String, OnlineReindexer<?, ?, ?>> reindexers;
|
|
||||||
private final ElasticIndexVersionDiscovery versionDiscovery;
|
|
||||||
private final SitePaths sitePaths;
|
|
||||||
private final boolean onlineUpgrade;
|
|
||||||
private final String runReindexMsg;
|
|
||||||
private final String prefix;
|
private final String prefix;
|
||||||
|
private final ElasticIndexVersionDiscovery versionDiscovery;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
ElasticVersionManager(
|
ElasticVersionManager(
|
||||||
@@ -74,147 +46,23 @@ public class ElasticVersionManager implements LifecycleListener {
|
|||||||
SitePaths sitePaths,
|
SitePaths sitePaths,
|
||||||
Collection<IndexDefinition<?, ?, ?>> defs,
|
Collection<IndexDefinition<?, ?, ?>> defs,
|
||||||
ElasticIndexVersionDiscovery versionDiscovery) {
|
ElasticIndexVersionDiscovery versionDiscovery) {
|
||||||
this.sitePaths = sitePaths;
|
super(cfg, sitePaths, defs);
|
||||||
this.versionDiscovery = versionDiscovery;
|
this.versionDiscovery = versionDiscovery;
|
||||||
this.defs = Maps.newHashMapWithExpectedSize(defs.size());
|
|
||||||
for (IndexDefinition<?, ?, ?> def : defs) {
|
|
||||||
this.defs.put(def.getName(), def);
|
|
||||||
}
|
|
||||||
|
|
||||||
prefix = MoreObjects.firstNonNull(cfg.getString("index", null, "prefix"), "gerrit");
|
prefix = MoreObjects.firstNonNull(cfg.getString("index", null, "prefix"), "gerrit");
|
||||||
reindexers = Maps.newHashMapWithExpectedSize(defs.size());
|
|
||||||
onlineUpgrade = cfg.getBoolean("index", null, "onlineUpgrade", true);
|
|
||||||
runReindexMsg =
|
|
||||||
"No index versions ready; run java -jar "
|
|
||||||
+ sitePaths.gerrit_war.toAbsolutePath()
|
|
||||||
+ " reindex";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void start() {
|
protected <V> boolean isDirty(Collection<Version<V>> inUse, Version<V> v) {
|
||||||
try {
|
return !inUse.contains(v);
|
||||||
for (IndexDefinition<?, ?, ?> def : defs.values()) {
|
|
||||||
initIndex(def);
|
|
||||||
}
|
|
||||||
} catch (IOException e) {
|
|
||||||
ProvisionException ex = new ProvisionException("Error scanning indexes");
|
|
||||||
ex.initCause(e);
|
|
||||||
throw ex;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private <K, V, I extends Index<K, V>> void initIndex(IndexDefinition<K, V, I> def)
|
@Override
|
||||||
throws IOException {
|
protected <K, V, I extends Index<K, V>> TreeMap<Integer, Version<V>> scanVersions(
|
||||||
TreeMap<Integer, Version<V>> versions = scanVersions(def);
|
IndexDefinition<K, V, I> def, GerritIndexStatus cfg) {
|
||||||
// Search from the most recent ready version.
|
|
||||||
// Write to the most recent ready version and the most recent version.
|
|
||||||
Version<V> search = null;
|
|
||||||
List<Version<V>> write = Lists.newArrayListWithCapacity(2);
|
|
||||||
for (Version<V> v : versions.descendingMap().values()) {
|
|
||||||
if (v.schema == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (write.isEmpty() && onlineUpgrade) {
|
|
||||||
write.add(v);
|
|
||||||
}
|
|
||||||
if (v.ready) {
|
|
||||||
search = v;
|
|
||||||
if (!write.contains(v)) {
|
|
||||||
write.add(v);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (search == null) {
|
|
||||||
throw new ProvisionException(runReindexMsg);
|
|
||||||
}
|
|
||||||
|
|
||||||
IndexFactory<K, V, I> factory = def.getIndexFactory();
|
|
||||||
I searchIndex = factory.create(search.schema);
|
|
||||||
IndexCollection<K, V, I> indexes = def.getIndexCollection();
|
|
||||||
indexes.setSearchIndex(searchIndex);
|
|
||||||
for (Version<V> v : write) {
|
|
||||||
if (v.schema != null) {
|
|
||||||
if (v.version != search.version) {
|
|
||||||
indexes.addWriteIndex(factory.create(v.schema));
|
|
||||||
} else {
|
|
||||||
indexes.addWriteIndex(searchIndex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
markNotReady(def.getName(), versions.values(), write);
|
|
||||||
|
|
||||||
synchronized (this) {
|
|
||||||
if (!reindexers.containsKey(def.getName())) {
|
|
||||||
int latest = write.get(0).version;
|
|
||||||
OnlineReindexer<K, V, I> reindexer = new OnlineReindexer<>(def, latest);
|
|
||||||
reindexers.put(def.getName(), reindexer);
|
|
||||||
if (onlineUpgrade && latest != search.version) {
|
|
||||||
reindexer.start();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Start the online reindexer if the current index is not already the latest.
|
|
||||||
*
|
|
||||||
* @param name index name
|
|
||||||
* @param force start re-index
|
|
||||||
* @return true if started, otherwise false.
|
|
||||||
* @throws ReindexerAlreadyRunningException
|
|
||||||
*/
|
|
||||||
public synchronized boolean startReindexer(String name, boolean force)
|
|
||||||
throws ReindexerAlreadyRunningException {
|
|
||||||
OnlineReindexer<?, ?, ?> reindexer = reindexers.get(name);
|
|
||||||
validateReindexerNotRunning(reindexer);
|
|
||||||
if (force || !isLatestIndexVersion(name, reindexer)) {
|
|
||||||
reindexer.start();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Activate the latest index if the current index is not already the latest.
|
|
||||||
*
|
|
||||||
* @param name index name
|
|
||||||
* @return true if index was activated, otherwise false.
|
|
||||||
* @throws ReindexerAlreadyRunningException
|
|
||||||
*/
|
|
||||||
public synchronized boolean activateLatestIndex(String name)
|
|
||||||
throws ReindexerAlreadyRunningException {
|
|
||||||
OnlineReindexer<?, ?, ?> reindexer = reindexers.get(name);
|
|
||||||
validateReindexerNotRunning(reindexer);
|
|
||||||
if (!isLatestIndexVersion(name, reindexer)) {
|
|
||||||
reindexer.activateIndex();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean isLatestIndexVersion(String name, OnlineReindexer<?, ?, ?> reindexer) {
|
|
||||||
int readVersion = defs.get(name).getIndexCollection().getSearchIndex().getSchema().getVersion();
|
|
||||||
return reindexer == null || reindexer.getVersion() == readVersion;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void validateReindexerNotRunning(OnlineReindexer<?, ?, ?> reindexer)
|
|
||||||
throws ReindexerAlreadyRunningException {
|
|
||||||
if (reindexer != null && reindexer.isRunning()) {
|
|
||||||
throw new ReindexerAlreadyRunningException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private <K, V, I extends Index<K, V>> TreeMap<Integer, Version<V>> scanVersions(
|
|
||||||
IndexDefinition<K, V, I> def) throws IOException {
|
|
||||||
TreeMap<Integer, Version<V>> versions = new TreeMap<>();
|
TreeMap<Integer, Version<V>> versions = new TreeMap<>();
|
||||||
for (Schema<V> schema : def.getSchemas().values()) {
|
for (Schema<V> schema : def.getSchemas().values()) {
|
||||||
int v = schema.getVersion();
|
int v = schema.getVersion();
|
||||||
versions.put(
|
versions.put(v, new Version<>(schema, v, cfg.getReady(def.getName(), v)));
|
||||||
v,
|
|
||||||
new Version<>(
|
|
||||||
schema, v, IndexUtils.getReady(sitePaths, def.getName(), schema.getVersion())));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -225,8 +73,7 @@ public class ElasticVersionManager implements LifecycleListener {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (!versions.containsKey(v)) {
|
if (!versions.containsKey(v)) {
|
||||||
versions.put(
|
versions.put(v, new Version<V>(null, v, cfg.getReady(def.getName(), v)));
|
||||||
v, new Version<V>(null, v, IndexUtils.getReady(sitePaths, def.getName(), v)));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@@ -234,18 +81,4 @@ public class ElasticVersionManager implements LifecycleListener {
|
|||||||
}
|
}
|
||||||
return versions;
|
return versions;
|
||||||
}
|
}
|
||||||
|
|
||||||
private <V> void markNotReady(
|
|
||||||
String name, Iterable<Version<V>> versions, Collection<Version<V>> inUse) throws IOException {
|
|
||||||
for (Version<V> v : versions) {
|
|
||||||
if (!inUse.contains(v)) {
|
|
||||||
IndexUtils.getReady(sitePaths, name, v.version);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void stop() {
|
|
||||||
// Do nothing; indexes are closed on demand by IndexCollection.
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ import com.google.inject.Inject;
|
|||||||
import com.google.inject.Singleton;
|
import com.google.inject.Singleton;
|
||||||
import io.searchbox.client.JestClientFactory;
|
import io.searchbox.client.JestClientFactory;
|
||||||
import io.searchbox.client.config.HttpClientConfig;
|
import io.searchbox.client.config.HttpClientConfig;
|
||||||
|
import io.searchbox.client.config.HttpClientConfig.Builder;
|
||||||
import io.searchbox.client.http.JestHttpClient;
|
import io.searchbox.client.http.JestHttpClient;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
@@ -32,12 +33,22 @@ class JestClientBuilder {
|
|||||||
|
|
||||||
JestHttpClient build() {
|
JestHttpClient build() {
|
||||||
JestClientFactory factory = new JestClientFactory();
|
JestClientFactory factory = new JestClientFactory();
|
||||||
factory.setHttpClientConfig(
|
Builder builder =
|
||||||
new HttpClientConfig.Builder(cfg.urls)
|
new HttpClientConfig.Builder(cfg.urls)
|
||||||
.multiThreaded(true)
|
.multiThreaded(true)
|
||||||
.discoveryEnabled(false)
|
.discoveryEnabled(false)
|
||||||
.discoveryFrequency(1L, TimeUnit.MINUTES)
|
.connTimeout((int) cfg.connectionTimeout)
|
||||||
.build());
|
.maxConnectionIdleTime(cfg.maxConnectionIdleTime, cfg.maxConnectionIdleUnit)
|
||||||
|
.maxTotalConnection(cfg.maxTotalConnection)
|
||||||
|
.readTimeout(cfg.readTimeout)
|
||||||
|
.requestCompressionEnabled(cfg.requestCompression)
|
||||||
|
.discoveryFrequency(1L, TimeUnit.MINUTES);
|
||||||
|
|
||||||
|
if (cfg.username != null && cfg.password != null) {
|
||||||
|
builder.defaultCredentials(cfg.username, cfg.password);
|
||||||
|
}
|
||||||
|
|
||||||
|
factory.setHttpClientConfig(builder.build());
|
||||||
return (JestHttpClient) factory.getObject();
|
return (JestHttpClient) factory.getObject();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,54 +14,37 @@
|
|||||||
|
|
||||||
package com.google.gerrit.lucene;
|
package com.google.gerrit.lucene;
|
||||||
|
|
||||||
import static com.google.common.base.Preconditions.checkArgument;
|
|
||||||
|
|
||||||
import com.google.common.collect.Lists;
|
|
||||||
import com.google.common.collect.Maps;
|
|
||||||
import com.google.common.primitives.Ints;
|
import com.google.common.primitives.Ints;
|
||||||
import com.google.gerrit.extensions.events.LifecycleListener;
|
import com.google.gerrit.extensions.events.LifecycleListener;
|
||||||
import com.google.gerrit.server.config.GerritServerConfig;
|
import com.google.gerrit.server.config.GerritServerConfig;
|
||||||
import com.google.gerrit.server.config.SitePaths;
|
import com.google.gerrit.server.config.SitePaths;
|
||||||
|
import com.google.gerrit.server.index.AbstractVersionManager;
|
||||||
import com.google.gerrit.server.index.GerritIndexStatus;
|
import com.google.gerrit.server.index.GerritIndexStatus;
|
||||||
import com.google.gerrit.server.index.Index;
|
import com.google.gerrit.server.index.Index;
|
||||||
import com.google.gerrit.server.index.IndexCollection;
|
|
||||||
import com.google.gerrit.server.index.IndexDefinition;
|
import com.google.gerrit.server.index.IndexDefinition;
|
||||||
import com.google.gerrit.server.index.IndexDefinition.IndexFactory;
|
|
||||||
import com.google.gerrit.server.index.OnlineReindexer;
|
|
||||||
import com.google.gerrit.server.index.ReindexerAlreadyRunningException;
|
|
||||||
import com.google.gerrit.server.index.Schema;
|
import com.google.gerrit.server.index.Schema;
|
||||||
import com.google.inject.Inject;
|
import com.google.inject.Inject;
|
||||||
import com.google.inject.ProvisionException;
|
|
||||||
import com.google.inject.Singleton;
|
import com.google.inject.Singleton;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.nio.file.DirectoryStream;
|
import java.nio.file.DirectoryStream;
|
||||||
import java.nio.file.Files;
|
import java.nio.file.Files;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.TreeMap;
|
import java.util.TreeMap;
|
||||||
import org.eclipse.jgit.errors.ConfigInvalidException;
|
|
||||||
import org.eclipse.jgit.lib.Config;
|
import org.eclipse.jgit.lib.Config;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
@Singleton
|
@Singleton
|
||||||
public class LuceneVersionManager implements LifecycleListener {
|
public class LuceneVersionManager extends AbstractVersionManager implements LifecycleListener {
|
||||||
private static final Logger log = LoggerFactory.getLogger(LuceneVersionManager.class);
|
private static final Logger log = LoggerFactory.getLogger(LuceneVersionManager.class);
|
||||||
|
|
||||||
private static class Version<V> {
|
private static class Version<V> extends AbstractVersionManager.Version<V> {
|
||||||
private final Schema<V> schema;
|
|
||||||
private final int version;
|
|
||||||
private final boolean exists;
|
private final boolean exists;
|
||||||
private final boolean ready;
|
|
||||||
|
|
||||||
private Version(Schema<V> schema, int version, boolean exists, boolean ready) {
|
private Version(Schema<V> schema, int version, boolean exists, boolean ready) {
|
||||||
checkArgument(schema == null || schema.getVersion() == version);
|
super(schema, version, ready);
|
||||||
this.schema = schema;
|
|
||||||
this.version = version;
|
|
||||||
this.exists = exists;
|
this.exists = exists;
|
||||||
this.ready = ready;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -69,156 +52,26 @@ public class LuceneVersionManager implements LifecycleListener {
|
|||||||
return sitePaths.index_dir.resolve(String.format("%s%04d", prefix, schema.getVersion()));
|
return sitePaths.index_dir.resolve(String.format("%s%04d", prefix, schema.getVersion()));
|
||||||
}
|
}
|
||||||
|
|
||||||
private final SitePaths sitePaths;
|
|
||||||
private final Map<String, IndexDefinition<?, ?, ?>> defs;
|
|
||||||
private final Map<String, OnlineReindexer<?, ?, ?>> reindexers;
|
|
||||||
private final boolean onlineUpgrade;
|
|
||||||
private final String runReindexMsg;
|
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
LuceneVersionManager(
|
LuceneVersionManager(
|
||||||
@GerritServerConfig Config cfg,
|
@GerritServerConfig Config cfg,
|
||||||
SitePaths sitePaths,
|
SitePaths sitePaths,
|
||||||
Collection<IndexDefinition<?, ?, ?>> defs) {
|
Collection<IndexDefinition<?, ?, ?>> defs) {
|
||||||
this.sitePaths = sitePaths;
|
super(cfg, sitePaths, defs);
|
||||||
this.defs = Maps.newHashMapWithExpectedSize(defs.size());
|
|
||||||
for (IndexDefinition<?, ?, ?> def : defs) {
|
|
||||||
this.defs.put(def.getName(), def);
|
|
||||||
}
|
|
||||||
|
|
||||||
reindexers = Maps.newHashMapWithExpectedSize(defs.size());
|
|
||||||
onlineUpgrade = cfg.getBoolean("index", null, "onlineUpgrade", true);
|
|
||||||
runReindexMsg =
|
|
||||||
"No index versions ready; run java -jar "
|
|
||||||
+ sitePaths.gerrit_war.toAbsolutePath()
|
|
||||||
+ " reindex";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void start() {
|
protected <V> boolean isDirty(
|
||||||
GerritIndexStatus cfg;
|
Collection<com.google.gerrit.server.index.AbstractVersionManager.Version<V>> inUse,
|
||||||
try {
|
com.google.gerrit.server.index.AbstractVersionManager.Version<V> v) {
|
||||||
cfg = new GerritIndexStatus(sitePaths);
|
return !inUse.contains(v) && ((Version<V>) v).exists;
|
||||||
} catch (ConfigInvalidException | IOException e) {
|
|
||||||
throw fail(e);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!Files.exists(sitePaths.index_dir)) {
|
@Override
|
||||||
throw new ProvisionException(runReindexMsg);
|
protected <K, V, I extends Index<K, V>>
|
||||||
} else if (!Files.exists(sitePaths.index_dir)) {
|
TreeMap<Integer, AbstractVersionManager.Version<V>> scanVersions(
|
||||||
log.warn("Not a directory: %s", sitePaths.index_dir.toAbsolutePath());
|
|
||||||
throw new ProvisionException(runReindexMsg);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (IndexDefinition<?, ?, ?> def : defs.values()) {
|
|
||||||
initIndex(def, cfg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private <K, V, I extends Index<K, V>> void initIndex(
|
|
||||||
IndexDefinition<K, V, I> def, GerritIndexStatus cfg) {
|
IndexDefinition<K, V, I> def, GerritIndexStatus cfg) {
|
||||||
TreeMap<Integer, Version<V>> versions = scanVersions(def, cfg);
|
TreeMap<Integer, AbstractVersionManager.Version<V>> versions = new TreeMap<>();
|
||||||
// Search from the most recent ready version.
|
|
||||||
// Write to the most recent ready version and the most recent version.
|
|
||||||
Version<V> search = null;
|
|
||||||
List<Version<V>> write = Lists.newArrayListWithCapacity(2);
|
|
||||||
for (Version<V> v : versions.descendingMap().values()) {
|
|
||||||
if (v.schema == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (write.isEmpty() && onlineUpgrade) {
|
|
||||||
write.add(v);
|
|
||||||
}
|
|
||||||
if (v.ready) {
|
|
||||||
search = v;
|
|
||||||
if (!write.contains(v)) {
|
|
||||||
write.add(v);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (search == null) {
|
|
||||||
throw new ProvisionException(runReindexMsg);
|
|
||||||
}
|
|
||||||
|
|
||||||
IndexFactory<K, V, I> factory = def.getIndexFactory();
|
|
||||||
I searchIndex = factory.create(search.schema);
|
|
||||||
IndexCollection<K, V, I> indexes = def.getIndexCollection();
|
|
||||||
indexes.setSearchIndex(searchIndex);
|
|
||||||
for (Version<V> v : write) {
|
|
||||||
if (v.schema != null) {
|
|
||||||
if (v.version != search.version) {
|
|
||||||
indexes.addWriteIndex(factory.create(v.schema));
|
|
||||||
} else {
|
|
||||||
indexes.addWriteIndex(searchIndex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
markNotReady(cfg, def.getName(), versions.values(), write);
|
|
||||||
|
|
||||||
int latest = write.get(0).version;
|
|
||||||
OnlineReindexer<K, V, I> reindexer = new OnlineReindexer<>(def, latest);
|
|
||||||
synchronized (this) {
|
|
||||||
if (!reindexers.containsKey(def.getName())) {
|
|
||||||
reindexers.put(def.getName(), reindexer);
|
|
||||||
if (onlineUpgrade && latest != search.version) {
|
|
||||||
reindexer.start();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Start the online reindexer if the current index is not already the latest.
|
|
||||||
*
|
|
||||||
* @param force start re-index
|
|
||||||
* @return true if started, otherwise false.
|
|
||||||
* @throws ReindexerAlreadyRunningException
|
|
||||||
*/
|
|
||||||
public synchronized boolean startReindexer(String name, boolean force)
|
|
||||||
throws ReindexerAlreadyRunningException {
|
|
||||||
OnlineReindexer<?, ?, ?> reindexer = reindexers.get(name);
|
|
||||||
validateReindexerNotRunning(reindexer);
|
|
||||||
if (force || !isCurrentIndexVersionLatest(name, reindexer)) {
|
|
||||||
reindexer.start();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Activate the latest index if the current index is not already the latest.
|
|
||||||
*
|
|
||||||
* @return true if index was activate, otherwise false.
|
|
||||||
* @throws ReindexerAlreadyRunningException
|
|
||||||
*/
|
|
||||||
public synchronized boolean activateLatestIndex(String name)
|
|
||||||
throws ReindexerAlreadyRunningException {
|
|
||||||
OnlineReindexer<?, ?, ?> reindexer = reindexers.get(name);
|
|
||||||
validateReindexerNotRunning(reindexer);
|
|
||||||
if (!isCurrentIndexVersionLatest(name, reindexer)) {
|
|
||||||
reindexer.activateIndex();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean isCurrentIndexVersionLatest(String name, OnlineReindexer<?, ?, ?> reindexer) {
|
|
||||||
int readVersion = defs.get(name).getIndexCollection().getSearchIndex().getSchema().getVersion();
|
|
||||||
return reindexer == null || reindexer.getVersion() == readVersion;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void validateReindexerNotRunning(OnlineReindexer<?, ?, ?> reindexer)
|
|
||||||
throws ReindexerAlreadyRunningException {
|
|
||||||
if (reindexer != null && reindexer.isRunning()) {
|
|
||||||
throw new ReindexerAlreadyRunningException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private <K, V, I extends Index<K, V>> TreeMap<Integer, Version<V>> scanVersions(
|
|
||||||
IndexDefinition<K, V, I> def, GerritIndexStatus cfg) {
|
|
||||||
TreeMap<Integer, Version<V>> versions = new TreeMap<>();
|
|
||||||
for (Schema<V> schema : def.getSchemas().values()) {
|
for (Schema<V> schema : def.getSchemas().values()) {
|
||||||
// This part is Lucene-specific.
|
// This part is Lucene-specific.
|
||||||
Path p = getDir(sitePaths, def.getName(), schema);
|
Path p = getDir(sitePaths, def.getName(), schema);
|
||||||
@@ -252,36 +105,4 @@ public class LuceneVersionManager implements LifecycleListener {
|
|||||||
}
|
}
|
||||||
return versions;
|
return versions;
|
||||||
}
|
}
|
||||||
|
|
||||||
private <V> void markNotReady(
|
|
||||||
GerritIndexStatus cfg,
|
|
||||||
String name,
|
|
||||||
Iterable<Version<V>> versions,
|
|
||||||
Collection<Version<V>> inUse) {
|
|
||||||
boolean dirty = false;
|
|
||||||
for (Version<V> v : versions) {
|
|
||||||
if (!inUse.contains(v) && v.exists) {
|
|
||||||
cfg.setReady(name, v.version, false);
|
|
||||||
dirty = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (dirty) {
|
|
||||||
try {
|
|
||||||
cfg.save();
|
|
||||||
} catch (IOException e) {
|
|
||||||
throw fail(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private ProvisionException fail(Throwable t) {
|
|
||||||
ProvisionException e = new ProvisionException("Error scanning indexes");
|
|
||||||
e.initCause(t);
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void stop() {
|
|
||||||
// Do nothing; indexes are closed on demand by IndexCollection.
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,223 @@
|
|||||||
|
// 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.index;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkArgument;
|
||||||
|
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
|
import com.google.common.collect.Maps;
|
||||||
|
import com.google.gerrit.extensions.events.LifecycleListener;
|
||||||
|
import com.google.gerrit.server.config.GerritServerConfig;
|
||||||
|
import com.google.gerrit.server.config.SitePaths;
|
||||||
|
import com.google.gerrit.server.index.IndexDefinition.IndexFactory;
|
||||||
|
import com.google.inject.ProvisionException;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.TreeMap;
|
||||||
|
import org.eclipse.jgit.errors.ConfigInvalidException;
|
||||||
|
import org.eclipse.jgit.lib.Config;
|
||||||
|
|
||||||
|
public abstract class AbstractVersionManager implements LifecycleListener {
|
||||||
|
public static class Version<V> {
|
||||||
|
public final Schema<V> schema;
|
||||||
|
public final int version;
|
||||||
|
public final boolean ready;
|
||||||
|
|
||||||
|
public Version(Schema<V> schema, int version, boolean ready) {
|
||||||
|
checkArgument(schema == null || schema.getVersion() == version);
|
||||||
|
this.schema = schema;
|
||||||
|
this.version = version;
|
||||||
|
this.ready = ready;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected final boolean onlineUpgrade;
|
||||||
|
protected final String runReindexMsg;
|
||||||
|
protected final SitePaths sitePaths;
|
||||||
|
protected final Map<String, IndexDefinition<?, ?, ?>> defs;
|
||||||
|
protected final Map<String, OnlineReindexer<?, ?, ?>> reindexers;
|
||||||
|
|
||||||
|
protected AbstractVersionManager(
|
||||||
|
@GerritServerConfig Config cfg,
|
||||||
|
SitePaths sitePaths,
|
||||||
|
Collection<IndexDefinition<?, ?, ?>> defs) {
|
||||||
|
this.sitePaths = sitePaths;
|
||||||
|
this.defs = Maps.newHashMapWithExpectedSize(defs.size());
|
||||||
|
for (IndexDefinition<?, ?, ?> def : defs) {
|
||||||
|
this.defs.put(def.getName(), def);
|
||||||
|
}
|
||||||
|
|
||||||
|
reindexers = Maps.newHashMapWithExpectedSize(defs.size());
|
||||||
|
onlineUpgrade = cfg.getBoolean("index", null, "onlineUpgrade", true);
|
||||||
|
runReindexMsg =
|
||||||
|
"No index versions ready; run java -jar "
|
||||||
|
+ sitePaths.gerrit_war.toAbsolutePath()
|
||||||
|
+ " reindex";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void start() {
|
||||||
|
GerritIndexStatus cfg = createIndexStatus();
|
||||||
|
for (IndexDefinition<?, ?, ?> def : defs.values()) {
|
||||||
|
initIndex(def, cfg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void stop() {
|
||||||
|
// Do nothing; indexes are closed on demand by IndexCollection.
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start the online reindexer if the current index is not already the latest.
|
||||||
|
*
|
||||||
|
* @param name index name
|
||||||
|
* @param force start re-index
|
||||||
|
* @return true if started, otherwise false.
|
||||||
|
* @throws ReindexerAlreadyRunningException
|
||||||
|
*/
|
||||||
|
public synchronized boolean startReindexer(String name, boolean force)
|
||||||
|
throws ReindexerAlreadyRunningException {
|
||||||
|
OnlineReindexer<?, ?, ?> reindexer = reindexers.get(name);
|
||||||
|
validateReindexerNotRunning(reindexer);
|
||||||
|
if (force || !isLatestIndexVersion(name, reindexer)) {
|
||||||
|
reindexer.start();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Activate the latest index if the current index is not already the latest.
|
||||||
|
*
|
||||||
|
* @param name index name
|
||||||
|
* @return true if index was activated, otherwise false.
|
||||||
|
* @throws ReindexerAlreadyRunningException
|
||||||
|
*/
|
||||||
|
public synchronized boolean activateLatestIndex(String name)
|
||||||
|
throws ReindexerAlreadyRunningException {
|
||||||
|
OnlineReindexer<?, ?, ?> reindexer = reindexers.get(name);
|
||||||
|
validateReindexerNotRunning(reindexer);
|
||||||
|
if (!isLatestIndexVersion(name, reindexer)) {
|
||||||
|
reindexer.activateIndex();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected <K, V, I extends Index<K, V>> void initIndex(
|
||||||
|
IndexDefinition<K, V, I> def, GerritIndexStatus cfg) {
|
||||||
|
TreeMap<Integer, Version<V>> versions = scanVersions(def, cfg);
|
||||||
|
// Search from the most recent ready version.
|
||||||
|
// Write to the most recent ready version and the most recent version.
|
||||||
|
Version<V> search = null;
|
||||||
|
List<Version<V>> write = Lists.newArrayListWithCapacity(2);
|
||||||
|
for (Version<V> v : versions.descendingMap().values()) {
|
||||||
|
if (v.schema == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (write.isEmpty() && onlineUpgrade) {
|
||||||
|
write.add(v);
|
||||||
|
}
|
||||||
|
if (v.ready) {
|
||||||
|
search = v;
|
||||||
|
if (!write.contains(v)) {
|
||||||
|
write.add(v);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (search == null) {
|
||||||
|
throw new ProvisionException(runReindexMsg);
|
||||||
|
}
|
||||||
|
|
||||||
|
IndexFactory<K, V, I> factory = def.getIndexFactory();
|
||||||
|
I searchIndex = factory.create(search.schema);
|
||||||
|
IndexCollection<K, V, I> indexes = def.getIndexCollection();
|
||||||
|
indexes.setSearchIndex(searchIndex);
|
||||||
|
for (Version<V> v : write) {
|
||||||
|
if (v.version != search.version) {
|
||||||
|
indexes.addWriteIndex(factory.create(v.schema));
|
||||||
|
} else {
|
||||||
|
indexes.addWriteIndex(searchIndex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
markNotReady(def.getName(), versions.values(), write);
|
||||||
|
|
||||||
|
synchronized (this) {
|
||||||
|
if (!reindexers.containsKey(def.getName())) {
|
||||||
|
int latest = write.get(0).version;
|
||||||
|
OnlineReindexer<K, V, I> reindexer = new OnlineReindexer<>(def, latest);
|
||||||
|
reindexers.put(def.getName(), reindexer);
|
||||||
|
if (onlineUpgrade && latest != search.version) {
|
||||||
|
reindexer.start();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected GerritIndexStatus createIndexStatus() {
|
||||||
|
try {
|
||||||
|
return new GerritIndexStatus(sitePaths);
|
||||||
|
} catch (ConfigInvalidException | IOException e) {
|
||||||
|
throw fail(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected abstract <V> boolean isDirty(Collection<Version<V>> inUse, Version<V> v);
|
||||||
|
|
||||||
|
protected abstract <K, V, I extends Index<K, V>> TreeMap<Integer, Version<V>> scanVersions(
|
||||||
|
IndexDefinition<K, V, I> def, GerritIndexStatus cfg);
|
||||||
|
|
||||||
|
private boolean isLatestIndexVersion(String name, OnlineReindexer<?, ?, ?> reindexer) {
|
||||||
|
int readVersion = defs.get(name).getIndexCollection().getSearchIndex().getSchema().getVersion();
|
||||||
|
return reindexer == null || reindexer.getVersion() == readVersion;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void validateReindexerNotRunning(OnlineReindexer<?, ?, ?> reindexer)
|
||||||
|
throws ReindexerAlreadyRunningException {
|
||||||
|
if (reindexer != null && reindexer.isRunning()) {
|
||||||
|
throw new ReindexerAlreadyRunningException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private <V> void markNotReady(
|
||||||
|
String name, Iterable<Version<V>> versions, Collection<Version<V>> inUse) {
|
||||||
|
GerritIndexStatus cfg = createIndexStatus();
|
||||||
|
boolean dirty = false;
|
||||||
|
for (Version<V> v : versions) {
|
||||||
|
if (isDirty(inUse, v)) {
|
||||||
|
cfg.setReady(name, v.version, false);
|
||||||
|
dirty = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (dirty) {
|
||||||
|
try {
|
||||||
|
cfg.save();
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw fail(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private ProvisionException fail(Throwable t) {
|
||||||
|
ProvisionException e = new ProvisionException("Error scanning indexes");
|
||||||
|
e.initCause(t);
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user