| // Copyright 2017 The Bazel Authors. All rights reserved. |
| // |
| // 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.google.devtools.build.lib.exec; |
| |
| import com.google.common.annotations.VisibleForTesting; |
| import com.google.common.collect.ImmutableList; |
| import com.google.devtools.build.lib.actions.Artifact; |
| import com.google.devtools.build.lib.analysis.test.TestResult; |
| import com.google.devtools.build.lib.analysis.test.TestRunnerAction; |
| import com.google.devtools.build.lib.buildeventstream.BuildEventConverters; |
| import com.google.devtools.build.lib.buildeventstream.BuildEventId; |
| import com.google.devtools.build.lib.buildeventstream.BuildEventStreamProtos; |
| import com.google.devtools.build.lib.buildeventstream.BuildEventWithOrderConstraint; |
| import com.google.devtools.build.lib.buildeventstream.GenericBuildEvent; |
| import com.google.devtools.build.lib.buildeventstream.PathConverter; |
| import com.google.devtools.build.lib.runtime.BuildEventStreamerUtils; |
| import com.google.devtools.build.lib.util.Pair; |
| import com.google.devtools.build.lib.vfs.Path; |
| import com.google.devtools.build.lib.view.test.TestStatus.BlazeTestStatus; |
| import com.google.devtools.build.lib.view.test.TestStatus.TestResultData; |
| import java.util.Collection; |
| import java.util.List; |
| |
| /** This event is raised whenever an individual test attempt is completed. */ |
| public class TestAttempt implements BuildEventWithOrderConstraint { |
| |
| private final TestRunnerAction testAction; |
| private final BlazeTestStatus status; |
| private final boolean cachedLocally; |
| private final int attempt; |
| private final boolean lastAttempt; |
| private final Collection<Pair<String, Path>> files; |
| private final List<String> testWarnings; |
| private final long durationMillis; |
| private final long startTimeMillis; |
| private final BuildEventStreamProtos.TestResult.ExecutionInfo executionInfo; |
| |
| /** |
| * Construct the event given the test action and attempt number. |
| * |
| * @param cachedLocally True if the reported attempt is taken from the tool's local cache. |
| * @param testAction The test that was run. |
| * @param attempt The number of the attempt for this action. |
| */ |
| public TestAttempt( |
| boolean cachedLocally, |
| TestRunnerAction testAction, |
| BuildEventStreamProtos.TestResult.ExecutionInfo executionInfo, |
| Integer attempt, |
| BlazeTestStatus status, |
| long startTimeMillis, |
| long durationMillis, |
| Collection<Pair<String, Path>> files, |
| List<String> testWarnings, |
| boolean lastAttempt) { |
| this.testAction = testAction; |
| this.executionInfo = executionInfo; |
| this.attempt = attempt; |
| this.status = status; |
| this.cachedLocally = cachedLocally; |
| this.startTimeMillis = startTimeMillis; |
| this.durationMillis = durationMillis; |
| this.files = files; |
| this.testWarnings = testWarnings; |
| this.lastAttempt = lastAttempt; |
| } |
| |
| public TestAttempt( |
| boolean cachedLocally, |
| TestRunnerAction testAction, |
| Integer attempt, |
| BlazeTestStatus status, |
| long startTimeMillis, |
| long durationMillis, |
| Collection<Pair<String, Path>> files, |
| List<String> testWarnings, |
| boolean lastAttempt) { |
| this(cachedLocally, testAction, |
| BuildEventStreamProtos.TestResult.ExecutionInfo.getDefaultInstance(), attempt, status, |
| startTimeMillis, durationMillis, files, testWarnings, lastAttempt); |
| } |
| |
| public TestAttempt( |
| TestRunnerAction testAction, |
| BuildEventStreamProtos.TestResult.ExecutionInfo executionInfo, |
| Integer attempt, |
| BlazeTestStatus status, |
| long startTimeMillis, |
| long durationMillis, |
| Collection<Pair<String, Path>> files, |
| List<String> testWarnings, |
| boolean lastAttempt) { |
| this(false, testAction, executionInfo, attempt, status, startTimeMillis, durationMillis, files, |
| testWarnings, lastAttempt); |
| } |
| |
| public TestAttempt( |
| TestRunnerAction testAction, |
| Integer attempt, |
| BlazeTestStatus status, |
| long startTimeMillis, |
| long durationMillis, |
| Collection<Pair<String, Path>> files, |
| List<String> testWarnings, |
| boolean lastAttempt) { |
| this(testAction, BuildEventStreamProtos.TestResult.ExecutionInfo.getDefaultInstance(), attempt, |
| status, startTimeMillis, durationMillis, files, testWarnings, lastAttempt); |
| } |
| |
| public static TestAttempt fromCachedTestResult(TestResult result) { |
| TestResultData data = result.getData(); |
| return new TestAttempt( |
| true, |
| result.getTestAction(), |
| result.getExecutionInfo(), |
| 1, |
| data.getStatus(), |
| data.getStartTimeMillisEpoch(), |
| data.getRunDurationMillis(), |
| result.getFiles(), |
| result.getData().getWarningList(), |
| true); |
| } |
| |
| @VisibleForTesting |
| public Artifact getTestStatusArtifact() { |
| return testAction.getCacheStatusArtifact(); |
| } |
| |
| @VisibleForTesting |
| public Collection<Pair<String, Path>> getFiles() { |
| return files; |
| } |
| |
| @Override |
| public BuildEventId getEventId() { |
| return BuildEventId.testResult( |
| testAction.getOwner().getLabel(), |
| testAction.getRunNumber(), |
| testAction.getShardNum(), |
| attempt, |
| testAction.getConfiguration().getEventId()); |
| } |
| |
| @Override |
| public Collection<BuildEventId> postedAfter() { |
| return ImmutableList.of( |
| BuildEventId.targetCompleted( |
| testAction.getOwner().getLabel(), testAction.getConfiguration().getEventId())); |
| } |
| |
| @Override |
| public Collection<BuildEventId> getChildrenEvents() { |
| if (lastAttempt) { |
| return ImmutableList.of(); |
| } else { |
| return ImmutableList.of( |
| BuildEventId.testResult( |
| testAction.getOwner().getLabel(), |
| testAction.getRunNumber(), |
| testAction.getShardNum(), |
| attempt + 1, |
| testAction.getConfiguration().getEventId())); |
| } |
| } |
| |
| @Override |
| public BuildEventStreamProtos.BuildEvent asStreamProto(BuildEventConverters converters) { |
| PathConverter pathConverter = converters.pathConverter(); |
| BuildEventStreamProtos.TestResult.Builder builder = |
| BuildEventStreamProtos.TestResult.newBuilder(); |
| builder.setStatus(BuildEventStreamerUtils.bepStatus(status)); |
| builder.setExecutionInfo(executionInfo); |
| builder.setCachedLocally(cachedLocally); |
| builder.setTestAttemptStartMillisEpoch(startTimeMillis); |
| builder.setTestAttemptDurationMillis(durationMillis); |
| builder.addAllWarning(testWarnings); |
| for (Pair<String, Path> file : files) { |
| builder.addTestActionOutput( |
| BuildEventStreamProtos.File.newBuilder() |
| .setName(file.getFirst()) |
| .setUri(pathConverter.apply(file.getSecond())) |
| .build()); |
| } |
| return GenericBuildEvent.protoChaining(this).setTestResult(builder.build()).build(); |
| } |
| } |