mirror of https://github.com/apache/lucene.git
Update Circuit Breaker configured as a standard plugin (#1785)
This commit is contained in:
parent
54a5dc2e6e
commit
6a7da3cd50
|
@ -78,6 +78,7 @@ import org.apache.solr.update.UpdateLog;
|
||||||
import org.apache.solr.update.processor.UpdateRequestProcessorChain;
|
import org.apache.solr.update.processor.UpdateRequestProcessorChain;
|
||||||
import org.apache.solr.update.processor.UpdateRequestProcessorFactory;
|
import org.apache.solr.update.processor.UpdateRequestProcessorFactory;
|
||||||
import org.apache.solr.util.DOMUtil;
|
import org.apache.solr.util.DOMUtil;
|
||||||
|
import org.apache.solr.util.circuitbreaker.CircuitBreakerManager;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.w3c.dom.Node;
|
import org.w3c.dom.Node;
|
||||||
|
@ -228,14 +229,6 @@ public class SolrConfig extends XmlConfigFile implements MapSerializable {
|
||||||
queryResultMaxDocsCached = getInt("query/queryResultMaxDocsCached", Integer.MAX_VALUE);
|
queryResultMaxDocsCached = getInt("query/queryResultMaxDocsCached", Integer.MAX_VALUE);
|
||||||
enableLazyFieldLoading = getBool("query/enableLazyFieldLoading", false);
|
enableLazyFieldLoading = getBool("query/enableLazyFieldLoading", false);
|
||||||
|
|
||||||
useCircuitBreakers = getBool("circuitBreakers/@enabled", false);
|
|
||||||
cpuCBEnabled = getBool("circuitBreakers/cpuBreaker/@enabled", false);
|
|
||||||
memCBEnabled = getBool("circuitBreakers/memBreaker/@enabled", false);
|
|
||||||
memCBThreshold = getInt("circuitBreakers/memBreaker/@threshold", 95);
|
|
||||||
cpuCBThreshold = getInt("circuitBreakers/cpuBreaker/@threshold", 95);
|
|
||||||
|
|
||||||
validateCircuitBreakerThresholds();
|
|
||||||
|
|
||||||
filterCacheConfig = CacheConfig.getConfig(this, "query/filterCache");
|
filterCacheConfig = CacheConfig.getConfig(this, "query/filterCache");
|
||||||
queryResultCacheConfig = CacheConfig.getConfig(this, "query/queryResultCache");
|
queryResultCacheConfig = CacheConfig.getConfig(this, "query/queryResultCache");
|
||||||
documentCacheConfig = CacheConfig.getConfig(this, "query/documentCache");
|
documentCacheConfig = CacheConfig.getConfig(this, "query/documentCache");
|
||||||
|
@ -365,6 +358,7 @@ public class SolrConfig extends XmlConfigFile implements MapSerializable {
|
||||||
.add(new SolrPluginInfo(IndexSchemaFactory.class, "schemaFactory", REQUIRE_CLASS))
|
.add(new SolrPluginInfo(IndexSchemaFactory.class, "schemaFactory", REQUIRE_CLASS))
|
||||||
.add(new SolrPluginInfo(RestManager.class, "restManager"))
|
.add(new SolrPluginInfo(RestManager.class, "restManager"))
|
||||||
.add(new SolrPluginInfo(StatsCache.class, "statsCache", REQUIRE_CLASS))
|
.add(new SolrPluginInfo(StatsCache.class, "statsCache", REQUIRE_CLASS))
|
||||||
|
.add(new SolrPluginInfo(CircuitBreakerManager.class, "circuitBreaker"))
|
||||||
.build();
|
.build();
|
||||||
public static final Map<String, SolrPluginInfo> classVsSolrPluginInfo;
|
public static final Map<String, SolrPluginInfo> classVsSolrPluginInfo;
|
||||||
|
|
||||||
|
@ -531,13 +525,6 @@ public class SolrConfig extends XmlConfigFile implements MapSerializable {
|
||||||
public final int queryResultMaxDocsCached;
|
public final int queryResultMaxDocsCached;
|
||||||
public final boolean enableLazyFieldLoading;
|
public final boolean enableLazyFieldLoading;
|
||||||
|
|
||||||
// Circuit Breaker Configuration
|
|
||||||
public final boolean useCircuitBreakers;
|
|
||||||
public final int memCBThreshold;
|
|
||||||
public final boolean memCBEnabled;
|
|
||||||
public final boolean cpuCBEnabled;
|
|
||||||
public final int cpuCBThreshold;
|
|
||||||
|
|
||||||
// IndexConfig settings
|
// IndexConfig settings
|
||||||
public final SolrIndexConfig indexConfig;
|
public final SolrIndexConfig indexConfig;
|
||||||
|
|
||||||
|
@ -817,16 +804,6 @@ public class SolrConfig extends XmlConfigFile implements MapSerializable {
|
||||||
loader.reloadLuceneSPI();
|
loader.reloadLuceneSPI();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void validateCircuitBreakerThresholds() {
|
|
||||||
if (useCircuitBreakers) {
|
|
||||||
if (memCBEnabled) {
|
|
||||||
if (memCBThreshold > 95 || memCBThreshold < 50) {
|
|
||||||
throw new IllegalArgumentException("Valid value range of memoryCircuitBreakerThresholdPct is 50 - 95");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getMultipartUploadLimitKB() {
|
public int getMultipartUploadLimitKB() {
|
||||||
return multipartUploadLimitKB;
|
return multipartUploadLimitKB;
|
||||||
}
|
}
|
||||||
|
@ -896,11 +873,7 @@ public class SolrConfig extends XmlConfigFile implements MapSerializable {
|
||||||
m.put("queryResultMaxDocsCached", queryResultMaxDocsCached);
|
m.put("queryResultMaxDocsCached", queryResultMaxDocsCached);
|
||||||
m.put("enableLazyFieldLoading", enableLazyFieldLoading);
|
m.put("enableLazyFieldLoading", enableLazyFieldLoading);
|
||||||
m.put("maxBooleanClauses", booleanQueryMaxClauseCount);
|
m.put("maxBooleanClauses", booleanQueryMaxClauseCount);
|
||||||
m.put("useCircuitBreakers", useCircuitBreakers);
|
|
||||||
m.put("cpuCircuitBreakerEnabled", cpuCBEnabled);
|
|
||||||
m.put("memoryCircuitBreakerEnabled", memCBEnabled);
|
|
||||||
m.put("memoryCircuitBreakerThresholdPct", memCBThreshold);
|
|
||||||
m.put("cpuCircuitBreakerThreshold", cpuCBThreshold);
|
|
||||||
for (SolrPluginInfo plugin : plugins) {
|
for (SolrPluginInfo plugin : plugins) {
|
||||||
List<PluginInfo> infos = getPluginInfos(plugin.clazz.getName());
|
List<PluginInfo> infos = getPluginInfos(plugin.clazz.getName());
|
||||||
if (infos == null || infos.isEmpty()) continue;
|
if (infos == null || infos.isEmpty()) continue;
|
||||||
|
|
|
@ -1174,7 +1174,8 @@ public final class SolrCore implements SolrInfoBean, Closeable {
|
||||||
}
|
}
|
||||||
|
|
||||||
private CircuitBreakerManager initCircuitBreakerManager() {
|
private CircuitBreakerManager initCircuitBreakerManager() {
|
||||||
CircuitBreakerManager circuitBreakerManager = CircuitBreakerManager.build(solrConfig);
|
final PluginInfo info = solrConfig.getPluginInfo(CircuitBreakerManager.class.getName());
|
||||||
|
CircuitBreakerManager circuitBreakerManager = CircuitBreakerManager.build(info);
|
||||||
|
|
||||||
return circuitBreakerManager;
|
return circuitBreakerManager;
|
||||||
}
|
}
|
||||||
|
|
|
@ -303,7 +303,7 @@ public class SearchHandler extends RequestHandlerBase implements SolrCoreAware,
|
||||||
|
|
||||||
final RTimerTree timer = rb.isDebug() ? req.getRequestTimer() : null;
|
final RTimerTree timer = rb.isDebug() ? req.getRequestTimer() : null;
|
||||||
|
|
||||||
if (req.getCore().getSolrConfig().useCircuitBreakers) {
|
if (req.getCore().getCircuitBreakerManager().isEnabled()) {
|
||||||
List<CircuitBreaker> trippedCircuitBreakers;
|
List<CircuitBreaker> trippedCircuitBreakers;
|
||||||
|
|
||||||
if (timer != null) {
|
if (timer != null) {
|
||||||
|
|
|
@ -21,7 +21,6 @@ import java.lang.invoke.MethodHandles;
|
||||||
import java.lang.management.ManagementFactory;
|
import java.lang.management.ManagementFactory;
|
||||||
import java.lang.management.OperatingSystemMXBean;
|
import java.lang.management.OperatingSystemMXBean;
|
||||||
|
|
||||||
import org.apache.solr.core.SolrConfig;
|
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
@ -52,11 +51,11 @@ public class CPUCircuitBreaker extends CircuitBreaker {
|
||||||
|
|
||||||
private static final ThreadLocal<Double> allowedCPUUsage = ThreadLocal.withInitial(() -> 0.0);
|
private static final ThreadLocal<Double> allowedCPUUsage = ThreadLocal.withInitial(() -> 0.0);
|
||||||
|
|
||||||
public CPUCircuitBreaker(SolrConfig solrConfig) {
|
public CPUCircuitBreaker(CircuitBreakerConfig config) {
|
||||||
super(solrConfig);
|
super(config);
|
||||||
|
|
||||||
this.enabled = solrConfig.cpuCBEnabled;
|
this.enabled = config.getCpuCBEnabled();
|
||||||
this.cpuUsageThreshold = solrConfig.cpuCBThreshold;
|
this.cpuUsageThreshold = config.getCpuCBThreshold();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -17,8 +17,6 @@
|
||||||
|
|
||||||
package org.apache.solr.util.circuitbreaker;
|
package org.apache.solr.util.circuitbreaker;
|
||||||
|
|
||||||
import org.apache.solr.core.SolrConfig;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Default class to define circuit breakers for Solr.
|
* Default class to define circuit breakers for Solr.
|
||||||
* <p>
|
* <p>
|
||||||
|
@ -35,16 +33,16 @@ import org.apache.solr.core.SolrConfig;
|
||||||
public abstract class CircuitBreaker {
|
public abstract class CircuitBreaker {
|
||||||
public static final String NAME = "circuitbreaker";
|
public static final String NAME = "circuitbreaker";
|
||||||
|
|
||||||
protected final SolrConfig solrConfig;
|
protected final CircuitBreakerConfig config;
|
||||||
|
|
||||||
public CircuitBreaker(SolrConfig solrConfig) {
|
public CircuitBreaker(CircuitBreakerConfig config) {
|
||||||
this.solrConfig = solrConfig;
|
this.config = config;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Global config for all circuit breakers. For specific circuit breaker configs, define
|
// Global config for all circuit breakers. For specific circuit breaker configs, define
|
||||||
// your own config.
|
// your own config.
|
||||||
protected boolean isEnabled() {
|
protected boolean isEnabled() {
|
||||||
return solrConfig.useCircuitBreakers;
|
return config.isEnabled();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -61,4 +59,41 @@ public abstract class CircuitBreaker {
|
||||||
* Get error message when the circuit breaker triggers
|
* Get error message when the circuit breaker triggers
|
||||||
*/
|
*/
|
||||||
public abstract String getErrorMessage();
|
public abstract String getErrorMessage();
|
||||||
|
|
||||||
|
public static class CircuitBreakerConfig {
|
||||||
|
private final boolean enabled;
|
||||||
|
private final boolean memCBEnabled;
|
||||||
|
private final int memCBThreshold;
|
||||||
|
private final boolean cpuCBEnabled;
|
||||||
|
private final int cpuCBThreshold;
|
||||||
|
|
||||||
|
public CircuitBreakerConfig(final boolean enabled, final boolean memCBEnabled, final int memCBThreshold,
|
||||||
|
final boolean cpuCBEnabled, final int cpuCBThreshold) {
|
||||||
|
this.enabled = enabled;
|
||||||
|
this.memCBEnabled = memCBEnabled;
|
||||||
|
this.memCBThreshold = memCBThreshold;
|
||||||
|
this.cpuCBEnabled = cpuCBEnabled;
|
||||||
|
this.cpuCBThreshold = cpuCBThreshold;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isEnabled() {
|
||||||
|
return enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean getMemCBEnabled() {
|
||||||
|
return memCBEnabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getMemCBThreshold() {
|
||||||
|
return memCBThreshold;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean getCpuCBEnabled() {
|
||||||
|
return cpuCBEnabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getCpuCBThreshold() {
|
||||||
|
return cpuCBThreshold;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -21,7 +21,9 @@ import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import com.google.common.annotations.VisibleForTesting;
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
import org.apache.solr.core.SolrConfig;
|
import org.apache.solr.common.util.NamedList;
|
||||||
|
import org.apache.solr.core.PluginInfo;
|
||||||
|
import org.apache.solr.util.plugin.PluginInfoInitialized;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Manages all registered circuit breaker instances. Responsible for a holistic view
|
* Manages all registered circuit breaker instances. Responsible for a holistic view
|
||||||
|
@ -37,7 +39,7 @@ import org.apache.solr.core.SolrConfig;
|
||||||
* NOTE: The current way of registering new default circuit breakers is minimal and not a long term
|
* NOTE: The current way of registering new default circuit breakers is minimal and not a long term
|
||||||
* solution. There will be a follow up with a SIP for a schema API design.
|
* solution. There will be a follow up with a SIP for a schema API design.
|
||||||
*/
|
*/
|
||||||
public class CircuitBreakerManager {
|
public class CircuitBreakerManager implements PluginInfoInitialized {
|
||||||
// Class private to potentially allow "family" of circuit breakers to be enabled or disabled
|
// Class private to potentially allow "family" of circuit breakers to be enabled or disabled
|
||||||
private final boolean enableCircuitBreakerManager;
|
private final boolean enableCircuitBreakerManager;
|
||||||
|
|
||||||
|
@ -47,6 +49,18 @@ public class CircuitBreakerManager {
|
||||||
this.enableCircuitBreakerManager = enableCircuitBreakerManager;
|
this.enableCircuitBreakerManager = enableCircuitBreakerManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void init(PluginInfo pluginInfo) {
|
||||||
|
CircuitBreaker.CircuitBreakerConfig circuitBreakerConfig = buildCBConfig(pluginInfo);
|
||||||
|
|
||||||
|
// Install the default circuit breakers
|
||||||
|
CircuitBreaker memoryCircuitBreaker = new MemoryCircuitBreaker(circuitBreakerConfig);
|
||||||
|
CircuitBreaker cpuCircuitBreaker = new CPUCircuitBreaker(circuitBreakerConfig);
|
||||||
|
|
||||||
|
register(memoryCircuitBreaker);
|
||||||
|
register(cpuCircuitBreaker);
|
||||||
|
}
|
||||||
|
|
||||||
public void register(CircuitBreaker circuitBreaker) {
|
public void register(CircuitBreaker circuitBreaker) {
|
||||||
circuitBreakerList.add(circuitBreaker);
|
circuitBreakerList.add(circuitBreaker);
|
||||||
}
|
}
|
||||||
|
@ -121,19 +135,40 @@ public class CircuitBreakerManager {
|
||||||
*
|
*
|
||||||
* Any default circuit breakers should be registered here.
|
* Any default circuit breakers should be registered here.
|
||||||
*/
|
*/
|
||||||
public static CircuitBreakerManager build(SolrConfig solrConfig) {
|
@SuppressWarnings({"rawtypes"})
|
||||||
CircuitBreakerManager circuitBreakerManager = new CircuitBreakerManager(solrConfig.useCircuitBreakers);
|
public static CircuitBreakerManager build(PluginInfo pluginInfo) {
|
||||||
|
CircuitBreakerManager circuitBreakerManager = new CircuitBreakerManager(Boolean.parseBoolean(pluginInfo.attributes.get("enabled")));
|
||||||
|
|
||||||
// Install the default circuit breakers
|
circuitBreakerManager.init(pluginInfo);
|
||||||
CircuitBreaker memoryCircuitBreaker = new MemoryCircuitBreaker(solrConfig);
|
|
||||||
CircuitBreaker cpuCircuitBreaker = new CPUCircuitBreaker(solrConfig);
|
|
||||||
|
|
||||||
circuitBreakerManager.register(memoryCircuitBreaker);
|
|
||||||
circuitBreakerManager.register(cpuCircuitBreaker);
|
|
||||||
|
|
||||||
return circuitBreakerManager;
|
return circuitBreakerManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
@SuppressWarnings({"rawtypes"})
|
||||||
|
public static CircuitBreaker.CircuitBreakerConfig buildCBConfig(PluginInfo pluginInfo) {
|
||||||
|
boolean enabled = Boolean.parseBoolean(pluginInfo.attributes.get("enabled"));
|
||||||
|
boolean cpuCBEnabled = false;
|
||||||
|
boolean memCBEnabled = false;
|
||||||
|
int memCBThreshold = 100;
|
||||||
|
int cpuCBThreshold = 100;
|
||||||
|
|
||||||
|
NamedList args = pluginInfo.initArgs;
|
||||||
|
|
||||||
|
if (args != null) {
|
||||||
|
cpuCBEnabled = args.getBooleanArg("cpuEnabled");
|
||||||
|
memCBEnabled = args.getBooleanArg("memEnabled");
|
||||||
|
memCBThreshold = Integer.parseInt((String) args.get("memThreshold"));
|
||||||
|
cpuCBThreshold = Integer.parseInt((String) args.get("cpuThreshold"));
|
||||||
|
}
|
||||||
|
|
||||||
|
return new CircuitBreaker.CircuitBreakerConfig(enabled, memCBEnabled, memCBThreshold, cpuCBEnabled, cpuCBThreshold);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isEnabled() {
|
||||||
|
return enableCircuitBreakerManager;
|
||||||
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
public List<CircuitBreaker> getRegisteredCircuitBreakers() {
|
public List<CircuitBreaker> getRegisteredCircuitBreakers() {
|
||||||
return circuitBreakerList;
|
return circuitBreakerList;
|
||||||
|
|
|
@ -21,7 +21,6 @@ import java.lang.invoke.MethodHandles;
|
||||||
import java.lang.management.ManagementFactory;
|
import java.lang.management.ManagementFactory;
|
||||||
import java.lang.management.MemoryMXBean;
|
import java.lang.management.MemoryMXBean;
|
||||||
|
|
||||||
import org.apache.solr.core.SolrConfig;
|
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
@ -50,10 +49,10 @@ public class MemoryCircuitBreaker extends CircuitBreaker {
|
||||||
private static final ThreadLocal<Long> seenMemory = ThreadLocal.withInitial(() -> 0L);
|
private static final ThreadLocal<Long> seenMemory = ThreadLocal.withInitial(() -> 0L);
|
||||||
private static final ThreadLocal<Long> allowedMemory = ThreadLocal.withInitial(() -> 0L);
|
private static final ThreadLocal<Long> allowedMemory = ThreadLocal.withInitial(() -> 0L);
|
||||||
|
|
||||||
public MemoryCircuitBreaker(SolrConfig solrConfig) {
|
public MemoryCircuitBreaker(CircuitBreakerConfig config) {
|
||||||
super(solrConfig);
|
super(config);
|
||||||
|
|
||||||
this.enabled = solrConfig.memCBEnabled;
|
this.enabled = config.getMemCBEnabled();
|
||||||
|
|
||||||
long currentMaxHeap = MEMORY_MX_BEAN.getHeapMemoryUsage().getMax();
|
long currentMaxHeap = MEMORY_MX_BEAN.getHeapMemoryUsage().getMax();
|
||||||
|
|
||||||
|
@ -61,7 +60,7 @@ public class MemoryCircuitBreaker extends CircuitBreaker {
|
||||||
throw new IllegalArgumentException("Invalid JVM state for the max heap usage");
|
throw new IllegalArgumentException("Invalid JVM state for the max heap usage");
|
||||||
}
|
}
|
||||||
|
|
||||||
int thresholdValueInPercentage = solrConfig.memCBThreshold;
|
int thresholdValueInPercentage = config.getMemCBThreshold();
|
||||||
double thresholdInFraction = thresholdValueInPercentage / (double) 100;
|
double thresholdInFraction = thresholdValueInPercentage / (double) 100;
|
||||||
heapMemoryThreshold = (long) (currentMaxHeap * thresholdInFraction);
|
heapMemoryThreshold = (long) (currentMaxHeap * thresholdInFraction);
|
||||||
|
|
||||||
|
|
|
@ -55,17 +55,6 @@
|
||||||
"queryResultMaxDocsCached":1,
|
"queryResultMaxDocsCached":1,
|
||||||
"enableLazyFieldLoading":1,
|
"enableLazyFieldLoading":1,
|
||||||
"boolTofilterOptimizer":1,
|
"boolTofilterOptimizer":1,
|
||||||
"circuitBreakers":{
|
|
||||||
"enabled":10,
|
|
||||||
"memBreaker":{
|
|
||||||
"enabled":10,
|
|
||||||
"threshold":20
|
|
||||||
},
|
|
||||||
"cpuBreaker":{
|
|
||||||
"enabled":10,
|
|
||||||
"threshold":20
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"maxBooleanClauses":1},
|
"maxBooleanClauses":1},
|
||||||
"jmx":{
|
"jmx":{
|
||||||
"agentId":0,
|
"agentId":0,
|
||||||
|
|
|
@ -78,10 +78,12 @@
|
||||||
|
|
||||||
</query>
|
</query>
|
||||||
|
|
||||||
<circuitBreakers enabled="true">
|
<circuitBreaker class="solr.CircuitBreakerManager" enabled="true">
|
||||||
<cpuBreaker enabled="true" threshold="75"/>
|
<str name="memEnabled">true</str>
|
||||||
<memBreaker enabled="true" threshold="75"/>
|
<str name="memThreshold">75</str>
|
||||||
</circuitBreakers>
|
<str name="cpuEnabled">true</str>
|
||||||
|
<str name="cpuThreshold">75</str>
|
||||||
|
</circuitBreaker>
|
||||||
|
|
||||||
<initParams path="/select">
|
<initParams path="/select">
|
||||||
<lst name="defaults">
|
<lst name="defaults">
|
||||||
|
|
|
@ -266,9 +266,6 @@ public class SolrCoreTest extends SolrTestCaseJ4 {
|
||||||
assertEquals("wrong config for maxBooleanClauses", 1024, solrConfig.booleanQueryMaxClauseCount);
|
assertEquals("wrong config for maxBooleanClauses", 1024, solrConfig.booleanQueryMaxClauseCount);
|
||||||
assertEquals("wrong config for enableLazyFieldLoading", true, solrConfig.enableLazyFieldLoading);
|
assertEquals("wrong config for enableLazyFieldLoading", true, solrConfig.enableLazyFieldLoading);
|
||||||
assertEquals("wrong config for queryResultWindowSize", 10, solrConfig.queryResultWindowSize);
|
assertEquals("wrong config for queryResultWindowSize", 10, solrConfig.queryResultWindowSize);
|
||||||
assertEquals("wrong config for useCircuitBreakers", false, solrConfig.useCircuitBreakers);
|
|
||||||
assertEquals("wrong config for memoryCircuitBreakerThresholdPct", 95, solrConfig.memCBThreshold);
|
|
||||||
assertEquals("wrong config for cpuCircuitBreakerThreshold", 95, solrConfig.cpuCBThreshold);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -46,11 +46,6 @@ public class TestConfigOverlay extends SolrTestCase {
|
||||||
assertTrue(isEditableProp("query.queryResultMaxDocsCached", false, null));
|
assertTrue(isEditableProp("query.queryResultMaxDocsCached", false, null));
|
||||||
assertTrue(isEditableProp("query.enableLazyFieldLoading", false, null));
|
assertTrue(isEditableProp("query.enableLazyFieldLoading", false, null));
|
||||||
assertTrue(isEditableProp("query.boolTofilterOptimizer", false, null));
|
assertTrue(isEditableProp("query.boolTofilterOptimizer", false, null));
|
||||||
assertTrue(isEditableProp("query.circuitBreakers.enabled", false, null));
|
|
||||||
assertTrue(isEditableProp("query.circuitBreakers.memBreaker.enabled", false, null));
|
|
||||||
assertTrue(isEditableProp("query.circuitBreakers.memBreaker.threshold", false, null));
|
|
||||||
assertTrue(isEditableProp("query.circuitBreakers.cpuBreaker.enabled", false, null));
|
|
||||||
assertTrue(isEditableProp("query.circuitBreakers.cpuBreaker.threshold", false, null));
|
|
||||||
assertTrue(isEditableProp("jmx.agentId", false, null));
|
assertTrue(isEditableProp("jmx.agentId", false, null));
|
||||||
assertTrue(isEditableProp("jmx.serviceUrl", false, null));
|
assertTrue(isEditableProp("jmx.serviceUrl", false, null));
|
||||||
assertTrue(isEditableProp("jmx.rootName", false, null));
|
assertTrue(isEditableProp("jmx.rootName", false, null));
|
||||||
|
|
|
@ -32,10 +32,11 @@ import org.apache.solr.common.SolrException;
|
||||||
import org.apache.solr.common.params.CommonParams;
|
import org.apache.solr.common.params.CommonParams;
|
||||||
import org.apache.solr.common.util.ExecutorUtil;
|
import org.apache.solr.common.util.ExecutorUtil;
|
||||||
import org.apache.solr.common.util.SolrNamedThreadFactory;
|
import org.apache.solr.common.util.SolrNamedThreadFactory;
|
||||||
import org.apache.solr.core.SolrConfig;
|
import org.apache.solr.core.PluginInfo;
|
||||||
import org.apache.solr.search.QueryParsing;
|
import org.apache.solr.search.QueryParsing;
|
||||||
import org.apache.solr.util.circuitbreaker.CPUCircuitBreaker;
|
import org.apache.solr.util.circuitbreaker.CPUCircuitBreaker;
|
||||||
import org.apache.solr.util.circuitbreaker.CircuitBreaker;
|
import org.apache.solr.util.circuitbreaker.CircuitBreaker;
|
||||||
|
import org.apache.solr.util.circuitbreaker.CircuitBreakerManager;
|
||||||
import org.apache.solr.util.circuitbreaker.MemoryCircuitBreaker;
|
import org.apache.solr.util.circuitbreaker.MemoryCircuitBreaker;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
import org.junit.BeforeClass;
|
import org.junit.BeforeClass;
|
||||||
|
@ -93,7 +94,11 @@ public class TestCircuitBreaker extends SolrTestCaseJ4 {
|
||||||
|
|
||||||
removeAllExistingCircuitBreakers();
|
removeAllExistingCircuitBreakers();
|
||||||
|
|
||||||
CircuitBreaker circuitBreaker = new MockCircuitBreaker(h.getCore().getSolrConfig());
|
PluginInfo pluginInfo = h.getCore().getSolrConfig().getPluginInfo(CircuitBreakerManager.class.getName());
|
||||||
|
|
||||||
|
CircuitBreaker.CircuitBreakerConfig circuitBreakerConfig = CircuitBreakerManager.buildCBConfig(pluginInfo);
|
||||||
|
|
||||||
|
CircuitBreaker circuitBreaker = new MockCircuitBreaker(circuitBreakerConfig);
|
||||||
|
|
||||||
h.getCore().getCircuitBreakerManager().register(circuitBreaker);
|
h.getCore().getCircuitBreakerManager().register(circuitBreaker);
|
||||||
|
|
||||||
|
@ -110,7 +115,10 @@ public class TestCircuitBreaker extends SolrTestCaseJ4 {
|
||||||
|
|
||||||
removeAllExistingCircuitBreakers();
|
removeAllExistingCircuitBreakers();
|
||||||
|
|
||||||
CircuitBreaker circuitBreaker = new FakeMemoryPressureCircuitBreaker(h.getCore().getSolrConfig());
|
PluginInfo pluginInfo = h.getCore().getSolrConfig().getPluginInfo(CircuitBreakerManager.class.getName());
|
||||||
|
|
||||||
|
CircuitBreaker.CircuitBreakerConfig circuitBreakerConfig = CircuitBreakerManager.buildCBConfig(pluginInfo);
|
||||||
|
CircuitBreaker circuitBreaker = new FakeMemoryPressureCircuitBreaker(circuitBreakerConfig);
|
||||||
|
|
||||||
h.getCore().getCircuitBreakerManager().register(circuitBreaker);
|
h.getCore().getCircuitBreakerManager().register(circuitBreaker);
|
||||||
|
|
||||||
|
@ -132,7 +140,10 @@ public class TestCircuitBreaker extends SolrTestCaseJ4 {
|
||||||
try {
|
try {
|
||||||
removeAllExistingCircuitBreakers();
|
removeAllExistingCircuitBreakers();
|
||||||
|
|
||||||
CircuitBreaker circuitBreaker = new BuildingUpMemoryPressureCircuitBreaker(h.getCore().getSolrConfig());
|
PluginInfo pluginInfo = h.getCore().getSolrConfig().getPluginInfo(CircuitBreakerManager.class.getName());
|
||||||
|
|
||||||
|
CircuitBreaker.CircuitBreakerConfig circuitBreakerConfig = CircuitBreakerManager.buildCBConfig(pluginInfo);
|
||||||
|
CircuitBreaker circuitBreaker = new BuildingUpMemoryPressureCircuitBreaker(circuitBreakerConfig);
|
||||||
|
|
||||||
h.getCore().getCircuitBreakerManager().register(circuitBreaker);
|
h.getCore().getCircuitBreakerManager().register(circuitBreaker);
|
||||||
|
|
||||||
|
@ -185,7 +196,10 @@ public class TestCircuitBreaker extends SolrTestCaseJ4 {
|
||||||
try {
|
try {
|
||||||
removeAllExistingCircuitBreakers();
|
removeAllExistingCircuitBreakers();
|
||||||
|
|
||||||
CircuitBreaker circuitBreaker = new FakeCPUCircuitBreaker(h.getCore().getSolrConfig());
|
PluginInfo pluginInfo = h.getCore().getSolrConfig().getPluginInfo(CircuitBreakerManager.class.getName());
|
||||||
|
|
||||||
|
CircuitBreaker.CircuitBreakerConfig circuitBreakerConfig = CircuitBreakerManager.buildCBConfig(pluginInfo);
|
||||||
|
CircuitBreaker circuitBreaker = new FakeCPUCircuitBreaker(circuitBreakerConfig);
|
||||||
|
|
||||||
h.getCore().getCircuitBreakerManager().register(circuitBreaker);
|
h.getCore().getCircuitBreakerManager().register(circuitBreaker);
|
||||||
|
|
||||||
|
@ -260,8 +274,8 @@ public class TestCircuitBreaker extends SolrTestCaseJ4 {
|
||||||
|
|
||||||
private static class MockCircuitBreaker extends MemoryCircuitBreaker {
|
private static class MockCircuitBreaker extends MemoryCircuitBreaker {
|
||||||
|
|
||||||
public MockCircuitBreaker(SolrConfig solrConfig) {
|
public MockCircuitBreaker(CircuitBreakerConfig config) {
|
||||||
super(solrConfig);
|
super(config);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -278,8 +292,8 @@ public class TestCircuitBreaker extends SolrTestCaseJ4 {
|
||||||
|
|
||||||
private static class FakeMemoryPressureCircuitBreaker extends MemoryCircuitBreaker {
|
private static class FakeMemoryPressureCircuitBreaker extends MemoryCircuitBreaker {
|
||||||
|
|
||||||
public FakeMemoryPressureCircuitBreaker(SolrConfig solrConfig) {
|
public FakeMemoryPressureCircuitBreaker(CircuitBreakerConfig config) {
|
||||||
super(solrConfig);
|
super(config);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -292,8 +306,8 @@ public class TestCircuitBreaker extends SolrTestCaseJ4 {
|
||||||
private static class BuildingUpMemoryPressureCircuitBreaker extends MemoryCircuitBreaker {
|
private static class BuildingUpMemoryPressureCircuitBreaker extends MemoryCircuitBreaker {
|
||||||
private AtomicInteger count;
|
private AtomicInteger count;
|
||||||
|
|
||||||
public BuildingUpMemoryPressureCircuitBreaker(SolrConfig solrConfig) {
|
public BuildingUpMemoryPressureCircuitBreaker(CircuitBreakerConfig config) {
|
||||||
super(solrConfig);
|
super(config);
|
||||||
|
|
||||||
this.count = new AtomicInteger(0);
|
this.count = new AtomicInteger(0);
|
||||||
}
|
}
|
||||||
|
@ -321,8 +335,8 @@ public class TestCircuitBreaker extends SolrTestCaseJ4 {
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class FakeCPUCircuitBreaker extends CPUCircuitBreaker {
|
private static class FakeCPUCircuitBreaker extends CPUCircuitBreaker {
|
||||||
public FakeCPUCircuitBreaker(SolrConfig solrConfig) {
|
public FakeCPUCircuitBreaker(CircuitBreakerConfig config) {
|
||||||
super(solrConfig);
|
super(config);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -550,7 +550,7 @@
|
||||||
Circuit Breaker Section - This section consists of configurations for
|
Circuit Breaker Section - This section consists of configurations for
|
||||||
circuit breakers
|
circuit breakers
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -->
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -->
|
||||||
<circuitBreaker>
|
<circuitBreaker class="solr.CircuitBreakerManager" enabled="true">
|
||||||
<!-- Enable Circuit Breakers
|
<!-- Enable Circuit Breakers
|
||||||
|
|
||||||
Circuit breakers are designed to allow stability and predictable query
|
Circuit breakers are designed to allow stability and predictable query
|
||||||
|
@ -563,7 +563,12 @@
|
||||||
they are free to add their specific configuration but need to ensure that this flag is always
|
they are free to add their specific configuration but need to ensure that this flag is always
|
||||||
respected - this should have veto over all independent configuration flags.
|
respected - this should have veto over all independent configuration flags.
|
||||||
-->
|
-->
|
||||||
<useCircuitBreakers>false</useCircuitBreakers>
|
|
||||||
|
<!-- Memory Circuit Breaker Control Flag
|
||||||
|
|
||||||
|
Use the following flag to control the behaviour of this circuit breaker
|
||||||
|
-->
|
||||||
|
<str name="memEnabled">true</str>
|
||||||
|
|
||||||
<!-- Memory Circuit Breaker Threshold In Percentage
|
<!-- Memory Circuit Breaker Threshold In Percentage
|
||||||
|
|
||||||
|
@ -580,7 +585,20 @@
|
||||||
in logs and the corresponding error message should tell you what transpired (if the failure
|
in logs and the corresponding error message should tell you what transpired (if the failure
|
||||||
was caused by tripped circuit breakers).
|
was caused by tripped circuit breakers).
|
||||||
-->
|
-->
|
||||||
<memoryCircuitBreakerThresholdPct>100</memoryCircuitBreakerThresholdPct>
|
<str name="memThreshold">75</str>
|
||||||
|
|
||||||
|
<!-- CPU Based Circuit Breaker Control Flag
|
||||||
|
|
||||||
|
Use the following flag to control the behaviour of this circuit breaker
|
||||||
|
-->
|
||||||
|
<str name="cpuEnabled">true</str>
|
||||||
|
|
||||||
|
<!-- CPU Based Circuit Breaker Triggering Threshold
|
||||||
|
|
||||||
|
The triggering threshold is defined in units of CPU utilization. The configuration to control this is as below:
|
||||||
|
-->
|
||||||
|
<str name="cpuThreshold">75</str>
|
||||||
|
|
||||||
|
|
||||||
</circuitBreaker>
|
</circuitBreaker>
|
||||||
|
|
||||||
|
|
|
@ -27,19 +27,24 @@ are enabled, requests may be rejected under the condition of high node duress wi
|
||||||
It is up to the client to handle this error and potentially build a retrial logic as this should ideally be a transient situation.
|
It is up to the client to handle this error and potentially build a retrial logic as this should ideally be a transient situation.
|
||||||
|
|
||||||
== Circuit Breaker Configurations
|
== Circuit Breaker Configurations
|
||||||
The following flag controls the global activation/deactivation of circuit breakers. If this flag is disabled, all circuit breakers
|
All circuit breaker configurations are listed in the circuitBreaker tags in solrconfig.xml as shown below:
|
||||||
will be disabled globally. Per circuit breaker configurations are specified in their respective sections later.
|
|
||||||
|
|
||||||
[source,xml]
|
[source,xml]
|
||||||
----
|
----
|
||||||
<circuitBreakers enabled="true">
|
<circuitBreaker class="solr.CircuitBreakerManager" enabled="true">
|
||||||
<!-- All specific configs in this section -->
|
<!-- All specific configs in this section -->
|
||||||
</circuitBreakers>
|
</circuitBreaker>
|
||||||
----
|
----
|
||||||
|
|
||||||
This flag acts as the highest authority and global controller of circuit breakers. For using specific circuit breakers, each one
|
The "enabled" attribute controls the global activation/deactivation of circuit breakers. If this flag is disabled, all circuit breakers
|
||||||
|
will be disabled globally. Per circuit breaker configurations are specified in their respective sections later.
|
||||||
|
|
||||||
|
This attribute acts as the highest authority and global controller of circuit breakers. For using specific circuit breakers, each one
|
||||||
needs to be individually enabled in addition to this flag being enabled.
|
needs to be individually enabled in addition to this flag being enabled.
|
||||||
|
|
||||||
|
CircuitBreakerManager is the default manager for all circuit breakers that should be defined in the tag unless the user wishes to use
|
||||||
|
a custom implementation.
|
||||||
|
|
||||||
== Currently Supported Circuit Breakers
|
== Currently Supported Circuit Breakers
|
||||||
|
|
||||||
=== JVM Heap Usage Based Circuit Breaker
|
=== JVM Heap Usage Based Circuit Breaker
|
||||||
|
@ -51,11 +56,18 @@ Configuration for JVM heap usage based circuit breaker:
|
||||||
|
|
||||||
[source,xml]
|
[source,xml]
|
||||||
----
|
----
|
||||||
<memBreaker enabled="true" threshold="75"/>
|
<str name="memEnabled">true</str>
|
||||||
----
|
----
|
||||||
|
|
||||||
Note that this configuration will be overridden by the global circuit breaker flag -- if circuit breakers are disabled, this flag
|
Note that this configuration will be overridden by the global circuit breaker flag -- if circuit breakers are disabled, this flag
|
||||||
will not help you. Also, the triggering threshold is defined as a percentage of the max heap allocated to the JVM.
|
will not help you.
|
||||||
|
|
||||||
|
The triggering threshold is defined as a percentage of the max heap allocated to the JVM. It is controlled by the below configuration:
|
||||||
|
|
||||||
|
[source,xml]
|
||||||
|
----
|
||||||
|
<str name="memThreshold">75</str>
|
||||||
|
----
|
||||||
|
|
||||||
It does not logically make sense to have a threshold below 50% and above 95% of the max heap allocated to the JVM. Hence, the range
|
It does not logically make sense to have a threshold below 50% and above 95% of the max heap allocated to the JVM. Hence, the range
|
||||||
of valid values for this parameter is [50, 95], both inclusive.
|
of valid values for this parameter is [50, 95], both inclusive.
|
||||||
|
@ -76,11 +88,18 @@ Configuration for CPU utilization based circuit breaker:
|
||||||
|
|
||||||
[source,xml]
|
[source,xml]
|
||||||
----
|
----
|
||||||
<cpuBreaker enabled="true" threshold="20"/>
|
<str name="cpuEnabled">true</str>
|
||||||
----
|
----
|
||||||
|
|
||||||
Note that this configuration will be overridden by the global circuit breaker flag -- if circuit breakers are disabled, this flag
|
Note that this configuration will be overridden by the global circuit breaker flag -- if circuit breakers are disabled, this flag
|
||||||
will not help you. The triggering threshold is defined in units of CPU utilization.
|
will not help you.
|
||||||
|
|
||||||
|
The triggering threshold is defined in units of CPU utilization. The configuration to control this is as below:
|
||||||
|
|
||||||
|
[source,xml]
|
||||||
|
----
|
||||||
|
<str name="cpuThreshold">75</str>
|
||||||
|
----
|
||||||
|
|
||||||
== Performance Considerations
|
== Performance Considerations
|
||||||
It is worth noting that while JVM or CPU circuit breakers do not add any noticeable overhead per query, having too many
|
It is worth noting that while JVM or CPU circuit breakers do not add any noticeable overhead per query, having too many
|
||||||
|
|
|
@ -172,24 +172,49 @@ This parameter sets the maximum number of documents to cache for any entry in th
|
||||||
<queryResultMaxDocsCached>200</queryResultMaxDocsCached>
|
<queryResultMaxDocsCached>200</queryResultMaxDocsCached>
|
||||||
----
|
----
|
||||||
|
|
||||||
=== useCircuitBreakers
|
=== circuitBreaker
|
||||||
|
|
||||||
Global control flag for enabling circuit breakers.
|
This set of configurations control the behaviour of circuit breakers.
|
||||||
|
|
||||||
[source,xml]
|
[source,xml]
|
||||||
----
|
----
|
||||||
<useCircuitBreakers>true</useCircuitBreakers>
|
<circuitBreaker class="solr.CircuitBreakerManager" enabled="true">
|
||||||
|
<!-- All specific configs in this section -->
|
||||||
|
</circuitBreaker>
|
||||||
----
|
----
|
||||||
|
|
||||||
=== memoryCircuitBreakerThresholdPct
|
To control whether Circuit Breakers are globally enabled, use the "enabled" attribute.
|
||||||
|
|
||||||
|
=== Memory Circuit Breaker Settings
|
||||||
|
|
||||||
|
To turn memory circuit breaker on/off, use the following flag:
|
||||||
|
[source,xml]
|
||||||
|
----
|
||||||
|
<str name="memEnabled">true</str>
|
||||||
|
----
|
||||||
|
|
||||||
Memory threshold in percentage for JVM heap usage defined in percentage of maximum heap allocated
|
Memory threshold in percentage for JVM heap usage defined in percentage of maximum heap allocated
|
||||||
to the JVM (-Xmx). Ideally, this value should be in the range of 75-80% of maximum heap allocated
|
to the JVM (-Xmx). Ideally, this value should be in the range of 75-80% of maximum heap allocated
|
||||||
to the JVM.
|
to the JVM. The enabled flag can be used to control the specific toggle for this circuit breaker.
|
||||||
|
|
||||||
[source,xml]
|
[source,xml]
|
||||||
----
|
----
|
||||||
<memoryCircuitBreakerThresholdPct>75</memoryCircuitBreakerThresholdPct>
|
<str name="memThreshold">75</str>
|
||||||
|
----
|
||||||
|
|
||||||
|
=== CPU Circuit Breaker Settings
|
||||||
|
|
||||||
|
To control turning on/off this feature, use the following flag:
|
||||||
|
[source,xml]
|
||||||
|
----
|
||||||
|
<str name="cpuEnabled">true</str>
|
||||||
|
----
|
||||||
|
|
||||||
|
Defines the triggering threshold in terms of the average per minute CPU load. The enabled flag can be used to control the specific toggle for this circuit breaker.
|
||||||
|
|
||||||
|
[source,xml]
|
||||||
|
----
|
||||||
|
<str name="cpuThreshold">75</str>
|
||||||
----
|
----
|
||||||
|
|
||||||
=== useColdSearcher
|
=== useColdSearcher
|
||||||
|
|
Loading…
Reference in New Issue