org.specs2.reporter.JUnitPrinter.scala Maven / Gradle / Ivy
package org.specs2
package reporter
import junit.framework.AssertionFailedError
import text.NotNullStrings._
import org.junit.runner.Description
import org.junit.runner.notification.RunNotifier
import specification.core._
import text.AnsiColors
import execute._
import org.junit.ComparisonFailure
import main.Arguments
import control.{Actions, _}
import origami._
import control.ExecutionOrigin._
import org.specs2.fp.syntax._
/**
* The JUnitPrinter sends notifications to JUnit's RunNotifier
*/
trait JUnitPrinter extends Printer { outer =>
def prepare(env: Env, specifications: List[SpecStructure]): Action[Unit] = Actions.unit
def finalize(env: Env, specifications: List[SpecStructure]): Action[Unit] = Actions.unit
/** the junit notifier to use */
def notifier: RunNotifier
/** the descriptions of the specification fragments */
def descriptions: Map[Fragment, Description]
/** description for the whole specification */
def description: Description
// test run start and finish must not be notified if we execute the test from
// the JUnitCore runner because it is already doing that.
// Otherwise this could lead to double reporting see #440
//
// The value should be evaluated once in the outer scope, because the
// original stack trace is lost inside the callbacks.
def sink(env: Env, spec: SpecStructure): AsyncSink[Fragment] = {
val shouldNotify = !excludeFromReporting
fold.bracket[ActionStack, Fragment, RunNotifier](
open = Actions.protect { if (shouldNotify) notifier.fireTestRunStarted(description); notifier })(
step = (notifier: RunNotifier, fragment: Fragment) => notifyJUnit(env.arguments)(fragment).as(notifier))(
close = (notifier: RunNotifier) => Actions.protect(if (shouldNotify) notifier.fireTestRunFinished(new org.junit.runner.Result) else ())
)
}
def notifyJUnit(args: Arguments): Fragment => Action[Unit] = { fragment =>
if (Fragment.isExampleOrStep(fragment)) {
val description = findDescription(fragment)
fragment.executionResult.map { result =>
description.foreach { description: Description =>
if (Fragment.isExample(fragment))
notifyTestResult(description, result)(args)
else
notifyStepError(description, result)(args)
}
}
} else Actions.unit
}
private def findDescription(fragment: Fragment) = {
// find the fragment with the same description and same location
descriptions.find { case (f, d) =>
f.description == fragment.description && f.location == fragment.location
}.map(_._2)
}
private def notifyTestResult(description: Description, result: Result)(implicit args: Arguments) =
result match {
case f @ Failure(m, e, st, d) => failWith(description, junitFailure(f))
case e @ Error(m, st) => failWith(description, args.traceFilter(e.exception))
case DecoratedResult(_, f @ Failure(m, e, st, d)) => failWith(description, junitFailure(f))
case DecoratedResult(_, e @ Error(m, st)) => failWith(description, args.traceFilter(e.exception))
case Pending(_) | Skipped(_, _) => notifier.fireTestIgnored(description)
case Success(_, _) | DecoratedResult(_, _) => successWith(description)
}
private def notifyStepError(description: Description, result: Result)(implicit args: Arguments) =
result match {
case f @ Failure(m, e, st, d) => specFailWith(description, junitFailure(f))
case e @ Error(m, st) => specFailWith(description, args.traceFilter(e.exception))
case DecoratedResult(_, f @ Failure(m, e, st, d)) => specFailWith(description, junitFailure(f))
case DecoratedResult(_, e @ Error(m, st)) => specFailWith(description, args.traceFilter(e.exception))
case _ => ()
}
private def failWith(description: Description, failure: Throwable) = {
notifier.fireTestStarted(description)
notifier.fireTestFailure(new org.junit.runner.notification.Failure(description, failure))
notifier.fireTestFinished(description)
}
private def successWith(description: Description) = {
notifier.fireTestStarted(description)
notifier.fireTestFinished(description)
}
private def specFailWith(description: Description, failure: Throwable) =
notifier.fireTestFailure(new org.junit.runner.notification.Failure(description, failure))
/** @return a Throwable expected by JUnit Failure object */
private def junitFailure(f: Failure)(implicit args: Arguments): Throwable = f match {
case Failure(m, e, st, NoDetails) =>
new SpecFailureAssertionFailedError(Throwablex.exception(AnsiColors.removeColors(m), args.traceFilter(st)))
case Failure(m, e, st, FromNotImplementedError) =>
new SpecFailureAssertionFailedError(Throwablex.exception(AnsiColors.removeColors(m), args.traceFilter(st)))
case Failure(m, e, st, FromJUnitAssertionError) =>
new SpecFailureAssertionFailedError(Throwablex.exception(AnsiColors.removeColors(m), args.traceFilter(st)))
case Failure(m, e, st, FromExpectationError) =>
new SpecFailureAssertionFailedError(Throwablex.exception(AnsiColors.removeColors(m), args.traceFilter(st)))
case Failure(m, e, st, FailureDetails(actual, expected)) => new ComparisonFailure(AnsiColors.removeColors(m), expected, actual) {
private val e = args.traceFilter(f.exception)
override def getStackTrace = e.getStackTrace
override def getCause = e.getCause
override def printStackTrace(): Unit = { e.printStackTrace() }
override def printStackTrace(w: java.io.PrintStream): Unit = { e.printStackTrace(w) }
override def printStackTrace(w: java.io.PrintWriter): Unit = { e.printStackTrace(w) }
}
case Failure(m, e, st, FailureSeqDetails(actual, expected)) =>
val details =
if (args.diffs.showSeq(actual, expected, ordered = true)) {
val (added, missing) = args.diffs.showSeqDiffs(actual, expected, ordered = true)
List(showValues("Added", added), showValues("Missing", missing)).mkString(" / ")
} else ""
new ComparisonFailure(AnsiColors.removeColors(m+details), expected.mkString("\n"), actual.mkString("\n")) {
private val e = args.traceFilter(f.exception)
override def getStackTrace = e.getStackTrace
override def getCause = e.getCause
override def printStackTrace(): Unit = { e.printStackTrace() }
override def printStackTrace(w: java.io.PrintStream): Unit = { e.printStackTrace(w) }
override def printStackTrace(w: java.io.PrintWriter): Unit = { e.printStackTrace(w) }
}
case Failure(m, e, st, details @ FailureSetDetails(actual, expected)) =>
val details =
if (args.diffs.showSeq(actual.toSeq, expected.toSeq, ordered = false)) {
val (added, missing) = args.diffs.showSeqDiffs(actual.toSeq, expected.toSeq, ordered = false)
List(showValues("Added", added), showValues("Missing", missing)).mkString(" / ")
} else ""
new ComparisonFailure(AnsiColors.removeColors(m+details), expected.mkString("\n"), actual.mkString("\n")) {
private val e = args.traceFilter(f.exception)
override def getStackTrace = e.getStackTrace
override def getCause = e.getCause
override def printStackTrace(): Unit = { e.printStackTrace() }
override def printStackTrace(w: java.io.PrintStream): Unit = { e.printStackTrace(w) }
override def printStackTrace(w: java.io.PrintWriter): Unit = { e.printStackTrace(w) }
}
case Failure(m, e, st, details @ FailureMapDetails(actual, expected)) =>
val details =
if (args.diffs.showMap(actual, expected)) {
val (added, missing, different) = args.diffs.showMapDiffs(actual, expected)
List(showValues("Added", added), showValues("Missing", missing), showValues("Different", different)).mkString(" / ")
} else ""
new ComparisonFailure(AnsiColors.removeColors(m+details), expected.mkString("\n"), actual.mkString("\n")) {
private val e = args.traceFilter(f.exception)
override def getStackTrace = e.getStackTrace
override def getCause = e.getCause
override def printStackTrace(): Unit = { e.printStackTrace() }
override def printStackTrace(w: java.io.PrintStream): Unit = { e.printStackTrace(w) }
override def printStackTrace(w: java.io.PrintWriter): Unit = { e.printStackTrace(w) }
}
}
/** show values as a string with a description */
def showValues(description: String, values: Seq[Any]): String =
if (values.nonEmpty) s"$description ${values.map(notNullPair).mkString("\n", "\n", "\n\n")}" else ""
}
/**
* This class refines the `AssertionFailedError` from junit
* and provides the stackTrace of an exception which occurred during the specification execution
*/
class SpecFailureAssertionFailedError(e: Exception) extends AssertionFailedError(e.getMessage.notNull) {
override def toString = e.toString
override def getStackTrace = e.getStackTrace
override def getCause = e.getCause
override def printStackTrace(): Unit = { e.printStackTrace() }
override def printStackTrace(w: java.io.PrintStream): Unit = { e.printStackTrace(w) }
override def printStackTrace(w: java.io.PrintWriter): Unit = { e.printStackTrace(w) }
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy