HDFS-7898. Change TestAppendSnapshotTruncate to fail-fast. Contributed by Tsz Wo Nicholas Sze.
This commit is contained in:
parent
5578e22ce9
commit
e43882e84a
@ -734,6 +734,9 @@ Release 2.7.0 - UNRELEASED
|
|||||||
HDFS-7411. Change decommission logic to throttle by blocks rather than
|
HDFS-7411. Change decommission logic to throttle by blocks rather than
|
||||||
nodes in each interval. (Andrew Wang via cdouglas)
|
nodes in each interval. (Andrew Wang via cdouglas)
|
||||||
|
|
||||||
|
HDFS-7898. Change TestAppendSnapshotTruncate to fail-fast.
|
||||||
|
(Tsz Wo Nicholas Sze via jing9)
|
||||||
|
|
||||||
OPTIMIZATIONS
|
OPTIMIZATIONS
|
||||||
|
|
||||||
HDFS-7454. Reduce memory footprint for AclEntries in NameNode.
|
HDFS-7454. Reduce memory footprint for AclEntries in NameNode.
|
||||||
|
@ -41,10 +41,6 @@
|
|||||||
import org.apache.hadoop.fs.FileStatus;
|
import org.apache.hadoop.fs.FileStatus;
|
||||||
import org.apache.hadoop.fs.FileUtil;
|
import org.apache.hadoop.fs.FileUtil;
|
||||||
import org.apache.hadoop.fs.Path;
|
import org.apache.hadoop.fs.Path;
|
||||||
import org.apache.hadoop.hdfs.DFSConfigKeys;
|
|
||||||
import org.apache.hadoop.hdfs.DFSUtil;
|
|
||||||
import org.apache.hadoop.hdfs.DistributedFileSystem;
|
|
||||||
import org.apache.hadoop.hdfs.HdfsConfiguration;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.NameNode;
|
import org.apache.hadoop.hdfs.server.namenode.NameNode;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.TestFileTruncate;
|
import org.apache.hadoop.hdfs.server.namenode.TestFileTruncate;
|
||||||
import org.apache.hadoop.test.GenericTestUtils;
|
import org.apache.hadoop.test.GenericTestUtils;
|
||||||
@ -69,6 +65,9 @@ public class TestAppendSnapshotTruncate {
|
|||||||
private static final int BLOCK_SIZE = 1024;
|
private static final int BLOCK_SIZE = 1024;
|
||||||
private static final int DATANODE_NUM = 3;
|
private static final int DATANODE_NUM = 3;
|
||||||
private static final short REPLICATION = 3;
|
private static final short REPLICATION = 3;
|
||||||
|
private static final int FILE_WORKER_NUM = 3;
|
||||||
|
private static final long TEST_TIME_SECOND = 10;
|
||||||
|
private static final long TEST_TIMEOUT_SECOND = TEST_TIME_SECOND + 60;
|
||||||
|
|
||||||
static final int SHORT_HEARTBEAT = 1;
|
static final int SHORT_HEARTBEAT = 1;
|
||||||
static final String[] EMPTY_STRINGS = {};
|
static final String[] EMPTY_STRINGS = {};
|
||||||
@ -106,7 +105,7 @@ public static void tearDown() throws IOException {
|
|||||||
|
|
||||||
|
|
||||||
/** Test randomly mixing append, snapshot and truncate operations. */
|
/** Test randomly mixing append, snapshot and truncate operations. */
|
||||||
@Test
|
@Test(timeout=TEST_TIMEOUT_SECOND*1000)
|
||||||
public void testAST() throws Exception {
|
public void testAST() throws Exception {
|
||||||
final String dirPathString = "/dir";
|
final String dirPathString = "/dir";
|
||||||
final Path dir = new Path(dirPathString);
|
final Path dir = new Path(dirPathString);
|
||||||
@ -121,12 +120,12 @@ public void testAST() throws Exception {
|
|||||||
}
|
}
|
||||||
localDir.mkdirs();
|
localDir.mkdirs();
|
||||||
|
|
||||||
final DirWorker w = new DirWorker(dir, localDir, 3);
|
final DirWorker w = new DirWorker(dir, localDir, FILE_WORKER_NUM);
|
||||||
w.startAllFiles();
|
w.startAllFiles();
|
||||||
w.start();
|
w.start();
|
||||||
Worker.sleep(10L*1000);
|
Worker.sleep(TEST_TIME_SECOND * 1000);
|
||||||
w.stop();
|
w.stop();
|
||||||
w.stoptAllFiles();
|
w.stopAllFiles();
|
||||||
w.checkEverything();
|
w.checkEverything();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -259,7 +258,7 @@ void startAllFiles() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void stoptAllFiles() throws InterruptedException {
|
void stopAllFiles() throws InterruptedException {
|
||||||
for(FileWorker f : files) {
|
for(FileWorker f : files) {
|
||||||
f.stop();
|
f.stop();
|
||||||
}
|
}
|
||||||
@ -269,12 +268,12 @@ void checkEverything() throws IOException {
|
|||||||
LOG.info("checkEverything");
|
LOG.info("checkEverything");
|
||||||
for(FileWorker f : files) {
|
for(FileWorker f : files) {
|
||||||
f.checkFullFile();
|
f.checkFullFile();
|
||||||
Preconditions.checkState(f.state.get() != State.ERROR);
|
f.checkErrorState();
|
||||||
}
|
}
|
||||||
for(String snapshot : snapshotPaths.keySet()) {
|
for(String snapshot : snapshotPaths.keySet()) {
|
||||||
checkSnapshot(snapshot);
|
checkSnapshot(snapshot);
|
||||||
}
|
}
|
||||||
Preconditions.checkState(state.get() != State.ERROR);
|
checkErrorState();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -364,7 +363,7 @@ private boolean truncate(long newLength, StringBuilder b) throws IOException {
|
|||||||
b.append(", newLength=").append(newLength)
|
b.append(", newLength=").append(newLength)
|
||||||
.append(", isReady=").append(isReady);
|
.append(", isReady=").append(isReady);
|
||||||
if (!isReady) {
|
if (!isReady) {
|
||||||
TestFileTruncate.checkBlockRecovery(file, dfs);
|
TestFileTruncate.checkBlockRecovery(file, dfs, 100, 300L);
|
||||||
}
|
}
|
||||||
return isReady;
|
return isReady;
|
||||||
}
|
}
|
||||||
@ -407,6 +406,7 @@ enum State {
|
|||||||
IDLE(false), RUNNING(false), STOPPED(true), ERROR(true);
|
IDLE(false), RUNNING(false), STOPPED(true), ERROR(true);
|
||||||
|
|
||||||
final boolean isTerminated;
|
final boolean isTerminated;
|
||||||
|
|
||||||
State(boolean isTerminated) {
|
State(boolean isTerminated) {
|
||||||
this.isTerminated = isTerminated;
|
this.isTerminated = isTerminated;
|
||||||
}
|
}
|
||||||
@ -417,10 +417,28 @@ enum State {
|
|||||||
final AtomicBoolean isCalling = new AtomicBoolean();
|
final AtomicBoolean isCalling = new AtomicBoolean();
|
||||||
final AtomicReference<Thread> thread = new AtomicReference<Thread>();
|
final AtomicReference<Thread> thread = new AtomicReference<Thread>();
|
||||||
|
|
||||||
|
private Throwable thrown = null;
|
||||||
|
|
||||||
Worker(String name) {
|
Worker(String name) {
|
||||||
this.name = name;
|
this.name = name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
State checkErrorState() {
|
||||||
|
final State s = state.get();
|
||||||
|
if (s == State.ERROR) {
|
||||||
|
throw new IllegalStateException(name + " has " + s, thrown);
|
||||||
|
}
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setErrorState(Throwable t) {
|
||||||
|
checkErrorState();
|
||||||
|
|
||||||
|
LOG.error("Worker " + name + " failed.", t);
|
||||||
|
state.set(State.ERROR);
|
||||||
|
thrown = t;
|
||||||
|
}
|
||||||
|
|
||||||
void start() {
|
void start() {
|
||||||
Preconditions.checkState(state.compareAndSet(State.IDLE, State.RUNNING));
|
Preconditions.checkState(state.compareAndSet(State.IDLE, State.RUNNING));
|
||||||
|
|
||||||
@ -429,14 +447,13 @@ void start() {
|
|||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
final Random r = DFSUtil.getRandom();
|
final Random r = DFSUtil.getRandom();
|
||||||
for(State s; (s = state.get()) == State.RUNNING || s == State.IDLE;) {
|
for(State s; !(s = checkErrorState()).isTerminated;) {
|
||||||
if (s == State.RUNNING) {
|
if (s == State.RUNNING) {
|
||||||
isCalling.set(true);
|
isCalling.set(true);
|
||||||
try {
|
try {
|
||||||
LOG.info(call());
|
LOG.info(call());
|
||||||
} catch (Exception e) {
|
} catch(Throwable t) {
|
||||||
LOG.error("Worker " + name + " failed.", e);
|
setErrorState(t);
|
||||||
state.set(State.ERROR);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
isCalling.set(false);
|
isCalling.set(false);
|
||||||
@ -451,7 +468,11 @@ public void run() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
boolean isPaused() {
|
boolean isPaused() {
|
||||||
return state.get() == State.IDLE && !isCalling.get();
|
final State s = checkErrorState();
|
||||||
|
if (s == State.STOPPED) {
|
||||||
|
throw new IllegalStateException(name + " is " + s);
|
||||||
|
}
|
||||||
|
return s == State.IDLE && !isCalling.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
void pause() {
|
void pause() {
|
||||||
@ -459,9 +480,7 @@ void pause() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void stop() throws InterruptedException {
|
void stop() throws InterruptedException {
|
||||||
if (state.get() == State.ERROR) {
|
checkErrorState();
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
state.set(State.STOPPED);
|
state.set(State.STOPPED);
|
||||||
thread.get().join();
|
thread.get().join();
|
||||||
|
@ -1151,8 +1151,13 @@ static void checkBlockRecovery(Path p) throws IOException {
|
|||||||
|
|
||||||
public static void checkBlockRecovery(Path p, DistributedFileSystem dfs)
|
public static void checkBlockRecovery(Path p, DistributedFileSystem dfs)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
|
checkBlockRecovery(p, dfs, SUCCESS_ATTEMPTS, SLEEP);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void checkBlockRecovery(Path p, DistributedFileSystem dfs,
|
||||||
|
int attempts, long sleepMs) throws IOException {
|
||||||
boolean success = false;
|
boolean success = false;
|
||||||
for(int i = 0; i < SUCCESS_ATTEMPTS; i++) {
|
for(int i = 0; i < attempts; i++) {
|
||||||
LocatedBlocks blocks = getLocatedBlocks(p, dfs);
|
LocatedBlocks blocks = getLocatedBlocks(p, dfs);
|
||||||
boolean noLastBlock = blocks.getLastLocatedBlock() == null;
|
boolean noLastBlock = blocks.getLastLocatedBlock() == null;
|
||||||
if(!blocks.isUnderConstruction() &&
|
if(!blocks.isUnderConstruction() &&
|
||||||
@ -1160,9 +1165,9 @@ public static void checkBlockRecovery(Path p, DistributedFileSystem dfs)
|
|||||||
success = true;
|
success = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
try { Thread.sleep(SLEEP); } catch (InterruptedException ignored) {}
|
try { Thread.sleep(sleepMs); } catch (InterruptedException ignored) {}
|
||||||
}
|
}
|
||||||
assertThat("inode should complete in ~" + SLEEP * SUCCESS_ATTEMPTS + " ms.",
|
assertThat("inode should complete in ~" + sleepMs * attempts + " ms.",
|
||||||
success, is(true));
|
success, is(true));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user