
- •Practical Unit Testing with JUnit and Mockito
- •Table of Contents
- •About the Author
- •Acknowledgments
- •Preface
- •Preface - JUnit
- •Part I. Developers' Tests
- •Chapter 1. On Tests and Tools
- •1.1. An Object-Oriented System
- •1.2. Types of Developers' Tests
- •1.2.1. Unit Tests
- •1.2.2. Integration Tests
- •1.2.3. End-to-End Tests
- •1.2.4. Examples
- •1.2.5. Conclusions
- •1.3. Verification and Design
- •1.5. Tools Introduction
- •Chapter 2. Unit Tests
- •2.1. What is a Unit Test?
- •2.2. Interactions in Unit Tests
- •2.2.1. State vs. Interaction Testing
- •2.2.2. Why Worry about Indirect Interactions?
- •Part II. Writing Unit Tests
- •3.2. Class To Test
- •3.3. Your First JUnit Test
- •3.3.1. Test Results
- •3.4. JUnit Assertions
- •3.5. Failing Test
- •3.6. Parameterized Tests
- •3.6.1. The Problem
- •3.6.2. The Solution
- •3.6.3. Conclusions
- •3.7. Checking Expected Exceptions
- •3.8. Test Fixture Setting
- •3.8.1. Test Fixture Examples
- •3.8.2. Test Fixture in Every Test Method
- •3.8.3. JUnit Execution Model
- •3.8.4. Annotations for Test Fixture Creation
- •3.9. Phases of a Unit Test
- •3.10. Conclusions
- •3.11. Exercises
- •3.11.1. JUnit Run
- •3.11.2. String Reverse
- •3.11.3. HashMap
- •3.11.4. Fahrenheits to Celcius with Parameterized Tests
- •3.11.5. Master Your IDE
- •Templates
- •Quick Navigation
- •Chapter 4. Test Driven Development
- •4.1. When to Write Tests?
- •4.1.1. Test Last (AKA Code First) Development
- •4.1.2. Test First Development
- •4.1.3. Always after a Bug is Found
- •4.2. TDD Rhythm
- •4.2.1. RED - Write a Test that Fails
- •How To Choose the Next Test To Write
- •Readable Assertion Message
- •4.2.2. GREEN - Write the Simplest Thing that Works
- •4.2.3. REFACTOR - Improve the Code
- •Refactoring the Tests
- •Adding Javadocs
- •4.2.4. Here We Go Again
- •4.3. Benefits
- •4.4. TDD is Not Only about Unit Tests
- •4.5. Test First Example
- •4.5.1. The Problem
- •4.5.2. RED - Write a Failing Test
- •4.5.3. GREEN - Fix the Code
- •4.5.4. REFACTOR - Even If Only a Little Bit
- •4.5.5. First Cycle Finished
- •‘The Simplest Thing that Works’ Revisited
- •4.5.6. More Test Cases
- •But is It Comparable?
- •Comparison Tests
- •4.6. Conclusions and Comments
- •4.7. How to Start Coding TDD
- •4.8. When not To Use Test-First?
- •4.9. Should I Follow It Blindly?
- •4.9.1. Write Good Assertion Messages from the Beginning
- •4.9.2. If the Test Passes "By Default"
- •4.10. Exercises
- •4.10.1. Password Validator
- •4.10.2. Regex
- •4.10.3. Booking System
- •Chapter 5. Mocks, Stubs, Test Spies
- •5.1. Introducing Mockito
- •5.1.1. Creating Test Doubles
- •5.1.2. Expectations
- •5.1.3. Verification
- •5.1.4. Conclusions
- •5.2. Types of Test Double
- •5.2.1. Code To Be Tested with Test Doubles
- •5.2.2. The Dummy Object
- •5.2.3. Test Stub
- •5.2.4. Test Spy
- •5.2.5. Mock
- •5.3. Putting it All Together
- •5.4. Example: TDD with Test Doubles
- •5.4.2. The Second Test: Send a Message to Multiple Subscribers
- •Refactoring
- •5.4.3. The Third Test: Send Messages to Subscribers Only
- •5.4.4. The Fourth Test: Subscribe More Than Once
- •Mockito: How Many Times?
- •5.4.5. The Fifth Test: Remove a Subscriber
- •5.4.6. TDD and Test Doubles - Conclusions
- •More Test Code than Production Code
- •The Interface is What Really Matters
- •Interactions Can Be Tested
- •Some Test Doubles are More Useful than Others
- •5.5. Always Use Test Doubles… or Maybe Not?
- •5.5.1. No Test Doubles
- •5.5.2. Using Test Doubles
- •No Winner So Far
- •5.5.3. A More Complicated Example
- •5.5.4. Use Test Doubles or Not? - Conclusion
- •5.6. Conclusions (with a Warning)
- •5.7. Exercises
- •5.7.1. User Service Tested
- •5.7.2. Race Results Enhanced
- •5.7.3. Booking System Revisited
- •5.7.4. Read, Read, Read!
- •Part III. Hints and Discussions
- •Chapter 6. Things You Should Know
- •6.1. What Values To Check?
- •6.1.1. Expected Values
- •6.1.2. Boundary Values
- •6.1.3. Strange Values
- •6.1.4. Should You Always Care?
- •6.1.5. Not Only Input Parameters
- •6.2. How to Fail a Test?
- •6.3. How to Ignore a Test?
- •6.4. More about Expected Exceptions
- •6.4.1. The Expected Exception Message
- •6.4.2. Catch-Exception Library
- •6.4.3. Testing Exceptions And Interactions
- •6.4.4. Conclusions
- •6.5. Stubbing Void Methods
- •6.6. Matchers
- •6.6.1. JUnit Support for Matcher Libraries
- •6.6.2. Comparing Matcher with "Standard" Assertions
- •6.6.3. Custom Matchers
- •6.6.4. Advantages of Matchers
- •6.7. Mockito Matchers
- •6.7.1. Hamcrest Matchers Integration
- •6.7.2. Matchers Warning
- •6.8. Rules
- •6.8.1. Using Rules
- •6.8.2. Writing Custom Rules
- •6.9. Unit Testing Asynchronous Code
- •6.9.1. Waiting for the Asynchronous Task to Finish
- •6.9.2. Making Asynchronous Synchronous
- •6.9.3. Conclusions
- •6.10. Testing Thread Safe
- •6.10.1. ID Generator: Requirements
- •6.10.2. ID Generator: First Implementation
- •6.10.3. ID Generator: Second Implementation
- •6.10.4. Conclusions
- •6.11. Time is not on Your Side
- •6.11.1. Test Every Date (Within Reason)
- •6.11.2. Conclusions
- •6.12. Testing Collections
- •6.12.1. The TDD Approach - Step by Step
- •6.12.2. Using External Assertions
- •Unitils
- •Testing Collections Using Matchers
- •6.12.3. Custom Solution
- •6.12.4. Conclusions
- •6.13. Reading Test Data From Files
- •6.13.1. CSV Files
- •6.13.2. Excel Files
- •6.14. Conclusions
- •6.15. Exercises
- •6.15.1. Design Test Cases: State Testing
- •6.15.2. Design Test Cases: Interactions Testing
- •6.15.3. Test Collections
- •6.15.4. Time Testing
- •6.15.5. Redesign of the TimeProvider class
- •6.15.6. Write a Custom Matcher
- •6.15.7. Preserve System Properties During Tests
- •6.15.8. Enhance the RetryTestRule
- •6.15.9. Make an ID Generator Bulletproof
- •Chapter 7. Points of Controversy
- •7.1. Access Modifiers
- •7.2. Random Values in Tests
- •7.2.1. Random Object Properties
- •7.2.2. Generating Multiple Test Cases
- •7.2.3. Conclusions
- •7.3. Is Set-up the Right Thing for You?
- •7.4. How Many Assertions per Test Method?
- •7.4.1. Code Example
- •7.4.2. Pros and Cons
- •7.4.3. Conclusions
- •7.5. Private Methods Testing
- •7.5.1. Verification vs. Design - Revisited
- •7.5.2. Options We Have
- •7.5.3. Private Methods Testing - Techniques
- •Reflection
- •Access Modifiers
- •7.5.4. Conclusions
- •7.6. New Operator
- •7.6.1. PowerMock to the Rescue
- •7.6.2. Redesign and Inject
- •7.6.3. Refactor and Subclass
- •7.6.4. Partial Mocking
- •7.6.5. Conclusions
- •7.7. Capturing Arguments to Collaborators
- •7.8. Conclusions
- •7.9. Exercises
- •7.9.1. Testing Legacy Code
- •Part IV. Listen and Organize
- •Chapter 8. Getting Feedback
- •8.1. IDE Feedback
- •8.1.1. Eclipse Test Reports
- •8.1.2. IntelliJ IDEA Test Reports
- •8.1.3. Conclusion
- •8.2. JUnit Default Reports
- •8.3. Writing Custom Listeners
- •8.4. Readable Assertion Messages
- •8.4.1. Add a Custom Assertion Message
- •8.4.2. Implement the toString() Method
- •8.4.3. Use the Right Assertion Method
- •8.5. Logging in Tests
- •8.6. Debugging Tests
- •8.7. Notifying The Team
- •8.8. Conclusions
- •8.9. Exercises
- •8.9.1. Study Test Output
- •8.9.2. Enhance the Custom Rule
- •8.9.3. Custom Test Listener
- •8.9.4. Debugging Session
- •Chapter 9. Organization Of Tests
- •9.1. Package for Test Classes
- •9.2. Name Your Tests Consistently
- •9.2.1. Test Class Names
- •Splitting Up Long Test Classes
- •Test Class Per Feature
- •9.2.2. Test Method Names
- •9.2.3. Naming of Test-Double Variables
- •9.3. Comments in Tests
- •9.4. BDD: ‘Given’, ‘When’, ‘Then’
- •9.4.1. Testing BDD-Style
- •9.4.2. Mockito BDD-Style
- •9.5. Reducing Boilerplate Code
- •9.5.1. One-Liner Stubs
- •9.5.2. Mockito Annotations
- •9.6. Creating Complex Objects
- •9.6.1. Mummy Knows Best
- •9.6.2. Test Data Builder
- •9.6.3. Conclusions
- •9.7. Conclusions
- •9.8. Exercises
- •9.8.1. Test Fixture Setting
- •9.8.2. Test Data Builder
- •Part V. Make Them Better
- •Chapter 10. Maintainable Tests
- •10.1. Test Behaviour, not Methods
- •10.2. Complexity Leads to Bugs
- •10.3. Follow the Rules or Suffer
- •10.3.1. Real Life is Object-Oriented
- •10.3.2. The Non-Object-Oriented Approach
- •Do We Need Mocks?
- •10.3.3. The Object-Oriented Approach
- •10.3.4. How To Deal with Procedural Code?
- •10.3.5. Conclusions
- •10.4. Rewriting Tests when the Code Changes
- •10.4.1. Avoid Overspecified Tests
- •10.4.2. Are You Really Coding Test-First?
- •10.4.3. Conclusions
- •10.5. Things Too Simple To Break
- •10.6. Conclusions
- •10.7. Exercises
- •10.7.1. A Car is a Sports Car if …
- •10.7.2. Stack Test
- •Chapter 11. Test Quality
- •11.1. An Overview
- •11.2. Static Analysis Tools
- •11.3. Code Coverage
- •11.3.1. Line and Branch Coverage
- •11.3.2. Code Coverage Reports
- •11.3.3. The Devil is in the Details
- •11.3.4. How Much Code Coverage is Good Enough?
- •11.3.5. Conclusion
- •11.4. Mutation Testing
- •11.4.1. How does it Work?
- •11.4.2. Working with PIT
- •11.4.3. Conclusions
- •11.5. Code Reviews
- •11.5.1. A Three-Minute Test Code Review
- •Size Heuristics
- •But do They Run?
- •Check Code Coverage
- •Conclusions
- •11.5.2. Things to Look For
- •Easy to Understand
- •Documented
- •Are All the Important Scenarios Verified?
- •Run Them
- •Date Testing
- •11.5.3. Conclusions
- •11.6. Refactor Your Tests
- •11.6.1. Use Meaningful Names - Everywhere
- •11.6.2. Make It Understandable at a Glance
- •11.6.3. Make Irrelevant Data Clearly Visible
- •11.6.4. Do not Test Many Things at Once
- •11.6.5. Change Order of Methods
- •11.7. Conclusions
- •11.8. Exercises
- •11.8.1. Clean this Mess
- •Appendix A. Automated Tests
- •A.1. Wasting Your Time by not Writing Tests
- •A.1.1. And what about Human Testers?
- •A.1.2. One More Benefit: A Documentation that is Always Up-To-Date
- •A.2. When and Where Should Tests Run?
- •Appendix B. Running Unit Tests
- •B.1. Running Tests with Eclipse
- •B.1.1. Debugging Tests with Eclipse
- •B.2. Running Tests with IntelliJ IDEA
- •B.2.1. Debugging Tests with IntelliJ IDEA
- •B.3. Running Tests with Gradle
- •B.3.1. Using JUnit Listeners with Gradle
- •B.3.2. Adding JARs to Gradle’s Tests Classpath
- •B.4. Running Tests with Maven
- •B.4.1. Using JUnit Listeners and Reporters with Maven
- •B.4.2. Adding JARs to Maven’s Tests Classpath
- •Appendix C. Test Spy vs. Mock
- •C.1. Different Flow - and Who Asserts?
- •C.2. Stop with the First Error
- •C.3. Stubbing
- •C.4. Forgiveness
- •C.5. Different Threads or Containers
- •C.6. Conclusions
- •Appendix D. Where Should I Go Now?
- •Bibliography
- •Glossary
- •Index
- •Thank You!

Chapter 7. Points of Controversy
Discussion is an exchange of knowledge; an argument an exchange of ignorance.
— Robert Quillen
There are many topics related to writing tests that are still up for debate within the developer community. In this section we will discuss some of these. Sometimes a definite answer will be given, sometimes a variety of options will be presented – for you, dear reader, to decide about.
7.1. Access Modifiers
In the code shown throughout the book, I follow unwritten standards of the Java world relating to the use of access modifiers. In particular, I used the private keyword for almost every instance variable. There is nothing wrong with this approach; however, some people may find it unnecessary.
The point here is that while with production code we tend to be obsessed with encapsulation and information hiding, this is not the case with test classes. They are almost never used by other classes 1, which means that the fact that some field or method is marked with private does not really make code any better. Hence, if you are in the habit of marking everything with the private keyword, then by all means go ahead and adopt this approach with test code as well. On the other hand, if you think that by omitting private you are going to gain some readability in respect of your test code, then do get rid of it. (And don’t let any zealots intimidate you into doing otherwise!)
7.2. Random Values in Tests
At first, testing and random values seem to fit together nicely. Some people apparently think that running tests each time with different values can better prove the robustness of their code than running it with some selected values. In my experience, this is a fallacy, and using random values in tests causes more harm than good. In fact, I can recall instances of randomness being used in tests where it proved futile, but none where it turned out to be worthwhile! This section shows some typical cases that are to be avoided.
There definitely are some areas where random tests are not only valid, but also vital! For example the test suites of Apache Lucene2 contain a lot of randomly generated test cases which allowed them to report an important Java 7 bug3.
Before we start discussing code, however, let me just say that there are many ways to generate random values, including:
•custom "utility methods",
•using libraries for general use; e.g. Apache Commons Lang library4 provides a RandomStringUtils class, which contains a plethora of methods for generating random Strings of the requested length and consisting of different character sets (e.g. only letters, only numbers, a combination of both, etc.),
1This is not 100% true, as sometimes we use inheritance, so some test classes inherit the fields and methods of a base class. 2http://lucene.apache.org
3See http://blog.thetaphi.de/2011/07/real-story-behind-java-7-ga-bugs.html 4http://commons.apache.org/lang/
146

Chapter 7. Points of Controversy
• using specialized test-focused frameworks like Quickcheck5.
The issues that I would like to discuss here are not really related to any particular way of generating random values, but rather to the idea of using randomness in your tests.
7.2.1. Random Object Properties
Let us assume that an SUT (of the UserToPersonConverter class) can translate an object of the User class into objects of the Person class by turning the user’s name and surname into a person’s nickname. The implementation of the SUT is shown below:
Listing 7.1. The UserToPersonConverter class
public class UserToPersonConverter {
public static Person convert(User user) {
return new Person(user.getName() + " " + user.getSurname());
}
}
A test of this class can take advantage of random values. Such an attempt is shown in Listing 7.2.
Listing 7.2. Test of the UserToPersonConverter class
public class UserToPersonConverterTest {
@Test
public void shouldConvertUserNamesIntoPersonNick() { String name = RandomStringUtils.randomAlphabetic(8);
String surname = RandomStringUtils.randomAlphabetic(12); User user = new User(name, surname);
Person person = UserToPersonConverter.convert(user);
assertEquals(name + " " + surname, person.getNick());
}
}
As far as I understand the idea behind the implementation shown in Listing 7.2, random values are to be used because they (supposedly) highlight the robustness of a given test. It is as if the test were to shout "Hey look, every time I execute the test it uses a different user’s name and still passes! Amazing, isn’t it?".
7.2.2. Generating Multiple Test Cases
A natural next step in the direction of making the test even more impressive would be to test the same functionality with more random values, within a single execution of a test method. This can be achieved using data providers, for example:
5http://java.net/projects/quickcheck
147

Chapter 7. Points of Controversy
Listing 7.3. Creation of multiple random test cases
@RunWith(JUnitParamsRunner.class)
public class UserToPersonConverterDataProvidersTest {
private static Object[] getRandomNames() { Object[] values = new Object[100];
for (int i = 0; i < values.length; i++) {
values[i] = $(RandomStringUtils.randomAlphabetic(8), RandomStringUtils.randomAlphabetic(12));
}
return values;
}
@Test
@Parameters(method = "getRandomNames")
public void shouldConvertUserNamesIntoPersonNick( String name, String surname) {
User user = new User(name, surname);
Person person = UserToPersonConverter.convert(user); assertEquals(name + " " + surname, person.getNick());
}
}
This test method is executed 100 times with different random values.
Even if the test in Listing 7.3 looks much more serious than the previous one, it is not really much stronger. It only tricks us into thinking that UserToPersonConverter has now been thoroughly tested. Unfortunately it hasn’t been.
Let us take another look at the implementation of the UserToPersonConverter class (shown in Listing 7.1). Has it been tested more effectively, just because we have passed 100 nonsensical, unpronounceable names (each time of the same length, namely 8 and 12)? I do not think so. The probability that tests 2 to 100 will reveal some bugs not discovered by the first test is minimal. The diversity of test parameters is very limited, and so is the value added by each of the tests. It would not increase, even if we were to up the number of randomly generated values from 100 to 1000.
When it comes to testing, I would rather prioritize the quality of test cases over the quantity. A good quality test case can expose some holes within the production code. I suggest thinking carefully about the possible scenarios (as discussed in Section 6.1) and then making a deliberate choice of some values. My test of the UserToPersonConverter class would definitely contain cases of names and surnames of varied length (including empty strings), with varied capitalization. I might even use some randomness to avoid putting so many names directly in the test code; however, I would make sure that some borderline cases (e.g. empty or extremely long strings) are verified. The variety of my test parameters would definitely be far greater than that generated by the data provider in Listing 7.3.
7.2.3. Conclusions
As you will have observed for yourself, generating multiple tests with random values does not automatically make the test suite stronger. Instead, it can complicate your tests and give rise to a number of issues. However, if randomness is really appropriate within the context of your testing domain, then at least try to do it right. Table 7.1 offers some hints and discusses issues relating to the use of random values in your tests.
148

Chapter 7. Points of Controversy
Table 7.1. Issues with the random values of parameters
(possible) issue |
comment |
Why should I test with |
You should not. "Random" does not have to mean "any". Take |
nonsensical values (e.g. |
control over how parameters values are generated, e.g. use |
"(*&KLNHF_98234" passed as String |
RandomStringUtils.randomAlphabetic() method from the |
name parameter)? |
Apache Commons Lang library in cases of the name parameter. |
|
|
I do not have control over random |
Do not rely solely on random values! Apart from random |
values, so how can I guarantee |
values, you also have to make sure boundary values (and any |
that some values (e.g. boundary |
other values which are important to your code) are being used. |
values) will always be checked? |
Probably you need two sources of parameters - one controlled by |
|
you, and one random. |
|
|
Hard to repeat. How can I repeat |
Repeatability is an important thing. If you do not have it, then |
something which is random? |
you might know there is a bug in your code, but you will not be |
|
able to nail it down - which will surely cost you a few gray hairs. |
|
However, it is possible to repeat your tests, even when random |
|
values are being used. |
|
First of all, XML files created by JUnit during the execution of |
|
tests contain comprehensive information on parameters passed to |
|
test methods, so you can see what values were used there. You |
|
can also control the way the random values are generated by |
|
choosing the seed, remembering it (e.g. by writing to some test |
|
report), and using it later to repeat the testsa. |
"Flickering tests" - every |
Every time the test fails, add the values which made it fail to |
execution of the test suite could |
your test. And of course, make it pass with these values. |
end with a different result. |
|
|
|
Weird error messages. |
Using random test values can result in obscure failure messages. |
|
For example, you can learn that the test failed because of some |
|
issue with user "IkldlDFg yw,cKxH.zDIF". This might be very |
|
confusing, especially if the test case has nothing to do with |
|
the user’s name. I would suggest using much simpler values, |
|
which reveal the intention much better - e.g. "ANY_NAME |
|
ANY_SURNAME" would be much more clear. |
aIf you are interested in details, please refer to Jaroslav Tulach’s article at http://wiki.apidesign.org/wiki/RandomizedTest
Some specialized tools, e.g. Quickcheck, provide solutions for some of the issues described above.
I have to admit I am not a big fan of using random values in tests. I do not see many cases where using them would improve the test. In general, I would rather suggest:
•spending more time pondering over important values which should be verified (see the discussion in Section 6.1) instead of generating thousands of random cases.
•using meaningful names instead of random garbage strings.
149