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

io.hekate.util.StateGuard Maven / Gradle / Ivy

/*
 * Copyright 2022 The Hekate Project
 *
 * The Hekate Project licenses this file to you 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 io.hekate.util;

import io.hekate.util.async.Waiting;
import io.hekate.util.format.ToString;
import io.hekate.util.format.ToStringIgnore;
import java.util.List;
import java.util.Locale;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock.ReadLock;
import java.util.concurrent.locks.ReentrantReadWriteLock.WriteLock;

import static io.hekate.util.StateGuard.State.INITIALIZED;
import static io.hekate.util.StateGuard.State.INITIALIZING;
import static io.hekate.util.StateGuard.State.TERMINATED;
import static io.hekate.util.StateGuard.State.TERMINATING;

/**
 * Helper class that provides support for components lifecycle management and locking.
 *
 * 

* Internally this class is backed by {@link ReentrantReadWriteLock}. *

*/ public class StateGuard { /** * Enumeration of {@link StateGuard} states. */ public enum State { /** See {@link StateGuard#becomeInitializing()}. */ INITIALIZING, /** See {@link StateGuard#becomeInitialized()}. */ INITIALIZED, /** See {@link StateGuard#becomeTerminating()}. */ TERMINATING, /** See {@link StateGuard#becomeTerminated()}. */ TERMINATED } /** * Guarded supplier. * * @param Result type. * @param Error type. */ @FunctionalInterface public interface GuardedSupplier { /** * Returns a value. * * @return Value. * * @throws E Error. */ T get() throws E; } /** * Guarded runnable. * * @param Error type. */ @FunctionalInterface public interface GuardedRunnable { /** * Runs this task. * * @throws E Error. */ void run() throws E; } private final Class type; @ToStringIgnore private final ReadLock readLock; @ToStringIgnore private final WriteLock writeLock; private State state = TERMINATED; /** * Constructs a new instance. * * @param type Guarded component type (for errors reporting). */ public StateGuard(Class type) { this.type = type; ReentrantReadWriteLock lock = new ReentrantReadWriteLock(); readLock = lock.readLock(); writeLock = lock.writeLock(); } /** * Conditionally executes the specified task in the {@link #readLock read}-locked context if this guard is in {@link #isInitialized() * initialized} state. * * @param task Task. * @param Error type. * * @return {@code true} if task was executed or {@code false} if this guard is not in the {@link #isInitialized() initialized} state. * * @throws E Task execution error. */ public boolean withReadLockIfInitialized(GuardedRunnable task) throws E { lockRead(); try { if (isInitialized()) { task.run(); return true; } } finally { unlockRead(); } return false; } /** * Conditionally executes the specified task in the {@link #writeLock write}-locked context if this guard is in {@link #isInitialized() * initialized} state. * * @param task Task. * @param Error type. * * @return {@code true} if task was executed or {@code false} if this guard is not in the {@link #isInitialized() initialized} state. * * @throws E Task execution error. */ public boolean withWriteLockIfInitialized(GuardedRunnable task) throws E { lockWrite(); try { if (isInitialized()) { task.run(); return true; } } finally { unlockWrite(); } return false; } /** * Executes the specified task while holding the {@link #lockRead() read} lock. * * @param task Task. * @param Error type. * * @throws E Task execution error. */ public void withReadLock(GuardedRunnable task) throws E { lockRead(); try { task.run(); } finally { unlockRead(); } } /** * Executes the specified task while holding the {@link #lockRead() read} lock. * * @param task Task. * @param Result type. * @param Error type. * * @return Task execution result. * * @throws E Task execution error. */ public T withReadLock(GuardedSupplier task) throws E { lockRead(); try { return task.get(); } finally { unlockRead(); } } /** * Executes the specified task while holding the {@link #lockWrite() write} lock. * * @param task Task. * @param Error type. * * @throws E Task execution error. */ public void withWriteLock(GuardedRunnable task) throws E { lockWrite(); try { task.run(); } finally { unlockWrite(); } } /** * Executes the specified task while holding the {@link #lockWrite() write} lock. * * @param task Task. * @param Result type. * @param Error type. * * @return Task execution result. * * @throws E Task execution error. */ public T withWriteLock(GuardedSupplier task) throws E { lockWrite(); try { return task.get(); } finally { unlockWrite(); } } /** * Switches this guard to {@link State#INITIALIZING} state. * *

* Important!!! This method must be invoked with {@link #lockWrite() write lock} being held. *

*/ public void becomeInitializing() { assert writeLock.isHeldByCurrentThread() : "Thread must hold write lock."; if (state == INITIALIZING || state == INITIALIZED) { throw new IllegalStateException(type.getSimpleName() + " is already " + state.name().toLowerCase(Locale.US) + '.'); } state = INITIALIZING; } /** * Returns {@code true} if this guard is in {@link State#INITIALIZING} state. * *

* Important!!! This method must be invoked with {@link #lockWrite() write lock} or {@link #lockRead() read lock} being held. *

* * @return {@code true} if this guard is in {@link State#INITIALIZING} state. */ public boolean isInitializing() { return state == INITIALIZING; } /** * Switches this guard to {@link State#INITIALIZED} state. * *

* Important!!! This method must be invoked with {@link #lockWrite() write lock} being held. *

*/ public void becomeInitialized() { assert writeLock.isHeldByCurrentThread() : "Thread must hold write lock."; if (state == INITIALIZED) { throw new IllegalStateException(type.getSimpleName() + " already initialized."); } state = INITIALIZED; } /** * Switches this guard to {@link State#INITIALIZED} state and executes the specified task. * * @param task Task. * @param Error type. * * @throws E Task execution error. */ public void becomeInitialized(GuardedRunnable task) throws E { withWriteLock(() -> { becomeInitialized(); task.run(); }); } /** * Returns {@code true} if this guard is in {@link State#INITIALIZED} state. * *

* Important!!! This method must be invoked with {@link #lockWrite() write lock} or {@link #lockRead() read lock} being held. *

* * @return {@code true} if this guard is in {@link State#INITIALIZED} state. */ public boolean isInitialized() { return state == INITIALIZED; } /** * Switches this guard to {@link State#TERMINATING} state if guard is currently in {@link State#INITIALIZING} or {@link * State#INITIALIZED} state. * *

* Important!!! This method must be invoked with {@link #lockWrite() write lock} being held. *

* * @return {@code true} if successfully switched to {@link State#TERMINATING} state. */ public boolean becomeTerminating() { assert writeLock.isHeldByCurrentThread() : "Thread must hold write lock."; if (state != TERMINATED && state != TERMINATING) { state = TERMINATING; return true; } return false; } /** * Tries to switch this guard to {@link State#TERMINATING} state and executes the specified task if switch happened. * * @param task Task. * @param Error type. * * @return {@link Waiting} object to await for the task execution. * * @throws E Task execution error. * @see #becomeTerminating() */ public Waiting becomeTerminating(GuardedSupplier, E> task) throws E { return withWriteLock(() -> { if (becomeTerminating()) { List result = task.get(); return result != null && !result.isEmpty() ? Waiting.awaitAll(result) : Waiting.NO_WAIT; } else { return Waiting.NO_WAIT; } }); } /** * Tries to switch this guard to {@link State#TERMINATING} state and executes the specified task if switch happened. * * @param task Task. * @param Error type. * * @throws E Task execution error. * @see #becomeTerminating() */ public void becomeTerminating(GuardedRunnable task) throws E { withWriteLock(() -> { if (becomeTerminating()) { task.run(); } }); } /** * Switches this guard to {@link State#TERMINATED} state if guard is currently in any other state besides {@link State#TERMINATED}. * *

* Important!!! This method must be invoked with {@link #lockWrite() write lock} being held. *

* * @return {@code true} if successfully switched to {@link State#TERMINATED} state. */ public boolean becomeTerminated() { assert writeLock.isHeldByCurrentThread() : "Thread must hold write lock."; if (state != TERMINATED) { state = TERMINATED; return true; } return false; } /** * Switches this guard to {@link State#TERMINATED} state and executes the specified task. * * @param task Task. * @param Error type. * * @return {@link Waiting} object to await for the task execution. * * @throws E Task execution error. */ public Waiting becomeTerminated(GuardedSupplier, E> task) throws E { return withWriteLock(() -> { if (becomeTerminated()) { List result = task.get(); return result != null && !result.isEmpty() ? Waiting.awaitAll(result) : Waiting.NO_WAIT; } else { return Waiting.NO_WAIT; } }); } /** * Switches this guard to {@link State#TERMINATED} state and executes the specified task. * * @param task Task. * @param Error type. * * @throws E Task execution error. */ public void becomeTerminated(GuardedRunnable task) throws E { withWriteLock(() -> { if (becomeTerminated()) { task.run(); } }); } /** * Acquires the read lock on this guard and checks its current state. If state is anything other than {@link State#INITIALIZED} then * {@link IllegalStateException} will be thrown and lock will be released. * * @throws IllegalStateException If guard is not in {@link State#INITIALIZED} state. */ public void lockReadWithStateCheck() { lockReadWithStateCheck(INITIALIZED); } /** * Acquires the read lock on this guard and checks its current state. If state is anything other than the specified {@code state} then * {@link IllegalStateException} will be thrown and lock will be released. * * @param state Expected state. * * @throws IllegalStateException If the specified {@code state} doesn't match the current state of this guard. */ public void lockReadWithStateCheck(State state) { lockRead(); if (this.state != state) { unlockRead(); throw new IllegalStateException(type.getSimpleName() + " is not " + state.name().toLowerCase(Locale.US) + '.'); } } /** * Executes the specified task while holding the {@link #lockRead() read} lock. * *

* If the current state is anything other than {@link State#INITIALIZED} then {@link IllegalStateException} will be thrown and lock will * be released without executing the task. *

* * @param task Task. * @param Result type. * @param Task execution error type. * * @return Task execution result. * * @throws E Task execution error. */ public T withReadLockAndStateCheck(GuardedSupplier task) throws E { lockReadWithStateCheck(); try { return task.get(); } finally { unlockRead(); } } /** * Executes the specified task while holding the {@link #lockRead() read} lock. * *

* If the current state is anything other than {@link State#INITIALIZED} then {@link IllegalStateException} will be thrown and lock will * be released without executing the task. *

* * @param task Task. * @param Task execution error type. * * @throws E Task execution error. */ public void withReadLockAndStateCheck(GuardedRunnable task) throws E { lockReadWithStateCheck(); try { task.run(); } finally { unlockRead(); } } /** * Tries to acquire the read lock on this guard only if current state is {@link State#INITIALIZED}. * * @return {@code true} if lock was acquired. */ public boolean tryLockReadWithStateCheck() { lockRead(); if (state != INITIALIZED) { unlockRead(); return false; } return true; } /** * Tries to acquire the read lock on this guard only if current state is the same with the specified one. * * @param state Expected state. * * @return {@code true} if lock was acquired. */ public boolean tryLockReadWithStateCheck(State state) { lockRead(); if (this.state != state) { unlockRead(); return false; } return true; } /** * Acquires the read lock. * * @see ReadLock#lock() */ public void lockRead() { readLock.lock(); } /** * Tries to acquire the read lock with the specified timeout. * * @param timeout Timeout. * @param unit Unit of timeout. * * @return {@code true} if lock was successfully acquired. * * @throws InterruptedException If thread was interrupted while waiting for lock acquisition. * @see ReadLock#tryLock(long, TimeUnit) */ public boolean tryLockRead(long timeout, TimeUnit unit) throws InterruptedException { return readLock.tryLock(timeout, unit); } /** * Releases the read lock. * * @see ReadLock#unlock() */ public void unlockRead() { readLock.unlock(); } /** * Acquires the write lock on this guard and checks its current state. If state is anything other than {@link State#INITIALIZED} then * {@link IllegalStateException} will be thrown and lock will be released. * * @throws IllegalStateException If guard is not in {@link State#INITIALIZED} state. */ public void lockWriteWithStateCheck() { lockWrite(); if (state != INITIALIZED) { unlockWrite(); throw new IllegalStateException(type.getSimpleName() + " is not " + INITIALIZED.name().toLowerCase(Locale.US) + '.'); } } /** * Acquires the write lock. * * @see WriteLock#lock() */ public void lockWrite() { writeLock.lock(); } /** * Tries to acquire the write lock with the specified timeout. * * @param timeout Timeout. * @param unit Unit of timeout. * * @return {@code true} if lock was successfully acquired. * * @throws InterruptedException If thread was interrupted while awaiting for lock acquisition. * @see WriteLock#tryLock(long, TimeUnit) */ public boolean tryLockWrite(long timeout, TimeUnit unit) throws InterruptedException { return writeLock.tryLock(timeout, unit); } /** * Releases the write lock. * * @see WriteLock#unlock() */ public void unlockWrite() { writeLock.unlock(); } /** * Returns {@code true} if the current thread holds a write lock. * * @return {@code true} if the current thread holds a write lock. * * @see WriteLock#isHeldByCurrentThread() */ public boolean isWriteLocked() { return writeLock.isHeldByCurrentThread(); } @Override public String toString() { readLock.lock(); try { return ToString.format(this); } finally { readLock.unlock(); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy