blob: f8aa29b63d2807affb082f00980ee21fb26b5b01 [file] [log] [blame]
/*
* Copyright 2012-present Facebook, Inc.
*
* 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.facebook.buck.cli;
import com.facebook.buck.graph.Dot;
import com.facebook.buck.java.HasClasspathEntries;
import com.facebook.buck.json.BuildFileParseException;
import com.facebook.buck.model.BuildTarget;
import com.facebook.buck.model.BuildTargetException;
import com.facebook.buck.parser.BuildTargetPatternParser;
import com.facebook.buck.parser.ParserConfig;
import com.facebook.buck.rules.ActionGraph;
import com.facebook.buck.rules.BuildRule;
import com.facebook.buck.rules.TargetGraph;
import com.facebook.buck.rules.TargetGraphToActionGraph;
import com.facebook.buck.rules.TargetGraphTransformer;
import com.facebook.buck.rules.TargetNode;
import com.facebook.buck.util.HumanReadableException;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Function;
import com.google.common.base.Functions;
import com.google.common.base.Preconditions;
import com.google.common.collect.FluentIterable;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.LinkedHashMultimap;
import com.google.common.collect.Multimap;
import com.google.common.collect.Sets;
import java.io.IOException;
import java.nio.file.Path;
import java.util.SortedSet;
import javax.annotation.Nullable;
public class AuditClasspathCommand extends AbstractCommandRunner<AuditCommandOptions> {
private final TargetGraphTransformer<ActionGraph> targetGraphTransformer;
public AuditClasspathCommand(CommandRunnerParams params) {
super(params);
this.targetGraphTransformer = new TargetGraphToActionGraph(
params.getBuckEventBus(),
new BuildTargetNodeToBuildRuleTransformer());
}
@Override
AuditCommandOptions createOptions(BuckConfig buckConfig) {
return new AuditCommandOptions(buckConfig);
}
@Override
int runCommandWithOptionsInternal(AuditCommandOptions options)
throws IOException, InterruptedException {
// Create a TargetGraph that is composed of the transitive closure of all of the dependent
// BuildRules for the specified BuildTargets.
final ImmutableSet<BuildTarget> targets = FluentIterable
.from(options.getArgumentsFormattedAsBuildTargets())
.transform(new Function<String, BuildTarget>() {
@Override
public BuildTarget apply(String input) {
return getParser().getBuildTargetParser().parse(
input,
BuildTargetPatternParser.fullyQualified(
getParser().getBuildTargetParser()));
}
})
.toSet();
if (targets.isEmpty()) {
console.printBuildFailure("Please specify at least one build target.");
return 1;
}
TargetGraph targetGraph;
try {
targetGraph = getParser().buildTargetGraphForBuildTargets(
targets,
new ParserConfig(options.getBuckConfig()),
getBuckEventBus(),
console,
environment,
options.getEnableProfiling());
} catch (BuildTargetException | BuildFileParseException e) {
console.printBuildFailureWithoutStacktrace(e);
return 1;
}
if (options.shouldGenerateDotOutput()) {
return printDotOutput(targetGraph);
} else if (options.shouldGenerateJsonOutput()) {
return printJsonClasspath(targetGraph, targets);
} else {
return printClasspath(targetGraph, targets);
}
}
@VisibleForTesting
int printDotOutput(TargetGraph targetGraph) {
Dot<TargetNode<?>> dot = new Dot<>(
targetGraph,
"target_graph",
new Function<TargetNode<?>, String>() {
@Override
public String apply(TargetNode<?> targetNode) {
return "\"" + targetNode.getBuildTarget().getFullyQualifiedName() + "\"";
}
},
getStdOut());
try {
dot.writeOutput();
} catch (IOException e) {
return 1;
}
return 0;
}
@VisibleForTesting
int printClasspath(TargetGraph targetGraph, ImmutableSet<BuildTarget> targets) {
ActionGraph graph = targetGraphTransformer.apply(targetGraph);
SortedSet<Path> classpathEntries = Sets.newTreeSet();
for (BuildTarget target : targets) {
BuildRule rule = Preconditions.checkNotNull(graph.findBuildRuleByTarget(target));
HasClasspathEntries hasClasspathEntries = getHasClasspathEntriesFrom(rule);
if (hasClasspathEntries != null) {
classpathEntries.addAll(hasClasspathEntries.getTransitiveClasspathEntries().values());
} else {
throw new HumanReadableException(rule.getFullyQualifiedName() + " is not a java-based" +
" build target");
}
}
for (Path path : classpathEntries) {
getStdOut().println(path);
}
return 0;
}
@VisibleForTesting
int printJsonClasspath(TargetGraph targetGraph, ImmutableSet<BuildTarget> targets)
throws IOException {
ActionGraph graph = targetGraphTransformer.apply(targetGraph);
Multimap<String, String> targetClasspaths = LinkedHashMultimap.create();
for (BuildTarget target : targets) {
BuildRule rule = Preconditions.checkNotNull(graph.findBuildRuleByTarget(target));
HasClasspathEntries hasClasspathEntries = getHasClasspathEntriesFrom(rule);
if (hasClasspathEntries == null) {
continue;
}
targetClasspaths.putAll(
target.getFullyQualifiedName(),
Iterables.transform(
hasClasspathEntries.getTransitiveClasspathEntries().values(),
Functions.toStringFunction()));
}
// Note: using `asMap` here ensures that the keys are sorted
getObjectMapper().writeValue(console.getStdOut(), targetClasspaths.asMap());
return 0;
}
@Nullable
private HasClasspathEntries getHasClasspathEntriesFrom(BuildRule rule) {
if (rule instanceof HasClasspathEntries) {
return (HasClasspathEntries) rule;
}
return null;
}
@Override
String getUsageIntro() {
return "provides facilities to audit build targets' classpaths";
}
}