org.apache.jena.tdb.StoreConnection Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jena-tdb Show documentation
Show all versions of jena-tdb Show documentation
TDB is a storage subsystem for Jena and ARQ, it is a native triple store providing persistent storage of triples/quads.
/*
* 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 ;
import java.util.HashMap ;
import java.util.HashSet ;
import java.util.Map ;
import java.util.Set ;
import org.apache.jena.query.ReadWrite ;
import org.apache.jena.query.TxnType;
import org.apache.jena.sparql.mgt.ARQMgt ;
import org.apache.jena.tdb.base.file.ChannelManager ;
import org.apache.jena.tdb.base.file.Location ;
import org.apache.jena.tdb.base.file.LocationLock ;
import org.apache.jena.tdb.setup.DatasetBuilderStd ;
import org.apache.jena.tdb.setup.StoreParams ;
import org.apache.jena.tdb.store.DatasetGraphTDB ;
import org.apache.jena.tdb.sys.ProcessUtils;
import org.apache.jena.tdb.sys.SystemTDB ;
import org.apache.jena.tdb.transaction.* ;
/** A StoreConnection is the reference to the underlying storage.
* There is JVM-wide cache of backing datasets.
* The work of transaction coordination is done in TransactionManager.
*/
public class StoreConnection
{
private final TransactionManager transactionManager ;
private final DatasetGraphTDB baseDSG ;
private boolean isValid = true ;
private volatile boolean haveUsedInTransaction = false ;
private StoreConnection(DatasetGraphTDB dsg) {
baseDSG = dsg;
transactionManager = new TransactionManager(baseDSG);
}
public boolean isValid() { return isValid ; }
private void checkValid() {
if (!isValid)
throw new TDBTransactionException("StoreConnection inValid (issued before a StoreConnection.release?)") ;
}
// Ensure that a dataset used non-transactionally has been flushed to disk
private void checkTransactional() {
// Access to booleans is atomic.
if ( !haveUsedInTransaction ) {
// See http://en.wikipedia.org/wiki/Double-checked_locking
// Except we don't have the delayed constructor problem.
synchronized (this) {
if ( !haveUsedInTransaction ) {
// Sync the underlying database in case used
// non-transactionally by the application.
baseDSG.sync();
}
haveUsedInTransaction = true;
}
}
}
public boolean haveUsedInTransaction() { return haveUsedInTransaction ; }
public Location getLocation() {
checkValid();
return baseDSG.getLocation();
}
/** Return a description of the transaction manager state */
public SysTxnState getTransMgrState() {
checkValid();
return transactionManager.state();
}
/**
* @deprecated Use {@link #begin(TxnType)}
*/
@Deprecated
public DatasetGraphTxn begin(ReadWrite mode) {
return begin(TxnType.convert(mode));
}
/**
* Begin a transaction. Terminate a write transaction with
* {@link Transaction#commit()} or {@link Transaction#abort()}.
* Terminate a write transaction with {@link Transaction#close()}.
*/
public DatasetGraphTxn begin(TxnType mode) {
checkValid();
checkTransactional();
haveUsedInTransaction = true;
return transactionManager.begin(mode, null);
}
/**
* @deprecated Use {@link #begin(TxnType, String)}
*/
@Deprecated
public DatasetGraphTxn begin(ReadWrite mode, String label) {
return begin(TxnType.convert(mode), label);
}
/**
* Begin a transaction, giving it a label. Terminate a write transaction
* with {@link Transaction#commit()} or {@link Transaction#abort()}.
* Terminate a write transaction with {@link Transaction#close()}.
*/
public DatasetGraphTxn begin(TxnType mode, String label) {
checkValid();
checkTransactional();
return transactionManager.begin(mode, label);
}
/**
* Internal operation - to get a dataset for application use, call a
* {@link TDBFactory} function. Do not use the base dataset without knowing how the
* transaction system uses it.
*/
public DatasetGraphTDB getBaseDataset() {
checkValid();
return baseDSG;
}
/** For internal use only */
public TransactionManager getTransactionManager() {
return transactionManager ;
}
/** Flush the delayed write queue to the base storage.
* This can only be done if there are no active transactions.
* If there are active transactions, nothing is done but this is safe to call.
*/
public void flush() {
if ( !haveUsedInTransaction() )
return;
checkValid();
transactionManager.flush();
}
/** Indicate whether there are any active transactions.
* @see #getTransMgrState
*/
public boolean activeTransactions() {
checkValid();
return transactionManager.activeTransactions();
}
/** Flush the journal regardless - use with great case - do not use when transactions may be active. */
public void forceRecoverFromJournal() {
JournalControl.recoverFromJournal(getBaseDataset().getConfig(), transactionManager.getJournal());
}
/** Highly risky! */
public void printJournal() {
JournalControl.print(transactionManager.getJournal());
}
private static Map cache = new HashMap<>() ;
// ---- statics managing the cache.
/** Obtain a StoreConnection for a particular location */
public static StoreConnection make(String location) {
return make(Location.create(location));
}
/** Stop managing all locations. Use with great care. */
public static synchronized void reset() {
// Copy to avoid potential CME.
Set x = new HashSet<>(cache.keySet()) ;
for (Location loc : x)
expel(loc, true) ;
cache.clear() ;
}
/** Stop managing a location. There should be no transactions running. */
public static synchronized void release(Location location) {
expel(location, false);
}
/** Stop managing a location. Use with great care (testing only). */
public static synchronized void expel(Location location, boolean force) {
StoreConnection sConn = cache.get(location) ;
if (sConn == null)
return ;
if (!force && sConn.transactionManager.activeTransactions())
throw new TDBTransactionException("Can't expel: Active transactions for location: " + location) ;
// No transactions at this point (or we don't care and are clearing up forcefully.)
sConn.transactionManager.closedown() ;
sConn.baseDSG.close() ;
sConn.isValid = false ;
cache.remove(location) ;
ChannelManager.release(sConn.transactionManager.getJournal().getFilename()) ;
// Release the lock
if (SystemTDB.DiskLocationMultiJvmUsagePrevention) {
if (location.getLock().isOwned()) {
location.getLock().release();
} else if (location.getLock().canLock()) {
SystemTDB.errlog.warn("Location " + location.getDirectoryPath() + " was not locked, if another JVM accessed this location simultaneously data corruption may have occurred");
}
}
}
/**
* Return a StoreConnection for a particular connection. This is used to
* create transactions for the database at the location.
*/
public static synchronized StoreConnection make(Location location, StoreParams params) {
StoreConnection sConn = cache.get(location) ;
if (sConn != null)
return sConn ;
DatasetGraphTDB dsg = build(location, params) ;
sConn = _makeAndCache(dsg) ;
return sConn ;
}
/**
* Build storage {@link DatasetGraphTDB}.
* This operation is the primitive that creates the storage-level DatasetGraphTDB.
*/
private static DatasetGraphTDB build(Location location, StoreParams params) {
return DatasetBuilderStd.create(location, params) ;
}
/** Make a StoreConnection based on any StoreParams at the location or the system defaults. */
public static StoreConnection make(Location location) {
return make(location, null) ;
}
/**
* Return the StoreConnection if one already exists for this location, else
* return null
*/
public static synchronized StoreConnection getExisting(Location location) {
return cache.get(location);
}
private static StoreConnection _makeAndCache(DatasetGraphTDB dsg) {
Location location = dsg.getLocation() ;
StoreConnection sConn = cache.get(location) ;
if (sConn == null)
{
sConn = new StoreConnection(dsg) ;
if ( SystemTDB.DiskLocationMultiJvmUsagePrevention ) {
// Obtain the lock ASAP
LocationLock lock = location.getLock();
if (lock.canLock()) {
if (!lock.canObtain()) {
int here = ProcessUtils.getPid(0);
throw new TDBException("Process ID "+here+" can't open database at location " + location.getDirectoryPath() + " because it is already locked by the process with PID " + lock.getOwner() + ". "+
"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");
}
lock.obtain();
// There's an interesting race condition here that two JVMs might write out the lock file one after another without
// colliding and causing an IO error in either. The best way to check for this is simply to check we now own the lock
// and if not error
if (!lock.isOwned()) {
int here = ProcessUtils.getPid(0);
throw new TDBException("Process ID "+here+" failed to open database at location " + location.getDirectoryPath() + " because it is already locked by the process with PID " + lock.getOwner() + ". "+
"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");
}
}
}
sConn.forceRecoverFromJournal() ;
if (!location.isMemUnique())
// Don't cache use-once in-memory datasets.
cache.put(location, sConn) ;
String NS = TDB.PATH ;
TransactionInfo txInfo = new TransactionInfo(sConn.transactionManager) ;
ARQMgt.register(NS + ".system:type=Transactions", txInfo) ;
}
return sConn ;
}
/**
* Return a StoreConnection backed by in-memory
* datastructures (for testing).
*/
public static StoreConnection createMemUncached() {
DatasetGraphTDB dsg = DatasetBuilderStd.create(Location.mem(), null);
return new StoreConnection(dsg);
}
}