org.eclipse.jnosql.mapping.graph.AbstractGraphTemplate Maven / Gradle / Ivy
/*
* Copyright (c) 2022 Contributors to the Eclipse Foundation
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* and Apache License v2.0 which accompanies this distribution.
* The Eclipse Public License is available at http://www.eclipse.org/legal/epl-v10.html
* and the Apache License v2.0 is available at http://www.opensource.org/licenses/apache2.0.php.
*
* You may elect to redistribute this code under either of these licenses.
*
* Contributors:
*
* Otavio Santana
*/
package org.eclipse.jnosql.mapping.graph;
import jakarta.data.exceptions.EmptyResultException;
import jakarta.data.exceptions.NonUniqueResultException;
import jakarta.nosql.PreparedStatement;
import jakarta.nosql.QueryMapper;
import org.eclipse.jnosql.mapping.core.Converters;
import org.apache.tinkerpop.gremlin.process.traversal.Traverser;
import org.apache.tinkerpop.gremlin.process.traversal.dsl.graph.GraphTraversal;
import org.apache.tinkerpop.gremlin.process.traversal.dsl.graph.GraphTraversalSource;
import org.apache.tinkerpop.gremlin.structure.Direction;
import org.apache.tinkerpop.gremlin.structure.Edge;
import org.apache.tinkerpop.gremlin.structure.Graph;
import org.apache.tinkerpop.gremlin.structure.Transaction;
import org.apache.tinkerpop.gremlin.structure.Vertex;
import org.eclipse.jnosql.mapping.IdNotFoundException;
import org.eclipse.jnosql.mapping.metadata.EntitiesMetadata;
import org.eclipse.jnosql.mapping.metadata.EntityMetadata;
import org.eclipse.jnosql.mapping.metadata.FieldMetadata;
import org.eclipse.jnosql.mapping.core.util.ConverterUtil;
import org.eclipse.jnosql.mapping.metadata.InheritanceMetadata;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.UnaryOperator;
import java.util.function.Supplier;
import java.util.function.Consumer;
import java.util.stream.Stream;
import java.util.stream.StreamSupport;
import static java.util.Objects.isNull;
import static java.util.Objects.requireNonNull;
import static org.apache.tinkerpop.gremlin.structure.T.id;
public abstract class AbstractGraphTemplate implements GraphTemplate {
private static final Function, GraphTraversal> INITIAL_VERTEX =
g -> (GraphTraversal) g;
private static final Function, GraphTraversal> INITIAL_EDGE =
g -> (GraphTraversal) g;
protected abstract Graph getGraph();
protected abstract EntitiesMetadata getEntities();
protected abstract GraphConverter getConverter();
protected abstract Converters getConverters();
protected abstract GraphEventPersistManager getEventManager();
private GremlinExecutor gremlinExecutor;
private GremlinExecutor getExecutor() {
if (Objects.isNull(gremlinExecutor)) {
this.gremlinExecutor = new GremlinExecutor(getConverter());
}
return gremlinExecutor;
}
@Override
public T insert(T entity) {
requireNonNull(entity, "entity is required");
checkId(entity);
UnaryOperator save = v -> {
GraphTransactionUtil.transaction(getGraph());
return v;
};
return persist(entity, save);
}
@Override
public T insert(T entity, Duration ttl) {
throw new UnsupportedOperationException("GraphTemplate does not support insert with TTL");
}
@Override
public Iterable insert(Iterable entities, Duration ttl) {
throw new UnsupportedOperationException("GraphTemplate does not support insert with TTL");
}
@Override
public T update(T entity) {
requireNonNull(entity, "entity is required");
checkId(entity);
if (isIdNull(entity)) {
throw new IllegalStateException("to update a graph id cannot be null");
}
vertex(entity).orElseThrow(() -> new EmptyResultException("Entity does not find in the update"));
UnaryOperator update = e -> {
final Vertex vertex = getConverter().toVertex(entity);
GraphTransactionUtil.transaction(getGraph());
return vertex;
};
return persist(entity, update);
}
@Override
public Optional find(Class type, K id) {
requireNonNull(type, "type is required");
requireNonNull(id, "id is required");
EntityMetadata entityMetadata = getEntities().get(type);
FieldMetadata idField = entityMetadata.id()
.orElseThrow(() -> IdNotFoundException.newInstance(type));
Object value = ConverterUtil.getValue(id, entityMetadata, idField.fieldName(), getConverters());
final Optional vertex = traversal().V(value).hasLabel(entityMetadata.name()).tryNext();
return vertex.map(getConverter()::toEntity);
}
@Override
public void delete(T idValue) {
requireNonNull(idValue, "id is required");
traversal().V(idValue).toStream().forEach(Vertex::remove);
}
@Override
public void delete(Iterable ids) {
requireNonNull(ids, "ids is required");
final Object[] vertexIds = StreamSupport.stream(ids.spliterator(), false).toArray(Object[]::new);
traversal().V(vertexIds).toStream().forEach(Vertex::remove);
}
@Override
public void delete(Class type, K id) {
requireNonNull(type, "type is required");
requireNonNull(id, "id is required");
EntityMetadata mapping = getEntities().get(type);
traversal()
.V(id)
.hasLabel(mapping.name())
.toStream()
.forEach(Vertex::remove);
}
@Override
public void deleteEdge(T idEdge) {
requireNonNull(idEdge, "idEdge is required");
traversal().E(idEdge).toStream().forEach(Edge::remove);
}
@Override
public Optional find(K idValue) {
requireNonNull(idValue, "id is required");
Optional vertex = traversal().V(idValue).tryNext();
return vertex.map(getConverter()::toEntity);
}
@Override
public Iterable insert(Iterable entities) {
requireNonNull(entities, "entities is required");
return StreamSupport.stream(entities.spliterator(), false)
.map(this::insert).toList();
}
@Override
public Iterable update(Iterable entities) {
requireNonNull(entities, "entities is required");
return StreamSupport.stream(entities.spliterator(), false)
.map(this::update).toList();
}
@Override
public void deleteEdge(Iterable ids) {
requireNonNull(ids, "ids is required");
final Object[] edgeIds = StreamSupport.stream(ids.spliterator(), false).toArray(Object[]::new);
traversal().E(edgeIds).toStream().forEach(Edge::remove);
}
@Override
public EdgeEntity edge(O outgoing, String label, I incoming) {
requireNonNull(incoming, "incoming is required");
requireNonNull(label, "label is required");
requireNonNull(outgoing, "outgoing is required");
checkId(outgoing);
checkId(incoming);
if (isIdNull(outgoing)) {
throw new IllegalStateException("outgoing Id field is required");
}
if (isIdNull(incoming)) {
throw new IllegalStateException("incoming Id field is required");
}
Vertex outVertex = vertex(outgoing).orElseThrow(() -> new EmptyResultException("Outgoing entity does not found"));
Vertex inVertex = vertex(incoming).orElseThrow(() -> new EmptyResultException("Incoming entity does not found"));
final Predicate> predicate = t -> {
Edge e = t.get();
return e.inVertex().id().equals(inVertex.id())
&& e.outVertex().id().equals(outVertex.id());
};
Optional edge = traversal().V(outVertex.id())
.out(label).has(id, inVertex.id()).inE(label).filter(predicate).tryNext();
return edge.map(edge1 -> new DefaultEdgeEntity<>(edge1, incoming, outgoing))
.orElseGet(() -> new DefaultEdgeEntity<>(getEdge(label, outVertex, inVertex), incoming, outgoing));
}
@Override
public Optional edge(E edgeId) {
requireNonNull(edgeId, "edgeId is required");
Optional edgeOptional = traversal().E(edgeId).tryNext();
if (edgeOptional.isPresent()) {
Edge edge = edgeOptional.get();
return Optional.of(getConverter().toEdgeEntity(edge));
}
return Optional.empty();
}
@Override
public Collection edges(T entity, Direction direction) {
return edgesImpl(entity, direction);
}
@Override
public Collection edges(T entity, Direction direction, String... labels) {
return edgesImpl(entity, direction, labels);
}
@SafeVarargs
@Override
public final Collection edges(T entity, Direction direction, Supplier... labels) {
checkLabelsSupplier(labels);
return edgesImpl(entity, direction, Stream.of(labels).map(Supplier::get).toArray(String[]::new));
}
@Override
public Collection edgesById(K id, Direction direction, String... labels) {
return edgesByIdImpl(id, direction, labels);
}
@Override
public Collection edgesById(K id, Direction direction) {
return edgesByIdImpl(id, direction);
}
@SafeVarargs
@Override
public final Collection edgesById(K id, Direction direction, Supplier... labels) {
checkLabelsSupplier(labels);
return edgesByIdImpl(id, direction, Stream.of(labels).map(Supplier::get).toArray(String[]::new));
}
@Override
public VertexTraversal traversalVertex(Object... vertexIds) {
if (Stream.of(vertexIds).anyMatch(Objects::isNull)) {
throw new IllegalStateException("No one vertexId element cannot be null");
}
return new DefaultVertexTraversal(() -> traversal().V(vertexIds), INITIAL_VERTEX, getConverter());
}
@Override
public EdgeTraversal traversalEdge(Object... edgeIds) {
if (Stream.of(edgeIds).anyMatch(Objects::isNull)) {
throw new IllegalStateException("No one edgeId element cannot be null");
}
return new DefaultEdgeTraversal(() -> traversal().E(edgeIds), INITIAL_EDGE, getConverter());
}
@Override
public Transaction transaction() {
return getGraph().tx();
}
@Override
public Stream query(String gremlin) {
requireNonNull(gremlin, "query is required");
return getExecutor().executeGremlin(traversal(), gremlin);
}
@Override
public Optional singleResult(String gremlin) {
Stream entities = query(gremlin);
final Iterator iterator = entities.iterator();
if (!iterator.hasNext()) {
return Optional.empty();
}
final T entity = iterator.next();
if (!iterator.hasNext()) {
return Optional.ofNullable(entity);
}
throw new NonUniqueResultException("The gremlin query returns more than one result: " + gremlin);
}
@Override
public PreparedStatement prepare(String gremlin) {
requireNonNull(gremlin, "query is required");
return new DefaultPreparedStatement(getExecutor(), gremlin, traversal());
}
@Override
public long count(String label) {
Objects.requireNonNull(label, "label is required");
return traversal().V().hasLabel(label).count().tryNext().orElse(0L);
}
@Override
public long count(Class type) {
Objects.requireNonNull(type, "entity class is required");
var metadata = getEntities().get(type);
if(metadata.inheritance().isPresent()){
InheritanceMetadata inheritanceMetadata = metadata.inheritance().orElseThrow();
if(!inheritanceMetadata.parent().equals(metadata.type())){
return traversal().V().hasLabel(metadata.name())
.has(inheritanceMetadata.discriminatorColumn(), inheritanceMetadata.discriminatorValue())
.count().tryNext().orElse(0L);
}
}
return count(getEntities().get(type).name());
}
@Override
public QueryMapper.MapperFrom select(Class type) {
Objects.requireNonNull(type, "type is required");
EntityMetadata metadata = getEntities().get(type);
GraphTraversal traversal = traversal().V().hasLabel(metadata.name());
return new GraphMapperSelect(metadata,getConverters(), traversal, getConverter());
}
@Override
public QueryMapper.MapperDeleteFrom delete(Class type) {
Objects.requireNonNull(type, "type is required");
EntityMetadata metadata = getEntities().get(type);
GraphTraversal traversal = traversal().V().hasLabel(metadata.name());
return new GraphMapperDelete(metadata,getConverters(), traversal, getConverter());
}
@Override
public Stream findAll(Class type) {
Objects.requireNonNull(type, "type is required");
EntityMetadata metadata = getEntities().get(type);
if(metadata.inheritance().isPresent()){
InheritanceMetadata inheritanceMetadata = metadata.inheritance().orElseThrow();
if(!inheritanceMetadata.parent().equals(metadata.type())){
return traversal().V().hasLabel(metadata.name())
.has(inheritanceMetadata.discriminatorColumn(), inheritanceMetadata.discriminatorValue())
.toStream().map(getConverter()::toEntity);
}
}
return traversal().V().hasLabel(metadata.name())
.toStream().map(getConverter()::toEntity);
}
@Override
public void deleteAll(Class type) {
Objects.requireNonNull(type, "type is required");
EntityMetadata metadata = getEntities().get(type);
if(metadata.inheritance().isPresent()){
InheritanceMetadata inheritanceMetadata = metadata.inheritance().orElseThrow();
if(!inheritanceMetadata.parent().equals(metadata.type())){
traversal().V().hasLabel(metadata.name())
.has(inheritanceMetadata.discriminatorColumn(), inheritanceMetadata.discriminatorValue())
.toStream().forEach(Vertex::remove);
return;
}
}
traversal().V().hasLabel(metadata.name()).toStream().forEach(Vertex::remove);
}
protected GraphTraversalSource traversal() {
return getGraph().traversal();
}
protected Iterator vertices(Object id) {
return getGraph().vertices(id);
}
private Edge getEdge(String label, Vertex outVertex, Vertex inVertex) {
final Edge edge = outVertex.addEdge(label, inVertex);
GraphTransactionUtil.transaction(getGraph());
return edge;
}
private Collection edgesByIdImpl(K id, Direction direction, String... labels) {
requireNonNull(id, "id is required");
requireNonNull(direction, "direction is required");
Iterator vertices = vertices(id);
if (vertices.hasNext()) {
List edges = new ArrayList<>();
vertices.next().edges(direction, labels).forEachRemaining(edges::add);
return edges.stream().map(getConverter()::toEdgeEntity).toList();
}
return Collections.emptyList();
}
private Optional vertex(T entity) {
EntityMetadata entityMetadata = getEntities().get(entity.getClass());
FieldMetadata field = entityMetadata.id().orElseThrow(() -> IdNotFoundException.newInstance(entity.getClass()));
Object id = field.read(entity);
Iterator vertices = vertices(id);
if (vertices.hasNext()) {
return Optional.of(vertices.next());
}
return Optional.empty();
}
private Collection edgesImpl(T entity, Direction direction, String... labels) {
requireNonNull(entity, "entity is required");
if (isIdNull(entity)) {
throw new IllegalStateException("Entity id is required");
}
if (vertex(entity).isEmpty()) {
return Collections.emptyList();
}
Object id = getConverter().toVertex(entity).id();
return edgesByIdImpl(id, direction, labels);
}
private void checkLabelsSupplier(Supplier[] labels) {
if (Stream.of(labels).anyMatch(Objects::isNull)) {
throw new IllegalStateException("Item cannot be null");
}
}
private boolean isIdNull(T entity) {
EntityMetadata entityMetadata = getEntities().get(entity.getClass());
FieldMetadata field = entityMetadata.id().orElseThrow(() -> IdNotFoundException.newInstance(entity.getClass()));
return isNull(field.read(entity));
}
private void checkId(T entity) {
EntityMetadata entityMetadata = getEntities().get(entity.getClass());
entityMetadata.id().orElseThrow(() -> IdNotFoundException.newInstance(entity.getClass()));
}
protected T persist(T entity, UnaryOperator persistAction) {
return Stream.of(entity)
.map(toUnary(getEventManager()::firePreEntity))
.map(getConverter()::toVertex)
.map(persistAction)
.map(t -> getConverter().toEntity(entity, t))
.map(toUnary(getEventManager()::firePostEntity))
.findFirst()
.orElseThrow();
}
private UnaryOperator toUnary(Consumer consumer) {
return t -> {
consumer.accept(t);
return t;
};
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy