org.specs2.reporter.JUnitXmlPrinter.scala Maven / Gradle / Ivy
package org.specs2
package reporter
import org.junit.runner.Description
import java.net.InetAddress
import main.Arguments
import execute._
import io.FileName
import control._
import Actions._
import io._
import scala.collection.JavaConverters._
import Exceptions._
import specification.core._
import specification.process._
import text.NotNullStrings._
import JUnitDescriptions._
import org.specs2.concurrent.ExecutionEnv
import origami._
/**
* The JUnitXmlPrinter creates an xml file with the specification execution results
*/
trait JUnitXmlPrinter extends Printer {
def prepare(env: Env, specs: List[SpecStructure]): Action[Unit] = Actions.unit
def finalize(env: Env, specs: List[SpecStructure]): Action[Unit] = Actions.unit
def sink(env: Env, spec: SpecStructure): AsyncSink[Fragment] =
(Statistics.fold zip fold.list[Fragment].into[Action]).
mapFlatten(saveResults(env, spec))
def saveResults(env: Env, spec: SpecStructure): ((Stats, List[Fragment])) => Action[Unit] = { case (stats, fs) =>
descriptionFold(spec, stats, env).run(descriptions(spec, fs)(env.specs2ExecutionEnv).toList).flatMap { suite =>
env.fileSystem.writeFile(outputDirectory(env.arguments) | FileName.unsafe(spec.specClassName+".xml"), suite.xml)
}
}
def descriptionFold(spec: SpecStructure, stats: Stats, env: Env): AsyncFold[(Fragment, Description), TestSuite] = {
val suite = TestSuite(specDescription(spec), spec.specClassName, stats.errors, stats.failures, stats.skipped, stats.timer.totalMillis)
fold.fromFoldLeft[Action, (Fragment, Description), TestSuite](suite) { case (res, (f, d)) =>
if (Fragment.isExample(f))
f.executedResult.map { case ExecutedResult(result, timer) =>
res.addTest(new TestCase(d, result, timer.totalMillis)(env.arguments))
}
else ok(res)
}
}
def descriptions(spec: SpecStructure, fragments: List[Fragment])(ee: ExecutionEnv) =
JUnitDescriptions.fragmentDescriptions(spec.setFragments(Fragments(fragments:_*)))(ee)
def outputDirectory(arguments: Arguments): DirectoryPath =
arguments.commandLine.directoryOr("junit.outdir", "target" / "test-reports")
case class TestSuite(description: Description, className: String, errors: Int, failures: Int, skipped: Int, time: Long = 0, tests: Seq[TestCase] = Seq()) {
def addTest(t: TestCase) = copy(tests = tests :+ t)
def xml =
s"""|
|
| $properties
| ${tests.map(_.xml).mkString("\n")}
|
|
| """.stripMargin
/**
* output properties
*/
def properties =
s"""
${System.getProperties.entrySet.asScala.toSeq.map(p => s""" """).mkString("\n")}
"""
}
case class TestCase(desc: Description, result: Result, time: Long)(implicit args: Arguments) {
def xml =
s"""
$testError$testFailure$testSkipped$testPending
"""
def testError = result match {
case er @ Error(m, e) => s"""${escape(args.traceFilter(er.stackTrace).mkString("\n"))} """
case _ => ""
}
def testFailure = result match {
case f @ Failure(m, e, st, d) => s"""${escape(args.traceFilter(st).mkString("\n"))} """
case _ => ""
}
def testPending = result match {
case Pending(m) => " "
case _ => ""
}
def testSkipped = result match {
case Skipped(m, e) => """ """
case _ => ""
}
}
private def escape(s: =>String): String =
scala.xml.Utility.escape(s.notNull)
private def formatTime(t: Long) = "%.3f" format (t / 1000.0)
}
object JUnitXmlPrinter extends JUnitXmlPrinter
© 2015 - 2025 Weber Informatics LLC | Privacy Policy