org.apache.tinkerpop.gremlin.tinkergraph.structure.SpecializedTinkerEdge Maven / Gradle / Ivy
/*
* 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.tinkerpop.gremlin.tinkergraph.structure;
import org.apache.tinkerpop.gremlin.structure.*;
import org.apache.tinkerpop.gremlin.structure.util.ElementHelper;
import org.apache.tinkerpop.gremlin.util.iterator.IteratorUtils;
import java.util.*;
import java.util.concurrent.Semaphore;
public abstract class SpecializedTinkerEdge extends TinkerEdge {
/** `dirty` flag for serialization to avoid superfluous serialization */
// TODO re-implement/verify this optimization: only re-serialize if element has been changed
private boolean modifiedSinceLastSerialization = true;
private Semaphore modificationSemaphore = new Semaphore(1);
private final Set specificKeys;
protected SpecializedTinkerEdge(TinkerGraph graph, Long id, Vertex outVertex, String label, Vertex inVertex, Set specificKeys) {
super(graph, id, outVertex, label, inVertex);
this.specificKeys = specificKeys;
if (graph.referenceManager != null) {
graph.referenceManager.applyBackpressureMaybe();
}
}
@Override
public Set keys() {
return specificKeys;
}
@Override
public Property property(String key) {
return specificProperty(key);
}
/* implement in concrete specialised instance to avoid using generic HashMaps */
protected abstract Property specificProperty(String key);
@Override
public Iterator> properties(String... propertyKeys) {
if (propertyKeys.length == 0) {
return (Iterator) specificKeys.stream().map(key -> property(key)).filter(vp -> vp.isPresent()).iterator();
} else if (propertyKeys.length == 1) { // treating as special case for performance
// return IteratorUtils.of(property(propertyKeys[0]));
final Property prop = property(propertyKeys[0]);
return prop.isPresent() ? IteratorUtils.of(prop) : Collections.emptyIterator();
} else {
return Arrays.stream(propertyKeys).map(key -> (Property) property(key)).filter(vp -> vp.isPresent()).iterator();
}
}
@Override
public Property property(String key, V value) {
if (this.removed) throw elementAlreadyRemoved(Edge.class, id);
ElementHelper.validateProperty(key, value);
final Property oldProperty = super.property(key);
acquireModificationLock();
modifiedSinceLastSerialization = true;
final Property p = updateSpecificProperty(key, value);
TinkerHelper.autoUpdateIndex(this, key, value, oldProperty.isPresent() ? oldProperty.value() : null);
releaseModificationLock();
return p;
}
protected abstract Property updateSpecificProperty(String key, V value);
public void removeProperty(String key) {
acquireModificationLock();
modifiedSinceLastSerialization = true;
removeSpecificProperty(key);
releaseModificationLock();
}
protected abstract void removeSpecificProperty(String key);
@Override
public Graph graph() {
return this.graph;
}
@Override
public void remove() {
super.remove();
graph.getElementsByLabel(graph.edgesByLabel, label).remove(this);
modifiedSinceLastSerialization = true;
}
public void setModifiedSinceLastSerialization(boolean modifiedSinceLastSerialization) {
this.modifiedSinceLastSerialization = modifiedSinceLastSerialization;
}
public boolean isModifiedSinceLastSerialization() {
return modifiedSinceLastSerialization;
}
public void acquireModificationLock() {
try {
modificationSemaphore.acquire();
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
public void releaseModificationLock() {
modificationSemaphore.release();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy