/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF 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 org.apache.jena.tdb.base.file; import java.io.BufferedWriter; import java.io.File; import java.io.FileWriter; import java.io.IOException; import org.apache.jena.atlas.io.IO; import org.apache.jena.tdb.TDBException; import org.apache.jena.tdb.sys.ProcessUtils; import org.apache.jena.tdb.sys.SystemTDB; /** * Represents a lock on a TDB location * */ public class LocationLock { private static final int NO_OWNER = 0; private static final String LOCK_FILENAME = "tdb.lock"; private Location location; public LocationLock(Location location) { if (location == null) throw new NullPointerException("Location cannot be null"); this.location = location; } /** * Gets whether the location is lockable i.e. is it an on-disk location * where we can use a lock file to prevent multi-process access and the * potential data corruption that ensues * * @return True if the location is lockable */ public boolean canLock() { return !location.isMem(); } /** * Gets whether the location is currently locked * * @return boolean */ public boolean isLocked() { // Memory locations are never considered locked if (location.isMem()) return false; return getOwner() != NO_OWNER; } /** * Gets whether the lock is owned by the current process * * @return True if the location is locked and owned by the process, false * otherwise */ public boolean isOwned() { // Memory locations are never considered locked if (location.isMem()) return false; int owner = getOwner(); if (owner == NO_OWNER) return false; return owner == ProcessUtils.getPid(NO_OWNER); } /** * Gets the current owner of this locations lock. * * @return Process ID of owner if locked, zero if the location cannot be * locked or not currently locked */ public int getOwner() { // Memory locations are never considered locked if (location.isMem()) return NO_OWNER; File lockFile = getLockFile(); if (lockFile.exists()) { checkLockFileForRead(lockFile); // Can read lock owner from the file try { String rawLockInfo = IO.readWholeFileAsUTF8(lockFile.getAbsolutePath()); int owner = Integer.parseInt(rawLockInfo); return owner; } catch (IOException e) { throw new FileException("Unable to check TDB lock owner due to an IO error reading the lock file", e); } catch (NumberFormatException e) { throw new FileException("Unable to check TDB lock owner as the lock file contains invalid data", e); } } else { // No lock file so nobody owns the lock currently return NO_OWNER; } } /** * Gets whether the current JVM can obtain the lock * * @return True if the lock can be obtained (or is already held), false * otherwise */ public boolean canObtain() { // Memory locations cannot be locked if (location.isMem()) return false; int owner = this.getOwner(); int pid = ProcessUtils.getPid(NO_OWNER); if (owner == NO_OWNER) { // Can obtain provided we have a valid PID if (pid != NO_OWNER) return true; } else if (owner == pid) { // Already held return true; } // Owned by another process, only obtainable if other process is dead if (!ProcessUtils.isAlive(owner)) return true; // Otherwise not obtainable return false; } /** * Obtains the lock in order to prevent other JVMs using the location */ public void obtain() { // Memory locations cannot be locked so nothing to do if (location.isMem()) return; // Get current owner int owner = this.getOwner(); int pid = ProcessUtils.getPid(NO_OWNER); if (owner == NO_OWNER) { // No owner currently so try to obtain the lock if (pid == NO_OWNER) { // In the case where we cannot obtain our PID then we cannot // obtain a lock SystemTDB.errlog.warn("Location " + location.getDirectoryPath() + " cannot be locked as unable to obtain PID of current process, if another JVM accessed this location while this process is accessing it then data corruption may occur"); return; } takeLock(pid); } else if (owner == ProcessUtils.getPid(NO_OWNER)) { // We already own the lock so nothing to do } else { // Someone other process potentially owns the lock on this location // Check if the owner is alive if (ProcessUtils.isAlive(owner)) { throw new TDBException("Location " + location.getDirectoryPath() + " is currently locked by PID " + owner + "(this process is PID "+pid+")" + ". TDB databases do not permit concurrent usage across JVMs so in order to prevent possible data corruption you cannot open this location from the JVM that does not own the lock for the dataset"); } // Otherwise the previous owner is dead so we can take the lock takeLock(ProcessUtils.getPid(NO_OWNER)); } } private void takeLock(int pid) { File lockFile = getLockFile(); checkLockFileForWrite(lockFile); // Write our PID to the lock file try (BufferedWriter writer = new BufferedWriter(new FileWriter(lockFile))) { writer.write(Integer.toString(pid)); writer.close(); } catch (IOException e) { throw new TDBException("Failed to obtain a lock on the location " + location.getDirectoryPath(), e); } // Mark lock for deletion on JVM exit // This does not guarantee that the lock file gets cleaned up because // such deletions only succeed for normal JVM termination but it should // clean up the lock for normal JVM terminations lockFile.deleteOnExit(); } /** * Releases the lock so that other JVMs can use the location */ public void release() { // Memory locations cannot be locked so nothing to do if (location.isMem()) return; int owner = this.getOwner(); // Nobody owned the lock so nothing to do if (owner == NO_OWNER) return; // Some other process owns the lock so we can't release it if (owner != ProcessUtils.getPid(NO_OWNER)) throw new TDBException("Cannot release the lock on location " + location.getDirectoryPath() + " since this process does not own the lock"); File lockFile = getLockFile(); // No lock file exists so nothing to do if (!lockFile.exists()) return; // Try and delete the lock file thereby releasing the lock if (!lockFile.delete()) throw new TDBException("Failed to release the lock on location " + location.getDirectoryPath() + ", it may be necessary to manually remove the lock file"); } /** * Gets the lock file * * @return Lock file */ private File getLockFile() { return new File(location.getPath(LOCK_FILENAME)); } /** * Checks if a lock file is valid throwing an exception if it is not * * @param lockFile * Lock file * @exception FileException * Thrown if the lock file is invalid */ private void checkLockFileForRead(File lockFile) { if (!lockFile.exists()) return; if (!lockFile.isFile() || !lockFile.canRead()) { // Unable to read lock owner because it isn't a file or we don't // have read permission throw new FileException( "Unable to check TDB lock owner for this location since the expected lock file is not a file/not readable"); } } /** * Checks if a lock file is valid throwing an exception if it is not * * @param lockFile * Lock file * @exception FileException * Thrown if the lock file is invalid */ private void checkLockFileForWrite(File lockFile) { if (!lockFile.exists()) return; if (!lockFile.isFile() || !lockFile.canWrite()) { // Unable to read lock owner because it isn't a file or we don't // have read permission throw new FileException( "Unable to check TDB lock owner for this location since the expected lock file is not a file/not writable"); } } }