-->

Полный гайд по тестированию на Flutter. Часть 2: Простые модульные тесты

Полный гайд по тестированию на Flutter. Часть 2: Простые модульные тесты

Hola, Amigos! На связи Павел Гершевич, Mobile Team Lead агентства разработки сайтов и мобильных приложений Amiga. Продолжаем нашу серию статей переводов о тестировании в Flutter. В этой и нескольких следующих частях поговорим о модульном (Unit) тестировании.

Простой Unit-тест

Представим, что нам нужно протестировать функцию входа в приложение с функциями валидации email и пароля.

Используем две статичные функции.

class Validator {

static bool validateEmail(String value) {

return value.isNotEmpty;

}


static bool validatePassword(String value) {

return value.isNotEmpty;

}

}


Затем метод расширения под названием isNullOrEmpty.

extension StringExtension on String? {

bool get isNullOrEmpty => this == null || this!.isEmpty;

}

И наконец, функция входа.

import 'package:testing_examples/part2/ext/extension.dart';

import 'package:testing_examples/part2/util/utils.dart';


bool login(String? email, String? password) {

if (email.isNullOrEmpty || password.isNullOrEmpty) {

return false;

}

return Validator.validateEmail(email!) && Validator.validatePassword(password!);

}

Полный исходный код можно найти по ссылке: https://github.com/ntminhdn/testing_examples/tree/main/lib/part2

Необходимо протестировать 4 функции, которые находятся в 3 разных файлах, поэтому создаем 3 файла для тестов в папке test. Чтобы различать файлы unit-тестов и файлы, относящиеся к другим методам тестирования, таким как Widget-тесты, назовем папку unit_test внутри папки test.

Naming convention для файлов с тестами гласит, что для названия файла нужно использовать код и суффикс _test.dart. И еще одно правило — структура папки test должна повторять структуру папки lib, как показано в примере:

Сначала напишем тесты для функции validateEmail. Каждый файл должен начинаться с функции main() как точки входа. Для написания Unit-тестов нужно импортировать пакет flutter_test.


import 'package:flutter_test/flutter_test.dart';


void main() {

}


Для того, чтобы создать Unit-тест, используем функцию test, передавая ей 2 параметра: description и body.


void main() {

test('validateEmail should return true when the email is not empty', () {

// body

});

}

Не важно короткие или длинные наименования самих тестов. Главное, чтобы они были простыми для понимания без чтения кода.

Для тела теста обычно используется паттерн ААА: сначала, подготавливаем все необходимое (Arrange), потом выполняем нужное действие (Act) и проверяем его результат (Assert).

test('validateEmail should return true when the email is not empty', () {

// Arrange

String validEmail = 'test@example.com';


// Act

bool result = Validator.validateEmail(validEmail);


// Assert

expect(result, true);

});

  • Arrange — шаг, где создаем переменные и входные данные перед вызовом функции, которую хотим протестировать.

    Например, если нужно проверить функцию validateEmail когда email не пустой, то нужно создать переменную String validEmail = ‘test@example.com’.

  • Act — вызов функции, которую нужно протестировать с уже подготовленными на прошлом шаге входными данными: Validator.validateEmail(validEmail).

  • Assert — шаг, где проверяем соответствует ли результат ожиданиям, используя функцию expect.

    Например: expect(result, true), expect(result, 1000), expect(result, “Minh”), …

Итак, один Unit-тест описан. Теперь функция validateEmail должна быть протестирована еще для одного случая: когда email пустой, она должна вернуть false.

test('validateEmail should return false when the email is empty', () {

// Arrange

String invalidEmail = '';


// Act

bool result = Validator.validateEmail(invalidEmail);


// Assert

expect(result, false);

});

Напишем Unit-тесты для функции validatePassword в похожем виде.

test('validatePassword should return true when the password is not empty', () {

// Arrange

String validPassword = 'password123';


// Act

bool result = Validator.validatePassword(validPassword);


// Assert

expect(result, true);

});


test('validatePassword should return false when the password is empty', () {

// Arrange

String invalidPassword = '';


// Act

bool result = Validator.validatePassword(invalidPassword);


// Assert

expect(result, false);

});

Теперь в файле utils_test.dart есть 4 тестовых кейса. Нужно сгруппировать их по самим функциям, используя group.

group('validateEmail', () {

test('validateEmail should return true when the email is not empty', () {

// body

});


test('validateEmail should return false when the email is empty', () {

// body

});

});


group('validatePassword', () {

test('validatePassword should return true when the password is not empty',

// body

});


test('validatePassword should return false when the password is empty', () {

// body

});

});

Для того, чтобы запустить Unit-тесты, набираем в консоли команду flutter test или нажимаем Run или Debug в IDE, как на картинке ниже. Если в консоль выводится фраза «All tests passed!», то все тесты успешно прошли. Если какой-то из них провалится, то появится лог в консоли.

group('login', () {

test('login should return false when the email is empty', () {

// Arrange

String? email;

String password = 'password123';


// Act

bool result = login(email, password);


// Assert

expect(result, false);

});


test('login should return false when the password is empty', () {

// Arrange

String email = 'ntminh@gmail.vn';

String? password;


// Act

bool result = login(email, password);


// Assert

expect(result, false);

});


test('login should return false when the email and password are empty', () {

// Arrange

String? email;

String? password;


// Act

bool result = login(email, password);


// Assert

expect(result, false);

});


test('login should return true when the email and password are not empty',

() {

// Arrange

String email = 'ntminh@gmail.vn';

String password = 'password123';


// Act

bool result = login(email, password);


// Assert

expect(result, true);

});

});

Не стоит волноваться, если кто-то случайно удалит код, который сейчас тестируется. Во время рефакторинга тесты покажут ошибку!

Функция expect и Matcher

Функция expect используется для проверки соответствия результата условию (matcher).

expect(actual, matcher);

Matcher может быть булевым значением, например true, false, строковым значением, например «OK», или числом: 0, -1 и т. д. Также он может быть комплексным выражением, таким как:

  • isNull: используется для проверки, что текущее значение равно null.

  • isNotNull: используется для проверки, что текущее значение не равно null.

  • isTrue: одинаково со сравнением с true.

  • isFalse: одинаково со сравнением с false.

  • isList: используется для проверки, что текущее значение это List.

  • isMap: используется для проверки, что текущее значение это Map.

  • isA<T>(): используется для проверки, что текущее значение имеет тип T.

  • isException: используется для проверки, что текущее значение это Exception.

  • throwsArgumentError: используется для проверки, что во время выполнения кидает ArgumentError.

Мы познакомимся с большим количеством Matcher в следующих частях.

Заключение

В данной статье, мы написали простой Unit-тест. В следующих выпусках продолжим писать Unit-тесты для более сложных случаев с использованием продвинутых техник: Mock, Fake и Stub.

Подписывайтесь на телеграмм-канале Flutter. Много, чтобы не пропустить следующую статью!

Хотите связаться с владельцами компании напрямую?
Константин Франгуриди
Константин Франгуриди
Account director

НАПИСАТЬ

Дмитрий Тарасов
Дмитрий Тарасов
СЕО

НАПИСАТЬ