blob: 6e7ee51103d2fe0f4d8c76568f1fc4f61034b1c6 [file] [log] [blame]
// Copyright 2020 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.skyframe;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.devtools.build.lib.buildeventstream.BuildEventIdUtil.configurationId;
import com.google.common.collect.ImmutableMap;
import com.google.devtools.build.lib.actions.CompletionContext;
import com.google.devtools.build.lib.actions.CompletionContext.PathResolverFactory;
import com.google.devtools.build.lib.analysis.AspectCompleteEvent;
import com.google.devtools.build.lib.analysis.AspectValue;
import com.google.devtools.build.lib.analysis.ConfiguredTargetValue;
import com.google.devtools.build.lib.analysis.TopLevelArtifactHelper.ArtifactsInOutputGroup;
import com.google.devtools.build.lib.analysis.TopLevelArtifactHelper.ArtifactsToBuild;
import com.google.devtools.build.lib.bugreport.BugReporter;
import com.google.devtools.build.lib.buildeventstream.BuildEventStreamProtos.BuildEventId;
import com.google.devtools.build.lib.causes.Cause;
import com.google.devtools.build.lib.causes.LabelCause;
import com.google.devtools.build.lib.collect.nestedset.NestedSet;
import com.google.devtools.build.lib.events.Event;
import com.google.devtools.build.lib.events.ExtendedEventHandler;
import com.google.devtools.build.lib.skyframe.AspectCompletionValue.AspectCompletionKey;
import com.google.devtools.build.lib.skyframe.AspectKeyCreator.AspectKey;
import com.google.devtools.build.lib.skyframe.CompletionFunction.Completor;
import com.google.devtools.build.skyframe.SkyFunction;
import com.google.devtools.build.skyframe.SkyFunction.Environment;
import javax.annotation.Nullable;
/** Manages completing builds for aspects. */
class AspectCompletor
implements Completor<AspectValue, AspectCompletionValue, AspectCompletionKey, BuildEventId> {
static SkyFunction aspectCompletionFunction(
PathResolverFactory pathResolverFactory,
SkyframeActionExecutor skyframeActionExecutor,
MetadataConsumerForMetrics.FilesMetricConsumer topLevelArtifactsMetric,
BugReporter bugReporter) {
return new CompletionFunction<>(
pathResolverFactory,
new AspectCompletor(),
skyframeActionExecutor,
topLevelArtifactsMetric,
bugReporter);
}
@Override
public Event getRootCauseError(
AspectValue value, AspectCompletionKey key, LabelCause rootCause, Environment env)
throws InterruptedException {
AspectKey aspectKey = key.actionLookupKey();
// Skyframe lookups here should not have large effect on the number of dependency edges as
// they are only needed for failed top-level aspects.
ConfiguredTargetValue baseTargetValue =
(ConfiguredTargetValue) env.getValue(aspectKey.getBaseConfiguredTargetKey());
checkNotNull(baseTargetValue, "Base configured target value should be ready!");
ConfiguredTargetAndData configuredTargetAndData =
ConfiguredTargetAndData.fromConfiguredTargetInSkyframe(
baseTargetValue.getConfiguredTarget(), env);
checkNotNull(configuredTargetAndData, "Configured target and data should be ready!");
return Event.error(
configuredTargetAndData.getLocation(),
String.format(
"%s, aspect %s: %s",
aspectKey.getLabel(), aspectKey.getAspectClass().getName(), rootCause.getMessage()));
}
@Override
public String getLocationIdentifier(AspectValue value, AspectCompletionKey key, Environment env) {
AspectKey aspectKey = key.actionLookupKey();
return aspectKey.getLabel() + ", aspect " + aspectKey.getAspectClass().getName();
}
@Override
public AspectCompletionValue getResult() {
return AspectCompletionValue.INSTANCE;
}
@Override
public BuildEventId getFailureData(AspectCompletionKey key, AspectValue value, Environment env) {
// TODO(b/261521010): this isn't used anymore, and exists only for consistency with the
// interface. See if there's a way to clean it up.
return configurationId(key.actionLookupKey().getConfigurationKey());
}
@Override
public ExtendedEventHandler.Postable createFailed(
AspectCompletionKey skyKey,
NestedSet<Cause> rootCauses,
CompletionContext ctx,
ImmutableMap<String, ArtifactsInOutputGroup> outputs,
BuildEventId configurationEventId) {
return AspectCompleteEvent.createFailed(skyKey.actionLookupKey(), ctx, rootCauses, outputs);
}
@Nullable
@Override
public AspectCompleteEvent createSucceeded(
AspectCompletionKey skyKey,
AspectValue value,
CompletionContext completionContext,
ArtifactsToBuild artifactsToBuild,
Environment env)
throws InterruptedException {
return AspectCompleteEvent.createSuccessful(
skyKey.actionLookupKey(),
completionContext,
artifactsToBuild.getAllArtifactsByOutputGroup());
}
}