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

com.univocity.parsers.annotations.helpers.AnnotationHelper Maven / Gradle / Ivy

/*******************************************************************************
 * Copyright 2014 uniVocity Software Pty Ltd
 *
 * 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.univocity.parsers.annotations.helpers;

import com.univocity.parsers.annotations.*;
import com.univocity.parsers.annotations.Format;
import com.univocity.parsers.common.*;
import com.univocity.parsers.common.beans.*;
import com.univocity.parsers.common.input.*;
import com.univocity.parsers.conversions.*;

import java.lang.annotation.Annotation;
import java.lang.reflect.*;
import java.math.*;
import java.text.*;
import java.util.*;

/**
 * Helper class to process fields annotated with {@link Parsed}
 *
 * @author uniVocity Software Pty Ltd - [email protected]
 */
public class AnnotationHelper {

	private AnnotationHelper() {

	}

	/**
	 * Converts the special "null" strings that might be provided by {@link Parsed#defaultNullRead() and  Parsed#defaultNullWrite()}
	 *
	 * @param defaultValue The string returned by {@link Parsed#defaultNullRead() and  Parsed#defaultNullWrite()}
	 *
	 * @return the default value if it is not the String literal "null" or "'null'".
	 * 

If "null" was provided, then null will be returned. *

If "'null'" was provided, then "null" will be returned. */ private static String getNullValue(String defaultValue) { if ("null".equals(defaultValue)) { return null; } if ("'null'".equals(defaultValue)) { return "null"; } return defaultValue; } private static String getNullWriteValue(Parsed parsed) { if (parsed == null) { return null; } return getNullValue(parsed.defaultNullWrite()); } private static String getNullReadValue(Parsed parsed) { if (parsed == null) { return null; } return getNullValue(parsed.defaultNullRead()); } /** * Identifies the proper conversion for a given Field and an annotation from the package {@link com.univocity.parsers.annotations} * * @param target The field or method to have conversions applied to * @param annotation the annotation from {@link com.univocity.parsers.annotations} that identifies a {@link Conversion} instance. * * @return The {@link Conversion} that should be applied to the field */ @SuppressWarnings("rawtypes") public static Conversion getConversion(AnnotatedElement target, Annotation annotation) { return getConversion(getType(target), target, annotation); } /** * Identifies the proper conversion for a given type and an annotation from the package {@link com.univocity.parsers.annotations} * * @param classType the type to have conversions applied to * @param annotation the annotation from {@link com.univocity.parsers.annotations} that identifies a {@link Conversion} instance. * * @return The {@link Conversion} that should be applied to the type */ @SuppressWarnings("rawtypes") public static Conversion getConversion(Class classType, Annotation annotation) { return getConversion(classType, null, annotation); } @SuppressWarnings({"rawtypes", "unchecked"}) private static Conversion getConversion(Class fieldType, AnnotatedElement target, Annotation annotation) { try { Parsed parsed = target == null ? null : findAnnotation(target, Parsed.class); Class annType = annotation.annotationType(); String nullRead = getNullReadValue(parsed); String nullWrite = getNullWriteValue(parsed); if (annType == NullString.class) { String[] nulls = ((NullString) annotation).nulls(); return Conversions.toNull(nulls); } else if (annType == EnumOptions.class) { if (!fieldType.isEnum()) { if (target == null) { throw new IllegalStateException("Invalid " + EnumOptions.class.getName() + " instance for converting class " + fieldType.getName() + ". Not an enum type."); } else { throw new IllegalStateException("Invalid " + EnumOptions.class.getName() + " annotation on " + describeElement(target) + ". Attribute must be an enum type."); } } EnumOptions enumOptions = ((EnumOptions) annotation); String element = enumOptions.customElement().trim(); if (element.isEmpty()) { element = null; } Enum nullReadValue = nullRead == null ? null : Enum.valueOf(fieldType, nullRead); return new EnumConversion(fieldType, nullReadValue, nullWrite, element, enumOptions.selectors()); } else if (annType == Trim.class) { int length = ((Trim) annotation).length(); if (length == -1) { return Conversions.trim(); } else { return Conversions.trim(length); } } else if (annType == LowerCase.class) { return Conversions.toLowerCase(); } else if (annType == UpperCase.class) { return Conversions.toUpperCase(); } else if (annType == Replace.class) { Replace replace = ((Replace) annotation); return Conversions.replace(replace.expression(), replace.replacement()); } else if (annType == BooleanString.class) { if (fieldType != boolean.class && fieldType != Boolean.class) { if (target == null) { throw new DataProcessingException("Invalid usage of " + BooleanString.class.getName() + ". Got type " + fieldType.getName() + " instead of boolean."); } else { throw new DataProcessingException("Invalid annotation: " + describeElement(target) + " has type " + fieldType.getName() + " instead of boolean."); } } BooleanString boolString = ((BooleanString) annotation); String[] falseStrings = boolString.falseStrings(); String[] trueStrings = boolString.trueStrings(); Boolean valueForNull = nullRead == null ? null : BooleanConversion.getBoolean(nullRead, trueStrings, falseStrings); if (valueForNull == null && fieldType == boolean.class) { valueForNull = Boolean.FALSE; } return Conversions.toBoolean(valueForNull, nullWrite, trueStrings, falseStrings); } else if (annType == Format.class) { Format format = ((Format) annotation); String[] formats = format.formats(); String[] options = format.options(); Locale locale = extractLocale(options); Conversion conversion = null; if (fieldType == BigDecimal.class) { BigDecimal defaultForNull = nullRead == null ? null : new BigDecimal(nullRead); conversion = Conversions.formatToBigDecimal(defaultForNull, nullWrite, formats); } else if (Number.class.isAssignableFrom(fieldType)) { conversion = Conversions.formatToNumber(formats); ((NumericConversion) conversion).setNumberType(fieldType); } else { Date dateIfNull = null; if (nullRead != null) { if ("now".equalsIgnoreCase(nullRead)) { dateIfNull = new Date(); } else { if (formats.length == 0) { throw new DataProcessingException("No format defined"); } SimpleDateFormat sdf = new SimpleDateFormat(formats[0], locale); dateIfNull = sdf.parse(nullRead); } } if (Date.class == fieldType) { conversion = Conversions.toDate(locale, dateIfNull, nullWrite, formats); } else if (Calendar.class == fieldType) { Calendar calendarIfNull = null; if (dateIfNull != null) { calendarIfNull = Calendar.getInstance(); calendarIfNull.setTime(dateIfNull); } conversion = Conversions.toCalendar(locale, calendarIfNull, nullWrite, formats); } } if (conversion != null) { if (options.length > 0) { //noinspection ConstantConditions if (conversion instanceof FormattedConversion) { Object[] formatters = ((FormattedConversion) conversion).getFormatterObjects(); for (Object formatter : formatters) { applyFormatSettings(formatter, options); } } else { throw new DataProcessingException("Options '" + Arrays.toString(options) + "' not supported by conversion of type '" + conversion.getClass() + "'. It must implement " + FormattedConversion.class); } } return conversion; } } else if (annType == Convert.class) { Convert convert = ((Convert) annotation); String[] args = convert.args(); Class conversionClass = convert.conversionClass(); return (Conversion) newInstance(Conversion.class, conversionClass, args); } if (fieldType == String.class && (nullRead != null || nullWrite != null)) { return new ToStringConversion(nullRead, nullWrite); } return null; } catch (DataProcessingException ex) { throw ex; } catch (Throwable ex) { if (target == null) { throw new DataProcessingException("Unexpected error identifying conversions to apply over type " + fieldType, ex); } else { throw new DataProcessingException("Unexpected error identifying conversions to apply over " + describeElement(target), ex); } } } private static Locale extractLocale(String[] options) { for (int i = 0; i < options.length; i++) { if (options[i] != null && options[i].startsWith("locale=")) { String locale = options[i].substring("locale=".length()); String languageCode; String countryCode; String variant; CharAppender appender = new DefaultCharAppender(100, "", 0); int j = 0; char ch; for (; j < locale.length() && Character.isLetterOrDigit((ch = locale.charAt(j))); j++, appender.append(ch)) ; languageCode = appender.getAndReset(); for (++j; j < locale.length() && Character.isLetterOrDigit((ch = locale.charAt(j))); j++, appender.append(ch)) ; countryCode = appender.getAndReset(); for (++j; j < locale.length() && Character.isLetterOrDigit((ch = locale.charAt(j))); j++, appender.append(ch)) ; variant = appender.getAndReset(); options[i] = null; return new Locale(languageCode, countryCode, variant); } } return Locale.getDefault(); } public static T newInstance(Class parent, Class type, String[] args) { if (!parent.isAssignableFrom(type)) { throw new DataProcessingException("Not a valid " + parent.getSimpleName() + " class: '" + type.getSimpleName() + "' (" + type.getName() + ')'); } try { Constructor constructor = type.getConstructor(String[].class); return (T) constructor.newInstance((Object) args); } catch (NoSuchMethodException e) { if (args.length == 0) { try { return type.newInstance(); } catch (Exception ex) { throw new DataProcessingException("Unexpected error instantiating custom " + parent.getSimpleName() + " class '" + type.getSimpleName() + "' (" + type.getName() + ')', e); } } throw new DataProcessingException("Could not find a public constructor with a String[] parameter in custom " + parent.getSimpleName() + " class '" + type.getSimpleName() + "' (" + type.getName() + ')', e); } catch (Exception e) { throw new DataProcessingException("Unexpected error instantiating custom " + parent.getSimpleName() + " class '" + type.getSimpleName() + "' (" + type.getName() + ')', e); } } /** * Identifies the proper conversion for a given type * * @param fieldType The type of field to have conversions applied to. * @param parsed the {@link Parsed} annotation from {@link com.univocity.parsers.annotations}. * * @return The {@link Conversion} that should be applied to the field type */ @SuppressWarnings({"rawtypes", "unchecked"}) public static Conversion getDefaultConversion(Class fieldType, Parsed parsed) { String nullRead = getNullReadValue(parsed); Object valueIfStringIsNull = null; ObjectConversion conversion = null; if (fieldType == Boolean.class || fieldType == boolean.class) { conversion = Conversions.toBoolean(); valueIfStringIsNull = nullRead == null ? null : Boolean.valueOf(nullRead); } else if (fieldType == Character.class || fieldType == char.class) { conversion = Conversions.toChar(); if (nullRead != null && nullRead.length() > 1) { throw new DataProcessingException("Invalid default value for character '" + nullRead + "'. It should contain one character only."); } valueIfStringIsNull = nullRead == null ? null : nullRead.charAt(0); } else if (fieldType == Byte.class || fieldType == byte.class) { conversion = Conversions.toByte(); valueIfStringIsNull = nullRead == null ? null : Byte.valueOf(nullRead); } else if (fieldType == Short.class || fieldType == short.class) { conversion = Conversions.toShort(); valueIfStringIsNull = nullRead == null ? null : Short.valueOf(nullRead); } else if (fieldType == Integer.class || fieldType == int.class) { conversion = Conversions.toInteger(); valueIfStringIsNull = nullRead == null ? null : Integer.valueOf(nullRead); } else if (fieldType == Long.class || fieldType == long.class) { conversion = Conversions.toLong(); valueIfStringIsNull = nullRead == null ? null : Long.valueOf(nullRead); } else if (fieldType == Float.class || fieldType == float.class) { conversion = Conversions.toFloat(); valueIfStringIsNull = nullRead == null ? null : Float.valueOf(nullRead); } else if (fieldType == Double.class || fieldType == double.class) { conversion = Conversions.toDouble(); valueIfStringIsNull = nullRead == null ? null : Double.valueOf(nullRead); } else if (fieldType == BigInteger.class) { conversion = Conversions.toBigInteger(); valueIfStringIsNull = nullRead == null ? null : new BigInteger(nullRead); } else if (fieldType == BigDecimal.class) { conversion = Conversions.toBigDecimal(); valueIfStringIsNull = nullRead == null ? null : new BigDecimal(nullRead); } else if (Enum.class.isAssignableFrom(fieldType)) { conversion = Conversions.toEnum(fieldType); } if (conversion != null) { conversion.setValueIfStringIsNull(valueIfStringIsNull); conversion.setValueIfObjectIsNull(getNullWriteValue(parsed)); } return conversion; } /** * Returns the default {@link Conversion} that should be applied to the field based on its type. * * @param target The field or method whose values must be converted from a given parsed String. * * @return The default {@link Conversion} applied to the given field. */ @SuppressWarnings("rawtypes") public static Conversion getDefaultConversion(AnnotatedElement target) { Parsed parsed = findAnnotation(target, Parsed.class); return getDefaultConversion(getType(target), parsed); } /** * Applied the configuration of a formatter object ({@link SimpleDateFormat}, {@link NumberFormat} and others). * * @param formatter the formatter instance * @param propertiesAndValues a sequence of key-value pairs, where the key is a property of the formatter * object to be set to the following value via reflection */ public static void applyFormatSettings(Object formatter, String[] propertiesAndValues) { if (propertiesAndValues.length == 0) { return; } Map values = new HashMap(); for (String setting : propertiesAndValues) { if (setting == null) { continue; } String[] pair = setting.split("="); if (pair.length != 2) { throw new DataProcessingException("Illegal format setting '" + setting + "' among: " + Arrays.toString(propertiesAndValues)); } values.put(pair[0], pair[1]); } try { for (PropertyWrapper property : BeanHelper.getPropertyDescriptors(formatter.getClass())) { String name = property.getName(); String value = values.remove(name); if (value != null) { invokeSetter(formatter, property, value); } if ("decimalFormatSymbols".equals(property.getName())) { DecimalFormatSymbols modifiedDecimalSymbols = new DecimalFormatSymbols(); boolean modified = false; try { for (PropertyWrapper prop : BeanHelper.getPropertyDescriptors(modifiedDecimalSymbols.getClass())) { value = values.remove(prop.getName()); if (value != null) { invokeSetter(modifiedDecimalSymbols, prop, value); modified = true; } } if (modified) { Method writeMethod = property.getWriteMethod(); if (writeMethod != null) { writeMethod.invoke(formatter, modifiedDecimalSymbols); } else { throw new IllegalStateException("No write method defined for property " + property.getName()); } } } catch (Throwable ex) { throw new DataProcessingException("Error trying to configure decimal symbols of formatter '" + formatter.getClass() + '.', ex); } } } } catch (Exception e) { //ignore and proceed } if (!values.isEmpty()) { throw new DataProcessingException("Cannot find properties in formatter of type '" + formatter.getClass() + "': " + values); } } private static void invokeSetter(Object formatter, PropertyWrapper property, final String value) { Method writeMethod = property.getWriteMethod(); if (writeMethod == null) { DataProcessingException exception = new DataProcessingException("Cannot set property '" + property.getName() + "' of formatter '" + formatter.getClass() + "' to '{value}'. No setter defined"); exception.setValue(value); throw exception; } Class parameterType = writeMethod.getParameterTypes()[0]; Object parameterValue = null; if (parameterType == String.class) { parameterValue = value; } else if (parameterType == Integer.class || parameterType == int.class) { parameterValue = Integer.parseInt(value); } else if (parameterType == Character.class || parameterType == char.class) { parameterValue = value.charAt(0); } else if (parameterType == Currency.class) { parameterValue = Currency.getInstance(value); } else if (parameterType == Boolean.class || parameterType == boolean.class) { parameterValue = Boolean.valueOf(value); } else if (parameterType == TimeZone.class) { parameterValue = TimeZone.getTimeZone(value); } else if (parameterType == DateFormatSymbols.class) { parameterValue = DateFormatSymbols.getInstance(new Locale(value)); } if (parameterValue == null) { DataProcessingException exception = new DataProcessingException("Cannot set property '" + property.getName() + "' of formatter '" + formatter.getClass() + ". Cannot convert '{value}' to instance of " + parameterType); exception.setValue(value); throw exception; } try { writeMethod.invoke(formatter, parameterValue); } catch (Throwable e) { DataProcessingException exception = new DataProcessingException("Error setting property '" + property.getName() + "' of formatter '" + formatter.getClass() + ", with '{parameterValue}' (converted from '{value}')", e); exception.setValue("parameterValue", parameterValue); exception.setValue(value); throw exception; } } private static boolean allFieldsIndexOrNameBased(boolean searchName, Class beanClass, MethodFilter filter) { boolean hasAnnotation = false; for (TransformedHeader header : getFieldSequence(beanClass, true, null, filter)) { if (header == null || header.getTarget() == null) { continue; } AnnotatedElement element = header.getTarget(); if (element instanceof Method && filter.reject((Method) element)) { continue; } Parsed annotation = findAnnotation(element, Parsed.class); if (annotation != null) { hasAnnotation = true; if ((annotation.index() != -1 && searchName) || (annotation.index() == -1 && !searchName)) { return false; } } } return hasAnnotation; } /** * Runs through all annotations of a given class to identify whether all annotated fields and methods * (with the {@link Parsed} annotation) are mapped to a column by index. * * @param beanClass a class whose {@link Parsed} annotations will be processed. * * @return {@code true} if every field and method annotated with {@link Parsed} in the given class maps to an index, otherwise {@code false}. */ public static boolean allFieldsIndexBasedForParsing(Class beanClass) { return allFieldsIndexOrNameBased(false, beanClass, MethodFilter.ONLY_SETTERS); } /** * Runs through all annotations of a given class to identify whether all annotated fields and methods * (with the {@link Parsed} annotation) are mapped to a column by name. * * @param beanClass a class whose {@link Parsed} annotations will be processed. * * @return {@code true} if every field and method annotated with {@link Parsed} in the given class maps to a header name, otherwise {@code false}. */ public static boolean allFieldsNameBasedForParsing(Class beanClass) { return allFieldsIndexOrNameBased(true, beanClass, MethodFilter.ONLY_SETTERS); } /** * Runs through all annotations of a given class to identify whether all annotated fields and methods * (with the {@link Parsed} annotation) are mapped to a column by index. * * @param beanClass a class whose {@link Parsed} annotations will be processed. * * @return {@code true} if every field and method annotated with {@link Parsed} in the given class maps to an index, otherwise {@code false}. */ public static boolean allFieldsIndexBasedForWriting(Class beanClass) { return allFieldsIndexOrNameBased(false, beanClass, MethodFilter.ONLY_GETTERS); } /** * Runs through all annotations of a given class to identify whether all annotated fields and methods * (with the {@link Parsed} annotation) are mapped to a column by name. * * @param beanClass a class whose {@link Parsed} annotations will be processed. * * @return {@code true} if every field and method annotated with {@link Parsed} in the given class maps to a header name, otherwise {@code false}. */ public static boolean allFieldsNameBasedForWriting(Class beanClass) { return allFieldsIndexOrNameBased(true, beanClass, MethodFilter.ONLY_GETTERS); } /** * Runs through all {@link Parsed} annotations of a given class to identify all indexes associated with its fields * * @param beanClass a class whose {@link Parsed} annotations will be processed. * @param filter filter to apply over annotated methods when the class is being used for reading data from beans (to write values to an output) * or when writing values into beans (while parsing). It is used to choose either a "get" or a "set" * method annotated with {@link Parsed}, when both methods target the same field. * * @return an array of column indexes used by the given class */ public static Integer[] getSelectedIndexes(Class beanClass, MethodFilter filter) { List indexes = new ArrayList(); for (TransformedHeader header : getFieldSequence(beanClass, true, null, filter)) { if (header == null) { continue; } int index = header.getHeaderIndex(); if (index != -1) { if (filter == MethodFilter.ONLY_GETTERS && indexes.contains(index)) { //allows the same column to be mapped to multiple fields when parsing, but not when writing. throw new IllegalArgumentException("Duplicate field index '" + index + "' found in attribute '" + header.getTargetName() + "' of class " + beanClass.getName()); } indexes.add(index); } } return indexes.toArray(new Integer[indexes.size()]); } /** * Runs through all {@link Parsed} annotations of a given class to identify all header names associated with its fields * * @param beanClass a class whose {@link Parsed} annotations will be processed. * @param filter a filter to exclude annotated methods that won't be used for parsing or writing * * @return an array of column names used by the given class */ public static String[] deriveHeaderNamesFromFields(Class beanClass, MethodFilter filter) { List sequence = getFieldSequence(beanClass, true, null, filter); List out = new ArrayList(sequence.size()); for (TransformedHeader field : sequence) { if (field == null) { return ArgumentUtils.EMPTY_STRING_ARRAY; // some field has an index that goes beyond list of header names, can't derive. } out.add(field.getHeaderName()); } return out.toArray(new String[out.size()]); } /** * Searches for the {@link Headers} annotation in the hierarchy of a class * * @param beanClass the class whose hierarchy will be searched * * @return the {@link Headers} annotation of the given class or its most immediate parent, or {@code null} if not found. */ public static Headers findHeadersAnnotation(Class beanClass) { Headers headers; Class parent = beanClass; do { headers = parent.getAnnotation(Headers.class); if (headers != null) { return headers; } else { for (Class iface : parent.getInterfaces()) { headers = findHeadersAnnotation(iface); if (headers != null) { return headers; } } } parent = parent.getSuperclass(); } while (parent != null); return null; } public static Class getType(AnnotatedElement element) { if (element instanceof Field) { return ((Field) element).getType(); } Method method = (Method) element; Class[] params = method.getParameterTypes(); if (params.length == 1) { return params[0]; } else if (params.length > 1) { throw new IllegalArgumentException("Method " + describeElement(element) + " cannot have multiple parameters"); } Class returnType = method.getReturnType(); if (returnType != void.class) { return returnType; } throw new IllegalArgumentException("Method " + describeElement(element) + " must return a value if it has no input parameter"); } public static Class getDeclaringClass(AnnotatedElement element) { if (element instanceof Field) { return ((Field) element).getDeclaringClass(); } else { return ((Method) element).getDeclaringClass(); } } public static String getName(AnnotatedElement element) { if (element instanceof Field) { return ((Field) element).getName(); } else { return ((Method) element).getName(); } } static String describeElement(AnnotatedElement element) { String description; if (element instanceof Field) { description = "attribute '" + ((Field) element).getName() + "'"; } else { description = "method '" + ((Method) element).getName() + "'"; } return description + " of class " + getDeclaringClass(element).getName(); } private static void processAnnotations(AnnotatedElement element, boolean processNested, List indexes, List tmp, Map> nestedReplacements, HeaderTransformer transformer, MethodFilter filter) { Parsed annotation = findAnnotation(element, Parsed.class); if (annotation != null) { TransformedHeader header = new TransformedHeader(element, transformer); if (filter == MethodFilter.ONLY_GETTERS && header.getHeaderIndex() >= 0 && indexes.contains(header.getHeaderIndex())) { //allows the same column to be mapped to multiple fields when parsing, but not when writing. throw new IllegalArgumentException("Duplicate field index '" + header.getHeaderIndex() + "' found in " + describeElement(element)); } tmp.add(header); indexes.add(header.getHeaderIndex()); } if (processNested) { Nested nested = findAnnotation(element, Nested.class); if (nested != null) { tmp.add(new TransformedHeader(element, null)); Class nestedBeanType = nested.type(); if (nestedBeanType == Object.class) { nestedBeanType = getType(element); } Class transformerType = nested.headerTransformer(); if (transformerType != HeaderTransformer.class) { String[] args = nested.args(); HeaderTransformer innerTransformer = AnnotationHelper.newInstance(HeaderTransformer.class, transformerType, args); nestedReplacements.put(element, getFieldSequence(nestedBeanType, true, indexes, innerTransformer, filter)); } else { nestedReplacements.put(element, getFieldSequence(nestedBeanType, true, indexes, transformer, filter)); } } } } /** * Returns a list of fields with {@link Parsed} annotations in the sequence they should be processed for parsing * or writing. The sequence is ordered taking into account their original order in the annotated class, unless * {@link Parsed#index()} is set to a non-negative number. * * @param beanClass the class whose field sequence will be returned. * @param processNested flag indicating whether {@link Nested} annotations should be processed * @param transformer a {@link HeaderTransformer} instance to be used for transforming headers of a given {@link Nested} attribute. * @param filter filter to apply over annotated methods when the class is being used for reading data from beans (to write values to an output) * or when writing values into beans (while parsing). It is used to choose either a "get" or a "set" * method annotated with {@link Parsed}, when both methods target the same field. * * @return a list of fields ordered by their processing sequence */ public static List getFieldSequence(Class beanClass, boolean processNested, HeaderTransformer transformer, MethodFilter filter) { List indexes = new ArrayList(); List tmp = getFieldSequence(beanClass, processNested, indexes, transformer, filter); Collections.sort(tmp, new Comparator() { @Override public int compare(TransformedHeader t1, TransformedHeader t2) { int i1 = t1.getHeaderIndex(); int i2 = t2.getHeaderIndex(); return i1 < i2 ? -1 : (i1 == i2 ? 0 : 1); } }); int col = -1; for (int i : indexes) { col++; if (i < 0) { continue; } if (i != col) { while (i >= tmp.size()) { tmp.add(null); } Collections.swap(tmp, i, col); } } return tmp; } private static List getFieldSequence(Class beanClass, boolean processNested, List indexes, HeaderTransformer transformer, MethodFilter filter) { List tmp = new ArrayList(); Map> nestedReplacements = new LinkedHashMap>(); for (Field field : getAllFields(beanClass).keySet()) { processAnnotations(field, processNested, indexes, tmp, nestedReplacements, transformer, filter); } for (Method method : getAnnotatedMethods(beanClass, filter)) { processAnnotations(method, processNested, indexes, tmp, nestedReplacements, transformer, filter); } if (!nestedReplacements.isEmpty()) { int size = tmp.size(); for (int i = size - 1; i >= 0; i--) { TransformedHeader field = tmp.get(i); List nestedFields = nestedReplacements.remove(field.getTarget()); if (nestedFields != null) { tmp.remove(i); tmp.addAll(i, nestedFields); if (nestedReplacements.isEmpty()) { break; } } } } return tmp; } /** * Returns all fields available from a given class. * * @param beanClass a class whose fields will be returned. * * @return a map of {@link Field} and the corresponding {@link PropertyWrapper} */ public static Map getAllFields(Class beanClass) { Map properties = new LinkedHashMap(); try { for (PropertyWrapper property : BeanHelper.getPropertyDescriptors(beanClass)) { String name = property.getName(); if (name != null) { properties.put(name, property); } } } catch (Exception e) { //ignore and proceed to get fields directly } Set used = new HashSet(); Class clazz = beanClass; Map out = new LinkedHashMap(); do { Field[] declared = clazz.getDeclaredFields(); for (Field field : declared) { if (used.contains(field.getName())) { continue; } used.add(field.getName()); out.put(field, properties.get(field.getName())); } clazz = clazz.getSuperclass(); } while (clazz != null && clazz != Object.class); return out; } /** * Returns all methods available from a given class that have an annotation. * * @param beanClass a class whose methods will be returned. * @param filter filter to apply over annotated methods when the class is being used for reading data from beans (to write values to an output) * or when writing values into beans (while parsing). It is used to choose either a "get" or a "set" * method annotated with {@link Parsed}, when both methods target the same field. * * @return a map of {@link Method} and the corresponding {@link PropertyWrapper} */ public static List getAnnotatedMethods(Class beanClass, MethodFilter filter) { List out = new ArrayList(); Class clazz = beanClass; do { Method[] declared = clazz.getDeclaredMethods(); outer: for (Method method : declared) { Annotation[] annotations = method.getDeclaredAnnotations(); for (Annotation annotation : annotations) { if (isCustomAnnotation(annotation)) { if (filter.reject(method)) { continue outer; } out.add(method); continue outer; } } } clazz = clazz.getSuperclass(); } while (clazz != null && clazz != Object.class); return out; } private static AnnotatedElement lastProcessedElement; private static Class lastProcessedAnnotationType; private static Annotation lastAnnotationFound; /** * Searches for an annotation of a given type that's been applied to an element either directly (as a regular annotation) * or indirectly (as a meta-annotations, i.e. an annotation that has annotations). * * @param annotatedElement the element whose annotations will be searched * @param annotationType the type of annotation to search for * @param the type of the annotation being searched for * * @return the annotation associated with the given element, or {@code null} if not found. */ public synchronized static A findAnnotation(AnnotatedElement annotatedElement, Class annotationType) { if (annotatedElement == null || annotationType == null) { return null; } if (annotatedElement.equals(lastProcessedElement) && annotationType == lastProcessedAnnotationType) { return (A) lastAnnotationFound; } lastProcessedElement = annotatedElement; lastProcessedAnnotationType = annotationType; Stack path = new Stack(); A annotation = findAnnotation(annotatedElement, annotationType, new HashSet(), path); if (annotation == null || path.isEmpty()) { lastAnnotationFound = annotation; return annotation; } while (!path.isEmpty()) { Annotation parent = path.pop(); Annotation target = path.isEmpty() ? annotation : path.peek(); for (Method method : parent.annotationType().getDeclaredMethods()) { Copy copy = method.getAnnotation(Copy.class); if (copy != null) { Class targetClass = copy.to(); String targetProperty = copy.property(); if (targetProperty.trim().isEmpty()) { targetProperty = method.getName(); } Object value = invoke(parent, method); if (targetClass == Parsed.class && targetProperty.equals("field") && value.getClass() == String.class) { value = new String[]{(String) value}; } setAnnotationValue(getTargetAnnotation(annotatedElement, targetClass, target), targetProperty, value); } } } lastAnnotationFound = annotation; return annotation; } private static Annotation getTargetAnnotation(AnnotatedElement annotatedElement, Class targetClass, Annotation current) { if (targetClass == current.annotationType()) { return current; } throw new IllegalStateException("Can't process @Copy annotation on '" + current + "' of field '" + annotatedElement + "'.\n" + "Target class '" + targetClass.getName() + "' could not be found."); } private static void setAnnotationValue(Annotation annotation, String attribute, Object newValue) { Object handler = Proxy.getInvocationHandler(annotation); try { Field field = handler.getClass().getDeclaredField("memberValues"); field.setAccessible(true); Map memberValues = (Map) field.get(handler); memberValues.put(attribute, newValue); } catch (Exception e) { throw new IllegalStateException("Can't process @Copy annotation to assign value '" + newValue + "' to attribute '" + attribute + "' of annotation " + annotation + ".", e); } } private static Object invoke(Annotation annotation, Method method) { try { return method.invoke(annotation, (Object[]) null); } catch (Exception e) { throw new IllegalStateException("Can't read value from annotation " + annotation, e); } } private static A findAnnotation(AnnotatedElement annotatedElement, Class annotationType, Set visited, Stack path) { Annotation[] declaredAnnotations = annotatedElement.getDeclaredAnnotations(); for (int i = 0; i < declaredAnnotations.length; i++) { Annotation ann = declaredAnnotations[i]; if (ann.annotationType() == annotationType) { return (A) ann; } } for (int i = 0; i < declaredAnnotations.length; i++) { Annotation ann = declaredAnnotations[i]; if (isCustomAnnotation(ann) && visited.add(ann)) { A annotation = findAnnotation(ann.annotationType(), annotationType, visited, path); path.push(ann); if (annotation != null) { return annotation; } } } return null; } private static final Set javaLangAnnotationTypes = new HashSet(); private static final Set customAnnotationTypes = new HashSet(); private static boolean isCustomAnnotation(Annotation annotation) { Class annotationType = annotation.annotationType(); if (customAnnotationTypes.contains(annotationType)) { return true; } if (javaLangAnnotationTypes.contains(annotationType)) { return false; } if (annotationType.getName().startsWith("java.lang.annotation")) { javaLangAnnotationTypes.add(annotationType); return false; } else { customAnnotationTypes.add(annotationType); return true; } } /** * Returns all annotations applied to an element, excluding the ones not in a given package. * * @param annotatedElement the element (method, field, etc) whose annotations will be extracted * @param aPackage the package of the annotations that should be returned * * @return the list of annotation elements applied to the given element, that are also members of the given package. */ public static List findAllAnnotationsInPackage(AnnotatedElement annotatedElement, Package aPackage) { final ArrayList found = new ArrayList(); findAllAnnotationsInPackage(annotatedElement, aPackage, found, new HashSet()); return found; } private static void findAllAnnotationsInPackage(AnnotatedElement annotatedElement, Package aPackage, ArrayList found, Set visited) { Annotation[] declaredAnnotations = annotatedElement.getDeclaredAnnotations(); for (int i = 0; i < declaredAnnotations.length; i++) { Annotation ann = declaredAnnotations[i]; if (aPackage.equals(ann.annotationType().getPackage())) { found.add(ann); } if (isCustomAnnotation(ann) && visited.add(ann)) { findAllAnnotationsInPackage(ann.annotationType(), aPackage, found, visited); } } } /** * Returns Java's default value for a given type, in a primitive type wrapper. * * @param type the primitive type whose default value will be returned. * * @return the default value for the given primitive type, or {@code null} if the type is not primitive. */ public static final Object getDefaultPrimitiveValue(Class type) { if (type == int.class) { return Integer.valueOf(0); } else if (type == double.class) { return 0.0D; } else if (type == boolean.class) { return Boolean.FALSE; } else if (type == long.class) { return Long.valueOf(0L); } else if (type == float.class) { return 0.0F; } else if (type == byte.class) { return Byte.valueOf((byte) 0); } else if (type == char.class) { return Character.valueOf('\0'); } else if (type == short.class) { return Short.valueOf((short) 0); } return null; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy