org.glassfish.common.util.admin.ManagedFile Maven / Gradle / Ivy
/*
* Copyright (c) 2010, 2018 Oracle and/or its affiliates. All rights reserved.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0, which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the
* Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
* version 2 with the GNU Classpath Exception, which is available at
* https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
*/
package org.glassfish.common.util.admin;
import com.sun.enterprise.util.CULoggerInfo;
import com.sun.enterprise.util.LocalStringManagerImpl;
import java.io.File;
import java.io.IOException;
import java.io.RandomAccessFile;
import java.nio.channels.FileChannel;
import java.nio.channels.FileLock;
import java.nio.channels.OverlappingFileLockException;
import java.util.Queue;
import java.util.Timer;
import java.util.TimerTask;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.logging.Level;
import java.util.logging.Logger;
/**
* Defines the notion of a managed file with a classic Read-Write locking policy.
* A managed file can be locked for multiple concurrent reads or a single write.
*
* A simple example could follow this :
*
* ManagedFile managedFile = new ManagedFile(new File(...), 1000, -1);
* Lock writeLock;
* try {
* writeLock = managedFile.writeAccess();
* // write or delete the file
* } finally {
* writeLock.unlock();
* }
*
* @author Jerome Dochez
*/
public class ManagedFile {
final File file;
final int maxHoldingTime;
final int timeOut;
final ReentrantReadWriteLock rwl = new ReentrantReadWriteLock();
final ManagedFile.RefCounterLock rl = new ManagedFile.RefCounterLock(rwl.readLock(), true);
final ManagedFile.RefCounterLock wl = new ManagedFile.RefCounterLock(rwl.writeLock(), false);
final Queue waiters = new ConcurrentLinkedQueue();
final static Logger logger = CULoggerInfo.getLogger();
final static LocalStringManagerImpl localStrings =
new LocalStringManagerImpl(ParamTokenizer.class);
public interface ManagedLock extends java.util.concurrent.locks.Lock {
public RandomAccessFile getLockedFile();
}
/**
* Creates a new managed file.
*
* @param file the file to manage
* @param timeOut the max time in milliseconds to wait for a read or write lock
* @param maxHoldingTime the max time in milliseconds to hold the read or write lock
* @throws IOException when the file cannot be locked
*/
public ManagedFile(File file, int timeOut, int maxHoldingTime) throws IOException {
this.file = file;
this.maxHoldingTime = maxHoldingTime;
this.timeOut = timeOut;
}
/**
* Blocks for {@link ManagedFile#timeOut} milliseconds for the write access
* to the managed file.
*
* @return the lock instance on the locked file.
* @throws IOException if the file cannot be locked
* @throws TimeoutException if the lock cannot be obtained before the timeOut
* expiration.
*/
public ManagedLock accessWrite() throws IOException, TimeoutException {
wl._lock();
return wl;
}
/**
* Blocks for {@link ManagedFile#timeOut} milliseconds for the read access
* to the managed file.
*
* @return the lock instance on the locked file.
* @throws IOException if the file cannot be locked
* @throws TimeoutException if the lock cannot be obtained before the timeOut
* expiration.
*/
public ManagedLock accessRead() throws IOException, TimeoutException {
rl._lock();
return rl;
}
/**
* Many threads can be requesting the shared read lock, we must keep track
* through a reference counter when all these users are done. Each thread
* must call {@link RefCounterLock#unlock()} to release the lock, when the
* reference counter returns to zero, the file lock is release.
*/
private class RefCounterLock implements ManagedLock {
final java.util.concurrent.locks.Lock lock;
final boolean read;
final AtomicInteger refs = new AtomicInteger(0);
FileLock fileLock;
RandomAccessFile raf;
FileChannel fc;
Timer timer;
public synchronized RandomAccessFile getLockedFile() {
return raf;
}
private FileLock get(FileChannel fc, boolean shared) throws IOException, TimeoutException {
FileLock fl;
boolean wasInterrupted = false;
Thread current = Thread.currentThread();
waiters.add(current);
// calculate how much time we want to be blocked for
long endTime = System.currentTimeMillis() + timeOut;
// so far, we wait in 1/10th increment of the requested timeOut.
final int individualWaitTime = timeOut / 10;
// Block while not first in queue or cannot acquire lock
while (waiters.peek() != current ||
(fl = getLock(fc, shared)) == null) {
// I cannot just park the thread and signal it since the
// the lock maybe owned by a different process. I just need
// to wait...
if (logger.isLoggable(Level.FINE)) {
logger.fine("Waiting..." + individualWaitTime);
}
if (System.currentTimeMillis() > endTime) {
throw new TimeoutException(localStrings.getLocalString("FileLockTimeOut",
"time out expired on locking {0}", file.getPath()));
}
try {
Thread.sleep(individualWaitTime);
} catch (InterruptedException e) {
wasInterrupted = true;
}
if (Thread.interrupted()) // ignore interrupts while waiting
wasInterrupted = true;
}
waiters.remove();
if (wasInterrupted) // reassert interrupt status on exit
current.interrupt();
return fl;
}
private FileLock getLock(FileChannel fc, boolean shared) throws IOException {
try {
return fc.lock(0, Long.MAX_VALUE, shared);
} catch (OverlappingFileLockException e) {
return null;
}
}
private synchronized FileLock access(boolean shared, String mode, int timeOut) throws IOException, TimeoutException {
raf = new RandomAccessFile(file, mode);
fc = raf.getChannel();
final FileLock fl = get(fc, shared);
if (maxHoldingTime != -1) {
timer = new Timer();
timer.schedule(new TimerTask() {
@Override
public void run() {
try {
if (fl.isValid()) {
logger.log(Level.SEVERE,
CULoggerInfo.fileLockNotReleased,
file.getPath());
release(fl);
}
} catch (IOException e) {
e.printStackTrace();
}
}
}, timeOut);
}
return fl;
}
private synchronized void release(FileLock lock) throws IOException {
lock.release();
if (timer != null) {
timer.cancel();
timer = null;
}
if (raf != null) {
raf.close();
raf = null;
}
if (fc != null) {
fc.close();
fc = null;
}
}
private RefCounterLock(java.util.concurrent.locks.Lock lock, boolean read) {
this.lock = lock;
this.read = read;
}
public void _lock() throws IOException, TimeoutException {
lock.lock();
if (refs.incrementAndGet() == 1) {
// create the file lock.
if (read) {
fileLock = access(true, "r", maxHoldingTime);
} else {
fileLock = access(false, "rw", maxHoldingTime);
}
}
}
@Override
public void lock() {
throw new UnsupportedOperationException();
}
@Override
public void lockInterruptibly() throws InterruptedException {
lock.lockInterruptibly();
refs.incrementAndGet();
}
@Override
public boolean tryLock() {
boolean result = lock.tryLock();
if (result) refs.incrementAndGet();
return result;
}
@Override
public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
boolean result = lock.tryLock(time, unit);
if (result) refs.incrementAndGet();
return result;
}
@Override
public void unlock() {
lock.unlock();
if (refs.decrementAndGet() == 0) {
try {
release(fileLock);
} catch (IOException e) {
e.printStackTrace();
}
}
}
@Override
public Condition newCondition() {
return lock.newCondition();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy