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

org.kiwiproject.beans.BeanConverter Maven / Gradle / Ivy

Go to download

Kiwi is a utility library. We really like Google's Guava, and also use Apache Commons. But if they don't have something we need, and we think it is useful, this is where we put it.

There is a newer version: 4.5.2
Show newest version
package org.kiwiproject.beans;

import static com.google.common.base.Preconditions.checkState;
import static java.util.Objects.isNull;
import static java.util.function.Predicate.not;
import static java.util.stream.Collectors.toUnmodifiableSet;

import com.google.common.collect.Sets;
import lombok.Getter;
import lombok.Setter;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.BeanWrapper;
import org.springframework.beans.BeanWrapperImpl;
import org.springframework.beans.NotReadablePropertyException;
import org.springframework.beans.NotWritablePropertyException;

import java.beans.PropertyDescriptor;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import java.util.function.Function;
import java.util.stream.Stream;

/**
 * Simple way to convert one bean to another.  This utility uses spring-beans to attempt the conversion at first.
 * If attempting to convert maps, it will attempt to do simple copies of the key-value pairs.
 * 

* Exclusion lists can be provided to ignore specific fields. *

* Also, custom mappers can be provided per field for more control over how the fields are converted. *

* NOTE: This class requires spring-beans as a dependency. */ @Slf4j public class BeanConverter { /** * Custom mappers that map specific field names to a function that will accept the data and convert it to the new value. */ @Getter private final Map> mappers = new HashMap<>(); /** * Set of property names to exclude from copying over. Defaults to {@code ["class", "new"] } */ @Getter @Setter private Set exclusions = Sets.newHashSet("class", "new"); /** * Allows for a failed conversion to throw an exception if {@code true}; otherwise just logs a failure if {@code false} */ @Getter @Setter private boolean failOnError; /** * This conversion method takes a single parameter and modifies the object in place. * * @param input the object to perform the conversion on * @return the same object that was passed in */ public T convert(T input) { return convert(input, input); } /** * This conversion method takes two parameters, and it copies properties from one object to another * * @param input the object to copy the properties from * @param target the object to copy the properties too (destination) * @param the type of object being returned * @return the modified target object */ public R convert(T input, R target) { if (isNull(input)) { return null; } // perform standard mapping first var inputWrapper = new BeanWrapperImpl(input); var targetWrapper = new BeanWrapperImpl(target); var propertyNames = getPropertySet(input, inputWrapper); // This cannot be a foreach because if failOnError is true, the exceptions need to bubble. for (String propName : propertyNames) { if (hasPropertyMapper(propName)) { // custom mapper var func = getPropertyMapper(propName); func.apply(input); } else if (input != target) { // only need to map simple properties if it's a different object var inputValue = readBeanValue(input, inputWrapper, propName); writeBeanValue(target, targetWrapper, propName, inputValue); } } return target; } protected Set getPropertySet(T input, BeanWrapper inputWrapper) { var propertyNames = getPropertyNamesAsSet(input, inputWrapper); // remove exclusions return propertyNames.stream() .filter(not(prop -> exclusions.contains(prop))) .collect(toUnmodifiableSet()); } @SuppressWarnings({"unchecked", "rawtypes"}) private static Set getPropertyNamesAsSet(T input, BeanWrapper inputWrapper) { if (input instanceof Map inputMap) { return inputMap.keySet(); } return Stream.of(inputWrapper.getPropertyDescriptors()) .map(PropertyDescriptor::getName) .collect(toUnmodifiableSet()); } /** * Checks to see if a property mapper exists for a given property name * * @param propertyName the property name * @return true if a property mapper was found for that property */ public boolean hasPropertyMapper(String propertyName) { return mappers.containsKey(propertyName); } /** * Get the property mapper function for a specific property name. *

* It is assumed the caller knows the correct return type for the mapper, otherwise a * {@link ClassCastException} will be thrown at runtime. * * @param propertyName the property name * @param the result type of the mapper for the given property * @return the Function that will be triggered */ @SuppressWarnings("unchecked") public Function getPropertyMapper(String propertyName) { return (Function) mappers.get(propertyName); } @SuppressWarnings("unchecked") protected Object readBeanValue(T input, BeanWrapper inputWrapper, String propName) { try { return inputWrapper.getPropertyValue(propName); } catch (NotReadablePropertyException e) { // try direct access if map if (input instanceof Map) { return ((Map) input).get(propName); } else { // if no approach works logOrFail("Exception trying to read value", propName, e); } } return null; } protected void logOrFail(String msg, String propName, RuntimeException e) { if (failOnError) { throw e; } else { LOG.debug("{} - property: {}", msg, propName); LOG.trace("Exception: ", e); } } @SuppressWarnings({"unchecked", "rawtypes"}) protected void writeBeanValue(R target, BeanWrapper targetWrapper, String propName, Object inputValue) { try { // default - grab value directly from input targetWrapper.setPropertyValue(propName, inputValue); } catch (NotWritablePropertyException e) { // try direct access if map if (target instanceof Map targetMap) { targetMap.put(propName, inputValue); } else { // if no approach works logOrFail("Exception trying to write value", propName, e); } } } /** * Adds a property mapper function for a specific property name * * @param propertyName the property name * @param function the Function that will be triggered * @throws IllegalStateException if a mapper is already registered on the given property */ public void addPropertyMapper(String propertyName, Function function) { checkState(!mappers.containsKey(propertyName), "Mapper already registered for property: %s", propertyName); mappers.put(propertyName, function); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy