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();
}
}
}