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

com.tuyang.beanutils.internal.dump.BeanCopyDump Maven / Gradle / Ivy

Go to download

BeanUtils library is a Java bean copy utility with powerful functionality and high performance.

The newest version!
package com.tuyang.beanutils.internal.dump;

import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;

import com.tuyang.beanutils.annotation.BeanCopySource;
import com.tuyang.beanutils.annotation.CopyFeature;
import com.tuyang.beanutils.config.BeanCopyConfig;
import com.tuyang.beanutils.config.BeanCopyConfig.DumpOption;
import com.tuyang.beanutils.internal.cache.BeanCopyCache;
import com.tuyang.beanutils.internal.cache.BeanCopyPropertyItem;
import com.tuyang.beanutils.internal.logger.Logger;
import com.tuyang.beanutils.internal.utils.PropertyUtils;

public class BeanCopyDump {
	
	private static Logger logger = Logger.getLogger(BeanCopyDump.class);
	
	private static final int LineCharCount = 93;
	private static ThreadLocal localDumpLevel = new ThreadLocal<>();
	private static ThreadLocal> localDumpStack = new ThreadLocal<>();
	
	public static void beginDump() {
		
		if( BeanCopyConfig.instance().getDumpOption() == DumpOption.AutoDumpNone )
			return;
		
		Integer dumpLevel = localDumpLevel.get();
		if( dumpLevel == null ) {
			dumpLevel = -1;
		}
		if(dumpLevel == -1 ) {
			localDumpStack.set(new ArrayList());
		}
		localDumpLevel.set(++dumpLevel);
	}
	
	public static void endDump() {
		if( BeanCopyConfig.instance().getDumpOption() == DumpOption.AutoDumpNone )
			return;

		Integer dumpLevel = localDumpLevel.get();
		localDumpLevel.set(--dumpLevel);
		if(dumpLevel == -1 && localDumpStack.get().size() > 0 ) {
			logger.info("=============================================================================================");
		}
	}
	
	public static void dumpPropertyMapping(Class sourceClass, Class targetClass , Class optionClass) {
		dumpPropertyMapping(sourceClass, targetClass, optionClass, null);
	}
	
	public static void dumpPropertyMapping(Class sourceClass, Class targetClass , Class optionClass, List itemList) {
		
		List stackCacheKeyList = localDumpStack.get();
		if( stackCacheKeyList == null ) {
			stackCacheKeyList = new ArrayList<>();
			localDumpStack.set(stackCacheKeyList);
		}

		long cacheKey = (((long)sourceClass.hashCode() ) << 16 )+ (long) targetClass.hashCode();
		if( optionClass != null ) {
			cacheKey = (cacheKey <<16)  + (long)optionClass.hashCode();
		} else {
			cacheKey = (cacheKey <<16)  + (long)targetClass.hashCode();
		}
		
		if( stackCacheKeyList.contains(cacheKey) ) {
			return;
		}
		if( stackCacheKeyList.isEmpty() ) {
			logger.info("=============================================================================================");
			logger.info("Dump Bean Copy Property Mapping:");
		}
		
		Integer dumpLevel = localDumpLevel.get();
		if( dumpLevel == null ) {
			dumpLevel = -1;
		}
		if(dumpLevel == -1 ) {
			localDumpStack.set(new ArrayList());
		}
		
		localDumpLevel.set(++dumpLevel);
		
		dumpPropertyMappingInternal(sourceClass, targetClass, optionClass, itemList);
		
		localDumpLevel.set(--dumpLevel);
		
		if(dumpLevel == -1 && localDumpStack.get().size() > 0 ) {
			logger.info("=============================================================================================");
		}
	}
	
	private static void dumpPropertyMappingInternal(Class sourceClass, Class targetClass , Class optionClass, List itemList ) {
		if( optionClass == null )
			optionClass = targetClass;
		
		List stackCacheKeyList = localDumpStack.get();
		long cacheKey = (((long)sourceClass.hashCode() ) << 16 )+ (long) targetClass.hashCode();
		if( optionClass != null ) {
			cacheKey = (cacheKey <<16)  + (long)optionClass.hashCode();
		} else {
			cacheKey = (cacheKey <<16)  + (long)targetClass.hashCode();
		}
		if( stackCacheKeyList.contains(cacheKey) ) {
			return;
		}
		stackCacheKeyList.add(cacheKey);
		
		logger.info("---------------------------------------------------------------------------------------------");
		logger.info("From: [" + sourceClass.getSimpleName() + "] To: [" + targetClass.getSimpleName() + "] Option: [" + optionClass.getSimpleName() + "]");
		logger.info("---------------------------------------------------------------------------------------------");
		CopyFeature[] features = null;
		if( optionClass != null && optionClass.isAnnotationPresent(BeanCopySource.class)) {
			BeanCopySource copyAnnotation = optionClass.getAnnotation(BeanCopySource.class);
			features = copyAnnotation.features();
		} else if( targetClass.isAnnotationPresent(BeanCopySource.class)) {
			BeanCopySource copyAnnotation = targetClass.getAnnotation(BeanCopySource.class);
			features = copyAnnotation.features();
		}
		if( features == null || features.length == 0) {
			logger.info("CopyFeature: (NONE)");
		} else {
			logger.info("CopyFeature:");
			for( CopyFeature f : features) {
				logger.info("    " + f.toString());
			}
		}
		logger.info("---------------------------------------------------------------------------------------------");

		if( itemList == null )
			itemList = BeanCopyCache.buildBeanCopyPropertyItem(sourceClass, targetClass, optionClass);
		
		PropertyDescriptor[] sourcePds = PropertyUtils.getPropertyDescriptors(sourceClass);
		PropertyDescriptor[] targetPds = PropertyUtils.getPropertyDescriptors(targetClass);
		List ignoredSourcePds = new ArrayList<>();
		List ignoredTargetPds = new ArrayList<>();
		for( PropertyDescriptor pDescriptor : sourcePds ) {
			ignoredSourcePds.add(pDescriptor);
		}
		for( PropertyDescriptor pDescriptor : targetPds ) {
			ignoredTargetPds.add(pDescriptor);
		}
		
		for( BeanCopyPropertyItem item : itemList ) {
			
			Method readMethod = item.readMethods[0];
			Method writeMethod = item.writeMethod;
			
			PropertyDescriptor readPd  = findPropertyDescriptor( sourcePds, readMethod, true);
			PropertyDescriptor writePd = findPropertyDescriptor( targetPds, writeMethod, false);
			
			ignoredSourcePds.remove(readPd);
			ignoredTargetPds.remove(writePd);

			if( readMethod.getDeclaringClass().equals(Object.class)) {
				continue;
			}
			Class readType = item.readMethods[item.readMethods.length-1].getReturnType() ;
			Class writeType = writeMethod.getParameterTypes()[0];
			
			StringBuilder sBuilder = new StringBuilder();
			
			sBuilder.append(readType.getSimpleName()).append(" ");
			
			for( int i =0; i< item.readMethods.length; i++ ) {
				Method method = item.readMethods[i];
				if( i != 0 )
					sBuilder.append(".");
				sBuilder.append(method.getName()).append("()");
			}
			
			for( int i = sBuilder.length(); i < LineCharCount/2; i++ ) {
				sBuilder.insert(0, ' ');
			}
			sBuilder.append("|");
			sBuilder.append(writeType.getSimpleName()).append(" ");
			
			sBuilder.append(item.writeMethod.getName()).append("()");
			logger.info( sBuilder.toString() );
		}
		
		if( !ignoredSourcePds.isEmpty() ) {
			for( PropertyDescriptor pd : ignoredSourcePds ) {
				Method method = pd.getReadMethod() != null ? pd.getReadMethod(): pd.getWriteMethod();
				
				Class belongClass = method.getDeclaringClass();
				if( belongClass.equals(Object.class))
					continue;
				
				method = pd.getReadMethod();
				
				if( method != null ) {
					Class readType = method.getReturnType() ;
					StringBuilder sBuilder = new StringBuilder();
					sBuilder.append(readType.getSimpleName()).append(" ");
					sBuilder.append(method.getName()).append("()");
					for( int i = sBuilder.length(); i < LineCharCount/2; i++ ) {
						sBuilder.insert(0, ' ');
					}
					sBuilder.append("|(ignored)");
					logger.info( sBuilder.toString() );
				} else {

					StringBuilder sBuilder = new StringBuilder();
					sBuilder.append("(No get method for ").append(pd.getName()).append(")");
					for( int i = sBuilder.length(); i < LineCharCount/2; i++ ) {
						sBuilder.insert(0, ' ');
					}
					sBuilder.append("|(ignored)");
					logger.info( sBuilder.toString() );
				}
				
			}
		}
		
		if( !ignoredTargetPds.isEmpty() ) {
			for( PropertyDescriptor pd : ignoredTargetPds ) {
				Method method = pd.getWriteMethod() != null ? pd.getWriteMethod() : pd.getReadMethod();
				Class belongClass = method.getDeclaringClass();
				if( belongClass.equals(Object.class))
					continue;
				method = pd.getWriteMethod();
				if( method != null ) {
					Class writeType = method.getParameterTypes()[0];
					StringBuilder sBuilder = new StringBuilder();
					sBuilder.append("(ignored)");
					for( int i = sBuilder.length(); i < LineCharCount/2; i++ ) {
						sBuilder.insert(0, ' ');
					}
					sBuilder.append("|");
					sBuilder.append(writeType.getSimpleName()).append(" ");
					sBuilder.append( method.getName() ).append("()");
					logger.info( sBuilder.toString() );
				} else {
					StringBuilder sBuilder = new StringBuilder();
					sBuilder.append("(ignored)");
					for( int i = sBuilder.length(); i < LineCharCount/2; i++ ) {
						sBuilder.insert(0, ' ');
					}
					sBuilder.append("|");
					sBuilder.append("(No set method for "+ pd.getName() +")");
					logger.info( sBuilder.toString() );
				}
				
			}
		}
		
		for( BeanCopyPropertyItem item : itemList ) {
			if( item.useBeanCopy || item.optionClass != null ) {
				Class sourceProertyType = item.readMethods[item.readMethods.length-1].getReturnType();
				Class targetPropertyType = item.writeMethod.getParameterTypes()[0];
				if( targetPropertyType.isArray() || PropertyUtils.isInterfaceType(targetPropertyType, Collection.class) )
					continue;
				if( sourceProertyType.isEnum() || targetPropertyType.isEnum() )
					continue;
				dumpPropertyMappingInternal(sourceProertyType, targetPropertyType, item.optionClass, null );
			}
		}
	}

	private static PropertyDescriptor findPropertyDescriptor(PropertyDescriptor[] sourcePds, Method method, boolean isRead ) {
		
		for( PropertyDescriptor pDescriptor : sourcePds ) {
			if( isRead ) {
				if ( pDescriptor.getReadMethod() != null && pDescriptor.getReadMethod().equals(method) ) {
					return pDescriptor;
				}
			}
			if( !isRead ) {
				if( pDescriptor.getWriteMethod() != null && pDescriptor.getWriteMethod().equals(method) ) {
					return pDescriptor;
				}
			}
		}
		
		return null;
	}
	
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy