Skip to content

Commit c584e2a

Browse files
tim-aeroreugn
andauthored
Support interface superclass (#140)
* Allowed the top of the hierarchy to be an interface. If classes implement an interface instead of extending an abstract class, the Java Object Mapper should respect any AerospikeRecords annotations on the interface. The Object Mapper will look for definitions in the config file, then any AerospikeRecords on the specific class, then the superclasses and finally anything in the interface hierarchy. The first one which defines a namespace will be the definition which is used. * Changed the interface scanning to not store information which is not needed. * Added convenience methods for getting namespace/set/key/RecordKey * Apply suggestions from code review Co-authored-by: Eugene R. <[email protected]> * Update InterfaceHierarchyTest.java Utilised "addAll" to add vararg children to parent, and removed redundant saving of nested children * Modified reactive interface * Implemented reactive methods * Fixed the reactive test * More reactive changes --------- Co-authored-by: Eugene R. <[email protected]>
1 parent 9f3487f commit c584e2a

9 files changed

+420
-19
lines changed

README.md

+9
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ The documentation for this project can be found on [javadoc.io](https://www.java
3030
+ 9.2. [Subclasses](#Subclasses)
3131
+ 9.2.1. [Data Inheritance](#Data-Inheritance)
3232
+ 9.2.2. [Subclass Inheritance](#Subclass-Inheritance)
33+
+ 9.2.3 [Using Interfaces](#Using-Interfaces)
3334
+ 9.3. [Custom Object Converters](#Custom-Object-Converters)
3435
10. [External Configuration File](#External-Configuration-File)
3536
+ 10.1. [File Structure](#File-Structure)
@@ -1704,6 +1705,14 @@ public List customers;
17041705

17051706
The former is considered better style in Java and also provides the Java Object Mapper with information about the elements in the list, so it will optimize its workings to know how to store a list of Customers. The latter gives it no type information so it must derive the type -- and hence how to map it to Aerospike -- for every element in this list. This can have a noticeable performance impact for large lists, as well as consuming more database space (as it must store the runtime type of each element in the list in addition to the data).
17061707

1708+
### Using Interfaces
1709+
Sometimes it is better to have an interface to group common types rather an an abstract superclass. In this case the Object Mapper supports placing the `@AerospikeReocrd` annotation on the interface and it will behave as if the annotation was on a superclass. There are multiple different was of placing the `@AerospikeRecord` annotation on a single class, and the order the Object Mapper looks for them in is:
1710+
1. Configuration file
1711+
2. Class level definition
1712+
3. First parent class with `@AerospikeRecord` annotation (of any ancestor)
1713+
4. Interface with `@AerospikeRecord` annotation (first one found)
1714+
1715+
Once the Object Mapper finds an appropriate annotation it ignores any further annotations and uses the definitions on the first one found.
17071716

17081717
----
17091718

src/main/java/com/aerospike/mapper/tools/AeroMapper.java

+24
Original file line numberDiff line numberDiff line change
@@ -726,4 +726,28 @@ private Policy getPolicyByClassAndType(Class<?> clazz, PolicyType policyType) {
726726
throw new UnsupportedOperationException("Provided unsupported policy.");
727727
}
728728
}
729+
730+
@Override
731+
public String getNamespace(Class<?> clazz) {
732+
ClassCacheEntry<?> entry = ClassCache.getInstance().loadClass(clazz, this);
733+
return entry == null ? null : entry.getNamespace();
734+
}
735+
736+
@Override
737+
public String getSet(Class<?> clazz) {
738+
ClassCacheEntry<?> entry = ClassCache.getInstance().loadClass(clazz, this);
739+
return entry == null ? null : entry.getSetName();
740+
}
741+
742+
@Override
743+
public Object getKey(Object obj) {
744+
ClassCacheEntry<?> entry = ClassCache.getInstance().loadClass(obj.getClass(), this);
745+
return entry == null ? null : entry.getKey(obj);
746+
}
747+
748+
@Override
749+
public Key getRecordKey(Object obj) {
750+
ClassCacheEntry<?> entry = ClassCache.getInstance().loadClass(obj.getClass(), this);
751+
return entry == null ? null : new Key(entry.getNamespace(), entry.getSetName(), Value.get(entry.getKey(obj)));
752+
}
729753
}

src/main/java/com/aerospike/mapper/tools/ClassCache.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,8 @@ public <T> ClassCacheEntry<T> loadClass(@NotNull Class<T> clazz, IBaseAeroMapper
4545

4646
@SuppressWarnings("unchecked")
4747
public <T> ClassCacheEntry<T> loadClass(@NotNull Class<T> clazz, IBaseAeroMapper mapper, boolean requireRecord) {
48-
if (clazz.isPrimitive() || clazz.equals(Object.class) || clazz.equals(String.class)
48+
// Clazz can be null if an interface is passed
49+
if (clazz == null || clazz.isPrimitive() || clazz.equals(Object.class) || clazz.equals(String.class)
4950
|| clazz.equals(Character.class) || Number.class.isAssignableFrom(clazz)) {
5051
return null;
5152
}

src/main/java/com/aerospike/mapper/tools/ClassCacheEntry.java

+40-14
Original file line numberDiff line numberDiff line change
@@ -122,16 +122,7 @@ private enum FactoryMethodType {
122122
throw new NotAnnotatedClass(String.format("Class %s is not augmented by the @AerospikeRecord annotation",
123123
clazz.getName()));
124124
} else if (recordDescription != null) {
125-
this.namespace = ParserUtils.getInstance().get(recordDescription.namespace());
126-
this.setName = ParserUtils.getInstance().get(recordDescription.set());
127-
this.ttl = recordDescription.ttl();
128-
this.mapAll = recordDescription.mapAll();
129-
this.version = recordDescription.version();
130-
this.sendKey = recordDescription.sendKey();
131-
this.durableDelete = recordDescription.durableDelete();
132-
this.shortenedClassName = recordDescription.shortName();
133-
this.factoryClass = recordDescription.factoryClass();
134-
this.factoryMethod = recordDescription.factoryMethod();
125+
this.setPropertiesFromAerospikeRecord(recordDescription);
135126
}
136127
this.config = config;
137128
}
@@ -142,16 +133,24 @@ public ClassCacheEntry<T> construct() {
142133
this.overrideSettings(config);
143134
}
144135

136+
if (this.namespace == null || this.namespace.isEmpty()) {
137+
List<AerospikeRecord> aerospikeInterfaceRecords = this.loadAerospikeRecordsFromInterfaces(this.clazz);
138+
for (int i = 0; (this.namespace == null || this.namespace.isEmpty()) && i < aerospikeInterfaceRecords.size(); i++) {
139+
this.setPropertiesFromAerospikeRecord(aerospikeInterfaceRecords.get(i));
140+
}
141+
}
145142
this.loadFieldsFromClass();
146143
this.loadPropertiesFromClass();
147144
this.superClazz = ClassCache.getInstance().loadClass(this.clazz.getSuperclass(), this.mapper, !this.mapAll);
148145
this.binCount = this.values.size() + (superClazz != null ? superClazz.binCount : 0);
149146
this.formOrdinalsFromValues();
150147
Method factoryConstructorMethod = findConstructorFactoryMethod();
151-
if (factoryConstructorMethod == null) {
152-
this.findConstructor();
153-
} else {
154-
this.setConstructorFactoryMethod(factoryConstructorMethod);
148+
if (!this.clazz.isInterface()) {
149+
if (factoryConstructorMethod == null) {
150+
this.findConstructor();
151+
} else {
152+
this.setConstructorFactoryMethod(factoryConstructorMethod);
153+
}
155154
}
156155
if (StringUtils.isBlank(this.shortenedClassName)) {
157156
this.shortenedClassName = clazz.getSimpleName();
@@ -236,6 +235,33 @@ public boolean isChildClass() {
236235
return isChildClass;
237236
}
238237

238+
private List<AerospikeRecord> loadAerospikeRecordsFromInterfaces(Class<?> clazz) {
239+
List<AerospikeRecord> results = new ArrayList<>();
240+
Class<?>[] interfaces = clazz.getInterfaces();
241+
for (int i = 0; i < interfaces.length; i++) {
242+
Class<?> thisInterface = interfaces[i];
243+
AerospikeRecord[] aerospikeRecords = thisInterface.getAnnotationsByType(AerospikeRecord.class);
244+
for (int j = 0; j < aerospikeRecords.length; j++) {
245+
results.add(aerospikeRecords[j]);
246+
}
247+
results.addAll(loadAerospikeRecordsFromInterfaces(thisInterface));
248+
}
249+
return results;
250+
}
251+
252+
private void setPropertiesFromAerospikeRecord(AerospikeRecord recordDescription) {
253+
this.namespace = ParserUtils.getInstance().get(recordDescription.namespace());
254+
this.setName = ParserUtils.getInstance().get(recordDescription.set());
255+
this.ttl = recordDescription.ttl();
256+
this.mapAll = recordDescription.mapAll();
257+
this.version = recordDescription.version();
258+
this.sendKey = recordDescription.sendKey();
259+
this.durableDelete = recordDescription.durableDelete();
260+
this.shortenedClassName = recordDescription.shortName();
261+
this.factoryClass = recordDescription.factoryClass();
262+
this.factoryMethod = recordDescription.factoryMethod();
263+
}
264+
239265
private void checkRecordSettingsAgainstSuperClasses() {
240266
if (!StringUtils.isBlank(this.namespace) && !StringUtils.isBlank(this.setName)) {
241267
// This class defines its own namespace + set, it is only a child class if its closest named superclass is the same as ours.

src/main/java/com/aerospike/mapper/tools/IAeroMapper.java

+31
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77

88
import com.aerospike.client.AerospikeException;
99
import com.aerospike.client.IAerospikeClient;
10+
import com.aerospike.client.Key;
1011
import com.aerospike.client.Operation;
1112
import com.aerospike.client.policy.BatchPolicy;
1213
import com.aerospike.client.policy.Policy;
@@ -392,5 +393,35 @@ public interface IAeroMapper extends IBaseAeroMapper {
392393
*/
393394
<T> VirtualList<T> asBackedList(@NotNull Class<?> owningClazz, @NotNull Object key, @NotNull String binName, Class<T> elementClazz);
394395

396+
/**
397+
* Get the IAerospikeClient which was used to create this mapper.
398+
* @return the underlying mapper.
399+
*/
395400
IAerospikeClient getClient();
401+
402+
/**
403+
* Get the namespace associated with the passed class
404+
* @param clazz - the class to retrieve the namespace of
405+
* @return the namespace
406+
*/
407+
String getNamespace(Class<?> clazz);
408+
409+
/**
410+
* Get the set associated with the passed class
411+
* @param clazz - the class to retrieve the set of
412+
* @return the set
413+
*/
414+
String getSet(Class<?> clazz);
415+
416+
/**
417+
* Get the primary id associated with the passed object
418+
*/
419+
Object getKey(Object obj);
420+
421+
/**
422+
* Get the Aerospike Key used to read the record associated with the passed key
423+
* @param obj - the object to return the key of
424+
* @return the key which could be used to retrieve this record.
425+
*/
426+
Key getRecordKey(Object obj);
396427
}

src/main/java/com/aerospike/mapper/tools/IReactiveAeroMapper.java

+37-4
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,24 @@
11
package com.aerospike.mapper.tools;
22

3+
import java.util.function.Function;
4+
5+
import javax.validation.constraints.NotNull;
6+
37
import com.aerospike.client.AerospikeException;
8+
import com.aerospike.client.Key;
49
import com.aerospike.client.Operation;
5-
import com.aerospike.client.policy.*;
10+
import com.aerospike.client.policy.BatchPolicy;
11+
import com.aerospike.client.policy.Policy;
12+
import com.aerospike.client.policy.QueryPolicy;
13+
import com.aerospike.client.policy.ScanPolicy;
14+
import com.aerospike.client.policy.WritePolicy;
615
import com.aerospike.client.query.Filter;
716
import com.aerospike.client.reactor.IAerospikeReactorClient;
817
import com.aerospike.mapper.tools.virtuallist.ReactiveVirtualList;
18+
919
import reactor.core.publisher.Flux;
1020
import reactor.core.publisher.Mono;
1121

12-
import javax.validation.constraints.NotNull;
13-
import java.util.function.Function;
14-
1522
public interface IReactiveAeroMapper extends IBaseAeroMapper {
1623

1724
/**
@@ -321,4 +328,30 @@ public interface IReactiveAeroMapper extends IBaseAeroMapper {
321328
<T> ReactiveVirtualList<T> asBackedList(@NotNull Class<?> owningClazz, @NotNull Object key, @NotNull String binName, Class<T> elementClazz);
322329

323330
IAerospikeReactorClient getReactorClient();
331+
/**
332+
* Get the namespace associated with the passed class
333+
* @param clazz - the class to retrieve the namespace of
334+
* @return the namespace
335+
*/
336+
<T> Mono<String> getNamespace(Class<T> clazz);
337+
338+
/**
339+
* Get the set associated with the passed class
340+
* @param clazz - the class to retrieve the set of
341+
* @return the set
342+
*/
343+
<T> Mono<String> getSet(Class<T> clazz);
344+
345+
/**
346+
* Get the primary id associated with the passed object
347+
*/
348+
Mono<Object> getKey(Object obj);
349+
350+
/**
351+
* Get the Aerospike Key used to read the record associated with the passed key
352+
*
353+
* @param obj - the object to return the key of
354+
* @return the key which could be used to retrieve this record.
355+
*/
356+
Mono<Key> getRecordKey(Object obj);
324357
}

src/main/java/com/aerospike/mapper/tools/ReactiveAeroMapper.java

+24
Original file line numberDiff line numberDiff line change
@@ -556,4 +556,28 @@ private Throwable translateError(Throwable e) {
556556
}
557557
return e;
558558
}
559+
560+
@Override
561+
public <T> Mono<String> getNamespace(Class<T> clazz) {
562+
ClassCacheEntry<T> entry = MapperUtils.getEntryAndValidateNamespace(clazz, this);
563+
return entry == null ? null : Mono.just(entry.getNamespace());
564+
}
565+
566+
@Override
567+
public <T> Mono<String> getSet(Class<T> clazz) {
568+
ClassCacheEntry<?> entry = ClassCache.getInstance().loadClass(clazz, this);
569+
return entry == null ? null : Mono.just(entry.getSetName());
570+
}
571+
572+
@Override
573+
public Mono<Object> getKey(Object obj) {
574+
ClassCacheEntry<?> entry = ClassCache.getInstance().loadClass(obj.getClass(), this);
575+
return entry == null ? null : Mono.just(entry.getKey(obj));
576+
}
577+
578+
@Override
579+
public Mono<Key> getRecordKey(Object obj) {
580+
ClassCacheEntry<?> entry = ClassCache.getInstance().loadClass(obj.getClass(), this);
581+
return entry == null ? null : Mono.just(new Key(entry.getNamespace(), entry.getSetName(), Value.get(entry.getKey(obj))));
582+
}
559583
}

0 commit comments

Comments
 (0)