HDFS-9812. Streamer threads leak if failure happens when closing DFSOutputStream. Contributed by Lin Yiqun.

This commit is contained in:
Akira Ajisaka 2016-03-08 10:43:17 +09:00
parent 391da36d93
commit 352d299cf8
2 changed files with 11 additions and 6 deletions

View File

@ -770,14 +770,19 @@ protected synchronized void closeImpl() throws IOException {
flushInternal(); // flush all data to Datanodes
// get last block before destroying the streamer
ExtendedBlock lastBlock = getStreamer().getBlock();
closeThreads(false);
try (TraceScope ignored =
dfsClient.getTracer().newScope("completeFile")) {
completeFile(lastBlock);
}
} catch (ClosedChannelException ignored) {
} finally {
setClosed();
// Failures may happen when flushing data.
// Streamers may keep waiting for the new block information.
// Thus need to force closing these threads.
// Don't need to call setClosed() because closeThreads(true)
// calls setClosed() in the finally block.
closeThreads(true);
}
}

View File

@ -507,7 +507,7 @@ private void initDataStreaming() {
}
protected void endBlock() {
LOG.debug("Closing old block " + block);
LOG.debug("Closing old block {}", block);
this.setName("DataStreamer for file " + src);
closeResponder();
closeStream();
@ -591,7 +591,7 @@ public void run() {
LOG.debug("stage=" + stage + ", " + this);
}
if (stage == BlockConstructionStage.PIPELINE_SETUP_CREATE) {
LOG.debug("Allocating new block: " + this);
LOG.debug("Allocating new block: {}", this);
setPipeline(nextBlockOutputStream());
initDataStreaming();
} else if (stage == BlockConstructionStage.PIPELINE_SETUP_APPEND) {
@ -644,7 +644,7 @@ public void run() {
}
}
LOG.debug(this + " sending " + one);
LOG.debug("{} sending {}", this, one);
// write out data to remote datanode
try (TraceScope ignored = dfsClient.getTracer().
@ -1766,7 +1766,7 @@ void queuePacket(DFSPacket packet) {
packet.addTraceParent(Tracer.getCurrentSpanId());
dataQueue.addLast(packet);
lastQueuedSeqno = packet.getSeqno();
LOG.debug("Queued " + packet + ", " + this);
LOG.debug("Queued {}, {}", packet, this);
dataQueue.notifyAll();
}
}