Close the file in case of an IOException in AbstractMemoryHttpData.renameTo. (#10163)

Motivation:

An `IOException` may be thrown from `FileChannel.write` or `FileChannel.force`, and cause the fd leak.

Modification:

Close the file in a finally block.

Result:

Avoid fd leak.
This commit is contained in:
feijermu 2020-04-06 16:12:46 +08:00 committed by Norman Maurer
parent 7402a3a55a
commit 3808777c32
2 changed files with 58 additions and 15 deletions

View File

@ -235,9 +235,11 @@ public abstract class AbstractMemoryHttpData extends AbstractHttpData {
return true; return true;
} }
int length = byteBuf.readableBytes(); int length = byteBuf.readableBytes();
RandomAccessFile accessFile = new RandomAccessFile(dest, "rw");
FileChannel fileChannel = accessFile.getChannel();
int written = 0; int written = 0;
RandomAccessFile accessFile = new RandomAccessFile(dest, "rw");
try {
FileChannel fileChannel = accessFile.getChannel();
try {
if (byteBuf.nioBufferCount() == 1) { if (byteBuf.nioBufferCount() == 1) {
ByteBuffer byteBuffer = byteBuf.nioBuffer(); ByteBuffer byteBuffer = byteBuf.nioBuffer();
while (written < length) { while (written < length) {
@ -249,10 +251,13 @@ public abstract class AbstractMemoryHttpData extends AbstractHttpData {
written += fileChannel.write(byteBuffers); written += fileChannel.write(byteBuffers);
} }
} }
fileChannel.force(false); fileChannel.force(false);
} finally {
fileChannel.close(); fileChannel.close();
}
} finally {
accessFile.close(); accessFile.close();
}
return written == length; return written == length;
} }

View File

@ -24,6 +24,7 @@ import org.junit.Test;
import java.io.ByteArrayInputStream; import java.io.ByteArrayInputStream;
import java.io.File; import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream; import java.io.FileOutputStream;
import java.nio.charset.Charset; import java.nio.charset.Charset;
import java.security.SecureRandom; import java.security.SecureRandom;
@ -64,6 +65,43 @@ public class AbstractMemoryHttpDataTest {
test.delete(); test.delete();
} }
} }
@Test
public void testRenameTo() throws Exception {
TestHttpData test = new TestHttpData("test", UTF_8, 0);
try {
File tmpFile = File.createTempFile(UUID.randomUUID().toString(), ".tmp");
tmpFile.deleteOnExit();
final int totalByteCount = 4096;
byte[] bytes = new byte[totalByteCount];
ThreadLocalRandom.current().nextBytes(bytes);
ByteBuf content = Unpooled.wrappedBuffer(bytes);
test.setContent(content);
boolean succ = test.renameTo(tmpFile);
assertTrue(succ);
FileInputStream fis = new FileInputStream(tmpFile);
try {
byte[] buf = new byte[totalByteCount];
int count = 0;
int offset = 0;
int size = totalByteCount;
while ((count = fis.read(buf, offset, size)) > 0) {
offset += count;
size -= count;
if (offset >= totalByteCount || size <= 0) {
break;
}
}
assertArrayEquals(bytes, buf);
assertEquals(0, fis.available());
} finally {
fis.close();
}
} finally {
//release the ByteBuf in AbstractMemoryHttpData
test.delete();
}
}
/** /**
* Provide content into HTTP data with input stream. * Provide content into HTTP data with input stream.
* *