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

org.gradle.internal.model.StateTransitionController 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.model;

import org.gradle.internal.DisplayName;
import org.gradle.internal.UncheckedException;
import org.gradle.internal.build.ExecutionResult;
import org.gradle.internal.work.Synchronizer;

import javax.annotation.Nullable;
import javax.annotation.concurrent.ThreadSafe;
import java.util.List;
import java.util.function.Function;
import java.util.function.Supplier;

/**
 * Manages the transition between states of some object with mutable state.
 *
 * Adds validation to ensure that the object is in an expected state and applies thread safety.
 */
@ThreadSafe
public class StateTransitionController {
    private final DisplayName displayName;
    private final Synchronizer synchronizer;
    // This structure is immutable, and this field is mutated only by the thread that owns the lock
    private volatile CurrentState state;

    public StateTransitionController(DisplayName displayName, T initialState, Synchronizer synchronizer) {
        this.displayName = displayName;
        this.synchronizer = synchronizer;
        this.state = new InState<>(displayName, initialState, null);
    }

    /**
     * Verifies that the current state is not the given state. Ignores any transition in progress and failures of previous operations.
     *
     * 

You should try to not use this method, as it does not provide any thread safety for the code that follows the call.

*/ public void assertNotInState(T forbidden) { if (state.state == forbidden) { throw new IllegalStateException(displayName.getCapitalizedDisplayName() + " should not be in state " + forbidden + "."); } } /** * Verifies that the current state is the given state or some later state. Ignores any transition in progress and failures of previous operations. * *

You should try to not use this method, as it does not provide any thread safety for the code that follows the call.

*/ public void assertInStateOrLater(T expected) { CurrentState current = state; if (!current.hasSeenStateIgnoringTransitions(expected)) { throw new IllegalStateException(displayName.getCapitalizedDisplayName() + " should be in state " + expected + " or later."); } } /** * Calculates a value when the current state is not the given state. Allows concurrent access to the state and does not block other threads from transitioning the state. * Fails if the current state is the given state or if a transition to the given state is happening or a previous transition has failed. * *

You should try to not use this method, as it does not provide full thread safety.

*/ public S notInStateIgnoreOtherThreads(T forbidden, Supplier supplier) { CurrentState current = state; current.asResult().rethrow(); current.assertNotInState(forbidden); try { return supplier.get(); } catch (Throwable t) { // TODO - remove the need for locking here synchronizer.withLock(() -> { state = state.failed(ExecutionResult.failed(t)); }); throw UncheckedException.throwAsUncheckedException(t); } } /** * Runs the given action, verifying the current state is the expected state. * Fails if the current state is not the given state or a previous operation has failed. * Blocks until other operations are complete. */ public void inState(T expected, Runnable action) { inState(expected, () -> { action.run(); return null; }); } /** * Runs the given action, verifying the current state is the expected state. * Fails if the current state is not the given state, the current thread is transitioning the state, or a previous operation has failed. * Blocks until other operations are complete. */ public S inState(T expected, Supplier action) { return synchronizer.withLock(() -> { CurrentState current = state; current.assertInState(expected); try { return action.get(); } catch (Throwable t) { state = current.failed(ExecutionResult.failed(t)); throw state.rethrow(); } }); } /** * Runs the given action, verifying the current state is not the forbidden state. * Fails if the current state is the given state, the current thread is transitioning the state, or a previous operation has failed. * Blocks until other operations are complete. */ public S notInState(T forbidden, Supplier action) { return synchronizer.withLock(() -> { CurrentState current = state; current.assertNotInState(forbidden); try { return action.get(); } catch (Throwable t) { state = current.failed(ExecutionResult.failed(t)); throw state.rethrow(); } }); } /** * Resets the state to the given state. * Fails if the current state is not the given state, ignores failures from previous operations. * Blocks until other operations are complete. */ public void restart(T fromState, T toState, Runnable action) { synchronizer.withLock(() -> { CurrentState current = state; current.assertCanTransition(fromState, toState, true); action.run(); state = new InState<>(displayName, toState, null); }); } /** * Transitions to the given "to" state. * Fails if the current state is not the given "from" state, the current thread is transitioning the state, or a previous operation has failed. * Blocks until other operations are complete. */ public void transition(T fromState, T toState, Runnable action) { synchronizer.withLock(() -> doTransition(fromState, toState, action)); } /** * Transitions to the given "to" state. * Fails if the current state is not the given "from" state, the current thread is transitioning the state, or a previous operation has failed. * Blocks until other operations are complete. */ public S transition(T fromState, T toState, Supplier action) { return synchronizer.withLock(() -> doTransition(fromState, toState, () -> ExecutionResult.succeeded(action.get())).getValueOrRethrow()); } /** * Transitions to the given "to" state, returning any failure in the result object. * Fails if the current state is not the given "from" state, the current thread is transitioning the state, or a previous operation has failed. */ public ExecutionResult tryTransition(T fromState, T toState, Supplier> action) { return synchronizer.withLock(() -> doTransition(fromState, toState, action)); } /** * Transitions to the given "to" state. Does nothing if the current state is the "to" state. * Fails if the current state is not either of the given "to" or "from" state, the current thread is currently transitioning the state, or a previous operation has failed. * Blocks until other operations are complete. */ public void maybeTransition(T fromState, T toState, Runnable action) { synchronizer.withLock(() -> { if (state.inStateAndNotTransitioning(toState)) { return; } doTransition(fromState, toState, action); }); } public void maybeTransitionIfNotCurrentlyTransitioning(T fromState, T toState, Runnable action) { synchronizer.withLock(() -> { if (state.inStateOrTransitioningTo(toState)) { return; } doTransition(fromState, toState, action); }); } /** * Transitions to the given "to" state. Does nothing if the "to" state has already been transitioned to at some point in the past (but is not necessarily the current state). * Fails if the current state is not the given "from" state, the current thread is currently transitioning the state, or a previous operation has failed. * Blocks until other operations are complete. */ public void transitionIfNotPreviously(T fromState, T toState, Runnable action) { synchronizer.withLock(() -> { if (state.hasSeenStateAndNotTransitioning(toState)) { return; } doTransition(fromState, toState, action); }); } /** * Transitions to a final state, taking any failures from previous transitions and transforming them. */ public ExecutionResult transition(T fromState, T toState, Function, ExecutionResult> action) { return synchronizer.withLock(() -> { CurrentState current = state; current.assertCanTransition(fromState, toState, true); return doTransitionWithFailures(toState, action, current); }); } public ExecutionResult transition(List fromStates, T toState, Function, ExecutionResult> action) { return synchronizer.withLock(() -> { CurrentState current = state; current.assertCanTransition(fromStates, toState, true); return doTransitionWithFailures(toState, action, current); }); } private ExecutionResult doTransitionWithFailures(T toState, Function, ExecutionResult> action, CurrentState current) { ExecutionResult currentResult = current.asResult(); state = current.transitioningTo(toState); ExecutionResult result; try { result = action.apply(currentResult); } catch (Throwable t) { result = ExecutionResult.failed(t); } if (!result.getFailures().isEmpty()) { state = state.failed(result); } else { state = state.nextState(toState); } return result; } private void doTransition(T fromState, T toState, Runnable action) { doTransition(fromState, toState, () -> { action.run(); return ExecutionResult.succeeded(); }).getValueOrRethrow(); } private ExecutionResult doTransition(T fromState, T toState, Supplier> action) { CurrentState current = state; current.assertCanTransition(fromState, toState); state = current.transitioningTo(toState); ExecutionResult result; try { result = action.get(); } catch (Throwable t) { result = ExecutionResult.failed(t); } if (!result.getFailures().isEmpty()) { state = state.failed(result); } else { state = state.nextState(toState); } return result; } private static abstract class CurrentState { final DisplayName displayName; final T state; public CurrentState(DisplayName displayName, T state) { this.displayName = displayName; this.state = state; } public abstract void assertInState(T expected); public abstract void assertNotInState(T forbidden); public void assertCanTransition(T fromState, T toState) { assertCanTransition(fromState, toState, false); } public abstract void assertCanTransition(T fromState, T toState, boolean ignoreFailures); public abstract void assertCanTransition(List fromStates, T toState, boolean ignoreFailures); public abstract boolean inStateAndNotTransitioning(T toState); public abstract boolean inStateOrTransitioningTo(T toState); public abstract boolean hasSeenStateAndNotTransitioning(T toState); public abstract boolean hasSeenStateIgnoringTransitions(T toState); public CurrentState failed(ExecutionResult failure) { return new Failed<>(displayName, state, failure); } public RuntimeException rethrow() { throw new IllegalStateException(); } public ExecutionResult asResult() { return ExecutionResult.succeeded(); } public CurrentState transitioningTo(T toState) { return new TransitioningToNewState(toState, this); } public abstract CurrentState nextState(T toState); } /** * Currently in the given state. */ private static class InState extends CurrentState { private final DisplayName displayName; @Nullable private final InState previous; public InState(DisplayName displayName, T state, @Nullable InState previous) { super(displayName, state); this.displayName = displayName; this.previous = previous; } @Override public void assertInState(T expected) { if (state != expected) { throw new IllegalStateException("Expected " + displayName.getDisplayName() + " to be in state " + expected + " but is in state " + state + "."); } } @Override public void assertNotInState(T forbidden) { if (state == forbidden) { throw new IllegalStateException(displayName.getCapitalizedDisplayName() + " should not be in state " + forbidden + "."); } } @Override public void assertCanTransition(T fromState, T toState, boolean ignoreFailures) { if (state != fromState) { throw new IllegalStateException("Can only transition " + displayName.getCapitalizedDisplayName() + " to state " + toState + " from state " + fromState + " however it is currently in state " + state + "."); } } @Override public void assertCanTransition(List fromStates, T toState, boolean ignoreFailures) { if (!fromStates.contains(state)) { throw new IllegalStateException("Can only transition " + displayName.getCapitalizedDisplayName() + " to state " + toState + " from states " + fromStates + " however it is currently in state " + state + "."); } } @Override public boolean inStateAndNotTransitioning(T toState) { return state == toState; } @Override public boolean inStateOrTransitioningTo(T toState) { return inStateAndNotTransitioning(toState); } @Override public boolean hasSeenStateAndNotTransitioning(T toState) { if (state == toState) { return true; } if (previous != null) { return previous.hasSeenStateAndNotTransitioning(toState); } return false; } @Override public boolean hasSeenStateIgnoringTransitions(T toState) { return hasSeenStateAndNotTransitioning(toState); } @Override public CurrentState nextState(T toState) { return new InState<>(displayName, toState, this); } } /** * Currently transitioning to a new state. */ private static class TransitioningToNewState extends CurrentState { final T targetState; final CurrentState fromState; public TransitioningToNewState(T targetState, CurrentState fromState) { super(fromState.displayName, fromState.state); this.targetState = targetState; this.fromState = fromState; } @Override public boolean inStateAndNotTransitioning(T toState) { throw new IllegalStateException("Expected " + displayName.getDisplayName() + " to be in state " + toState + " but is in state " + state + " and transitioning to " + targetState + "."); } @Override public boolean inStateOrTransitioningTo(T toState) { if (targetState == toState) { return true; } throw new IllegalStateException("Expected " + displayName.getDisplayName() + " to be in state " + toState + " but is in state " + state + " and transitioning to " + targetState + "."); } @Override public void assertInState(T expected) { throw new IllegalStateException("Expected " + displayName.getDisplayName() + " to be in state " + expected + " but is in state " + state + " and transitioning to " + targetState + "."); } @Override public void assertNotInState(T forbidden) { throw new IllegalStateException(displayName.getCapitalizedDisplayName() + " should not be in state " + forbidden + " but is in state " + state + " and transitioning to " + targetState + "."); } @Override public void assertCanTransition(T fromState, T toState, boolean ignoreFailures) { failDueToTransition(toState); } @Override public void assertCanTransition(List fromStates, T toState, boolean ignoreFailures) { failDueToTransition(toState); } private void failDueToTransition(T toState) { if (targetState == toState) { throw new IllegalStateException("Cannot transition " + displayName.getDisplayName() + " to state " + toState + " as already transitioning to this state."); } else { throw new IllegalStateException("Cannot transition " + displayName.getDisplayName() + " to state " + toState + " as already transitioning to state " + targetState + "."); } } @Override public boolean hasSeenStateAndNotTransitioning(T toState) { throw new IllegalStateException("Expected " + displayName.getDisplayName() + " to be in state " + toState + " or later but is in state " + state + " and transitioning to " + targetState + "."); } @Override public boolean hasSeenStateIgnoringTransitions(T toState) { return fromState.hasSeenStateIgnoringTransitions(toState); } @Override public CurrentState nextState(T toState) { return fromState.nextState(toState); } } /** * A previous operation has failed. */ private static class Failed extends CurrentState { final ExecutionResult failure; public Failed(DisplayName displayName, T state, ExecutionResult failure) { super(displayName, state); this.failure = failure; } public void throwFailure() { failure.rethrow(); } @Override public void assertInState(T expected) { throwFailure(); } @Override public void assertNotInState(T forbidden) { throwFailure(); } @Override public void assertCanTransition(List fromStates, T toState, boolean ignoreFailures) { if (!ignoreFailures) { throwFailure(); } } @Override public void assertCanTransition(T fromState, T toState, boolean ignoreFailures) { if (!ignoreFailures) { throwFailure(); } } @Override public boolean hasSeenStateAndNotTransitioning(T toState) { throwFailure(); return false; } @Override public ExecutionResult asResult() { return failure.asFailure(); } @Override public boolean inStateAndNotTransitioning(T toState) { throwFailure(); return false; } @Override public boolean inStateOrTransitioningTo(T toState) { throwFailure(); return false; } @Override public boolean hasSeenStateIgnoringTransitions(T toState) { throwFailure(); return false; } @Override public RuntimeException rethrow() { failure.rethrow(); throw new IllegalStateException(); } @Override public CurrentState failed(ExecutionResult failure) { return new Failed<>(displayName, state, this.failure.withFailures(failure.asFailure())); } @Override public CurrentState nextState(T toState) { return new Failed<>(displayName, toState, failure); } } public interface State { } }