org.scalatest.tools.DashboardReporter.scala Maven / Gradle / Ivy
/*
* Copyright 2001-2013 Artima, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.scalatest.tools
import org.scalatest.events._
import org.scalatest.Reporter
import org.scalatest.events.MotionToSuppress
import org.scalatest.exceptions.StackDepthException
import java.io.PrintWriter
import java.io.BufferedOutputStream
import java.io.FileOutputStream
import java.io.File
import java.util.Date
import java.util.TimeZone
import java.text.SimpleDateFormat
import java.util.regex.Matcher.quoteReplacement
import scala.collection.mutable
import scala.collection.mutable.Stack
import scala.collection.mutable.ListBuffer
import scala.xml.XML
import scala.xml.NodeSeq
import scala.xml.Elem
import scala.xml.Node
/**
* A Reporter
that writes test status information in xml format
* for use by Flex formatter.
*/
private[scalatest] class DashboardReporter(directory: String,
numOldFilesToKeep: Int)
extends Reporter
{
final val BufferSize = 4096
private val events = ListBuffer[Event]()
private var index = 0
private var timestamp = "" // gets set at RunStarting event
private final val TimestampPattern = """\d{4}-\d{2}-\d{2}-\d{6}-\d{3}"""
private val runsDir = new File(directory + "/runs")
private val durationsDir = new File(directory + "/durations")
private val summariesDir = new File(directory + "/summaries")
private val summaryFile = new File(directory + "/summary.xml")
private val durationsFile = new File(directory + "/durations.xml")
runsDir.mkdir()
durationsDir.mkdir()
summariesDir.mkdir()
//
// Records events as they are received. Initiates processing once
// a run-termination event comes in.
//
// Ignores info and markup events.
//
def apply(event: Event) {
event match {
case _: DiscoveryStarting =>
case _: DiscoveryCompleted =>
case _: RunStarting => timestamp = formatCurrentTime
case _: InfoProvided =>
case _: AlertProvided =>
case _: NoteProvided =>
case _: ScopeOpened =>
case _: ScopeClosed =>
case _: ScopePending =>
case _: MarkupProvided =>
case _: RunCompleted => writeFiles(event)
case _: RunStopped => writeFiles(event)
case _: RunAborted => writeFiles(event)
case _ => events += event
}
}
//
// Formats current time for use as timestamp to identify current run.
//
// Uses GMT time zone to avoid sequencing errors that could occur with
// time shifts due to daylight savings time or laptop travel if local
// time zone were used.
//
def formatCurrentTime: String = {
val df = new SimpleDateFormat("yyyy-MM-dd-HHmmss-SSS")
df.setTimeZone(TimeZone.getTimeZone("GMT"))
df.format(new Date)
}
//
// Provides sequential index values for xml entries.
//
def nextIndex(): Int = {
index += 1
index
}
//
// Throws exception for specified unexpected event.
//
def unexpectedEvent(e: Event) {
throw new RuntimeException("unexpected event [" + e + "]")
}
//
// Escapes html entities and curly braces in specified string.
//
def escape(s: String): String =
scala.xml.Utility.escape(s).
replaceAll("""\{""", """\\{""").
replaceAll("""\}""", """\\}""")
//
// Formats date for inclusion in as 'date' attribute in xml.
//
// E.g.: "Mon May 30 10:29:58 PDT 2011"
//
def formatDate(timeStamp: Long): String = {
val df = new SimpleDateFormat("EEE MMM d kk:mm:ss zzz yyyy")
df.format(new Date(timeStamp))
}
//
// Reads existing summary.xml file, or, if none exists, returns a
// xml containing all empty elements.
//
def getOldSummaryXml: Elem = {
if (summaryFile.exists)
XML.loadFile(summaryFile)
else
}
//
// If a summary file containing previous run histories exists, moves
// it to the summaries/ subdirectory and renames it to a filename
// containing the timestamp of the most recent run the file contains.
//
// Ditto for the durations.xml file.
//
def archiveOldFiles(oldRunsXml: NodeSeq) {
val previousRunTimestamp =
if (oldRunsXml.size > 0) Some("" + oldRunsXml(0) \ "@id")
else None
if (previousRunTimestamp.isDefined) {
if (summaryFile.exists)
summaryFile.renameTo(
new File(
summariesDir + "/summary-" + previousRunTimestamp.get + ".xml"))
if (durationsFile.exists)
durationsFile.renameTo(
new File(
durationsDir + "/duration-" + previousRunTimestamp.get + ".xml"))
purgeDir(summariesDir, "summary-")
purgeDir(durationsDir, "duration-")
}
}
//
// Deletes older files from specified archive directory.
//
// We keep timestamped old copies of summary.xml and durations.xml in
// summaries/ and durations/ subdirectories. This method trims the
// oldest archived files from those directories to maintain a specified
// maximum number of archived copies.
//
def purgeDir(directory: File, prefix: String) {
directory.
listFiles().
filter(_.getName.matches(prefix + TimestampPattern + """\.xml""")).
sorted.
dropRight(numOldFilesToKeep).
foreach(_.delete())
}
//
// Writes dashboard reporter summary, duration, and run files at completion
// of a run. Archives old copies of summary and duration files into
// summaries/ and durations/ subdirectories.
//
def writeFiles(terminatingEvent: Event) {
val durations = Durations(durationsFile)
val oldSummaryXml = getOldSummaryXml
val oldRunsXml = oldSummaryXml \\ "run"
archiveOldFiles(oldRunsXml)
val thisRunFile = new File(runsDir, "run-" + timestamp + ".xml")
writeRunFile(terminatingEvent, thisRunFile)
val thisRunXml = XML.loadFile(thisRunFile)
durations.addTests(timestamp, thisRunXml)
writeDurationsFile(durations)
writeSummaryFile(
terminatingEvent, oldSummaryXml, oldRunsXml, thisRunXml, durations)
}
//
// Writes the durations.xml file.
//
def writeDurationsFile(durations: Durations) {
writeFile("durations.xml", durations.toXml)
}
//
// Writes the summary.xml file.
//
def writeSummaryFile(terminatingEvent: Event, oldSummaryXml: NodeSeq,
oldRunsXml: NodeSeq, thisRunXml: NodeSeq,
durations: Durations)
{
val SummaryTemplate =
"""|
|
|$runs$
|
|$regressions$
|
|$recentlySlower$
|
|""".stripMargin
//
// Formats a element of summary file.
//
def formatRun(id: String, succeeded: String, failed: String,
ignored: String, canceled: String, pending: String): String =
{
" \n"
}
//
// Generates the summary file element for the current run.
//
def genThisRun(terminatingEvent: Event): String = {
val summaryOption =
terminatingEvent match {
case e: RunCompleted => e.summary
case e: RunAborted => e.summary
case e: RunStopped => e.summary
case _ => unexpectedEvent(terminatingEvent); None
}
val summary = summaryOption.getOrElse(Summary(0, 0, 0, 0, 0, 0, 0, 0))
formatRun(timestamp,
"" + summary.testsSucceededCount,
"" + summary.testsFailedCount,
"" + summary.testsIgnoredCount,
"" + summary.testsCanceledCount,
"" + summary.testsPendingCount)
}
//
// Formats elements for previous runs.
//
// (We could let scala do this its way, but it makes kind of a mess.)
//
def formatOldRuns(oldRunsXml: NodeSeq): String = {
val buf = new StringBuilder
for (run <- oldRunsXml) {
val id = "" + (run \ "@id")
val succeeded = "" + (run \ "@succeeded")
val failed = "" + (run \ "@failed")
val ignored = "" + (run \ "@ignored")
val canceled = "" + (run \ "@canceled")
val pending = "" + (run \ "@pending")
buf.append(
formatRun(id, succeeded, failed, ignored, canceled, pending))
}
buf.toString
}
//
// Generates elements for output summary file.
//
def genRegressions(oldSummaryXml: NodeSeq, thisRunXml: NodeSeq): String =
{
//
// Searches through regressions from previous summary to try and
// find one matching specified test.
//
def getOldRegression(suite: Node, test: Node,
oldRegressionsXml: NodeSeq): Option[Node] =
{
oldRegressionsXml.find(
node => (((node \ "@testName") == (test \ "@name")) &&
((node \ "@suiteId") == (suite \ "@id"))))
}
//
// Formats a element.
//
def formatRegression(suite: Node, test: Node, result: String,
lastSucceeded: String, firstRegressed: String):
String =
{
" \n"
}
//
// Gets the timestamp of the previous run.
//
def getLastRunId: Option[String] = {
val previousRuns = oldSummaryXml \\ "run"
if (previousRuns.size > 0) Some("" + (previousRuns(0) \ "@id"))
else None
}
//
// Retrieves xml of the previous run if available, else Empty.
//
def getLastRunXml(lastRunId: Option[String]): NodeSeq = {
if (lastRunId.isDefined)
XML.loadFile(directory + "/runs/run-" + lastRunId.get + ".xml")
else
NodeSeq.Empty
}
//
// Checks xml from previous run to see if it contains a test with
// success status that matches specified test.
//
def lastRunSucceeded(suite: Node, test: Node, lastRunXml: NodeSeq):
Boolean =
{
var found = false
var succeeded = false
val oldSuitesIt = (lastRunXml \\ "suite").iterator
while (!found && oldSuitesIt.hasNext) {
val oldSuite = oldSuitesIt.next()
if (oldSuite \ "@id" == suite \ "@id") {
val oldTests = oldSuite \ "test"
val matchingTest =
oldTests.find(node => (node \ "@name") == test \ "@name")
if (matchingTest.isDefined) {
found = true
val result = "" + matchingTest.get \ "@result"
succeeded = (result == "succeeded")
}
}
}
succeeded
}
//
// genRegressions main
//
val buf = new StringBuilder
val suites = thisRunXml \\ "suite"
val oldRegressionsXml = oldSummaryXml \\ "regressedTest"
val lastRunId = getLastRunId
val lastRunXml = getLastRunXml(lastRunId)
for (suite <- suites) {
for (test <- suite \ "test") {
val result = "" + (test \ "@result")
if (result != "succeeded") {
val oldRegression = getOldRegression(suite, test,
oldRegressionsXml)
val lastSucceeded =
if (oldRegression.isDefined)
"" + oldRegression.get \ "@lastSucceeded"
else if (lastRunSucceeded(suite, test, lastRunXml))
lastRunId.get
else
"never"
val firstRegressed =
if (oldRegression.isDefined)
"" + oldRegression.get \ "@firstRegressed"
else
timestamp
if (!((result == "pending") && (lastSucceeded == "never")))
buf.append(
formatRegression(
suite, test, result, lastSucceeded, firstRegressed))
}
}
}
buf.toString
}
def genRecentlySlower(durations: Durations): String = {
var slowRecords = List[SlowRecord]()
case class SlowRecord(suite: Durations#Suite, test: Durations#Test,
oldAvg: Int, newAvg: Int, percentSlower: Int)
{
def toXml: String = {
val SlowerTestTemplate =
"""
|""".stripMargin
SlowerTestTemplate.
replaceFirst("""\$suiteId\$""", quoteReplacement(suite.suiteId)).
replaceFirst("""\$suiteName\$""", quoteReplacement(suite.suiteName)).
replaceFirst("""\$testName\$""", quoteReplacement(test.name)).
replaceFirst("""\$oldAvg\$""", "" + oldAvg).
replaceFirst("""\$newAvg\$""", "" + newAvg)
}
}
def toXml: String = {
val buf = new StringBuilder
for (slowRecord <- slowRecords) buf.append(slowRecord.toXml)
buf.toString
}
for (suite <- durations.suites) {
for (test <- suite.tests) {
if (test.runCount > 10) {
val oldAvg = test.previousAverage
val newAvg = test.computeNewAvg
if ((newAvg - oldAvg > 1) && (oldAvg > 0)) {
val percentSlower =
(((newAvg - oldAvg).toDouble / oldAvg.toDouble) * 100).toInt
if (percentSlower > 10)
slowRecords ::=
SlowRecord(suite, test, oldAvg, newAvg, percentSlower)
}
}
}
}
slowRecords =
slowRecords.sortBy(r => r.percentSlower).take(20)
toXml
}
//
// writeSummaryFile main
//
val thisRun = genThisRun(terminatingEvent)
val oldRuns = formatOldRuns(oldRunsXml)
val regressions = genRegressions(oldSummaryXml, thisRunXml)
val recentlySlower = genRecentlySlower(durations)
val summaryText =
SummaryTemplate.
replaceFirst("""\$runs\$""", quoteReplacement(thisRun + oldRuns)).
replaceFirst("""\$regressions\$""", quoteReplacement(regressions)).
replaceFirst("""\$recentlySlower\$""",
quoteReplacement(recentlySlower))
writeFile("summary.xml", summaryText)
}
//
// Writes specified text to specified file in output directory.
//
def writeFile(filename: String, text: String) {
val out = new PrintWriter(directory + "/" + filename)
out.print(text)
out.close()
}
//
// Writes timestamped output file to 'runs' subdirectory beneath specified
// output dir. Format of file name is, e.g. for timestamp
// "2011-10-24-105759-563", "run-2011-10-24-105759-563.xml".
//
// We write the file piece-by-piece directly, instead of creating a string
// and writing that,
//
def writeRunFile(event: Event, thisRunFile: File) {
index = 0
var suiteRecord: SuiteRecord = null
val stack = new Stack[SuiteRecord]
val pw =
new PrintWriter(
new BufferedOutputStream(
new FileOutputStream(thisRunFile), BufferSize))
//
// Formats element of output xml.
//
def formatSummary(event: Event): String = {
val (summaryOption, durationOption) =
event match {
case e: RunCompleted => (e.summary, e.duration)
case e: RunAborted => (e.summary, e.duration)
case e: RunStopped => (e.summary, e.duration)
case _ => unexpectedEvent(event); (None, None)
}
val summary = summaryOption.getOrElse(Summary(0, 0, 0, 0, 0, 0, 0, 0))
val duration = durationOption.getOrElse(0)
" \n"
}
//
// Closes out a SuiteRecord. Gets called upon receipt of a
// SuiteCompleted or SuiteAborted event.
//
// If the suite being closed is nested within another suite, its
// completed record is added to the record of the suite it is nested
// in. Otherwise its xml is written to the output file.
//
def endSuite(e: Event) {
suiteRecord.addEndEvent(e)
val prevRecord = stack.pop()
if (prevRecord != null)
prevRecord.addNestedElement(suiteRecord)
else
pw.print(suiteRecord.toXml)
suiteRecord = prevRecord
}
//
// writeRunFile main
//
pw.println("")
pw.print(formatSummary(event))
for (event <- events.sorted) {
event match {
case e: SuiteStarting =>
stack.push(suiteRecord)
suiteRecord = new SuiteRecord(e)
case e: TestStarting => suiteRecord.addNestedElement(e)
case e: TestSucceeded => suiteRecord.addNestedElement(e)
case e: TestIgnored => suiteRecord.addNestedElement(e)
case e: TestFailed => suiteRecord.addNestedElement(e)
case e: TestPending => suiteRecord.addNestedElement(e)
case e: TestCanceled => suiteRecord.addNestedElement(e)
case e: SuiteCompleted => endSuite(e)
case e: SuiteAborted => endSuite(e)
case e: DiscoveryStarting => unexpectedEvent(e)
case e: DiscoveryCompleted => unexpectedEvent(e)
case e: RunStarting => unexpectedEvent(e)
case e: RunCompleted => unexpectedEvent(e)
case e: RunStopped => unexpectedEvent(e)
case e: RunAborted => unexpectedEvent(e)
case e: InfoProvided => unexpectedEvent(e)
case e: AlertProvided => unexpectedEvent(e)
case e: NoteProvided => unexpectedEvent(e)
case e: ScopeOpened => unexpectedEvent(e)
case e: ScopeClosed => unexpectedEvent(e)
case e: ScopePending => unexpectedEvent(e)
case e: MarkupProvided => unexpectedEvent(e)
}
}
pw.println(" ")
pw.flush()
pw.close()
}
//
// Generates xml for a TestIgnored event.
//
def formatTestIgnored(event: TestIgnored): String = {
" \n"
}
//
// Extracts message from specified formatter if there is one, otherwise
// returns test name.
//
def testMessage(testName: String, formatter: Option[Formatter]): String = {
val message =
formatter match {
case Some(IndentedText(_, rawText, _)) => rawText
case _ => testName
}
escape(message)
}
//
// Class that aggregates events that make up a suite.
//
// Holds all the events encountered from SuiteStarting through its
// corresponding end event (e.g. SuiteCompleted). Once the end event
// is received, this class's toXml method can be called to generate the
// complete xml string for the element.
//
class SuiteRecord(startEvent: SuiteStarting) {
var nestedElements = List[Any]()
var endEvent: Event = null
//
// Adds either an Event or a nested SuiteRecord to this object's
// list of elements.
//
def addNestedElement(element: Any) {
nestedElements ::= element
}
//
// Adds suite closing event (SuiteCompleted or SuiteAborted) to the
// object.
//
def addEndEvent(event: Event) {
def isEndEvent(e: Event): Boolean = {
e match {
case _: SuiteCompleted => true
case _: SuiteAborted => true
case _ => false
}
}
require(endEvent == null)
require(isEndEvent(event))
endEvent = event
}
//
// Generates value to be used in element's 'result' attribute.
//
def result: String = {
endEvent match {
case _: SuiteCompleted => "completed"
case _: SuiteAborted => "aborted"
case _ => unexpectedEvent(endEvent); ""
}
}
//
// Generates xml string representation of SuiteRecord object.
//
def toXml: String = {
val buf = new StringBuilder
var testRecord: TestRecord = null
//
// Generates opening element
//
def formatStartOfSuite: String = {
val duration = endEvent.timeStamp - startEvent.timeStamp
"\n" +
"\n"
}
//
// Indicates whether a test record is currently open during
// event processing.
//
def inATest: Boolean =
(testRecord != null) && (testRecord.endEvent == null)
//
// toXml main
//
if (startEvent.suiteName != "DiscoverySuite")
buf.append(formatStartOfSuite)
for (element <- nestedElements.reverse) {
if (inATest) {
testRecord.addEvent(element.asInstanceOf[Event])
if (testRecord.isComplete)
buf.append(testRecord.toXml)
}
else {
element match {
case e: TestIgnored => buf.append(formatTestIgnored(e))
case e: SuiteRecord => buf.append(e.toXml)
case e: TestStarting => testRecord = new TestRecord(e)
case _ =>
throw new RuntimeException("unexpected [" + element + "]")
}
}
}
if (startEvent.suiteName != "DiscoverySuite")
buf.append(" \n")
buf.toString
}
}
//
// Class that aggregates events that make up a test.
//
// Holds all the events encountered from TestStarting through its
// corresponding end event (e.g. TestSucceeded). Once the end event
// is received, this class's toXml method can be called to generate
// the complete xml string for the element.
//
// (We no longer record info-provided or markup events for tests,
// so nested events within the TestRecord have been removed.)
//
class TestRecord(startEvent: TestStarting) {
var endEvent: Event = null
//
// Adds specified event to object's list of nested events.
//
def addEvent(event: Event) {
def isEndEvent: Boolean = {
event match {
case _: TestSucceeded => true
case _: TestFailed => true
case _: TestPending => true
case _: TestCanceled => true
case _ => false
}
}
if (isEndEvent)
endEvent = event
else
unexpectedEvent(event)
}
//
// Indicates whether an end event has been received yet for this
// record.
//
def isComplete: Boolean = (endEvent != null)
//
// Generates value for use as 'result' attribute of element.
//
def result: String = {
endEvent match {
case _: TestSucceeded => "succeeded"
case _: TestFailed => "failed"
case _: TestPending => "pending"
case _: TestCanceled => "canceled"
case _ => unexpectedEvent(endEvent); ""
}
}
object Duration {
def unapply(event: Event): Option[Long] =
event match {
case TestSucceeded(_, _, _, _, _, _, _, duration, _, _, _, _, _, _)
=> duration
case TestFailed(_, _, _, _, _, _, _, _, _, duration, _, _, _, _, _, _)
=> duration
case TestPending(_, _, _, _, _, _, _, duration, _, _, _, _, _)
=> duration
case TestCanceled(_, _, _, _, _, _, _, _, _, duration, _, _, _, _, _, _)
=> duration
case _ => None
}
}
//
// Generates initial element of object's xml.
//
def formatTestStart: String = {
val duration =
endEvent match {
case Duration(d) => d
case _ => endEvent.timeStamp - startEvent.timeStamp
}
"\n"
}
//
// Generates xml for a test failure.
//
def formatException(event: TestFailed): String = {
val buf = new StringBuilder
var depth = -1
def nextDepth: Int = {
depth += 1
depth
}
buf.append("\n")
if (event.throwable.isDefined) {
val throwable = event.throwable.get
val stackTrace = throwable.getStackTrace
require(stackTrace.size > 0)
// PCData will enclose the message in CDATA.
buf.append("" + scala.xml.PCData(event.message) + " \n")
if (throwable.isInstanceOf[StackDepthException]) {
val sde = throwable.asInstanceOf[StackDepthException]
if (sde.failedCodeFileName.isDefined &&
sde.failedCodeLineNumber.isDefined)
{
buf.append(
"\n" +
"" + sde.failedCodeStackDepth + " \n" +
"" + sde.failedCodeFileName.get + " \n" +
"" +
sde.failedCodeLineNumber.get +
" \n" +
" \n")
}
}
buf.append("\n")
for (frame <- stackTrace) {
buf.append(
"" +
frame.getClassName + "(" + frame.getFileName + ":" +
frame.getLineNumber + ")" +
" \n")
}
buf.append(" \n")
}
buf.append(" \n")
buf.toString
}
//
// Generates xml string representation of object.
//
def toXml: String = {
val buf = new StringBuilder
if (endEvent == null)
throw new IllegalStateException("toXml called without endEvent")
buf.append(formatTestStart)
if (endEvent.isInstanceOf[TestFailed])
buf.append(formatException(endEvent.asInstanceOf[TestFailed]))
buf.append(" \n")
buf.toString
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy