com.datastax.driver.mapping.DefaultPropertyMapper Maven / Gradle / Ivy
Show all versions of cassandra-driver-mapping Show documentation
/*
* Copyright (C) 2012-2017 DataStax Inc.
*
* 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.
*/
package com.datastax.driver.mapping;
import com.datastax.driver.core.Metadata;
import com.datastax.driver.mapping.annotations.*;
import com.google.common.base.Throwables;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Sets;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.annotation.Annotation;
import java.lang.reflect.AccessibleObject;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.*;
import static com.google.common.base.Preconditions.checkNotNull;
/**
* The default {@link PropertyMapper} used by the mapper.
*
* This mapper can be configured to scan for fields, getters and setters, or both.
* The default is to scan for both.
*
* This mapper can also be configured to skip transient properties.
* By default, all properties will be mapped (non-transient),
* unless explicitly marked with {@link Transient @Transient}.
*
* This mapper recognizes standard getter and setter methods
* (as defined by the Java Beans specification),
* and also "relaxed" setter methods, i.e., setter methods
* whose return type are not {@code void}.
*
* @see DefaultMappedProperty
*/
public class DefaultPropertyMapper implements PropertyMapper {
private static final Logger LOGGER = LoggerFactory.getLogger(DefaultPropertyMapper.class);
private static final HashSet DEFAULT_TRANSIENT_PROPERTY_NAMES = Sets.newHashSet(
"class",
// JAVA-1279: exclude Groovy's metaClass property
"metaClass"
);
private static final Set> NON_TRANSIENT_ANNOTATIONS = ImmutableSet.>of(
Column.class,
PartitionKey.class,
ClusteringColumn.class,
com.datastax.driver.mapping.annotations.Field.class,
Computed.class,
Frozen.class,
FrozenKey.class,
FrozenValue.class
);
/**
* Annotations allowed on a property that maps to a table column.
*/
private static final Set> VALID_COLUMN_ANNOTATIONS = ImmutableSet.>builder()
.add(Column.class)
.add(Computed.class)
.add(ClusteringColumn.class)
.add(Frozen.class)
.add(FrozenKey.class)
.add(FrozenValue.class)
.add(PartitionKey.class)
.add(Transient.class)
.build();
/**
* Annotations allowed on a property that maps to a UDT field.
*/
private static final Set> VALID_FIELD_ANNOTATIONS = ImmutableSet.of(
com.datastax.driver.mapping.annotations.Field.class,
Frozen.class,
FrozenKey.class,
FrozenValue.class,
Transient.class);
private PropertyAccessStrategy propertyAccessStrategy = PropertyAccessStrategy.BOTH;
private PropertyTransienceStrategy propertyTransienceStrategy = PropertyTransienceStrategy.OPT_OUT;
private HierarchyScanStrategy hierarchyScanStrategy = new DefaultHierarchyScanStrategy();
private NamingStrategy namingStrategy = new DefaultNamingStrategy();
private Set transientPropertyNames = new HashSet(DEFAULT_TRANSIENT_PROPERTY_NAMES);
/**
* Sets the {@link PropertyAccessStrategy property access strategy} to use.
* The default is {@link PropertyAccessStrategy#BOTH}.
*
* @param propertyAccessStrategy the {@link PropertyAccessStrategy property access strategy} to use; may not be {@code null}.
* @return this {@link DefaultPropertyMapper} instance (to allow for fluent builder pattern).
*/
public DefaultPropertyMapper setPropertyAccessStrategy(PropertyAccessStrategy propertyAccessStrategy) {
this.propertyAccessStrategy = checkNotNull(propertyAccessStrategy);
return this;
}
/**
* Sets the {@link PropertyTransienceStrategy property transience strategy} to use.
* The default is {@link PropertyTransienceStrategy#OPT_OUT}.
*
* @param propertyTransienceStrategy the {@link PropertyTransienceStrategy property transience strategy} to use; may not be {@code null}.
* @return this {@link DefaultPropertyMapper} instance (to allow for fluent builder pattern).
*/
public DefaultPropertyMapper setPropertyTransienceStrategy(PropertyTransienceStrategy propertyTransienceStrategy) {
this.propertyTransienceStrategy = checkNotNull(propertyTransienceStrategy);
return this;
}
/**
* Sets the {@link HierarchyScanStrategy hierarchy scan strategy} to use.
* The default is {@link DefaultHierarchyScanStrategy}.
*
* @param hierarchyScanStrategy the {@link HierarchyScanStrategy hierarchy scan strategy} to use; may not be {@code null}.
* @return this {@link DefaultPropertyMapper} instance (to allow for fluent builder pattern).
*/
public DefaultPropertyMapper setHierarchyScanStrategy(HierarchyScanStrategy hierarchyScanStrategy) {
this.hierarchyScanStrategy = checkNotNull(hierarchyScanStrategy);
return this;
}
/**
* Sets the {@link NamingStrategy naming strategy} to use.
* The default is {@link DefaultNamingStrategy}.
*
* @param namingStrategy the {@link NamingStrategy naming strategy} to use; may not be {@code null}.
* @return this {@link DefaultPropertyMapper} instance (to allow for fluent builder pattern).
*/
public DefaultPropertyMapper setNamingStrategy(NamingStrategy namingStrategy) {
this.namingStrategy = checkNotNull(namingStrategy);
return this;
}
/**
* Sets transient property names. This
* will completely replace any names already configured for this object.
*
* The default set comprises the following property names:
* {@code class} and {@code metaClass}.
* These properties pertain to the {@link Object} class –
* {@code metaClass} being specific to the Groovy language.
*
* Property names provided here will always be considered transient;
* if a more fine-grained tuning is required, it is also possible
* to use the {@link Transient @Transient} annotation
* on a specific property.
*
* Subclasses can also override {@link #isTransient(String, Field, Method, Method, Map)} to gain
* complete control over which properties should be considered transient.
*
* @param transientPropertyNames a set of property names to exclude from mapping; may not be {@code null}. This
* will completely replace any names already configured for this object.
*/
public DefaultPropertyMapper setTransientPropertyNames(Set transientPropertyNames) {
this.transientPropertyNames = checkNotNull(transientPropertyNames);
return this;
}
/**
* Adds new values to the existing set of transient property names.
*
* The default set comprises the following property names:
* {@code class} and {@code metaClass}.
* These properties pertain to the {@link Object} class –
* {@code metaClass} being specific to the Groovy language.
*
* Property names provided here will always be considered transient;
* if a more fine-grained tuning is required, it is also possible
* to use the {@link Transient @Transient} annotation
* on a specific property.
*
* Subclasses can also override {@link #isTransient(String, Field, Method, Method, Map)} to gain
* complete control over which properties should be considered transient.
*
* @param transientPropertyNames the values to add; may not be {@code null}.
*/
public DefaultPropertyMapper addTransientPropertyNames(String... transientPropertyNames) {
return addTransientPropertyNames(Arrays.asList(checkNotNull(transientPropertyNames)));
}
/**
* Adds new values to the existing set of transient property names.
*
* The default set comprises the following property names:
* {@code class} and {@code metaClass}.
* These properties pertain to the {@link Object} class –
* {@code metaClass} being specific to the Groovy language.
*
* Property names provided here will always be considered transient;
* if a more fine-grained tuning is required, it is also possible
* to use the {@link Transient @Transient} annotation
* on a specific property.
*
* Subclasses can also override {@link #isTransient(String, Field, Method, Method, Map)} to gain
* complete control over which properties should be considered transient.
*
* @param transientPropertyNames the values to add; may not be {@code null}.
*/
public DefaultPropertyMapper addTransientPropertyNames(Collection transientPropertyNames) {
this.transientPropertyNames.addAll(checkNotNull(transientPropertyNames));
return this;
}
@Override
public Set extends MappedProperty>> mapTable(Class> tableClass) {
return mapTableOrUdt(tableClass, VALID_COLUMN_ANNOTATIONS);
}
@Override
public Set extends MappedProperty>> mapUdt(Class> udtClass) {
return mapTableOrUdt(udtClass, VALID_FIELD_ANNOTATIONS);
}
private Set extends MappedProperty>> mapTableOrUdt(Class> entityClass, Collection extends Class extends Annotation>> allowed) {
Map fieldsGettersAndSetters = new HashMap();
List> classHierarchy = hierarchyScanStrategy.filterClassHierarchy(entityClass);
if (propertyAccessStrategy.isFieldScanAllowed()) {
Map fields = scanFields(classHierarchy);
for (Map.Entry entry : fields.entrySet()) {
String propertyName = entry.getKey();
Field field = tryMakeAccessible(entry.getValue());
fieldsGettersAndSetters.put(propertyName, new Object[]{field, null, null});
}
}
if (propertyAccessStrategy.isGetterSetterScanAllowed()) {
Map properties = scanProperties(classHierarchy);
for (Map.Entry entry : properties.entrySet()) {
PropertyDescriptor property = entry.getValue();
Method getter = tryMakeAccessible(locateGetter(entityClass, property));
Method setter = tryMakeAccessible(locateSetter(entityClass, property));
Object[] value = fieldsGettersAndSetters.get(entry.getKey());
if (value != null) {
value[1] = getter;
value[2] = setter;
} else if (getter != null || setter != null) {
fieldsGettersAndSetters.put(entry.getKey(), new Object[]{null, getter, setter});
}
}
}
Set> mappedProperties = new HashSet>(fieldsGettersAndSetters.size());
for (Map.Entry entry : fieldsGettersAndSetters.entrySet()) {
String propertyName = entry.getKey();
Field field = (Field) entry.getValue()[0];
Method getter = (Method) entry.getValue()[1];
Method setter = (Method) entry.getValue()[2];
Map, Annotation> annotations = scanPropertyAnnotations(field, getter);
AnnotationChecks.validateAnnotations(propertyName, annotations, allowed);
if (isTransient(propertyName, field, getter, setter, annotations)) {
LOGGER.debug(String.format("Property '%s' is transient and will not be mapped", propertyName));
continue;
}
if (!annotations.containsKey(Computed.class) && field == null && getter == null) {
throw new IllegalArgumentException(String.format("Property '%s' is not readable", propertyName));
}
if (field == null && setter == null) {
throw new IllegalArgumentException(String.format("Property '%s' is not writable", propertyName));
}
String mappedName = inferMappedName(entityClass, propertyName, annotations);
MappedProperty> property = createMappedProperty(entityClass, propertyName, mappedName, field, getter, setter, annotations);
mappedProperties.add(property);
}
return mappedProperties;
}
/**
* Returns {@code true} if the given property is transient,
* {@code false} otherwise.
*
* If this method returns {@code true} the given property will not be mapped.
* The implementation provided here relies on the
* {@link #setPropertyTransienceStrategy(PropertyTransienceStrategy) transience strategy}
* and the {@link #setTransientPropertyNames(Set) transient property names}
* configured on this mapper.
*
* Subclasses may override this method to take full control of which properties
* should be mapped and which should be considered transient.
*
* @param propertyName the property name; may not be {@code null}.
* @param field the property field; may be {@code null}.
* @param getter the getter method for this property; may be {@code null}.
* @param setter the setter method for this property; may be {@code null}.
* @param annotations the annotations found on this property; may be empty but never {@code null}.
* @return {@code true} if the given property is transient (i.e., non-mapped), {@code false} otherwise.
*/
@SuppressWarnings("unused")
protected boolean isTransient(String propertyName, Field field, Method getter, Method setter, Map, Annotation> annotations) {
if (propertyTransienceStrategy == PropertyTransienceStrategy.OPT_OUT)
return annotations.containsKey(Transient.class)
|| (transientPropertyNames.contains(propertyName)
&& Collections.disjoint(annotations.keySet(), NON_TRANSIENT_ANNOTATIONS));
else
return Collections.disjoint(annotations.keySet(), NON_TRANSIENT_ANNOTATIONS);
}
/**
* Locates a getter method for the given mapped class and given property.
*
* Most users should rely on the implementation provided here.
* It is however possible to return any non-standard method, as long as it does
* not take parameters, and its return type is assignable to (and covariant with) the property's type.
* This might be particularly useful for boolean properties whose names are verbs, e.g. "{@code hasAccount}":
* one could then return the non-standard method {@code boolean hasAccount()} as its getter.
*
* This method is never called if {@link PropertyAccessStrategy#isGetterSetterScanAllowed()} returns {@code false}.
* Besides, implementors are free to return {@code null} if access to the property through reflection is not required
* (in which case, they will likely have to provide a custom implementation of {@link MappedProperty}).
*
* @param mappedClass The mapped class; this is necessarily a class annotated with
* either {@link Table @Table} or
* {@link UDT @UDT}.
* @param property The property to locate a getter for; never {@code null}.
* @return The getter method for the given base class and given property, or {@code null} if no getter was found, or reflection is not required.
*/
protected Method locateGetter(@SuppressWarnings("unused") Class> mappedClass, PropertyDescriptor property) {
return property.getReadMethod();
}
/**
* Locates a setter method for the given mapped class and given property.
*
* Most users should rely on the implementation provided here.
* It is however possible to return any non-standard method, as long as it accepts one single parameter type
* that is contravariant with the property's type.
*
* This method is never called if {@link PropertyAccessStrategy#isGetterSetterScanAllowed()} returns {@code false}.
* Besides, implementors are free to return {@code null} if access to the property through reflection is not required
* (in which case, they will likely have to provide a custom implementation of {@link MappedProperty}).
*
* @param mappedClass The mapped class; this is necessarily a class annotated with
* either {@link Table @Table} or
* {@link UDT @UDT}.
* @param property The property to locate a setter for; never {@code null}.
* @return The setter method for the given base class and given property, or {@code null} if no setter was found, or reflection is not required.
*/
protected Method locateSetter(Class> mappedClass, PropertyDescriptor property) {
Method setter = property.getWriteMethod();
if (setter == null) {
// JAVA-984: look for a "relaxed" setter, ie. a setter whose return type may be anything
String propertyName = property.getName();
String setterName = "set" + propertyName.substring(0, 1).toUpperCase() + propertyName.substring(1);
try {
Method m = mappedClass.getMethod(setterName, property.getPropertyType());
if (!Modifier.isStatic(m.getModifiers()))
setter = m;
} catch (NoSuchMethodException ignored) {
}
}
return setter;
}
/**
* Infers the Cassandra object name corresponding to given the property name.
*
* Most users should rely on the implementation provided here.
* It relies on annotation values and ultimately resorts to the
* {@link NamingStrategy} configured on this mapper.
*
* Subclasses may override this method if they need full control
* over generating Cassandra object names.
*
* @param mappedClass The mapped class; this is necessarily a class annotated with
* either {@link Table @Table} or
* {@link UDT @UDT}.
* @param propertyName The property name; may not be {@code null} nor empty.
* @param annotations The property annotations (found on its field and getter method); may not be {@code null} but can be empty.
* @return The inferred Cassandra object name.
*/
protected String inferMappedName(@SuppressWarnings("unused") Class> mappedClass, String propertyName, Map, Annotation> annotations) {
if (annotations.containsKey(Computed.class)) {
String expression = ((Computed) annotations.get(Computed.class)).value();
if (expression.isEmpty())
throw new IllegalArgumentException(String.format("Property '%s': attribute 'value' of annotation @Computed is mandatory for computed properties", propertyName));
return expression;
}
// If a name is explicitly provided with @Column or @Field, use it
boolean caseSensitive = false;
String mappedName = null;
if (annotations.containsKey(Column.class)) {
Column column = (Column) annotations.get(Column.class);
caseSensitive = column.caseSensitive();
if (!column.name().isEmpty())
mappedName = column.name();
} else if (annotations.containsKey(com.datastax.driver.mapping.annotations.Field.class)) {
com.datastax.driver.mapping.annotations.Field udtMappedField =
(com.datastax.driver.mapping.annotations.Field) annotations.get(com.datastax.driver.mapping.annotations.Field.class);
caseSensitive = udtMappedField.caseSensitive();
if (!udtMappedField.name().isEmpty())
mappedName = udtMappedField.name();
}
if (mappedName != null) {
return caseSensitive ? Metadata.quote(mappedName) : mappedName.toLowerCase();
}
// Otherwise delegate to the naming strategy
mappedName = namingStrategy.toCassandraName(propertyName);
if (mappedName == null || mappedName.isEmpty())
throw new IllegalArgumentException(String.format("Property '%s': could not infer mapped name", propertyName));
return Metadata.quoteIfNecessary(mappedName);
}
/**
* Creates a {@link MappedProperty} instance.
*
* Instances returned by the implementation below will use the Java reflection API to read and write values.
* Subclasses may override this method if they are capable of accessing
* properties without incurring the cost of reflection.
*
* @param mappedClass The mapped class; this is necessarily a class annotated with
* either {@link Table @Table} or
* {@link UDT @UDT}.
* @param propertyName The property name; may not be {@code null} nor empty.
* @param mappedName The mapped name; may not be {@code null} nor empty.
* @param field The property field; may be {@code null}.
* @param getter The property getter method; may be {@code null}.
* @param setter The property setter method; may be {@code null}.
* @param annotations The property annotations (found on its field and getter method); may not be {@code null} but can be empty.
* @return a newly-allocated {@link MappedProperty} instance.
*/
protected MappedProperty> createMappedProperty(Class> mappedClass, String propertyName, String mappedName, Field field, Method getter, Method setter, Map, Annotation> annotations) {
return DefaultMappedProperty.create(mappedClass, propertyName, mappedName, field, getter, setter, annotations);
}
private static Map scanFields(List> classHierarchy) {
HashMap fields = new HashMap();
for (Class> clazz : classHierarchy) {
for (Field field : clazz.getDeclaredFields()) {
if (field.isSynthetic() || Modifier.isStatic(field.getModifiers()) || Modifier.isTransient(field.getModifiers()))
continue;
// never override a more specific field masking another one declared in a superclass
if (!fields.containsKey(field.getName()))
fields.put(field.getName(), field);
}
}
return fields;
}
private static Map scanProperties(List> classHierarchy) {
Map properties = new HashMap();
for (Class> clazz : classHierarchy) {
// each time extract only current class properties
BeanInfo beanInfo;
try {
beanInfo = Introspector.getBeanInfo(clazz, clazz.getSuperclass());
} catch (IntrospectionException e) {
throw Throwables.propagate(e);
}
for (PropertyDescriptor property : beanInfo.getPropertyDescriptors()) {
if (!properties.containsKey(property.getName())) {
properties.put(property.getName(), property);
}
}
}
return properties;
}
private static Map, Annotation> scanPropertyAnnotations(Field field, Method getter) {
Map, Annotation> annotations = new HashMap, Annotation>();
// annotations on getters should have precedence over annotations on fields
if (field != null)
scanFieldAnnotations(field, annotations);
if (getter != null)
scanMethodAnnotations(getter, annotations);
return annotations;
}
private static Map, Annotation> scanFieldAnnotations(Field field, Map, Annotation> annotations) {
for (Annotation annotation : field.getAnnotations()) {
annotations.put(annotation.annotationType(), annotation);
}
return annotations;
}
private static Map, Annotation> scanMethodAnnotations(Method method, Map, Annotation> annotations) {
// 1. direct method annotations
for (Annotation annotation : method.getAnnotations()) {
annotations.put(annotation.annotationType(), annotation);
}
// 2. Class hierarchy: check for annotations in overridden methods in superclasses
Class> getterClass = method.getDeclaringClass();
for (Class> clazz = getterClass.getSuperclass(); clazz != null && !clazz.equals(Object.class); clazz = clazz.getSuperclass()) {
maybeAddOverriddenMethodAnnotations(annotations, method, clazz);
}
// 3. Interfaces: check for annotations in implemented interfaces
for (Class> clazz = getterClass; !clazz.equals(Object.class); clazz = clazz.getSuperclass()) {
for (Class> itf : clazz.getInterfaces()) {
maybeAddOverriddenMethodAnnotations(annotations, method, itf);
}
}
return annotations;
}
private static void maybeAddOverriddenMethodAnnotations(Map, Annotation> annotations, Method getter, Class> clazz) {
try {
Method overriddenGetter = clazz.getDeclaredMethod(getter.getName(), (Class>[]) getter.getParameterTypes());
for (Annotation annotation : overriddenGetter.getAnnotations()) {
// do not override a more specific version of the annotation type being scanned
if (!annotations.containsKey(annotation.annotationType()))
annotations.put(annotation.annotationType(), annotation);
}
} catch (NoSuchMethodException e) {
//ok
}
}
private static T tryMakeAccessible(T object) {
if (object != null && !object.isAccessible()) {
try {
object.setAccessible(true);
} catch (SecurityException e) {
// ok
}
}
return object;
}
}