AssertJ как способ значительно улучшить код ваших тестов

Привет, Хабр!

В 2019–2020 годах на одном из проектов я был идейным вдохновителем перехода на JUnit 5. Для проверок мы использовали стандартные ассерты и Hamcrest. Тогда мне казалось, что этого более чем достаточно. Один из наших lead-инженеров предлагал AssertJ как более «модное и молодёжное» решение, но поддержки эта идея не получила. Я был одним из тех, кто выступал против AssertJ. Каюсь, был грешен :)

За последние пару лет, несмотря на менеджерскую позицию, я написал свыше пятисот тестов, и мой подход к тестированию претерпел значительные изменения. В этой статье я постараюсь объяснить, почему AssertJ — это лучшее решение для проверок в тестах, существующее сегодня (год 2022 от Р.X.). Разумеется, всё ниже сказанное — это моё субъективное мнение.

1. Переходите на JUnit 5, если ещё нет

Да, совет «капитанский», но действительно важный. Старые версии должны кануть в Лету, в том числе и JUnit 4. Во всех проектах, где я участвую, явным образом через checkstyle запрещаю использование классов из JUnit 4 (пример тут).


    
    
    

Делаю так, потому что полностью убрать JUnit 4 с classpath часто невозможно, например, из-за Testcontainers (см. issue).

2. Структурируйте свои тесты

Я приверженец AAA-подхода: Arrange-Act-Assert. Вы можете также использовать Given-When-Then — принципиально сути это не меняет. Требуйте от разработчиков, чтобы в каждом тесте был ассерт! Для контроля можно (и нужно!) использовать статический анализатор, например, PMD и его JUnitTestsShouldIncludeAssert.

Я не фанат слепого поклонения каким-либо правилам и допускаю в одном тесте несколько действий и несколько проверок. Вместе с тем использование AssertJ сильно облегчает переход к парадигме один тест — один ассерт. Достигается это за счёт fluent API — одной из ключевых особенностей AssertJ.

@Test
void shouldSatisfyContract() {
    assertThat(check)
        .hasType(Index.class)
        .hasDiagnostic(Diagnostic.INVALID_INDEXES)
        .hasHost(PgHostImpl.ofPrimary());
}

Больше примеров тут.

3. Откажитесь от традиционных ассертов

Ассерты в стиле JUnit были очень хороши… лет 20 назад. Сейчас они устарели и представляют собой пример не самого удачного дизайна. Сможете сходу вспомнить порядок следования expected и actual?

final String actual = doSomething();
// Так?
Assertions.assertEquals(actual, "expected");
// Или так?
Assertions.assertEquals("expected", actual);

А сможете научить всех своих инженеров, включая новичков, не путать их местами?

AssertJ by design лишён этой проблемы:

assertThat(actual).isEqualTo("expected");

4. Делайте ваши тесты более читаемыми, используя естественный язык

При использовании традиционных ассертов код ваших тестов выглядит искусственным и с трудом читается вслух. Сравните:

final Account a = makeAccount();
assertEquals("RussianAccount{id=1, currency=BaseCurrency(isoCode=RUB), number=30102810100000000001, active=true, balance=0, holder=Party{Revolut LLC, type=LEGAL_PERSON, tax identification number=7703408188, id=1}, chapter=BALANCE}",
    a.toString());

и

final Account a = makeAccount();
assertThat(a)
    .hasToString("RussianAccount{id=1, currency=BaseCurrency(isoCode=RUB), number=30102810100000000001, active=true, balance=0, holder=Party{Revolut LLC, type=LEGAL_PERSON, tax identification number=7703408188, id=1}, chapter=BALANCE}");

Вот ещё пример:

// JUnit
assertEquals(1, cache.size());

// AssertJ
assertThat(cache)
    .hasSize(1);

AssertJ предоставляет красивые и удобные методы для проверки типовых вещей: эквивалентности, хэш кода, размера коллекции и т.д.

assertThat(second)
    .isNotEqualTo(first)
    .doesNotHaveSameHashCodeAs(first);
assertThat(index.getIndexNames())
    .hasSize(2)
    .containsExactly("index3", "index4")
    .isUnmodifiable();

Если вы работали с BigDecimal в тестах, то, вероятно, сталкивались с проблемой проверки значений из-за разного масштаба: обычно вместо equals приходится применять compareTo. AssertJ частично устраняет эту проблему за счёт метода isEqualByComparingTo:

@Test
void moneyProblem() {
    final BigDecimal one = new BigDecimal("1.000");
  
    // AssertJ
    assertThat(one).isEqualByComparingTo(BigDecimal.ONE); // pass
  
    // JUnit
    Assertions.assertEquals(0, one.compareTo(BigDecimal.ONE)); // pass
    Assertions.assertEquals(BigDecimal.ONE, one); // fail
}

5. Полностью откажитесь от Hamcrest

Любой код живёт, развивается и рано или поздно умирает. Какие-то вещи сначала становятся популярными, а потом выходят из моды. Не цепляйтесь за устаревающие проекты. Hamcrest не радует нас новыми версиями с октября 2019. Просто замените его более современным решением:

// Было - Hamcrest
assertThat(indexes.stream()
    .map(TableNameAware::getTableName)
    .collect(Collectors.toSet()), containsInAnyOrder("t", "demo.t", "test.t"));

// Стало - AssertJ
assertThat(indexes.stream()
    .map(TableNameAware::getTableName)
    .collect(Collectors.toSet())).containsExactlyInAnyOrder("t", "demo.t", "test.t");

6. Используйте возможности функционального подхода

AssertJ поддерживает функциональный подход и позволяет вам преобразовывать исходные данные. Предыдущий пример можно значительно улучшить, сократив количество кода и увеличив его читаемость:

assertThat(indexes)
    .flatExtracting(TableNameAware::getTableName)
    .containsExactlyInAnyOrder("t", "demo.t", "test.t");

 А вот как можно работать с Optional<>:

assertThat(statisticsMaintenance.getLastStatsResetTimestamp())
    .isPresent()
    .get()
    .satisfies(t -> assertThat(t).isAfter(testStartTime));

7. Расширяйте AssertJ для использования с вашими собственными типами

AssertJ предоставляет абстрактный базовый класс AbstractAssert<>, расширяя который, вы можете добавить поддержку своих собственных типов и методов для проверки. В некоторых случаях это позволяет заметно сократить количество тестового кода и повысить его выразительность. Пример:

assertThat(check)
    .hasType(Column.class)
    .hasDiagnostic(Diagnostic.COLUMNS_WITHOUT_DESCRIPTION)
    .hasHost(PgHostImpl.ofPrimary())
    .executing()
    .isEmpty();

8. Получайте понятные логи, если тест упал

Выше я совсем не упомянул про поддержку в IntelliJ IDEA (code completion) и про то, что AssertJ даёт весьма подробные и читаемые логи, если тест падает:

[All diagnostics must be logged] 
Actual and expected should have same size but actual size is:
  10
while expected size is:
  12
Actual was:
  ["1999-12-31T23:59:59Z	db_indexes_health	invalid_indexes	0",
...

Можно добавить описание к последующему шагу теста:

@Test
void completenessTest() {
    assertThat(logger.logAll(Exclusions.empty()))
        .as("All diagnostics must be logged")
        .hasSameSizeAs(Diagnostic.values());
}

 А ещё можно переопределить сообщение об ошибке через overridingErrorMessage(), но в большинстве случаев это не требуется.

9. Защищайте себя от неправильного использования AssertJ

Если вы ещё не поняли, то я обожаю статический анализ кода. Эта волшебная штука, если её правильно приготовить, может делать за вас огромное количество работы!

Код AssertJ активно использует аннотацию @CheckReturnValue: методы assertThat(), as(), overridingErrorMessage() и некоторые другие размечены ею.

Если вы забудете после assertThat() вызвать какой-нибудь метод проверки, то SpotBugs упадёт с ошибкой RV_RETURN_VALUE_IGNORED.

Ошибка от статического анализатораОшибка от статического анализатора

Основная хитрость здесь в том, что нужно для SpotBugs явно выставить порог предупреждений в Low.

Для Maven-плагина:


    true
    Max
    Low

Для Gradle-плагина:

spotbugs {
    effort = 'max'
    reportLevel = 'low'
}

* * *

На этом всё. Больше примеров использования AssertJ вы сможете найти в моих проектах на GitHub.

Надеюсь, эта статья поможет сделать код ваших тестов чуточку лучше.

© Habrahabr.ru