com.querydsl.core.util.BeanMap Maven / Gradle / Ivy
/*
* Copyright 2001-2004 The Apache Software Foundation
*
* 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.querydsl.core.util;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.AbstractMap;
import java.util.AbstractSet;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.Function;
/**
* An implementation of Map for JavaBeans which uses introspection to get and put properties in the
* bean.
*
* If an exception occurs during attempts to get or set a property then the property is
* considered non existent in the Map
*
*
*
* @author James Strachan
* @author Matt Hall, John Watkinson, Stephen Colebourne
* @version $Revision: 1.1 $ $Date: 2005/10/11 17:05:19 $
* @since Commons Collections 1.0
*/
@SuppressWarnings("rawtypes")
public class BeanMap extends AbstractMap implements Cloneable {
private transient Object bean;
private transient Map readMethods = new HashMap<>();
private transient Map writeMethods = new HashMap<>();
private transient Map> types = new HashMap<>();
/** An empty array. Used to invoke accessors via reflection. */
private static final Object[] NULL_ARGUMENTS = {};
/**
* Maps primitive Class types to transformers. The transformer transform strings into the
* appropriate primitive wrapper.
*/
private static final Map, Function, ?>> defaultFunctions = new HashMap<>();
static {
defaultFunctions.put(
Boolean.TYPE,
new Function() {
@Override
public Object apply(Object input) {
return Boolean.valueOf(input.toString());
}
});
defaultFunctions.put(
Character.TYPE,
new Function() {
@Override
public Object apply(Object input) {
return input.toString().charAt(0);
}
});
defaultFunctions.put(
Byte.TYPE,
new Function() {
@Override
public Object apply(Object input) {
return Byte.valueOf(input.toString());
}
});
defaultFunctions.put(
Short.TYPE,
new Function() {
@Override
public Object apply(Object input) {
return Short.valueOf(input.toString());
}
});
defaultFunctions.put(
Integer.TYPE,
new Function() {
@Override
public Object apply(Object input) {
return Integer.valueOf(input.toString());
}
});
defaultFunctions.put(
Long.TYPE,
new Function() {
@Override
public Object apply(Object input) {
return Long.valueOf(input.toString());
}
});
defaultFunctions.put(
Float.TYPE,
new Function() {
@Override
public Object apply(Object input) {
return Float.valueOf(input.toString());
}
});
defaultFunctions.put(
Double.TYPE,
new Function() {
@Override
public Object apply(Object input) {
return Double.valueOf(input.toString());
}
});
}
// Constructors
// -------------------------------------------------------------------------
/** Constructs a new empty {@code BeanMap}. */
public BeanMap() {}
/**
* Constructs a new {@code BeanMap} that operates on the specified bean. If the given bean is
* {@code null}, then this map will be empty.
*
* @param bean the bean for this map to operate on
*/
public BeanMap(Object bean) {
this.bean = bean;
initialise();
}
// Map interface
// -------------------------------------------------------------------------
@Override
public String toString() {
return "BeanMap<" + bean + ">";
}
/**
* Clone this bean map using the following process:
*
*
*
*
* - If there is no underlying bean, return a cloned BeanMap without a bean.
*
*
- Since there is an underlying bean, try to instantiate a new bean of the same type using
* Class.newInstance().
*
*
- If the instantiation fails, throw a CloneNotSupportedException
*
*
- Clone the bean map and set the newly instantiated bean as the underlying bean for the
* bean map.
*
*
- Copy each property that is both readable and writable from the existing object to a
* cloned bean map.
*
*
- If anything fails along the way, throw a CloneNotSupportedException.
*
*
*/
@Override
public Object clone() throws CloneNotSupportedException {
var newMap = (BeanMap) super.clone();
if (bean == null) {
// no bean, just an empty bean map at the moment. return a newly
// cloned and empty bean map.
return newMap;
}
Object newBean = null;
Class> beanClass = null;
try {
beanClass = bean.getClass();
newBean = beanClass.getDeclaredConstructor().newInstance();
} catch (Exception e) {
// unable to instantiate
throw new CloneNotSupportedException(
"Unable to instantiate the underlying bean \"" + beanClass.getName() + "\": " + e);
}
try {
newMap.setBean(newBean);
} catch (Exception exception) {
throw new CloneNotSupportedException(
"Unable to set bean in the cloned bean map: " + exception);
}
try {
// copy only properties that are readable and writable. If its
// not readable, we can't get the value from the old map. If
// its not writable, we can't write a value into the new map.
for (String key : readMethods.keySet()) {
if (getWriteMethod(key) != null) {
newMap.put(key, get(key));
}
}
} catch (Exception exception) {
throw new CloneNotSupportedException(
"Unable to copy bean values to cloned bean map: " + exception);
}
return newMap;
}
/**
* Puts all of the writable properties from the given BeanMap into this BeanMap. Read-only and
* Write-only properties will be ignored.
*
* @param map the BeanMap whose properties to put
*/
public void putAllWriteable(BeanMap map) {
for (String key : map.readMethods.keySet()) {
if (getWriteMethod(key) != null) {
this.put(key, map.get(key));
}
}
}
/**
* This method reinitializes the bean map to have default values for the bean's properties. This
* is accomplished by constructing a new instance of the bean which the map uses as its underlying
* data source. This behavior for {@code Map#clear()} differs from the Map contract in that the
* mappings are not actually removed from the map (the mappings for a BeanMap are fixed).
*/
@Override
public void clear() {
if (bean == null) {
return;
}
Class> beanClass = null;
try {
beanClass = bean.getClass();
bean = beanClass.getDeclaredConstructor().newInstance();
} catch (Exception e) {
throw new UnsupportedOperationException(
"Could not create new instance of class: " + beanClass);
}
}
/**
* Returns true if the bean defines a property with the given name.
*
* The given name must be a {@code String}; if not, this method returns false. This method will
* also return false if the bean does not define a property with that name.
*
*
Write-only properties will not be matched as the test operates against property read
* methods.
*
* @param name the name of the property to check
* @return false if the given name is null or is not a {@code String}; false if the bean does not
* define a property with that name; or true if the bean does define a property with that name
*/
public boolean containsKey(String name) {
var method = getReadMethod(name);
return method != null;
}
/**
* Returns the value of the bean's property with the given name.
*
*
The given name must be a {@code String} and must not be null; otherwise, this method returns
* {@code null}. If the bean defines a property with the given name, the value of that property is
* returned. Otherwise, {@code null} is returned.
*
*
Write-only properties will not be matched as the test operates against property read
* methods.
*
* @param name the name of the property whose value to return
* @return the value of the property with that name
*/
public Object get(String name) {
if (bean != null) {
var method = getReadMethod(name);
if (method != null) {
try {
return method.invoke(bean, NULL_ARGUMENTS);
} catch (IllegalAccessException
| NullPointerException
| InvocationTargetException
| IllegalArgumentException e) {
}
}
}
return null;
}
/**
* Sets the bean property with the given name to the given value.
*
* @param name the name of the property to set
* @param value the value to set that property to
* @return the previous value of that property
*/
@Override
public Object put(String name, Object value) {
if (bean != null) {
var oldValue = get(name);
var method = getWriteMethod(name);
if (method == null) {
throw new IllegalArgumentException(
"The bean of type: " + bean.getClass().getName() + " has no property called: " + name);
}
try {
var arguments = createWriteMethodArguments(method, value);
method.invoke(bean, arguments);
var newValue = get(name);
firePropertyChange(name, oldValue, newValue);
} catch (InvocationTargetException | IllegalAccessException e) {
throw new IllegalArgumentException(e.getMessage());
}
return oldValue;
}
return null;
}
/**
* Returns the number of properties defined by the bean.
*
* @return the number of properties defined by the bean
*/
@Override
public int size() {
return readMethods.size();
}
/**
* Get the keys for this BeanMap.
*
*
Write-only properties are not included in the returned set of property names,
* although it is possible to set their value and to get their type.
*
* @return BeanMap keys. The Set returned by this method is not modifiable.
*/
@Override
public Set keySet() {
return readMethods.keySet();
}
/**
* Gets a Set of MapEntry objects that are the mappings for this BeanMap.
*
* Each MapEntry can be set but not removed.
*
* @return the unmodifiable set of mappings
*/
@Override
public Set> entrySet() {
return new AbstractSet<>() {
@Override
public Iterator> iterator() {
return entryIterator();
}
@Override
public int size() {
return BeanMap.this.readMethods.size();
}
};
}
/**
* Returns the values for the BeanMap.
*
* @return values for the BeanMap. The returned collection is not modifiable.
*/
@Override
public Collection