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

com.firefly.core.XmlApplicationContext Maven / Gradle / Ivy

The newest version!
package com.firefly.core;

import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import com.firefly.annotation.Inject;
import com.firefly.core.support.BeanDefinition;
import com.firefly.core.support.annotation.AnnotationBeanDefinition;
import com.firefly.core.support.annotation.AnnotationBeanReader;
import com.firefly.core.support.xml.ManagedArray;
import com.firefly.core.support.xml.ManagedList;
import com.firefly.core.support.xml.ManagedMap;
import com.firefly.core.support.xml.ManagedRef;
import com.firefly.core.support.xml.ManagedValue;
import com.firefly.core.support.xml.XmlBeanDefinition;
import com.firefly.core.support.xml.XmlBeanReader;
import com.firefly.core.support.xml.XmlManagedNode;
import com.firefly.utils.ConvertUtils;
import com.firefly.utils.ReflectUtils;
import com.firefly.utils.ReflectUtils.BeanMethodFilter;
import com.firefly.utils.VerifyUtils;
import com.firefly.utils.log.Log;
import com.firefly.utils.log.LogFactory;

/**
 * The core application context mixed XML and annotation bean management
 * 
 * @author JJ Xu & Alvin Qiu
 */
public class XmlApplicationContext extends AbstractApplicationContext {

	private static Log log = LogFactory.getInstance().getLog("firefly-system");

	public XmlApplicationContext() {
		this(null);
	}

	public XmlApplicationContext(String file) {
		super(file);
	}

	@Override
	protected List getBeanDefinitions(String file) {
		List list1 = new AnnotationBeanReader(file)
				.loadBeanDefinitions();
		List list2 = new XmlBeanReader(file)
				.loadBeanDefinitions();
		if (list1 != null && list2 != null) {
			log.debug("mixed bean");
			list1.addAll(list2);
			return list1;
		} else if (list1 != null) {
			log.debug("annotation bean");
			return list1;
		} else if (list2 != null) {
			log.debug("xml bean");
			return list2;
		}
		return null;
	}

	@Override
	protected Object inject(BeanDefinition beanDef) {
		if (beanDef instanceof XmlBeanDefinition)
			return xmlInject(beanDef);
		else if (beanDef instanceof AnnotationBeanDefinition)
			return annotationInject(beanDef);
		else
			return null;
	}

	private Object xmlInject(BeanDefinition beanDef) {
		final XmlBeanDefinition beanDefinition = (XmlBeanDefinition) beanDef;
		Class clazz = null;
		Object object = null;
		
		try {
			clazz = XmlApplicationContext.class.getClassLoader().loadClass(beanDefinition.getClassName());
			if(beanDefinition.getContructorParameters().size() <= 0) {
				object = clazz.newInstance();
			} else {
				List constructorParameters = new ArrayList();
				for (int i = 0; i < beanDefinition.getContructorParameters().size(); i++) {
					Object p = getInjectArg(beanDefinition.getContructorParameters().get(i), beanDefinition.getConstructor().getParameterTypes()[i]);
					constructorParameters.add(p);
				}
//				System.out.println(constructorParameters.toString());
				object = beanDefinition.getConstructor().newInstance(constructorParameters.toArray());
			}
		} catch (Throwable t) {
			log.error("object initiate error", t);
		}
		
		final Object instance = object;
		ReflectUtils.getSetterMethods(clazz, new BeanMethodFilter(){

			@Override
			public boolean accept(String propertyName, Method method) {
				XmlManagedNode value = beanDefinition.getProperties().get(propertyName);
				if (value != null) {
					try {
						method.invoke(instance, getInjectArg(value, method.getParameterTypes()[0]));
					} catch (Throwable t) {
						log.error("xml inject error", t);
					}
				}
				return false;
			}});

		addObjectToContext(beanDefinition, object);
		return instance;
	}

	@SuppressWarnings("unchecked")
	private Object getInjectArg(XmlManagedNode value, Class parameterType) {
		if (value instanceof ManagedValue) {
			ManagedValue managedValue = (ManagedValue)value;
			String typeName = null;
			if (parameterType == null) {
				typeName = VerifyUtils.isEmpty(managedValue.getTypeName()) ? null
						: managedValue.getTypeName();
			} else {
				typeName = VerifyUtils.isEmpty(managedValue.getTypeName()) ? parameterType.getName() : managedValue.getTypeName();
			}
			log.debug("value type [{}]", typeName);
			return getValueArg(managedValue, typeName);
		} else if (value instanceof ManagedRef) {
			return getRefArg((ManagedRef)value);
		} else if (value instanceof ManagedList) {
			return getListArg((ManagedList)value, parameterType);
		} else if (value instanceof ManagedArray) {
			return getArrayArg((ManagedArray)value, parameterType);
		} else if (value instanceof ManagedMap) {
			return getMapArg((ManagedMap)value, parameterType);
		} else
			return null;
	}

	private Object getValueArg(ManagedValue managedValue, String typeName) {
		return ConvertUtils.convert(managedValue.getValue(), typeName);
	}

	private Object getRefArg(ManagedRef ref) {
		Object instance = map.get(ref.getBeanName());
		if (instance == null) {
			BeanDefinition b = findBeanDefinition(ref.getBeanName());
			if (b != null)
				instance = inject(b);
		}
		return instance;
	}

	@SuppressWarnings("unchecked")
	private Object getListArg(ManagedList values, Class setterParamType) {
		Collection collection = null;
		
		if (VerifyUtils.isNotEmpty(values.getTypeName())) {
			try {
				collection = (Collection) XmlApplicationContext.class
						.getClassLoader().loadClass(values.getTypeName())
						.newInstance();
			} catch (Throwable t) {
				log.error("list inject error", t);
			}
		} else {
			collection = (setterParamType == null ? new ArrayList()
					: ConvertUtils.getCollectionObj(setterParamType));
		}

		for (XmlManagedNode item : values) {
			Object listValue = getInjectArg(item, null);
			collection.add(listValue);
		}
		return collection;
	}

	private Object getArrayArg(ManagedArray values, Class setterParamType) {
		Collection collection = new ArrayList();
		for (XmlManagedNode item : values) {
			Object listValue = getInjectArg(item, null);
			collection.add(listValue);
		}
		return ConvertUtils.convert(collection, setterParamType);
	}


	@SuppressWarnings("unchecked")
	private Object getMapArg(ManagedMap values, Class setterParamType) {
		Map m = null;
		if (VerifyUtils.isNotEmpty(values.getTypeName())) {
			try {
				m = (Map) XmlApplicationContext.class.getClassLoader()
						.loadClass(values.getTypeName())
						.newInstance();
			} catch (Throwable t) {
				log.error("map inject error", t);
			}
		} else { 
			m = (setterParamType == null ? new HashMap() : ConvertUtils.getMapObj(setterParamType));
			log.debug("map ret [{}]", m.getClass().getName());
		}
		for (XmlManagedNode o : values.keySet()) {
			Object k = getInjectArg(o, null);
			Object v = getInjectArg(values.get(o), null);
			m.put(k, v);
		}
		return m;
	}

	/**
	 * annotation injecting
	 * 
	 * @param beanDef
	 * @return
	 */
	private Object annotationInject(BeanDefinition beanDef) {
		AnnotationBeanDefinition beanDefinition = (AnnotationBeanDefinition) beanDef;
		// constructor injecting
		Object object = constructorInject(beanDefinition);
		beanDefinition.setInjectedInstance(object);
		fieldInject(beanDefinition, object);
		methodInject(beanDefinition, object);
		addObjectToContext(beanDefinition, object);
		return object;
	}
	
	private Object constructorInject(AnnotationBeanDefinition beanDefinition) {
		Class[] params = beanDefinition.getConstructor().getParameterTypes();
		Object[] p = new Object[params.length];
		for (int i = 0; i < p.length; i++) {
			String key = params[i].getName();
			Object instance = map.get(key);
			if (instance != null) {
				p[i] = instance;
			} else {
				BeanDefinition b = findBeanDefinition(key);
				if (b != null)
					p[i] = inject(b);
			}
		}
		Object instance = null;
		try {
			instance = beanDefinition.getConstructor().newInstance(p);
		} catch (Throwable t) {
			log.error("constructor injecting error", t);
		}
		return instance;
	}

	private void fieldInject(AnnotationBeanDefinition beanDefinition, final Object object) {
		for (Field field : beanDefinition.getInjectFields()) {
			field.setAccessible(true);
			Class clazz = field.getType();
			String id = field.getAnnotation(Inject.class).value();
			String key = VerifyUtils.isNotEmpty(id) ? id : clazz.getName();
			Object instance = map.get(key);
			if (instance == null) {
				BeanDefinition b = findBeanDefinition(key);
				if (b != null)
					instance = inject(b);
			}
			if (instance != null) {
				try {
					field.set(object, instance);
				} catch (Throwable t) {
					log.error("field injecting error", t);
				}
			}
		}
	}

	private void methodInject(AnnotationBeanDefinition beanDefinition, final Object object) {
		for (Method method : beanDefinition.getInjectMethods()) {
			method.setAccessible(true);
			Class[] params = method.getParameterTypes();
			Object[] p = new Object[params.length];
			for (int i = 0; i < p.length; i++) {
				String key = params[i].getName();
				Object instance = map.get(key);
				if (instance != null) {
					p[i] = instance;
				} else {
					BeanDefinition b = findBeanDefinition(key);
					if (b != null)
						p[i] = inject(b);
				}
			}
			try {
				method.invoke(object, p);
			} catch (Throwable t) {
				log.error("method injecting error", t);
			}
		}
	}
}