-
Notifications
You must be signed in to change notification settings - Fork 25k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[TEST] More MetadataStateFormat tests #78577
Merged
grcevski
merged 17 commits into
elastic:master
from
grcevski:tests/metadata_state_format
Oct 7, 2021
Merged
Changes from 15 commits
Commits
Show all changes
17 commits
Select commit
Hold shift + click to select a range
94b4b51
[TEST] More MetadataStateFormat tests
b1799df
Fix style.
6027cea
Merge branch 'master' into tests/metadata_state_format
6da40e2
Improve tests, add delete tests.
138d929
Merge branch 'master' into tests/metadata_state_format
1a1e801
Update test based on expected behaviour.
877c998
Merge branch 'master' into tests/metadata_state_format
0122f8d
Merge branch 'master' into tests/metadata_state_format
84ccc3b
Implement PR suggestions.
1a8e8b1
Update server/src/test/java/org/elasticsearch/gateway/MetadataStateFo…
grcevski a3e1bb7
Update server/src/test/java/org/elasticsearch/gateway/MetadataStateFo…
grcevski e1b415e
Merge branch 'tests/metadata_state_format' of github.com:grcevski/ela…
b3eeb4e
More PR suggestions implementation.
e1a2714
Merge branch 'master' into tests/metadata_state_format
9467926
Fix import.
2b37ee5
Update server/src/test/java/org/elasticsearch/gateway/MetadataStateFo…
grcevski 0b4b055
Fix line width.
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -16,6 +16,7 @@ | |
import org.apache.lucene.store.MockDirectoryWrapper; | ||
import org.apache.lucene.store.NIOFSDirectory; | ||
import org.apache.lucene.util.LuceneTestCase; | ||
import org.elasticsearch.ElasticsearchException; | ||
import org.elasticsearch.cluster.ClusterModule; | ||
import org.elasticsearch.cluster.metadata.Metadata; | ||
import org.elasticsearch.common.xcontent.NamedXContentRegistry; | ||
|
@@ -34,6 +35,7 @@ | |
import java.nio.file.Path; | ||
import java.nio.file.StandardOpenOption; | ||
import java.util.HashSet; | ||
import java.util.List; | ||
import java.util.Set; | ||
import java.util.stream.StreamSupport; | ||
|
||
|
@@ -344,6 +346,124 @@ public void testFailRandomlyAndReadAnyState() throws IOException { | |
writeAndReadStateSuccessfully(format, paths); | ||
} | ||
|
||
public void testInconsistentMultiPathState() throws IOException { | ||
Path paths[] = new Path[3]; | ||
for (int i = 0; i < paths.length; i++) { | ||
paths[i] = createTempDir(); | ||
} | ||
Format format = new Format("foo-"); | ||
|
||
DummyState state = new DummyState(randomRealisticUnicodeOfCodepointLengthBetween(1, 100), randomInt(), randomLong(), | ||
randomDouble(), randomBoolean()); | ||
// Call write without clean-up to simulate multi-write transaction. | ||
long genId = format.write(state, paths); | ||
assertEquals(state, format.loadLatestState(logger, NamedXContentRegistry.EMPTY, paths)); | ||
ensureOnlyOneStateFile(paths); | ||
|
||
for (Path path : paths) { | ||
assertEquals(genId, format.findMaxGenerationId("foo-", path)); | ||
} | ||
assertEquals(0, format.findStateFilesByGeneration(-1, paths).size()); | ||
assertEquals(paths.length, format.findStateFilesByGeneration(genId, paths).size()); | ||
|
||
Path badPath = paths[paths.length-1]; | ||
|
||
format.failOnPaths(badPath.resolve(MetadataStateFormat.STATE_DIR_NAME)); | ||
format.failOnMethods(Format.FAIL_RENAME_TMP_FILE); | ||
|
||
DummyState newState = new DummyState(randomRealisticUnicodeOfCodepointLengthBetween(1, 100), randomInt(), randomLong(), | ||
randomDouble(), randomBoolean()); | ||
|
||
expectThrows(WriteStateException.class, () -> format.write(newState, paths)); | ||
long firstPathId = format.findMaxGenerationId("foo-", paths[0]); | ||
assertEquals(firstPathId, format.findMaxGenerationId("foo-", paths[1])); | ||
assertEquals(genId, format.findMaxGenerationId("foo-", badPath)); | ||
assertEquals(genId, firstPathId-1); | ||
|
||
// Since at least one path has the latest generation, we should find the latest | ||
// generation when we supply all paths. | ||
long allPathsId = format.findMaxGenerationId("foo-", paths); | ||
assertEquals(firstPathId, allPathsId); | ||
|
||
// Assert that we can find the new state since one path successfully wrote it. | ||
assertEquals(newState, format.loadLatestState(logger, NamedXContentRegistry.EMPTY, paths)); | ||
} | ||
|
||
public void testDeleteMetaState() throws IOException { | ||
Path paths[] = new Path[3]; | ||
for (int i = 0; i < paths.length; i++) { | ||
paths[i] = createTempDir(); | ||
} | ||
Format format = new Format("foo-"); | ||
|
||
DummyState state = new DummyState(randomRealisticUnicodeOfCodepointLengthBetween(1, 100), randomInt(), randomLong(), | ||
randomDouble(), randomBoolean()); | ||
long genId = format.write(state, paths); | ||
assertEquals(state, format.loadLatestState(logger, NamedXContentRegistry.EMPTY, paths)); | ||
ensureOnlyOneStateFile(paths); | ||
|
||
for (Path path : paths) { | ||
assertEquals(genId, format.findMaxGenerationId("foo-", path)); | ||
} | ||
assertEquals(0, format.findStateFilesByGeneration(-1, paths).size()); | ||
assertEquals(paths.length, format.findStateFilesByGeneration(genId, paths).size()); | ||
|
||
Format.deleteMetaState(paths); | ||
|
||
assertEquals(0, format.findStateFilesByGeneration(genId, paths).size()); | ||
|
||
for (Path path : paths) { | ||
assertEquals(false, Files.exists(path.resolve(MetadataStateFormat.STATE_DIR_NAME))); | ||
} | ||
|
||
// We shouldn't find any state or state generations anymore | ||
assertEquals(-1, format.findMaxGenerationId("foo-", paths)); | ||
assertNull(format.loadLatestState(logger, NamedXContentRegistry.EMPTY, paths)); | ||
} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can we also verify that |
||
|
||
public void testCleanupOldFilesWithErrorPath() throws IOException { | ||
Path paths[] = new Path[3]; | ||
for (int i = 0; i < paths.length; i++) { | ||
paths[i] = createTempDir(); | ||
} | ||
Format format = new Format("foo-"); | ||
|
||
DummyState state = new DummyState(randomRealisticUnicodeOfCodepointLengthBetween(1, 100), randomInt(), randomLong(), | ||
randomDouble(), randomBoolean()); | ||
long genId = format.write(state, paths); | ||
assertEquals(state, format.loadLatestState(logger, NamedXContentRegistry.EMPTY, paths)); | ||
ensureOnlyOneStateFile(paths); | ||
|
||
for (Path path : paths) { | ||
assertEquals(genId, format.findMaxGenerationId("foo-", path)); | ||
} | ||
assertEquals(0, format.findStateFilesByGeneration(-1, paths).size()); | ||
assertEquals(paths.length, format.findStateFilesByGeneration(genId, paths).size()); | ||
|
||
List<Path> stateFiles = format.findStateFilesByGeneration(genId, paths); | ||
|
||
final int badDirIndex = 1; | ||
|
||
format.failOnPaths(paths[badDirIndex].resolve(MetadataStateFormat.STATE_DIR_NAME)); | ||
format.failOnMethods(Format.FAIL_DELETE_TMP_FILE); | ||
|
||
// Ensure clean-up old files doesn't fail with one bad dir. We pretend we want to | ||
// keep a newer generation that doesn't exist (genId + 1). | ||
format.cleanupOldFiles(genId + 1, paths); | ||
|
||
// We simulated failure on deleting one stale state file, there should be one that's remaining from the old state. | ||
// We'll corrupt this remaining file and check to see if loading the state throws an exception. | ||
// All other state files, including the first directory uncorrupted state files should be cleaned up. | ||
corruptFile(stateFiles.get(badDirIndex), logger); | ||
|
||
try { | ||
format.loadLatestStateWithGeneration(logger, xContentRegistry(), paths); | ||
fail("Reading corrupted state should fail"); | ||
} catch (ElasticsearchException esException) { | ||
assertThat(esException.getMessage(), is("java.io.IOException: failed to read " + stateFiles.get(badDirIndex))); | ||
} | ||
grcevski marked this conversation as resolved.
Show resolved
Hide resolved
|
||
} | ||
|
||
private static class Format extends MetadataStateFormat<DummyState> { | ||
private enum FailureMode { | ||
NO_FAILURES, | ||
|
@@ -353,6 +473,7 @@ private enum FailureMode { | |
|
||
private FailureMode failureMode; | ||
private String[] failureMethods; | ||
private Path[] failurePaths; | ||
|
||
static final String FAIL_CREATE_OUTPUT_FILE = "createOutput"; | ||
static final String FAIL_WRITE_TO_OUTPUT_FILE = "writeBytes"; | ||
|
@@ -361,6 +482,7 @@ private enum FailureMode { | |
static final String FAIL_FSYNC_STATE_DIRECTORY = "syncMetaData"; | ||
static final String FAIL_DELETE_TMP_FILE = "deleteFile"; | ||
static final String FAIL_OPEN_STATE_FILE_WHEN_COPYING = "openInput"; | ||
static final String FAIL_LIST_ALL = "listAll"; | ||
|
||
/** | ||
* Constructs a MetadataStateFormat object for storing/retrieving DummyState. | ||
|
@@ -392,21 +514,37 @@ public void failOnMethods(String... failureMethods) { | |
this.failureMethods = failureMethods; | ||
} | ||
|
||
public void failOnPaths(Path... paths) { | ||
this.failurePaths = paths; | ||
} | ||
|
||
public void failRandomly() { | ||
this.failureMode = FailureMode.FAIL_RANDOMLY; | ||
} | ||
|
||
private void throwDirectoryExceptionCheckPaths(Path dir) throws MockDirectoryWrapper.FakeIOException { | ||
if (failurePaths != null) { | ||
for (Path p : failurePaths) { | ||
if (p.equals(dir)) { | ||
throw new MockDirectoryWrapper.FakeIOException(); | ||
} | ||
} | ||
} else { | ||
throw new MockDirectoryWrapper.FakeIOException(); | ||
} | ||
} | ||
|
||
@Override | ||
protected Directory newDirectory(Path dir) { | ||
MockDirectoryWrapper mock = newMockFSDirectory(dir); | ||
if (failureMode == FailureMode.FAIL_ON_METHOD) { | ||
final String failMethod = randomFrom(failureMethods); | ||
MockDirectoryWrapper.Failure fail = new MockDirectoryWrapper.Failure() { | ||
@Override | ||
public void eval(MockDirectoryWrapper dir) throws IOException { | ||
public void eval(MockDirectoryWrapper directory) throws IOException { | ||
for (StackTraceElement e : Thread.currentThread().getStackTrace()) { | ||
if (failMethod.equals(e.getMethodName())) { | ||
throw new MockDirectoryWrapper.FakeIOException(); | ||
throwDirectoryExceptionCheckPaths(dir); | ||
} | ||
} | ||
} | ||
|
@@ -415,9 +553,9 @@ public void eval(MockDirectoryWrapper dir) throws IOException { | |
} else if (failureMode == FailureMode.FAIL_RANDOMLY) { | ||
MockDirectoryWrapper.Failure fail = new MockDirectoryWrapper.Failure() { | ||
@Override | ||
public void eval(MockDirectoryWrapper dir) throws IOException { | ||
public void eval(MockDirectoryWrapper directory) throws IOException { | ||
if (randomIntBetween(0, 20) == 0) { | ||
throw new MockDirectoryWrapper.FakeIOException(); | ||
throwDirectoryExceptionCheckPaths(dir); | ||
} | ||
} | ||
}; | ||
|
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we can also verify that we can find max genId and read state when supplying all paths (and get the new state out)?