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

grails.databinding.SimpleDataBinder.groovy Maven / Gradle / Ivy

There is a newer version: 2023.2.0-M1
Show newest version
/*
 * Copyright 2014-2023 the original author or authors.
 *
 * 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
 *
 *      https://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 grails.databinding

import java.lang.annotation.Annotation
import java.lang.reflect.Array
import java.lang.reflect.Field
import java.lang.reflect.Modifier
import java.lang.reflect.ParameterizedType
import java.lang.reflect.Type
import java.security.ProtectionDomain
import java.util.regex.Matcher

import groovy.transform.CompileStatic
import groovy.transform.TypeCheckingMode
import groovy.xml.slurpersupport.GPathResult

import grails.databinding.converters.FormattedValueConverter
import grails.databinding.converters.ValueConverter
import grails.databinding.events.DataBindingListener
import grails.databinding.initializers.ValueInitializer

import org.grails.databinding.ClosureValueConverter
import org.grails.databinding.ClosureValueInitializer
import org.grails.databinding.IndexedPropertyReferenceDescriptor
import org.grails.databinding.converters.ConversionService
import org.grails.databinding.converters.FormattedDateValueConverter
import org.grails.databinding.converters.StructuredCalendarBindingEditor
import org.grails.databinding.converters.StructuredDateBindingEditor
import org.grails.databinding.converters.StructuredSqlDateBindingEditor
import org.grails.databinding.errors.SimpleBindingError
import org.grails.databinding.xml.GPathResultMap

/**
 * A data binder that will bind nested Maps to an object.
 *
 
 class Person {
    String firstName
    Address homeAddress
 }

 class Address {
    String city
    String state
 }

 def person = new Person()
 def binder = new SimpleDataBinder()
 binder.bind person, [firstName: 'Steven', homeAddress: [city: 'St. Louis', state: 'Missouri']]
 assert person.firstName == 'Steven'
 assert person.homeAddress.city == 'St. Louis'
 assert person.homeAddress.state == 'Missouri'

 
* * @author Jeff Brown * @author Graeme Rocher * * @since 3.0 */ @CompileStatic class SimpleDataBinder implements DataBinder { protected Map structuredEditors = new HashMap() ConversionService conversionService protected Map> conversionHelpers = [:].withDefault { c -> [] } protected Map formattedValueConversionHelpers = new HashMap() protected static final List BASIC_TYPES = [ String, Boolean, Byte, Short, Integer, Long, Float, Double, Character ] as List static final INDEXED_PROPERTY_REGEX = /(.*)\[\s*([^\s]*)\s*\]\s*$/ int autoGrowCollectionLimit = 256 SimpleDataBinder() { registerStructuredEditor Date, new StructuredDateBindingEditor() registerStructuredEditor java.sql.Date, new StructuredSqlDateBindingEditor() registerStructuredEditor Calendar, new StructuredCalendarBindingEditor() registerFormattedValueConverter new FormattedDateValueConverter() } void registerStructuredEditor(Class clazz, StructuredBindingEditor editor) { structuredEditors[clazz] = editor } void registerConverter(ValueConverter converter) { conversionHelpers[converter.targetType] << converter } void registerFormattedValueConverter(FormattedValueConverter converter) { formattedValueConversionHelpers[converter.targetType] = converter } /** * * @param obj The object being bound to * @param source The data binding source * @see DataBindingSource */ @Override void bind(Object obj, DataBindingSource source) { bind(obj, source, null, null, null, null) } /** * * @param obj The object being bound to * @param source The data binding source * @param listener A listener which will be notified of data binding events triggered * by this binding * @see DataBindingSource * @see DataBindingListener */ @Override void bind(Object obj, DataBindingSource source, DataBindingListener listener) { bind(obj, source, null, null, null, listener) } /** * * @param obj The object being bound to * @param source The data binding source * @param whiteList A list of property names to be included during this data binding. * All other properties represented in the binding source will be ignored. * @see DataBindingSource */ @Override void bind(Object obj, DataBindingSource source, List whiteList) { bind(obj, source, null, whiteList, null, null) } /** * * @param obj The object being bound to * @param source The data binding source * @param whiteList A list of property names to be included during this data binding. * All other properties represented in the binding source will be ignored * @param blackList A list of properties names to be excluded during this data binding. * @see DataBindingSource */ @Override void bind(Object obj, DataBindingSource source, List whiteList, List blackList) { bind(obj, source, null, whiteList, blackList, null) } /** * * @param obj The object being bound to * @param gpath A GPathResult which represents the data being bound. * @see DataBindingSource */ @Override void bind(Object obj, GPathResult gpath) { bind(obj, new SimpleMapDataBindingSource(new GPathResultMap(gpath))) } /** * * @param obj The object being bound to * @param source The data binding source * @param filter Only properties beginning with filter will be included in the * data binding. For example, if filter is "person" and the binding * source contains data for properties "person.name" and "author.name" * the value of "person.name" will be bound to obj.name. The value of * "author.name" will be ignored. * @param whiteList A list of property names to be included during this data binding. * All other properties represented in the binding source will be ignored * @param blackList A list of properties names to be excluded during this data binding. * @see DataBindingSource */ @Override void bind(Object obj, DataBindingSource source, String filter, List whiteList, List blackList) { bind(obj, source, filter, whiteList, blackList, null) } /** * * @param obj The object being bound to * @param source The data binding source * @param filter Only properties beginning with filter will be included in the * data binding. For example, if filter is "person" and the binding * source contains data for properties "person.name" and "author.name" * the value of "person.name" will be bound to obj.name. * The value of "author.name" will be ignored. * @param whiteList A list of property names to be included during this data binding. * All other properties represented in the binding source will be ignored * @param blackList A list of properties names to be excluded during this data binding. * @param listener A listener which will be notified of data binding events triggered by this binding * @see DataBindingSource * @see DataBindingListener */ @Override void bind(Object obj, DataBindingSource source, String filter, List whiteList, List blackList, DataBindingListener listener) { doBind(obj, source, filter, whiteList, blackList, listener, null) } protected void doBind(Object obj, DataBindingSource source, String filter, List whiteList, List blackList, DataBindingListener listener, Object errors) { Set keys = source.getPropertyNames() for (String key in keys) { if (!filter || key.startsWith(filter + '.')) { String propName = key if (filter) { propName = key[(1 + filter.size())..-1] } MetaProperty metaProperty = obj.metaClass.getMetaProperty propName if (metaProperty) { // normal property if (isOkToBind(metaProperty, whiteList, blackList)) { Object val = source[key] try { ValueConverter converter = getValueConverter(obj, metaProperty.name) if (converter) { bindProperty(obj, source, metaProperty, converter.convert(source), listener, errors) } else { processProperty(obj, metaProperty, preprocessValue(val), source, listener, errors) } } catch (Exception e) { addBindingError(obj, propName, val, e, listener, errors) } } } else { IndexedPropertyReferenceDescriptor descriptor = getIndexedPropertyReferenceDescriptor propName if (descriptor) { // indexed property metaProperty = obj.metaClass.getMetaProperty descriptor.propertyName if (metaProperty && isOkToBind(metaProperty, whiteList, blackList)) { Object val = source.getPropertyValue(key) processIndexedProperty(obj, metaProperty, descriptor, val, source, listener, errors) } } else if (propName.startsWith('_') && propName.length() > 1) { // boolean special handling String restOfPropertyName = propName[1..-1] if (!source.containsProperty(restOfPropertyName)) { metaProperty = obj.metaClass.getMetaProperty restOfPropertyName if (metaProperty && isOkToBind(metaProperty, whiteList, blackList)) { if ((Boolean == metaProperty.type || Boolean.TYPE == metaProperty.type)) { bindProperty(obj, source, metaProperty, false, listener, errors) } } } } } } } } protected boolean isOkToBind(String propName, List whiteList, List blackList) { 'class' != propName && 'classLoader' != propName && 'protectionDomain' != propName && 'metaClass' != propName && !blackList?.contains(propName) && (!whiteList || whiteList.contains(propName) || whiteList.find { it -> it?.toString()?.startsWith(propName + '.') }) } protected boolean isOkToBind(MetaProperty property, List whitelist, List blacklist) { isOkToBind(property.name, whitelist, blacklist) && (property.type != null && !(ClassLoader.isAssignableFrom(property.type) || ProtectionDomain.isAssignableFrom(property.type))) } protected IndexedPropertyReferenceDescriptor getIndexedPropertyReferenceDescriptor(String propName) { IndexedPropertyReferenceDescriptor descriptor = null Matcher matcher = (propName =~ INDEXED_PROPERTY_REGEX) if (matcher) { String indexedPropertyName = matcher.group(1) String index = matcher.group(2) if (index.size() > 2 && ((index.startsWith("'") && index.endsWith("'")) || (index.startsWith('"') && index.endsWith('"')))) { index = index[1..-2] } descriptor = new IndexedPropertyReferenceDescriptor(propertyName: indexedPropertyName, index: index) } descriptor } protected void processProperty(Object obj, MetaProperty metaProperty, Object val, DataBindingSource source, DataBindingListener listener, Object errors) { String propName = metaProperty.name Class propertyType = metaProperty.type if (structuredEditors.containsKey(propertyType) && (val == 'struct' || val == 'date.struct')) { StructuredBindingEditor structuredEditor = structuredEditors[propertyType] val = structuredEditor.getPropertyValue(obj, propName, source) } bindProperty(obj, source, metaProperty, val, listener, errors) } protected SimpleMapDataBindingSource splitIndexedStruct(IndexedPropertyReferenceDescriptor indexedPropertyReferenceDescriptor, DataBindingSource source) { String propName = indexedPropertyReferenceDescriptor.propertyName Map structValues = new HashMap() String prefix = indexedPropertyReferenceDescriptor for (String propertyName : source.propertyNames) { if (propertyName.startsWith(prefix)) { String deIndexedPropertyName = propName String[] parts = propertyName.split('_') if (parts.length > 1) { deIndexedPropertyName = deIndexedPropertyName + '_' + parts[1] } structValues.put(deIndexedPropertyName, source.getPropertyValue(propertyName)) } } new SimpleMapDataBindingSource(structValues) } protected void processIndexedProperty(Object obj, MetaProperty metaProperty, IndexedPropertyReferenceDescriptor indexedPropertyReferenceDescriptor, Object val, DataBindingSource source, DataBindingListener listener, Object errors) { String propName = indexedPropertyReferenceDescriptor.propertyName Class propertyType = metaProperty.type Class genericType = getReferencedTypeForCollection(propName, obj) if (structuredEditors.containsKey(genericType) && (val == 'struct' || val == 'date.struct')) { StructuredBindingEditor structuredEditor = structuredEditors[genericType] val = structuredEditor.getPropertyValue(obj, propName, splitIndexedStruct(indexedPropertyReferenceDescriptor, source)) } if (propertyType.isArray()) { int index = Integer.parseInt(indexedPropertyReferenceDescriptor.index) Object[] array = initializeArray(obj, propName, propertyType.componentType, index) if (array != null) { addElementToArrayAt(array, index, val) } } else if (Collection.isAssignableFrom(propertyType)) { int index = Integer.parseInt(indexedPropertyReferenceDescriptor.index) Collection collectionInstance = initializeCollection(obj, propName, propertyType) Object indexedInstance = null if (!(Set.isAssignableFrom(propertyType))) { indexedInstance = collectionInstance[index] } if (indexedInstance == null) { if (genericType) { if (genericType.isAssignableFrom(val?.getClass())) { addElementToCollectionAt(obj, propName, collectionInstance, index, val) } else if (isBasicType(genericType)) { addElementToCollectionAt(obj, propName, collectionInstance, index, convert(genericType, val)) } else if (val instanceof Map) { indexedInstance = genericType.newInstance() bind(indexedInstance, new SimpleMapDataBindingSource(val), listener) addElementToCollectionAt(obj, propName, collectionInstance, index, indexedInstance) } else if (val instanceof DataBindingSource) { indexedInstance = genericType.newInstance() bind(indexedInstance, val, listener) addElementToCollectionAt(obj, propName, collectionInstance, index, indexedInstance) } else if (genericType.isEnum() && val instanceof CharSequence) { Object enumValue = convertStringToEnum(genericType, val.toString()) addElementToCollectionAt(obj, propName, collectionInstance, index, enumValue) } else { addElementToCollectionAt(obj, propName, collectionInstance, index, convert(genericType, val)) } } else { addElementToCollectionAt(obj, propName, collectionInstance, index, val) } } else { if (val instanceof Map) { bind(indexedInstance, new SimpleMapDataBindingSource(val), listener) } else if (val instanceof DataBindingSource) { bind(indexedInstance, val, listener) } else if (val == null && indexedInstance != null) { addElementToCollectionAt(obj, propName, collectionInstance, index, null) } } } else if (Map.isAssignableFrom(propertyType)) { Map mapInstance = initializeMap(obj, propName) if (mapInstance.size() < autoGrowCollectionLimit || mapInstance.containsKey(indexedPropertyReferenceDescriptor.index)) { Class referencedType = getReferencedTypeForCollection(propName, obj) if (referencedType != null) { if (val instanceof Map) { mapInstance[indexedPropertyReferenceDescriptor.index] = referencedType.newInstance(val) } else { mapInstance[indexedPropertyReferenceDescriptor.index] = convert(referencedType, val) } } else { mapInstance[indexedPropertyReferenceDescriptor.index] = val } } } } @CompileStatic(TypeCheckingMode.SKIP) protected Object[] initializeArray(obj, String propertyName, Class arrayType, int index) { Object[] array = obj[propertyName] if (array == null && index < autoGrowCollectionLimit) { array = Array.newInstance(arrayType, index + 1) obj[propertyName] = array } else if (array != null && array.length <= index && index < autoGrowCollectionLimit) { Object newArray = Array.newInstance(arrayType, index + 1) System.arraycopy(array, 0, newArray, 0, array.length) array = newArray obj[propertyName] = newArray } array } protected boolean isBasicType(Class c) { BASIC_TYPES.contains(c) || c.isPrimitive() } protected Class getReferencedTypeForCollectionInClass(String propertyName, Class clazz) { Class referencedType Field field = getField(clazz, propertyName) if (field) { Type genericType = field.genericType if (genericType instanceof ParameterizedType) { ParameterizedType pt = (ParameterizedType) genericType Class rawType = pt.getRawType() if (Map.isAssignableFrom(rawType)) { referencedType = pt.getActualTypeArguments()[1] } else { referencedType = pt.getActualTypeArguments()[0] } } } referencedType } protected Class getReferencedTypeForCollection(String propertyName, Object obj) { getReferencedTypeForCollectionInClass(propertyName, obj.getClass()) } protected boolean isOkToAddElementAt(Collection collection, int index) { boolean isOk if (collection instanceof Set) { isOk = collection.size() < autoGrowCollectionLimit } else { isOk = (index < autoGrowCollectionLimit || index < collection.size()) } isOk } @CompileStatic(TypeCheckingMode.SKIP) protected void addElementToCollectionAt(Object obj, String propertyName, Collection collection, int index, Object val) { if (isOkToAddElementAt(collection, index)) { if (collection instanceof Set) { collection.add(val) } else { collection[index] = val } } } @CompileStatic(TypeCheckingMode.SKIP) protected addElementToArrayAt(Object[] array, int index, Object val) { if (array.length > index) { array[index] = convert(array.class.componentType, val) } } protected Map initializeMap(Object obj, String propertyName) { if (obj[propertyName] == null) { obj[propertyName] = [:] } (Map) obj[propertyName] } protected Collection initializeCollection(Object obj, String propertyName, Class type, boolean reuseExistingCollectionIfExists = true) { Object val = null if (reuseExistingCollectionIfExists) { val = obj[propertyName] } if (val == null) { val = getDefaultCollectionInstanceForType(type) obj[propertyName] = val } (Collection) val } protected Object getDefaultCollectionInstanceForType(Class type) { Object val if (List.isAssignableFrom(type)) { val = [] } else if (SortedSet.isAssignableFrom(type)) { val = new TreeSet() } else if (LinkedHashSet.isAssignableFrom(type)) { val = new LinkedHashSet() } else if (Collection.isAssignableFrom(type)) { val = new HashSet() } val } /** * Get a ValueConverter for field * * @param field The field to retrieve a converter for * @param formattingValue The format that the converter will use * @return a ValueConverter for field which uses formattingValue for its format * @see BindingFormat */ protected ValueConverter getFormattedConverter(Field field, String formattingValue) { ValueConverter converter FormattedValueConverter formattedConverter = formattedValueConversionHelpers[field.type] if (formattedConverter) { converter = { SimpleMapDataBindingSource source -> Object value = preprocessValue(source.getPropertyValue(field.name)) Object convertedValue = null if (value != null) { convertedValue = formattedConverter.convert(value, formattingValue) } convertedValue } as ValueConverter } converter } protected Field getField(Class clazz, String fieldName) { Field field = null try { field = clazz.getDeclaredField(fieldName) } catch (NoSuchFieldException ignored) { Class superClass = clazz.getSuperclass() if (superClass != Object) { field = getField(superClass, fieldName) } } field } protected ValueConverter getValueConverterForField(Object obj, String propName) { ValueConverter converter try { Field field = getField(obj.getClass(), propName) if (field) { Annotation annotation = field.getAnnotation(BindUsing) if (annotation) { Class valueClass = getValueOfBindUsing(annotation) if (Closure.isAssignableFrom(valueClass)) { Closure closure = (Closure) valueClass.newInstance(null, null) converter = new ClosureValueConverter(converterClosure: closure.curry(obj), targetType: field.type) } } else { annotation = field.getAnnotation(BindingFormat) if (annotation) { converter = getFormattedConverter(field, getFormatString(annotation)) } } } } catch (Exception ignored) { } converter } /** * @param annotation An instance of grails.databinding.BindingUsing or org.grails.databinding.BindingUsing * @return the value Class of the annotation */ protected Class getValueOfBindUsing(Annotation annotation) { assert annotation instanceof BindUsing Class value if (annotation instanceof BindUsing) { value = ((BindUsing) annotation).value() } value } /** * @param annotation An instance of grails.databinding.BindingFormat or org.grails.databinding.BindingFormat * @return the value String of the annotation */ protected String getFormatString(Annotation annotation) { assert annotation instanceof BindingFormat String formatString if (annotation instanceof BindingFormat) { formatString = ((BindingFormat) annotation).value() } formatString } protected ValueConverter getValueConverterForClass(Object obj, String propName) { ValueConverter converter Class objClass = obj.getClass() Annotation annotation = objClass.getAnnotation(BindUsing) if (annotation) { Class valueClass = getValueOfBindUsing(annotation) if (BindingHelper.isAssignableFrom(valueClass)) { BindingHelper dataConverter = (BindingHelper) valueClass.newInstance() converter = new ClosureValueConverter(converterClosure: { DataBindingSource it -> dataConverter.getPropertyValue(obj, propName, it) }) } } converter } protected ValueConverter getValueConverter(Object obj, String propName) { ValueConverter converter = getValueConverterForField obj, propName converter = converter ?: getValueConverterForClass(obj, propName) converter } @CompileStatic(TypeCheckingMode.SKIP) protected convertStringToEnum(Class enumClass, String value) { try { enumClass.valueOf(value) } catch (IllegalArgumentException ignored) { } } protected Object preprocessValue(Object propertyValue) { propertyValue } protected void setPropertyValue(Object obj, DataBindingSource source, MetaProperty metaProperty, Object propertyValue, DataBindingListener listener) { boolean convertCollectionElements = false if (propertyValue instanceof Collection) { Class referencedType = getReferencedTypeForCollection(metaProperty.name, obj) if (referencedType) { Object nonAssignableValue = propertyValue.find { it != null && !(referencedType.isAssignableFrom(it.getClass())) } if (nonAssignableValue != null) { convertCollectionElements = true } } } setPropertyValue(obj, source, metaProperty, propertyValue, listener, convertCollectionElements) } protected void setPropertyValue(Object obj, DataBindingSource source, MetaProperty metaProperty, Object propertyValue, DataBindingListener listener, boolean convertCollectionElements) { String propName = metaProperty.name Type propertyType MetaMethod propertyGetter if (metaProperty instanceof MetaBeanProperty) { MetaBeanProperty mbp = (MetaBeanProperty) metaProperty propertyType = mbp.getter?.returnType ?: mbp.field?.type if (propertyType && (propertyType.interface || Modifier.isAbstract(propertyType.modifiers))) { propertyType = mbp.field?.type } propertyGetter = mbp.getter } if (propertyType == null || propertyType == Object) { propertyType = metaProperty.type if (propertyType == null || propertyType == Object) { propertyType = getField(obj.getClass(), propName)?.type ?: Object } } if (propertyValue == null || propertyType == Object || propertyType.isAssignableFrom(propertyValue.getClass())) { if (convertCollectionElements && ((!(propertyValue instanceof Range) && propertyValue instanceof Collection && Collection.isAssignableFrom(propertyType) && propertyGetter))) { addElementsToCollection(obj, propName, propertyValue, true) } else { obj[propName] = propertyValue } } else if (propertyValue instanceof List && Set.isAssignableFrom(propertyType) && !SortedSet.isAssignableFrom(propertyType)) { addElementsToCollection(obj, propName, propertyValue, true) } else { if (propertyValue instanceof Map) { if (Collection.isAssignableFrom(propertyType) && propertyValue.size() == 1 && ((Map) propertyValue)[propertyValue.keySet()[0]] instanceof List) { Object key = propertyValue.keySet()[0] List list = (List) ((Map) propertyValue)[key] addElementsToCollection(obj, propName, list) } else { if (obj[propName] == null) { initializeProperty(obj, propName, propertyType, source) } bind(obj[propName], new SimpleMapDataBindingSource(propertyValue), listener) } } else if (propertyValue instanceof DataBindingSource) { if (Collection.isAssignableFrom(propertyType) && propertyValue.size() == 1 && ((Map) propertyValue)[propertyValue.getPropertyNames()[0]] instanceof List) { String key = propertyValue.getPropertyNames()[0] List list = (List) ((Map) propertyValue)[key] addElementsToCollection(obj, propName, list) } else { if (obj[propName] == null) { initializeProperty(obj, propName, propertyType, source) } bind(obj[propName], propertyValue, listener) } } else if (Collection.isAssignableFrom(propertyType) && propertyValue instanceof String) { addElementToCollection(obj, propName, propertyType, propertyValue, true) } else if (Collection.isAssignableFrom(propertyType) && propertyValue instanceof Number) { addElementToCollection(obj, propName, propertyType, propertyValue, true) } else if (Collection.isAssignableFrom(propertyType) && propertyValue.getClass().isArray()) { addElementsToCollection(obj, propName, propertyValue as Collection, true) } else { obj[propName] = convert(propertyType, propertyValue) } } } protected boolean addElementToCollection(Object obj, String propName, Class propertyType, Object propertyValue, boolean clearCollection) { boolean isSet = false Collection coll = initializeCollection(obj, propName, propertyType) if (coll != null) { if (clearCollection) { coll.clear() } Class referencedType = getReferencedTypeForCollection propName, obj if (referencedType != null) { if (propertyValue == null || referencedType.isAssignableFrom(propertyValue.getClass())) { coll << propertyValue isSet = true } else { coll << convert(referencedType, propertyValue) isSet = true } } } isSet } protected bindProperty(Object obj, DataBindingSource source, MetaProperty metaProperty, Object propertyValue, DataBindingListener listener, Object errors) { String propName = metaProperty.name if (listener == null || listener.beforeBinding(obj, propName, propertyValue, errors) != false) { try { setPropertyValue(obj, source, metaProperty, propertyValue, listener) } catch (Exception e) { addBindingError(obj, propName, propertyValue, e, listener, errors) } } else if (listener != null && propertyValue instanceof Map && obj[propName] != null) { bind(obj[propName], new SimpleMapDataBindingSource(propertyValue)) } listener?.afterBinding(obj, propName, errors) } protected void addBindingError(Object obj, String propName, propertyValue, Exception e, DataBindingListener listener, Object errors) { if (listener) { SimpleBindingError error = new SimpleBindingError(obj, propName, propertyValue, e.cause ?: e) listener.bindingError error, errors } } private void addElementsToCollection(Object obj, String collectionPropertyName, Collection collection, boolean removeExistingElements = false) { Class propertyType = obj.metaClass.getMetaProperty(collectionPropertyName).type Class referencedType = getReferencedTypeForCollection(collectionPropertyName, obj) Collection coll = initializeCollection(obj, collectionPropertyName, propertyType, !removeExistingElements) if (removeExistingElements) { coll.clear() } for (element in collection) { if (element == null || referencedType == null || referencedType.isAssignableFrom(element.getClass())) { coll << element } else { coll << convert(referencedType, element) } } obj[collectionPropertyName] = coll } protected void initializeProperty(Object obj, String propName, Class propertyType, DataBindingSource source) { ValueInitializer initializer = getPropertyInitializer(obj, propName) if (initializer) { obj[propName] = initializer.initialize() } else { obj[propName] = propertyType.newInstance() } } protected ValueInitializer getPropertyInitializer(Object obj, String propName) { ValueInitializer initializer = getValueInitializerForField obj, propName initializer } protected ValueInitializer getValueInitializerForField(Object obj, String propName) { ValueInitializer initializer try { Field field = getField(obj.getClass(), propName) if (field) { Annotation annotation = field.getAnnotation(BindInitializer) if (annotation) { Class valueClass = getValueOfBindInitializer(annotation) if (Closure.isAssignableFrom(valueClass)) { Closure closure = (Closure) valueClass.newInstance(null, null) initializer = new ClosureValueInitializer(initializerClosure: closure.curry(obj), targetType: field.type) } } } } catch (Exception ignored) { } initializer } /** * @param annotation An instance of grails.databinding.BindInitializer * @return the value Class of the annotation */ protected Class getValueOfBindInitializer(Annotation annotation) { assert annotation instanceof BindInitializer Class value if (annotation instanceof BindInitializer) { value = ((BindInitializer) annotation).value() } value } protected Object convert(Class typeToConvertTo, Object value) { if (value == null || typeToConvertTo.isAssignableFrom(value?.getClass())) { return value } if (conversionHelpers.containsKey(typeToConvertTo)) { ValueConverter converter = getConverter(typeToConvertTo, value) if (converter) { return converter.convert(value) } } if (conversionService?.canConvert(value.getClass(), typeToConvertTo)) { return conversionService.convert(value, typeToConvertTo) } if (Collection.isAssignableFrom(typeToConvertTo) && value instanceof String[]) { if (typeToConvertTo == Set) { return value as Set } if (typeToConvertTo == List) { return value as List } } else if (typeToConvertTo.isPrimitive() || typeToConvertTo.isArray()) { return value } else if (value instanceof Map) { Object obj = typeToConvertTo.newInstance() bind(obj, new SimpleMapDataBindingSource(value)) return obj } else if (Enum.isAssignableFrom(typeToConvertTo) && value instanceof String) { return convertStringToEnum((Class) typeToConvertTo, value) } typeToConvertTo.newInstance(value) } protected ValueConverter getConverter(Class typeToConvertTo, Object value) { List converters = conversionHelpers.get(typeToConvertTo) converters?.find { ValueConverter c -> c.canConvert(value) } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy