org.exist.util.Leasable Maven / Gradle / Ivy
/*
* eXist Open Source Native XML Database
* Copyright (C) 2001-2018 The eXist Project
* http://exist-db.org
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
package org.exist.util;
import net.jcip.annotations.GuardedBy;
import net.jcip.annotations.ThreadSafe;
import javax.annotation.Nullable;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.function.Consumer;
/**
* A container for an Object upon which
* leases may be taken and returned.
*
* The container keeps a reference count of
* the number of active leases.
*
* If a `closer` is provided then when the
* number of active leases returns to zero,
* the closer is invoked. Once the closer is
* invoked, all subsequent leases of the object
* are invalid.
*
* This is useful for when you have an object which
* makes use of resources that have to be freed when
* you are finished with the object, but you want to be
* able to freely share the object around.
*
* @param The type of the leasable object.
*
* @author Adam Retter
*/
@ThreadSafe
public class Leasable {
private final T object;
@Nullable private final Consumer closer;
private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
@GuardedBy("lock") private int leases;
@GuardedBy("lock") private boolean closed;
public Leasable(final T object) {
this(object, null);
}
public Leasable(final T object, @Nullable final Consumer closer) {
this.object = object;
this.closer = closer;
}
/**
* Creates a {@code Leasable} from an {@code U extends AutoCloseable}.
*
* The {@link AutoCloseable#close()} method will be involed as the `closer`
* and exception thrown by {@link AutoCloseable#close()} will be promoted
* to an {@link IllegalLeasableState} exception.
*
*
* @param the type of the auto-closeable object.
*
* @param object The object to setup for leasing.
*
* @return The leasable object.
*/
public static Leasable fromCloseable(final U object) {
return new Leasable<>(object, _object -> {
try {
_object.close();
} catch (final Exception e) {
throw new IllegalLeasableState(e);
}
});
}
/**
* Take a lease on the object.
*
* This will increment the reference
* count of the object by 1.
*
* Note, callers are expected to call
* {@link Lease#close()} when they are
* finished with the lease.
*
* @return a lease of the object.
*/
public Lease lease() {
lock.writeLock().lock();
try {
if (closed) {
throw new IllegalLeasableState("Object is closed");
}
final Lease lease = new Lease();
leases++;
return lease;
} finally {
lock.writeLock().unlock();
}
}
/**
* Determines if the object is leased.
*
* @return true if the object is leased.
*/
public boolean isLeased() {
lock.readLock().lock();
try {
if (closed) {
return false;
}
return leases > 0;
} finally {
lock.readLock().unlock();
}
}
/**
* Represents a lease of the leasable object.
*/
public class Lease implements AutoCloseable {
/**
* Access to the leased object.
*
* @return the leased object.
*/
public T get() {
lock.readLock().lock();
try {
if (closed) {
throw new IllegalLeasableState("Object is closed");
}
if(leases == 0) {
throw new IllegalLeasableState("Lease was returned");
}
return object;
} finally {
lock.readLock().unlock();
}
}
/**
* Releases the lease on the leasable object.
*/
@Override
public void close() {
lock.writeLock().lock();
try {
if (closed) {
throw new IllegalLeasableState("Object is closed");
}
if(leases == 0) {
throw new IllegalLeasableState("Lease was already returned");
}
leases--;
if (leases == 0 && closer != null) {
closer.accept(object);
closed = true;
}
} finally {
lock.writeLock().unlock();
}
}
}
/**
* Represents an illegal state of the leasable/leased object.
*/
public static class IllegalLeasableState extends IllegalStateException {
public IllegalLeasableState(final String message) {
super(message);
}
public IllegalLeasableState(final Throwable cause) {
super(cause);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy