Settings: Ensure fields are overriden and not merged when using arrays
In the case you try to merge two settings, one being an array and one being a field, together, the settings were merged instead of being overridden. First config my.value: 1 Second config my.value: [ 2, 3 ] If you execute settingsBuilder().put(settings1).put(settings2).build() now only values 2,3 will be in the final settings Closes #8381
This commit is contained in:
parent
a5127d2ffd
commit
8626c18ad9
|
@ -20,7 +20,9 @@
|
||||||
package org.elasticsearch.common.settings;
|
package org.elasticsearch.common.settings;
|
||||||
|
|
||||||
import com.google.common.base.Charsets;
|
import com.google.common.base.Charsets;
|
||||||
|
import com.google.common.base.Predicate;
|
||||||
import com.google.common.collect.ImmutableMap;
|
import com.google.common.collect.ImmutableMap;
|
||||||
|
import com.google.common.collect.Iterables;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
import com.google.common.collect.Maps;
|
import com.google.common.collect.Maps;
|
||||||
import org.elasticsearch.ElasticsearchIllegalArgumentException;
|
import org.elasticsearch.ElasticsearchIllegalArgumentException;
|
||||||
|
@ -45,6 +47,8 @@ import java.nio.file.Files;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.regex.Matcher;
|
||||||
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
import static org.elasticsearch.common.Strings.toCamelCase;
|
import static org.elasticsearch.common.Strings.toCamelCase;
|
||||||
import static org.elasticsearch.common.unit.ByteSizeValue.parseBytesSizeValue;
|
import static org.elasticsearch.common.unit.ByteSizeValue.parseBytesSizeValue;
|
||||||
|
@ -57,6 +61,7 @@ import static org.elasticsearch.common.unit.TimeValue.parseTimeValue;
|
||||||
public class ImmutableSettings implements Settings {
|
public class ImmutableSettings implements Settings {
|
||||||
|
|
||||||
public static final Settings EMPTY = new Builder().build();
|
public static final Settings EMPTY = new Builder().build();
|
||||||
|
private final static Pattern ARRAY_PATTERN = Pattern.compile("(.*)\\.\\d+$");
|
||||||
|
|
||||||
private ImmutableMap<String, String> settings;
|
private ImmutableMap<String, String> settings;
|
||||||
private final ImmutableMap<String, String> forcedUnderscoreSettings;
|
private final ImmutableMap<String, String> forcedUnderscoreSettings;
|
||||||
|
@ -873,6 +878,7 @@ public class ImmutableSettings implements Settings {
|
||||||
* Sets all the provided settings.
|
* Sets all the provided settings.
|
||||||
*/
|
*/
|
||||||
public Builder put(Settings settings) {
|
public Builder put(Settings settings) {
|
||||||
|
removeNonArraysFieldsIfNewSettingsContainsFieldAsArray(settings.getAsMap());
|
||||||
map.putAll(settings.getAsMap());
|
map.putAll(settings.getAsMap());
|
||||||
classLoader = settings.getClassLoaderIfSet();
|
classLoader = settings.getClassLoaderIfSet();
|
||||||
return this;
|
return this;
|
||||||
|
@ -882,10 +888,42 @@ public class ImmutableSettings implements Settings {
|
||||||
* Sets all the provided settings.
|
* Sets all the provided settings.
|
||||||
*/
|
*/
|
||||||
public Builder put(Map<String, String> settings) {
|
public Builder put(Map<String, String> settings) {
|
||||||
|
removeNonArraysFieldsIfNewSettingsContainsFieldAsArray(settings);
|
||||||
map.putAll(settings);
|
map.putAll(settings);
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes non array values from the existing map, if settings contains an array value instead
|
||||||
|
*
|
||||||
|
* Example:
|
||||||
|
* Existing map contains: {key:value}
|
||||||
|
* New map contains: {key:[value1,value2]} (which has been flattened to {}key.0:value1,key.1:value2})
|
||||||
|
*
|
||||||
|
* This ensure that that the 'key' field gets removed from the map in order to override all the
|
||||||
|
* data instead of merging
|
||||||
|
*/
|
||||||
|
private void removeNonArraysFieldsIfNewSettingsContainsFieldAsArray(Map<String, String> settings) {
|
||||||
|
List<String> prefixesToRemove = new ArrayList<>();
|
||||||
|
for (final Map.Entry<String, String> entry : settings.entrySet()) {
|
||||||
|
final Matcher matcher = ARRAY_PATTERN.matcher(entry.getKey());
|
||||||
|
if (matcher.matches()) {
|
||||||
|
prefixesToRemove.add(matcher.group(1));
|
||||||
|
} else if (Iterables.any(map.keySet(), startsWith(entry.getKey() + "."))) {
|
||||||
|
prefixesToRemove.add(entry.getKey());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (String prefix : prefixesToRemove) {
|
||||||
|
Iterator<Map.Entry<String, String>> iterator = map.entrySet().iterator();
|
||||||
|
while (iterator.hasNext()) {
|
||||||
|
Map.Entry<String, String> entry = iterator.next();
|
||||||
|
if (entry.getKey().startsWith(prefix + ".") || entry.getKey().equals(prefix)) {
|
||||||
|
iterator.remove();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets all the provided settings.
|
* Sets all the provided settings.
|
||||||
*/
|
*/
|
||||||
|
@ -1090,4 +1128,22 @@ public class ImmutableSettings implements Settings {
|
||||||
return new ImmutableSettings(Collections.unmodifiableMap(map), classLoader);
|
return new ImmutableSettings(Collections.unmodifiableMap(map), classLoader);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static StartsWithPredicate startsWith(String prefix) {
|
||||||
|
return new StartsWithPredicate(prefix);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final class StartsWithPredicate implements Predicate<String> {
|
||||||
|
|
||||||
|
private String prefix;
|
||||||
|
|
||||||
|
public StartsWithPredicate(String prefix) {
|
||||||
|
this.prefix = prefix;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(String input) {
|
||||||
|
return input.startsWith(prefix);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,10 +21,12 @@ package org.elasticsearch.common.settings;
|
||||||
|
|
||||||
import org.elasticsearch.common.settings.bar.BarTestClass;
|
import org.elasticsearch.common.settings.bar.BarTestClass;
|
||||||
import org.elasticsearch.common.settings.foo.FooTestClass;
|
import org.elasticsearch.common.settings.foo.FooTestClass;
|
||||||
|
import org.elasticsearch.common.settings.loader.YamlSettingsLoader;
|
||||||
import org.elasticsearch.test.ElasticsearchTestCase;
|
import org.elasticsearch.test.ElasticsearchTestCase;
|
||||||
import org.hamcrest.Matchers;
|
import org.hamcrest.Matchers;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
@ -216,6 +218,110 @@ public class ImmutableSettingsTests extends ElasticsearchTestCase {
|
||||||
assertThat(names.size(), equalTo(2));
|
assertThat(names.size(), equalTo(2));
|
||||||
assertTrue(names.contains("bar"));
|
assertTrue(names.contains("bar"));
|
||||||
assertTrue(names.contains("baz"));
|
assertTrue(names.contains("baz"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testThatArraysAreOverriddenCorrectly() throws IOException {
|
||||||
|
// overriding a single value with an array
|
||||||
|
Settings settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value", "1").build())
|
||||||
|
.put(settingsBuilder().putArray("value", "2", "3").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value"), arrayContaining("2", "3"));
|
||||||
|
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().put("value", "1").build())
|
||||||
|
.put(settingsBuilder().putArray("value", "2", "3").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value"), arrayContaining("2", "3"));
|
||||||
|
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(new YamlSettingsLoader().load("value: 1"))
|
||||||
|
.put(new YamlSettingsLoader().load("value: [ 2, 3 ]"))
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value"), arrayContaining("2", "3"));
|
||||||
|
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().put("value.with.deep.key", "1").build())
|
||||||
|
.put(settingsBuilder().putArray("value.with.deep.key", "2", "3").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value.with.deep.key"), arrayContaining("2", "3"));
|
||||||
|
|
||||||
|
// overriding an array with a shorter array
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value", "1", "2").build())
|
||||||
|
.put(settingsBuilder().putArray("value", "3").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value"), arrayContaining("3"));
|
||||||
|
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value", "1", "2", "3").build())
|
||||||
|
.put(settingsBuilder().putArray("value", "4", "5").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value"), arrayContaining("4", "5"));
|
||||||
|
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value.deep.key", "1", "2", "3").build())
|
||||||
|
.put(settingsBuilder().putArray("value.deep.key", "4", "5").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value.deep.key"), arrayContaining("4", "5"));
|
||||||
|
|
||||||
|
// overriding an array with a longer array
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value", "1", "2").build())
|
||||||
|
.put(settingsBuilder().putArray("value", "3", "4", "5").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value"), arrayContaining("3", "4", "5"));
|
||||||
|
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value.deep.key", "1", "2", "3").build())
|
||||||
|
.put(settingsBuilder().putArray("value.deep.key", "4", "5").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value.deep.key"), arrayContaining("4", "5"));
|
||||||
|
|
||||||
|
// overriding an array with a single value
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value", "1", "2").build())
|
||||||
|
.put(settingsBuilder().put("value", "3").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value"), arrayContaining("3"));
|
||||||
|
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value.deep.key", "1", "2").build())
|
||||||
|
.put(settingsBuilder().put("value.deep.key", "3").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value.deep.key"), arrayContaining("3"));
|
||||||
|
|
||||||
|
// test that other arrays are not overridden
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value", "1", "2", "3").putArray("a", "b", "c").build())
|
||||||
|
.put(settingsBuilder().putArray("value", "4", "5").putArray("d", "e", "f").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value"), arrayContaining("4", "5"));
|
||||||
|
assertThat(settings.getAsArray("a"), arrayContaining("b", "c"));
|
||||||
|
assertThat(settings.getAsArray("d"), arrayContaining("e", "f"));
|
||||||
|
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value.deep.key", "1", "2", "3").putArray("a", "b", "c").build())
|
||||||
|
.put(settingsBuilder().putArray("value.deep.key", "4", "5").putArray("d", "e", "f").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value.deep.key"), arrayContaining("4", "5"));
|
||||||
|
assertThat(settings.getAsArray("a"), notNullValue());
|
||||||
|
assertThat(settings.getAsArray("d"), notNullValue());
|
||||||
|
|
||||||
|
// overriding a deeper structure with an array
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().put("value.data", "1").build())
|
||||||
|
.put(settingsBuilder().putArray("value", "4", "5").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.getAsArray("value"), arrayContaining("4", "5"));
|
||||||
|
|
||||||
|
// overriding an array with a deeper structure
|
||||||
|
settings = settingsBuilder()
|
||||||
|
.put(settingsBuilder().putArray("value", "4", "5").build())
|
||||||
|
.put(settingsBuilder().put("value.data", "1").build())
|
||||||
|
.build();
|
||||||
|
assertThat(settings.get("value.data"), is("1"));
|
||||||
|
assertThat(settings.get("value"), is(nullValue()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue