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

com.esotericsoftware.yamlbeans.Beans Maven / Gradle / Ivy

There is a newer version: 2.0.32
Show newest version
/*
 * Copyright (c) 2008 Nathan Sweet
 * 
 * Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation
 * files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy,
 * modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software
 * is furnished to do so, subject to the following conditions:
 * 
 * The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
 * 
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
 * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
 * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR
 * IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */

package com.esotericsoftware.yamlbeans;

import java.lang.annotation.Annotation;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.lang.reflect.WildcardType;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.TreeSet;

import com.esotericsoftware.yamlbeans.YamlConfig.ConstructorParameters;

/** Utility for dealing with beans and public fields.
 * @author Nathan Sweet */
class Beans {
	private Beans () {
	}

	static public boolean isScalar (Class c) {
		return c.isPrimitive() || c == String.class || c == Integer.class || c == Boolean.class || c == Float.class
			|| c == Long.class || c == Double.class || c == Short.class || c == Byte.class || c == Character.class;
	}

	static public DeferredConstruction getDeferredConstruction (Class type, YamlConfig config) {
		ConstructorParameters parameters = config.readConfig.constructorParameters.get(type);
		if (parameters != null) return new DeferredConstruction(parameters.constructor, parameters.parameterNames);
		try {
			Class constructorProperties = Class.forName("java.beans.ConstructorProperties");
			for (Constructor typeConstructor : type.getConstructors()) {
				Annotation annotation = typeConstructor.getAnnotation(constructorProperties);
				if (annotation == null) continue;
				String[] parameterNames = (String[])constructorProperties.getMethod("value").invoke(annotation, (Object[])null);
				return new DeferredConstruction(typeConstructor, parameterNames);
			}
		} catch (Exception ignored) {
		}
		return null;
	}

	static public Object createObject (Class type, boolean privateConstructors) throws InvocationTargetException {
		// Use no-arg constructor.
		Constructor constructor = null;
		for (Constructor typeConstructor : type.getConstructors()) {
			if (typeConstructor.getParameterTypes().length == 0) {
				constructor = typeConstructor;
				break;
			}
		}

		if (constructor == null && privateConstructors) {
			// Try a private constructor.
			try {
				constructor = type.getDeclaredConstructor();
				constructor.setAccessible(true);
			} catch (SecurityException ignored) {
			} catch (NoSuchMethodException ignored) {
			}
		}

		// Otherwise try to use a common implementation.
		if (constructor == null) {
			try {
				if (List.class.isAssignableFrom(type)) {
					constructor = ArrayList.class.getConstructor(new Class[0]);
				} else if (Set.class.isAssignableFrom(type)) {
					constructor = HashSet.class.getConstructor(new Class[0]);
				} else if (Map.class.isAssignableFrom(type)) {
					constructor = HashMap.class.getConstructor(new Class[0]);
				}
			} catch (Exception ex) {
				throw new InvocationTargetException(ex, "Error getting constructor for class: " + type.getName());
			}
		}

		if (constructor == null)
			throw new InvocationTargetException(null, "Unable to find a no-arg constructor for class: " + type.getName());

		try {
			return constructor.newInstance();
		} catch (Exception ex) {
			throw new InvocationTargetException(ex, "Error constructing instance of class: " + type.getName());
		}
	}

	static public Set getProperties (Class type, boolean beanProperties, boolean privateFields, YamlConfig config) {
		if (type == null) throw new IllegalArgumentException("type cannot be null.");
		Set properties = config.writeConfig.keepBeanPropertyOrder ? new LinkedHashSet() : new TreeSet();
		for (Field field : getAllFields(type)) {
			Property property = getProperty(type, beanProperties, privateFields, config, field);
			if (property != null) {
				properties.add(property);
			}
		}
		return properties;
	}

	static private String toJavaIdentifier (String name) {
		StringBuilder buffer = new StringBuilder();
		for (int i = 0, n = name.length(); i < n; i++) {
			char c = name.charAt(i);
			if (Character.isJavaIdentifierPart(c)) buffer.append(c);
		}
		return buffer.toString();
	}

	static public Property getProperty (Class type, String name, boolean beanProperties, boolean privateFields,
		YamlConfig config) {
		if (type == null) throw new IllegalArgumentException("type cannot be null.");
		if (name == null || name.length() == 0) throw new IllegalArgumentException("name cannot be null or empty.");
		name = toJavaIdentifier(name);

		for (Field field : getAllFields(type))
			if (field.getName().equals(name)) return getProperty(type, beanProperties, privateFields, config, field);

		// If a property is not found, try prepending `_` for reserved field names (eg "class" in the data can be a "_class" field).
		if (!name.startsWith("_")) return getProperty(type, "_" + name, beanProperties, privateFields, config);

		return null;
	}

	private static Property getProperty(Class type, boolean beanProperties, boolean privateFields, YamlConfig config,
			Field field) {
		Property property = null;
		if (beanProperties) {
			String name = field.getName();
			DeferredConstruction deferredConstruction = getDeferredConstruction(type, config);
			boolean constructorProperty = deferredConstruction != null && deferredConstruction.hasParameter(name);

			String nameUpper = Character.toUpperCase(name.charAt(0)) + name.substring(1);
			String setMethodName = "set" + nameUpper;
			String getMethodName = "get" + nameUpper;
			if (field.getType().equals(Boolean.class) || field.getType().equals(boolean.class)) {
				setMethodName = name.startsWith("is")
						? "set" + Character.toUpperCase(name.charAt(2)) + name.substring(3)
						: setMethodName;
				getMethodName = name.startsWith("is") ? name : "is" + nameUpper;
			}

			Method getMethod = null;
			Method setMethod = null;
			try {
				setMethod = type.getMethod(setMethodName, field.getType());
			} catch (Exception ignored) {
			}
			try {
				getMethod = type.getMethod(getMethodName);
			} catch (Exception ignored) {
			}

			// field.getType() is boolean, but getMethodName is getFieldName(E.g: getBool).
			if (getMethod == null && (field.getType().equals(Boolean.class) || field.getType().equals(boolean.class))) {
				try {
					getMethod = type.getMethod("get" + nameUpper);
				} catch (Exception ignored) {
				}
			}
			if (getMethod != null && (setMethod != null || constructorProperty)) {
				return new MethodProperty(name, setMethod, getMethod);
			}
		}

		int modifiers = field.getModifiers();
		if (!Modifier.isStatic(modifiers) && !Modifier.isTransient(modifiers)
				&& (Modifier.isPublic(modifiers) || privateFields)) {
			field.setAccessible(true);
			property = new FieldProperty(field);
		}
		return property;
	}

	static private ArrayList getAllFields (Class type) {
		ArrayList classes = new ArrayList();
		Class nextClass = type;
		while (nextClass != null && nextClass != Object.class) {
			classes.add(nextClass);
			nextClass = nextClass.getSuperclass();
		}
		ArrayList allFields = new ArrayList();
		for (int i = classes.size() - 1; i >= 0; i--) {
			Collections.addAll(allFields, classes.get(i).getDeclaredFields());
		}
		return allFields;
	}

	static public class MethodProperty extends Property {
		private final Method setMethod, getMethod;

		public MethodProperty (String name, Method setMethod, Method getMethod) {
			super(getMethod.getDeclaringClass(), name, getMethod.getReturnType(), getMethod.getGenericReturnType());
			this.setMethod = setMethod;
			this.getMethod = getMethod;
		}

		public void set (Object object, Object value) throws Exception {
			if (object instanceof DeferredConstruction) {
				((DeferredConstruction)object).storeProperty(this, value);
				return;
			}
			setMethod.invoke(object, value);
		}

		public Object get (Object object) throws Exception {
			return getMethod.invoke(object);
		}
	}

	static public class FieldProperty extends Property {
		private final Field field;

		public FieldProperty (Field field) {
			super(field.getDeclaringClass(), field.getName(), field.getType(), field.getGenericType());
			this.field = field;
		}

		public void set (Object object, Object value) throws Exception {
			if (object instanceof DeferredConstruction) {
				((DeferredConstruction)object).storeProperty(this, value);
				return;
			}
			field.set(object, value);
		}

		public Object get (Object object) throws Exception {
			return field.get(object);
		}
	}

	static public abstract class Property implements Comparable {
		private final Class declaringClass;
		private final String name;
		private final Class type;
		private final Class elementType;

		Property (Class declaringClass, String name, Class type, Type genericType) {
			this.declaringClass = declaringClass;
			this.name = name;
			this.type = type;
			this.elementType = getElementTypeFromGenerics(genericType);
		}

		private Class getElementTypeFromGenerics (Type type) {
			if (type instanceof ParameterizedType) {
				ParameterizedType parameterizedType = (ParameterizedType)type;
				Type rawType = parameterizedType.getRawType();

				if (isCollection(rawType) || isMap(rawType)) {
					Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
					if (actualTypeArguments.length > 0) {
						final Type cType = actualTypeArguments[actualTypeArguments.length - 1];
						if (cType instanceof Class) {
							return (Class)cType;
						} else if (cType instanceof WildcardType) {
							WildcardType t = (WildcardType)cType;
							final Type bound = t.getUpperBounds()[0];
							return bound instanceof Class ? (Class)bound : null;
						} else if (cType instanceof ParameterizedType) {
							ParameterizedType t = (ParameterizedType)cType;
							final Type rt = t.getRawType();
							return rt instanceof Class ? (Class)rt : null;
						}
					}
				}
			}
			return null;
		}

		private boolean isMap (Type type) {
			return Map.class.isAssignableFrom((Class)type);
		}

		private boolean isCollection (Type type) {
			return Collection.class.isAssignableFrom((Class)type);
		}

		public int hashCode () {
			final int prime = 31;
			int result = 1;
			result = prime * result + ((declaringClass == null) ? 0 : declaringClass.hashCode());
			result = prime * result + ((name == null) ? 0 : name.hashCode());
			result = prime * result + ((type == null) ? 0 : type.hashCode());
			result = prime * result + ((elementType == null) ? 0 : elementType.hashCode());
			return result;
		}

		public boolean equals (Object obj) {
			if (this == obj) return true;
			if (obj == null) return false;
			if (getClass() != obj.getClass()) return false;
			Property other = (Property)obj;
			if (declaringClass == null) {
				if (other.declaringClass != null) return false;
			} else if (!declaringClass.equals(other.declaringClass)) return false;
			if (name == null) {
				if (other.name != null) return false;
			} else if (!name.equals(other.name)) return false;
			if (type == null) {
				if (other.type != null) return false;
			} else if (!type.equals(other.type)) return false;
			if (elementType == null) {
				if (other.elementType != null) return false;
			} else if (!elementType.equals(other.elementType)) return false;
			return true;
		}

		public Class getDeclaringClass () {
			return declaringClass;
		}

		public Class getElementType () {
			return elementType;
		}

		public Class getType () {
			return type;
		}

		public String getName () {
			return name;
		}

		public String toString () {
			return name;
		}

		public int compareTo (Property o) {
			int comparison = name.compareTo(o.name);
			if (comparison != 0) {
				// Sort id and name above all other fields.
				if (name.equals("id")) return -1;
				if (o.name.equals("id")) return 1;
				if (name.equals("name")) return -1;
				if (o.name.equals("name")) return 1;
			}
			return comparison;
		}

		abstract public void set (Object object, Object value) throws Exception;

		abstract public Object get (Object object) throws Exception;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy