| /* |
| * Copyright (C) 2013 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.builder.internal.testing; |
| |
| import com.android.annotations.NonNull; |
| import com.android.annotations.Nullable; |
| import com.android.ddmlib.testrunner.TestIdentifier; |
| import com.android.ddmlib.testrunner.TestResult; |
| import com.android.ddmlib.testrunner.XmlTestRunListener; |
| import com.android.utils.ILogger; |
| import com.google.common.collect.ImmutableMap; |
| import com.google.common.collect.Maps; |
| import com.google.common.collect.Sets; |
| |
| import java.io.File; |
| import java.io.IOException; |
| import java.util.Map; |
| import java.util.Set; |
| |
| /** |
| * Custom version of {@link com.android.ddmlib.testrunner.ITestRunListener}. |
| */ |
| public class CustomTestRunListener extends XmlTestRunListener { |
| |
| @NonNull |
| private final String mDeviceName; |
| @NonNull |
| private final String mProjectName; |
| @NonNull |
| private final String mFlavorName; |
| private final ILogger mLogger; |
| private final Set<TestIdentifier> mFailedTests = Sets.newHashSet(); |
| |
| |
| public CustomTestRunListener(@NonNull String deviceName, |
| @NonNull String projectName, @NonNull String flavorName, |
| @Nullable ILogger logger) { |
| mDeviceName = deviceName; |
| mProjectName = projectName; |
| mFlavorName = flavorName; |
| mLogger = logger; |
| } |
| |
| @Override |
| protected File getResultFile(File reportDir) throws IOException { |
| return new File(reportDir, |
| "TEST-" + mDeviceName + "-" + mProjectName + "-" + mFlavorName + ".xml"); |
| } |
| |
| @Override |
| protected String getTestSuiteName() { |
| // in order for the gradle report to look good we put the test suite name as one of the |
| // test class name. |
| |
| Map<TestIdentifier, TestResult> testResults = getRunResult().getTestResults(); |
| if (testResults.isEmpty()) { |
| return null; |
| } |
| |
| Map.Entry<TestIdentifier, TestResult> testEntry = testResults.entrySet().iterator().next(); |
| return testEntry.getKey().getClassName(); |
| } |
| |
| @Override |
| protected Map<String, String> getPropertiesAttributes() { |
| Map<String, String> propertiesAttributes = Maps.newLinkedHashMap(super.getPropertiesAttributes()); |
| propertiesAttributes.put("device", mDeviceName); |
| propertiesAttributes.put("flavor", mFlavorName); |
| propertiesAttributes.put("project", mProjectName); |
| return ImmutableMap.copyOf(propertiesAttributes); |
| } |
| |
| @Override |
| public void testRunStarted(String runName, int testCount) { |
| if (mLogger != null) { |
| mLogger.info("Starting %1$d tests on %2$s", testCount, mDeviceName); |
| } |
| super.testRunStarted(runName, testCount); |
| } |
| |
| @Override |
| public void testFailed(TestIdentifier test, String trace) { |
| if (mLogger != null) { |
| mLogger.warning("\n%1$s > %2$s[%3$s] \033[31mFAILED \033[0m", |
| test.getClassName(), test.getTestName(), mDeviceName); |
| mLogger.warning(getModifiedTrace(trace)); |
| } |
| |
| mFailedTests.add(test); |
| |
| super.testFailed(test, trace); |
| } |
| |
| @Override |
| public void testAssumptionFailure(TestIdentifier test, String trace) { |
| if (mLogger != null) { |
| mLogger.warning("\n%1$s > %2$s[%3$s] \033[33mSKIPPED \033[0m\n%4$s", |
| test.getClassName(), test.getTestName(), mDeviceName, getModifiedTrace(trace)); |
| } |
| super.testAssumptionFailure(test, trace); |
| } |
| |
| @Override |
| public void testEnded(TestIdentifier test, Map<String, String> testMetrics) { |
| if (!mFailedTests.remove(test)) { |
| // if wasn't present in the list, then the test succeeded. |
| if (mLogger != null) { |
| mLogger.info("\n%1$s > %2$s[%3$s] \033[32mSUCCESS \033[0m", |
| test.getClassName(), test.getTestName(), mDeviceName); |
| } |
| |
| } |
| super.testEnded(test, testMetrics); |
| } |
| |
| @Override |
| public void testRunFailed(String errorMessage) { |
| if (mLogger != null) { |
| mLogger.warning("Tests on %1$s failed: %2$s", mDeviceName, errorMessage); |
| } |
| super.testRunFailed(errorMessage); |
| } |
| |
| @Override |
| public void testIgnored(TestIdentifier test) { |
| if (mLogger != null) { |
| mLogger.warning("\n%1$s > %2$s[%3$s] \033[33mSKIPPED \033[0m", |
| test.getClassName(), test.getTestName(), mDeviceName); |
| } |
| super.testIgnored(test); |
| } |
| |
| private String getModifiedTrace(String trace) { |
| // split lines |
| String[] lines = trace.split("\n"); |
| |
| if (lines.length < 2) { |
| return trace; |
| } |
| |
| // get the first two lines, and prepend \t on them |
| return "\t" + lines[0] + "\n\t" + lines[1]; |
| } |
| } |