Use JUnit’s expected exceptions sparingly, from the JOOQ blog, says not to use JUnit‘s “expected exceptions” feature. (Or, more accurately, to use it “sparingly.”) It’s good advice, but it’s also largely limited to JUnit (with one main exception).
What they’re addressing is the transition from this pattern:
@Test
public void testValueOfIntInvalid() {
try {
ubyte((UByte.MIN_VALUE) - 1);
fail();
}
catch (NumberFormatException e) {}
}
To this pattern:
@Test(expected = NumberFormatException.class)
public void testValueOfShortInvalidCase1() {
ubyte((short) ((UByte.MIN_VALUE) - 1));
}
They have four reasons that this doesn’t actually get you anything:
- We’re not really gaining anything in terms of number of lines of code
- We’ll have to refactor it back anyway
- The single method call is not the unit
- Annotations are always a bad choice for control flow structuring
Of these, “The single method call is not the unit” is by far the strongest point.
We’re not really gaining anything in terms of number of lines of code
It’s a matter of opinion, but we generally are gaining something in terms of the code. The refactored test has one line of actual code: ubyte((short) ((UByte.MIN_VALUE) - 1));
. Everything around it is window dressing. The original test has the try/catch
code, which adds a lot of noise – and discards the “error condition,” which is the actual metric for success (it fails the test if there’s no error.)
Is this a big deal? … No, it’s not. But it’s also not a point worth making.
We’ll have to refactor it back anyway
Here’s where JUnit itself messes things up. The authors actually have a really good point, if you’re locked into JUnit:
In the annotation-driven approach, all I can do is test for the exception type. I cannot make any assumptions about the exception message for instance, in case I do want to add further tests, later on. Consider this:
try { ubyte((UByte.MIN_VALUE) - 1); fail("Reason for failing"); } catch (NumberFormatException e) { assertEquals("some message", e.getMessage()); assertNull(e.getCause()); ... }
Ah, JUnit. JOOQ is making the assertion (see what I did there?) that they can’t examine Reason for failing
or some message
with the annotation.
They’re right… if you’re locked into JUnit. TestNG can, of course, look at the message and validate it. If the message doesn’t fit the regex, then the test fails, just as expected (and desired).
@Test(expectedExceptions={NumberFormatException.class},
expectedExceptionsMessageRegExp="Reason for failing")
public void testThings() {
...
The single method call is not the unit
Here’s what JOOQ says:
The unit test was called
testValueOfIntInvalid()
. So, the semantic “unit†being tested is that of theUByte
type’svalueOf()
behaviour in the event of invalid input in general. Not for a single value, such asUByte.MIN_VALUE - 1
.
They’re running yet again into a limitation of JUnit. The actual point they’re making is correct, but TestNG provides a @DataProvider
mechanism, where you’d actually give the test a signature of testValueOfIntInvalid(int)
and pass in a set of integers. That means you can test a range and corner cases, with a single appropriate test.
People have written a data-provider-like feature for JUnit (see junit-dataprovider) – which only highlights how JUnit is affecting the choices made by the JOOQ team.
Annotations are always a bad choice for control flow structuring
Hard to argue with this one: Java has excellent control flow, even if it’s verbose at times. However, the annotation is direct enough (with TestNG, at least) that you aren’t really violating control flow: you’re saying “this is a test, it has these execution rules, and this is how you measure success.” With JUnit, you don’t have the same control.
JUnit.next
According to JOOQ, the next generation of JUnit (junit-lambda) fixes things a little, by offering a lambda solution to exceptions:
Exception e = expectThrows(NumberFormatException.class, () ->
ubyte((UByte.MIN_VALUE) - 1));
assertEquals("abc", e.getMessage());
That’s admirable, but they’re still behind TestNG by a few generations – and there’s still no data factory mechanism.
In Conclusion
Their closing statement is actually really good:
Annotations were abused for a lot of logic, mostly in the JavaEE and Spring environments, which were all too eager to move XML configuration back into Java code. This has gone the wrong way, and the example provided here clearly shows that there is almost always a better way to write out control flow logic explicitly both using object orientation or functional programming, than by using annotations.
It’s worth noting that the statement was actually driven by a lack of features in JUnit (when compared to TestNG) but the point remains.