Published
- 6 min read
Java Unit Tests Made Easy — Random Values with PODAM

In this article Java Unit Tests make easy — Random Values with PODAM, we discuss how to generate random values to object, collections we use on Junit test using PODAM library.
When we write unit test cases for functions, a big headache for developers is filling mock data to our objects, collections etc.
For example, let see what would be the test case to fetch an object from a function.
Please look below code snippet. In this object contains the list of child objects too.
@Test
void testFindAuthorByUserId() {
when(authorRepository.findById(anyLong()))
.thenReturn(
new AuthorDoc(
Long.valueOf(1),
"orgCode",
"firstName",
"lastName",
new GregorianCalendar(2020, Calendar.DECEMBER, 14, 10, 50).getTime(),
"email",
Arrays.asList(
new BookDoc(
Long.valueOf(1),
"book Name 1",
"isbn1",
LocalDateTime.of(2020, Month.DECEMBER, 14, 10, 50, 28),
LocalDateTime.of(2020, Month.DECEMBER, 14, 10, 50, 28),
"createdBy",
"modifiedBy"
),
new BookDoc(
Long.valueOf(2),
"book Name 2",
"isbn2",
LocalDateTime.of(2020, Month.DECEMBER, 14, 10, 50, 28),
LocalDateTime.of(2020, Month.DECEMBER, 14, 10, 50, 28),
"createdBy",
"modifiedBy"
)
),
LocalDateTime.of(2020, Month.DECEMBER, 14, 10, 50, 28),
LocalDateTime.of(2020, Month.DECEMBER, 14, 10, 50, 28),
"createdBy",
"modifiedBy"
)
);
AuthorDto result = authorService.findAuthorByUserId(Long.valueOf(1));
Assertions.assertEquals(
new AuthorDto(
Long.valueOf(1),
"orgCode",
"firstName",
"lastName",
new GregorianCalendar(2020, Calendar.DECEMBER, 14, 10, 50).getTime(),
"email",
Arrays.asList(
new BookDto(Long.valueOf(1), "book Name 1", "isbn1"),
new BookDto(Long.valueOf(2), "book Name 2", "isbn2")
)
),
result
);
}
In this code snippet, we can see a lot of hardcoded data that we have to use to fill the object with values.
Now, let’s see what we can do for this…
Any good solution??
Yes, we have PODAM.
What is PODAM?
PODAM is a lightweight tool to auto-fill Java POJOs with data. This comes in handy when developing unit tests. Thanks to PODAM, users now have a one-liner that does all the work for them.
What are the features of PODAM?
- Automatic introspection of your POJOs to fill all possible attributes with random data. You can customize this behavior.
- Extension points to customize the data strategy, and fields exclusions based on names or annotations.
- Support for your custom annotations to either provide data or exclude fields.
- The option to select minimal constructors (for lightweight POJOs) or full constructors.
- Support for Java types, enumerations, arrays, and collections (lists, sets, maps).
- Support for generics.
- Support for your factories when PODAM cannot introspect enough information to generate a type (e.g.,
XMLGregorianCalendar
). - Support for arbitrary method executions when the typical lifecycle has ended.
Using PODAM in a Spring Boot Application
We have to add PODAM dependencies to the Spring Boot project.
Please use the latest version. When I’m writing this article, the latest version is 7.2.5.RELEASE.
For Maven
For Maven
<dependency>
<groupId>uk.co.jemos.podam</groupId>
<artifactId>podam</artifactId>
<version>7.2.5.RELEASE</version>
<scope>test</scope>
</dependency>
For Gradle
testCompile group: 'uk.co.jemos.podam', name: 'podam', version: '7.2.5.RELEASE'
Now we are going to refactor the above mentions code using Podam.
## **Pojo using podom**
<table>
<tr>
<td><code>@Test</code></td>
</tr>
<tr>
<td><code>void testFindAuthorByUserId() {</code></td>
</tr>
<tr>
<td><code> PodamFactory factory = new PodamFactoryImpl();</code></td>
</tr>
<tr>
<td><code> AuthorDoc authorDoc = factory.manufacturePojo(AuthorDoc.class);</code></td>
</tr>
<tr>
<td><code> when(authorRepository.findById(anyLong())).thenReturn(authorDoc);</code></td>
</tr>
<tr>
<td><code> AuthorDto result = authorService.findAuthorByUserId(Long.valueOf(1));</code></td>
</tr>
<tr>
<td><code> Assertions.assertEquals(authorDoc.getUserId(), result.getUserId());</code></td>
</tr>
<tr>
<td><code> Assertions.assertEquals(authorDoc.getFirstName(), result.getFirstName());</code></td>
</tr>
<tr>
<td><code> Assertions.assertEquals(authorDoc.getBooks().size(), result.getBooks().size());</code></td>
</tr>
<tr>
<td><code>}</code></td>
</tr>
</table>
Can you believe this?
46 line of code we reduce to less than ten lines.
Let’s see what happen there. Let’s break down.
First, we implement PodamFactory instance.
PodamFactory factory = new PodamFactoryImpl();
Then using the above PodamFactory, we can create an object with values.
AuthorDoc authorDoc = factory.manufacturePojo(AuthorDoc.class);
if we debug this, we can see the assigned values to the object like below
Let see more examples
List (String) using podom
Let’s see how we can build a List of String using Podam.
List (Pojo) using podom
If we need to test some function with the list object, it is an immense pain.
<table>
<tr>
<td><code>@Test</code></td>
</tr>
<tr>
<td><code>void testFindByOrgCode() {</code></td>
</tr>
<tr>
<td><code>when(authorRepository.findByOrgCode(anyString()))</code></td>
</tr>
<tr>
<td><code>.thenReturn(</code></td>
</tr>
<tr>
<td><code>Arrays.asList(</code></td>
</tr>
<tr>
<td><code>new AuthorDoc(</code></td>
</tr>
<tr>
<td><code>Long.valueOf(1),</code></td>
</tr>
<tr>
<td><code>"orgCode",</code></td>
</tr>
<tr>
<td><code>"firstName",</code></td>
</tr>
<tr>
<td><code>"lastName",</code></td>
</tr>
<tr>
<td><code>new GregorianCalendar(2020, Calendar.DECEMBER, 14, 10, 50).getTime(),</code></td>
</tr>
<tr>
<td><code>"email",</code></td>
</tr>
<tr>
<td><code>Arrays.asList(</code></td>
</tr>
<tr>
<td><code>new BookDoc(</code></td>
</tr>
<tr>
<td><code>Long.valueOf(1),</code></td>
</tr>
<tr>
<td><code>"book Name 1",</code></td>
</tr>
<tr>
<td><code>"isbn1",</code></td>
</tr>
<tr>
<td><code>LocalDateTime.of(2020, Month.DECEMBER, 14, 10, 50, 28),</code></td>
</tr>
<tr>
<td><code>LocalDateTime.of(2020, Month.DECEMBER, 14, 10, 50, 28),</code></td>
</tr>
<tr>
<td><code>"createdBy",</code></td>
</tr>
<tr>
<td><code>"modifiedBy")</code></td>
</tr>
<tr>
<td><code>)),</code></td>
</tr>
<tr>
<td><code>LocalDateTime.of(2020, Month.DECEMBER, 14, 10, 50, 28),</code></td>
</tr>
<tr>
<td><code>LocalDateTime.of(2020, Month.DECEMBER, 14, 10, 50, 28),</code></td>
</tr>
<tr>
<td><code>"createdBy",</code></td>
</tr>
<tr>
<td><code>"modifiedBy")));</code></td>
</tr>
<tr>
<td><code>List<AuthorDto> result = authorService.findByOrgCode("orgCode");</code></td>
</tr>
<tr>
<td><code>Assertions.assertEquals(</code></td>
</tr>
<tr>
<td><code>Arrays.asList(</code></td>
</tr>
<tr>
<td><code>new AuthorDto(</code></td>
</tr>
<tr>
<td><code>Long.valueOf(1),</code></td>
</tr>
<tr>
<td><code>"orgCode",</code></td>
</tr>
<tr>
<td><code>"firstName",</code></td>
</tr>
<tr>
<td><code>"lastName",</code></td>
</tr>
<tr>
<td><code>new GregorianCalendar(2020, Calendar.DECEMBER, 14, 10, 50).getTime(),</code></td>
</tr>
<tr>
<td><code>"email",</code></td>
</tr>
<tr>
<td><code>Arrays.asList(</code></td>
</tr>
<tr>
<td><code>new BookDto(Long.valueOf(1), "book Name 1", "isbn1"),</code></td>
</tr>
<tr>
<td><code>new BookDto(Long.valueOf(2), "book Name 2", "isbn2")))),</code></td>
</tr>
<tr>
<td><code>result);</code></td>
</tr>
<tr>
<td><code>}</code></td>
</tr>
</table>
How many code lines we saved???
Map(String, Pojo) using podom
Same like the above example, we can quickly build a map of date using podam.
<table>
<tr>
<td><code>@Test</code></td>
</tr>
<tr>
<td><code>void testFindAuthorMap() {</code></td>
</tr>
<tr>
<td><code> PodamFactory factory = new PodamFactoryImpl();</code></td>
</tr>
<tr>
<td><code> Map<String, AuthorDoc> authorDocMap =</code></td>
</tr>
<tr>
<td><code> factory.manufacturePojo(HashMap.class, String.class, AuthorDoc.class);</code></td>
</tr>
<tr>
<td><code> when(authorRepository.findMapByOrgCode(anyString())).thenReturn(authorDocMap);</code></td>
</tr>
<tr>
<td><code> Map<String, AuthorDto> result = authorService.findAuthorMap("orgCode");</code></td>
</tr>
<tr>
<td><code> result.forEach(</code></td>
</tr>
<tr>
<td><code> (key, value) -></code></td>
</tr>
<tr>
<td><code> Assertions.assertEquals(authorDocMap.get(key).getUserId(), value.getUserId()));</code></td>
</tr>
<tr>
<td><code>}</code></td>
</tr>
</table>
Defining an attribute-level strategy
So far, we see podam assign sample values to our POJO attributes. What happened if there is a validation. For example, if its an Email field we have to give an exact email address.
for that, we can use data strategies at the attribute level
Lets create a EmailStrategy class
<table>
<tr>
<td><code>public class EmailStrategy implements AttributeStrategy<String> {</code></td>
</tr>
<tr>
<td><code> @Override</code></td>
</tr>
<tr>
<td><code> public String getValue(Class<?> attrType, List<Annotation> attrAnnotations) {</code></td>
</tr>
<tr>
<td><code> return "aaa.bbb@ccc.ddd";</code></td>
</tr>
<tr>
<td><code> }</code></td>
</tr>
<tr>
<td><code>}</code></td>
</tr>
</table>
we can use this in 2 ways,
<table>
<tr>
<td><code>@PodamStrategyValue(EmailStrategy.class)</code>
</td>
<td>
</td>
</tr>
<tr>
<td>
</td>
<td><code>private String email;</code>
</td>
</tr>
</table>
- config strategy when we create PadomFactory. then it will auto-assign POJO level validations
PodamFactory factory = new PodamFactoryImpl();Class<? extends Annotation> annotation = Email.class;AttributeStrategy<?> strategy = new EmailStrategy();((RandomDataProviderStrategy)factory.getStrategy()).addOrReplaceAttributeStrategy(annotation, strategy);
Limit collection size.
In our previous example, we have a list of books in the author object. Let’s see how we can limit the size of this list. For that, we can use @PodamCollection annotation.
We added collection elements size as 3. so this will only create 3 records of book attribute on that list.
For a full list of supported annotations, please refer to the podam Annotations page.
Conclusion
Thanks for reading the article Java Unit Tests make easy — Random Values with PODAM.
You can find source codes for this tutorial from our Github.
Originally published at https://onloadcode.com on December 14, 2020.
📖 Read all my articles on Medium: @madukajayawardana.