Skip to content

Commit 6abe4c9

Browse files
committed
Add HierarchyCircuitBreakerService
Adds a breaker for request BigArrays, which are used for parent/child queries as well as some aggregations. Certain operations like Netty HTTP responses and transport responses increment the breaker, but will not trip. This also changes the output of the nodes' stats endpoint to show the parent breaker as well as the fielddata and request breakers. There are a number of new settings for breakers now: `indices.breaker.total.limit`: starting limit for all memory-use breaker, defaults to 70% `indices.breaker.fielddata.limit`: starting limit for fielddata breaker, defaults to 60% `indices.breaker.fielddata.overhead`: overhead for fielddata breaker estimations, defaults to 1.03 (the fielddata breaker settings also use the backwards-compatible setting `indices.fielddata.breaker.limit` and `indices.fielddata.breaker.overhead`) `indices.breaker.request.limit`: starting limit for request breaker, defaults to 40% `indices.breaker.request.overhead`: request breaker estimation overhead, defaults to 1.0 The breaker service infrastructure is now generic and opens the path to adding additional circuit breakers in the future. Fixes #6129 Conflicts: src/main/java/org/elasticsearch/index/fielddata/IndexFieldData.java src/main/java/org/elasticsearch/index/fielddata/IndexFieldDataService.java src/main/java/org/elasticsearch/index/fielddata/RamAccountingTermsEnum.java src/main/java/org/elasticsearch/index/fielddata/ordinals/GlobalOrdinalsBuilder.java src/main/java/org/elasticsearch/index/fielddata/ordinals/InternalGlobalOrdinalsBuilder.java src/main/java/org/elasticsearch/index/fielddata/plain/AbstractIndexOrdinalsFieldData.java src/main/java/org/elasticsearch/index/fielddata/plain/DisabledIndexFieldData.java src/main/java/org/elasticsearch/index/fielddata/plain/IndexIndexFieldData.java src/main/java/org/elasticsearch/index/fielddata/plain/NonEstimatingEstimator.java src/main/java/org/elasticsearch/index/fielddata/plain/PackedArrayIndexFieldData.java src/main/java/org/elasticsearch/index/fielddata/plain/ParentChildIndexFieldData.java src/main/java/org/elasticsearch/index/fielddata/plain/SortedSetDVOrdinalsIndexFieldData.java src/main/java/org/elasticsearch/node/internal/InternalNode.java src/test/java/org/elasticsearch/index/aliases/IndexAliasesServiceTests.java src/test/java/org/elasticsearch/index/codec/CodecTests.java src/test/java/org/elasticsearch/index/fielddata/AbstractFieldDataTests.java src/test/java/org/elasticsearch/index/fielddata/IndexFieldDataServiceTests.java src/test/java/org/elasticsearch/index/mapper/MapperTestUtils.java src/test/java/org/elasticsearch/index/query/IndexQueryParserFilterCachingTests.java src/test/java/org/elasticsearch/index/query/SimpleIndexQueryParserTests.java src/test/java/org/elasticsearch/index/query/guice/IndexQueryParserModuleTests.java src/test/java/org/elasticsearch/index/search/FieldDataTermsFilterTests.java src/test/java/org/elasticsearch/index/search/child/ChildrenConstantScoreQueryTests.java src/test/java/org/elasticsearch/index/similarity/SimilarityTests.java
1 parent be86556 commit 6abe4c9

File tree

59 files changed

+1640
-415
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

59 files changed

+1640
-415
lines changed

docs/reference/index-modules/fielddata.asciidoc

+48-13
Original file line numberDiff line numberDiff line change
@@ -24,28 +24,63 @@ field data after a certain time of inactivity. Defaults to `-1`. For
2424
example, can be set to `5m` for a 5 minute expiry.
2525
|=======================================================================
2626

27+
[float]
28+
[[circuit-breaker]]
29+
=== Circuit Breaker
30+
31+
coming[1.4.0,Prior to 1.4.0 there was only a single circuit breaker for fielddata]
32+
33+
Elasticsearch contains multiple circuit breakers used to prevent operations from
34+
causing an OutOfMemoryError. Each breaker specifies a limit for how much memory
35+
it can use. Additionally, there is a parent-level breaker that specifies the
36+
total amount of memory that can be used across all breakers.
37+
38+
The parent-level breaker can be configured with the following setting:
39+
40+
`indices.breaker.total.limit`::
41+
Starting limit for overall parent breaker, defaults to 70% of JVM heap
42+
43+
All circuit breaker settings can be changed dynamically using the cluster update
44+
settings API.
45+
2746
[float]
2847
[[fielddata-circuit-breaker]]
29-
=== Field data circuit breaker
48+
==== Field data circuit breaker
3049
The field data circuit breaker allows Elasticsearch to estimate the amount of
3150
memory a field will required to be loaded into memory. It can then prevent the
3251
field data loading by raising an exception. By default the limit is configured
3352
to 60% of the maximum JVM heap. It can be configured with the following
3453
parameters:
3554

36-
[cols="<,<",options="header",]
37-
|=======================================================================
38-
|Setting |Description
39-
|`indices.fielddata.breaker.limit` |Maximum size of estimated field data
40-
to allow loading. Defaults to 60% of the maximum JVM heap.
41-
|`indices.fielddata.breaker.overhead` |A constant that all field data
42-
estimations are multiplied with to determine a final estimation. Defaults to
43-
1.03
44-
|=======================================================================
55+
`indices.breaker.fielddata.limit`::
56+
Limit for fielddata breaker, defaults to 60% of JVM heap
57+
58+
`indices.breaker.fielddata.overhead`::
59+
A constant that all field data estimations are multiplied with to determine a
60+
final estimation. Defaults to 1.03
61+
62+
`indices.fielddata.breaker.limit`::
63+
deprecated[1.4.0,Replaced by `indices.breaker.fielddata.limit`]
64+
65+
`indices.fielddata.breaker.overhead`::
66+
deprecated[1.4.0,Replaced by `indices.breaker.fielddata.overhead`]
67+
68+
[float]
69+
[[request-circuit-breaker]]
70+
==== Request circuit breaker
71+
72+
coming[1.4.0]
73+
74+
The request circuit breaker allows Elasticsearch to prevent per-request data
75+
structures (for example, memory used for calculating aggregations during a
76+
request) from exceeding a certain amount of memory.
77+
78+
`indices.breaker.request.limit`::
79+
Limit for request breaker, defaults to 40% of JVM heap
4580

46-
Both the `indices.fielddata.breaker.limit` and
47-
`indices.fielddata.breaker.overhead` can be changed dynamically using the
48-
cluster update settings API.
81+
`indices.breaker.request.overhead`::
82+
A constant that all request estimations are multiplied with to determine a
83+
final estimation. Defaults to 1
4984

5085
[float]
5186
[[fielddata-monitoring]]

src/main/java/org/elasticsearch/action/admin/cluster/node/stats/NodeStats.java

+5-5
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@
2828
import org.elasticsearch.common.xcontent.XContentBuilder;
2929
import org.elasticsearch.http.HttpStats;
3030
import org.elasticsearch.indices.NodeIndicesStats;
31-
import org.elasticsearch.indices.fielddata.breaker.FieldDataBreakerStats;
31+
import org.elasticsearch.indices.breaker.AllCircuitBreakerStats;
3232
import org.elasticsearch.monitor.fs.FsStats;
3333
import org.elasticsearch.monitor.jvm.JvmStats;
3434
import org.elasticsearch.monitor.network.NetworkStats;
@@ -75,15 +75,15 @@ public class NodeStats extends NodeOperationResponse implements ToXContent {
7575
private HttpStats http;
7676

7777
@Nullable
78-
private FieldDataBreakerStats breaker;
78+
private AllCircuitBreakerStats breaker;
7979

8080
NodeStats() {
8181
}
8282

8383
public NodeStats(DiscoveryNode node, long timestamp, @Nullable NodeIndicesStats indices,
8484
@Nullable OsStats os, @Nullable ProcessStats process, @Nullable JvmStats jvm, @Nullable ThreadPoolStats threadPool,
8585
@Nullable NetworkStats network, @Nullable FsStats fs, @Nullable TransportStats transport, @Nullable HttpStats http,
86-
@Nullable FieldDataBreakerStats breaker) {
86+
@Nullable AllCircuitBreakerStats breaker) {
8787
super(node);
8888
this.timestamp = timestamp;
8989
this.indices = indices;
@@ -174,7 +174,7 @@ public HttpStats getHttp() {
174174
}
175175

176176
@Nullable
177-
public FieldDataBreakerStats getBreaker() {
177+
public AllCircuitBreakerStats getBreaker() {
178178
return this.breaker;
179179
}
180180

@@ -215,7 +215,7 @@ public void readFrom(StreamInput in) throws IOException {
215215
if (in.readBoolean()) {
216216
http = HttpStats.readHttpStats(in);
217217
}
218-
breaker = FieldDataBreakerStats.readOptionalCircuitBreakerStats(in);
218+
breaker = AllCircuitBreakerStats.readOptionalAllCircuitBreakerStats(in);
219219
}
220220

221221
@Override

src/main/java/org/elasticsearch/client/transport/TransportClient.java

+2
Original file line numberDiff line numberDiff line change
@@ -76,6 +76,7 @@
7676
import org.elasticsearch.common.transport.TransportAddress;
7777
import org.elasticsearch.env.Environment;
7878
import org.elasticsearch.env.EnvironmentModule;
79+
import org.elasticsearch.indices.breaker.CircuitBreakerModule;
7980
import org.elasticsearch.monitor.MonitorService;
8081
import org.elasticsearch.node.internal.InternalSettingsPreparer;
8182
import org.elasticsearch.plugins.PluginsModule;
@@ -187,6 +188,7 @@ public TransportClient(Settings pSettings, boolean loadConfigSettings) throws El
187188
modules.add(new TransportModule(this.settings));
188189
modules.add(new ActionModule(true));
189190
modules.add(new ClientTransportModule());
191+
modules.add(new CircuitBreakerModule(this.settings));
190192

191193
injector = modules.createInjector();
192194

src/main/java/org/elasticsearch/cluster/settings/ClusterDynamicSettingsModule.java

+7-1
Original file line numberDiff line numberDiff line change
@@ -28,8 +28,9 @@
2828
import org.elasticsearch.common.inject.AbstractModule;
2929
import org.elasticsearch.discovery.DiscoverySettings;
3030
import org.elasticsearch.discovery.zen.elect.ElectMasterService;
31+
import org.elasticsearch.indices.breaker.HierarchyCircuitBreakerService;
3132
import org.elasticsearch.indices.cache.filter.IndicesFilterCache;
32-
import org.elasticsearch.indices.fielddata.breaker.InternalCircuitBreakerService;
33+
import org.elasticsearch.indices.breaker.InternalCircuitBreakerService;
3334
import org.elasticsearch.indices.recovery.RecoverySettings;
3435
import org.elasticsearch.indices.store.IndicesStore;
3536
import org.elasticsearch.indices.ttl.IndicesTTLService;
@@ -85,6 +86,11 @@ public ClusterDynamicSettingsModule() {
8586
clusterDynamicSettings.addDynamicSetting(InternalCircuitBreakerService.CIRCUIT_BREAKER_OVERHEAD_SETTING, Validator.NON_NEGATIVE_DOUBLE);
8687
clusterDynamicSettings.addDynamicSetting(DestructiveOperations.REQUIRES_NAME);
8788
clusterDynamicSettings.addDynamicSetting(DiscoverySettings.PUBLISH_TIMEOUT, Validator.TIME_NON_NEGATIVE);
89+
clusterDynamicSettings.addDynamicSetting(HierarchyCircuitBreakerService.TOTAL_CIRCUIT_BREAKER_LIMIT_SETTING, Validator.MEMORY_SIZE);
90+
clusterDynamicSettings.addDynamicSetting(HierarchyCircuitBreakerService.FIELDDATA_CIRCUIT_BREAKER_LIMIT_SETTING, Validator.MEMORY_SIZE);
91+
clusterDynamicSettings.addDynamicSetting(HierarchyCircuitBreakerService.FIELDDATA_CIRCUIT_BREAKER_OVERHEAD_SETTING, Validator.NON_NEGATIVE_DOUBLE);
92+
clusterDynamicSettings.addDynamicSetting(HierarchyCircuitBreakerService.REQUEST_CIRCUIT_BREAKER_LIMIT_SETTING, Validator.MEMORY_SIZE);
93+
clusterDynamicSettings.addDynamicSetting(HierarchyCircuitBreakerService.REQUEST_CIRCUIT_BREAKER_OVERHEAD_SETTING, Validator.NON_NEGATIVE_DOUBLE);
8894
}
8995

9096
public void addDynamicSettings(String... settings) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,224 @@
1+
/*
2+
* Licensed to Elasticsearch under one or more contributor
3+
* license agreements. See the NOTICE file distributed with
4+
* this work for additional information regarding copyright
5+
* ownership. Elasticsearch licenses this file to you under
6+
* the Apache License, Version 2.0 (the "License"); you may
7+
* not use this file except in compliance with the License.
8+
* You may obtain a copy of the License at
9+
*
10+
* http://www.apache.org/licenses/LICENSE-2.0
11+
*
12+
* Unless required by applicable law or agreed to in writing,
13+
* software distributed under the License is distributed on an
14+
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15+
* KIND, either express or implied. See the License for the
16+
* specific language governing permissions and limitations
17+
* under the License.
18+
*/
19+
20+
package org.elasticsearch.common.breaker;
21+
22+
import org.elasticsearch.common.logging.ESLogger;
23+
import org.elasticsearch.common.unit.ByteSizeValue;
24+
import org.elasticsearch.indices.breaker.BreakerSettings;
25+
import org.elasticsearch.indices.breaker.HierarchyCircuitBreakerService;
26+
27+
import java.util.concurrent.atomic.AtomicLong;
28+
29+
/**
30+
* Breaker that will check a parent's when incrementing
31+
*/
32+
public class ChildMemoryCircuitBreaker implements CircuitBreaker {
33+
34+
private final long memoryBytesLimit;
35+
private final BreakerSettings settings;
36+
private final double overheadConstant;
37+
private final AtomicLong used;
38+
private final AtomicLong trippedCount;
39+
private final ESLogger logger;
40+
private final HierarchyCircuitBreakerService parent;
41+
private final Name name;
42+
43+
/**
44+
* Create a circuit breaker that will break if the number of estimated
45+
* bytes grows above the limit. All estimations will be multiplied by
46+
* the given overheadConstant. This breaker starts with 0 bytes used.
47+
* @param settings settings to configure this breaker
48+
* @param parent parent circuit breaker service to delegate tripped breakers to
49+
* @param name the name of the breaker
50+
*/
51+
public ChildMemoryCircuitBreaker(BreakerSettings settings, ESLogger logger,
52+
HierarchyCircuitBreakerService parent, Name name) {
53+
this(settings, null, logger, parent, name);
54+
}
55+
56+
/**
57+
* Create a circuit breaker that will break if the number of estimated
58+
* bytes grows above the limit. All estimations will be multiplied by
59+
* the given overheadConstant. Uses the given oldBreaker to initialize
60+
* the starting offset.
61+
* @param settings settings to configure this breaker
62+
* @param parent parent circuit breaker service to delegate tripped breakers to
63+
* @param name the name of the breaker
64+
* @param oldBreaker the previous circuit breaker to inherit the used value from (starting offset)
65+
*/
66+
public ChildMemoryCircuitBreaker(BreakerSettings settings, ChildMemoryCircuitBreaker oldBreaker,
67+
ESLogger logger, HierarchyCircuitBreakerService parent, Name name) {
68+
this.name = name;
69+
this.settings = settings;
70+
this.memoryBytesLimit = settings.getLimit();
71+
this.overheadConstant = settings.getOverhead();
72+
if (oldBreaker == null) {
73+
this.used = new AtomicLong(0);
74+
this.trippedCount = new AtomicLong(0);
75+
} else {
76+
this.used = oldBreaker.used;
77+
this.trippedCount = oldBreaker.trippedCount;
78+
}
79+
this.logger = logger;
80+
if (logger.isTraceEnabled()) {
81+
logger.trace("creating ChildCircuitBreaker with settings {}", this.settings);
82+
}
83+
this.parent = parent;
84+
}
85+
86+
/**
87+
* Method used to trip the breaker, delegates to the parent to determine
88+
* whether to trip the breaker or not
89+
*/
90+
@Override
91+
public void circuitBreak(String fieldName, long bytesNeeded) {
92+
this.trippedCount.incrementAndGet();
93+
throw new CircuitBreakingException("[" + this.name + "] Data too large, data for [" +
94+
fieldName + "] would be larger than limit of [" +
95+
memoryBytesLimit + "/" + new ByteSizeValue(memoryBytesLimit) + "]",
96+
bytesNeeded, this.memoryBytesLimit);
97+
}
98+
99+
/**
100+
* Add a number of bytes, tripping the circuit breaker if the aggregated
101+
* estimates are above the limit. Automatically trips the breaker if the
102+
* memory limit is set to 0. Will never trip the breaker if the limit is
103+
* set < 0, but can still be used to aggregate estimations.
104+
* @param bytes number of bytes to add to the breaker
105+
* @return number of "used" bytes so far
106+
* @throws CircuitBreakingException
107+
*/
108+
@Override
109+
public double addEstimateBytesAndMaybeBreak(long bytes, String label) throws CircuitBreakingException {
110+
// short-circuit on no data allowed, immediately throwing an exception
111+
if (memoryBytesLimit == 0) {
112+
circuitBreak(label, bytes);
113+
}
114+
115+
long newUsed;
116+
// If there is no limit (-1), we can optimize a bit by using
117+
// .addAndGet() instead of looping (because we don't have to check a
118+
// limit), which makes the RamAccountingTermsEnum case faster.
119+
if (this.memoryBytesLimit == -1) {
120+
newUsed = this.used.addAndGet(bytes);
121+
if (logger.isTraceEnabled()) {
122+
logger.trace("[{}] Adding [{}][{}] to used bytes [new used: [{}], limit: [-1b]]",
123+
this.name, new ByteSizeValue(bytes), label, new ByteSizeValue(newUsed));
124+
}
125+
} else {
126+
// Otherwise, check the addition and commit the addition, looping if
127+
// there are conflicts. May result in additional logging, but it's
128+
// trace logging and shouldn't be counted on for additions.
129+
long currentUsed;
130+
do {
131+
currentUsed = this.used.get();
132+
newUsed = currentUsed + bytes;
133+
long newUsedWithOverhead = (long) (newUsed * overheadConstant);
134+
if (logger.isTraceEnabled()) {
135+
logger.trace("[{}] Adding [{}][{}] to used bytes [new used: [{}], limit: {} [{}], estimate: {} [{}]]",
136+
this.name,
137+
new ByteSizeValue(bytes), label, new ByteSizeValue(newUsed),
138+
memoryBytesLimit, new ByteSizeValue(memoryBytesLimit),
139+
newUsedWithOverhead, new ByteSizeValue(newUsedWithOverhead));
140+
}
141+
if (memoryBytesLimit > 0 && newUsedWithOverhead > memoryBytesLimit) {
142+
logger.error("[{}] New used memory {} [{}] from field [{}] would be larger than configured breaker: {} [{}], breaking",
143+
this.name,
144+
newUsedWithOverhead, new ByteSizeValue(newUsedWithOverhead), label,
145+
memoryBytesLimit, new ByteSizeValue(memoryBytesLimit));
146+
circuitBreak(label, newUsedWithOverhead);
147+
}
148+
// Attempt to set the new used value, but make sure it hasn't changed
149+
// underneath us, if it has, keep trying until we are able to set it
150+
} while (!this.used.compareAndSet(currentUsed, newUsed));
151+
}
152+
153+
// Additionally, we need to check that we haven't exceeded the parent's limit
154+
try {
155+
parent.checkParentLimit(label);
156+
} catch (CircuitBreakingException e) {
157+
// If the parent breaker is tripped, this breaker has to be
158+
// adjusted back down because the allocation is "blocked" but the
159+
// breaker has already been incremented
160+
this.used.addAndGet(-bytes);
161+
throw e;
162+
}
163+
return newUsed;
164+
}
165+
166+
/**
167+
* Add an <b>exact</b> number of bytes, not checking for tripping the
168+
* circuit breaker. This bypasses the overheadConstant multiplication.
169+
*
170+
* Also does not check with the parent breaker to see if the parent limit
171+
* has been exceeded.
172+
*
173+
* @param bytes number of bytes to add to the breaker
174+
* @return number of "used" bytes so far
175+
*/
176+
@Override
177+
public long addWithoutBreaking(long bytes) {
178+
long u = used.addAndGet(bytes);
179+
if (logger.isTraceEnabled()) {
180+
logger.trace("[{}] Adjusted breaker by [{}] bytes, now [{}]", this.name, bytes, u);
181+
}
182+
assert u >= 0 : "Used bytes: [" + u + "] must be >= 0";
183+
return u;
184+
}
185+
186+
/**
187+
* @return the number of aggregated "used" bytes so far
188+
*/
189+
@Override
190+
public long getUsed() {
191+
return this.used.get();
192+
}
193+
194+
/**
195+
* @return the number of bytes that can be added before the breaker trips
196+
*/
197+
@Override
198+
public long getLimit() {
199+
return this.memoryBytesLimit;
200+
}
201+
202+
/**
203+
* @return the constant multiplier the breaker uses for aggregations
204+
*/
205+
@Override
206+
public double getOverhead() {
207+
return this.overheadConstant;
208+
}
209+
210+
/**
211+
* @return the number of times the breaker has been tripped
212+
*/
213+
@Override
214+
public long getTrippedCount() {
215+
return this.trippedCount.get();
216+
}
217+
218+
/**
219+
* @return the name of the breaker
220+
*/
221+
public Name getName() {
222+
return this.name;
223+
}
224+
}

0 commit comments

Comments
 (0)