SOLR-11255: Use concurrent set for metric names.

This commit is contained in:
Andrzej Bialecki 2017-08-21 15:24:29 +02:00
parent fa1198be73
commit 5d9b9d7a23
14 changed files with 99 additions and 22 deletions

View File

@ -97,6 +97,8 @@ Bug Fixes
* SOLR-11084: Issue with starting script with solr.home (-s) == solr (Leil Ireson, Amrit Sarkar via Erick Erickson) * SOLR-11084: Issue with starting script with solr.home (-s) == solr (Leil Ireson, Amrit Sarkar via Erick Erickson)
* SOLR-11255: Fix occasional ConcurrentModificationException when using SolrInfoMBeanHandler. (ab)
Optimizations Optimizations
---------------------- ----------------------

View File

@ -222,7 +222,7 @@ public final class SolrCore implements SolrInfoBean, SolrMetricProducer, Closeab
private Counter newSearcherOtherErrorsCounter; private Counter newSearcherOtherErrorsCounter;
private final CoreContainer coreContainer; private final CoreContainer coreContainer;
private Set<String> metricNames = new HashSet<>(); private Set<String> metricNames = ConcurrentHashMap.newKeySet();
public Set<String> getMetricNames() { public Set<String> getMetricNames() {
return metricNames; return metricNames;

View File

@ -66,6 +66,7 @@ public interface SolrInfoBean {
* Modifiable set of metric names that this component reports (default is null, * Modifiable set of metric names that this component reports (default is null,
* which means none). If not null then this set is used by {@link #registerMetricName(String)} * which means none). If not null then this set is used by {@link #registerMetricName(String)}
* to capture what metrics names are reported from this component. * to capture what metrics names are reported from this component.
* <p><b>NOTE: this set has to allow iteration under modifications.</b></p>
*/ */
default Set<String> getMetricNames() { default Set<String> getMetricNames() {
return null; return null;

View File

@ -18,8 +18,8 @@ package org.apache.solr.handler;
import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles;
import java.util.Collection; import java.util.Collection;
import java.util.HashSet;
import java.util.Set; import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import com.codahale.metrics.MetricRegistry; import com.codahale.metrics.MetricRegistry;
import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableList;
@ -74,7 +74,7 @@ public abstract class RequestHandlerBase implements SolrRequestHandler, SolrInfo
private PluginInfo pluginInfo; private PluginInfo pluginInfo;
private Set<String> metricNames = new HashSet<>(); private Set<String> metricNames = ConcurrentHashMap.newKeySet();
private MetricRegistry registry; private MetricRegistry registry;
@SuppressForbidden(reason = "Need currentTimeMillis, used only for stats output") @SuppressForbidden(reason = "Need currentTimeMillis, used only for stats output")

View File

@ -19,9 +19,9 @@ package org.apache.solr.handler.component;
import java.io.IOException; import java.io.IOException;
import java.util.Collections; import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet;
import java.util.Map; import java.util.Map;
import java.util.Set; import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import com.codahale.metrics.MetricRegistry; import com.codahale.metrics.MetricRegistry;
import org.apache.solr.common.util.NamedList; import org.apache.solr.common.util.NamedList;
@ -42,7 +42,7 @@ public abstract class SearchComponent implements SolrInfoBean, NamedListInitiali
*/ */
private String name = this.getClass().getName(); private String name = this.getClass().getName();
protected Set<String> metricNames = new HashSet<>(); protected Set<String> metricNames = ConcurrentHashMap.newKeySet();
protected MetricRegistry registry; protected MetricRegistry registry;
/** /**

View File

@ -16,8 +16,8 @@
*/ */
package org.apache.solr.highlight; package org.apache.solr.highlight;
import java.util.HashSet;
import java.util.Set; import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import com.codahale.metrics.Counter; import com.codahale.metrics.Counter;
import com.codahale.metrics.MetricRegistry; import com.codahale.metrics.MetricRegistry;
@ -35,7 +35,7 @@ public abstract class HighlightingPluginBase implements SolrInfoBean, SolrMetric
{ {
protected Counter numRequests; protected Counter numRequests;
protected SolrParams defaults; protected SolrParams defaults;
protected Set<String> metricNames = new HashSet<>(1); protected Set<String> metricNames = ConcurrentHashMap.newKeySet(1);
protected MetricRegistry registry; protected MetricRegistry registry;
public void init(NamedList args) { public void init(NamedList args) {

View File

@ -16,8 +16,8 @@
*/ */
package org.apache.solr.search; package org.apache.solr.search;
import java.util.HashSet;
import java.util.Set; import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import com.codahale.metrics.MetricRegistry; import com.codahale.metrics.MetricRegistry;
import org.apache.solr.core.SolrInfoBean; import org.apache.solr.core.SolrInfoBean;
@ -36,7 +36,7 @@ public class SolrFieldCacheBean implements SolrInfoBean, SolrMetricProducer {
private boolean disableJmxEntryList = Boolean.getBoolean("disableSolrFieldCacheMBeanEntryListJmx"); private boolean disableJmxEntryList = Boolean.getBoolean("disableSolrFieldCacheMBeanEntryListJmx");
private MetricRegistry registry; private MetricRegistry registry;
private Set<String> metricNames = new HashSet<>(); private Set<String> metricNames = ConcurrentHashMap.newKeySet();
@Override @Override
public String getName() { return this.getClass().getName(); } public String getName() { return this.getClass().getName(); }

View File

@ -25,11 +25,11 @@ import java.util.Collections;
import java.util.Comparator; import java.util.Comparator;
import java.util.Date; import java.util.Date;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Objects; import java.util.Objects;
import java.util.Set; import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong; import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReference; import java.util.concurrent.atomic.AtomicReference;
@ -137,7 +137,7 @@ public class SolrIndexSearcher extends IndexSearcher implements Closeable, SolrI
private final String path; private final String path;
private boolean releaseDirectory; private boolean releaseDirectory;
private Set<String> metricNames = new HashSet<>(); private Set<String> metricNames = ConcurrentHashMap.newKeySet();
private static DirectoryReader getReader(SolrCore core, SolrIndexConfig config, DirectoryFactory directoryFactory, private static DirectoryReader getReader(SolrCore core, SolrIndexConfig config, DirectoryFactory directoryFactory,
String path) throws IOException { String path) throws IOException {

View File

@ -16,8 +16,8 @@
*/ */
package org.apache.solr.store.blockcache; package org.apache.solr.store.blockcache;
import java.util.HashSet;
import java.util.Set; import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicLong; import java.util.concurrent.atomic.AtomicLong;
import com.codahale.metrics.MetricRegistry; import com.codahale.metrics.MetricRegistry;
@ -55,8 +55,7 @@ public class Metrics extends SolrCacheBase implements SolrInfoBean, SolrMetricPr
private MetricsMap metricsMap; private MetricsMap metricsMap;
private MetricRegistry registry; private MetricRegistry registry;
private Set<String> metricNames = new HashSet<>(); private Set<String> metricNames = ConcurrentHashMap.newKeySet();
private long previous = System.nanoTime(); private long previous = System.nanoTime();
@Override @Override

View File

@ -19,7 +19,6 @@ package org.apache.solr.store.hdfs;
import java.io.IOException; import java.io.IOException;
import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles;
import java.util.Arrays; import java.util.Arrays;
import java.util.HashSet;
import java.util.Iterator; import java.util.Iterator;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
@ -51,7 +50,7 @@ public class HdfsLocalityReporter implements SolrInfoBean, SolrMetricProducer {
private String hostname; private String hostname;
private final ConcurrentMap<HdfsDirectory,ConcurrentMap<FileStatus,BlockLocation[]>> cache; private final ConcurrentMap<HdfsDirectory,ConcurrentMap<FileStatus,BlockLocation[]>> cache;
private final Set<String> metricNames = new HashSet<>(); private final Set<String> metricNames = ConcurrentHashMap.newKeySet();
private MetricRegistry registry; private MetricRegistry registry;
public HdfsLocalityReporter() { public HdfsLocalityReporter() {

View File

@ -19,9 +19,9 @@ package org.apache.solr.update;
import java.io.IOException; import java.io.IOException;
import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles;
import java.util.HashSet;
import java.util.Set; import java.util.Set;
import java.util.Vector; import java.util.Vector;
import java.util.concurrent.ConcurrentHashMap;
import com.codahale.metrics.MetricRegistry; import com.codahale.metrics.MetricRegistry;
import org.apache.solr.core.DirectoryFactory; import org.apache.solr.core.DirectoryFactory;
@ -58,7 +58,7 @@ public abstract class UpdateHandler implements SolrInfoBean {
protected final UpdateLog ulog; protected final UpdateLog ulog;
protected Set<String> metricNames = new HashSet<>(); protected Set<String> metricNames = ConcurrentHashMap.newKeySet();
protected MetricRegistry registry; protected MetricRegistry registry;
private void parseEventListeners() { private void parseEventListeners() {

View File

@ -17,8 +17,8 @@
package org.apache.solr.update; package org.apache.solr.update;
import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles;
import java.util.HashSet;
import java.util.Set; import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ExecutorService; import java.util.concurrent.ExecutorService;
import java.util.concurrent.ThreadFactory; import java.util.concurrent.ThreadFactory;
@ -66,7 +66,7 @@ public class UpdateShardHandler implements SolrMetricProducer, SolrInfoBean {
private final InstrumentedHttpRequestExecutor httpRequestExecutor; private final InstrumentedHttpRequestExecutor httpRequestExecutor;
private final Set<String> metricNames = new HashSet<>(); private final Set<String> metricNames = ConcurrentHashMap.newKeySet();
private MetricRegistry registry; private MetricRegistry registry;
public UpdateShardHandler(UpdateShardHandlerConfig cfg) { public UpdateShardHandler(UpdateShardHandlerConfig cfg) {

View File

@ -16,8 +16,8 @@
*/ */
package org.apache.solr.core; package org.apache.solr.core;
import java.util.HashSet;
import java.util.Set; import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import com.codahale.metrics.MetricRegistry; import com.codahale.metrics.MetricRegistry;
import org.apache.solr.metrics.MetricsMap; import org.apache.solr.metrics.MetricsMap;
@ -25,7 +25,7 @@ import org.apache.solr.metrics.SolrMetricManager;
import org.apache.solr.metrics.SolrMetricProducer; import org.apache.solr.metrics.SolrMetricProducer;
class MockInfoBean implements SolrInfoBean, SolrMetricProducer { class MockInfoBean implements SolrInfoBean, SolrMetricProducer {
Set<String> metricNames = new HashSet<>(); Set<String> metricNames = ConcurrentHashMap.newKeySet();
MetricRegistry registry; MetricRegistry registry;
@Override @Override

View File

@ -19,14 +19,20 @@ package org.apache.solr.handler.admin;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.regex.Matcher; import java.util.regex.Matcher;
import java.util.regex.Pattern; import java.util.regex.Pattern;
import com.codahale.metrics.MetricRegistry;
import org.apache.solr.SolrTestCaseJ4; import org.apache.solr.SolrTestCaseJ4;
import org.apache.solr.common.params.CommonParams; import org.apache.solr.common.params.CommonParams;
import org.apache.solr.common.util.ContentStream; import org.apache.solr.common.util.ContentStream;
import org.apache.solr.common.util.ContentStreamBase; import org.apache.solr.common.util.ContentStreamBase;
import org.apache.solr.common.util.NamedList; import org.apache.solr.common.util.NamedList;
import org.apache.solr.core.SolrInfoBean;
import org.apache.solr.request.LocalSolrQueryRequest; import org.apache.solr.request.LocalSolrQueryRequest;
import org.junit.BeforeClass; import org.junit.BeforeClass;
import org.junit.Test; import org.junit.Test;
@ -120,4 +126,74 @@ public class MBeansHandlerTest extends SolrTestCaseJ4 {
assertTrue("external entity ignored properly", true); assertTrue("external entity ignored properly", true);
} }
boolean runSnapshots;
@Test
public void testMetricsSnapshot() throws Exception {
final CountDownLatch counter = new CountDownLatch(500);
MetricRegistry registry = new MetricRegistry();
Set<String> names = ConcurrentHashMap.newKeySet();
SolrInfoBean bean = new SolrInfoBean() {
@Override
public String getName() {
return "foo";
}
@Override
public String getDescription() {
return "foo";
}
@Override
public Category getCategory() {
return Category.ADMIN;
}
@Override
public Set<String> getMetricNames() {
return names;
}
@Override
public MetricRegistry getMetricRegistry() {
return registry;
}
};
runSnapshots = true;
Thread modifier = new Thread(() -> {
int i = 0;
while (runSnapshots) {
bean.registerMetricName("name-" + i++);
try {
Thread.sleep(31);
} catch (InterruptedException e) {
runSnapshots = false;
break;
}
}
});
Thread reader = new Thread(() -> {
while (runSnapshots) {
try {
bean.getMetricsSnapshot();
} catch (Exception e) {
runSnapshots = false;
e.printStackTrace();
fail("Exception getting metrics snapshot: " + e.toString());
}
try {
Thread.sleep(53);
} catch (InterruptedException e) {
runSnapshots = false;
break;
}
counter.countDown();
}
});
modifier.start();
reader.start();
counter.await(30, TimeUnit.SECONDS);
runSnapshots = false;
}
} }