All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.datastax.driver.mapping.DefaultPropertyMapper Maven / Gradle / Ivy

There is a newer version: 3.11.5
Show newest version
/*
 * Copyright 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 static com.google.common.base.Preconditions.checkNotNull;

import com.datastax.driver.core.Metadata;
import com.datastax.driver.mapping.annotations.ClusteringColumn;
import com.datastax.driver.mapping.annotations.Column;
import com.datastax.driver.mapping.annotations.Computed;
import com.datastax.driver.mapping.annotations.Frozen;
import com.datastax.driver.mapping.annotations.FrozenKey;
import com.datastax.driver.mapping.annotations.FrozenValue;
import com.datastax.driver.mapping.annotations.PartitionKey;
import com.datastax.driver.mapping.annotations.Table;
import com.datastax.driver.mapping.annotations.Transient;
import com.datastax.driver.mapping.annotations.UDT;
import com.google.common.base.Throwables;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Sets;
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.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * 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> mapTable(Class tableClass) { return mapTableOrUdt(tableClass, VALID_COLUMN_ANNOTATIONS); } @Override public Set> mapUdt(Class udtClass) { return mapTableOrUdt(udtClass, VALID_FIELD_ANNOTATIONS); } private Set> mapTableOrUdt( Class entityClass, Collection> 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; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy