org.gradle.composite.internal.DefaultBuildControllers Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-test-kit Show documentation
Show all versions of gradle-test-kit Show documentation
Gradle 6.2.1 API redistribution.
/*
* Copyright 2017 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.composite.internal;
import com.google.common.collect.ImmutableList;
import org.gradle.api.artifacts.component.BuildIdentifier;
import org.gradle.api.internal.artifacts.DefaultBuildIdentifier;
import org.gradle.execution.plan.PlanExecutor;
import org.gradle.internal.UncheckedException;
import org.gradle.internal.build.BuildState;
import org.gradle.internal.build.ExecutionResult;
import org.gradle.internal.concurrent.CompositeStoppable;
import org.gradle.internal.concurrent.ManagedExecutor;
import org.gradle.internal.work.WorkerLeaseService;
import java.util.Comparator;
import java.util.Map;
import java.util.TreeMap;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
class DefaultBuildControllers implements BuildControllers {
// Always iterate over the controllers in a fixed order
private final Map controllers = new TreeMap<>(idComparator());
private final ManagedExecutor executorService;
private final WorkerLeaseService workerLeaseService;
private final PlanExecutor planExecutor;
private final int monitoringPollTime;
private final TimeUnit monitoringPollTimeUnit;
DefaultBuildControllers(ManagedExecutor executorService, WorkerLeaseService workerLeaseService, PlanExecutor planExecutor, int monitoringPollTime, TimeUnit monitoringPollTimeUnit) {
this.executorService = executorService;
this.workerLeaseService = workerLeaseService;
this.planExecutor = planExecutor;
this.monitoringPollTime = monitoringPollTime;
this.monitoringPollTimeUnit = monitoringPollTimeUnit;
}
@Override
public BuildController getBuildController(BuildState build) {
BuildController buildController = controllers.get(build.getBuildIdentifier());
if (buildController != null) {
return buildController;
}
BuildController newBuildController = new DefaultBuildController(build, workerLeaseService);
controllers.put(build.getBuildIdentifier(), newBuildController);
return newBuildController;
}
@Override
public void populateWorkGraphs() {
boolean tasksDiscovered = true;
while (tasksDiscovered) {
tasksDiscovered = false;
for (BuildController buildController : ImmutableList.copyOf(controllers.values())) {
if (buildController.scheduleQueuedTasks()) {
tasksDiscovered = true;
}
}
}
for (BuildController buildController : controllers.values()) {
buildController.finalizeWorkGraph();
}
}
@Override
public ExecutionResult execute() {
CountDownLatch complete = new CountDownLatch(controllers.size());
Map> results = new ConcurrentHashMap<>();
// Start work in each build
for (BuildController buildController : controllers.values()) {
buildController.startExecution(executorService, result -> {
results.put(buildController, result);
complete.countDown();
});
}
awaitCompletion(complete);
// Collect the failures in deterministic order
ExecutionResult result = ExecutionResult.succeeded();
for (BuildController buildController : controllers.values()) {
result = result.withFailures(results.get(buildController));
}
return result;
}
private void awaitCompletion(CountDownLatch complete) {
while (true) {
// Wake for the work in all builds to complete. Periodically wake up and check the executor health
AtomicBoolean done = new AtomicBoolean();
// Ensure that this thread does not hold locks while waiting and so prevent this work from completing
workerLeaseService.blocking(() -> {
try {
done.set(complete.await(monitoringPollTime, monitoringPollTimeUnit));
} catch (InterruptedException e) {
throw UncheckedException.throwAsUncheckedException(e);
}
});
if (done.get()) {
return;
}
planExecutor.assertHealthy();
}
}
@Override
public void close() {
CompositeStoppable.stoppable(controllers.values()).stop();
}
private static Comparator idComparator() {
return (id1, id2) -> {
// Root is always last
if (id1.equals(DefaultBuildIdentifier.ROOT)) {
if (id2.equals(DefaultBuildIdentifier.ROOT)) {
return 0;
} else {
return 1;
}
}
if (id2.equals(DefaultBuildIdentifier.ROOT)) {
return -1;
}
return id1.getBuildPath().compareTo(id2.getBuildPath());
};
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy