Testing content of list ignoring some of the fields

homaxto picture homaxto · Nov 29, 2017 · Viewed 13.5k times · Source

I have a scenario where I receive a list from a method call and I would like to assert that the list contains the correct elements. One way to do this would be to look for some detail in each element to see which expected element to compare with - eg. a name. However the elements also contains a random generated UUID that I do not care about comparing.
And then I thought a test tool might come to my rescue. Take the following simplified example.

I have a class Dog:

public class Dog {
    private String name;
    private Integer age;
}

They are contained in a list:

List<Dog> dogs = ... many dogs

Now I want to test that the list contains the expected dogs, but for some reason I do not know some of the fields - let us say age. I have tried both using assertj and hamcrest but I cannot find the correct solution that both compares two lists while also ignoring some of the fields.

Until now this is what I have (using hamcrest):

List<Dog> actualDogs = codeUndertest.findDogs(new Owner("Basti"));
List<Dog> expectedDogs = createExpectedListOfDogsWithoutTheAge();

Matcher.assertThat(actualDogs.get(0), Matcher.is(com.shazam.shazamcrest.matcher.Matchers
    .sameBeanAs(expectedDogs.(0))
    .ignoring("age")
))

This works but it only compares two objects of class Dog. How do I compare all the dogs in the two list?
Bonus question: how do I compare the lists without knowing the order or if I only need to assert that the expected dogs are contained in the list.

Answer

Joel Costigliola picture Joel Costigliola · Nov 30, 2017

Give a try to AssertJ's usingElementComparatorIgnoringFields:

Employee bill = new Employee("Bill", 60, "Micro$oft");
Employee appleBill = new Employee("Billie", 60, "Apple");
List<Employee> employees = newArrayList(bill, appleBill);

Employees[] expectedEmployees = { new Employee("Bill", 60, "Google"), 
                                  new Employee("Billie", 60, "Facebook") };
// this assertion succeeds as we don't compare the company field.     
assertThat(employees).usingElementComparatorIgnoringFields("company")
                     .contains(expectedEmployees);

edit: The new recursive comparison API can be used, it gives much finer control about what is being compared: https://assertj.github.io/doc/#assertj-core-recursive-comparison-ignoring-fields