8313739: ZipOutputStream.close() should always close the wrapped stream
Reviewed-by: jpai, lancea
This commit is contained in:
parent
adc3604027
commit
f613e13397
src/java.base/share/classes/java/util/zip
test/jdk/java/util/zip/ZipOutputStream
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 1996, 2022, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 1996, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||
*
|
||||
* This code is free software; you can redistribute it and/or modify it
|
||||
@ -240,14 +240,30 @@ public class DeflaterOutputStream extends FilterOutputStream {
|
||||
*/
|
||||
public void close() throws IOException {
|
||||
if (!closed) {
|
||||
closed = true;
|
||||
IOException finishException = null;
|
||||
try {
|
||||
finish();
|
||||
} catch (IOException ioe){
|
||||
finishException = ioe;
|
||||
throw ioe;
|
||||
} finally {
|
||||
if (usesDefaultDeflater)
|
||||
if (usesDefaultDeflater) {
|
||||
def.end();
|
||||
}
|
||||
if (finishException == null) {
|
||||
out.close();
|
||||
} else {
|
||||
try {
|
||||
out.close();
|
||||
} catch (IOException ioe) {
|
||||
if (finishException != ioe) {
|
||||
ioe.addSuppressed(finishException);
|
||||
}
|
||||
throw ioe;
|
||||
}
|
||||
}
|
||||
}
|
||||
out.close();
|
||||
closed = true;
|
||||
}
|
||||
}
|
||||
|
||||
|
261
test/jdk/java/util/zip/ZipOutputStream/CloseWrappedStream.java
Normal file
261
test/jdk/java/util/zip/ZipOutputStream/CloseWrappedStream.java
Normal file
@ -0,0 +1,261 @@
|
||||
/*
|
||||
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
|
||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||
*
|
||||
* This code is free software; you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License version 2 only, as
|
||||
* published by the Free Software Foundation.
|
||||
*
|
||||
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||
* version 2 for more details (a copy is included in the LICENSE file that
|
||||
* accompanied this code).
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License version
|
||||
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||
*
|
||||
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||
* or visit www.oracle.com if you need additional information or have any
|
||||
* questions.
|
||||
*/
|
||||
|
||||
/* @test
|
||||
@bug 8313739
|
||||
@summary Verify that ZipOutputStream closes the wrapped stream even after failed writes
|
||||
@run junit CloseWrappedStream
|
||||
*/
|
||||
|
||||
import org.junit.jupiter.api.Test;
|
||||
|
||||
import java.io.*;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.util.concurrent.atomic.AtomicReference;
|
||||
import java.util.zip.ZipEntry;
|
||||
import java.util.zip.ZipOutputStream;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.*;
|
||||
|
||||
public class CloseWrappedStream {
|
||||
|
||||
/**
|
||||
* Verify that closing a ZipOutputStream closes the wrapped output stream,
|
||||
* also when the wrapped stream throws while remaining data is flushed
|
||||
*/
|
||||
@Test
|
||||
public void exceptionDuringFinish() {
|
||||
// A wrapped stream which should be closed even after a write failure
|
||||
WrappedOutputStream wrappedStream = new WrappedOutputStream();
|
||||
|
||||
IOException exception = assertThrows(IOException.class, () -> {
|
||||
try (ZipOutputStream zo = new ZipOutputStream(wrappedStream)) {
|
||||
zo.putNextEntry(new ZipEntry("file.txt"));
|
||||
zo.write("hello".getBytes(StandardCharsets.UTF_8));
|
||||
// Make finish() throw IOException
|
||||
wrappedStream.failOnWrite = true;
|
||||
} // Close throws when deflated data is flushed to wrapped stream
|
||||
});
|
||||
|
||||
// Check that we failed with the expected message
|
||||
assertEquals(WrappedOutputStream.WRITE_MSG, exception.getMessage());
|
||||
|
||||
// Verify that the wrapped stream was closed once
|
||||
assertEquals(1, wrappedStream.timesClosed,
|
||||
"Expected wrapped output stream to be closed once");
|
||||
}
|
||||
|
||||
/**
|
||||
* Sanity check that the wrapped stream is closed also for the
|
||||
* normal case where the wrapped stream does not throw
|
||||
*
|
||||
* @throws IOException if an unexpected IOException occurs
|
||||
*/
|
||||
@Test
|
||||
public void noExceptions() throws IOException {
|
||||
|
||||
WrappedOutputStream wrappedStream = new WrappedOutputStream();
|
||||
|
||||
try (ZipOutputStream zo = new ZipOutputStream(wrappedStream)) {
|
||||
zo.putNextEntry(new ZipEntry("file.txt"));
|
||||
zo.write("hello".getBytes(StandardCharsets.UTF_8));
|
||||
}
|
||||
|
||||
// Verify that the wrapped stream was closed once
|
||||
assertEquals(1, wrappedStream.timesClosed,
|
||||
"Expected wrapped output stream to be closed once");
|
||||
}
|
||||
|
||||
/**
|
||||
* Check that an exception thrown while closing the wrapped
|
||||
* stream is propagated to the caller without any suppressed exceptions
|
||||
*
|
||||
* @throws IOException if an unexpected IOException occurs
|
||||
*/
|
||||
@Test
|
||||
public void exceptionDuringClose() throws IOException {
|
||||
|
||||
WrappedOutputStream wrappedStream = new WrappedOutputStream();
|
||||
|
||||
IOException exception = assertThrows(IOException.class, () -> {
|
||||
try (ZipOutputStream zo = new ZipOutputStream(wrappedStream)) {
|
||||
zo.putNextEntry(new ZipEntry("file.txt"));
|
||||
zo.write("hello".getBytes(StandardCharsets.UTF_8));
|
||||
wrappedStream.failOnClose = true;
|
||||
}
|
||||
});
|
||||
|
||||
// Check that we failed with the expected message
|
||||
assertEquals(WrappedOutputStream.CLOSE_MSG, exception.getMessage());
|
||||
|
||||
// There should be no suppressed exceptions
|
||||
assertEquals(0, exception.getSuppressed().length);
|
||||
|
||||
// Verify that the wrapped stream was closed once
|
||||
assertEquals(1, wrappedStream.timesClosed,
|
||||
"Expected wrapped output stream to be closed once");
|
||||
}
|
||||
|
||||
/**
|
||||
* Check that if an exception is thrown while closing the wrapped stream,
|
||||
* then later close attempts will not close the wrapped stream again
|
||||
*
|
||||
* @throws IOException if an unexpected IOException occurs
|
||||
*/
|
||||
@Test
|
||||
public void doubleCloseShouldCloseWrappedStreamOnce() throws IOException {
|
||||
|
||||
WrappedOutputStream wrappedStream = new WrappedOutputStream();
|
||||
|
||||
// Keep a reference so we can double-close
|
||||
AtomicReference<ZipOutputStream> ref = new AtomicReference<>();
|
||||
|
||||
IOException exception = assertThrows(IOException.class, () -> {
|
||||
try (ZipOutputStream zo = new ZipOutputStream(wrappedStream)) {
|
||||
ref.set(zo);
|
||||
zo.putNextEntry(new ZipEntry("file.txt"));
|
||||
zo.write("hello".getBytes(StandardCharsets.UTF_8));
|
||||
wrappedStream.failOnClose = true;
|
||||
}
|
||||
});
|
||||
|
||||
// Double-closing the ZipOutputStream should have no effect
|
||||
ref.get().close();
|
||||
|
||||
// Check that we failed with the expected message
|
||||
assertEquals(WrappedOutputStream.CLOSE_MSG, exception.getMessage());
|
||||
|
||||
// There should be no suppressed exceptions
|
||||
assertEquals(0, exception.getSuppressed().length);
|
||||
|
||||
// Verify that the wrapped stream was closed once
|
||||
assertEquals(1, wrappedStream.timesClosed,
|
||||
"Expected wrapped output stream to be closed once");
|
||||
}
|
||||
|
||||
/**
|
||||
* Check that when the wrapped stream throws while calling finish
|
||||
* AND while being closed, then the propagated exception is the one
|
||||
* from the close operation, with the exception thrown during finish
|
||||
* added as a suppressed exception.
|
||||
*
|
||||
* @throws IOException if an unexpected IOException occurs
|
||||
*/
|
||||
@Test
|
||||
public void exceptionDuringFinishAndClose() throws IOException {
|
||||
|
||||
WrappedOutputStream wrappedStream = new WrappedOutputStream();
|
||||
|
||||
IOException exception = assertThrows(IOException.class, () -> {
|
||||
try (ZipOutputStream zo = new ZipOutputStream(wrappedStream)) {
|
||||
|
||||
zo.putNextEntry(new ZipEntry("file.txt"));
|
||||
zo.write("hello".getBytes(StandardCharsets.UTF_8));
|
||||
// Will cause wrapped stream to throw during finish()
|
||||
wrappedStream.failOnWrite = true;
|
||||
// Will cause wrapped stream to throw during close()
|
||||
wrappedStream.failOnClose = true;
|
||||
}
|
||||
});
|
||||
|
||||
// We expect the top-level exception to be from the close operation
|
||||
assertEquals(WrappedOutputStream.CLOSE_MSG, exception.getMessage());
|
||||
|
||||
// The exception from the finish operation should be suppressed
|
||||
assertEquals(1, exception.getSuppressed().length, "Expected suppressed exception");
|
||||
assertEquals(WrappedOutputStream.WRITE_MSG, exception.getSuppressed()[0].getMessage());
|
||||
|
||||
// Verify that the wrapped stream was closed once
|
||||
assertEquals(1, wrappedStream.timesClosed,
|
||||
"Expected wrapped output stream to be closed once");
|
||||
}
|
||||
|
||||
/**
|
||||
* Check that when the wrapped stream throws the same IOException
|
||||
* (identical instance) for write and close operations, then no attempt
|
||||
* is made to add the exception instance to itself as a suppressed exception.
|
||||
*
|
||||
* @throws IOException if an unexpected IOException occurs
|
||||
*/
|
||||
@Test
|
||||
public void sameExceptionDuringFinishAndClose() throws IOException {
|
||||
|
||||
WrappedOutputStream wrappedStream = new WrappedOutputStream();
|
||||
|
||||
IOException exception = assertThrows(IOException.class, () -> {
|
||||
try (ZipOutputStream zo = new ZipOutputStream(wrappedStream)) {
|
||||
zo.putNextEntry(new ZipEntry("file.txt"));
|
||||
zo.write("hello".getBytes(StandardCharsets.UTF_8));
|
||||
// Make operations fail with the same exception instance
|
||||
wrappedStream.failException = new IOException("same fail");
|
||||
wrappedStream.failOnWrite = true;
|
||||
wrappedStream.failOnClose = true;
|
||||
}
|
||||
});
|
||||
// We expect the top-level exception to be from the close operation
|
||||
assertSame(wrappedStream.failException, exception);
|
||||
|
||||
// The exception should not suppress itself
|
||||
assertEquals(0, exception.getSuppressed().length, "Expected no suppressed exception");
|
||||
|
||||
// Verify that the wrapped stream was closed once
|
||||
assertEquals(1, wrappedStream.timesClosed,
|
||||
"Expected wrapped output stream to be closed once");
|
||||
}
|
||||
|
||||
/**
|
||||
* Output stream which conditionally throws IOException on writes
|
||||
* and tracks its close status.
|
||||
*/
|
||||
static class WrappedOutputStream extends FilterOutputStream {
|
||||
static final String WRITE_MSG = "fail during write";
|
||||
static final String CLOSE_MSG = "fail during close";
|
||||
boolean failOnWrite = false;
|
||||
boolean failOnClose = false;
|
||||
IOException failException = null;
|
||||
int timesClosed = 0;
|
||||
|
||||
public WrappedOutputStream() {
|
||||
super(new ByteArrayOutputStream());
|
||||
}
|
||||
|
||||
@Override
|
||||
public synchronized void write(byte[] b, int off, int len) throws IOException{
|
||||
if (failOnWrite) {
|
||||
throw failException != null ? failException : new IOException(WRITE_MSG);
|
||||
} else {
|
||||
super.write(b, off, len);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void close() throws IOException {
|
||||
timesClosed++;
|
||||
if (failOnClose) {
|
||||
throw failException != null ? failException : new IOException(CLOSE_MSG);
|
||||
} else {
|
||||
super.close();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user