Dynamiczna analiza kodu dla SBT - testy jednostkowe
Dynamiczna analiza programu to analiza oprogramowania komputerowego wykonywanego przez wykonywanie programów na rzeczywistym lub wirtualnym procesorze. Korzystanie z metryk testów, takich jak pokrycie kodu, zapewnia, że przetestowano odpowiednią ilość możliwych zachować programu. Aby analiza dynamiczna programu była skuteczna, program docelowy musi być wykonany z wystarczającą ilością danych wejściowych do testów, aby uzyskać interesujące zachowanie. Testy jednostkowe, testy integracyjne, testy systemowe i testy akceptacyjne wykorzystują dynamiczną analizę programu.
Za wikipedią.
W tym poście skupię się tylko na frameworkach do testów, testach jednostkowych i mierzeniu pokrycia kodu testami.
Frameworki dla testów
- ScalaTest - jest to chyba najbardziej znany i rozbudowany framework dla testów do języka Scala. Wspiera Scala.js w wersji 0.6.x. Ostatnia wersja snapshot wspiera Scala Native. Posiada osiem różnych stylów pisania testów, jednak autorzy zachęcają do wybrania dwóch dla projektu. Jeden styl dla testów jednostkowo-integracyjnych, drugi - dla akceptacyjnych. Najciekawsze style to:
FunSuite
iFlatSpec
- są to proste, płaskie style pisania testów podobne do JUnit. Jeśli jednak są dla kogoś zbyt awangardowe jest możliwość pisania testów w Scali z użyciem JUnit i assercji z ScalaTestFunSpec
,FreeSpec
iWordSpec
- są to style testów umożliwiające pisanie zagnieżdżonych testów, jednak zagnieżdżenia nie są wymagane (za wyjątkiem styluFunSpec
, gdzie trzeba użyć przynajmniej jeden poziomDescribe
)FeatureSpec
- jest to zaawansowany styl dla pisania testów akceptacyjnych.
- specs2 - jest to drugi najbardziej znany framework do pisania testów dla języka Scala. Od wersji czwartej wspiera Scala.js w wersji 0.6.x. Ma dwa style pisania testów:
org.specs2.mutable.Specificatio
- najbardziej przypominaFreeSpec
i jest przewidziany do pisania testów jednostkowych i integracyjnych.org.specs2.Specification
- styl testów przewidziany do pisania testów akceptacyjnych,
- uTest - czarny koń tego zestawienia. Wspiera Scala.js w wersji 0.6.x i 1.0.x oraz Scala Native. Autor frameworka skromnie chwali się, że wspiera wszystkie wersje języka Scala. Ma jeden styl pisania testów który wygląda jak
FreeTest
. - MiniTest - stworzony przez Monix do testowania swojej biblioteki. Wspiera Scala.js w wersji 0.6.x. Ma jeden styl pisania testów, który wygląda jak
FunSuite
. - Greenlight - projekt niestety umarł. Wspiera Scala.js w wersji 0.6.x. Ma jeden styl pisania testów, który wygląda jak
FlatSpec
Test jednostkowy w uTest
W build.sbt
dodajemy uTest
do zależności projektu:
libraryDependencies += "com.lihaoyi" %%% "utest" % "0.6.5" % "test"
i ustawiamy jako framework testowy:
testFrameworks += new TestFramework("utest.runner.Framework"),
Tworzymy klasę Calculator
, którą będziemy testować :
package pl.writeonly.re.shared
class Calculator {
type T = Int
def add(a: T, b: T): T = a * b
def mul(a: T, b: T): T = a + b
def leq(a: T, b: T): Boolean = a < b
}
Oraz testy dla niej:
package pl.writeonly.re.shared
import utest._
object CalculatorTest extends TestSuite {
override val tests: Tests = Tests {
val calculator = new Calculator()
'addition - {
val addition: (Int, Int) => Int = (x, y) => calculator.add(x, y)
"0 + 0 == 0" - {
assert(addition(0, 0) == 0)
}
"2 + 2 == 4" - {
assert(addition(2, 2) == 4)
}
}
'multiplication - {
val multiplication: (Int, Int) => Int = (x, y) => calculator.mul(x, y)
"0 + 0 == 0" - {
assert(multiplication(0, 0) == 0)
}
"2 + 2 == 4" - {
assert(multiplication(2, 2) == 4)
}
}
'less_or_equal - {
val less_or_equal: (Int, Int) => Boolean = (x, y) => calculator.leq(x, y)
"0 <= 2 == true" - {
assert(less_or_equal(0, 2))
}
"2 <= 0 == false" - {
assert(!less_or_equal(2, 0))
}
}
}
}
Wszystko kompilujemy i uruchamiamy testy za pomocą polecenia:
sbt clean compile test
Testy przeszły, jesteśmy szczęśliwi:
[info] -------------------------------- Running Tests --------------------------------
[info] + pl.writeonly.re.shared.CalculatorTest.addition.0 + 0 == 0 0ms
[info] + pl.writeonly.re.shared.CalculatorTest.addition.2 + 2 == 4 0ms
[info] + pl.writeonly.re.shared.CalculatorTest.multiplication.0 + 0 == 0 0ms
[info] + pl.writeonly.re.shared.CalculatorTest.multiplication.2 + 2 == 4 0ms
[info] + pl.writeonly.re.shared.CalculatorTest.less_or_equal.0 <= 2 == true 0ms
[info] + pl.writeonly.re.shared.CalculatorTest.less_or_equal.2 <= 0 == false 0ms
[info] Tests: 6, Passed: 6, Failed: 0
Testowanie testów - mierzenie pokrycia kodu testami
Skąd mamy mieć pewność, że przetestowaliśmy klasę Calculator
w wystarczający sposób? Możemy to częściowo sprawdzić mierząc pokrycie kodu produkcyjnego (tj. klasy Calculator
) testami.
Jeśli chodzi o narzędzia do mierzenia pokrycia kodu testami to tutaj król jest jeden i jest nim scoverage. Posiada on wtyczki do:
Przy czym użyjemy tutaj tylko pierwszej z nich.
Dodajemy sbt-scoverage
do build.sbt
:
addSbtPlugin("org.scoverage" % "sbt-scoverage" % "1.5.1")
I wykonujemy:
sbt clean coverage test && sbt coverageReport
gdzie:
coverage
- wykonuje kompilacje z instrumentacją koducoverageReport
- generuje raport
Niestety powyższe polecenie działa tylko dla implementacji Scala/JVM i Scala.js. Scala Native nie wspiera instrumentacji kodu. Dlatego projekt resentiment trzeba kompilować za pomocą polecenia:
sbt clean compile re/test coverage reJVM/test reJS/test && sbt coverageReport
Teraz możemy otworzyć pliki <folder_projektu>/re/js/target/scala-2.11/scoverage-report/index.html
oraz <folder_projektu>/re/jvm/target/scala-2.11/scoverage-report/index.html
i zobaczyć, że klasa Calculator
ma 100% pokrycia kodu testami. Lider nietechniczny i Product Owner powinni być z nas zadowoleni.