Statyczna analiza kodu dla języka Scala w SBT - część 1.
Statyczna analiza programu to analiza oprogramowania komputerowego wykonywanego bez faktycznego uruchamiania programów, w przeciwieństwie do analizy dynamicznej, która jest analizą wykonywaną na programach podczas ich wykonywania.
Termin ten jest zwykle stosowany do analizy wykonywanej przez zautomatyzowane narzędzie, analiza wykonywana przez człowieka jest nazywana przeglądem kodu.
Za wikipedią.
Jest to moja ulubiona część konfigurowania projektu, ponieważ odpowiednio dobrany zestaw wtyczek do statycznej analizy kodu potrafi znacząco skrócić czas potrzebny do zrobienia przeglądu kodu.
Wtyczki modyfikujące kod źródłowy
Z powodu ogromu różnego rodzaju wtyczek do statycznej analizy kodu dla języka Scala w tym poście skupię się tylko na wtyczkach modyfikujących kod źródłowy.
sbt-scalariform - sbt plugin adding support for source code formatting using Scalariform
sbt-scalariform to wtyczka sbt dodająca obsługę formatowania kodu źródłowego przy użyciu formatera kodu Scalariform
Dodajemy scalariform
do pliku projektu/plugins.sbt
:
addSbtPlugin("org.scalariform" % "sbt-scalariform" % "1.8.2")
Konfiguracja jest możliwa za pomocą:
- zmiennej
scalariformPreferences
w plikubuild.sbt
- pliku
.scalariform.conf
Formatowanie kodu jest wykonywane automatycznie podczas kompilacji:
sbt compile test:compile it:compile
Scalafmt - Code formatter for Scala
Scalafmt to formater kodu dla języka Scala.
Dodajemy scalafmt
do pliku projektu/plugins.sbt
:
addSbtPlugin("com.geirsson" % "sbt-scalafmt" % "1.5.1")
Konfiguracja jest możliwa za pomocą pliku .scalafmt.conf
, np.:
style=IntelliJ
maxColumn=80
Formatowanie domyślnie nie wykonuje się podczas kompilacji. Aby sformatować kod należy wykonać:
sbt scalafmtSbt scalafmt test:scalafmt it:scalafmt
Kolejno formatowany jest plik build.sbt
, kod produkcyjny, testy jednostkowe i testy integracyjne.
Możliwe jest także tylko sprawdzenie czy kod jest sformatowany poprawnie za pomocą polecenia:
sbt scalafmtSbtCheck scalafmtCheck test:scalafmtCheck it:scalafmtCheck
Scalafix - Refactoring and linting tool for Scala
Scalafix to narzędzie do analizy statycznej kodu w języku Scala. Jest to jedyne narzędzie, które potrafi znalezione błędy poprawić.
Dodajemy scalafix
do pliku projektu/plugins.sbt
:
addSbtPlugin("ch.epfl.scala" % "sbt-scalafix" % "0.9.0")
W pliku build.sbt
dodajemy następujące linie:
addCompilerPlugin(scalafixSemanticdb)
scalacOptions ++= Seq(
//"-Xfatal-warnings", // it should be disabled for scalafix
"-Ywarn-adapted-args", // for NoAutoTupling
"-Ywarn-unused", // for RemoveUnused
)
Konfiguracja jest możliwa za pomocą pliku .scalafix.conf
. Poniższa konfiguracja włącza wszystkie istniejące obecnie reguły:
rules = [
// Semantic Rules - Reguły semantyczne
NoAutoTupling
RemoveUnused
// Syntactic Rules - Reguły składniowe
DisableSyntax
LeakingImplicitClassVal
NoValInForComprehension
ProcedureSyntax
]
DisableSyntax.noVars = true
DisableSyntax.noThrows = true
DisableSyntax.noNulls = true
DisableSyntax.noReturns = true
DisableSyntax.noAsInstanceOf = true
DisableSyntax.noIsInstanceOf = true
DisableSyntax.noXml = true
DisableSyntax.noDefaultArgs = true
DisableSyntax.noFinalVal = true
DisableSyntax.noFinalize = true
DisableSyntax.noValPatterns = true
DisableSyntax.noUniversalEquality = true
DisableSyntax.noUniversalEqualityMessage = "== is unsafe since it allows comparing two unrelated types"
DisableSyntax.regex = []
Aby naprawić kod należy wykonać:
sbt scalafix test:scalafix it:scalafix
Kolejno naprawiany jest kod produkcyjny, testy jednostkowe i testy integracyjne.
Możliwe jest także tylko sprawdzenie czy kod nie zawiera błędów za pomocą polecenia:
sbt 'scalafix --check' 'test:scalafix --check' 'it:scalafix --check'
Podsumowanie
Wszystkie wymienione wyżej wtyczki dodałem do projektu resentiment.
Na chwilę obecna moje polecenie do zbudowania tego projektu to:
sbt clean compile test:compile it:compile re/test && \
sbt coverage reJVM/test reJS/test && \
sbt coverageReport
Wcześniej jednak powinienem wywołać polecenie refaktoryzującą i formatującą kod:
sbt scalafix test:scalafix it:scalafix && \
sbt scalafmtSbt scalafmt test:scalafmt it:scalafmt
Ewentualnie, gdy robię przegląd kodu mogę sprawdzić czy kod jest poprawny za pomocą polecenia:
sbt 'scalafix --check' 'test:scalafix --check' 'it:scalafix --check' && \
sbt scalafmtSbtCheck scalafmtCheck test:scalafmtCheck it:scalafmtCheck
Niestety ponieważ w testach używam porównania ==
musiałem zakomentować linię DisableSyntax.noUniversalEquality
w pliku .scalafix.conf
. Problem ten rozwiąże w następnym poście.