Use GatewayMetaState in CoordinatorTests rarely (#36897)
This commit adds one more underlying implementation of MockPersistedState. Previously only InMemoryPersistentState was used, not GatewayMetaState is used rarely. When adding GatewayMetaState support the main question was: do we want to emulate exceptions as we do today in MockPersistedState before delegating to GatewayMetaState or do we want these exceptions to propagate from the lower level, i.e. file system exceptions? On the one hand, lower level exception propagation is already tested in GatewayMetaStateTests, so this won't improve the coverage. On the other hand, the benefit of low-level exceptions is to see how all these components work in conjunction. Finally, we abandoned the idea of low-level exceptions because we don't have a way to deal with IOError today in CoordinatorTests, but hacking GatewayMetaState not to throw IOError seems unnatural. So MockPersistedState rarely throws an exception before delegating to GatewayMetaState, which is not supposed to throw the exception. This commit required two changes: Move GatewayMetaStateUT to upper-level from GatewayMetaStatePersistedStateTests, because otherwise, it's not easy to construct GatewayMetaState instance in CoordinatorTests. Move addition of STATE_NOT_RECOVERED_BLOCK from GatewayMetaState constructor to GatewayMetaState.applyClusterUpdaters, because CoordinatorTests class assumes that there is no such block and most of them fail.
This commit is contained in:
parent
f8d80dff7c
commit
42fd68ed38
|
@ -130,6 +130,13 @@ public class ClusterStateUpdaters {
|
||||||
.build();
|
.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static ClusterState addStateNotRecoveredBlock(ClusterState state) {
|
||||||
|
return ClusterState.builder(state)
|
||||||
|
.blocks(ClusterBlocks.builder()
|
||||||
|
.blocks(state.blocks()).addGlobalBlock(GatewayService.STATE_NOT_RECOVERED_BLOCK).build())
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
|
||||||
static ClusterState mixCurrentStateAndRecoveredState(final ClusterState currentState, final ClusterState recoveredState) {
|
static ClusterState mixCurrentStateAndRecoveredState(final ClusterState currentState, final ClusterState recoveredState) {
|
||||||
assert currentState.metaData().indices().isEmpty();
|
assert currentState.metaData().indices().isEmpty();
|
||||||
|
|
||||||
|
|
|
@ -28,7 +28,6 @@ import org.elasticsearch.cluster.ClusterChangedEvent;
|
||||||
import org.elasticsearch.cluster.ClusterName;
|
import org.elasticsearch.cluster.ClusterName;
|
||||||
import org.elasticsearch.cluster.ClusterState;
|
import org.elasticsearch.cluster.ClusterState;
|
||||||
import org.elasticsearch.cluster.ClusterStateApplier;
|
import org.elasticsearch.cluster.ClusterStateApplier;
|
||||||
import org.elasticsearch.cluster.block.ClusterBlocks;
|
|
||||||
import org.elasticsearch.cluster.coordination.CoordinationState;
|
import org.elasticsearch.cluster.coordination.CoordinationState;
|
||||||
import org.elasticsearch.cluster.coordination.CoordinationState.PersistedState;
|
import org.elasticsearch.cluster.coordination.CoordinationState.PersistedState;
|
||||||
import org.elasticsearch.cluster.coordination.InMemoryPersistedState;
|
import org.elasticsearch.cluster.coordination.InMemoryPersistedState;
|
||||||
|
@ -64,8 +63,6 @@ import java.util.function.Consumer;
|
||||||
import java.util.function.Function;
|
import java.util.function.Function;
|
||||||
import java.util.function.UnaryOperator;
|
import java.util.function.UnaryOperator;
|
||||||
|
|
||||||
import static org.elasticsearch.gateway.GatewayService.STATE_NOT_RECOVERED_BLOCK;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class is responsible for storing/retrieving metadata to/from disk.
|
* This class is responsible for storing/retrieving metadata to/from disk.
|
||||||
* When instance of this class is created, constructor ensures that this version is compatible with state stored on disk and performs
|
* When instance of this class is created, constructor ensures that this version is compatible with state stored on disk and performs
|
||||||
|
@ -124,11 +121,9 @@ public class GatewayMetaState implements ClusterStateApplier, CoordinationState.
|
||||||
previousManifest = manifestAndMetaData.v1();
|
previousManifest = manifestAndMetaData.v1();
|
||||||
|
|
||||||
final MetaData metaData = manifestAndMetaData.v2();
|
final MetaData metaData = manifestAndMetaData.v2();
|
||||||
final ClusterBlocks.Builder blocks = ClusterBlocks.builder().addGlobalBlock(STATE_NOT_RECOVERED_BLOCK);
|
|
||||||
|
|
||||||
previousClusterState = ClusterState.builder(clusterName)
|
previousClusterState = ClusterState.builder(clusterName)
|
||||||
.version(previousManifest.getClusterStateVersion())
|
.version(previousManifest.getClusterStateVersion())
|
||||||
.blocks(blocks.build())
|
|
||||||
.metaData(metaData).build();
|
.metaData(metaData).build();
|
||||||
|
|
||||||
logger.debug("took {} to load state", TimeValue.timeValueMillis(TimeValue.nsecToMSec(System.nanoTime() - startNS)));
|
logger.debug("took {} to load state", TimeValue.timeValueMillis(TimeValue.nsecToMSec(System.nanoTime() - startNS)));
|
||||||
|
@ -139,6 +134,7 @@ public class GatewayMetaState implements ClusterStateApplier, CoordinationState.
|
||||||
assert transportService.getLocalNode() != null : "transport service is not yet started";
|
assert transportService.getLocalNode() != null : "transport service is not yet started";
|
||||||
|
|
||||||
previousClusterState = Function.<ClusterState>identity()
|
previousClusterState = Function.<ClusterState>identity()
|
||||||
|
.andThen(ClusterStateUpdaters::addStateNotRecoveredBlock)
|
||||||
.andThen(state -> ClusterStateUpdaters.setLocalNode(state, transportService.getLocalNode()))
|
.andThen(state -> ClusterStateUpdaters.setLocalNode(state, transportService.getLocalNode()))
|
||||||
.andThen(state -> ClusterStateUpdaters.upgradeAndArchiveUnknownOrInvalidSettings(state, clusterService.getClusterSettings()))
|
.andThen(state -> ClusterStateUpdaters.upgradeAndArchiveUnknownOrInvalidSettings(state, clusterService.getClusterSettings()))
|
||||||
.andThen(state -> ClusterStateUpdaters.closeBadIndices(state, indicesService))
|
.andThen(state -> ClusterStateUpdaters.closeBadIndices(state, indicesService))
|
||||||
|
|
|
@ -57,12 +57,15 @@ import org.elasticsearch.common.transport.TransportAddress;
|
||||||
import org.elasticsearch.common.unit.TimeValue;
|
import org.elasticsearch.common.unit.TimeValue;
|
||||||
import org.elasticsearch.discovery.zen.PublishClusterStateStats;
|
import org.elasticsearch.discovery.zen.PublishClusterStateStats;
|
||||||
import org.elasticsearch.discovery.zen.UnicastHostsProvider.HostsResolver;
|
import org.elasticsearch.discovery.zen.UnicastHostsProvider.HostsResolver;
|
||||||
|
import org.elasticsearch.env.NodeEnvironment;
|
||||||
|
import org.elasticsearch.gateway.MockGatewayMetaState;
|
||||||
import org.elasticsearch.indices.cluster.FakeThreadPoolMasterService;
|
import org.elasticsearch.indices.cluster.FakeThreadPoolMasterService;
|
||||||
import org.elasticsearch.test.ESTestCase;
|
import org.elasticsearch.test.ESTestCase;
|
||||||
import org.elasticsearch.test.disruption.DisruptableMockTransport;
|
import org.elasticsearch.test.disruption.DisruptableMockTransport;
|
||||||
import org.elasticsearch.test.disruption.DisruptableMockTransport.ConnectionStatus;
|
import org.elasticsearch.test.disruption.DisruptableMockTransport.ConnectionStatus;
|
||||||
import org.elasticsearch.transport.TransportService;
|
import org.elasticsearch.transport.TransportService;
|
||||||
import org.hamcrest.Matcher;
|
import org.hamcrest.Matcher;
|
||||||
|
import org.junit.After;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
@ -131,6 +134,16 @@ import static org.hamcrest.Matchers.startsWith;
|
||||||
|
|
||||||
public class CoordinatorTests extends ESTestCase {
|
public class CoordinatorTests extends ESTestCase {
|
||||||
|
|
||||||
|
private final List<NodeEnvironment> nodeEnvironments = new ArrayList<>();
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void closeNodeEnvironmentsAfterEachTest() {
|
||||||
|
for (NodeEnvironment nodeEnvironment : nodeEnvironments) {
|
||||||
|
nodeEnvironment.close();
|
||||||
|
}
|
||||||
|
nodeEnvironments.clear();
|
||||||
|
}
|
||||||
|
|
||||||
@Before
|
@Before
|
||||||
public void resetPortCounterBeforeEachTest() {
|
public void resetPortCounterBeforeEachTest() {
|
||||||
resetPortCounter();
|
resetPortCounter();
|
||||||
|
@ -1102,8 +1115,8 @@ public class CoordinatorTests extends ESTestCase {
|
||||||
private final Set<String> blackholedNodes = new HashSet<>();
|
private final Set<String> blackholedNodes = new HashSet<>();
|
||||||
private final Map<Long, ClusterState> committedStatesByVersion = new HashMap<>();
|
private final Map<Long, ClusterState> committedStatesByVersion = new HashMap<>();
|
||||||
|
|
||||||
private final Function<DiscoveryNode, PersistedState> defaultPersistedStateSupplier = localNode -> new MockPersistedState(0L,
|
private final Function<DiscoveryNode, MockPersistedState> defaultPersistedStateSupplier =
|
||||||
clusterState(0L, 0L, localNode, VotingConfiguration.EMPTY_CONFIG, VotingConfiguration.EMPTY_CONFIG, 0L));
|
localNode -> new MockPersistedState(localNode);
|
||||||
|
|
||||||
Cluster(int initialNodeCount) {
|
Cluster(int initialNodeCount) {
|
||||||
this(initialNodeCount, true);
|
this(initialNodeCount, true);
|
||||||
|
@ -1490,35 +1503,77 @@ public class CoordinatorTests extends ESTestCase {
|
||||||
return getAnyNode();
|
return getAnyNode();
|
||||||
}
|
}
|
||||||
|
|
||||||
class MockPersistedState extends InMemoryPersistedState {
|
class MockPersistedState implements PersistedState {
|
||||||
MockPersistedState(long term, ClusterState acceptedState) {
|
private final PersistedState delegate;
|
||||||
super(term, acceptedState);
|
private final NodeEnvironment nodeEnvironment;
|
||||||
|
|
||||||
|
MockPersistedState(DiscoveryNode localNode) {
|
||||||
|
try {
|
||||||
|
if (rarely()) {
|
||||||
|
nodeEnvironment = newNodeEnvironment();
|
||||||
|
nodeEnvironments.add(nodeEnvironment);
|
||||||
|
delegate = new MockGatewayMetaState(Settings.EMPTY, nodeEnvironment, xContentRegistry(), localNode)
|
||||||
|
.getPersistedState(Settings.EMPTY, null);
|
||||||
|
} else {
|
||||||
|
nodeEnvironment = null;
|
||||||
|
delegate = new InMemoryPersistedState(0L,
|
||||||
|
clusterState(0L, 0L, localNode, VotingConfiguration.EMPTY_CONFIG, VotingConfiguration.EMPTY_CONFIG, 0L));
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new UncheckedIOException("Unable to create MockPersistedState", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
MockPersistedState(DiscoveryNode newLocalNode, MockPersistedState oldState) {
|
||||||
|
try {
|
||||||
|
if (oldState.nodeEnvironment != null) {
|
||||||
|
nodeEnvironment = oldState.nodeEnvironment;
|
||||||
|
delegate = new MockGatewayMetaState(Settings.EMPTY, nodeEnvironment, xContentRegistry(), newLocalNode)
|
||||||
|
.getPersistedState(Settings.EMPTY, null);
|
||||||
|
} else {
|
||||||
|
nodeEnvironment = null;
|
||||||
|
BytesStreamOutput outStream = new BytesStreamOutput();
|
||||||
|
outStream.setVersion(Version.CURRENT);
|
||||||
|
oldState.getLastAcceptedState().writeTo(outStream);
|
||||||
|
StreamInput inStream = new NamedWriteableAwareStreamInput(outStream.bytes().streamInput(),
|
||||||
|
new NamedWriteableRegistry(ClusterModule.getNamedWriteables()));
|
||||||
|
delegate = new InMemoryPersistedState(oldState.getCurrentTerm(), ClusterState.readFrom(inStream,
|
||||||
|
newLocalNode)); // adapts it to new localNode instance
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new UncheckedIOException("Unable to create MockPersistedState", e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void possiblyFail(String description) {
|
private void possiblyFail(String description) {
|
||||||
if (disruptStorage && rarely()) {
|
if (disruptStorage && rarely()) {
|
||||||
// TODO revisit this when we've decided how PersistedState should throw exceptions
|
|
||||||
logger.trace("simulating IO exception [{}]", description);
|
logger.trace("simulating IO exception [{}]", description);
|
||||||
if (randomBoolean()) {
|
// In the real-life IOError might be thrown, for example if state fsync fails.
|
||||||
throw new UncheckedIOException(new IOException("simulated IO exception [" + description + ']'));
|
// This will require node restart and we're not emulating it here.
|
||||||
} else {
|
throw new UncheckedIOException(new IOException("simulated IO exception [" + description + ']'));
|
||||||
throw new CoordinationStateRejectedException("simulated IO exception [" + description + ']');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long getCurrentTerm() {
|
||||||
|
return delegate.getCurrentTerm();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ClusterState getLastAcceptedState() {
|
||||||
|
return delegate.getLastAcceptedState();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void setCurrentTerm(long currentTerm) {
|
public void setCurrentTerm(long currentTerm) {
|
||||||
possiblyFail("before writing term of " + currentTerm);
|
possiblyFail("before writing term of " + currentTerm);
|
||||||
super.setCurrentTerm(currentTerm);
|
delegate.setCurrentTerm(currentTerm);
|
||||||
// TODO possiblyFail() here if that's a failure mode of the storage layer
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void setLastAcceptedState(ClusterState clusterState) {
|
public void setLastAcceptedState(ClusterState clusterState) {
|
||||||
possiblyFail("before writing last-accepted state of term=" + clusterState.term() + ", version=" + clusterState.version());
|
possiblyFail("before writing last-accepted state of term=" + clusterState.term() + ", version=" + clusterState.version());
|
||||||
super.setLastAcceptedState(clusterState);
|
delegate.setLastAcceptedState(clusterState);
|
||||||
// TODO possiblyFail() here if that's a failure mode of the storage layer
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1528,7 +1583,7 @@ public class CoordinatorTests extends ESTestCase {
|
||||||
private final int nodeIndex;
|
private final int nodeIndex;
|
||||||
private Coordinator coordinator;
|
private Coordinator coordinator;
|
||||||
private final DiscoveryNode localNode;
|
private final DiscoveryNode localNode;
|
||||||
private final PersistedState persistedState;
|
private final MockPersistedState persistedState;
|
||||||
private FakeClusterApplier clusterApplier;
|
private FakeClusterApplier clusterApplier;
|
||||||
private AckedFakeThreadPoolMasterService masterService;
|
private AckedFakeThreadPoolMasterService masterService;
|
||||||
private TransportService transportService;
|
private TransportService transportService;
|
||||||
|
@ -1540,7 +1595,7 @@ public class CoordinatorTests extends ESTestCase {
|
||||||
this(nodeIndex, createDiscoveryNode(nodeIndex, masterEligible), defaultPersistedStateSupplier);
|
this(nodeIndex, createDiscoveryNode(nodeIndex, masterEligible), defaultPersistedStateSupplier);
|
||||||
}
|
}
|
||||||
|
|
||||||
ClusterNode(int nodeIndex, DiscoveryNode localNode, Function<DiscoveryNode, PersistedState> persistedStateSupplier) {
|
ClusterNode(int nodeIndex, DiscoveryNode localNode, Function<DiscoveryNode, MockPersistedState> persistedStateSupplier) {
|
||||||
this.nodeIndex = nodeIndex;
|
this.nodeIndex = nodeIndex;
|
||||||
this.localNode = localNode;
|
this.localNode = localNode;
|
||||||
persistedState = persistedStateSupplier.apply(localNode);
|
persistedState = persistedStateSupplier.apply(localNode);
|
||||||
|
@ -1608,19 +1663,7 @@ public class CoordinatorTests extends ESTestCase {
|
||||||
UUIDs.randomBase64UUID(random()), // generated deterministically for repeatable tests
|
UUIDs.randomBase64UUID(random()), // generated deterministically for repeatable tests
|
||||||
address.address().getHostString(), address.getAddress(), address, Collections.emptyMap(),
|
address.address().getHostString(), address.getAddress(), address, Collections.emptyMap(),
|
||||||
localNode.isMasterNode() ? EnumSet.allOf(Role.class) : emptySet(), Version.CURRENT);
|
localNode.isMasterNode() ? EnumSet.allOf(Role.class) : emptySet(), Version.CURRENT);
|
||||||
final PersistedState newPersistedState;
|
return new ClusterNode(nodeIndex, newLocalNode, node -> new MockPersistedState(newLocalNode, persistedState));
|
||||||
try {
|
|
||||||
BytesStreamOutput outStream = new BytesStreamOutput();
|
|
||||||
outStream.setVersion(Version.CURRENT);
|
|
||||||
persistedState.getLastAcceptedState().writeTo(outStream);
|
|
||||||
StreamInput inStream = new NamedWriteableAwareStreamInput(outStream.bytes().streamInput(),
|
|
||||||
new NamedWriteableRegistry(ClusterModule.getNamedWriteables()));
|
|
||||||
newPersistedState = new MockPersistedState(persistedState.getCurrentTerm(),
|
|
||||||
ClusterState.readFrom(inStream, newLocalNode)); // adapts it to new localNode instance
|
|
||||||
} catch (IOException e) {
|
|
||||||
throw new UncheckedIOException(e);
|
|
||||||
}
|
|
||||||
return new ClusterNode(nodeIndex, newLocalNode, node -> newPersistedState);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private PersistedState getPersistedState() {
|
private PersistedState getPersistedState() {
|
||||||
|
|
|
@ -46,6 +46,7 @@ import java.util.stream.Collectors;
|
||||||
import java.util.stream.Stream;
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
import static org.elasticsearch.cluster.metadata.MetaData.CLUSTER_READ_ONLY_BLOCK;
|
import static org.elasticsearch.cluster.metadata.MetaData.CLUSTER_READ_ONLY_BLOCK;
|
||||||
|
import static org.elasticsearch.gateway.ClusterStateUpdaters.addStateNotRecoveredBlock;
|
||||||
import static org.elasticsearch.gateway.ClusterStateUpdaters.closeBadIndices;
|
import static org.elasticsearch.gateway.ClusterStateUpdaters.closeBadIndices;
|
||||||
import static org.elasticsearch.gateway.ClusterStateUpdaters.hideStateIfNotRecovered;
|
import static org.elasticsearch.gateway.ClusterStateUpdaters.hideStateIfNotRecovered;
|
||||||
import static org.elasticsearch.gateway.ClusterStateUpdaters.mixCurrentStateAndRecoveredState;
|
import static org.elasticsearch.gateway.ClusterStateUpdaters.mixCurrentStateAndRecoveredState;
|
||||||
|
@ -182,6 +183,24 @@ public class ClusterStateUpdatersTests extends ESTestCase {
|
||||||
assertFalse(newState.blocks().hasGlobalBlock(STATE_NOT_RECOVERED_BLOCK));
|
assertFalse(newState.blocks().hasGlobalBlock(STATE_NOT_RECOVERED_BLOCK));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void testAddStateNotRecoveredBlock() {
|
||||||
|
final MetaData.Builder metaDataBuilder = MetaData.builder()
|
||||||
|
.persistentSettings(Settings.builder().put("test", "test").build());
|
||||||
|
final IndexMetaData indexMetaData = createIndexMetaData("test", Settings.EMPTY);
|
||||||
|
metaDataBuilder.put(indexMetaData, false);
|
||||||
|
|
||||||
|
final ClusterState initialState = ClusterState
|
||||||
|
.builder(ClusterState.EMPTY_STATE)
|
||||||
|
.metaData(metaDataBuilder)
|
||||||
|
.build();
|
||||||
|
assertFalse(initialState.blocks().hasGlobalBlock(STATE_NOT_RECOVERED_BLOCK));
|
||||||
|
|
||||||
|
final ClusterState newState = addStateNotRecoveredBlock(initialState);
|
||||||
|
|
||||||
|
assertMetaDataEquals(initialState, newState);
|
||||||
|
assertTrue(newState.blocks().hasGlobalBlock(STATE_NOT_RECOVERED_BLOCK));
|
||||||
|
}
|
||||||
|
|
||||||
public void testCloseBadIndices() throws IOException {
|
public void testCloseBadIndices() throws IOException {
|
||||||
final IndicesService indicesService = mock(IndicesService.class);
|
final IndicesService indicesService = mock(IndicesService.class);
|
||||||
final IndexMetaData good = createIndexMetaData("good", Settings.EMPTY);
|
final IndexMetaData good = createIndexMetaData("good", Settings.EMPTY);
|
||||||
|
|
|
@ -27,49 +27,20 @@ import org.elasticsearch.cluster.coordination.CoordinationMetaData.VotingConfigE
|
||||||
import org.elasticsearch.cluster.metadata.IndexMetaData;
|
import org.elasticsearch.cluster.metadata.IndexMetaData;
|
||||||
import org.elasticsearch.cluster.metadata.Manifest;
|
import org.elasticsearch.cluster.metadata.Manifest;
|
||||||
import org.elasticsearch.cluster.metadata.MetaData;
|
import org.elasticsearch.cluster.metadata.MetaData;
|
||||||
import org.elasticsearch.cluster.metadata.MetaDataIndexUpgradeService;
|
|
||||||
import org.elasticsearch.cluster.node.DiscoveryNode;
|
import org.elasticsearch.cluster.node.DiscoveryNode;
|
||||||
import org.elasticsearch.cluster.node.DiscoveryNodes;
|
import org.elasticsearch.cluster.node.DiscoveryNodes;
|
||||||
import org.elasticsearch.cluster.service.ClusterService;
|
|
||||||
import org.elasticsearch.common.settings.Settings;
|
import org.elasticsearch.common.settings.Settings;
|
||||||
import org.elasticsearch.common.util.set.Sets;
|
import org.elasticsearch.common.util.set.Sets;
|
||||||
import org.elasticsearch.env.NodeEnvironment;
|
import org.elasticsearch.env.NodeEnvironment;
|
||||||
import org.elasticsearch.indices.IndicesService;
|
|
||||||
import org.elasticsearch.plugins.MetaDataUpgrader;
|
|
||||||
import org.elasticsearch.test.ESTestCase;
|
import org.elasticsearch.test.ESTestCase;
|
||||||
import org.elasticsearch.transport.TransportService;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
|
|
||||||
import static org.hamcrest.Matchers.equalTo;
|
import static org.hamcrest.Matchers.equalTo;
|
||||||
import static org.hamcrest.Matchers.not;
|
import static org.hamcrest.Matchers.not;
|
||||||
import static org.mockito.Mockito.mock;
|
|
||||||
|
|
||||||
public class GatewayMetaStatePersistedStateTests extends ESTestCase {
|
public class GatewayMetaStatePersistedStateTests extends ESTestCase {
|
||||||
private class GatewayMetaStateUT extends GatewayMetaState {
|
|
||||||
private final DiscoveryNode localNode;
|
|
||||||
|
|
||||||
GatewayMetaStateUT(Settings settings, NodeEnvironment nodeEnvironment, DiscoveryNode localNode) throws IOException {
|
|
||||||
super(settings, nodeEnvironment, new MetaStateService(nodeEnvironment, xContentRegistry()),
|
|
||||||
mock(MetaDataIndexUpgradeService.class), mock(MetaDataUpgrader.class),
|
|
||||||
mock(TransportService.class), mock(ClusterService.class),
|
|
||||||
mock(IndicesService.class));
|
|
||||||
this.localNode = localNode;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void upgradeMetaData(MetaDataIndexUpgradeService metaDataIndexUpgradeService, MetaDataUpgrader metaDataUpgrader) {
|
|
||||||
// MetaData upgrade is tested in GatewayMetaStateTests, we override this method to NOP to make mocking easier
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void applyClusterStateUpdaters() {
|
|
||||||
// Just set localNode here, not to mess with ClusterService and IndicesService mocking
|
|
||||||
previousClusterState = ClusterStateUpdaters.setLocalNode(previousClusterState, localNode);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private NodeEnvironment nodeEnvironment;
|
private NodeEnvironment nodeEnvironment;
|
||||||
private ClusterName clusterName;
|
private ClusterName clusterName;
|
||||||
private Settings settings;
|
private Settings settings;
|
||||||
|
@ -91,13 +62,13 @@ public class GatewayMetaStatePersistedStateTests extends ESTestCase {
|
||||||
super.tearDown();
|
super.tearDown();
|
||||||
}
|
}
|
||||||
|
|
||||||
private GatewayMetaStateUT newGateway() throws IOException {
|
private MockGatewayMetaState newGateway() throws IOException {
|
||||||
GatewayMetaStateUT gateway = new GatewayMetaStateUT(settings, nodeEnvironment, localNode);
|
MockGatewayMetaState gateway = new MockGatewayMetaState(settings, nodeEnvironment, xContentRegistry(), localNode);
|
||||||
gateway.applyClusterStateUpdaters();
|
gateway.applyClusterStateUpdaters();
|
||||||
return gateway;
|
return gateway;
|
||||||
}
|
}
|
||||||
|
|
||||||
private GatewayMetaStateUT maybeNew(GatewayMetaStateUT gateway) throws IOException {
|
private MockGatewayMetaState maybeNew(MockGatewayMetaState gateway) throws IOException {
|
||||||
if (randomBoolean()) {
|
if (randomBoolean()) {
|
||||||
return newGateway();
|
return newGateway();
|
||||||
}
|
}
|
||||||
|
@ -105,7 +76,7 @@ public class GatewayMetaStatePersistedStateTests extends ESTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testInitialState() throws IOException {
|
public void testInitialState() throws IOException {
|
||||||
GatewayMetaStateUT gateway = newGateway();
|
MockGatewayMetaState gateway = newGateway();
|
||||||
ClusterState state = gateway.getLastAcceptedState();
|
ClusterState state = gateway.getLastAcceptedState();
|
||||||
assertThat(state.getClusterName(), equalTo(clusterName));
|
assertThat(state.getClusterName(), equalTo(clusterName));
|
||||||
assertTrue(MetaData.isGlobalStateEquals(state.metaData(), MetaData.EMPTY_META_DATA));
|
assertTrue(MetaData.isGlobalStateEquals(state.metaData(), MetaData.EMPTY_META_DATA));
|
||||||
|
@ -117,7 +88,7 @@ public class GatewayMetaStatePersistedStateTests extends ESTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testSetCurrentTerm() throws IOException {
|
public void testSetCurrentTerm() throws IOException {
|
||||||
GatewayMetaStateUT gateway = newGateway();
|
MockGatewayMetaState gateway = newGateway();
|
||||||
|
|
||||||
for (int i = 0; i < randomIntBetween(1, 5); i++) {
|
for (int i = 0; i < randomIntBetween(1, 5); i++) {
|
||||||
final long currentTerm = randomNonNegativeLong();
|
final long currentTerm = randomNonNegativeLong();
|
||||||
|
@ -171,7 +142,7 @@ public class GatewayMetaStatePersistedStateTests extends ESTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testSetLastAcceptedState() throws IOException {
|
public void testSetLastAcceptedState() throws IOException {
|
||||||
GatewayMetaStateUT gateway = newGateway();
|
MockGatewayMetaState gateway = newGateway();
|
||||||
final long term = randomNonNegativeLong();
|
final long term = randomNonNegativeLong();
|
||||||
|
|
||||||
for (int i = 0; i < randomIntBetween(1, 5); i++) {
|
for (int i = 0; i < randomIntBetween(1, 5); i++) {
|
||||||
|
@ -194,7 +165,7 @@ public class GatewayMetaStatePersistedStateTests extends ESTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testSetLastAcceptedStateTermChanged() throws IOException {
|
public void testSetLastAcceptedStateTermChanged() throws IOException {
|
||||||
GatewayMetaStateUT gateway = newGateway();
|
MockGatewayMetaState gateway = newGateway();
|
||||||
|
|
||||||
final String indexName = randomAlphaOfLength(10);
|
final String indexName = randomAlphaOfLength(10);
|
||||||
final int numberOfShards = randomIntBetween(1, 5);
|
final int numberOfShards = randomIntBetween(1, 5);
|
||||||
|
@ -218,7 +189,7 @@ public class GatewayMetaStatePersistedStateTests extends ESTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testCurrentTermAndTermAreDifferent() throws IOException {
|
public void testCurrentTermAndTermAreDifferent() throws IOException {
|
||||||
GatewayMetaStateUT gateway = newGateway();
|
MockGatewayMetaState gateway = newGateway();
|
||||||
|
|
||||||
long currentTerm = randomNonNegativeLong();
|
long currentTerm = randomNonNegativeLong();
|
||||||
long term = randomValueOtherThan(currentTerm, () -> randomNonNegativeLong());
|
long term = randomValueOtherThan(currentTerm, () -> randomNonNegativeLong());
|
||||||
|
@ -233,7 +204,7 @@ public class GatewayMetaStatePersistedStateTests extends ESTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testMarkAcceptedConfigAsCommitted() throws IOException {
|
public void testMarkAcceptedConfigAsCommitted() throws IOException {
|
||||||
GatewayMetaStateUT gateway = newGateway();
|
MockGatewayMetaState gateway = newGateway();
|
||||||
|
|
||||||
//generate random coordinationMetaData with different lastAcceptedConfiguration and lastCommittedConfiguration
|
//generate random coordinationMetaData with different lastAcceptedConfiguration and lastCommittedConfiguration
|
||||||
CoordinationMetaData coordinationMetaData;
|
CoordinationMetaData coordinationMetaData;
|
||||||
|
|
|
@ -0,0 +1,66 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elasticsearch under one or more contributor
|
||||||
|
* license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright
|
||||||
|
* ownership. Elasticsearch licenses this file to you under
|
||||||
|
* the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
* not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing,
|
||||||
|
* software distributed under the License is distributed on an
|
||||||
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
||||||
|
* KIND, either express or implied. See the License for the
|
||||||
|
* specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.elasticsearch.gateway;
|
||||||
|
|
||||||
|
import org.elasticsearch.cluster.metadata.MetaDataIndexUpgradeService;
|
||||||
|
import org.elasticsearch.cluster.node.DiscoveryNode;
|
||||||
|
import org.elasticsearch.cluster.service.ClusterService;
|
||||||
|
import org.elasticsearch.common.settings.Settings;
|
||||||
|
import org.elasticsearch.common.xcontent.NamedXContentRegistry;
|
||||||
|
import org.elasticsearch.env.NodeEnvironment;
|
||||||
|
import org.elasticsearch.indices.IndicesService;
|
||||||
|
import org.elasticsearch.plugins.MetaDataUpgrader;
|
||||||
|
import org.elasticsearch.transport.TransportService;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link GatewayMetaState} constructor accepts a lot of arguments.
|
||||||
|
* It's not always easy / convenient to construct these dependencies.
|
||||||
|
* This class constructor takes far fewer dependencies and constructs usable {@link GatewayMetaState} with 2 restrictions:
|
||||||
|
* no metadata upgrade will be performed and no cluster state updaters will be run. This is sufficient for most of the tests.
|
||||||
|
* Metadata upgrade is tested in {@link GatewayMetaStateTests} and different {@link ClusterStateUpdaters} in
|
||||||
|
* {@link ClusterStateUpdatersTests}.
|
||||||
|
*/
|
||||||
|
public class MockGatewayMetaState extends GatewayMetaState {
|
||||||
|
private final DiscoveryNode localNode;
|
||||||
|
|
||||||
|
public MockGatewayMetaState(Settings settings, NodeEnvironment nodeEnvironment,
|
||||||
|
NamedXContentRegistry xContentRegistry, DiscoveryNode localNode) throws IOException {
|
||||||
|
super(settings, nodeEnvironment, new MetaStateService(nodeEnvironment, xContentRegistry),
|
||||||
|
mock(MetaDataIndexUpgradeService.class), mock(MetaDataUpgrader.class),
|
||||||
|
mock(TransportService.class), mock(ClusterService.class),
|
||||||
|
mock(IndicesService.class));
|
||||||
|
this.localNode = localNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void upgradeMetaData(MetaDataIndexUpgradeService metaDataIndexUpgradeService, MetaDataUpgrader metaDataUpgrader) {
|
||||||
|
// MetaData upgrade is tested in GatewayMetaStateTests, we override this method to NOP to make mocking easier
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void applyClusterStateUpdaters() {
|
||||||
|
// Just set localNode here, not to mess with ClusterService and IndicesService mocking
|
||||||
|
previousClusterState = ClusterStateUpdaters.setLocalNode(previousClusterState, localNode);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue