| /* |
| * Copyright (C) 2016 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.tradefed.testtype; |
| |
| import com.android.tradefed.build.BuildRetrievalError; |
| import com.android.tradefed.config.ConfigurationException; |
| import com.android.tradefed.config.DynamicRemoteFileResolver; |
| import com.android.tradefed.config.Option; |
| import com.android.tradefed.config.OptionSetter; |
| import com.android.tradefed.invoker.TestInformation; |
| import com.android.tradefed.invoker.tracing.CloseableTraceScope; |
| import com.android.tradefed.metrics.proto.MetricMeasurement.Metric; |
| import com.android.tradefed.result.InputStreamSource; |
| import com.android.tradefed.result.LogDataType; |
| import com.android.tradefed.testtype.MetricTestCase.LogHolder; |
| import com.android.tradefed.testtype.junit4.AfterClassWithInfo; |
| import com.android.tradefed.testtype.junit4.BeforeClassWithInfo; |
| import com.android.tradefed.testtype.junit4.CarryDnaeError; |
| import com.android.tradefed.testtype.junit4.RunAftersWithInfo; |
| import com.android.tradefed.testtype.junit4.RunBeforesWithInfo; |
| import com.android.tradefed.testtype.junit4.RunNotifierWrapper; |
| import com.android.tradefed.util.FileUtil; |
| import com.android.tradefed.util.proto.TfMetricProtoUtil; |
| |
| import com.google.common.annotations.VisibleForTesting; |
| |
| import org.junit.rules.ExternalResource; |
| import org.junit.rules.TestRule; |
| import org.junit.runner.Description; |
| import org.junit.runner.notification.RunNotifier; |
| import org.junit.runners.BlockJUnit4ClassRunner; |
| import org.junit.runners.model.FrameworkMethod; |
| import org.junit.runners.model.InitializationError; |
| import org.junit.runners.model.Statement; |
| |
| import java.io.File; |
| import java.lang.annotation.Annotation; |
| import java.util.ArrayList; |
| import java.util.HashMap; |
| import java.util.List; |
| import java.util.Map; |
| import java.util.Objects; |
| import java.util.Set; |
| import java.util.concurrent.LinkedBlockingQueue; |
| |
| /** |
| * JUnit4 test runner that also accommodates {@link IDeviceTest}. Should be specified above JUnit4 |
| * Test with a RunWith annotation. |
| */ |
| public class DeviceJUnit4ClassRunner extends BlockJUnit4ClassRunner |
| implements IAbiReceiver, ISetOptionReceiver, ITestInformationReceiver { |
| private IAbi mAbi; |
| private TestInformation mTestInformation; |
| |
| /** Keep track of the list of downloaded files. */ |
| private List<File> mDownloadedFiles = new ArrayList<>(); |
| |
| @Option(name = HostTest.SET_OPTION_NAME, description = HostTest.SET_OPTION_DESC) |
| private List<String> mKeyValueOptions = new ArrayList<>(); |
| |
| public DeviceJUnit4ClassRunner(Class<?> klass) throws InitializationError { |
| super(klass); |
| } |
| |
| /** |
| * We override createTest in order to set the device. |
| */ |
| @Override |
| protected Object createTest() throws Exception { |
| Object testObj = super.createTest(); |
| if (testObj instanceof IDeviceTest) { |
| ((IDeviceTest) testObj).setDevice(mTestInformation.getDevice()); |
| } |
| if (testObj instanceof IBuildReceiver) { |
| ((IBuildReceiver) testObj).setBuild(mTestInformation.getBuildInfo()); |
| } |
| // We are more flexible about abi information since not always available. |
| if (testObj instanceof IAbiReceiver) { |
| ((IAbiReceiver) testObj).setAbi(mAbi); |
| } |
| if (testObj instanceof IInvocationContextReceiver) { |
| ((IInvocationContextReceiver) testObj) |
| .setInvocationContext(mTestInformation.getContext()); |
| } |
| if (testObj instanceof ITestInformationReceiver) { |
| ((ITestInformationReceiver) testObj).setTestInformation(mTestInformation); |
| } |
| // Set options of test object |
| HostTest.setOptionToLoadedObject(testObj, mKeyValueOptions); |
| mDownloadedFiles.addAll(resolveRemoteFileForObject(testObj)); |
| return testObj; |
| } |
| |
| @Override |
| protected void runChild(FrameworkMethod method, RunNotifier notifier) { |
| RunNotifierWrapper wrapper = new RunNotifierWrapper(notifier); |
| try { |
| super.runChild(method, wrapper); |
| } finally { |
| for (File f : mDownloadedFiles) { |
| FileUtil.recursiveDelete(f); |
| } |
| } |
| if (wrapper.getDeviceNotAvailableException() != null) { |
| throw new CarryDnaeError(wrapper.getDeviceNotAvailableException()); |
| } |
| } |
| |
| @Override |
| protected Statement withBeforeClasses(Statement statement) { |
| Statement s = super.withBeforeClasses(statement); |
| |
| List<FrameworkMethod> beforesWithDevice = |
| getTestClass().getAnnotatedMethods(BeforeClassWithInfo.class); |
| return beforesWithDevice.isEmpty() |
| ? s |
| : new RunBeforesWithInfo(statement, beforesWithDevice, mTestInformation); |
| } |
| |
| @Override |
| protected Statement withAfterClasses(Statement statement) { |
| Statement s = super.withAfterClasses(statement); |
| |
| List<FrameworkMethod> aftersWithDevice = |
| getTestClass().getAnnotatedMethods(AfterClassWithInfo.class); |
| return aftersWithDevice.isEmpty() |
| ? s |
| : new RunAftersWithInfo(statement, aftersWithDevice, mTestInformation); |
| } |
| |
| @Override |
| public void run(RunNotifier notifier) { |
| RunNotifierWrapper wrapper = new RunNotifierWrapper(notifier); |
| super.run(wrapper); |
| |
| if (wrapper.getDeviceNotAvailableException() != null) { |
| throw new CarryDnaeError(wrapper.getDeviceNotAvailableException()); |
| } |
| } |
| |
| @Override |
| public void setAbi(IAbi abi) { |
| mAbi = abi; |
| } |
| |
| @Override |
| public IAbi getAbi() { |
| return mAbi; |
| } |
| |
| @Override |
| public void setTestInformation(TestInformation testInformation) { |
| mTestInformation = testInformation; |
| } |
| |
| @Override |
| public TestInformation getTestInformation() { |
| return mTestInformation; |
| } |
| |
| @VisibleForTesting |
| DynamicRemoteFileResolver createResolver() { |
| DynamicRemoteFileResolver resolver = new DynamicRemoteFileResolver(); |
| if (mTestInformation != null) { |
| resolver.setDevice(mTestInformation.getDevice()); |
| } |
| return resolver; |
| } |
| |
| private Set<File> resolveRemoteFileForObject(Object obj) { |
| try (CloseableTraceScope ignore = new CloseableTraceScope("junit4:resolveRemoteFiles")) { |
| OptionSetter setter = new OptionSetter(obj); |
| return setter.validateRemoteFilePath(createResolver()); |
| } catch (BuildRetrievalError | ConfigurationException e) { |
| throw new RuntimeException(e); |
| } |
| } |
| |
| /** |
| * Implementation of {@link ExternalResource} and {@link TestRule}. This rule allows to log |
| * metrics during a test case (inside @Test). It guarantees that the metrics map is cleaned |
| * between tests, so the same rule object can be re-used. |
| * |
| * <pre>Example: |
| * @Rule |
| * public TestMetrics metrics = new TestMetrics(); |
| * |
| * @Test |
| * public void testFoo() { |
| * metrics.addTestMetric("key", "value"); |
| * metrics.addTestMetric("key2", "value2"); |
| * } |
| * |
| * @Test |
| * public void testFoo2() { |
| * metrics.addTestMetric("key3", "value3"); |
| * } |
| * </pre> |
| */ |
| public static class TestMetrics extends ExternalResource { |
| |
| Description mDescription; |
| private Map<String, String> mMetrics = new HashMap<>(); |
| private HashMap<String, Metric> mProtoMetrics = new HashMap<>(); |
| |
| @Override |
| public Statement apply(Statement base, Description description) { |
| mDescription = description; |
| return super.apply(base, description); |
| } |
| |
| /** |
| * Log a metric entry for the test case. Each key within a test case must be unique |
| * otherwise it will override the previous value. |
| * |
| * @param key The key of the metric. |
| * @param value The value associated to the key. |
| */ |
| public void addTestMetric(String key, String value) { |
| mMetrics.put(key, value); |
| } |
| |
| /** |
| * Log a metric entry in proto format for the test case. Each key within a test case must be |
| * unique otherwise it will override the previous value. |
| * |
| * @param key The key of the metric. |
| * @param metric The value associated to the key. |
| */ |
| public void addTestMetric(String key, Metric metric) { |
| mProtoMetrics.put(key, metric); |
| } |
| |
| @Override |
| protected void before() throws Throwable { |
| mMetrics = new HashMap<>(); |
| mProtoMetrics = new HashMap<>(); |
| } |
| |
| @Override |
| protected void after() { |
| // we inject a Description with an annotation carrying metrics. |
| // We have to go around, since Description cannot be extended and RunNotifier |
| // does not give us a lot of flexibility to find our metrics back. |
| mProtoMetrics.putAll(TfMetricProtoUtil.upgradeConvert(mMetrics)); |
| mDescription.addChild( |
| Description.createTestDescription( |
| "METRICS", "METRICS", new MetricAnnotation(mProtoMetrics))); |
| } |
| } |
| |
| /** Fake annotation meant to carry metrics to the reporters. */ |
| public static class MetricAnnotation implements Annotation { |
| |
| public HashMap<String, Metric> mMetrics = new HashMap<>(); |
| |
| public MetricAnnotation(HashMap<String, Metric> metrics) { |
| mMetrics.putAll(metrics); |
| } |
| |
| @Override |
| public Class<? extends Annotation> annotationType() { |
| return null; |
| } |
| |
| @Override |
| public boolean equals(Object other) { |
| if (other == this) { |
| return true; |
| } |
| if (!(other instanceof MetricAnnotation)) { |
| return false; |
| } |
| MetricAnnotation o = (MetricAnnotation) other; |
| return Objects.equals(mMetrics, o.mMetrics); |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hash(mMetrics); |
| } |
| } |
| |
| /** |
| * Implementation of {@link ExternalResource} and {@link TestRule}. This rule allows to log logs |
| * during a test case (inside @Test). It guarantees that the log list is cleaned between tests, |
| * so the same rule object can be re-used. |
| * |
| * <pre>Example: |
| * @Rule |
| * public TestLogData logs = new TestLogData(); |
| * |
| * @Test |
| * public void testFoo() { |
| * logs.addTestLog("logcat", LogDataType.LOGCAT, new FileInputStreamSource(logcatFile)); |
| * } |
| * |
| * @Test |
| * public void testFoo2() { |
| * logs.addTestLog("logcat2", LogDataType.LOGCAT, new FileInputStreamSource(logcatFile2)); |
| * } |
| * </pre> |
| */ |
| public static class TestLogData extends ExternalResource { |
| private Description mDescription; |
| /** |
| * Using synchronous Queue here to mitigate possible concurrency issues in {@link |
| * com.android.tradefed.testtype.junit4.JUnit4ResultForwarder} that consumes the logs |
| */ |
| private LinkedBlockingQueue<LogHolder> mLogs = new LinkedBlockingQueue<>(); |
| |
| @Override |
| public Statement apply(Statement base, Description description) { |
| mDescription = description; |
| // we inject a Description with an annotation carrying logs. |
| // We have to go around, since Description cannot be extended and RunNotifier |
| // does not give us a lot of flexibility to find our logs back. |
| mDescription.addChild( |
| Description.createTestDescription("LOGS", "LOGS", new LogAnnotation(mLogs))); |
| return super.apply(base, description); |
| } |
| |
| public final void addTestLog( |
| String dataName, LogDataType dataType, InputStreamSource dataStream) { |
| mLogs.add(new LogHolder(dataName, dataType, dataStream)); |
| } |
| } |
| |
| /** Fake annotation meant to carry logs to the reporters. */ |
| public static class LogAnnotation implements Annotation { |
| |
| public LinkedBlockingQueue<LogHolder> mLogs; |
| |
| public LogAnnotation(LinkedBlockingQueue<LogHolder> logs) { |
| mLogs = logs; |
| } |
| |
| @Override |
| public Class<? extends Annotation> annotationType() { |
| return null; |
| } |
| |
| @Override |
| public boolean equals(Object other) { |
| if (other == this) { |
| return true; |
| } |
| if (!(other instanceof LogAnnotation)) { |
| return false; |
| } |
| TestLogData o = (TestLogData) other; |
| return Objects.equals(mLogs, o.mLogs); |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hash(mLogs); |
| } |
| } |
| } |