/* * Copyright 2015-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. */ // Copyright 2014 Google Inc. 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.facebook.buck.query; import com.facebook.buck.query.QueryEnvironment.Argument; import com.facebook.buck.query.QueryEnvironment.ArgumentType; import com.facebook.buck.query.QueryEnvironment.QueryFunction; import com.facebook.buck.util.MoreSets; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableSet; import com.google.common.util.concurrent.ListeningExecutorService; import java.util.ArrayList; import java.util.Collection; import java.util.Set; /** * A allpaths(from, to) expression, which computes all paths between the build targets in the set * 'from' and the build targets in the set 'to', by following the dependencies between nodes in the * target graph. * * <pre>expr ::= ALLPATHS '(' expr ',' expr ')'</pre> */ public class AllPathsFunction implements QueryFunction { private static final ImmutableList<ArgumentType> ARGUMENT_TYPES = ImmutableList.of(ArgumentType.EXPRESSION, ArgumentType.EXPRESSION); public AllPathsFunction() {} @Override public String getName() { return "allpaths"; } @Override public int getMandatoryArguments() { return 2; } @Override public ImmutableList<ArgumentType> getArgumentTypes() { return ARGUMENT_TYPES; } @Override public ImmutableSet<QueryTarget> eval( QueryEnvironment env, ImmutableList<Argument> args, ListeningExecutorService executor) throws QueryException, InterruptedException { QueryExpression from = args.get(0).getExpression(); QueryExpression to = args.get(1).getExpression(); Set<QueryTarget> fromSet = from.eval(env, executor); Set<QueryTarget> toSet = to.eval(env, executor); // Algorithm: // 1) compute "reachableFromX", the forward transitive closure of the "from" set; // 2) find the intersection of "reachableFromX" with the "to" set, and traverse the graph using // the reverse dependencies. This will effectively compute the intersection between the nodes // reachable from the "from" set and the reverse transitive closure of the "to" set. env.buildTransitiveClosure(fromSet, Integer.MAX_VALUE, executor); Set<QueryTarget> reachableFromX = env.getTransitiveClosure(fromSet); Set<QueryTarget> result = MoreSets.intersection(reachableFromX, toSet); Collection<QueryTarget> worklist = result; while (!worklist.isEmpty()) { Collection<QueryTarget> reverseDeps = env.getReverseDeps(worklist); worklist = new ArrayList<>(); for (QueryTarget target : reverseDeps) { if (reachableFromX.contains(target) && result.add(target)) { worklist.add(target); } } } return ImmutableSet.copyOf(result); } }