/*
* Copyright 2014-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.rules;
import com.facebook.buck.graph.AbstractBreadthFirstTraversal;
import com.facebook.buck.model.BuildTarget;
import com.facebook.buck.util.HumanReadableException;
import com.facebook.buck.util.MoreCollectors;
import com.google.common.base.Predicate;
import com.google.common.collect.FluentIterable;
import com.google.common.collect.ImmutableCollection;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ImmutableSortedSet;
import com.google.common.collect.Iterables;
import java.util.Optional;
public class BuildRules {
private BuildRules() {
// Utility class.
}
public static ImmutableSortedSet<BuildRule> toBuildRulesFor(
BuildTarget invokingBuildTarget,
BuildRuleResolver ruleResolver,
Iterable<BuildTarget> buildTargets) {
ImmutableSortedSet.Builder<BuildRule> buildRules = ImmutableSortedSet.naturalOrder();
for (BuildTarget target : buildTargets) {
Optional<BuildRule> buildRule = ruleResolver.getRuleOptional(target);
if (buildRule.isPresent()) {
buildRules.add(buildRule.get());
} else {
throw new HumanReadableException(
"No rule for %s found when processing %s",
target, invokingBuildTarget.getFullyQualifiedName());
}
}
return buildRules.build();
}
public static Predicate<BuildRule> isBuildRuleWithTarget(final BuildTarget target) {
return input -> input.getBuildTarget().equals(target);
}
/**
* @return the set of {@code BuildRule}s exported by {@code ExportDependencies} from the given
* rules.
*/
public static ImmutableSortedSet<BuildRule> getExportedRules(
Iterable<? extends BuildRule> rules) {
final ImmutableSortedSet.Builder<BuildRule> exportedRules = ImmutableSortedSet.naturalOrder();
AbstractBreadthFirstTraversal<ExportDependencies> visitor =
new AbstractBreadthFirstTraversal<ExportDependencies>(
Iterables.filter(rules, ExportDependencies.class)) {
@Override
public ImmutableSet<ExportDependencies> visit(ExportDependencies exporter) {
Iterable<BuildRule> exported = exporter.getExportedDeps();
exportedRules.addAll(exported);
return FluentIterable.from(exported).filter(ExportDependencies.class).toSet();
}
};
visitor.start();
return exportedRules.build();
}
public static ImmutableSet<BuildTarget> getTransitiveRuntimeDeps(
HasRuntimeDeps rule, BuildRuleResolver resolver) {
final ImmutableSet.Builder<BuildTarget> runtimeDeps = ImmutableSet.builder();
AbstractBreadthFirstTraversal<BuildTarget> visitor =
new AbstractBreadthFirstTraversal<BuildTarget>(
rule.getRuntimeDeps().collect(MoreCollectors.toImmutableSet())) {
@Override
public ImmutableCollection<BuildTarget> visit(BuildTarget runtimeDep) {
runtimeDeps.add(runtimeDep);
Optional<BuildRule> rule = resolver.getRuleOptional(runtimeDep);
if (rule.isPresent() && rule.get() instanceof HasRuntimeDeps) {
return ((HasRuntimeDeps) rule.get())
.getRuntimeDeps()
.collect(MoreCollectors.toImmutableSet());
}
return ImmutableSet.of();
}
};
visitor.start();
return runtimeDeps.build();
}
}