Skip to content

Commit 1d2019f

Browse files
committed
The Consumer to IOUtils.close(Closeable, IOConsumer) now accepts wrapped
Exception, not just IOException
1 parent 3b77766 commit 1d2019f

File tree

3 files changed

+36
-15
lines changed

3 files changed

+36
-15
lines changed

src/changes/changes.xml

+1
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,7 @@ The <action> type attribute can be add,update,fix,remove.
5656
<action dev="ggregory" type="fix" issue="IO-868" due-to="Julian Reschke, Gary Gregory">Improve Javadoc for a BoundedInputStream builder() throwing IOException.</action>
5757
<action dev="ggregory" type="fix" issue="IO-868" due-to="Julian Reschke, Gary Gregory">Improve Javadoc for all implementations of AbstractOriginSupplier#get().</action>
5858
<action dev="ggregory" type="fix" due-to="Gary Gregory">The Consumer to IOUtils.closeQuietly(Closeable, Consumer) now accepts Exception, not just IOException.</action>
59+
<action dev="ggregory" type="fix" due-to="Gary Gregory">The Consumer to IOUtils.close(Closeable, IOConsumer) now accepts wrapped Exception, not just IOException.</action>
5960
<!-- ADD -->
6061
<action dev="ggregory" type="add" issue="IO-860" due-to="Nico Strecker, Gary Gregory">Add ThrottledInputStream.Builder.setMaxBytes(long, ChronoUnit).</action>
6162
<action dev="ggregory" type="add" due-to="Gary Gregory">Add IOIterable.</action>

src/main/java/org/apache/commons/io/IOUtils.java

+4
Original file line numberDiff line numberDiff line change
@@ -450,6 +450,10 @@ public static void close(final Closeable closeable, final IOConsumer<IOException
450450
if (consumer != null) {
451451
consumer.accept(e);
452452
}
453+
} catch (final Exception e) {
454+
if (consumer != null) {
455+
consumer.accept(new IOException(e));
456+
}
453457
}
454458
}
455459
}

src/test/java/org/apache/commons/io/IOUtilsTest.java

+31-15
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,7 @@
8181
import org.apache.commons.io.test.TestUtils;
8282
import org.apache.commons.io.test.ThrowOnCloseReader;
8383
import org.apache.commons.lang3.StringUtils;
84+
import org.apache.commons.lang3.exception.ExceptionUtils;
8485
import org.junit.jupiter.api.AfterAll;
8586
import org.junit.jupiter.api.BeforeAll;
8687
import org.junit.jupiter.api.BeforeEach;
@@ -321,28 +322,43 @@ public void testClose() {
321322

322323
@Test
323324
public void testCloseConsumer() {
325+
// null consumer
324326
final Closeable nullCloseable = null;
325-
assertDoesNotThrow(() -> IOUtils.close(nullCloseable, null)); // null consumer
326-
assertDoesNotThrow(() -> IOUtils.close(new StringReader("s"), null)); // null consumer
327-
assertDoesNotThrow(() -> IOUtils.close(new ThrowOnCloseReader(new StringReader("s")), null)); // null consumer
328-
329-
final IOConsumer<IOException> nullConsumer = null; // null consumer doesn't throw
327+
assertDoesNotThrow(() -> IOUtils.close(nullCloseable, null));
328+
assertDoesNotThrow(() -> IOUtils.close(new StringReader("s"), null));
329+
assertDoesNotThrow(() -> IOUtils.close(new ThrowOnCloseReader(new StringReader("s")), null));
330+
// null consumer doesn't throw
331+
final IOConsumer<IOException> nullConsumer = null;
330332
assertDoesNotThrow(() -> IOUtils.close(nullCloseable, nullConsumer));
331333
assertDoesNotThrow(() -> IOUtils.close(new StringReader("s"), nullConsumer));
332334
assertDoesNotThrow(() -> IOUtils.close(new ThrowOnCloseReader(new StringReader("s")), nullConsumer));
333-
334-
final IOConsumer<IOException> silentConsumer = IOConsumer.noop(); // noop consumer doesn't throw
335+
// noop consumer doesn't throw
336+
final IOConsumer<IOException> silentConsumer = IOConsumer.noop();
335337
assertDoesNotThrow(() -> IOUtils.close(nullCloseable, silentConsumer));
336338
assertDoesNotThrow(() -> IOUtils.close(new StringReader("s"), silentConsumer));
337339
assertDoesNotThrow(() -> IOUtils.close(new ThrowOnCloseReader(new StringReader("s")), silentConsumer));
338-
339-
final IOConsumer<IOException> noisyConsumer = i -> {
340-
throw i;
341-
}; // consumer passes on the throw
342-
assertDoesNotThrow(() -> IOUtils.close(nullCloseable, noisyConsumer)); // no throw
343-
assertDoesNotThrow(() -> IOUtils.close(new StringReader("s"), noisyConsumer)); // no throw
344-
assertThrows(IOException.class,
345-
() -> IOUtils.close(new ThrowOnCloseReader(new StringReader("s")), noisyConsumer)); // closeable throws
340+
// consumer passes on the throw
341+
final IOConsumer<IOException> noisyConsumer = ExceptionUtils::rethrow;
342+
// no throw
343+
assertDoesNotThrow(() -> IOUtils.close(nullCloseable, noisyConsumer));
344+
// no throw
345+
assertDoesNotThrow(() -> IOUtils.close(new StringReader("s"), noisyConsumer));
346+
// closeable throws
347+
assertThrows(IOException.class, () -> IOUtils.close(new ThrowOnCloseReader(new StringReader("s")), noisyConsumer));
348+
// consumes other than IOException
349+
final AtomicBoolean b = new AtomicBoolean();
350+
final IOConsumer<IOException> consumer = e -> b.set(true);
351+
// IOException subclass
352+
assertDoesNotThrow(() -> IOUtils.close(new BrokenOutputStream((Throwable) new EOFException()), consumer));
353+
assertTrue(b.get());
354+
b.set(false);
355+
// RuntimeException
356+
assertDoesNotThrow(() -> IOUtils.close(new BrokenOutputStream(new RuntimeException()), consumer));
357+
assertTrue(b.get());
358+
b.set(false);
359+
// RuntimeException subclass
360+
assertDoesNotThrow(() -> IOUtils.close(new BrokenOutputStream(new UnsupportedOperationException()), consumer));
361+
assertTrue(b.get());
346362
}
347363

348364
@Test

0 commit comments

Comments
 (0)