org.gradle.internal.progress.DefaultBuildOperationExecutor 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 2014 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.progress;
import org.gradle.api.Action;
import org.gradle.api.Transformer;
import org.gradle.internal.Transformers;
import org.gradle.internal.UncheckedException;
import org.gradle.internal.logging.events.OperationIdentifier;
import org.gradle.internal.logging.progress.ProgressLogger;
import org.gradle.internal.logging.progress.ProgressLoggerFactory;
import org.gradle.internal.operations.BuildOperationContext;
import org.gradle.internal.time.TimeProvider;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
public class DefaultBuildOperationExecutor implements BuildOperationExecutor {
private final BuildOperationListener listener;
private final TimeProvider timeProvider;
private final ProgressLoggerFactory progressLoggerFactory;
private final AtomicLong nextId = new AtomicLong();
private final ThreadLocal currentOperation = new ThreadLocal();
public DefaultBuildOperationExecutor(BuildOperationListener listener, TimeProvider timeProvider, ProgressLoggerFactory progressLoggerFactory) {
this.listener = listener;
this.timeProvider = timeProvider;
this.progressLoggerFactory = progressLoggerFactory;
}
@Override
public Operation getCurrentOperation() {
OperationDetails current = currentOperation.get();
if (current == null) {
throw new IllegalStateException("No operation is currently running.");
}
return current;
}
@Override
public void run(String displayName, Action super BuildOperationContext> action) {
run(BuildOperationDetails.displayName(displayName).build(), Transformers.toTransformer(action));
}
@Override
public void run(BuildOperationDetails operationDetails, Action super BuildOperationContext> action) {
run(operationDetails, Transformers.toTransformer(action));
}
@Override
public T run(String displayName, Transformer factory) {
return run(BuildOperationDetails.displayName(displayName).build(), factory);
}
@Override
public T run(BuildOperationDetails operationDetails, Transformer factory) {
OperationDetails operationBefore = currentOperation.get();
OperationDetails parent = operationDetails.getParent() != null ? (OperationDetails) operationDetails.getParent() : operationBefore;
OperationIdentifier parentId;
if (parent == null) {
parentId = null;
} else {
if (!parent.running.get()) {
throw new IllegalStateException(String.format("Cannot start operation (%s) as parent operation (%s) has already completed.", operationDetails.getDisplayName(), parent.operationDetails.getDisplayName()));
}
parentId = parent.id;
}
OperationIdentifier id = new OperationIdentifier(nextId.getAndIncrement());
OperationDetails currentOperation = new OperationDetails(parent, id, operationDetails);
currentOperation.running.set(true);
this.currentOperation.set(currentOperation);
try {
long startTime = timeProvider.getCurrentTime();
BuildOperationInternal operation = new BuildOperationInternal(id, parentId, operationDetails.getName(), operationDetails.getDisplayName(), operationDetails.getOperationDescriptor());
listener.started(operation, new OperationStartEvent(startTime));
T result = null;
Throwable failure = null;
BuildOperationContextImpl context = new BuildOperationContextImpl();
try {
ProgressLogger progressLogger;
if (operationDetails.getProgressDisplayName() != null) {
progressLogger = progressLoggerFactory.newOperation(DefaultBuildOperationExecutor.class);
progressLogger.setDescription(operationDetails.getDisplayName());
progressLogger.setShortDescription(operationDetails.getProgressDisplayName());
progressLogger.started();
} else {
progressLogger = null;
}
try {
result = factory.transform(context);
} finally {
if (progressLogger != null) {
progressLogger.completed();
}
}
if (parent != null && !parent.running.get()) {
throw new IllegalStateException(String.format("Parent operation (%s) completed before this operation (%s).", parent.operationDetails.getDisplayName(), operationDetails.getDisplayName()));
}
} catch (Throwable t) {
context.failed(t);
failure = t;
}
long endTime = timeProvider.getCurrentTime();
listener.finished(operation, new OperationResult(startTime, endTime, context.failure));
if (failure != null) {
throw UncheckedException.throwAsUncheckedException(failure);
}
return result;
} finally {
this.currentOperation.set(operationBefore);
currentOperation.running.set(false);
}
}
private static class OperationDetails implements Operation {
final AtomicBoolean running = new AtomicBoolean();
final OperationDetails parent;
final OperationIdentifier id;
final BuildOperationDetails operationDetails;
OperationDetails(OperationDetails parent, OperationIdentifier id, BuildOperationDetails operationDetails) {
this.parent = parent;
this.id = id;
this.operationDetails = operationDetails;
}
@Override
public Object getId() {
return id;
}
@Override
public Object getParentId() {
return parent.id;
}
}
private static class BuildOperationContextImpl implements BuildOperationContext {
Throwable failure;
@Override
public void failed(Throwable t) {
failure = t;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy