Тестирование кода и TDD
В этом уроке мы разберем концепцию тестирования нашего программного кода, научимся это делать на примере простейших библиотек, а также изучим подход в разработке через тестирование.
Тестирование
Давайте рассмотрим пример программы для подсчета факториала (Факториалом числа n называется произведение всех натуральных чисел от 1 до n включительно):
class Calculator
def factorial(max)
sum = 1
1.upto max do |num|
sum *= num
end
sum
end
endЧто будет есть мы передадим в качестве аргумента nil ? Да, мы можем протестировать нашу программу вручную, может обработать этот случай, но есть ли у нас хоть какие-то гарантии, что через несколько обновлений эта программа будет стабильно продолжать работать ? Если мы будем самостоятельно тестировать каждое обновление нашей программы, то это будет занимать колосальное кол-во времени.
Для решения этой проблемы существует тестирование нашей программы прямо внутри нашего кода. Это напоминает наш main.rb файл, в котором мы вызывали методы нашего объекта и смотрели, что получилось, но здесь мы уже задействуем специальные библиотеки, которые делают наше тестирование более прозрачным, структурированным.
Для ознакомления с этим механизмом первым делом необходимо установить библиотеку. На данный ммомент наиболее популярными являются minitest и rspec. В рамках данного урока мы познакмимся с minitest за счет ее простоты для первоначального знакомства, в дальнейшем при знакомстве с Rails мы изучим rspec.
Начнем с установки minitest через bundler. Добавьте библиотеку в Gemfile:
# Оставим nokogiri для сохранения консистентности
gem 'nokogiri', '~> 1.14'
# Добавляем minitest
gem 'minitest', '~> 5.18'После чего произведем установку с помощью bundle install. Готово, теперь мы можем начать пользоваться нашей библиотекой.
Юнит тесты тестируют не всю нашу программу, а только ее часть. В ООП парадигме блоками нашей программы являются классы, соответственно мы тестируем наши классы. Один класс - один тестовый класс. Тестовые классы следует называть точно также как и тестируемые классы с добавлением приписки Test, Допустим наш класс называется BookLibrary, значит тестовый класс будет называться BookLibraryTest. Кроме того, тестовые классы и классы с бизнес-логикой необходимо разделять между собой. Для этого можно создавать в директории продекта новый каталог tests, который будет содержать наши тетсовые классы.
Для начала давайте подумаем как все возможные сценария использования вашей программы, которые возможны:
передано целое положительное число
передано число с плавающей запятой
передан 0
передано отрицательное число
Отлично ! Теперь может приступить к тестированию. Создаем каталог, который будет содержать тесты. У меня класс Calculator лежит по пути 8_libraries/3_testing/calculator.rb, значит я создам каталог с тестами в месте 8_libraries/3_testing/tests с названием файла calculator.rb и названием класса CalculatorTest
Для того, чтобы возспользоваться библиотекой minitest необходимо добавить зависимость в файл с тестовым файлом и наследоваться от специального класса Minitest::Test следующим образом:
Далее нам неободимо создать методы, которые будут соотвествовать каждому нашему тестовому сценарию, которые мы определили ранее
Как можете заметить каждый метод имеет название тестируемого метода, а также описание тестового сценария.
Теперь нам необходимо создать наш объект, который будем тестировать и воспроизвести сценарии в каждом методе:
И в конце нам нужно определиться, какое поведение мы ожидаем от нашей программы. Наш код является безопасным в тот момент, когда мы получаем ожидаемое поведение в каждом случае. Давайте разберем все наши кейсы
Если передано положительное число, то мы ожидаем получить корректное положительное число
Число с плавающей запятой не может передоваться в факториал, поэтому самый лучший способ - ожидать ошибку.
При передаче 0 мы должны получить в качестве результата 1
Отрицательное число передавать в факториал мы тоже не может, поэтому тоже следует ожидать ошибку
Изучив наши сценарии и ожидаемый ответ, давайте исправим нашу программу.
Ранее мы изучали тему выброса ошибок. Вместо того, чтобы создавать свои собственные ошибки, можно воспользоваться уже существующими классами в Ruby. Например, в наших сценариях лкчше всего по смыслу подойдет ошибка ArgumentError, так как в обоих ошибочных сценариях передан аргумент, который мы не ожидаем получить.
Теперь когда наша программа должна отрабатывать корректно, убедимся в этом с помощью тестов.
В minitest для различных проверок существуют специальные методы Сравнения. обычно все эти методы начинаются с assert.
Самый базовый метод assert принимает в качестве первого аргумента какой-то boolean. Если он равен true, значит этот тест пройден, если false, значит тест не пройден. Вторым аргументом он принимает сообщение, которое будет показано в случае если тест не пройден. Этот аргумент опционален, поэтому передавать его необязательно. Посмотрим на примере ожидания корректного результата
В целом почти любую проверку можно выразить с помощью true/false, но minitest нам предоставляет дополнительные методы-обертки для более удобных и понятных проверок.
Например сравнение можно заменить на assert_equal, который первым аргументом принимает ожидаемое значение, а вторым реально полученное (на самом деле всегда есть + 1 аргумент для текстового описания на случай непройденного текста, просто держите это в уме)
Помимо сравнения результата наш код может выбрасывать ошибки, которые нам необходимо тоже проверять. Можно и свои костыли написать через begin...end, но будет выглядеть это, мягко говоря, не очень красиво. Для этого minitest предлагает нам специальный метод для проверки ошибки assert_raises, который принимает в качестве аргумента класс ошибки, а также блок кода, внутри которого мы должны вызывать наш метод
Помимо проверки типа ошибки, мы можем получить сам объект ошибки и проверить сообщение, которое находится внутри него.
Более подробно с различными вариантами метода assert вы сможете ознакомиться в допольнительных материалах. А пока давайте допишем наши оставшиеся тесты
TDD
В прошлых примерах мы сначала написали нашу программу, а уже потом обложили ее тестами. Существует методика, главный принцип которой - делать все наоборот. По TDD мы сначала пишем тесты, а уже после этого пишем нашу программу, чтобы она удовлетворяла условиям этих тестов. Данный способ написания программы считается наиболее безопасным, но с непривычки писать по такой методике достаточно тяжело.
Задание string_utils
Напишите класс
StringUtilsи реализуйте в нем методcamel_to_snake_case, который принимает строку в форматеcamelCaseи возвращает эту же строку в форматеsnake_caseСоставьте все возможные сценарии использования этого метода
Напишите тесты с помощью
minitestдля вашего класса
Last updated