org.gradle.internal.build.DefaultBuildWorkGraphController Maven / Gradle / Ivy
/*
* Copyright 2021 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.internal.build;
import com.google.common.util.concurrent.Runnables;
import org.gradle.api.Task;
import org.gradle.api.artifacts.component.BuildIdentifier;
import org.gradle.api.internal.TaskInternal;
import org.gradle.api.specs.Spec;
import org.gradle.composite.internal.IncludedBuildTaskResource;
import org.gradle.composite.internal.TaskIdentifier;
import org.gradle.execution.EntryTaskSelector;
import org.gradle.execution.plan.BuildWorkPlan;
import org.gradle.execution.plan.LocalTaskNode;
import org.gradle.execution.plan.QueryableExecutionPlan;
import org.gradle.execution.plan.TaskNode;
import org.gradle.execution.plan.TaskNodeFactory;
import org.gradle.internal.UncheckedException;
import org.gradle.internal.work.WorkerLeaseService;
import javax.annotation.Nullable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.BiConsumer;
import java.util.function.Consumer;
public class DefaultBuildWorkGraphController implements BuildWorkGraphController {
private final TaskNodeFactory taskNodeFactory;
private final BuildLifecycleController controller;
private final BuildIdentifier buildIdentifier;
private final WorkerLeaseService workerLeaseService;
private final Map nodesByPath = new ConcurrentHashMap<>();
private final Object lock = new Object();
private Thread currentOwner;
private final Set pendingGraphs = new HashSet<>();
private DefaultBuildWorkGraph currentlyRunning;
public DefaultBuildWorkGraphController(TaskNodeFactory taskNodeFactory, BuildLifecycleController controller, BuildState buildState, WorkerLeaseService workerLeaseService) {
this.taskNodeFactory = taskNodeFactory;
this.controller = controller;
this.buildIdentifier = buildState.getBuildIdentifier();
this.workerLeaseService = workerLeaseService;
}
@Override
public void resetState() {
synchronized (lock) {
if (currentOwner != null) {
throw new IllegalStateException("Cannot reset work graph state as another thread is currently using the work graph.");
}
nodesByPath.clear();
}
taskNodeFactory.resetState();
}
@Override
public ExportedTaskNode locateTask(TaskIdentifier taskIdentifier) {
DefaultExportedTaskNode node = doLocate(taskIdentifier);
if (taskIdentifier instanceof TaskIdentifier.TaskBasedTaskIdentifier) {
node.maybeBindTask(((TaskIdentifier.TaskBasedTaskIdentifier) taskIdentifier).getTask());
}
return node;
}
@Override
public BuildWorkGraph newWorkGraph() {
synchronized (lock) {
while (currentOwner != null && currentOwner != Thread.currentThread()) {
workerLeaseService.blocking(() -> {
try {
lock.wait();
} catch (InterruptedException e) {
throw UncheckedException.throwAsUncheckedException(e);
}
});
}
currentOwner = Thread.currentThread();
DefaultBuildWorkGraph workGraph = new DefaultBuildWorkGraph();
pendingGraphs.add(workGraph);
return workGraph;
}
}
private DefaultExportedTaskNode doLocate(TaskIdentifier taskIdentifier) {
return nodesByPath.computeIfAbsent(taskIdentifier.getTaskPath(), DefaultExportedTaskNode::new);
}
@Nullable
private TaskInternal findTaskNode(String taskPath) {
for (Task task : taskNodeFactory.getTasks()) {
if (task.getPath().equals(taskPath)) {
return (TaskInternal) task;
}
}
return null;
}
private class DefaultBuildWorkGraph implements BuildWorkGraph {
private final Thread owner;
BuildWorkPlan plan;
public DefaultBuildWorkGraph() {
this.owner = Thread.currentThread();
}
@Override
public void stop() {
if (plan != null) {
plan.stop();
}
synchronized (lock) {
assert currentOwner == Thread.currentThread();
pendingGraphs.remove(this);
if (pendingGraphs.isEmpty()) {
currentOwner = null;
lock.notifyAll();
}
}
}
@Override
public boolean schedule(Collection taskNodes) {
assertIsOwner();
List tasks = new ArrayList<>();
for (ExportedTaskNode taskNode : taskNodes) {
DefaultExportedTaskNode node = (DefaultExportedTaskNode) taskNode;
if (nodesByPath.get(node.taskPath) != taskNode) {
throw new IllegalArgumentException();
}
if (node.shouldSchedule()) {
// Not already in task graph
tasks.add(node.getTask());
}
}
if (tasks.isEmpty()) {
return false;
}
controller.getGradle().getOwner().getProjects().withMutableStateOfAllProjects(() -> {
createPlan();
controller.prepareToScheduleTasks();
controller.populateWorkGraph(plan, workGraph -> workGraph.addEntryTasks(tasks));
});
return true;
}
@Override
public void populateWorkGraph(Consumer super BuildLifecycleController.WorkGraphBuilder> action) {
BuildWorkPlan ownedPlan = getOwnedPlan();
controller.prepareToScheduleTasks();
controller.populateWorkGraph(ownedPlan, action);
}
@Override
public void addFilter(Spec filter) {
getOwnedPlan().addFilter(filter);
}
@Override
public void addFinalization(BiConsumer finalization) {
getOwnedPlan().addFinalization(finalization);
}
private BuildWorkPlan getOwnedPlan() {
assertIsOwner();
createPlan();
return plan;
}
private void createPlan() {
if (plan == null) {
plan = controller.newWorkGraph();
plan.onComplete(this::nodeComplete);
}
}
private void nodeComplete(LocalTaskNode node) {
DefaultExportedTaskNode exportedNode = nodesByPath.get(node.getTask().getPath());
if (exportedNode != null) {
exportedNode.fireCompleted();
}
}
@Override
public void finalizeGraph() {
assertIsOwner();
if (plan != null) {
controller.finalizeWorkGraph(plan);
}
}
@Override
public ExecutionResult runWork() {
synchronized (lock) {
if (currentlyRunning != null) {
throw new IllegalStateException("Build '" + buildIdentifier + "' is currently already running work.");
}
currentlyRunning = this;
}
try {
if (plan != null) {
return controller.executeTasks(plan);
} else {
return ExecutionResult.succeeded();
}
} finally {
synchronized (lock) {
currentlyRunning = null;
}
}
}
private void assertIsOwner() {
if (Thread.currentThread() != owner) {
throw new IllegalStateException("Current thread is not the owner of this work graph.");
}
}
}
private class DefaultExportedTaskNode implements ExportedTaskNode {
final String taskPath;
TaskNode taskNode;
Runnable action = Runnables.doNothing();
DefaultExportedTaskNode(String taskPath) {
this.taskPath = taskPath;
}
void maybeBindTask(TaskInternal task) {
synchronized (lock) {
if (taskNode == null) {
taskNode = taskNodeFactory.getOrCreateNode(task);
}
}
}
@Override
public void onComplete(Runnable action) {
synchronized (lock) {
Runnable previous = this.action;
this.action = () -> {
previous.run();
action.run();
};
}
}
@Override
public TaskInternal getTask() {
synchronized (lock) {
if (taskNode == null) {
TaskInternal task = findTaskNode(taskPath);
if (task == null) {
throw new IllegalStateException("Task '" + taskPath + "' was never scheduled for execution.");
}
taskNode = taskNodeFactory.getOrCreateNode(task);
}
return taskNode.getTask();
}
}
@Override
public IncludedBuildTaskResource.State getTaskState() {
synchronized (lock) {
if (taskNode == null) {
TaskInternal task = findTaskNode(taskPath);
if (task == null) {
// Assume not scheduled yet
return IncludedBuildTaskResource.State.NotScheduled;
}
taskNode = taskNodeFactory.getOrCreateNode(task);
}
if (taskNode.isExecuted() && taskNode.isSuccessful()) {
return IncludedBuildTaskResource.State.Success;
} else if (taskNode.isExecuted()) {
return IncludedBuildTaskResource.State.Failed;
} else if (taskNode.isComplete()) {
// Not scheduled
return IncludedBuildTaskResource.State.NotScheduled;
} else {
// Scheduled but not completed
return IncludedBuildTaskResource.State.Scheduled;
}
}
}
boolean shouldSchedule() {
synchronized (lock) {
return taskNode == null || !taskNode.isRequired();
}
}
@Override
public String healthDiagnostics() {
synchronized (lock) {
return "exportedTaskState=" + getTaskState();
}
}
public void fireCompleted() {
synchronized (lock) {
action.run();
action = Runnables.doNothing();
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy