Use original settings on full-cluster restart (#30780)

When doing a node restart using the test framework, the restarted node does not only use the
settings provided to the original node, but also additional settings provided by plugin extensions,
which does not correspond to the settings that a node would have on a true restart.
This commit is contained in:
Yannick Welsch 2018-05-23 09:02:01 +02:00 committed by GitHub
parent cceaa9a0f1
commit 30b004f582
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 38 additions and 9 deletions

View File

@ -230,6 +230,7 @@ public class Node implements Closeable {
private final Lifecycle lifecycle = new Lifecycle(); private final Lifecycle lifecycle = new Lifecycle();
private final Injector injector; private final Injector injector;
private final Settings settings; private final Settings settings;
private final Settings originalSettings;
private final Environment environment; private final Environment environment;
private final NodeEnvironment nodeEnvironment; private final NodeEnvironment nodeEnvironment;
private final PluginsService pluginsService; private final PluginsService pluginsService;
@ -260,6 +261,7 @@ public class Node implements Closeable {
logger.info("initializing ..."); logger.info("initializing ...");
} }
try { try {
originalSettings = environment.settings();
Settings tmpSettings = Settings.builder().put(environment.settings()) Settings tmpSettings = Settings.builder().put(environment.settings())
.put(Client.CLIENT_TYPE_SETTING_S.getKey(), CLIENT_TYPE).build(); .put(Client.CLIENT_TYPE_SETTING_S.getKey(), CLIENT_TYPE).build();
@ -563,7 +565,14 @@ public class Node implements Closeable {
} }
/** /**
* The settings that were used to create the node. * The original settings that were used to create the node
*/
public Settings originalSettings() {
return originalSettings;
}
/**
* The settings that are used by this node. Contains original settings as well as additional settings provided by plugins.
*/ */
public Settings settings() { public Settings settings() {
return this.settings; return this.settings;

View File

@ -906,7 +906,7 @@ public final class InternalTestCluster extends TestCluster {
private void createNewNode(final Settings newSettings) { private void createNewNode(final Settings newSettings) {
final long newIdSeed = NodeEnvironment.NODE_ID_SEED_SETTING.get(node.settings()) + 1; // use a new seed to make sure we have new node id final long newIdSeed = NodeEnvironment.NODE_ID_SEED_SETTING.get(node.settings()) + 1; // use a new seed to make sure we have new node id
Settings finalSettings = Settings.builder().put(node.settings()).put(newSettings).put(NodeEnvironment.NODE_ID_SEED_SETTING.getKey(), newIdSeed).build(); Settings finalSettings = Settings.builder().put(node.originalSettings()).put(newSettings).put(NodeEnvironment.NODE_ID_SEED_SETTING.getKey(), newIdSeed).build();
if (DISCOVERY_ZEN_MINIMUM_MASTER_NODES_SETTING.exists(finalSettings) == false) { if (DISCOVERY_ZEN_MINIMUM_MASTER_NODES_SETTING.exists(finalSettings) == false) {
throw new IllegalStateException(DISCOVERY_ZEN_MINIMUM_MASTER_NODES_SETTING.getKey() + throw new IllegalStateException(DISCOVERY_ZEN_MINIMUM_MASTER_NODES_SETTING.getKey() +
" is not configured after restart of [" + name + "]"); " is not configured after restart of [" + name + "]");

View File

@ -469,9 +469,11 @@ public class InternalTestClusterTests extends ESTestCase {
}; };
String nodePrefix = "test"; String nodePrefix = "test";
Path baseDir = createTempDir(); Path baseDir = createTempDir();
List<Class<? extends Plugin>> plugins = new ArrayList<>(mockPlugins());
plugins.add(NodeAttrCheckPlugin.class);
InternalTestCluster cluster = new InternalTestCluster(randomLong(), baseDir, false, true, 2, 2, InternalTestCluster cluster = new InternalTestCluster(randomLong(), baseDir, false, true, 2, 2,
"test", nodeConfigurationSource, 0, nodePrefix, "test", nodeConfigurationSource, 0, nodePrefix,
mockPlugins(), Function.identity()); plugins, Function.identity());
try { try {
cluster.beforeTest(random(), 0.0); cluster.beforeTest(random(), 0.0);
assertMMNinNodeSetting(cluster, 2); assertMMNinNodeSetting(cluster, 2);
@ -502,4 +504,26 @@ public class InternalTestClusterTests extends ESTestCase {
cluster.close(); cluster.close();
} }
} }
/**
* Plugin that adds a simple node attribute as setting and checks if that node attribute is not already defined.
* Allows to check that the full-cluster restart logic does not copy over plugin-derived settings.
*/
public static class NodeAttrCheckPlugin extends Plugin {
private final Settings settings;
public NodeAttrCheckPlugin(Settings settings) {
this.settings = settings;
}
@Override
public Settings additionalSettings() {
if (settings.get("node.attr.dummy") != null) {
fail("dummy setting already exists");
}
return Settings.builder().put("node.attr.dummy", true).build();
}
}
} }

View File

@ -316,12 +316,8 @@ public class MachineLearning extends Plugin implements ActionPlugin, AnalysisPlu
} }
private void addMlNodeAttribute(Settings.Builder additionalSettings, String attrName, String value) { private void addMlNodeAttribute(Settings.Builder additionalSettings, String attrName, String value) {
// Unfortunately we cannot simply disallow any value, because the internal cluster integration
// test framework will restart nodes with settings copied from the node immediately before it
// was stopped. The best we can do is reject inconsistencies, and report this in a way that
// makes clear that setting the node attribute directly is not allowed.
String oldValue = settings.get(attrName); String oldValue = settings.get(attrName);
if (oldValue == null || oldValue.equals(value)) { if (oldValue == null) {
additionalSettings.put(attrName, value); additionalSettings.put(attrName, value);
} else { } else {
reportClashingNodeAttribute(attrName); reportClashingNodeAttribute(attrName);