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

com.univocity.parsers.conversions.EnumConversion Maven / Gradle / Ivy

/*******************************************************************************
 * Copyright 2015 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.conversions;

import com.univocity.parsers.common.*;

import java.lang.reflect.*;
import java.util.*;

/**
 * Converts Strings to enumeration constants and vice versa.
 *
 * 

This class supports multiple types of identification of enumeration constants. For example, you can match the literal ({@link Enum#name()} the ordinal ({@link Enum#ordinal()} or * the result of a method defined in your enumeration. * *

The reverse conversion from an enumeration to String (in {@link EnumConversion#revert(Enum)} will return a String using the first {@link EnumSelector} provided in the constructor of this class. * * @param the enumeration type whose constants will be converted from/to {@code String} * @see ObjectConversion * @see EnumSelector * * @author uniVocity Software Pty Ltd - [email protected] * */ public class EnumConversion> extends ObjectConversion { private final Class enumType; private final Field customEnumField; private final Method customEnumMethod; private final EnumSelector[] selectors; private final Map[] conversions; /** * Defines a conversion for an enumeration type that will attempt to match Strings against * the results of the output produced by ({@link Enum#name()}, ({@link Enum#ordinal()} and ({@link Enum#toString()} * of each constant of the given enumeration (@link {@link Class#getEnumConstants()}). * * @param enumType the enumeration type to be converted from/to {@code String} */ public EnumConversion(Class enumType) { this(enumType, EnumSelector.NAME, EnumSelector.ORDINAL, EnumSelector.STRING); } /** * Defines a conversion for an enumeration type that will attempt to match Strings the list of {@link EnumSelector}s, in the specified order. * Each {@link EnumSelector} identifies which element of each constant of the enumeration class (@link {@link Class#getEnumConstants()} * should be used to match equivalent {@code String}s. * * @param enumType the enumeration type to be converted from/to {@code String} * @param selectors the selection elements of the enumeration to use for matching {@code String}s. */ public EnumConversion(Class enumType, EnumSelector... selectors) { this(enumType, null, null, null, selectors); } /** * Defines a conversion for an enumeration type that will attempt to match Strings the list of {@link EnumSelector}s, in the specified order. * Each {@link EnumSelector} identifies which element of each constant of the enumeration class (@link {@link Class#getEnumConstants()} * should be used to match equivalent {@code String}s. * * @param enumType the enumeration type to be converted from/to {@code String} * @param customEnumElement name of custom element of the enumeration (attribute or method) whose values should be used to match equivalent {@code String}s. * @param selectors the selection elements of the enumeration to use for matching {@code String}s. */ public EnumConversion(Class enumType, String customEnumElement, EnumSelector... selectors) { this(enumType, null, null, customEnumElement); } /** * Defines a conversion for an enumeration type that will attempt to match Strings the list of {@link EnumSelector}s, in the specified order. * Each {@link EnumSelector} identifies which element of each constant of the enumeration class (@link {@link Class#getEnumConstants()} * should be used to match equivalent {@code String}s. * * @param enumType the enumeration type to be converted from/to {@code String} * @param valueIfStringIsNull the default enumeration constant to use if the input {@code String} is {@code null} * @param valueIfEnumIsNull the default {@code String} value to use if the input {@code enum} constant is {@code null} * @param customEnumElement name of custom element of the enumeration (attribute or method) whose values should be used to match equivalent {@code String}s. * @param selectors the selection elements of the enumeration to use for matching {@code String}s. */ @SuppressWarnings("unchecked") public EnumConversion(Class enumType, T valueIfStringIsNull, String valueIfEnumIsNull, String customEnumElement, EnumSelector... selectors) { super(valueIfStringIsNull, valueIfEnumIsNull); this.enumType = enumType; if (customEnumElement != null) { customEnumElement = customEnumElement.trim(); if (customEnumElement.isEmpty()) { customEnumElement = null; } } LinkedHashSet selectorSet = new LinkedHashSet(); Collections.addAll(selectorSet, selectors); if ((selectorSet.contains(EnumSelector.CUSTOM_FIELD) || selectorSet.contains(EnumSelector.CUSTOM_METHOD)) && customEnumElement == null) { throw new IllegalArgumentException("Cannot create custom enum conversion without a field name to use"); } Field field = null; Method method = null; if (customEnumElement != null) { IllegalStateException fieldError = null; IllegalStateException methodError = null; try { field = enumType.getDeclaredField(customEnumElement); if (!field.isAccessible()) { field.setAccessible(true); } } catch (Throwable e) { fieldError = new IllegalStateException("Unable to access custom field '" + customEnumElement + "' in enumeration type " + enumType.getName(), e); } if (field == null) { try { method = enumType.getDeclaredMethod(customEnumElement); if (!method.isAccessible()) { method.setAccessible(true); } } catch (Throwable e) { methodError = new IllegalStateException("Unable to access custom method '" + customEnumElement + "' in enumeration type " + enumType.getName(), e); } if (method != null) { if (method.getReturnType() == void.class) { throw new IllegalArgumentException("Custom method '" + customEnumElement + "' in enumeration type " + enumType.getName() + " must return a value"); } if (!selectorSet.contains(EnumSelector.CUSTOM_METHOD)) { selectorSet.add(EnumSelector.CUSTOM_METHOD); } } } else if (!selectorSet.contains(EnumSelector.CUSTOM_FIELD)) { selectorSet.add(EnumSelector.CUSTOM_FIELD); } if (selectorSet.contains(EnumSelector.CUSTOM_FIELD) && fieldError != null) { throw fieldError; } if (selectorSet.contains(EnumSelector.CUSTOM_METHOD) && methodError != null) { throw methodError; } if (field == null && method == null) { throw new IllegalStateException("No method/field named '" + customEnumElement + "' found in enumeration type " + enumType.getName()); } } if (selectorSet.contains(EnumSelector.CUSTOM_FIELD) && selectorSet.contains(EnumSelector.CUSTOM_METHOD)) { throw new IllegalArgumentException("Cannot create custom enum conversion using both method and field values"); } if (selectorSet.isEmpty()) { throw new IllegalArgumentException("Selection of enum conversion types cannot be empty."); } this.customEnumField = field; this.customEnumMethod = method; this.selectors = selectorSet.toArray(new EnumSelector[selectorSet.size()]); this.conversions = new Map[selectorSet.size()]; initializeMappings(selectorSet); } private void initializeMappings(Set conversionTypes) { T[] constants = enumType.getEnumConstants(); int i = 0; for (EnumSelector conversionType : conversionTypes) { Map map = new HashMap(constants.length); conversions[i++] = map; for (T constant : constants) { String key = getKey(constant, conversionType); if (map.containsKey(key)) { throw new IllegalArgumentException("Enumeration element type " + conversionType + " does not uniquely identify elements of " + enumType.getName() + ". Got duplicate value '" + key + "' from constants '" + constant + "' and '" + map.get(key) + "'."); } map.put(key, constant); } } } private String getKey(T constant, EnumSelector conversionType) { switch (conversionType) { case NAME: return constant.name(); case ORDINAL: return String.valueOf(constant.ordinal()); case STRING: return constant.toString(); case CUSTOM_FIELD: try { return String.valueOf(customEnumField.get(constant)); } catch (Throwable e) { throw new IllegalStateException("Error reading custom field '" + customEnumField.getName() + "' from enumeration constant '" + constant + "' of type " + enumType.getName(), e); } case CUSTOM_METHOD: try { return String.valueOf(customEnumMethod.invoke(constant)); } catch (Throwable e) { throw new IllegalStateException("Error reading custom method '" + customEnumMethod.getName() + "' from enumeration constant '" + constant + "' of type " + enumType.getName(), e); } default: throw new IllegalStateException("Unsupported enumeration selector type " + conversionType); } } @Override public String revert(T input) { if (input == null) { return super.revert(null); } return getKey(input, selectors[0]); } @Override protected T fromString(String input) { for (Map conversion : conversions) { T value = conversion.get(input); if (value != null) { return value; } } DataProcessingException exception = new DataProcessingException("Cannot convert '{value}' to enumeration of type " + enumType.getName()); exception.setValue(input); exception.markAsNonFatal(); throw exception; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy