diff --git a/src/main/java/com/arangodb/springframework/core/convert/ArangoEntityReader.java b/src/main/java/com/arangodb/springframework/core/convert/ArangoEntityReader.java index 3a55bff1..b2de69b0 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/ArangoEntityReader.java +++ b/src/main/java/com/arangodb/springframework/core/convert/ArangoEntityReader.java @@ -27,7 +27,6 @@ /** * @author Mark Vollmary * @author Christian Lechner - * */ public interface ArangoEntityReader extends EntityReader { diff --git a/src/main/java/com/arangodb/springframework/core/convert/DefaultArangoConverter.java b/src/main/java/com/arangodb/springframework/core/convert/DefaultArangoConverter.java index c1ed39f4..7ff009a9 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/DefaultArangoConverter.java +++ b/src/main/java/com/arangodb/springframework/core/convert/DefaultArangoConverter.java @@ -222,8 +222,8 @@ private void readProperty( final String parentId, final PersistentPropertyAccessor accessor, final JsonNode source, - final ArangoPersistentProperty property) { - + final ArangoPersistentProperty property + ) { Object propertyValue = readPropertyValue(entity, parentId, source, property); if (propertyValue != null || !property.getType().isPrimitive()) { accessor.setProperty(property, propertyValue); @@ -234,8 +234,8 @@ private Object readPropertyValue( final ArangoPersistentEntity entity, final String parentId, final JsonNode source, - final ArangoPersistentProperty property) { - + final ArangoPersistentProperty property + ) { Optional ref = property.getRef(); if (ref.isPresent()) { return readReference(source, property, ref.get()).orElse(null); @@ -329,8 +329,8 @@ private Object readArray(final TypeInformation type, final JsonNode source) { private Optional readReference( final JsonNode source, final ArangoPersistentProperty property, - final Annotation annotation) { - + final Annotation annotation + ) { if (source.isMissingNode() || source.isNull()) { return Optional.empty(); } @@ -366,8 +366,8 @@ private Optional readRelation( final String parentId, final JsonNode source, final ArangoPersistentProperty property, - final A annotation) { - + final A annotation + ) { if (source.isNull()) { return Optional.empty(); } @@ -505,8 +505,8 @@ private DBDocumentEntity readDBDocumentEntity(final JsonNode source) { private ParameterValueProvider getParameterProvider( final ArangoPersistentEntity entity, - final JsonNode source) { - + final JsonNode source + ) { PropertyValueProvider provider = new ArangoPropertyValueProvider(entity, source); return new PersistentEntityParameterValueProvider<>(entity, provider, null); } @@ -781,7 +781,7 @@ private Optional getRefId(final Object source, final ArangoPersistentEnt .map(key -> { if (annotation != null) { return resolverFactory.getReferenceResolver(annotation) - .map(resolver -> resolver.write(source, entity, convertId(key), annotation)) + .map(resolver -> resolver.write(source, entity, convertId(key))) .orElseThrow(() -> new IllegalArgumentException("Missing reference resolver for " + annotation)); } else { return MetadataUtils.createIdFromCollectionAndKey(entity.getCollection(), convertId(key)); diff --git a/src/main/java/com/arangodb/springframework/core/convert/resolver/AbstractResolver.java b/src/main/java/com/arangodb/springframework/core/convert/resolver/AbstractResolver.java index 14da9cf4..a98372dc 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/resolver/AbstractResolver.java +++ b/src/main/java/com/arangodb/springframework/core/convert/resolver/AbstractResolver.java @@ -21,8 +21,8 @@ package com.arangodb.springframework.core.convert.resolver; import java.io.Serializable; -import java.lang.annotation.Annotation; import java.lang.reflect.Method; +import java.util.function.Supplier; import org.aopalliance.intercept.MethodInvocation; import org.springframework.aop.framework.ProxyFactory; @@ -39,9 +39,8 @@ /** * @author Mark Vollmary * @author Christian Lechner - * */ -public abstract class AbstractResolver { +public abstract class AbstractResolver { private static final Method GET_ENTITY_METHOD; private static final Method GET_REF_ID_METHOD; @@ -66,19 +65,11 @@ protected AbstractResolver(final ConversionService conversionService) { this.objenesis = new ObjenesisStd(true); } - public interface ResolverCallback { - - Object resolve(String id, TypeInformation type, A annotation); - - } - - @SuppressWarnings({ "rawtypes", "unchecked" }) protected Object proxy( final String id, final TypeInformation type, - final A annotation, - final ResolverCallback callback) { - final ProxyInterceptor interceptor = new ProxyInterceptor(id, type, annotation, callback, conversionService); + final Supplier callback) { + final ProxyInterceptor interceptor = new ProxyInterceptor(id, type, callback, conversionService); if (type.getType().isInterface()) { final ProxyFactory proxyFactory = new ProxyFactory(new Class[] { type.getType() }); for (final Class interf : type.getType().getInterfaces()) { @@ -102,24 +93,22 @@ private Class enhancedTypeFor(final Class type) { return enhancer.createClass(); } - static class ProxyInterceptor implements Serializable, + static class ProxyInterceptor implements Serializable, org.springframework.cglib.proxy.MethodInterceptor, org.aopalliance.intercept.MethodInterceptor { private static final long serialVersionUID = -6722757823918987065L; private final String id; final TypeInformation type; - private final A annotation; - private final ResolverCallback callback; + private final Supplier callback; private volatile boolean resolved; private Object result; private final ConversionService conversionService; - public ProxyInterceptor(final String id, final TypeInformation type, final A annotation, - final ResolverCallback callback, final ConversionService conversionService) { + public ProxyInterceptor(final String id, final TypeInformation type, + final Supplier callback, final ConversionService conversionService) { super(); this.id = id; this.type = type; - this.annotation = annotation; this.callback = callback; this.conversionService = conversionService; result = null; @@ -178,7 +167,7 @@ private Object ensureResolved() { private synchronized Object resolve() { if (!resolved) { - return convertIfNecessary(callback.resolve(id, type, annotation), type.getType()); + return convertIfNecessary(callback.get(), type.getType()); } return result; } diff --git a/src/main/java/com/arangodb/springframework/core/convert/resolver/DocumentFromResolver.java b/src/main/java/com/arangodb/springframework/core/convert/resolver/DocumentFromResolver.java index 46fa5ef1..4f32a8cc 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/resolver/DocumentFromResolver.java +++ b/src/main/java/com/arangodb/springframework/core/convert/resolver/DocumentFromResolver.java @@ -30,49 +30,50 @@ import java.util.Collection; import java.util.HashMap; import java.util.Map; +import java.util.function.Supplier; /** * @author Mark Vollmary * @author Christian Lechner - * */ -public class DocumentFromResolver extends AbstractResolver implements RelationResolver { +public class DocumentFromResolver extends AbstractResolver implements RelationResolver { - private final ArangoOperations template; + private final ArangoOperations template; - public DocumentFromResolver(final ArangoOperations template) { - super(template.getConverter().getConversionService()); - this.template = template; - } + public DocumentFromResolver(final ArangoOperations template) { + super(template.getConverter().getConversionService()); + this.template = template; + } - @Override - public Object resolveOne(final String id, final TypeInformation type, Collection> traversedTypes, final From annotation) { - return annotation.lazy() ? proxy(id, type, annotation, (i, t, a) -> _resolveOne(i, t)) - : _resolveOne(id, type); - } + @Override + public Object resolveOne(final String id, final TypeInformation type, Collection> traversedTypes, + final From annotation) { + Supplier supplier = () -> _resolveOne(id, type); + return annotation.lazy() ? proxy(id, type, supplier) : supplier.get(); + } - private Object _resolveOne(final String id, final TypeInformation type) { - ArangoCursor it = _resolve(id, type.getType(), true); - return it.hasNext() ? it.next() : null; - } + private Object _resolveOne(final String id, final TypeInformation type) { + ArangoCursor it = _resolve(id, type.getType(), true); + return it.hasNext() ? it.next() : null; + } - @Override - public Object resolveMultiple(final String id, final TypeInformation type, Collection> traversedTypes, final From annotation) { - return annotation.lazy() ? proxy(id, type, annotation, (i, t, a) -> _resolveMultiple(i, t)) - : _resolveMultiple(id, type); - } + @Override + public Object resolveMultiple(final String id, final TypeInformation type, Collection> traversedTypes, + final From annotation) { + Supplier supplier = () -> _resolveMultiple(id, type); + return annotation.lazy() ? proxy(id, type, supplier) : supplier.get(); + } - private Object _resolveMultiple(final String id, final TypeInformation type) { - return _resolve(id, getNonNullComponentType(type).getType(), false).asListRemaining(); - } + private Object _resolveMultiple(final String id, final TypeInformation type) { + return _resolve(id, getNonNullComponentType(type).getType(), false).asListRemaining(); + } - private ArangoCursor _resolve(final String id, final Class type, final boolean limit) { - final String query = String.format("FOR e IN @@edge FILTER e._from == @id %s RETURN e", limit ? "LIMIT 1" : ""); - Map bindVars = new HashMap<>(); - bindVars.put("@edge", type); - bindVars.put("id", id); - return template.query(query, bindVars, new AqlQueryOptions(), - type); - } + private ArangoCursor _resolve(final String id, final Class type, final boolean limit) { + final String query = String.format("FOR e IN @@edge FILTER e._from == @id %s RETURN e", limit ? "LIMIT 1" : ""); + Map bindVars = new HashMap<>(); + bindVars.put("@edge", type); + bindVars.put("id", id); + return template.query(query, bindVars, new AqlQueryOptions(), type); + } } diff --git a/src/main/java/com/arangodb/springframework/core/convert/resolver/DocumentToResolver.java b/src/main/java/com/arangodb/springframework/core/convert/resolver/DocumentToResolver.java index c4fdc7e0..e8440a4e 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/resolver/DocumentToResolver.java +++ b/src/main/java/com/arangodb/springframework/core/convert/resolver/DocumentToResolver.java @@ -30,47 +30,50 @@ import java.util.Collection; import java.util.HashMap; import java.util.Map; +import java.util.function.Supplier; /** * @author Mark Vollmary * @author Christian Lechner - * */ -public class DocumentToResolver extends AbstractResolver implements RelationResolver { +public class DocumentToResolver extends AbstractResolver implements RelationResolver { - private final ArangoOperations template; + private final ArangoOperations template; - public DocumentToResolver(final ArangoOperations template) { - super(template.getConverter().getConversionService()); - this.template = template; - } + public DocumentToResolver(final ArangoOperations template) { + super(template.getConverter().getConversionService()); + this.template = template; + } - @Override - public Object resolveOne(final String id, final TypeInformation type, Collection> traversedTypes, final To annotation) { - return annotation.lazy() ? proxy(id, type, annotation, (i, t, a) -> _resolveOne(i, t)) : _resolveOne(id, type); - } + @Override + public Object resolveOne(final String id, final TypeInformation type, Collection> traversedTypes, + final To annotation) { + Supplier supplier = () -> _resolveOne(id, type); + return annotation.lazy() ? proxy(id, type, supplier) : supplier.get(); + } - private Object _resolveOne(final String id, final TypeInformation type) { - ArangoCursor it = _resolve(id, type.getType(), true); - return it.hasNext() ? it.next() : null; - } + private Object _resolveOne(final String id, final TypeInformation type) { + ArangoCursor it = _resolve(id, type.getType(), true); + return it.hasNext() ? it.next() : null; + } - @Override - public Object resolveMultiple(final String id, final TypeInformation type, Collection> traversedTypes, final To annotation) { - return annotation.lazy() ? proxy(id, type, annotation, (i, t, a) -> _resolveMultiple(i, t)) - : _resolveMultiple(id, type); - } + @Override + public Object resolveMultiple(final String id, final TypeInformation type, Collection> traversedTypes, + final To annotation) { + Supplier supplier = () -> _resolveMultiple(id, type); + return annotation.lazy() ? proxy(id, type, supplier) : supplier.get(); + } - private Object _resolveMultiple(final String id, final TypeInformation type) { - return _resolve(id, getNonNullComponentType(type).getType(), false).asListRemaining(); - } + private Object _resolveMultiple(final String id, final TypeInformation type) { + return _resolve(id, getNonNullComponentType(type).getType(), false).asListRemaining(); + } - private ArangoCursor _resolve(final String id, final Class type, final boolean limit) { - final String query = String.format("FOR e IN @@edge FILTER e._to == @id %s RETURN e", limit ? "LIMIT 1" : ""); - Map bindVars = new HashMap<>(); - bindVars.put("@edge", type); - bindVars.put("id", id); - return template.query(query, bindVars, new AqlQueryOptions(), type); - } + private ArangoCursor _resolve(final String id, final Class type, final boolean limit) { + final String query = String.format("FOR e IN @@edge FILTER e._to == @id %s RETURN e", limit ? "LIMIT 1" : ""); + Map bindVars = new HashMap<>(); + bindVars.put("@edge", type); + bindVars.put("id", id); + return template.query(query, bindVars, new AqlQueryOptions(), type); + } } diff --git a/src/main/java/com/arangodb/springframework/core/convert/resolver/EdgeFromResolver.java b/src/main/java/com/arangodb/springframework/core/convert/resolver/EdgeFromResolver.java index 40ee75c8..a1dcc083 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/resolver/EdgeFromResolver.java +++ b/src/main/java/com/arangodb/springframework/core/convert/resolver/EdgeFromResolver.java @@ -26,12 +26,12 @@ import com.arangodb.springframework.core.ArangoOperations; import java.util.Collection; +import java.util.function.Supplier; /** * @author Mark Vollmary - * */ -public class EdgeFromResolver extends AbstractResolver implements RelationResolver { +public class EdgeFromResolver extends AbstractResolver implements RelationResolver { private final ArangoOperations template; @@ -41,8 +41,10 @@ public EdgeFromResolver(final ArangoOperations template) { } @Override - public Object resolveOne(final String id, final TypeInformation type, Collection> traversedTypes, final From annotation) { - return annotation.lazy() ? proxy(id, type, annotation, (i, t, a) -> _resolveOne(i, t)) : _resolveOne(id, type); + public Object resolveOne(final String id, final TypeInformation type, Collection> traversedTypes, + final From annotation) { + Supplier supplier = () -> _resolveOne(id, type); + return annotation.lazy() ? proxy(id, type, supplier) : supplier.get(); } private Object _resolveOne(final String id, final TypeInformation type) { @@ -51,7 +53,8 @@ private Object _resolveOne(final String id, final TypeInformation type) { } @Override - public Object resolveMultiple(final String id, final TypeInformation type, Collection> traversedTypes, final From annotation) { + public Object resolveMultiple(final String id, final TypeInformation type, Collection> traversedTypes, + final From annotation) { throw new UnsupportedOperationException("Edges with multiple 'from' values are not supported."); } diff --git a/src/main/java/com/arangodb/springframework/core/convert/resolver/EdgeToResolver.java b/src/main/java/com/arangodb/springframework/core/convert/resolver/EdgeToResolver.java index d16250be..69e92369 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/resolver/EdgeToResolver.java +++ b/src/main/java/com/arangodb/springframework/core/convert/resolver/EdgeToResolver.java @@ -26,12 +26,12 @@ import com.arangodb.springframework.core.ArangoOperations; import java.util.Collection; +import java.util.function.Supplier; /** * @author Mark Vollmary - * */ -public class EdgeToResolver extends AbstractResolver implements RelationResolver { +public class EdgeToResolver extends AbstractResolver implements RelationResolver { private final ArangoOperations template; @@ -41,8 +41,10 @@ public EdgeToResolver(final ArangoOperations template) { } @Override - public Object resolveOne(final String id, final TypeInformation type, Collection> traversedTypes, final To annotation) { - return annotation.lazy() ? proxy(id, type, annotation, (i, t, a) -> _resolveOne(i, t)) : _resolveOne(id, type); + public Object resolveOne(final String id, final TypeInformation type, Collection> traversedTypes, + final To annotation) { + Supplier supplier = () -> _resolveOne(id, type); + return annotation.lazy() ? proxy(id, type, supplier) : supplier.get(); } private Object _resolveOne(final String id, final TypeInformation type) { diff --git a/src/main/java/com/arangodb/springframework/core/convert/resolver/RefResolver.java b/src/main/java/com/arangodb/springframework/core/convert/resolver/RefResolver.java index 39d46895..6dfc8058 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/resolver/RefResolver.java +++ b/src/main/java/com/arangodb/springframework/core/convert/resolver/RefResolver.java @@ -21,6 +21,7 @@ package com.arangodb.springframework.core.convert.resolver; import java.util.Collection; +import java.util.function.Supplier; import java.util.stream.Collectors; import com.arangodb.springframework.core.mapping.ArangoPersistentEntity; @@ -33,43 +34,37 @@ /** * @author Mark Vollmary * @author Christian Lechner - * */ -public class RefResolver extends AbstractResolver - implements ReferenceResolver, AbstractResolver.ResolverCallback { - - private final ArangoOperations template; - - public RefResolver(final ArangoOperations template) { - super(template.getConverter().getConversionService()); - this.template = template; - } +public class RefResolver extends AbstractResolver implements ReferenceResolver { - @Override - public Object resolveOne(final String id, final TypeInformation type, final Ref annotation) { - return annotation.lazy() ? proxy(id, type, annotation, this) : _resolve(id, type); - } + private final ArangoOperations template; - @Override - public Object resolveMultiple(final Collection ids, final TypeInformation type, final Ref annotation) { - return ids.stream().map(id -> resolveOne(id, getNonNullComponentType(type), annotation)) - .collect(Collectors.toList()); - } + public RefResolver(final ArangoOperations template) { + super(template.getConverter().getConversionService()); + this.template = template; + } - @Override - public Object resolve(final String id, final TypeInformation type, final Ref annotation) { - return _resolve(id, type); - } + @Override + public Object resolveOne(final String id, final TypeInformation type, final Ref annotation) { + Supplier supplier = () -> _resolve(id, type); + return annotation.lazy() ? proxy(id, type, supplier) : supplier.get(); + } - public Object _resolve(final String id, final TypeInformation type) { - return template.find(id, type.getType()) - .orElseThrow(() -> cannotResolveException(id, type)); - } + @Override + public Object resolveMultiple(final Collection ids, final TypeInformation type, final Ref annotation) { + return ids.stream() + .map(id -> resolveOne(id, getNonNullComponentType(type), annotation)) + .collect(Collectors.toList()); + } - @Override - public String write(final Object source, final ArangoPersistentEntity entity, final Object id, final Ref annotation) { - return MetadataUtils.createIdFromCollectionAndKey(entity.getCollection(), String.valueOf(id)); - } + private Object _resolve(final String id, final TypeInformation type) { + return template.find(id, type.getType()) + .orElseThrow(() -> cannotResolveException(id, type)); + } + @Override + public String write(final Object source, final ArangoPersistentEntity entity, final Object id) { + return MetadataUtils.createIdFromCollectionAndKey(entity.getCollection(), String.valueOf(id)); + } } diff --git a/src/main/java/com/arangodb/springframework/core/convert/resolver/ReferenceResolver.java b/src/main/java/com/arangodb/springframework/core/convert/resolver/ReferenceResolver.java index 667aadb5..17c62411 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/resolver/ReferenceResolver.java +++ b/src/main/java/com/arangodb/springframework/core/convert/resolver/ReferenceResolver.java @@ -23,20 +23,14 @@ import java.lang.annotation.Annotation; import java.util.Collection; -import com.arangodb.springframework.annotation.Ref; import com.arangodb.springframework.core.mapping.ArangoPersistentEntity; import org.springframework.data.util.TypeInformation; /** * @author Mark Vollmary - * */ public interface ReferenceResolver { - Object resolveOne(String id, TypeInformation type, A annotation); - Object resolveMultiple(Collection ids, TypeInformation type, A annotation); - - public String write(Object source, ArangoPersistentEntity entity, Object id, Ref annotation); - + String write(Object source, ArangoPersistentEntity entity, Object id); } diff --git a/src/main/java/com/arangodb/springframework/core/convert/resolver/RelationResolver.java b/src/main/java/com/arangodb/springframework/core/convert/resolver/RelationResolver.java index b5af2b69..6cd62c18 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/resolver/RelationResolver.java +++ b/src/main/java/com/arangodb/springframework/core/convert/resolver/RelationResolver.java @@ -27,12 +27,9 @@ /** * @author Mark Vollmary - * */ public interface RelationResolver { - Object resolveOne(String id, TypeInformation type, Collection> traversedTypes, A annotation); - Object resolveMultiple(String id, TypeInformation type, Collection> traversedTypes, A annotation); } diff --git a/src/main/java/com/arangodb/springframework/core/convert/resolver/RelationsResolver.java b/src/main/java/com/arangodb/springframework/core/convert/resolver/RelationsResolver.java index e0f6aa07..7a69c892 100644 --- a/src/main/java/com/arangodb/springframework/core/convert/resolver/RelationsResolver.java +++ b/src/main/java/com/arangodb/springframework/core/convert/resolver/RelationsResolver.java @@ -21,6 +21,7 @@ package com.arangodb.springframework.core.convert.resolver; import java.util.*; +import java.util.function.Supplier; import java.util.stream.Collectors; import org.springframework.data.util.TypeInformation; @@ -32,79 +33,82 @@ /** * @author Mark Vollmary * @author Christian Lechner - * */ -public class RelationsResolver extends AbstractResolver implements RelationResolver { - - private final ArangoOperations template; - - public RelationsResolver(final ArangoOperations template) { - super(template.getConverter().getConversionService()); - this.template = template; - } - - @Override - public Object resolveOne(final String id, final TypeInformation type, final Collection> traversedTypes, final Relations annotation) { - return annotation.lazy() ? proxy(id, type, annotation, (i, t, a) -> _resolveOne(i, t, traversedTypes, a)) - : _resolveOne(id, type, traversedTypes, annotation); - } - - @Override - public Object resolveMultiple(final String id, final TypeInformation type, final Collection> traversedTypes, final Relations annotation) { - return annotation.lazy() ? proxy(id, type, annotation, (i, t, a) -> _resolveMultiple(i, t, traversedTypes, a)) - : _resolveMultiple(id, type, traversedTypes, annotation); - } - - private Object _resolveOne(final String id, final TypeInformation type, final Collection> traversedTypes, final Relations annotation) { - Collection> rawTypes = new ArrayList<>(); - for (TypeInformation it : traversedTypes) { - rawTypes.add(it.getType()); - } - ArangoCursor it = _resolve(id, type.getType(), rawTypes, annotation, true); - return it.hasNext() ? it.next() : null; - } - - private Object _resolveMultiple(final String id, final TypeInformation type, final Collection> traversedTypes, final Relations annotation) { - Collection> rawTypes = new ArrayList<>(); - for (TypeInformation it : traversedTypes) { - rawTypes.add(it.getType()); - } - return _resolve(id, getNonNullComponentType(type).getType(), rawTypes, annotation, false).asListRemaining(); - } - - private ArangoCursor _resolve( - final String id, - final Class type, - final Collection> traversedTypes, - final Relations annotation, - final boolean limit) { - - final String edges = Arrays.stream(annotation.edges()).map(e -> template.collection(e).name()) - .collect(Collectors.joining(",")); - - List> allTraversedTypes = new ArrayList<>(); - allTraversedTypes.add(type); - allTraversedTypes.addAll(traversedTypes); - - Map bindVars = new HashMap<>(); - bindVars.put("start", id); - StringBuilder withClause = new StringBuilder("WITH "); - for (int i = 0; i < allTraversedTypes.size(); i++) { - bindVars.put("@with" + i, allTraversedTypes.get(i)); - if (i > 0) withClause.append(", "); - withClause.append("@@with").append(i); - } - - final String query = String.format( - "%s FOR v IN %d .. %d %s @start %s OPTIONS {bfs: true, uniqueVertices: \"global\"} %s RETURN v", // - withClause, // - Math.max(1, annotation.minDepth()), // - Math.max(1, annotation.maxDepth()), // - annotation.direction(), // - edges, // - limit ? "LIMIT 1" : ""); - - return template.query(query, bindVars, type); - } +public class RelationsResolver extends AbstractResolver implements RelationResolver { + + private final ArangoOperations template; + + public RelationsResolver(final ArangoOperations template) { + super(template.getConverter().getConversionService()); + this.template = template; + } + + @Override + public Object resolveOne(final String id, final TypeInformation type, final Collection> traversedTypes, + final Relations annotation) { + Supplier supplier = () -> _resolveOne(id, type, traversedTypes, annotation); + return annotation.lazy() ? proxy(id, type, supplier) : supplier.get(); + } + + @Override + public Object resolveMultiple(final String id, final TypeInformation type, final Collection> traversedTypes, + final Relations annotation) { + Supplier supplier = () -> _resolveMultiple(id, type, traversedTypes, annotation); + return annotation.lazy() ? proxy(id, type, supplier) : supplier.get(); + } + + private Object _resolveOne(final String id, final TypeInformation type, final Collection> traversedTypes, + final Relations annotation) { + Collection> rawTypes = new ArrayList<>(); + for (TypeInformation it : traversedTypes) { + rawTypes.add(it.getType()); + } + ArangoCursor it = _resolve(id, type.getType(), rawTypes, annotation, true); + return it.hasNext() ? it.next() : null; + } + + private Object _resolveMultiple(final String id, final TypeInformation type, final Collection> traversedTypes, + final Relations annotation) { + Collection> rawTypes = new ArrayList<>(); + for (TypeInformation it : traversedTypes) { + rawTypes.add(it.getType()); + } + return _resolve(id, getNonNullComponentType(type).getType(), rawTypes, annotation, false).asListRemaining(); + } + + private ArangoCursor _resolve( + final String id, + final Class type, + final Collection> traversedTypes, + final Relations annotation, + final boolean limit) { + + final String edges = Arrays.stream(annotation.edges()).map(e -> template.collection(e).name()) + .collect(Collectors.joining(",")); + + List> allTraversedTypes = new ArrayList<>(); + allTraversedTypes.add(type); + allTraversedTypes.addAll(traversedTypes); + + Map bindVars = new HashMap<>(); + bindVars.put("start", id); + StringBuilder withClause = new StringBuilder("WITH "); + for (int i = 0; i < allTraversedTypes.size(); i++) { + bindVars.put("@with" + i, allTraversedTypes.get(i)); + if (i > 0) withClause.append(", "); + withClause.append("@@with").append(i); + } + + final String query = String.format( + "%s FOR v IN %d .. %d %s @start %s OPTIONS {bfs: true, uniqueVertices: \"global\"} %s RETURN v", // + withClause, // + Math.max(1, annotation.minDepth()), // + Math.max(1, annotation.maxDepth()), // + annotation.direction(), // + edges, // + limit ? "LIMIT 1" : ""); + + return template.query(query, bindVars, type); + } } diff --git a/src/main/java/com/arangodb/springframework/core/mapping/ArangoSimpleTypes.java b/src/main/java/com/arangodb/springframework/core/mapping/ArangoSimpleTypes.java index b02c2379..1f6dcd07 100644 --- a/src/main/java/com/arangodb/springframework/core/mapping/ArangoSimpleTypes.java +++ b/src/main/java/com/arangodb/springframework/core/mapping/ArangoSimpleTypes.java @@ -51,8 +51,9 @@ public final class ArangoSimpleTypes { static { final Set> simpleTypes = new HashSet<>(); - // com.arangodb.* simpleTypes.add(JsonNode.class); + + // com.arangodb.* simpleTypes.add(DBDocumentEntity.class); // java.math.* diff --git a/src/main/java/com/arangodb/springframework/repository/ArangoExampleConverter.java b/src/main/java/com/arangodb/springframework/repository/ArangoExampleConverter.java index 380b464a..db8f2555 100644 --- a/src/main/java/com/arangodb/springframework/repository/ArangoExampleConverter.java +++ b/src/main/java/com/arangodb/springframework/repository/ArangoExampleConverter.java @@ -130,7 +130,7 @@ private void traversePropertyTree(final Example example, final StringBuilder if (property.getRef().isPresent()) { final Optional> resolver = resolverFactory .getReferenceResolver(property.getRef().get()); - refIdValue = resolver.get().write(value, persistentEntity, idValue, property.getRef().get()); + refIdValue = resolver.get().write(value, persistentEntity, idValue); } else { refIdValue = String.format("%s/%s", persistentEntity.getCollection(), idValue); } diff --git a/src/main/java/com/arangodb/springframework/repository/query/AbstractArangoQuery.java b/src/main/java/com/arangodb/springframework/repository/query/AbstractArangoQuery.java index 2385326a..54c1f6af 100644 --- a/src/main/java/com/arangodb/springframework/repository/query/AbstractArangoQuery.java +++ b/src/main/java/com/arangodb/springframework/repository/query/AbstractArangoQuery.java @@ -201,8 +201,7 @@ private Object convertResult(final ArangoCursor result, final ArangoParameter } return (Integer) result.next() > 0; } - final ArangoResultConverter resultConverter = new ArangoResultConverter(accessor, result, operations, - domainClass); + final ArangoResultConverter resultConverter = new ArangoResultConverter<>(accessor, result, operations, domainClass); return resultConverter.convertResult(method.getReturnType().getType()); } diff --git a/src/main/java/com/arangodb/springframework/repository/query/ArangoResultConverter.java b/src/main/java/com/arangodb/springframework/repository/query/ArangoResultConverter.java index 6e2e6ae0..fda4664c 100644 --- a/src/main/java/com/arangodb/springframework/repository/query/ArangoResultConverter.java +++ b/src/main/java/com/arangodb/springframework/repository/query/ArangoResultConverter.java @@ -20,12 +20,9 @@ package com.arangodb.springframework.repository.query; -import java.lang.reflect.Method; import java.util.Collection; -import java.util.HashMap; import java.util.LinkedList; import java.util.List; -import java.util.Map; import java.util.Optional; import java.util.Set; import java.util.Spliterators; @@ -55,7 +52,7 @@ * @author Mark Vollmary * @author Christian Lechner */ -public class ArangoResultConverter { +public class ArangoResultConverter { private final static String MISSING_FULL_COUNT = "Query result does not contain the full result count! " + "The most likely cause is a forgotten LIMIT clause in the query."; @@ -63,43 +60,17 @@ public class ArangoResultConverter { private final ArangoParameterAccessor accessor; private final ArangoCursor result; private final ArangoOperations operations; - private final Class domainClass; + private final Class domainClass; - private static Map TYPE_MAP = new HashMap<>(); - - /** - * Build static map of all supported return types and the method used to convert them - */ - static { - try { - TYPE_MAP.put(List.class, ArangoResultConverter.class.getMethod("convertList")); - TYPE_MAP.put(Iterable.class, ArangoResultConverter.class.getMethod("convertList")); - TYPE_MAP.put(Collection.class, ArangoResultConverter.class.getMethod("convertList")); - TYPE_MAP.put(Page.class, ArangoResultConverter.class.getMethod("convertPage")); - TYPE_MAP.put(Slice.class, ArangoResultConverter.class.getMethod("convertPage")); - TYPE_MAP.put(Set.class, ArangoResultConverter.class.getMethod("convertSet")); - TYPE_MAP.put(ArangoCursor.class, ArangoResultConverter.class.getMethod("convertArangoCursor")); - TYPE_MAP.put(GeoResult.class, ArangoResultConverter.class.getMethod("convertGeoResult")); - TYPE_MAP.put(GeoResults.class, ArangoResultConverter.class.getMethod("convertGeoResults")); - TYPE_MAP.put(GeoPage.class, ArangoResultConverter.class.getMethod("convertGeoPage")); - TYPE_MAP.put(Optional.class, ArangoResultConverter.class.getMethod("convertOptional")); - TYPE_MAP.put("array", ArangoResultConverter.class.getMethod("convertArray")); - } catch (final NoSuchMethodException e) { - e.printStackTrace(); - } - } /** * @param accessor - * @param result - * the query result returned by the driver - * @param operations - * instance of arangoTemplate - * @param domainClass - * class type of documents + * @param result the query result returned by the driver + * @param operations instance of arangoTemplate + * @param domainClass class type of documents */ public ArangoResultConverter(final ArangoParameterAccessor accessor, final ArangoCursor result, - final ArangoOperations operations, final Class domainClass) { + final ArangoOperations operations, final Class domainClass) { this.accessor = accessor; this.result = result; this.operations = operations; @@ -114,23 +85,40 @@ public ArangoResultConverter(final ArangoParameterAccessor accessor, final Arang */ public Object convertResult(final Class type) { try { + return convert(type); + } catch (final Exception e) { + throw new MappingException(String.format("Can't convert result to type %s!", type.getName()), e); + } + } + + private Object convert(Class type) { if (type.isArray()) { - return TYPE_MAP.get("array").invoke(this); - } - if (!TYPE_MAP.containsKey(type)) { + return convertArray(); + } else if (List.class.equals(type) || Iterable.class.equals(type) || Collection.class.equals(type)) { + return convertList(); + } else if (Page.class.equals(type) || Slice.class.equals(type)) { + return convertPage(); + } else if (Set.class.equals(type)) { + return convertSet(); + } else if (ArangoCursor.class.equals(type)) { + return convertArangoCursor(); + } else if (GeoResult.class.equals(type)) { + return convertGeoResult(); + } else if (GeoResults.class.equals(type)) { + return convertGeoResults(); + } else if (GeoPage.class.equals(type)) { + return convertGeoPage(); + } else if (Optional.class.equals(type)) { + return convertOptional(); + } else { return getNext(result); } - return TYPE_MAP.get(type).invoke(this); - } catch (final Exception e) { - throw new MappingException(String.format("Can't convert result to type %s!", type.getName()), e); - } } /** * Creates a Set return type from the given cursor * - * @param cursor - * query result from driver + * @param cursor query result from driver * @return Set containing the results */ private Set buildSet(final ArangoCursor cursor) { @@ -140,73 +128,41 @@ private Set buildSet(final ArangoCursor cursor) { /** * Build a GeoResult from the given ArangoCursor * - * @param cursor - * query result from driver + * @param cursor query result from driver * @return GeoResult object */ - private GeoResult buildGeoResult(final ArangoCursor cursor) { - GeoResult geoResult = null; - while (cursor.hasNext() && geoResult == null) { - final Object object = cursor.next(); - if (!(object instanceof JsonNode)) { - continue; - } - - final JsonNode slice = (JsonNode) object; - final JsonNode distSlice = slice.get("_distance"); - final Double distanceInMeters = distSlice.isDouble() ? distSlice.doubleValue() : null; - if (distanceInMeters == null) { - continue; - } - - final Object entity = operations.getConverter().read(domainClass, slice); - final Distance distance = new Distance(distanceInMeters / 1000, Metrics.KILOMETERS); - geoResult = new GeoResult<>(entity, distance); - } - return geoResult; + private GeoResult buildGeoResult(final ArangoCursor cursor) { + return buildGeoResult(cursor.next()); } /** * Construct a GeoResult from the given object * - * @param object - * object representing one document in the result + * @param slice object representing one document in the result * @return GeoResult object */ - private GeoResult buildGeoResult(final Object object) { - if (object == null || !(object instanceof JsonNode)) { - return null; - } - - final JsonNode slice = (JsonNode) object; - final JsonNode distSlice = slice.get("_distance"); - final Double distanceInMeters = distSlice.isDouble() ? distSlice.doubleValue() : null; - if (distanceInMeters == null) { - return null; - } - - final Object entity = operations.getConverter().read(domainClass, slice); - final Distance distance = new Distance(distanceInMeters / 1000, Metrics.KILOMETERS); + private GeoResult buildGeoResult(final JsonNode slice) { + JsonNode distSlice = slice.get("_distance"); + Double distanceInMeters = distSlice.isDouble() ? distSlice.doubleValue() : null; + T entity = operations.getConverter().read(domainClass, slice); + // FIXME: Unboxing of 'distanceInMeters' may produce 'NullPointerException' + Distance distance = new Distance(distanceInMeters / 1000, Metrics.KILOMETERS); return new GeoResult<>(entity, distance); } /** * Build a GeoResults object with the ArangoCursor returned by query execution * - * @param cursor - * ArangoCursor containing query results + * @param cursor ArangoCursor containing query results * @return GeoResults object with all results */ - @SuppressWarnings({ "rawtypes", "unchecked" }) - private GeoResults buildGeoResults(final ArangoCursor cursor) { - final List> list = new LinkedList<>(); - cursor.forEachRemaining(o -> { - final GeoResult geoResult = buildGeoResult(o); - if (geoResult != null) { - list.add(geoResult); - } - }); - return new GeoResults(list); + private GeoResults buildGeoResults(final ArangoCursor cursor) { + final List> list = new LinkedList<>(); + cursor.forEachRemaining(o -> list.add(buildGeoResult(o))); + // FIXME: DE-803 + // convert geoResults to Metrics.NEUTRAL before + // invoking GeoResults.GeoResults(java.util.List>) + return new GeoResults<>(list); } public Optional convertOptional() { @@ -230,17 +186,20 @@ public ArangoCursor convertArangoCursor() { return result; } - public GeoResult convertGeoResult() { - return buildGeoResult(result); + @SuppressWarnings("unchecked") + public GeoResult convertGeoResult() { + return buildGeoResult((ArangoCursor) result); } - public GeoResults convertGeoResults() { - return buildGeoResults(result); + @SuppressWarnings("unchecked") + public GeoResults convertGeoResults() { + return buildGeoResults((ArangoCursor) result); } - public GeoPage convertGeoPage() { + @SuppressWarnings("unchecked") + public GeoPage convertGeoPage() { Assert.notNull(result.getStats().getFullCount(), MISSING_FULL_COUNT); - return new GeoPage<>(buildGeoResults(result), accessor.getPageable(), ((Number) result.getStats().getFullCount()).longValue()); + return new GeoPage<>(buildGeoResults((ArangoCursor) result), accessor.getPageable(), ((Number) result.getStats().getFullCount()).longValue()); } public Object convertArray() { diff --git a/src/test/java/com/arangodb/springframework/AbstractArangoTest.java b/src/test/java/com/arangodb/springframework/AbstractArangoTest.java index 362963df..4aea0a03 100644 --- a/src/test/java/com/arangodb/springframework/AbstractArangoTest.java +++ b/src/test/java/com/arangodb/springframework/AbstractArangoTest.java @@ -20,6 +20,7 @@ package com.arangodb.springframework; +import com.arangodb.ArangoDatabase; import org.junit.jupiter.api.AfterAll; import org.junit.jupiter.api.BeforeEach; import org.springframework.beans.factory.annotation.Autowired; @@ -37,6 +38,7 @@ public abstract class AbstractArangoTest { @Autowired protected ArangoOperations template; + protected ArangoDatabase db; protected final Class[] collections; protected AbstractArangoTest(final Class... collections) { @@ -50,6 +52,7 @@ public void before() { template.collection(collection).truncate(); } AbstractArangoTest.staticTemplate = template; + db = template.driver().db(ArangoTestConfiguration.DB); } @AfterAll diff --git a/src/test/java/com/arangodb/springframework/ArangoTestConfiguration.java b/src/test/java/com/arangodb/springframework/ArangoTestConfiguration.java index e0c7c14a..69972d23 100644 --- a/src/test/java/com/arangodb/springframework/ArangoTestConfiguration.java +++ b/src/test/java/com/arangodb/springframework/ArangoTestConfiguration.java @@ -49,7 +49,8 @@ @EnableArangoRepositories(basePackages = { "com.arangodb.springframework.repository", "com.arangodb.springframework.example.polymorphic.repository", - "com.arangodb.springframework.debug.repository"}, + "com.arangodb.springframework.debug.repository", + "com.arangodb.springframework.testdata.chess.repo"}, namedQueriesLocation = "classpath*:arango-named-queries-test.properties") @EnableArangoAuditing(auditorAwareRef = "auditorProvider") public class ArangoTestConfiguration implements ArangoConfiguration { diff --git a/src/test/java/com/arangodb/springframework/core/mapping/CustomMappingTest.java b/src/test/java/com/arangodb/springframework/core/mapping/CustomMappingTest.java index 60ce3a50..23631e8b 100644 --- a/src/test/java/com/arangodb/springframework/core/mapping/CustomMappingTest.java +++ b/src/test/java/com/arangodb/springframework/core/mapping/CustomMappingTest.java @@ -117,7 +117,7 @@ public void jsonNodeToCustom() { @Test public void customToJsonNodeFromDriver() { - ArangoCollection col = template.driver().db(ArangoTestConfiguration.DB).collection("customJsonNodeTestEntity"); + ArangoCollection col = db.collection("customJsonNodeTestEntity"); final DocumentEntity meta = col.insertDocument(new CustomJsonNodeTestEntity("abc")); final BaseDocument doc = col.getDocument(meta.getKey(), BaseDocument.class); assertThat(doc.getAttribute(FIELD), is("abc")); @@ -126,7 +126,7 @@ public void customToJsonNodeFromDriver() { @Test public void jsonNodeToCustomFromDriver() { - ArangoCollection col = template.driver().db(ArangoTestConfiguration.DB).collection("testEntity"); + ArangoCollection col = db.collection("testEntity"); final DocumentEntity meta = col.insertDocument(new TestEntity("abc")); final CustomJsonNodeTestEntity doc = col.getDocument(meta.getKey(), CustomJsonNodeTestEntity.class); assertThat(doc.getValue(), is("abc")); @@ -192,7 +192,7 @@ public void jsonNodeToDBEntity() { @Test public void customToDBEntityFromDriver() { - ArangoCollection col = template.driver().db(ArangoTestConfiguration.DB).collection("customDBEntityTestEntity"); + ArangoCollection col = db.collection("customDBEntityTestEntity"); final DocumentEntity meta = col.insertDocument(new CustomDBEntityTestEntity("abc")); final BaseDocument doc = col.getDocument(meta.getKey(), BaseDocument.class); assertThat(doc.getAttribute(FIELD), is("abc")); @@ -201,7 +201,7 @@ public void customToDBEntityFromDriver() { @Test public void jsonNodeToDBEntityFromDriver() { - ArangoCollection col = template.driver().db(ArangoTestConfiguration.DB).collection("testEntity"); + ArangoCollection col = db.collection("testEntity"); final DocumentEntity meta = col.insertDocument(new TestEntity("abc")); final CustomDBEntityTestEntity doc = col.getDocument(meta.getKey(), CustomDBEntityTestEntity.class); assertThat(doc.getValue(), is("abc")); diff --git a/src/test/java/com/arangodb/springframework/core/mapping/GeneralMappingTest.java b/src/test/java/com/arangodb/springframework/core/mapping/GeneralMappingTest.java index 34cb823d..aeaf1819 100644 --- a/src/test/java/com/arangodb/springframework/core/mapping/GeneralMappingTest.java +++ b/src/test/java/com/arangodb/springframework/core/mapping/GeneralMappingTest.java @@ -97,7 +97,7 @@ public void fieldNameAnnotation() { entity.test = "1234"; final DocumentEntity res = template.insert(entity); String colName = res.getId().split("/")[0]; - final ObjectNode slice = template.driver().db(ArangoTestConfiguration.DB) + final ObjectNode slice = db .collection(colName) .getDocument(res.getKey(), ObjectNode.class); assertThat(slice, is(notNullValue())); @@ -632,7 +632,7 @@ public void auditingTest() throws InterruptedException { assertThat(find.modifiedBy.getId(), is(createID)); } - final ObjectNode doc = template.driver().db(ArangoTestConfiguration.DB).collection("auditingTestEntity") + final ObjectNode doc = db.collection("auditingTestEntity") .getDocument(value.id, ObjectNode.class); assertThat(doc, is(notNullValue())); assertThat(doc.get("createdBy").isObject(), is(true)); diff --git a/src/test/java/com/arangodb/springframework/core/mapping/InheritanceMappingTest.java b/src/test/java/com/arangodb/springframework/core/mapping/InheritanceMappingTest.java index ef317505..d88e7258 100644 --- a/src/test/java/com/arangodb/springframework/core/mapping/InheritanceMappingTest.java +++ b/src/test/java/com/arangodb/springframework/core/mapping/InheritanceMappingTest.java @@ -331,8 +331,8 @@ static class SubClassWithOwnDocumentAnnotation extends BasicTestEntity { public void overrideDocumentAnnotation() { final SubClassWithOwnDocumentAnnotation doc = new SubClassWithOwnDocumentAnnotation(); template.insert(doc); - assertThat(template.driver().db(ArangoTestConfiguration.DB).collection("overrideDocAn").exists(), is(true)); - assertThat(template.driver().db(ArangoTestConfiguration.DB).collection("overrideDocAn").count().getCount(), + assertThat(db.collection("overrideDocAn").exists(), is(true)); + assertThat(db.collection("overrideDocAn").count().getCount(), is(1L)); } @@ -351,8 +351,8 @@ public void overrideEdgeAnnotation() { edge.from = from; edge.to = to; template.insert(edge); - assertThat(template.driver().db(ArangoTestConfiguration.DB).collection("overrideEdgeAn").exists(), is(true)); - assertThat(template.driver().db(ArangoTestConfiguration.DB).collection("overrideEdgeAn").count().getCount(), + assertThat(db.collection("overrideEdgeAn").exists(), is(true)); + assertThat(db.collection("overrideEdgeAn").count().getCount(), is(1L)); } diff --git a/src/test/java/com/arangodb/springframework/core/mapping/MultiTenancyCollectionLevelMappingTest.java b/src/test/java/com/arangodb/springframework/core/mapping/MultiTenancyCollectionLevelMappingTest.java index 1a83aa69..1ddf0b52 100644 --- a/src/test/java/com/arangodb/springframework/core/mapping/MultiTenancyCollectionLevelMappingTest.java +++ b/src/test/java/com/arangodb/springframework/core/mapping/MultiTenancyCollectionLevelMappingTest.java @@ -50,20 +50,20 @@ public void collectionLevel() { { tenantProvider.setId("tenant00"); template.insert(new MultiTenancyTestEntity()); - assertThat(template.driver().db(ArangoTestConfiguration.DB).collection("tenant00_collection").exists(), + assertThat(db.collection("tenant00_collection").exists(), is(true)); } { tenantProvider.setId("tenant01"); template.insert(new MultiTenancyTestEntity()); - assertThat(template.driver().db(ArangoTestConfiguration.DB).collection("tenant01_collection").exists(), + assertThat(db.collection("tenant01_collection").exists(), is(true)); } assertThat( - template.driver().db(ArangoTestConfiguration.DB).collection("tenant00_collection").count().getCount(), + db.collection("tenant00_collection").count().getCount(), is(1L)); assertThat( - template.driver().db(ArangoTestConfiguration.DB).collection("tenant01_collection").count().getCount(), + db.collection("tenant01_collection").count().getCount(), is(1L)); } diff --git a/src/test/java/com/arangodb/springframework/example/polymorphic/template/PolymorphicTemplate.java b/src/test/java/com/arangodb/springframework/example/polymorphic/template/PolymorphicTemplate.java index e1a883eb..54d82683 100644 --- a/src/test/java/com/arangodb/springframework/example/polymorphic/template/PolymorphicTemplate.java +++ b/src/test/java/com/arangodb/springframework/example/polymorphic/template/PolymorphicTemplate.java @@ -49,7 +49,7 @@ public class PolymorphicTemplate extends AbstractArangoTest { private ArangoCollection col() { - return template.driver().db(ArangoTestConfiguration.DB).collection("animals"); + return db.collection("animals"); } @BeforeEach diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/AbstractRepositoryTest.java b/src/test/java/com/arangodb/springframework/testdata/chess/AbstractRepositoryTest.java new file mode 100644 index 00000000..d82ce7ae --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/AbstractRepositoryTest.java @@ -0,0 +1,81 @@ +package com.arangodb.springframework.testdata.chess; + +import com.arangodb.springframework.AbstractArangoTest; +import com.arangodb.springframework.core.ArangoOperations; +import com.arangodb.springframework.testdata.chess.entity.Player; +import com.arangodb.springframework.testdata.chess.entity.Score; +import com.arangodb.springframework.testdata.chess.entity.Tournament; +import org.junit.jupiter.api.BeforeEach; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.data.geo.Point; + +import java.time.LocalDate; +import java.util.List; + +abstract class AbstractRepositoryTest extends AbstractArangoTest { + + @Autowired + private ArangoOperations ops; + + protected List players = List.of( + new Player("Magnus Carlsen", 2830, "Norway"), + new Player("Fabiano Caruana", 2803, "US"), + new Player("Maxime Vachier-Lagrave", 2732, "France"), + new Player("Hikaru Nakamura", 2789, "US"), + new Player("Ding Liren", 2762, "China"), + new Player("Wesley So", 2757, "US"), + new Player("Alireza Firouzja", 2760, "France"), + new Player("Anish Giri", 2745, "Netherlands"), + new Player("Ian Nepomniachtchi", 2758, "Russia") + ); + + protected List tournaments = List.of( + new Tournament( + "Tata Steel 2023", + LocalDate.of(2023, 1, 13), + "Wijk aan Zee", + new Point(4.6, 52.5) + ), + new Tournament( + "World Chess Championship 2023", + LocalDate.of(2023, 4, 9), + "Astana", + new Point(71.422222, 51.147222) + ), + new Tournament( + "Norway Chess 2023", + LocalDate.of(2023, 5, 30), + "Stavanger", + new Point(5.731389, 58.97) + ) + ); + + protected List scores = List.of( + new Score(players.get(7), tournaments.get(0), 1), + new Score(players.get(0), tournaments.get(0), 3), + new Score(players.get(5), tournaments.get(0), 4), + new Score(players.get(1), tournaments.get(0), 5), + new Score(players.get(4), tournaments.get(1), 1), + new Score(players.get(8), tournaments.get(1), 2), + new Score(players.get(3), tournaments.get(1), 4), + new Score(players.get(1), tournaments.get(1), 5), + new Score(players.get(6), tournaments.get(1), 6), + new Score(players.get(3), tournaments.get(2), 1), + new Score(players.get(1), tournaments.get(2), 2), + new Score(players.get(7), tournaments.get(2), 4), + new Score(players.get(5), tournaments.get(2), 5), + new Score(players.get(0), tournaments.get(2), 6) + ); + + protected AbstractRepositoryTest() { + super(Player.class, Score.class, Tournament.class); + } + + @BeforeEach + void importData() { + ops.insertAll(players, Player.class); + ops.insertAll(tournaments, Tournament.class); + ops.insertAll(scores, Score.class); + } + +} diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/PlayerRepositoryAbstract.java b/src/test/java/com/arangodb/springframework/testdata/chess/PlayerRepositoryAbstract.java new file mode 100644 index 00000000..190a03c0 --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/PlayerRepositoryAbstract.java @@ -0,0 +1,61 @@ +package com.arangodb.springframework.testdata.chess; + +import com.arangodb.springframework.testdata.chess.entity.*; +import com.arangodb.springframework.testdata.chess.repo.PlayerRepository; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.PageRequest; + +import java.util.Comparator; +import java.util.List; + +import static org.assertj.core.api.Assertions.assertThat; + +abstract class PlayerRepositoryAbstract extends AbstractRepositoryTest { + + @Autowired + @SuppressWarnings("SpringJavaInjectionPointsAutowiringInspection") + PlayerRepository repo; + + @Test + void findAllByCountry() { + List expected = players.stream() + .filter(it -> "US".equals(it.getCountry())) + .toList(); + Iterable found = repo.findAllByCountry("US"); + assertThat(found).containsExactlyInAnyOrderElementsOf(expected); + found.forEach(this::checkRefs); + } + + @Test + void findAllByRatingGreaterThan() { + int rating = 2780; + List expected = players.stream() + .filter(it -> it.getRating() > rating) + .sorted(Comparator.comparingInt(Player::getRating).reversed()) + .toList(); + + for (int i = 0; i < expected.size(); i++) { + Page page = repo.findAllByRatingGreaterThanOrderByRatingDesc(PageRequest.of(i, 1), rating); + assertThat(page.getTotalElements()).isEqualTo(expected.size()); + assertThat(page.getTotalPages()).isEqualTo(expected.size()); + Player current = page.iterator().next(); + assertThat(current).isEqualTo(expected.get(i)); + checkRefs(current); + } + } + + private void checkRefs(Player p) { + List expectedScores = scores.stream() + .filter(it -> it.player().equals(p)) + .toList(); + assertThat(p.getScores()).containsExactlyInAnyOrderElementsOf(expectedScores); + + List expectedTournaments = expectedScores.stream() + .map(Score::tournament) + .toList(); + assertThat(p.getTournaments()).containsExactlyInAnyOrderElementsOf(expectedTournaments); + } + +} diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/PlayerRepositoryTest.java b/src/test/java/com/arangodb/springframework/testdata/chess/PlayerRepositoryTest.java new file mode 100644 index 00000000..f3a48fd1 --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/PlayerRepositoryTest.java @@ -0,0 +1,4 @@ +package com.arangodb.springframework.testdata.chess; + +public class PlayerRepositoryTest extends PlayerRepositoryAbstract { +} diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/ScoreRepositoryAbstract.java b/src/test/java/com/arangodb/springframework/testdata/chess/ScoreRepositoryAbstract.java new file mode 100644 index 00000000..32387c45 --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/ScoreRepositoryAbstract.java @@ -0,0 +1,24 @@ +package com.arangodb.springframework.testdata.chess; + +import com.arangodb.springframework.testdata.chess.repo.ScoreRepository; +import org.junit.jupiter.api.Disabled; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; + +import static org.assertj.core.api.Assertions.assertThat; + +abstract class ScoreRepositoryAbstract extends AbstractRepositoryTest { + + @Autowired + @SuppressWarnings("SpringJavaInjectionPointsAutowiringInspection") + ScoreRepository repo; + + @Test + @Disabled("BTS-1859") + void findAll() { + assertThat(repo.findAll()) + .hasSize(scores.size()) + .containsExactlyInAnyOrderElementsOf(scores); + } + +} diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/ScoreRepositoryTest.java b/src/test/java/com/arangodb/springframework/testdata/chess/ScoreRepositoryTest.java new file mode 100644 index 00000000..0bc653f9 --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/ScoreRepositoryTest.java @@ -0,0 +1,4 @@ +package com.arangodb.springframework.testdata.chess; + +public class ScoreRepositoryTest extends ScoreRepositoryAbstract { +} diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/TournamentRepositoryAbstract.java b/src/test/java/com/arangodb/springframework/testdata/chess/TournamentRepositoryAbstract.java new file mode 100644 index 00000000..d7e41cc9 --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/TournamentRepositoryAbstract.java @@ -0,0 +1,122 @@ +package com.arangodb.springframework.testdata.chess; + +import com.arangodb.springframework.testdata.chess.entity.Player; +import com.arangodb.springframework.testdata.chess.entity.Score; +import com.arangodb.springframework.testdata.chess.entity.Tournament; +import com.arangodb.springframework.testdata.chess.repo.TournamentRepository; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.data.domain.PageRequest; +import org.springframework.data.geo.*; + +import java.time.LocalDate; +import java.util.List; +import java.util.Map; +import java.util.function.Function; + +import static java.util.stream.Collectors.toMap; +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.withinPercentage; + +abstract class TournamentRepositoryAbstract extends AbstractRepositoryTest { + + @Autowired + @SuppressWarnings("SpringJavaInjectionPointsAutowiringInspection") + TournamentRepository repo; + + // equirectangular approximation + static Distance calculateDistance(Point p1, Point p2) { + double lat1Rad = Math.toRadians(p1.getY()); + double lat2Rad = Math.toRadians(p2.getY()); + double lon1Rad = Math.toRadians(p1.getX()); + double lon2Rad = Math.toRadians(p2.getX()); + + double x = (lon2Rad - lon1Rad) * Math.cos((lat1Rad + lat2Rad) / 2); + double y = (lat2Rad - lat1Rad); + double distance = Math.sqrt(x * x + y * y) * 6371; + + return new Distance(distance, Metrics.KILOMETERS); + } + + @Test + void findAllByDateBetween() { + var start = LocalDate.of(2023, 1, 1); + var end = LocalDate.of(2023, 5, 1); + List expected = tournaments.stream() + .filter(it -> it.getDate().isAfter(start)) + .filter(it -> it.getDate().isBefore(end)) + .toList(); + List found = repo.findAllByDateBetween(start, end); + assertThat(found) + .hasSize(expected.size()) + .containsExactlyInAnyOrderElementsOf(expected); + found.forEach(this::checkRefs); + } + + @Test + void findByNameContainingIgnoreCase() { + String match = "2023"; + List expected = tournaments.stream() + .filter(it -> it.getName().contains(match)) + .toList(); + Iterable found = repo.findByNameContainingIgnoreCase(match); + assertThat(found) + .hasSize(expected.size()) + .containsExactlyInAnyOrderElementsOf(expected); + found.forEach(this::checkRefs); + } + + @Test + public void findAllByLocationWithin() { + Point p = new Point(4.893611, 52.372778); // Amsterdam + Distance d = new Distance(1_000, Metrics.KILOMETERS); + + List expected = tournaments.stream() + .filter(it -> calculateDistance(p, it.getLocation()).compareTo(d) < 0) + .toList(); + + List found = repo.findAllByLocationWithin(p, d); + assertThat(found) + .hasSize(expected.size()) + .containsExactlyInAnyOrderElementsOf(expected); + found.forEach(this::checkRefs); + } + + @Test + public void findAllByLocationWithinPageable() { + Point p = new Point(4.893611, 52.372778); // Amsterdam + Distance d = new Distance(1_000, Metrics.KILOMETERS); + + Map distances = tournaments.stream() + .collect(toMap(Function.identity(), t -> calculateDistance(p, t.getLocation()))); + List expected = tournaments.stream() + .filter(it -> distances.get(it).compareTo(d) < 0) + .toList(); + + for (int i = 0; i < expected.size(); i++) { + GeoPage page = repo.findAllByLocationWithin(PageRequest.of(i, 1), p, d); + assertThat(page.getTotalElements()).isEqualTo(expected.size()); + assertThat(page.getTotalPages()).isEqualTo(expected.size()); + GeoResult current = page.iterator().next(); + double expectedDistKm = distances.get(current.getContent()).in(Metrics.KILOMETERS).getValue(); + assertThat(current.getContent()).isEqualTo(expected.get(i)); + assertThat(current.getDistance().in(Metrics.KILOMETERS).getValue()).isCloseTo(expectedDistKm, withinPercentage(.01)); + // FIXME: DE-803 + // assertThat(page.getAverageDistance().in(Metrics.KILOMETERS).getValue()).isCloseTo(expectedDistKm, withinPercentage(.01)); + checkRefs(current.getContent()); + } + } + + private void checkRefs(Tournament t) { + List expectedScores = scores.stream() + .filter(it -> it.tournament().equals(t)) + .toList(); + assertThat(t.getStandings()).containsExactlyInAnyOrderElementsOf(expectedScores); + + List expectedPlayers = expectedScores.stream() + .map(Score::player) + .toList(); + assertThat(t.getPlayers()).containsExactlyInAnyOrderElementsOf(expectedPlayers); + } + +} diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/TournamentRepositoryTest.java b/src/test/java/com/arangodb/springframework/testdata/chess/TournamentRepositoryTest.java new file mode 100644 index 00000000..6842182c --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/TournamentRepositoryTest.java @@ -0,0 +1,4 @@ +package com.arangodb.springframework.testdata.chess; + +public class TournamentRepositoryTest extends TournamentRepositoryAbstract { +} diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/entity/Player.java b/src/test/java/com/arangodb/springframework/testdata/chess/entity/Player.java new file mode 100644 index 00000000..668cfb21 --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/entity/Player.java @@ -0,0 +1,58 @@ +/* + * DISCLAIMER + * + * Copyright 2018 ArangoDB GmbH, Cologne, Germany + * + * Licensed 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. + * + * Copyright holder is ArangoDB GmbH, Cologne, Germany + */ + +package com.arangodb.springframework.testdata.chess.entity; + +import com.arangodb.springframework.annotation.Document; +import com.arangodb.springframework.annotation.From; +import com.arangodb.springframework.annotation.Relations; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.RequiredArgsConstructor; +import org.springframework.data.annotation.Id; + +import java.util.List; + +@Document +@Data +@RequiredArgsConstructor +public class Player{ + @Id + private String id; + + private final String name; + + private final int rating; + + private final String country; + + @Relations( + edges = Score.class, + direction = Relations.Direction.OUTBOUND, + lazy = true + ) + @EqualsAndHashCode.Exclude + private List tournaments; + + @From(lazy = true) + @EqualsAndHashCode.Exclude + private List scores; + +} diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/entity/Score.java b/src/test/java/com/arangodb/springframework/testdata/chess/entity/Score.java new file mode 100644 index 00000000..f36548f9 --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/entity/Score.java @@ -0,0 +1,15 @@ +package com.arangodb.springframework.testdata.chess.entity; + +import com.arangodb.springframework.annotation.Edge; +import com.arangodb.springframework.annotation.From; +import com.arangodb.springframework.annotation.To; + +@Edge +public record Score( + @From + Player player, + @To + Tournament tournament, + int rank +) { +} diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/entity/Tournament.java b/src/test/java/com/arangodb/springframework/testdata/chess/entity/Tournament.java new file mode 100644 index 00000000..624ef1ec --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/entity/Tournament.java @@ -0,0 +1,40 @@ +package com.arangodb.springframework.testdata.chess.entity; + +import com.arangodb.springframework.annotation.*; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.RequiredArgsConstructor; +import org.springframework.data.annotation.Id; +import org.springframework.data.geo.Point; + +import java.time.LocalDate; +import java.util.List; + +@Document +@Data +@RequiredArgsConstructor +public class Tournament { + private @Id String id; + + private final String name; + + private final LocalDate date; + + private final String city; + + @GeoIndexed(geoJson = true) + private final Point location; + + @To(lazy = true) + @EqualsAndHashCode.Exclude + private List standings; + + @Relations( + edges = Score.class, + direction = Relations.Direction.INBOUND, + lazy = true + ) + @EqualsAndHashCode.Exclude + private List players; +} + diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/repo/PlayerRepository.java b/src/test/java/com/arangodb/springframework/testdata/chess/repo/PlayerRepository.java new file mode 100644 index 00000000..3efec4e6 --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/repo/PlayerRepository.java @@ -0,0 +1,15 @@ +package com.arangodb.springframework.testdata.chess.repo; + +import com.arangodb.springframework.repository.ArangoRepository; +import com.arangodb.springframework.testdata.chess.entity.Player; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.Pageable; + +public interface PlayerRepository extends ArangoRepository { + Iterable findAllByCountry(String country); + + Page findAllByRatingGreaterThanOrderByRatingDesc(Pageable pageable, int rating); +} + + + diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/repo/ScoreRepository.java b/src/test/java/com/arangodb/springframework/testdata/chess/repo/ScoreRepository.java new file mode 100644 index 00000000..2f0c8ffc --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/repo/ScoreRepository.java @@ -0,0 +1,7 @@ +package com.arangodb.springframework.testdata.chess.repo; + +import com.arangodb.springframework.repository.ArangoRepository; +import com.arangodb.springframework.testdata.chess.entity.Score; + +public interface ScoreRepository extends ArangoRepository { +} diff --git a/src/test/java/com/arangodb/springframework/testdata/chess/repo/TournamentRepository.java b/src/test/java/com/arangodb/springframework/testdata/chess/repo/TournamentRepository.java new file mode 100644 index 00000000..125c1187 --- /dev/null +++ b/src/test/java/com/arangodb/springframework/testdata/chess/repo/TournamentRepository.java @@ -0,0 +1,24 @@ +package com.arangodb.springframework.testdata.chess.repo; + +import com.arangodb.springframework.repository.ArangoRepository; +import com.arangodb.springframework.testdata.chess.entity.Tournament; +import org.springframework.data.domain.Pageable; +import org.springframework.data.geo.Distance; +import org.springframework.data.geo.GeoPage; +import org.springframework.data.geo.Point; + +import java.time.LocalDate; +import java.util.List; + +public interface TournamentRepository extends ArangoRepository { + List findAllByDateBetween(LocalDate start, LocalDate end); + + Iterable findByNameContainingIgnoreCase(String match); + + List findAllByLocationWithin(Point location, Distance distance); + + GeoPage findAllByLocationWithin(Pageable pageable, Point location, Distance distance); +} + + +