Package com.carrotsearch.ant.tasks.junit4.listeners.antxml

Source Code of com.carrotsearch.ant.tasks.junit4.listeners.antxml.AntXmlReport

package com.carrotsearch.ant.tasks.junit4.listeners.antxml;

import java.io.File;
import java.io.IOException;
import java.io.Reader;
import java.io.StringReader;
import java.io.StringWriter;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.List;
import java.util.Map;

import org.apache.tools.ant.BuildException;
import org.apache.tools.ant.Project;
import org.apache.tools.ant.filters.TokenFilter;
import org.junit.runner.Description;
import org.simpleframework.xml.core.Persister;
import org.simpleframework.xml.transform.RegistryMatcher;

import com.carrotsearch.ant.tasks.junit4.JUnit4;
import com.carrotsearch.ant.tasks.junit4.events.aggregated.AggregatedSuiteResultEvent;
import com.carrotsearch.ant.tasks.junit4.events.aggregated.AggregatedTestResultEvent;
import com.carrotsearch.ant.tasks.junit4.events.aggregated.TestStatus;
import com.carrotsearch.ant.tasks.junit4.events.mirrors.FailureMirror;
import com.carrotsearch.ant.tasks.junit4.listeners.AggregatedEventListener;
import com.google.common.base.Strings;
import com.google.common.base.Throwables;
import com.google.common.collect.Lists;
import com.google.common.eventbus.Subscribe;
import com.google.common.io.CharStreams;
import com.google.common.io.Files;

/**
* A report listener that produces XML files compatible with those produced by
* ANT's default <code>junit</code> task. These files do not include full
* information but many tools can parse them.
*/
public class AntXmlReport implements AggregatedEventListener {
  private JUnit4 junit4;
  private File dir;
  private boolean mavenExtensions = true;
  private List<TokenFilter> filters = Lists.newArrayList();

  /**
   * @see #setOutputStreams(boolean)
   */
  private boolean outputStreams = true;
 
  /**
   * Output directory to write reports to.
   */
  public void setDir(File dir) {
    this.dir = dir;
  }
 
  /**
   * Include output streams? Mind that with large outputs the report may OOM.
   */
  public void setOutputStreams(boolean outputStreams) {
    this.outputStreams = outputStreams;
  }
 
  /**
   * Emit maven elements in the XML (extensions compared to ANT).
   */
  public void setMavenExtensions(boolean mavenExtensions) {
    this.mavenExtensions = mavenExtensions;
  }

  /**
   * Adds method name filter.
   */
  public void addConfiguredTokenFilter(TokenFilter filter) {
    this.filters.add(filter);
  }

  /*
   *
   */
  @Override
  public void setOuter(JUnit4 junit4) {
    this.junit4 = junit4;
   
    if (this.dir == null) {
      throw new BuildException("'dir' attribute is required (target folder for reports).");
    }
   
    try {
      Files.createParentDirs(dir);
      dir.mkdir();
    } catch (IOException e) {
      throw new BuildException("Could not create parent folders of: "
          + dir, e);
    }
  }

  /**
   * Emit information about all of suite's tests.
   */
  @Subscribe
  public void onSuiteResult(AggregatedSuiteResultEvent e) {
    Description suiteDescription = e.getDescription();
    String displayName = suiteDescription.getDisplayName();
    if (displayName.trim().isEmpty()) {
      junit4.log("Could not emit XML report for suite (null description).",
          Project.MSG_WARN);
      return;
    }
   
    try {
      File reportFile = new File(dir, "TEST-" + displayName + ".xml");
      RegistryMatcher rm = new RegistryMatcher();
      rm.bind(String.class, new XmlStringTransformer());
      Persister persister = new Persister(rm);
      persister.write(buildModel(e), reportFile);
    } catch (Exception x) {
      junit4.log("Could not serialize report for suite "
          + displayName + ": " + x.toString(), Project.MSG_WARN);
    }
  }

  /**
   * Build data model for serialization.
   */
  private TestSuiteModel buildModel(AggregatedSuiteResultEvent e) throws IOException {
    SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss");

    TestSuiteModel suite = new TestSuiteModel();

    suite.hostname = "nohost.nodomain";
    suite.name = e.getDescription().getDisplayName();
    suite.properties = buildModel(e.getSlave().getSystemProperties());
    suite.time = e.getExecutionTime() / 1000.0;
    suite.timestamp = df.format(new Date(e.getStartTimestamp()));

    suite.testcases = buildModel(e.getTests());
    suite.tests = suite.testcases.size();

    if (mavenExtensions) {
      suite.skipped = 0;
    }

    // Suite-level failures and errors are simulated as test cases.
    for (FailureMirror m : e.getFailures()) {
      TestCaseModel model = new TestCaseModel();
      model.classname = "junit.framework.TestSuite"; // empirical ANT output.
      model.name = applyFilters(m.getDescription().getClassName());
      model.time = 0;
      if (m.isAssertionViolation()) {
        model.failures.add(buildModel(m));
      } else {
        model.errors.add(buildModel(m));
      }
      suite.testcases.add(model);
    }

    // Calculate test numbers that match limited view (no ignored tests,
    // faked suite-level errors).
    for (TestCaseModel tc : suite.testcases) {
      suite.errors += tc.errors.size();
      suite.failures += tc.failures.size();
      if (mavenExtensions && tc.skipped != null) {
        suite.skipped += 1;
      }
    }

    StringWriter sysout = new StringWriter();
    StringWriter syserr = new StringWriter();
    if (outputStreams) {
      e.getSlave().decodeStreams(e.getEventStream(), sysout, syserr);
    }
    suite.sysout = sysout.toString();
    suite.syserr = syserr.toString();

    return suite;
  }

  /* */
  private List<TestCaseModel> buildModel(List<AggregatedTestResultEvent> testEvents) {
    List<TestCaseModel> tests = Lists.newArrayList();
    for (AggregatedTestResultEvent e : testEvents) {
      TestCaseModel model = new TestCaseModel();

      if (e.getStatus() == TestStatus.IGNORED ||
          e.getStatus() == TestStatus.IGNORED_ASSUMPTION) {
        if (mavenExtensions) {
          // This emits an empty <skipped /> element.
          model.skipped = "";
        } else {
          // No way to report these in pure ANT XML.
          continue;
        }
      }

      model.name = applyFilters(e.getDescription().getMethodName());
      model.classname = e.getDescription().getClassName();
      model.time = e.getExecutionTime() / 1000.0;

      for (FailureMirror m : e.getFailures()) {
        if (m.isAssumptionViolation()) {
          // Assumptions are not represented in ANT or Maven XMLs.
          continue;
        } else if (m.isAssertionViolation()) {
          model.failures.add(buildModel(m));
        } else {
          model.errors.add(buildModel(m));
        }
      }

      tests.add(model);
    }
    return tests;
  }

  /**
   * Apply filters to a method name.
   * @param methodName
   */
  private String applyFilters(String methodName) {
    if (filters.isEmpty()) {
      return methodName;
    }

    Reader in = new StringReader(methodName);
    for (TokenFilter tf : filters) {
      in = tf.chain(in);
    }

    try {
      return CharStreams.toString(in);
    } catch (IOException e) {
      junit4.log("Could not apply filters to " + methodName +
          ": " + Throwables.getStackTraceAsString(e), Project.MSG_WARN);
      return methodName;
    }
  }

  /* */
  private FailureModel buildModel(FailureMirror f) {
    FailureModel model = new FailureModel();
    model.message = Strings.nullToEmpty(f.getMessage());
    model.text = f.getTrace();
    model.type = f.getThrowableClass();
    return model;
  }

  /* */
  private List<PropertyModel> buildModel(Map<String,String> properties) {
    List<PropertyModel> props = Lists.newArrayList();
    for (Map.Entry<String,String> e : properties.entrySet()) {
      props.add(new PropertyModel(e.getKey(), e.getValue()));
    }
    return props;
  }
}
TOP

Related Classes of com.carrotsearch.ant.tasks.junit4.listeners.antxml.AntXmlReport

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.