org.gradle.execution.taskgraph.DefaultTaskExecutionPlan Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* Copyright 2012 the original author or authors.
*
* 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 org.gradle.execution.taskgraph;
import com.google.common.base.Function;
import com.google.common.base.Predicate;
import com.google.common.base.StandardSystemProperty;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.common.collect.Sets;
import org.gradle.api.Action;
import org.gradle.api.BuildCancelledException;
import org.gradle.api.CircularReferenceException;
import org.gradle.api.Project;
import org.gradle.api.Task;
import org.gradle.api.Transformer;
import org.gradle.api.UncheckedIOException;
import org.gradle.api.internal.GradleInternal;
import org.gradle.api.internal.TaskInternal;
import org.gradle.api.internal.project.ProjectInternal;
import org.gradle.api.internal.tasks.CachingTaskDependencyResolveContext;
import org.gradle.api.internal.tasks.TaskContainerInternal;
import org.gradle.api.internal.tasks.TaskDestroyablesInternal;
import org.gradle.api.internal.tasks.TaskLocalStateInternal;
import org.gradle.api.specs.Spec;
import org.gradle.api.specs.Specs;
import org.gradle.execution.MultipleBuildFailures;
import org.gradle.execution.TaskFailureHandler;
import org.gradle.initialization.BuildCancellationToken;
import org.gradle.internal.Pair;
import org.gradle.internal.UncheckedException;
import org.gradle.internal.graph.CachingDirectedGraphWalker;
import org.gradle.internal.graph.DirectedGraph;
import org.gradle.internal.graph.DirectedGraphRenderer;
import org.gradle.internal.graph.GraphNodeRenderer;
import org.gradle.internal.logging.text.StyledTextOutput;
import org.gradle.internal.resources.ResourceDeadlockException;
import org.gradle.internal.resources.ResourceLock;
import org.gradle.internal.resources.ResourceLockCoordinationService;
import org.gradle.internal.resources.ResourceLockState;
import org.gradle.internal.work.WorkerLeaseRegistry.WorkerLease;
import org.gradle.internal.work.WorkerLeaseService;
import org.gradle.util.CollectionUtils;
import org.gradle.util.Path;
import org.gradle.util.TextUtil;
import javax.annotation.Nullable;
import java.io.File;
import java.io.IOException;
import java.io.StringWriter;
import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Deque;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.TreeSet;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicReference;
import static org.gradle.internal.resources.DefaultResourceLockCoordinationService.unlock;
import static org.gradle.internal.resources.ResourceLockState.Disposition.*;
/**
* A reusable implementation of TaskExecutionPlan. The {@link #addToTaskGraph(java.util.Collection)} and {@link #clear()} methods are NOT threadsafe, and callers must synchronize access to these
* methods.
*/
public class DefaultTaskExecutionPlan implements TaskExecutionPlan {
private final Set tasksInUnknownState = new LinkedHashSet();
private final Set entryTasks = new LinkedHashSet();
private final TaskInfoFactory nodeFactory = new TaskInfoFactory();
private final LinkedHashMap executionPlan = new LinkedHashMap();
private final List executionQueue = new LinkedList();
private final Map projectLocks = Maps.newHashMap();
private final List failures = new ArrayList();
private Spec super Task> filter = Specs.satisfyAll();
private TaskFailureHandler failureHandler = new RethrowingFailureHandler();
private final BuildCancellationToken cancellationToken;
private final Set runningTasks = Sets.newIdentityHashSet();
private final Set filteredTasks = Sets.newIdentityHashSet();
private final Map taskMutations = Maps.newIdentityHashMap();
private final Map canonicalizedFileCache = Maps.newIdentityHashMap();
private final Map, Boolean> reachableCache = Maps.newHashMap();
private final Set dependenciesCompleteCache = Sets.newHashSet();
private final ResourceLockCoordinationService coordinationService;
private final WorkerLeaseService workerLeaseService;
private final GradleInternal gradle;
private boolean tasksCancelled;
public DefaultTaskExecutionPlan(BuildCancellationToken cancellationToken, ResourceLockCoordinationService coordinationService, WorkerLeaseService workerLeaseService, GradleInternal gradle) {
this.cancellationToken = cancellationToken;
this.coordinationService = coordinationService;
this.workerLeaseService = workerLeaseService;
this.gradle = gradle;
}
@Override
public String getDisplayName() {
Path path = gradle.findIdentityPath();
if (path == null) {
return "gradle";
}
return path.toString();
}
public void addToTaskGraph(Collection extends Task> tasks) {
List queue = new ArrayList();
List sortedTasks = new ArrayList(tasks);
Collections.sort(sortedTasks);
for (Task task : sortedTasks) {
TaskInfo node = nodeFactory.createNode(task);
if (node.isMustNotRun()) {
requireWithDependencies(node);
} else if (filter.isSatisfiedBy(task)) {
node.require();
}
entryTasks.add(node);
queue.add(node);
}
Set visiting = new HashSet();
CachingTaskDependencyResolveContext context = new CachingTaskDependencyResolveContext();
while (!queue.isEmpty()) {
TaskInfo node = queue.get(0);
if (node.getDependenciesProcessed()) {
// Have already visited this task - skip it
queue.remove(0);
continue;
}
TaskInternal task = node.getTask();
boolean filtered = !filter.isSatisfiedBy(task);
if (filtered) {
// Task is not required - skip it
queue.remove(0);
node.dependenciesProcessed();
node.doNotRequire();
filteredTasks.add(task);
continue;
}
if (visiting.add(node)) {
// Have not seen this task before - add its dependencies to the head of the queue and leave this
// task in the queue
// Make sure it has been configured
((TaskContainerInternal) task.getProject().getTasks()).prepareForExecution(task);
Set extends Task> dependsOnTasks = context.getDependencies(task);
for (Task dependsOnTask : dependsOnTasks) {
TaskInfo targetNode = nodeFactory.createNode(dependsOnTask);
node.addDependencySuccessor(targetNode);
if (!visiting.contains(targetNode)) {
queue.add(0, targetNode);
}
}
for (Task finalizerTask : task.getFinalizedBy().getDependencies(task)) {
TaskInfo targetNode = nodeFactory.createNode(finalizerTask);
addFinalizerNode(node, targetNode);
if (!visiting.contains(targetNode)) {
queue.add(0, targetNode);
}
}
for (Task mustRunAfter : task.getMustRunAfter().getDependencies(task)) {
TaskInfo targetNode = nodeFactory.createNode(mustRunAfter);
node.addMustSuccessor(targetNode);
}
for (Task shouldRunAfter : task.getShouldRunAfter().getDependencies(task)) {
TaskInfo targetNode = nodeFactory.createNode(shouldRunAfter);
node.addShouldSuccessor(targetNode);
}
if (node.isRequired()) {
for (TaskInfo successor : node.getDependencySuccessors()) {
if (filter.isSatisfiedBy(successor.getTask())) {
successor.require();
}
}
} else {
tasksInUnknownState.add(node);
}
} else {
// Have visited this task's dependencies - add it to the graph
queue.remove(0);
visiting.remove(node);
node.dependenciesProcessed();
}
}
resolveTasksInUnknownState();
}
private void resolveTasksInUnknownState() {
List queue = new ArrayList(tasksInUnknownState);
Set visiting = new HashSet();
while (!queue.isEmpty()) {
TaskInfo task = queue.get(0);
if (task.isInKnownState()) {
queue.remove(0);
continue;
}
if (visiting.add(task)) {
for (TaskInfo hardPredecessor : task.getDependencyPredecessors()) {
if (!visiting.contains(hardPredecessor)) {
queue.add(0, hardPredecessor);
}
}
} else {
queue.remove(0);
visiting.remove(task);
task.mustNotRun();
for (TaskInfo predecessor : task.getDependencyPredecessors()) {
assert predecessor.isRequired() || predecessor.isMustNotRun();
if (predecessor.isRequired()) {
task.require();
break;
}
}
}
}
}
private void addFinalizerNode(TaskInfo node, TaskInfo finalizerNode) {
if (filter.isSatisfiedBy(finalizerNode.getTask())) {
node.addFinalizer(finalizerNode);
if (!finalizerNode.isInKnownState()) {
finalizerNode.mustNotRun();
}
finalizerNode.addMustSuccessor(node);
}
}
private void addAllReversed(List list, TreeSet set) {
List elements = CollectionUtils.toList(set);
Collections.reverse(elements);
list.addAll(elements);
}
private void requireWithDependencies(TaskInfo taskInfo) {
if (taskInfo.isMustNotRun() && filter.isSatisfiedBy(taskInfo.getTask())) {
taskInfo.require();
for (TaskInfo dependency : taskInfo.getDependencySuccessors()) {
requireWithDependencies(dependency);
}
}
}
public void determineExecutionPlan() {
List nodeQueue = Lists.newArrayList(Iterables.transform(entryTasks, new Function() {
int index;
public TaskInfoInVisitingSegment apply(TaskInfo taskInfo) {
return new TaskInfoInVisitingSegment(taskInfo, index++);
}
}));
int visitingSegmentCounter = nodeQueue.size();
HashMultimap visitingNodes = HashMultimap.create();
Deque walkedShouldRunAfterEdges = new ArrayDeque();
Deque path = new ArrayDeque();
HashMap planBeforeVisiting = new HashMap();
while (!nodeQueue.isEmpty()) {
TaskInfoInVisitingSegment taskInfoInVisitingSegment = nodeQueue.get(0);
int currentSegment = taskInfoInVisitingSegment.visitingSegment;
TaskInfo taskNode = taskInfoInVisitingSegment.taskInfo;
if (taskNode.isIncludeInGraph() || executionPlan.containsKey(taskNode.getTask())) {
nodeQueue.remove(0);
visitingNodes.remove(taskNode, currentSegment);
maybeRemoveProcessedShouldRunAfterEdge(walkedShouldRunAfterEdges, taskNode);
continue;
}
boolean alreadyVisited = visitingNodes.containsKey(taskNode);
visitingNodes.put(taskNode, currentSegment);
if (!alreadyVisited) {
// Have not seen this task before - add its dependencies to the head of the queue and leave this
// task in the queue
recordEdgeIfArrivedViaShouldRunAfter(walkedShouldRunAfterEdges, path, taskNode);
removeShouldRunAfterSuccessorsIfTheyImposeACycle(visitingNodes, taskInfoInVisitingSegment);
takePlanSnapshotIfCanBeRestoredToCurrentTask(planBeforeVisiting, taskNode);
ArrayList successors = new ArrayList();
addAllSuccessorsInReverseOrder(taskNode, successors);
for (TaskInfo successor : successors) {
if (visitingNodes.containsEntry(successor, currentSegment)) {
if (!walkedShouldRunAfterEdges.isEmpty()) {
//remove the last walked should run after edge and restore state from before walking it
GraphEdge toBeRemoved = walkedShouldRunAfterEdges.pop();
toBeRemoved.from.removeShouldRunAfterSuccessor(toBeRemoved.to);
restorePath(path, toBeRemoved);
restoreQueue(nodeQueue, visitingNodes, toBeRemoved);
restoreExecutionPlan(planBeforeVisiting, toBeRemoved);
break;
} else {
onOrderingCycle();
}
}
nodeQueue.add(0, new TaskInfoInVisitingSegment(successor, currentSegment));
}
path.push(taskNode);
} else {
// Have visited this task's dependencies - add it to the end of the plan
nodeQueue.remove(0);
maybeRemoveProcessedShouldRunAfterEdge(walkedShouldRunAfterEdges, taskNode);
visitingNodes.remove(taskNode, currentSegment);
path.pop();
executionPlan.put(taskNode.getTask(), taskNode);
Project project = taskNode.getTask().getProject();
projectLocks.put(project, getOrCreateProjectLock(project));
TaskMutationInfo taskMutationInfo = getOrCreateMutationsOf(taskNode);
for (TaskInfo dependency : taskNode.getDependencySuccessors()) {
getOrCreateMutationsOf(dependency).consumingTasks.add(taskNode);
taskMutationInfo.consumesOutputOf.add(dependency);
}
// Add any finalizers to the queue
ArrayList finalizerTasks = new ArrayList();
addAllReversed(finalizerTasks, taskNode.getFinalizers());
for (TaskInfo finalizer : finalizerTasks) {
if (!visitingNodes.containsKey(finalizer)) {
nodeQueue.add(finalizerTaskPosition(finalizer, nodeQueue), new TaskInfoInVisitingSegment(finalizer, visitingSegmentCounter++));
}
}
}
}
executionQueue.clear();
executionQueue.addAll(executionPlan.values());
}
private TaskMutationInfo getOrCreateMutationsOf(TaskInfo taskInfo) {
TaskMutationInfo mutations = taskMutations.get(taskInfo);
if (mutations == null) {
mutations = new TaskMutationInfo(taskInfo);
taskMutations.put(taskInfo, mutations);
}
return mutations;
}
private void maybeRemoveProcessedShouldRunAfterEdge(Deque walkedShouldRunAfterEdges, TaskInfo taskNode) {
if (!walkedShouldRunAfterEdges.isEmpty() && walkedShouldRunAfterEdges.peek().to.equals(taskNode)) {
walkedShouldRunAfterEdges.pop();
}
}
private void restoreExecutionPlan(HashMap planBeforeVisiting, GraphEdge toBeRemoved) {
Iterator> executionPlanIterator = executionPlan.entrySet().iterator();
for (int i = 0; i < planBeforeVisiting.get(toBeRemoved.from); i++) {
executionPlanIterator.next();
}
while (executionPlanIterator.hasNext()) {
executionPlanIterator.next();
executionPlanIterator.remove();
}
}
private void restoreQueue(List nodeQueue, HashMultimap visitingNodes, GraphEdge toBeRemoved) {
TaskInfoInVisitingSegment nextInQueue = null;
while (nextInQueue == null || !toBeRemoved.from.equals(nextInQueue.taskInfo)) {
nextInQueue = nodeQueue.get(0);
visitingNodes.remove(nextInQueue.taskInfo, nextInQueue.visitingSegment);
if (!toBeRemoved.from.equals(nextInQueue.taskInfo)) {
nodeQueue.remove(0);
}
}
}
private void restorePath(Deque path, GraphEdge toBeRemoved) {
TaskInfo removedFromPath = null;
while (!toBeRemoved.from.equals(removedFromPath)) {
removedFromPath = path.pop();
}
}
private void addAllSuccessorsInReverseOrder(TaskInfo taskNode, ArrayList dependsOnTasks) {
addAllReversed(dependsOnTasks, taskNode.getDependencySuccessors());
addAllReversed(dependsOnTasks, taskNode.getMustSuccessors());
addAllReversed(dependsOnTasks, taskNode.getShouldSuccessors());
}
private void removeShouldRunAfterSuccessorsIfTheyImposeACycle(final HashMultimap visitingNodes, final TaskInfoInVisitingSegment taskNodeWithVisitingSegment) {
TaskInfo taskNode = taskNodeWithVisitingSegment.taskInfo;
Iterables.removeIf(taskNode.getShouldSuccessors(), new Predicate() {
public boolean apply(TaskInfo input) {
return visitingNodes.containsEntry(input, taskNodeWithVisitingSegment.visitingSegment);
}
});
}
private void takePlanSnapshotIfCanBeRestoredToCurrentTask(HashMap planBeforeVisiting, TaskInfo taskNode) {
if (taskNode.getShouldSuccessors().size() > 0) {
planBeforeVisiting.put(taskNode, executionPlan.size());
}
}
private void recordEdgeIfArrivedViaShouldRunAfter(Deque walkedShouldRunAfterEdges, Deque path, TaskInfo taskNode) {
if (!path.isEmpty() && path.peek().getShouldSuccessors().contains(taskNode)) {
walkedShouldRunAfterEdges.push(new GraphEdge(path.peek(), taskNode));
}
}
/**
* Given a finalizer task, determine where in the current node queue that it should be inserted.
* The finalizer should be inserted after any of it's preceding tasks.
*/
private int finalizerTaskPosition(TaskInfo finalizer, final List nodeQueue) {
if (nodeQueue.size() == 0) {
return 0;
}
Set precedingTasks = getAllPrecedingTasks(finalizer);
Set precedingTaskIndices = CollectionUtils.collect(precedingTasks, new Transformer() {
public Integer transform(final TaskInfo dependsOnTask) {
return Iterables.indexOf(nodeQueue, new Predicate() {
public boolean apply(TaskInfoInVisitingSegment taskInfoInVisitingSegment) {
return taskInfoInVisitingSegment.taskInfo.equals(dependsOnTask);
}
});
}
});
return Collections.max(precedingTaskIndices) + 1;
}
private Set getAllPrecedingTasks(TaskInfo finalizer) {
Set precedingTasks = new HashSet();
Deque candidateTasks = new ArrayDeque();
// Consider every task that must run before the finalizer
candidateTasks.addAll(finalizer.getDependencySuccessors());
candidateTasks.addAll(finalizer.getMustSuccessors());
candidateTasks.addAll(finalizer.getShouldSuccessors());
// For each candidate task, add it to the preceding tasks.
while (!candidateTasks.isEmpty()) {
TaskInfo precedingTask = candidateTasks.pop();
if (precedingTasks.add(precedingTask)) {
// Any task that the preceding task must run after is also a preceding task.
candidateTasks.addAll(precedingTask.getMustSuccessors());
}
}
return precedingTasks;
}
private void onOrderingCycle() {
CachingDirectedGraphWalker graphWalker = new CachingDirectedGraphWalker(new DirectedGraph() {
public void getNodeValues(TaskInfo node, Collection super Void> values, Collection super TaskInfo> connectedNodes) {
connectedNodes.addAll(node.getDependencySuccessors());
connectedNodes.addAll(node.getMustSuccessors());
}
});
graphWalker.add(entryTasks);
final List firstCycle = new ArrayList(graphWalker.findCycles().get(0));
Collections.sort(firstCycle);
DirectedGraphRenderer graphRenderer = new DirectedGraphRenderer(new GraphNodeRenderer() {
public void renderTo(TaskInfo node, StyledTextOutput output) {
output.withStyle(StyledTextOutput.Style.Identifier).text(node.getTask().getIdentityPath());
}
}, new DirectedGraph() {
public void getNodeValues(TaskInfo node, Collection super Object> values, Collection super TaskInfo> connectedNodes) {
for (TaskInfo dependency : firstCycle) {
if (node.getDependencySuccessors().contains(dependency) || node.getMustSuccessors().contains(dependency)) {
connectedNodes.add(dependency);
}
}
}
});
StringWriter writer = new StringWriter();
graphRenderer.renderTo(firstCycle.get(0), writer);
throw new CircularReferenceException(String.format("Circular dependency between the following tasks:%n%s", writer.toString()));
}
public void clear() {
coordinationService.withStateLock(new Transformer() {
@Override
public ResourceLockState.Disposition transform(ResourceLockState resourceLockState) {
nodeFactory.clear();
entryTasks.clear();
executionPlan.clear();
executionQueue.clear();
projectLocks.clear();
failures.clear();
taskMutations.clear();
canonicalizedFileCache.clear();
reachableCache.clear();
dependenciesCompleteCache.clear();
runningTasks.clear();
return FINISHED;
}
});
}
public List getTasks() {
return new ArrayList(executionPlan.keySet());
}
@Override
public Set getFilteredTasks() {
return filteredTasks;
}
public void useFilter(Spec super Task> filter) {
this.filter = filter;
}
public void useFailureHandler(TaskFailureHandler handler) {
this.failureHandler = handler;
}
@Override
public boolean executeWithTask(final WorkerLease workerLease, final Action taskExecution) {
final AtomicReference selected = new AtomicReference();
final AtomicBoolean workRemaining = new AtomicBoolean();
coordinationService.withStateLock(new Transformer() {
@Override
public ResourceLockState.Disposition transform(ResourceLockState resourceLockState) {
if (cancellationToken.isCancellationRequested()) {
if (abortExecution()) {
tasksCancelled = true;
}
}
workRemaining.set(workRemaining());
if (!workRemaining.get()) {
return FINISHED;
}
if (allProjectsLocked()) {
return RETRY;
}
try {
selected.set(selectNextTask(workerLease));
} catch (Throwable t) {
abortAllAndFail(t);
workRemaining.set(false);
}
if (selected.get() == null && workRemaining.get()) {
return RETRY;
} else {
return FINISHED;
}
}
});
TaskInfo selectedTask = selected.get();
execute(selectedTask, workerLease, taskExecution);
return workRemaining.get();
}
private TaskInfo selectNextTask(final WorkerLease workerLease) {
final AtomicReference selected = new AtomicReference();
final Iterator iterator = executionQueue.iterator();
while (iterator.hasNext()) {
final TaskInfo taskInfo = iterator.next();
if (taskInfo.isReady() && allDependenciesComplete(taskInfo)) {
coordinationService.withStateLock(new Transformer() {
@Override
public ResourceLockState.Disposition transform(ResourceLockState resourceLockState) {
ResourceLock projectLock = getProjectLock(taskInfo);
// TODO: convert output file checks to a resource lock
if (!projectLock.tryLock() || !workerLease.tryLock() || !canRunWithCurrentlyExecutedTasks(taskInfo)) {
return FAILED;
}
selected.set(taskInfo);
if (taskInfo.allDependenciesSuccessful()) {
recordTaskStarted(taskInfo);
taskInfo.startExecution();
} else {
taskInfo.skipExecution();
}
iterator.remove();
return FINISHED;
}
});
if (selected.get() != null) {
break;
}
}
}
return selected.get();
}
private void execute(TaskInfo selectedTask, WorkerLease workerLease, Action taskExecution) {
if (selectedTask == null) {
return;
}
try {
if (!selectedTask.isComplete()) {
taskExecution.execute(selectedTask);
}
} finally {
coordinationService.withStateLock(unlock(workerLease, getProjectLock(selectedTask)));
}
}
private boolean allDependenciesComplete(TaskInfo taskInfo) {
if (dependenciesCompleteCache.contains(taskInfo)) {
return true;
}
boolean dependenciesComplete = taskInfo.allDependenciesComplete();
if (dependenciesComplete) {
dependenciesCompleteCache.add(taskInfo);
}
return dependenciesComplete;
}
private boolean allProjectsLocked() {
for (ResourceLock lock : projectLocks.values()) {
if (!lock.isLocked()) {
return false;
}
}
return true;
}
private ResourceLock getProjectLock(TaskInfo taskInfo) {
return projectLocks.get(taskInfo.getTask().getProject());
}
private ResourceLock getOrCreateProjectLock(Project project) {
String gradlePath = ((GradleInternal) project.getGradle()).getIdentityPath().toString();
String projectPath = ((ProjectInternal) project).getIdentityPath().toString();
return workerLeaseService.getProjectLock(gradlePath, projectPath);
}
private boolean canRunWithCurrentlyExecutedTasks(TaskInfo taskInfo) {
Set candidateTaskDestroyables = getDestroyablePaths(taskInfo);
if (!candidateTaskDestroyables.isEmpty() && !taskInfo.getTask().getOutputs().getFileProperties().isEmpty()) {
throw new IllegalStateException("Task " + taskInfo.getTask().getIdentityPath() + " has both outputs and destroyables defined. A task can define either outputs or destroyables, but not both.");
}
if (!candidateTaskDestroyables.isEmpty() && !taskInfo.getTask().getInputs().getFileProperties().isEmpty()) {
throw new IllegalStateException("Task " + taskInfo.getTask().getIdentityPath() + " has both inputs and destroyables defined. A task can define either inputs or destroyables, but not both.");
}
if (!runningTasks.isEmpty()) {
Set candidateTaskOutputs = getOutputPaths(taskInfo);
Set candidateTaskMutations = !candidateTaskOutputs.isEmpty() ? candidateTaskOutputs : candidateTaskDestroyables;
Pair overlap = firstRunningTaskWithOverlappingMutations(candidateTaskMutations);
if (overlap != null) {
return false;
}
}
Pair overlap = firstTaskWithDestroyedIntermediateInput(taskInfo, candidateTaskDestroyables);
if (overlap != null) {
return false;
}
return true;
}
private Set canonicalizedPaths(final Map cache, Iterable files) {
Function canonicalize = new Function() {
@Override
public String apply(File file) {
String path;
try {
path = cache.get(file);
if (path == null) {
path = file.getCanonicalPath();
cache.put(file, path);
}
} catch (IOException e) {
throw new UncheckedIOException(e);
}
return path;
}
};
return Sets.newHashSet(Iterables.transform(files, canonicalize));
}
@Nullable
private Pair firstRunningTaskWithOverlappingMutations(Set candidateTaskMutations) {
if (!candidateTaskMutations.isEmpty()) {
for (TaskInfo runningTask : runningTasks) {
TaskMutationInfo taskMutationInfo = taskMutations.get(runningTask);
Iterable runningTaskMutations = Iterables.concat(taskMutationInfo.outputPaths, taskMutationInfo.destroyablePaths);
String firstOverlap = findFirstOverlap(candidateTaskMutations, runningTaskMutations);
if (firstOverlap != null) {
return Pair.of(runningTask, firstOverlap);
}
}
}
return null;
}
@Nullable
private Pair firstTaskWithDestroyedIntermediateInput(final TaskInfo taskInfo, Set destroyablePaths) {
if (!destroyablePaths.isEmpty()) {
Iterator iterator = taskMutations.values().iterator();
while (iterator.hasNext()) {
TaskMutationInfo taskMutationInfo = iterator.next();
if (taskMutationInfo.task.isComplete() && !taskMutationInfo.consumingTasks.isEmpty()) {
String firstOverlap = findFirstOverlap(destroyablePaths, taskMutationInfo.outputPaths);
if (firstOverlap != null) {
for (TaskInfo consumingTask : taskMutationInfo.consumingTasks) {
if (consumingTask != taskInfo && !isReachableFrom(consumingTask, taskInfo)) {
return Pair.of(consumingTask, firstOverlap);
}
}
}
}
}
}
return null;
}
private boolean isReachableFrom(TaskInfo fromTask, TaskInfo toTask) {
Pair taskPair = Pair.of(fromTask, toTask);
if (reachableCache.get(taskPair) != null) {
return reachableCache.get(taskPair);
}
boolean reachable = false;
for (TaskInfo dependency : Iterables.concat(fromTask.getMustSuccessors(), fromTask.getDependencySuccessors())) {
if (!dependency.isComplete()) {
if (dependency == toTask) {
reachable = true;
}
if (isReachableFrom(dependency, toTask)) {
reachable = true;
}
}
}
reachableCache.put(taskPair, reachable);
return reachable;
}
private String findFirstOverlap(Iterable paths1, Iterable paths2) {
for (String path1 : paths1) {
for (String path2 : paths2) {
if (pathsOverlap(path1, path2)) {
return TextUtil.shorterOf(path1, path2);
}
}
}
return null;
}
private Set getOutputPaths(TaskInfo task) {
try {
return canonicalizedPaths(canonicalizedFileCache, Iterables.concat(
task.getTask().getOutputs().getFiles(),
((TaskLocalStateInternal) task.getTask().getLocalState()).getFiles()
));
} catch (ResourceDeadlockException e) {
throw new IllegalStateException("A deadlock was detected while resolving the task outputs for " + task.getTask().getIdentityPath() + ". This can be caused, for instance, by a task output causing dependency resolution.", e);
}
}
private Set getDestroyablePaths(TaskInfo task) {
return canonicalizedPaths(canonicalizedFileCache, ((TaskDestroyablesInternal) task.getTask().getDestroyables()).getFiles());
}
private boolean pathsOverlap(String firstPath, String secondPath) {
if (firstPath.equals(secondPath)) {
return true;
}
String shorter;
String longer;
if (firstPath.length() > secondPath.length()) {
shorter = secondPath;
longer = firstPath;
} else {
shorter = firstPath;
longer = secondPath;
}
return longer.startsWith(shorter + StandardSystemProperty.FILE_SEPARATOR.value());
}
private void recordTaskStarted(TaskInfo taskInfo) {
runningTasks.add(taskInfo);
TaskMutationInfo taskMutationInfo = taskMutations.get(taskInfo);
taskMutationInfo.outputPaths.addAll(getOutputPaths(taskInfo));
taskMutationInfo.destroyablePaths.addAll(getDestroyablePaths(taskInfo));
}
private void recordTaskCompleted(TaskInfo taskInfo) {
runningTasks.remove(taskInfo);
TaskMutationInfo taskMutationInfo = taskMutations.get(taskInfo);
for (TaskInfo producerTask : taskMutationInfo.consumesOutputOf) {
TaskMutationInfo producerTaskMutationInfo = taskMutations.get(producerTask);
if (producerTaskMutationInfo.consumingTasks.remove(taskInfo) && canRemoveTaskMutation(producerTaskMutationInfo)) {
taskMutations.remove(producerTask);
}
}
if (canRemoveTaskMutation(taskMutationInfo)) {
taskMutations.remove(taskInfo);
}
}
private boolean canRemoveTaskMutation(TaskMutationInfo taskMutationInfo) {
return taskMutationInfo != null && taskMutationInfo.task.isComplete() && taskMutationInfo.consumingTasks.isEmpty();
}
public void taskComplete(final TaskInfo taskInfo) {
coordinationService.withStateLock(new Transformer() {
@Override
public ResourceLockState.Disposition transform(ResourceLockState resourceLockState) {
enforceFinalizerTasks(taskInfo);
if (taskInfo.isFailed()) {
handleFailure(taskInfo);
}
taskInfo.finishExecution();
recordTaskCompleted(taskInfo);
return FINISHED;
}
});
}
private void enforceFinalizerTasks(TaskInfo taskInfo) {
for (TaskInfo finalizerNode : taskInfo.getFinalizers()) {
if (finalizerNode.isRequired() || finalizerNode.isMustNotRun()) {
enforceWithDependencies(finalizerNode, Sets.newHashSet());
}
}
}
private void enforceWithDependencies(TaskInfo nodeInfo, Set enforcedTasks) {
Deque candidateNodes = new ArrayDeque();
candidateNodes.add(nodeInfo);
while (!candidateNodes.isEmpty()) {
TaskInfo node = candidateNodes.pop();
if (!enforcedTasks.contains(node)) {
enforcedTasks.add(node);
candidateNodes.addAll(node.getDependencySuccessors());
if (node.isMustNotRun() || node.isRequired()) {
node.enforceRun();
}
}
}
}
private void abortAllAndFail(Throwable t) {
abortExecution(true);
this.failures.add(t);
}
private void handleFailure(TaskInfo taskInfo) {
Throwable executionFailure = taskInfo.getExecutionFailure();
if (executionFailure != null) {
// Always abort execution for an execution failure (as opposed to a task failure)
abortExecution();
this.failures.add(executionFailure);
return;
}
// Task failure
try {
failureHandler.onTaskFailure(taskInfo.getTask());
this.failures.add(taskInfo.getTaskFailure());
} catch (Exception e) {
// If the failure handler rethrows exception, then execution of other tasks is aborted. (--continue will collect failures)
abortExecution();
this.failures.add(e);
}
}
private boolean abortExecution() {
return abortExecution(false);
}
private boolean abortExecution(boolean abortAll) {
boolean aborted = false;
for (TaskInfo taskInfo : executionPlan.values()) {
// Allow currently executing and enforced tasks to complete, but skip everything else.
if (taskInfo.isRequired()) {
taskInfo.skipExecution();
aborted = true;
}
// If abortAll is set, also stop enforced tasks.
if (abortAll && taskInfo.isReady()) {
taskInfo.abortExecution();
aborted = true;
}
}
return aborted;
}
public void awaitCompletion() {
coordinationService.withStateLock(new Transformer() {
@Override
public ResourceLockState.Disposition transform(ResourceLockState resourceLockState) {
if (allTasksComplete()) {
rethrowFailures();
return FINISHED;
} else {
return RETRY;
}
}
});
}
private void rethrowFailures() {
if (tasksCancelled) {
failures.add(new BuildCancelledException());
}
if (failures.isEmpty()) {
return;
}
if (failures.size() > 1) {
throw new MultipleBuildFailures(failures);
}
throw UncheckedException.throwAsUncheckedException(failures.get(0));
}
private boolean allTasksComplete() {
for (TaskInfo taskInfo : executionPlan.values()) {
if (!taskInfo.isComplete()) {
return false;
}
}
return true;
}
private boolean workRemaining() {
for (TaskInfo taskInfo : executionQueue) {
if (!taskInfo.isComplete()) {
return true;
}
}
return false;
}
private static class GraphEdge {
private final TaskInfo from;
private final TaskInfo to;
private GraphEdge(TaskInfo from, TaskInfo to) {
this.from = from;
this.to = to;
}
}
private static class TaskInfoInVisitingSegment {
private final TaskInfo taskInfo;
private final int visitingSegment;
private TaskInfoInVisitingSegment(TaskInfo taskInfo, int visitingSegment) {
this.taskInfo = taskInfo;
this.visitingSegment = visitingSegment;
}
}
private static class RethrowingFailureHandler implements TaskFailureHandler {
public void onTaskFailure(Task task) {
task.getState().rethrowFailure();
}
}
private static class TaskMutationInfo {
final TaskInfo task;
final Set consumingTasks = Sets.newHashSet();
final Set consumesOutputOf = Sets.newHashSet();
final Set outputPaths = Sets.newHashSet();
final Set destroyablePaths = Sets.newHashSet();
TaskMutationInfo(TaskInfo task) {
this.task = task;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy