blob: c03ed2796f5e391769216508dfc6a18308d3996f [file] [log] [blame]
/*
* 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());
if (rawLockInfo.endsWith("\n")) {
// This is most likely down to trying to open a TDB2 database with TDB1
throw new FileException("Unable to check TDB lock owner, the lock file contents appear to be for a TDB2 database. Please try loading this location as a TDB2 database.");
}
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");
}
}
}