Unit test

How to do a Parameterized Unit Test in Quarkus?

Parameterized unit tests are beneficial for applications as they simplify the process of writing tests, allowing us to run a single test multiple times with different parameters.

Requirements

I’m using JDK 17, Intellij and Quarkus Version 3.4 on the project.

JDK17

Practice

I created this class:

package com.natancode.dto;

import jakarta.validation.constraints.*;

public class Task {

    @Pattern(regexp = "^[0-9]*$")
    @Size(max = 20)
    @NotNull(message = "Cannot be null")
    @NotBlank(message = "Cannot be empty")
    private String id;

    @Pattern(regexp = "[\\w\\W\\s]*")
    @Size(min=5, max = 20)
    @NotNull(message = "Cannot be null")
    @NotBlank(message = "Cannot be empty")
    private String name;

    private StatusTask status;

    public Task(String id, String name, StatusTask status) {
        this.id = id;
        this.name = name;
        this.status = status;
    }
}

These notations represent validations applied to the fields. For instance, in the ‘id‘ field, we use the ‘@Pattern‘ notation to verify if the ‘id‘ value conforms to the given regular expression. In this case, it only allows numeric values.

My current objective is to test the ‘id‘ field, so let’s proceed with a unit test to validate it.

Unit Test

We have multiple validations to perform on the ‘id‘ field. I aim to validate four aspects of the ‘id‘ value:

  • Whether it is null.
  • Whether it is empty.
  • Whether it is numeric.
  • Whether it adheres to the maximum value allowed.

To streamline this process, we don’t need to create four separate tests; we can utilize parameterized tests.

We require the Hibernate Validator library to conduct the validation, and I’ve developed a generic method for this purpose.

package com.natancode.util;

import com.natancode.dto.CustomException;
import com.natancode.dto.Error;
import jakarta.validation.ConstraintViolation;
import jakarta.validation.Validator;

import java.util.ArrayList;
import java.util.Set;

public class ValidationFields {

    public static void ValidateFiels(Object object, Validator validator) throws CustomException {
        Set<ConstraintViolation<Object>> violations = validator.validate(object);
        if(!violations.isEmpty()) {
            var errors = new ArrayList< Error >();
            violations.forEach(error -> {
                errors.add( new Error(500, error.getMessage(),
                        object.getClass().getName(), error.getPropertyPath().toString()) );
            });
            throw new CustomException(errors);
        }
    }
}

Now we can do the unit test.

package com.natancode.dto;

import com.natancode.util.ValidationFields;
import io.quarkus.test.junit.QuarkusTest;
import jakarta.inject.Inject;
import jakarta.validation.Validator;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.NullAndEmptySource;
import org.junit.jupiter.params.provider.ValueSource;

import static org.junit.jupiter.api.Assertions.assertThrows;

@QuarkusTest
public class TaskTest {

    @Inject
    Validator validator;

    @ParameterizedTest
    @NullAndEmptySource
    @ValueSource( strings = {"b", "111111111111111111111"})
    void shouldValidateTaskId(String id) {
        var task = new Task(id, "watch the course", StatusTask.BACKLOG);
        assertThrows(CustomException.class, () -> ValidationFields.ValidateFiels(task, validator) );
    }
}

We utilize @ParameterizedTest to indicate that it’s a parameterized test.

@NullAndEmptySource is employed to validate null and empty values.

For @ValueSource, we specify the required values. In this case, since the field type is String, we use strings within @ValueSource.

It’s worth noting that @ValueSource can also accommodate other field types such as integers and more.

The test method accepts these parameters and runs the test for each of them.

Conclusion

We achieved all four validations with a shorter and simpler code, resulting in cleaner code.

The code is available here:

https://github.com/natanlf/quarkus-unit-tests/tree/ParameterizedTest

Author

  • Natan Ferreira

    I am a seasoned Full Stack Software Developer with 8+ years of experience, including 6+ years specializing in Java with Spring and Quarkus. My core expertise lies in developing robust RESTful APIs integrated with Cosmos Db, MySQL, and cloud platforms like Azure and AWS. I have extensive experience designing and implementing microservices architectures, ensuring performance and reliability for high-traffic systems. In addition to backend development, I have experience with Angular to build user-friendly interfaces, leveraging my postgraduate degree in frontend web development to deliver seamless and responsive user experiences. My dedication to clean and secure code led me to present best practices to my company and clients, using tools like Sonar to ensure code quality and security. I am a critical thinker, problem solver, and team player, thriving in collaborative environments while tackling complex challenges. Beyond development, I share knowledge through my blog, NatanCode, where I write about Java, Spring, Quarkus, databases, and frontend development. My passion for learning and delivering innovative solutions drives me to excel in every project I undertake.