blob: 74c17e38b583a82d600e4317a67c25c9df92d4c7 [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.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}. The set may contain as direct entries {@link Artifact} or
* {@link ExpandedArtifact}.
*/
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) {
artifacts.add(
new LocalFile(
completionContext.pathResolver().toPath(expandedArtifact.artifact),
LocalFileType.OUTPUT));
} else {
artifacts.add(
new LocalFile(
completionContext.pathResolver().convertPath(expandedArtifact.target),
LocalFileType.OUTPUT));
}
}
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));
if (uri != null) {
builder.addFiles(newFileFromArtifact(expandedArtifact.artifact).setUri(uri));
}
} else {
String uri =
converters
.pathConverter()
.apply(completionContext.pathResolver().convertPath(expandedArtifact.target));
if (uri != null) {
builder.addFiles(
newFileFromArtifact(null, expandedArtifact.artifact, expandedArtifact.relPath)
.setUri(uri)
.build());
}
}
}
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);
}
}
for (NestedSet<?> succ : artifacts.getNonLeaves()) {
res.addTransitive(succ);
}
return res.build();
}
private static final class ExpandedArtifact {
public final Artifact artifact;
// These fields are used only for Fileset links.
@Nullable public final PathFragment relPath;
@Nullable public final Path target;
public ExpandedArtifact(Artifact artifact, PathFragment relPath, Path target) {
this.artifact = artifact;
this.relPath = relPath;
this.target = target;
}
}
}