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

com.github.nosan.embedded.cassandra.commons.FileLock Maven / Gradle / Ivy

There is a newer version: 5.0.0
Show newest version
/*
 * Copyright 2020-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
 *
 *      https://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 com.github.nosan.embedded.cassandra.commons;

import java.io.IOException;
import java.nio.channels.FileChannel;
import java.nio.channels.FileLockInterruptionException;
import java.nio.channels.OverlappingFileLockException;
import java.nio.file.Path;
import java.nio.file.StandardOpenOption;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit;

/**
 * Helper class to lock a file. It is recommended practice to lock a file with a try with resources block, such as:
 * 
 {@code
 * class X {
 *   public void m() {
 *     try(FileLock lock = FileLock.of(lockFile)){
 *         if(lock.tryLock(1, TimeUnit.MINUTES){
 *             // locked
 *         } else {
 *             // not locked
 *         }
 *     }
 * }}}
 * 
* An instance of this class must not be shared across different threads. * * @author Dmytro Nosan * @since 4.0.0 */ public final class FileLock implements AutoCloseable { private final FileChannel fileChannel; private final Map locks = new ConcurrentHashMap<>(); private FileLock(FileChannel fileChannel) { this.fileChannel = fileChannel; } /** * Creates a new {@link FileLock} instance for the specified file. * * @param file the file that should be locked * @return a new {@link FileLock} * @throws IOException in the case of I/O errors */ public static FileLock of(Path file) throws IOException { Objects.requireNonNull(file, "File must not be null"); return new FileLock(FileChannel.open(file, StandardOpenOption.WRITE, StandardOpenOption.CREATE)); } /** * Acquires an exclusive lock on the file. * * @param timeout the maximum time to wait * @param timeUnit the time unit of the {@code timeout} argument * @return {@code true} if lock has been acquired otherwise {@code false} * @throws FileLockInterruptionException If the invoking thread is interrupted while blocked in this method * @throws IOException If some other I/O error occurs */ public synchronized boolean tryLock(long timeout, TimeUnit timeUnit) throws FileLockInterruptionException, IOException { Objects.requireNonNull(timeUnit, "TimeUnit must not be null"); if (timeout < 0) { throw new IllegalArgumentException("Timeout must not be negative"); } java.nio.channels.FileLock fileLock = this.locks.get(Thread.currentThread()); if (fileLock != null && fileLock.isValid()) { return true; } long startTime = System.nanoTime(); long rem = timeUnit.toNanos(timeout); do { fileLock = lock(this.fileChannel); if (fileLock != null) { this.locks.put(Thread.currentThread(), fileLock); return true; } if (rem > 0) { try { Thread.sleep(Math.min(TimeUnit.NANOSECONDS.toMillis(rem) + 1, 100)); } catch (InterruptedException ex) { throw new FileLockInterruptionException(); } } rem = timeUnit.toNanos(timeout) - (System.nanoTime() - startTime); } while (rem > 0); return false; } /** * Closes the underlying {@link FileChannel} and releases all locks. * * @throws IOException If some other I/O error occurs */ @Override public synchronized void close() throws IOException { this.fileChannel.close(); this.locks.clear(); } private static java.nio.channels.FileLock lock(FileChannel fileChannel) throws IOException { try { return fileChannel.tryLock(); } catch (OverlappingFileLockException ex) { return null; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy