/* * Copyright (C) 2012 The Android Open Source Project * * 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 com.android.ddmlib.testrunner; import junit.framework.TestCase; import org.xml.sax.InputSource; import java.io.ByteArrayOutputStream; import java.io.File; import java.io.IOException; import java.io.OutputStream; import java.io.StringReader; import java.util.Collections; import java.util.Map; import javax.xml.xpath.XPath; import javax.xml.xpath.XPathExpressionException; import javax.xml.xpath.XPathFactory; /** * Unit tests for {@link XmlTestRunListener}. */ public class XmlTestRunListenerTest extends TestCase { private XmlTestRunListener mResultReporter; private ByteArrayOutputStream mOutputStream; private File mReportDir; /** * {@inheritDoc} */ @Override protected void setUp() throws Exception { super.setUp(); mOutputStream = new ByteArrayOutputStream(); mResultReporter = new XmlTestRunListener() { @Override OutputStream createOutputResultStream(File reportDir) throws IOException { return mOutputStream; } @Override String getTimestamp() { return "ignore"; } }; // TODO: use mock file dir instead mReportDir = createTmpDir(); mResultReporter.setReportDir(mReportDir); } private File createTmpDir() throws IOException { // create a temp file with unique name, then make it a directory File tmpDir = File.createTempFile("foo", "dir"); tmpDir.delete(); if (!tmpDir.mkdirs()) { throw new IOException("unable to create directory"); } return tmpDir; } /** * Recursively delete given file and all its contents */ private static void recursiveDelete(File rootDir) { if (rootDir.isDirectory()) { File[] childFiles = rootDir.listFiles(); if (childFiles != null) { for (File child : childFiles) { recursiveDelete(child); } } } rootDir.delete(); } @Override protected void tearDown() throws Exception { if (mReportDir != null) { recursiveDelete(mReportDir); } super.tearDown(); } /** * A simple test to ensure expected output is generated for test run with no tests. */ public void testEmptyGeneration() { final String expectedOutput = "<?xml version='1.0' encoding='UTF-8' ?>" + "<testsuite name=\"test\" tests=\"0\" failures=\"0\" errors=\"0\" skipped=\"0\" " + "time=\"#TIMEVALUE#\" " + "timestamp=\"ignore\" hostname=\"localhost\"> " + "<properties />" + "</testsuite>"; mResultReporter.testRunStarted("test", 1); mResultReporter.testRunEnded(1, Collections.<String, String> emptyMap()); // because the timestamp is impossible to hardcode, look for the actual timestamp and // replace it in the expected string. String output = getOutput(); String time = getTime(output); assertNotNull(time); String expectedTimedOutput = expectedOutput.replaceFirst("#TIMEVALUE#", time); assertEquals(expectedTimedOutput, output); } /** * A simple test to ensure expected output is generated for test run with a single passed test. */ public void testSinglePass() { Map<String, String> emptyMap = Collections.emptyMap(); final TestIdentifier testId = new TestIdentifier("FooTest", "testFoo"); mResultReporter.testRunStarted("run", 1); mResultReporter.testStarted(testId); mResultReporter.testEnded(testId, emptyMap); mResultReporter.testRunEnded(3, emptyMap); String output = getOutput(); // TODO: consider doing xml based compare assertTrue(output.contains("tests=\"1\" failures=\"0\" errors=\"0\"")); final String testCaseTag = String.format("<testcase name=\"%s\" classname=\"%s\"", testId.getTestName(), testId.getClassName()); assertTrue(output.contains(testCaseTag)); } /** * A simple test to ensure expected output is generated for test run with a single failed test. */ public void testSingleFail() { Map<String, String> emptyMap = Collections.emptyMap(); final TestIdentifier testId = new TestIdentifier("FooTest", "testFoo"); final String trace = "this is a trace"; mResultReporter.testRunStarted("run", 1); mResultReporter.testStarted(testId); mResultReporter.testFailed(testId, trace); mResultReporter.testEnded(testId, emptyMap); mResultReporter.testRunEnded(3, emptyMap); String output = getOutput(); // TODO: consider doing xml based compare assertTrue(output.contains("tests=\"1\" failures=\"1\" errors=\"0\"")); final String testCaseTag = String.format("<testcase name=\"%s\" classname=\"%s\"", testId.getTestName(), testId.getClassName()); assertTrue(output.contains(testCaseTag)); final String failureTag = String.format("<failure>%s</failure>", trace); assertTrue(output.contains(failureTag)); } /** * Gets the output produced, stripping it of extraneous whitespace characters. */ private String getOutput() { String output = mOutputStream.toString(); // ignore newlines and tabs whitespace output = output.replaceAll("[\\r\\n\\t]", ""); // replace two ws chars with one return output.replaceAll(" ", " "); } /** * Returns the value if the time attribute from the given XML content * * Actual XPATH: /testsuite/@time * * @param xml XML content. * @return */ private String getTime(String xml) { XPath xpath = XPathFactory.newInstance().newXPath(); try { return xpath.evaluate("/testsuite/@time", new InputSource(new StringReader(xml))); } catch (XPathExpressionException e) { // won't happen. } return null; } }