Geparametriseerd testen

Introductie tot testen in Java

Maria Milusheva

Senior Software Engineer

Voorbeeld: Validatie van gebruikersnaam

Denk aan de gebruikersnaam-validatie uit Hoofdstuk 1:

public static boolean isValidUsername(String username) {
    if (username == null || username.isEmpty() || username.contains(" ")) {
        return false;
    }
    return username.length() >= 3;
}
Introductie tot testen in Java

Voorbeeld: Herhalende validatietest

Herinner je dat de tests er zo uitzagen:

@Test
void isValidUsername_returnsFalse() {
    String username = "____";

    boolean actual = isValidUsername(username);

    assertFalse(actual);
}

We hoeven dit geen 3 keer te schrijven!

1 https://en.wikipedia.org/wiki/Don%27t_repeat_yourself
Introductie tot testen in Java

Geparametriseerde tests: syntax

In plaats van @Test gebruiken we @ParameterizedTest.

Deze importeer je uit:

import org.junit.jupiter.params.ParameterizedTest;

We hebben ook nog een annotatie nodig: @ValueSource.

Die importeer je uit:

import org.junit.jupiter.params.provider.ValueSource;
Introductie tot testen in Java

Geparametriseerde tests: voorbeeldtest

@ParameterizedTest
@ValueSource(strings = {"", "jane doe"}) // LET OP: het is "strings", niet String
void isValidUsername_returnsFalse(String username) { // Tests kunnen argumenten aannemen
    boolean actual = isValidUsername(username);

    assertFalse(actual);
}

JUnit voert dit uit als twee tests:

Tests run: 2, Failures: 0, Errors: 0, Skipped: 0
All tests passed!
Introductie tot testen in Java

Beperkingen van @ValueSource

@ValueSource is beperkt tot bepaalde types.

@ValueSource(bytes = {1, 2, 3, 42})
@ValueSource(shorts = {100, 200, 300})
@ValueSource(ints = {1, 2, 3, 42})
@ValueSource(longs = {1000L, 2000L, 3000L})
@ValueSource(floats = {1.0f, 3.14159f, 2.71828f})
@ValueSource(doubles = {3.14, 2.71828, 1.4142})
@ValueSource(chars = {'a', 'b', 'c', 'Z'})
@ValueSource(booleans = {true, false})
@ValueSource(strings = {"Hello", "JUnit", "5", "Parameter"})
@ValueSource(classes = {String.class, Integer.class, ValueSourceExamples.class})
1 https://junit.org/junit5/docs/5.7.1/api/org.junit.jupiter.params/org/junit/jupiter/params/provider/ValueSource.html
Introductie tot testen in Java

@NullSource

We kunnen geen null aan @ValueSource doorgeven; dat compileert niet:

@ParameterizedTest
@ValueSource(strings = {"", "jane doe", null}) // Compileert niet
void isValidUsername_returnsFalse(String username) {
    boolean actual = isValidUsername(username);

    assertFalse(actual);
}

Gebruik in plaats daarvan @NullSource.

Introductie tot testen in Java

De volledige ParameterizedTest draaien

Als we null aan @ValueSource doorgeven, compileert het niet.

@ParameterizedTest
@NullSource // Voegt een testgeval met null toe
@ValueSource(strings = {"", "jane doe"})
void isValidUsername_returnsFalse(String username) {
    boolean actual = isValidUsername(username);

    assertFalse(actual);
}
Tests run: 3, Failures: 0, Errors: 0, Skipped: 0
All tests passed!
Introductie tot testen in Java

Meerdere argumenten per test

Nog een beperking van @ValueSource: je kunt maar één waarde per test doorgeven.

Bekijk deze methode:

int countLetters(String input) {
    if (input != null) {
        return input.length();
    }
    return 0;
}
Introductie tot testen in Java

@CsvSource

Voor meerdere waarden gebruik je @CsvSource (CSV = comma-separated values):

@ParameterizedTest
@CsvSource({"Hello World, 11", "DataCamp, 8", "'', 0", ", 0"})
void countLetters_countsLetters(String input, int expected) {
    int actual = countLetters(input);

    assertEquals(expected, actual);
}

Let op: ", 0" wordt ingevoerd als null, 0. Voor een lege string gebruik je "'', 0" in JUnit 5.10

@CsvSource has similar type limitations as @ValueSource.

Introductie tot testen in Java

Let op: Source-annotaties importeren

Alle @____Source-annotaties kun je importeren uit:

import org.junit.jupiter.params.provider.*;
Introductie tot testen in Java

Laten we oefenen!

Introductie tot testen in Java

Preparing Video For Download...