org.vertexium.inmemory.InMemoryTableElement Maven / Gradle / Ivy
package org.vertexium.inmemory;
import org.vertexium.*;
import org.vertexium.inmemory.mutations.*;
import org.vertexium.property.MutablePropertyImpl;
import org.vertexium.property.StreamingPropertyValue;
import org.vertexium.property.StreamingPropertyValueRef;
import org.vertexium.util.ConvertingIterable;
import org.vertexium.util.FilterIterable;
import org.vertexium.util.IncreasingTime;
import org.vertexium.util.LookAheadIterable;
import java.io.Serializable;
import java.util.*;
public abstract class InMemoryTableElement implements Serializable {
private final String id;
private TreeSet mutations = new TreeSet<>();
protected InMemoryTableElement(String id) {
this.id = id;
}
public String getId() {
return id;
}
public void addAll(Mutation... newMutations) {
Collections.addAll(mutations, newMutations);
}
public long getFirstTimestamp() {
return findFirstMutation(ElementTimestampMutation.class).getTimestamp();
}
protected T findLastMutation(Class clazz) {
T last = null;
for (Mutation m : this.mutations) {
if (clazz.isAssignableFrom(m.getClass())) {
//noinspection unchecked
last = (T) m;
}
}
return last;
}
protected T findFirstMutation(Class clazz) {
for (Mutation m : this.mutations) {
if (clazz.isAssignableFrom(m.getClass())) {
//noinspection unchecked
return (T) m;
}
}
return null;
}
protected Iterable findMutations(final Class clazz) {
return new ConvertingIterable(new FilterIterable(this.mutations) {
@Override
protected boolean isIncluded(Mutation m) {
return clazz.isAssignableFrom(m.getClass());
}
}) {
@Override
protected T convert(Mutation o) {
//noinspection unchecked
return (T) o;
}
};
}
public Visibility getVisibility() {
return findLastMutation(AlterVisibilityMutation.class).getNewVisibility();
}
public long getTimestamp() {
return findLastMutation(ElementTimestampMutation.class).getTimestamp();
}
private List findPropertyMutations(Property p) {
return findPropertyMutations(p.getKey(), p.getName(), p.getVisibility());
}
public Property deleteProperty(String key, String name, Authorizations authorizations) {
return deleteProperty(key, name, null, authorizations);
}
public Property getProperty(String key, String name, Visibility visibility, Authorizations authorizations) {
List propertyMutations = findPropertyMutations(key, name, visibility);
if (propertyMutations == null || propertyMutations.size() == 0) {
return null;
}
return toProperty(propertyMutations, true, authorizations);
}
public Property deleteProperty(String key, String name, Visibility visibility, Authorizations authorizations) {
Property p = getProperty(key, name, visibility, authorizations);
if (p != null) {
deleteProperty(p);
}
return p;
}
protected void deleteProperty(Property p) {
List propertyMutations = findPropertyMutations(p);
this.mutations.removeAll(propertyMutations);
}
private List findPropertyMutations(String key, String name, Visibility visibility) {
List results = new ArrayList<>();
for (Mutation m : this.mutations) {
if (!(m instanceof PropertyMutation)) {
continue;
}
PropertyMutation pm = (PropertyMutation) m;
if ((key == null || pm.getPropertyKey().equals(key))
&& (name == null || pm.getPropertyName().equals(name))
&& (visibility == null || pm.getPropertyVisibility().equals(visibility))) {
results.add(pm);
}
}
return results;
}
public Iterable getHistoricalPropertyValues(
String key,
String name,
Visibility visibility,
Long startTime,
Long endTime,
Authorizations authorizations
) {
List propertyMutations = findPropertyMutations(key, name, visibility);
List historicalPropertyValues = new ArrayList<>();
Set hiddenVisibilities = new HashSet<>();
for (PropertyMutation m : propertyMutations) {
if (startTime != null && m.getTimestamp() < startTime) {
continue;
}
if (endTime != null && m.getTimestamp() > endTime) {
continue;
}
if (!canRead(m.getVisibility(), authorizations)) {
continue;
}
if (m instanceof SoftDeletePropertyMutation) {
continue;
}
if (m instanceof MarkPropertyHiddenMutation) {
hiddenVisibilities.add(m.getVisibility());
} else if (m instanceof MarkPropertyVisibleMutation) {
hiddenVisibilities.remove(m.getVisibility());
} else if (m instanceof AddPropertyValueMutation) {
AddPropertyValueMutation addPropertyValueMutation = (AddPropertyValueMutation) m;
Object value = addPropertyValueMutation.getValue();
value = loadIfStreamingPropertyValue(value);
HistoricalPropertyValue historicalPropertyValue = new HistoricalPropertyValue(
m.getPropertyKey(),
m.getPropertyName(),
m.getVisibility(),
m.getTimestamp(),
value,
addPropertyValueMutation.getMetadata(),
hiddenVisibilities
);
historicalPropertyValues.add(historicalPropertyValue);
} else {
throw new VertexiumException("Unhandled PropertyMutation: " + m.getClass().getName());
}
}
Collections.reverse(historicalPropertyValues);
return historicalPropertyValues;
}
public Iterable getProperties(final boolean includeHidden, Long endTime, final Authorizations authorizations) {
final TreeMap> propertiesMutations = new TreeMap<>();
for (PropertyMutation m : findMutations(PropertyMutation.class)) {
if (endTime != null && m.getTimestamp() > endTime) {
continue;
}
String mapKey = toMapKey(m);
List propertyMutations = propertiesMutations.get(mapKey);
if (propertyMutations == null) {
propertyMutations = new ArrayList<>();
propertiesMutations.put(mapKey, propertyMutations);
}
propertyMutations.add(m);
}
return new LookAheadIterable, Property>() {
@Override
protected boolean isIncluded(List src, Property property) {
return property != null;
}
@Override
protected Property convert(List propertyMutations) {
return toProperty(propertyMutations, includeHidden, authorizations);
}
@Override
protected Iterator> createIterator() {
return propertiesMutations.values().iterator();
}
};
}
private Property toProperty(List propertyMutations, boolean includeHidden, Authorizations authorizations) {
String propertyKey = null;
String propertyName = null;
Object value = null;
Metadata metadata = null;
long timestamp = 0;
Set hiddenVisibilities = new HashSet<>();
Visibility visibility = null;
boolean softDeleted = false;
boolean hidden = false;
for (PropertyMutation m : propertyMutations) {
if (!canRead(m.getVisibility(), authorizations)) {
continue;
}
propertyKey = m.getPropertyKey();
propertyName = m.getPropertyName();
visibility = m.getPropertyVisibility();
if (m.getTimestamp() > timestamp) {
timestamp = m.getTimestamp();
}
if (m instanceof AddPropertyValueMutation) {
value = ((AddPropertyValueMutation) m).getValue();
metadata = ((AddPropertyValueMutation) m).getMetadata();
softDeleted = false;
} else if (m instanceof SoftDeletePropertyMutation) {
softDeleted = true;
} else if (m instanceof MarkPropertyHiddenMutation) {
hidden = true;
hiddenVisibilities.add(m.getVisibility());
} else if (m instanceof MarkPropertyVisibleMutation) {
hidden = false;
hiddenVisibilities.remove(m.getVisibility());
} else {
throw new VertexiumException("Unhandled PropertyMutation: " + m.getClass().getName());
}
}
if (softDeleted) {
return null;
}
if (!includeHidden && hidden) {
return null;
}
if (propertyKey == null) {
return null;
}
value = loadIfStreamingPropertyValue(value);
return new MutablePropertyImpl(propertyKey, propertyName, value, metadata, timestamp, hiddenVisibilities, visibility);
}
private Object loadIfStreamingPropertyValue(Object value) {
if (value instanceof StreamingPropertyValueRef) {
value = loadStreamingPropertyValue((StreamingPropertyValueRef) value);
}
return value;
}
protected StreamingPropertyValue loadStreamingPropertyValue(StreamingPropertyValueRef> streamingPropertyValueRef) {
// There's no need to have a Graph object for the pure in-memory implementation. Subclasses should override.
return streamingPropertyValueRef.toStreamingPropertyValue(null);
}
private String toMapKey(PropertyMutation m) {
return m.getPropertyName() + m.getPropertyKey() + m.getPropertyVisibility().getVisibilityString();
}
public void appendSoftDeleteMutation(Long timestamp) {
if (timestamp == null) {
timestamp = IncreasingTime.currentTimeMillis();
}
this.mutations.add(new SoftDeleteMutation(timestamp));
}
public void appendMarkHiddenMutation(Visibility visibility) {
long timestamp = IncreasingTime.currentTimeMillis();
this.mutations.add(new MarkHiddenMutation(timestamp, visibility));
}
public void appendMarkVisibleMutation(Visibility visibility) {
long timestamp = IncreasingTime.currentTimeMillis();
this.mutations.add(new MarkVisibleMutation(timestamp, visibility));
}
public Property appendMarkPropertyHiddenMutation(String key, String name, Visibility propertyVisibility, Long timestamp, Visibility visibility, Authorizations authorizations) {
Property prop = getProperty(key, name, propertyVisibility, authorizations);
if (timestamp == null) {
timestamp = IncreasingTime.currentTimeMillis();
}
this.mutations.add(new MarkPropertyHiddenMutation(key, name, propertyVisibility, timestamp, visibility));
return prop;
}
public Property appendMarkPropertyVisibleMutation(String key, String name, Visibility propertyVisibility, Long timestamp, Visibility visibility, Authorizations authorizations) {
Property prop = getProperty(key, name, propertyVisibility, authorizations);
if (timestamp == null) {
timestamp = IncreasingTime.currentTimeMillis();
}
this.mutations.add(new MarkPropertyVisibleMutation(key, name, propertyVisibility, timestamp, visibility));
return prop;
}
public void appendSoftDeletePropertyMutation(String key, String name, Visibility propertyVisibility, Long timestamp) {
if (timestamp == null) {
timestamp = IncreasingTime.currentTimeMillis();
}
this.mutations.add(new SoftDeletePropertyMutation(timestamp, key, name, propertyVisibility));
}
public void appendAlterVisibilityMutation(Visibility newVisibility) {
long timestamp = IncreasingTime.currentTimeMillis();
this.mutations.add(new AlterVisibilityMutation(timestamp, newVisibility));
}
public void appendAddPropertyMutation(String key, String name, Object value, Metadata metadata, Visibility visibility, Long timestamp) {
if (timestamp == null) {
timestamp = IncreasingTime.currentTimeMillis();
}
this.mutations.add(new AddPropertyValueMutation(timestamp, key, name, value, metadata, visibility));
}
public void appendAlterEdgeLabelMutation(String newEdgeLabel) {
long timestamp = IncreasingTime.currentTimeMillis();
this.mutations.add(new AlterEdgeLabelMutation(timestamp, newEdgeLabel));
}
protected List getFilteredMutations(boolean includeHidden, Long endTime, Authorizations authorizations) {
List mutations = new ArrayList<>();
for (Mutation m : this.mutations) {
if (!canRead(m.getVisibility(), authorizations)) {
continue;
}
if (endTime != null && m.getTimestamp() > endTime) {
continue;
}
if (includeHidden) {
if (m instanceof MarkHiddenMutation || m instanceof MarkPropertyHiddenMutation) {
continue;
}
}
mutations.add(m);
}
return mutations;
}
public boolean canRead(Authorizations authorizations) {
// this is just a shortcut so that we don't need to construct evaluators and visibility objects to check for an empty string.
//noinspection SimplifiableIfStatement
if (getVisibility().getVisibilityString().length() == 0) {
return true;
}
return authorizations.canRead(getVisibility());
}
private static boolean canRead(Visibility visibility, Authorizations authorizations) {
// this is just a shortcut so that we don't need to construct evaluators and visibility objects to check for an empty string.
//noinspection SimplifiableIfStatement
if (visibility.getVisibilityString().length() == 0) {
return true;
}
return authorizations.canRead(visibility);
}
public Set getHiddenVisibilities() {
Set results = new HashSet<>();
for (Mutation m : this.mutations) {
if (m instanceof MarkHiddenMutation) {
results.add(m.getVisibility());
} else if (m instanceof MarkVisibleMutation) {
results.remove(m.getVisibility());
}
}
return results;
}
public boolean isHidden(Authorizations authorizations) {
for (Visibility visibility : getHiddenVisibilities()) {
if (authorizations.canRead(visibility)) {
return true;
}
}
return false;
}
public TElement createElement(InMemoryGraph graph, Authorizations authorizations) {
return createElement(graph, true, null, authorizations);
}
public final TElement createElement(InMemoryGraph graph, boolean includeHidden, Long endTime, Authorizations authorizations) {
if (endTime != null && getFirstTimestamp() > endTime) {
return null;
}
if (isDeleted(endTime, authorizations)) {
return null;
}
return createElementInternal(graph, includeHidden, endTime, authorizations);
}
public boolean isDeleted(Long endTime, Authorizations authorizations) {
boolean deleted = false;
for (Mutation m : this.mutations) {
if (!canRead(m.getVisibility(), authorizations)) {
continue;
}
if (endTime != null && m.getTimestamp() > endTime) {
continue;
}
if (m instanceof SoftDeleteMutation) {
deleted = true;
} else if (m instanceof ElementTimestampMutation) {
deleted = false;
}
}
return deleted;
}
protected abstract TElement createElementInternal(InMemoryGraph graph, boolean includeHidden, Long endTime, Authorizations authorizations);
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy