blob: 73835a637cdbde27a263698bbbaca8d67bc08eea [file] [log] [blame]
// Copyright 2016 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.runtime;
import static com.google.devtools.build.lib.analysis.TargetCompleteEvent.newFileFromArtifact;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.actions.CompletionContext;
import com.google.devtools.build.lib.actions.CompletionContext.ArtifactReceiver;
import com.google.devtools.build.lib.actions.FileArtifactValue;
import com.google.devtools.build.lib.buildeventstream.ArtifactGroupNamer;
import com.google.devtools.build.lib.buildeventstream.BuildEvent;
import com.google.devtools.build.lib.buildeventstream.BuildEvent.LocalFile.LocalFileType;
import com.google.devtools.build.lib.buildeventstream.BuildEventContext;
import com.google.devtools.build.lib.buildeventstream.BuildEventIdUtil;
import com.google.devtools.build.lib.buildeventstream.BuildEventStreamProtos;
import com.google.devtools.build.lib.buildeventstream.BuildEventStreamProtos.BuildEventId;
import com.google.devtools.build.lib.buildeventstream.GenericBuildEvent;
import com.google.devtools.build.lib.buildeventstream.PathConverter;
import com.google.devtools.build.lib.collect.nestedset.NestedSet;
import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder;
import com.google.devtools.build.lib.collect.nestedset.Order;
import com.google.devtools.build.lib.vfs.Path;
import com.google.devtools.build.lib.vfs.PathFragment;
import java.util.Collection;
import javax.annotation.Nullable;
/**
* A {@link BuildEvent} introducing a set of artifacts to be referred to later by its name. Those
* events are generated by the {@link BuildEventStreamer} upon seeing an {@link
* com.google.devtools.build.lib.actions.EventReportingArtifacts}, if necessary.
*/
class NamedArtifactGroup implements BuildEvent {
private final String name;
private final CompletionContext completionContext;
private final NestedSet<?> set; // of Artifact or ExpandedArtifact
/**
* Create a {@link NamedArtifactGroup}. Although the set may contain a mixture of Artifacts and
* ExpandedArtifacts, all its leaf successors ("direct elements") are ExpandedArtifacts.
*/
NamedArtifactGroup(String name, CompletionContext completionContext, NestedSet<?> set) {
this.name = name;
this.completionContext = completionContext;
this.set = set;
}
@Override
public BuildEventId getEventId() {
return BuildEventIdUtil.fromArtifactGroupName(name);
}
@Override
public Collection<BuildEventId> getChildrenEvents() {
return ImmutableSet.of();
}
@Override
public Collection<LocalFile> referencedLocalFiles() {
ImmutableList.Builder<LocalFile> artifacts = ImmutableList.builder();
for (Object elem : set.getLeaves()) {
ExpandedArtifact expandedArtifact = (ExpandedArtifact) elem;
if (expandedArtifact.relPath == null) {
FileArtifactValue metadata =
completionContext.getFileArtifactValue(expandedArtifact.artifact);
artifacts.add(
new LocalFile(
completionContext.pathResolver().toPath(expandedArtifact.artifact),
LocalFileType.forArtifact(expandedArtifact.artifact, metadata),
metadata));
} else {
// TODO(b/199940216): Can fileset metadata be properly handled here?
artifacts.add(
new LocalFile(
completionContext.pathResolver().convertPath(expandedArtifact.target),
LocalFileType.OUTPUT,
/* artifactMetadata= */ null));
}
}
return artifacts.build();
}
@Override
public BuildEventStreamProtos.BuildEvent asStreamProto(BuildEventContext converters) {
PathConverter pathConverter = converters.pathConverter();
ArtifactGroupNamer namer = converters.artifactGroupNamer();
BuildEventStreamProtos.NamedSetOfFiles.Builder builder =
BuildEventStreamProtos.NamedSetOfFiles.newBuilder();
for (Object elem : set.getLeaves()) {
ExpandedArtifact expandedArtifact = (ExpandedArtifact) elem;
if (expandedArtifact.relPath == null) {
String uri =
pathConverter.apply(completionContext.pathResolver().toPath(expandedArtifact.artifact));
BuildEventStreamProtos.File file =
newFileFromArtifact(
/* name= */ null, expandedArtifact.artifact, completionContext, uri);
// Omit files with unknown contents (e.g. if uploading failed).
if (file.getFileCase() != BuildEventStreamProtos.File.FileCase.FILE_NOT_SET) {
builder.addFiles(file);
}
} else {
String uri =
pathConverter.apply(
completionContext.pathResolver().convertPath(expandedArtifact.target));
BuildEventStreamProtos.File file =
newFileFromArtifact(
/* name= */ null,
expandedArtifact.artifact,
expandedArtifact.relPath,
completionContext,
uri);
// Omit files with unknown contents (e.g. if uploading failed).
if (file.getFileCase() != BuildEventStreamProtos.File.FileCase.FILE_NOT_SET) {
builder.addFiles(file);
}
}
}
for (NestedSet<?> succ : set.getNonLeaves()) {
builder.addFileSets(namer.apply(succ.toNode()));
}
return GenericBuildEvent.protoChaining(this).setNamedSetOfFiles(builder.build()).build();
}
/**
* Given a set whose leaf successors are {@link Artifact} and {@link ExpandedArtifact}, returns a
* new NestedSet whose leaf successors are all ExpandedArtifact. Non-leaf successors are
* unaltered.
*/
static NestedSet<?> expandSet(CompletionContext ctx, NestedSet<?> artifacts) {
NestedSetBuilder<Object> res = new NestedSetBuilder<>(Order.STABLE_ORDER);
for (Object artifact : artifacts.getLeaves()) {
if (artifact instanceof ExpandedArtifact) {
res.add(artifact);
} else if (artifact instanceof Artifact) {
ctx.visitArtifacts(
ImmutableList.of((Artifact) artifact),
new ArtifactReceiver() {
@Override
public void accept(Artifact artifact) {
res.add(new ExpandedArtifact(artifact, null, null));
}
@Override
public void acceptFilesetMapping(
Artifact fileset, PathFragment relName, Path targetFile) {
res.add(new ExpandedArtifact(fileset, relName, targetFile));
}
});
} else {
throw new IllegalStateException("Unexpected type in artifact set: " + artifact);
}
}
boolean noDirects = res.isEmpty();
ImmutableList<? extends NestedSet<?>> nonLeaves = artifacts.getNonLeaves();
for (NestedSet<?> succ : nonLeaves) {
res.addTransitive(succ);
}
NestedSet<?> result = res.build();
// If we have executed one call to res.addTransitive(x)
// and none to res.add, then res.build() simply returns x
// ("inlining"), which may violate our postcondition on elements.
// (This can happen if 'artifacts' contains one non-leaf successor
// and one or more leaf successors for which visitArtifacts is a no-op.)
//
// In that case we need to recursively apply the expansion. Ugh.
if (noDirects && nonLeaves.size() == 1) {
return expandSet(ctx, result);
}
return result;
}
private static final class ExpandedArtifact {
final Artifact artifact;
// These fields are used only for Fileset links.
@Nullable final PathFragment relPath;
@Nullable final Path target;
ExpandedArtifact(Artifact artifact, PathFragment relPath, Path target) {
this.artifact = artifact;
this.relPath = relPath;
this.target = target;
}
// TODO(adonovan): define equals/hashCode. Consider:
// //foo generates bar/ (a tree artifact), with a single child, bar/baz, with this child an
// explicitly named artifact (see b/70354083). Both of these artifacts are in its "files to
// build". Then bar/ will be expanded to bar/baz, and we will have two identical (but not equal)
// ExpandedArtifact objects, leading to a duplicate emission of bar/baz in BEP.
}
}