All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gradle.internal.execution.DefaultWorkExecutionTracker 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.execution;

import org.gradle.api.internal.TaskInternal;
import org.gradle.api.internal.tasks.execution.ExecuteTaskBuildOperationDetails;
import org.gradle.api.internal.tasks.execution.ExecuteTaskBuildOperationType;
import org.gradle.internal.operations.BuildOperationAncestryTracker;
import org.gradle.internal.operations.BuildOperationDescriptor;
import org.gradle.internal.operations.BuildOperationListener;
import org.gradle.internal.operations.BuildOperationListenerManager;
import org.gradle.internal.operations.CurrentBuildOperationRef;
import org.gradle.internal.operations.OperationFinishEvent;
import org.gradle.internal.operations.OperationIdentifier;
import org.gradle.internal.operations.OperationProgressEvent;
import org.gradle.internal.operations.OperationStartEvent;
import org.gradle.internal.operations.UncategorizedBuildOperations;

import java.io.Closeable;
import java.io.IOException;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;

import static java.lang.String.format;

public class DefaultWorkExecutionTracker implements WorkExecutionTracker, Closeable {

    private final BuildOperationAncestryTracker buildOperationAncestryTracker;
    private final BuildOperationListenerManager buildOperationListenerManager;
    private final CurrentBuildOperationRef currentBuildOperationRef = CurrentBuildOperationRef.instance();
    private final OperationListener operationListener = new OperationListener();

    public DefaultWorkExecutionTracker(
        BuildOperationAncestryTracker buildOperationAncestryTracker,
        BuildOperationListenerManager buildOperationListenerManager
    ) {
        this.buildOperationAncestryTracker = buildOperationAncestryTracker;
        this.buildOperationListenerManager = buildOperationListenerManager;
        buildOperationListenerManager.addListener(operationListener);
    }

    @Override
    public Optional getCurrentTask() {
        return buildOperationAncestryTracker
            .findClosestExistingAncestor(
                currentBuildOperationRef.getId(),
                operationListener.runningTasks::get
            );
    }

    @Override
    public boolean isExecutingTransformAction() {
        return buildOperationAncestryTracker.findClosestMatchingAncestor(
            currentBuildOperationRef.getId(), operationListener.runningTransformActions::contains
        ).isPresent();
    }

    @Override
    public void close() throws IOException {
        buildOperationListenerManager.removeListener(operationListener);
        assert !operationListener.hasRunningWork();
    }

    private static class OperationListener implements BuildOperationListener {

        final Map runningTasks = new ConcurrentHashMap<>();
        final Set runningTransformActions = ConcurrentHashMap.newKeySet();

        @Override
        public void started(BuildOperationDescriptor buildOperation, OperationStartEvent startEvent) {
            if (isTransformAction(buildOperation)) {
                runningTransformActions.add(mandatoryIdOf(buildOperation));
            } else {
                Object details = buildOperation.getDetails();
                if (details instanceof ExecuteTaskBuildOperationDetails) {
                    runningTasks.put(mandatoryIdOf(buildOperation), ((ExecuteTaskBuildOperationDetails) details).getTask());
                }
            }
        }

        @Override
        public void finished(BuildOperationDescriptor buildOperation, OperationFinishEvent finishEvent) {
            if (isTransformAction(buildOperation)) {
                runningTransformActions.remove(mandatoryIdOf(buildOperation));
            } else {
                Object details = buildOperation.getDetails();
                if (details instanceof ExecuteTaskBuildOperationType.Details) {
                    Object removed = runningTasks.remove(mandatoryIdOf(buildOperation));
                    if (removed == null) {
                        throw new IllegalStateException(format("Task build operation %s was finished without being started.", buildOperation));
                    }
                }
            }
        }

        private OperationIdentifier mandatoryIdOf(BuildOperationDescriptor buildOperation) {
            OperationIdentifier id = buildOperation.getId();
            if (id == null) {
                throw new IllegalStateException(format("Build operation %s has no valid id", buildOperation));
            }
            return id;
        }

        @Override
        public void progress(OperationIdentifier operationIdentifier, OperationProgressEvent progressEvent) {
        }

        public boolean hasRunningWork() {
            return !runningTasks.isEmpty() || !runningTransformActions.isEmpty();
        }

        private static boolean isTransformAction(BuildOperationDescriptor buildOperation) {
            return UncategorizedBuildOperations.TRANSFORM_ACTION.equals(buildOperation.getMetadata());
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy