com.graphaware.tx.event.improved.propertycontainer.snapshot.PropertyContainerSnapshot Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of tx-api Show documentation
Show all versions of tx-api Show documentation
Improved Transaction API with Graph Snapshotting
/*
* Copyright (c) 2013-2017 GraphAware
*
* This file is part of the GraphAware Framework.
*
* GraphAware Framework is free software: you can redistribute it and/or modify it under the terms of
* the GNU General Public License as published by the Free Software Foundation, either
* version 3 of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See the GNU General Public License for more details. You should have received a copy of
* the GNU General Public License along with this program. If not, see
* .
*/
package com.graphaware.tx.event.improved.propertycontainer.snapshot;
import com.graphaware.common.wrapper.BasePropertyContainerWrapper;
import com.graphaware.tx.event.improved.data.PropertyContainerTransactionData;
import com.graphaware.tx.event.improved.data.TransactionDataContainer;
import org.neo4j.graphdb.NotFoundException;
import org.neo4j.graphdb.PropertyContainer;
import org.neo4j.logging.Log;
import com.graphaware.common.log.LoggerFactory;
import java.util.Collection;
import java.util.HashSet;
/**
* A decorator of a {@link org.neo4j.graphdb.PropertyContainer} that represents a snapshot of a {@link org.neo4j.graphdb.PropertyContainer} before a
* transaction has started. It consults {@link com.graphaware.tx.event.improved.data.PropertyContainerTransactionData} wrapped in {@link com.graphaware.tx.event.improved.data.TransactionDataContainer},
* before returning information about contained properties in order to provide these as they were before the transactions started.
* Mutations are preformed as expected but only in the case that the mutated {@link org.neo4j.graphdb.PropertyContainer} has not been deleted
* in the transaction. If it has been deleted, an exception is thrown upon mutation.
*
* @param type of the wrapped property container.
*/
public abstract class PropertyContainerSnapshot extends BasePropertyContainerWrapper implements PropertyContainer {
private static final Log LOG = LoggerFactory.getLogger(PropertyContainerSnapshot.class);
protected final T wrapped;
protected final TransactionDataContainer transactionDataContainer;
/**
* Construct a new snapshot.
*
* @param wrapped property container.
* @param transactionDataContainer transaction data container.
*/
protected PropertyContainerSnapshot(T wrapped, TransactionDataContainer transactionDataContainer) {
this.wrapped = wrapped;
this.transactionDataContainer = transactionDataContainer;
}
/**
* {@inheritDoc}
*/
@Override
public T getWrapped() {
return wrapped;
}
/**
* Get data about the transaction relevant for the subclass.
*
* @return transaction data.
*/
protected abstract PropertyContainerTransactionData transactionData();
/**
* {@inheritDoc}
*/
@Override
public boolean hasProperty(String key) {
if (transactionData().hasBeenDeleted(wrapped)) {
return transactionData().propertiesOfDeletedContainer(wrapped).containsKey(key);
}
if (transactionData().hasBeenChanged(wrapped)) {
if (transactionData().hasPropertyBeenCreated(wrapped, key)) {
return false; //has been created in the transaction = wasn't there before
}
if (transactionData().hasPropertyBeenDeleted(wrapped, key)) {
return true; //has been deleted in the transaction = was there before
}
}
//at this point, we know that either of the following is the case:
//- the container has not been changed by the tx
//- the container has been changed and the property we're looking for has not
//- the container has been changed and the property we're looking for has been changed
// In either case, we can find out whether it was there by delegating to the wrapped one (its current version).
return super.hasProperty(key);
}
/**
* {@inheritDoc}
*/
@Override
public Object getProperty(String key) {
if (!hasProperty(key)) {
throw new NotFoundException("Snapshot of " + wrapped.toString() + " did not have a property with key " + key);
}
if (transactionData().hasBeenDeleted(wrapped)) {
return transactionData().propertiesOfDeletedContainer(wrapped).get(key);
}
if (transactionData().hasBeenChanged(wrapped)) {
if (transactionData().hasPropertyBeenChanged(wrapped, key)) {
return transactionData().changedProperties(wrapped).get(key).getPrevious();
}
if (transactionData().hasPropertyBeenDeleted(wrapped, key)) {
return transactionData().deletedProperties(wrapped).get(key);
}
}
return super.getProperty(key);
}
/**
* {@inheritDoc}
*/
@Override
public void setProperty(String key, Object value) {
checkCanBeMutated();
super.setProperty(key, value);
}
/**
* {@inheritDoc}
*/
@Override
public Object removeProperty(String key) {
checkCanBeMutated();
return super.removeProperty(key);
}
/**
* Check whether this property container has not been deleted and can thus be mutated.
* In case the container can be mutated, nothing happens. In the opposite case, an {@link IllegalStateException} exception is thrown.
*/
protected void checkCanBeMutated() {
if (transactionData().hasBeenDeleted(wrapped)) {
LOG.error("Deleted property container " + wrapped + " should not be mutated.");
throw new IllegalStateException("Deleted property container " + wrapped + " should not be mutated.");
}
}
/**
* {@inheritDoc}
*/
@Override
public Iterable getPropertyKeys() {
if (transactionData().hasBeenDeleted(wrapped)) {
return transactionData().propertiesOfDeletedContainer(wrapped).keySet();
}
if (!transactionData().hasBeenChanged(wrapped)) {
return super.getPropertyKeys();
}
Collection result = new HashSet<>();
for (String key : super.getPropertyKeys()) {
if (!transactionData().hasPropertyBeenCreated(wrapped, key)) {
result.add(key);
}
}
result.addAll(transactionData().deletedProperties(wrapped).keySet());
return result;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy