Converting a DataTable into a Java Object
In 'Different patterns for creating Cucumber Feature files' we learnt various ways/patters to represent feature or product specification. Here, we will continue the same but with more of a real world use case. Let us see 3 patterns to represent complex objects in feature files. Imagine we are creating a REST API. In order to test the API behavior, we need to call the API from step definition. If the payload for the API to test is expected as JSON, we need to convert the data in DataTable to a JSON so that it can be used as payload to call REST API from step definition. If we can convert DataTable to a Java Object, we can easily convert that to a JSON. Let us learn that here.
Let’s imagine we are working on creating a HR Software and we need to implement the following requirement/user story.
As a HR Staff,
I want to to save a new employee details,
So that I can refer to the details in the system when needed.
Let us assume the following fields represents the basic details of an employee,
import java.time.LocalDate;
import lombok.Getter;
import lombok.Setter;
import lombok.ToString;
@Getter
@Setter
@ToString
public class Employee {
private Long id;
private String firstName;
private String lastName;
private String email;
private LocalDate dateOfBirth;
private boolean remoteWorker;
private String jobTitle;
private String employeeNumber;
private String employmentStatus;
private String employmentType;
}
And the sample JSON representation is,
{
"id": 1,
"firstName": "Effie",
"lastName": "Slee",
"email": "eslee0@blueocean.com",
"dateOfBirth": "2000-01-01",
"remoteWorker": false,
"jobTitle": "Physical Therapy Assistant",
"employeeNumber": 101,
"employeeStatus": "Active",
"employmentType": "Full-Time"
}
Pattern: One Step Per Field
Let us convert the above said high level requirement or user story to a feature file,
Feature File
Feature: Create Employee
Scenario: Create employee with basic details
Given employee first name is 'Effie'
And employee last name is 'Slee'
And employee email is 'eslee0@blueocean.com'
And employee date of birth is '2000-01-01'
And employee remote worker flag is 'false'
And employee job title is 'Physical Therapy Assistant'
And employee number is 'E101'
And employment status is 'Active'
And employment type is 'Full-Time'
When user saves a new employee
Then the save IS SUCCESSFUL
Corresponding Step Definitions
import com.madrascoder.cucumberbooksample.dto.Employee;
import io.cucumber.java.en.And;
import io.cucumber.java.en.Given;
import io.cucumber.java.en.Then;
import io.cucumber.java.en.When;
import java.time.LocalDate;
public class EmployeeStepDefinitions {
private Employee employee = new Employee();
@Given("employee first name is {string}")
public void employeeFirstNameIs(String firstName) {
employee.setFirstName(firstName);
}
@And("employee last name is {string}")
public void employeeLastNameIs(String lastName) {
employee.setLastName(lastName);
}
@And("employee email is {string}")
public void employeeEmailIs(String email) {
employee.setEmail(email);
}
@And("employee date of birth is {string}")
public void employeeJoiningDateIs(String dateOfBirthStr) {
LocalDate dateOfBirth = LocalDate.parse(dateOfBirthStr);
employee.setDateOfBirth(dateOfBirth);
}
@And("employee remote worker flag is {boolean}")
public void employeeRemoteWorkerFlagIs(boolean remoteWorker) {
employee.setRemoteWorker(remoteWorker);
}
@And("employee job title is {string}")
public void employeeJobTitleIs(String jobTitle) {
employee.setJobTitle(jobTitle);
}
@And("employee number is {string}")
public void employeeNumberIs(String employeeNumber) {
employee.setEmployeeNumber(employeeNumber);
}
@And("employment status is {string}")
public void employmentStatusIs(String employmentStatus) {
employee.setEmploymentStatus(employmentStatus);
}
@And("employment type is {string}")
public void employmentTypeIs(String employmentType) {
employee.setEmploymentType(employmentType);
}
@When("user saves a new employee")
public void userSavesANewEmployee() {
// call REST API to create employee
// save REST API Response
}
@Then("the save IS SUCCESSFUL")
public void theSaveIsSuccessful() {
// assert if Response status is 201 Created or 200 Ok.
}
}
Note: Creating step definitions for each step is time consuming, representations are some time hierarchical and it becomes very hard to maintain.
So, let us try another approach.
Pattern: Map<Key, Value> DataTable
Feature File
Feature: Create Employee
Scenario: Create employee with basic details
Given user wants to create employee with following details
| firstName | Effie |
| lastName | Slee |
| email | eslee@blueocean.com |
| dateOfBirth | 2000-01-01 |
| remoteWorker | false |
| jobTitle | Physical Therapy Assistant |
| employeeNumber | E101 |
| employmentStatus | Active |
| employmentType | Full-Time |
When user saves a new employee
Then the save IS SUCCESSFUL
Corresponding Step Definitions
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import com.madrascoder.cucumberbooksample.dto.Employee;
import io.cucumber.java.en.Given;
import io.cucumber.java.en.Then;
import io.cucumber.java.en.When;
import java.util.Map;
public class EmployeeStepDefinitions {
private Employee employee;
@Given("user wants to create employee with following details")
public void userWantsToCreateEmployeeWithFollowingDetails(Map<String, Object> employeeMap) {
// Don't create ObjectMapper instance like the one below, it is just an example
// Create one instance of ObjectMapper and keep reusing every where, else autowire Spring's ObjectMapper bean
final ObjectMapper objectMapper = new ObjectMapper();
objectMapper.registerModule(new JavaTimeModule())
.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
this.employee = objectMapper.convertValue(employeeMap, Employee.class);
}
@When("user saves a new employee")
public void userSavesANewEmployee() {
// call REST API to create employee
// save REST API Response
}
@Then("the save IS SUCCESSFUL")
public void theSaveIsSuccessful() {
// assert if Response status is 201 Created or 200 Ok.
}
}
Using Map<String, Object> DataTable made life little easy. But reading the Map DataTable is not easy. Most of us are used to spreadsheet style, header row at the top and all values under it. Reading Key on the left and value on the right in a DataTable may add some cognitive load.
In the next pattern, let’s do spreadsheet like DataTable.
Pattern: Spreadsheet DataTable
Feature File
Feature: Create Employee
Scenario: Create employee with basic details
Given user wants to create employee with following details
| firstName | lastName | email | dateOfBirth | remoteWorker | jobTitle | employeeNumber | employeeStatus | employmentType |
| Effie | Slee | eslee@blueocean.com | 2000-01-01 | false | Physical Therapy Assistant | E101 | Active | Full-Time |
When user saves a new employee
Then the save IS SUCCESSFUL
IMPORTANT NOTE: Header row in the data table contains the camelCase field names of Employee.java. This makes life easy for Cucumber to automatically convert the DataTable into a Employee Object. This is one of the convention that is going to help us a lot.
If needed, we can make the the Header row as Title Case, we can write a few lines of code to convert it to Java field names if we follow some convention.
Prerequisite for Step Definitions
Before we implement the step definition, we need to create a generic data transformer class that converts any DataTable to corresponding Java Object. Here is that magic class,
Navigate to following location and create a class to automatically transform DataTable to a Java Object,
cd src/test/com/madrascoder/cucumberbooksample
touch DataTransformer.java
Add the following code,
import com.fasterxml.jackson.databind.ObjectMapper;
import io.cucumber.java.DefaultDataTableCellTransformer;
import io.cucumber.java.DefaultDataTableEntryTransformer;
import io.cucumber.java.DefaultParameterTransformer;
import java.lang.reflect.Type;
public class DataTransformer {
// Set spring.jackson.serialization.write-dates-as-timestamps=false
// in application.properties
// This is required to convert ISO 8601 Date String to LocalDate and
// many other java.time package use cases
private final ObjectMapper objectMapper;
// Spring will auto wire the ObjectMapper
public DataTransformer(ObjectMapper objectMapper) {
this.objectMapper = objectMapper;
}
@DefaultParameterTransformer
@DefaultDataTableEntryTransformer
@DefaultDataTableCellTransformer
public Object transformer(Object fromValue, Type toValueType) {
return objectMapper.convertValue(fromValue,
objectMapper.constructType(toValueType));
}
}
Also, set the following property in application.properties
. This helps converting ISO 8601 Date and Time formats to java.time.* classes.
spring.jackson.serialization.write-dates-as-timestamps=false
Corresponding Step Definitions
import com.madrascoder.cucumberbooksample.dto.Employee;
import io.cucumber.java.en.Given;
import io.cucumber.java.en.Then;
import io.cucumber.java.en.When;
import lombok.extern.slf4j.Slf4j;
@Slf4j
public class EmployeeStepDefinitions {
private Employee employee;
@Given("user wants to create employee with following details")
public void userWantsToCreateEmployeeWithFollowingDetails(Employee employee) {
log.info("Employee: {}", employee);
this.employee = employee;
}
@When("user saves a new employee")
public void userSavesANewEmployee() {
// call REST API to create employee
// save REST API Response
}
@Then("the save IS SUCCESSFUL")
public void theSaveIsSuccessful() {
// assert if Response status is 201 Created or 200 Ok.
}
}
Look at the above step definition file, compare it with other patterns stated above. You will know how DataTransformer.java
and Spreadsheet DataTable Pattern together made life easy.
There is one draw back here, if the object has more fields, viewing DataTable may require scrolling horizontally. If you are using a wide screen monitor, it can be avoided to some extent. Considering the fact that its easier for Product Owner to understand and developer to create step definitions, having to do the horizontal scroll is the trade off and it is worth it.
Quick Tip
You may use below strategy to create DataTable header row. Remember to replace Employee.class
with your respective class.
public static void main(String[] args) {
final String header = Stream.of(Employee.class.getDeclaredFields())
.map(Field::getName)
.collect(Collectors.joining("|", "|", "|"));
System.out.println(header);
}
[OR]
To generate mock data for DataTable, you may use Mockaroo custom formatter with pipe delimiter. For more information refer https://www.mockaroo.com/
Representing Hierarchical Data
In the real world, an Employee may have associated Addresses. How do we represent that in feature file and step definition.
Let us assume, Address has the following fields,
import lombok.EqualsAndHashCode;
import lombok.Getter;
import lombok.Setter;
import lombok.ToString;
@Getter
@Setter
@ToString
@EqualsAndHashCode
public class Address {
private Long id;
private String type;
private String addressLine1;
private String addressLine2;
private String city;
private String state;
private String zipcode;
private String country;
}
Employee has list of associated addresses,
import java.time.LocalDate;
import java.util.List;
import javax.validation.constraints.Email;
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.Past;
import lombok.EqualsAndHashCode;
import lombok.Getter;
import lombok.Setter;
import lombok.ToString;
@Getter
@Setter
@ToString
@EqualsAndHashCode
public class Employee {
private Long id;
@NotBlank(message = "First name is required")
private String firstName;
@NotBlank(message = "Last name is required")
private String lastName;
@Email(message = "Valid email id is required")
private String email;
@Past(message = "Expecting past date for date of birth")
private LocalDate dateOfBirth;
private boolean remoteWorker;
private String jobTitle;
private String employeeNumber;
private String employmentStatus;
private String employmentType;
private List<Address> addresses; // Look here
}
Feature File
Feature: Create Employee
Scenario: Create employee with basic details
Given user wants to create employee with following details
| firstName | lastName | email | dateOfBirth | remoteWorker | jobTitle | employeeNumber | employeeStatus | employmentType |
| Effie | Slee | eslee@blueocean.com | LocalDate.now-6570 | NO | Physical Therapy Assistant | E101 | Active | Full-Time |
And with following address
| type | addressLine1 | addressLine2 | city | state | country |
| Office | 1000 Blue Ocean Drive | Suite 200 | Miami | FL | USA |
When user saves a new employee
Then the save 'IS SUCCESSFUL'
Corresponding Step Definitions
import com.madrascoder.cucumberbooksample.dto.Address;
import com.madrascoder.cucumberbooksample.dto.Employee;
import io.cucumber.java.en.And;
import io.cucumber.java.en.Given;
import io.cucumber.java.en.When;
import io.restassured.mapper.TypeRef;
import io.restassured.response.Response;
import java.util.List;
import java.util.Map;
public class EmployeeStepDefinitions extends AbstractStepDefinitions {
@Given("user wants to create/update employee with following details")
public void userWantsToCreateEmployeeWithFollowingDetails(Employee employee) {
testContext().setPayload(employee);
}
@And("with following details address(es)")
public void withFollowingDetailsAddress(List<Address> addresses) {
final Employee payload = testContext().getPayload(Employee.class);
payload.setAddresses(addresses);
}
...
We can keep enhancing the payload with all the child or associated objects using different steps as stated above.
Conclusion
In this chapter, we saw 3 patterns
1. One Step Per Field Pattern
2. Map<Key, Object> DataTable Pattern
3. Spreadsheet DataTable Pattern
My recommendation is to use the Spreadsheet DataTable Pattern when ever you need to represent a complex object in feature file.
You may be wondering, how can to represent hierarchical information in feature file using Spreadsheet DataTable Pattern. In the future chapter, I will introduce that when handling
One Employee can have Many Address
use case.
In the next chapter, let’s apply Spreadsheet DataTable Pattern to test the REST API to ‘create employee’.
References
Credits
Photo by CHUTTERSNAP on Unsplash
Previous Chapter | Scroll Up to Top | Table of Contents | Next Chapter