Skip to content

Commit c9897b5

Browse files
authored
Merge pull request #26 from digipost/asUnchecked-enhancements
DiggExceptions.asUnchecked(..) enhancements
2 parents 460b8f2 + 08ba192 commit c9897b5

File tree

2 files changed

+104
-7
lines changed

2 files changed

+104
-7
lines changed

src/main/java/no/digipost/DiggExceptions.java

+70-6
Original file line numberDiff line numberDiff line change
@@ -15,12 +15,21 @@
1515
*/
1616
package no.digipost;
1717

18-
import no.digipost.function.*;
19-
18+
import no.digipost.function.ThrowingBiConsumer;
19+
import no.digipost.function.ThrowingBiFunction;
20+
import no.digipost.function.ThrowingConsumer;
21+
import no.digipost.function.ThrowingFunction;
22+
import no.digipost.function.ThrowingRunnable;
23+
import no.digipost.function.ThrowingSupplier;
24+
25+
import java.io.IOException;
26+
import java.io.UncheckedIOException;
2027
import java.util.function.Consumer;
2128
import java.util.function.Function;
2229
import java.util.stream.Stream;
2330

31+
import static no.digipost.DiggBase.friendlyName;
32+
2433
public final class DiggExceptions {
2534

2635
/**
@@ -35,16 +44,71 @@ public static Stream<Throwable> causalChainOf(Throwable t) {
3544
return causes.build();
3645
}
3746

47+
/**
48+
* Generate a concise description of an exception/throwable
49+
* containing the {@link DiggBase#friendlyName(Class) "friendly name"}
50+
* of the exception's type, and its {@link Throwable#getMessage() message}.
51+
*
52+
* @param t the exception/throwable
53+
*
54+
* @return the description
55+
*/
3856
public static String exceptionNameAndMessage(Throwable t) {
39-
return t.getClass().getSimpleName() + ": '" + t.getMessage() + "'";
57+
return friendlyName(t.getClass()) + ": '" + t.getMessage() + "'";
4058
}
4159

60+
61+
/**
62+
* Utility for acquiring a {@link RuntimeException} from any {@link Throwable}.
63+
* This method is appropriate to use when you have caught an exception which you have no
64+
* ability to handle, and you need to just throw it from a context where you are
65+
* not allowed to.
66+
* <p>
67+
* If you want to add more context to the thrown exception (which you probably should), consider
68+
* using {@link #asUnchecked(Throwable, String)} or {@link #asUnchecked(Throwable, Function)}.
69+
*
70+
* @param t the exception (Throwable)
71+
*
72+
* @return a new {@link RuntimeException} which has the given exception as its {@link Throwable#getCause() cause},
73+
* or {@code t} itself casted to {@code RuntimeException} if possible
74+
*/
4275
public static RuntimeException asUnchecked(Throwable t) {
43-
return asUnchecked(t, DiggExceptions::exceptionNameAndMessage);
76+
return t instanceof RuntimeException ? (RuntimeException) t : asUnchecked(t, DiggExceptions::exceptionNameAndMessage);
77+
}
78+
79+
80+
/**
81+
* Utility for acquiring a {@link RuntimeException} with a custom message from any {@link Throwable}.
82+
* The result from this method will <em>always</em> be a new exception instance.
83+
*
84+
* @param <X> the type of the given exception
85+
* @param t the exception (Throwable)
86+
* @param messageCreator a function which returns the message for the new exception
87+
*
88+
* @return a new {@link RuntimeException} which has the result from the {@code messageCreator} function as its
89+
* {@link Throwable#getMessage() message}, and the given exception as its {@link Throwable#getCause() cause}
90+
*/
91+
public static <X extends Throwable> RuntimeException asUnchecked(X t, Function<? super X, String> messageCreator) {
92+
return asUnchecked(t, messageCreator.apply(t));
4493
}
4594

46-
public static <X extends Throwable> RuntimeException asUnchecked(X t, Function<? super X, String> message) {
47-
return t instanceof RuntimeException ? (RuntimeException) t : new RuntimeException(message.apply(t), t);
95+
96+
/**
97+
* Utility for acquiring a {@link RuntimeException} with a custom message from any {@link Throwable}.
98+
* The result from this method will <em>always</em> be a new exception instance.
99+
*
100+
* @param t the exception (Throwable)
101+
* @param message the message for the new exception
102+
*
103+
* @return a new {@link RuntimeException} which has the given {@code message},
104+
* and the given exception as its {@link Throwable#getCause() cause}
105+
*/
106+
public static RuntimeException asUnchecked(Throwable t, String message) {
107+
if (t instanceof IOException) {
108+
return new UncheckedIOException(message, (IOException) t);
109+
} else {
110+
return new RuntimeException(message, t);
111+
}
48112
}
49113

50114
/**

src/test/java/no/digipost/DiggExceptionsTest.java

+34-1
Original file line numberDiff line numberDiff line change
@@ -16,30 +16,35 @@
1616
package no.digipost;
1717

1818
import no.digipost.concurrent.OneTimeToggle;
19+
import org.junit.jupiter.api.Nested;
1920
import org.junit.jupiter.api.Test;
2021

2122
import java.io.IOException;
23+
import java.io.UncheckedIOException;
2224
import java.util.List;
2325
import java.util.Optional;
2426
import java.util.function.Consumer;
2527

26-
import static uk.co.probablyfine.matchers.Java8Matchers.where;
2728
import static java.util.stream.Collectors.toList;
2829
import static no.digipost.DiggExceptions.applyUnchecked;
30+
import static no.digipost.DiggExceptions.asUnchecked;
2931
import static no.digipost.DiggExceptions.causalChainOf;
3032
import static no.digipost.DiggExceptions.getUnchecked;
3133
import static no.digipost.DiggExceptions.mayThrow;
3234
import static no.digipost.DiggExceptions.runUnchecked;
3335
import static org.hamcrest.MatcherAssert.assertThat;
3436
import static org.hamcrest.Matchers.contains;
37+
import static org.hamcrest.Matchers.containsString;
3538
import static org.hamcrest.Matchers.empty;
3639
import static org.hamcrest.Matchers.instanceOf;
3740
import static org.hamcrest.Matchers.is;
3841
import static org.hamcrest.Matchers.nullValue;
3942
import static org.hamcrest.Matchers.sameInstance;
43+
import static org.junit.jupiter.api.Assertions.assertAll;
4044
import static org.junit.jupiter.api.Assertions.assertThrows;
4145
import static org.mockito.Mockito.mock;
4246
import static org.mockito.Mockito.verify;
47+
import static uk.co.probablyfine.matchers.Java8Matchers.where;
4348

4449
public class DiggExceptionsTest {
4550

@@ -98,6 +103,34 @@ public void factoryMethodsForThrowingFunctionalInterfaces() throws Throwable {
98103
}
99104

100105

106+
@Nested
107+
class AsUnchecked {
108+
109+
@Test
110+
void castToUnchecked() {
111+
Exception e = new IllegalStateException();
112+
assertThat(e, where(DiggExceptions::asUnchecked, sameInstance(e)));
113+
}
114+
115+
@Test
116+
void UnknownCheckedExceptionBecomesRuntimeException() {
117+
class MyCheckedException extends Exception {
118+
public MyCheckedException() {
119+
super("Who in their right mind would define their own checked exception");
120+
}
121+
}
122+
RuntimeException uncheckedException = asUnchecked(new MyCheckedException());
123+
assertAll(
124+
() -> assertThat(uncheckedException, where(Object::getClass, is(RuntimeException.class))),
125+
() -> assertThat(uncheckedException, where(Exception::getMessage, containsString("DiggExceptionsTest.AsUnchecked.MyCheckedException"))),
126+
() -> assertThat(uncheckedException, where(Exception::getMessage, containsString("Who in their right mind"))));
127+
}
128+
129+
@Test
130+
void ioExceptionAsUncheckedIOException() {
131+
assertThat(new IOException("error"), where(DiggExceptions::asUnchecked, instanceOf(UncheckedIOException.class)));
132+
}
101133

134+
}
102135

103136
}

0 commit comments

Comments
 (0)