CglibProxyFactory.java

  1. /*
  2.  *    Copyright 2009-2024 the original author or authors.
  3.  *
  4.  *    Licensed under the Apache License, Version 2.0 (the "License");
  5.  *    you may not use this file except in compliance with the License.
  6.  *    You may obtain a copy of the License at
  7.  *
  8.  *       https://www.apache.org/licenses/LICENSE-2.0
  9.  *
  10.  *    Unless required by applicable law or agreed to in writing, software
  11.  *    distributed under the License is distributed on an "AS IS" BASIS,
  12.  *    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13.  *    See the License for the specific language governing permissions and
  14.  *    limitations under the License.
  15.  */
  16. package org.apache.ibatis.executor.loader.cglib;

  17. import java.lang.reflect.Method;
  18. import java.util.List;
  19. import java.util.Map;
  20. import java.util.Set;
  21. import java.util.concurrent.locks.ReentrantLock;

  22. import net.sf.cglib.proxy.Callback;
  23. import net.sf.cglib.proxy.Enhancer;
  24. import net.sf.cglib.proxy.MethodInterceptor;
  25. import net.sf.cglib.proxy.MethodProxy;

  26. import org.apache.ibatis.executor.loader.AbstractEnhancedDeserializationProxy;
  27. import org.apache.ibatis.executor.loader.AbstractSerialStateHolder;
  28. import org.apache.ibatis.executor.loader.ProxyFactory;
  29. import org.apache.ibatis.executor.loader.ResultLoaderMap;
  30. import org.apache.ibatis.executor.loader.WriteReplaceInterface;
  31. import org.apache.ibatis.io.Resources;
  32. import org.apache.ibatis.logging.Log;
  33. import org.apache.ibatis.logging.LogFactory;
  34. import org.apache.ibatis.reflection.ExceptionUtil;
  35. import org.apache.ibatis.reflection.factory.ObjectFactory;
  36. import org.apache.ibatis.reflection.property.PropertyCopier;
  37. import org.apache.ibatis.reflection.property.PropertyNamer;
  38. import org.apache.ibatis.session.Configuration;

  39. /**
  40.  * @author Clinton Begin
  41.  *
  42.  * @deprecated Since 3.5.10, use Javassist instead.
  43.  */
  44. @Deprecated
  45. public class CglibProxyFactory implements ProxyFactory {

  46.   private static final String FINALIZE_METHOD = "finalize";
  47.   private static final String WRITE_REPLACE_METHOD = "writeReplace";

  48.   public CglibProxyFactory() {
  49.     try {
  50.       Resources.classForName("net.sf.cglib.proxy.Enhancer");
  51.     } catch (Throwable e) {
  52.       throw new IllegalStateException(
  53.           "Cannot enable lazy loading because CGLIB is not available. Add CGLIB to your classpath.", e);
  54.     }
  55.   }

  56.   @Override
  57.   public Object createProxy(Object target, ResultLoaderMap lazyLoader, Configuration configuration,
  58.       ObjectFactory objectFactory, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {
  59.     return EnhancedResultObjectProxyImpl.createProxy(target, lazyLoader, configuration, objectFactory,
  60.         constructorArgTypes, constructorArgs);
  61.   }

  62.   public Object createDeserializationProxy(Object target, Map<String, ResultLoaderMap.LoadPair> unloadedProperties,
  63.       ObjectFactory objectFactory, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {
  64.     return EnhancedDeserializationProxyImpl.createProxy(target, unloadedProperties, objectFactory, constructorArgTypes,
  65.         constructorArgs);
  66.   }

  67.   static Object createStaticProxy(Class<?> type, Callback callback, List<Class<?>> constructorArgTypes,
  68.       List<Object> constructorArgs) {
  69.     LogHolder.log.warn("CglibProxyFactory is deprecated. Use another proxy factory implementation.");
  70.     Enhancer enhancer = new Enhancer();
  71.     enhancer.setCallback(callback);
  72.     enhancer.setSuperclass(type);
  73.     try {
  74.       type.getDeclaredMethod(WRITE_REPLACE_METHOD);
  75.       // ObjectOutputStream will call writeReplace of objects returned by writeReplace
  76.       if (LogHolder.log.isDebugEnabled()) {
  77.         LogHolder.log.debug(WRITE_REPLACE_METHOD + " method was found on bean " + type + ", make sure it returns this");
  78.       }
  79.     } catch (NoSuchMethodException e) {
  80.       enhancer.setInterfaces(new Class[] { WriteReplaceInterface.class });
  81.     } catch (SecurityException e) {
  82.       // nothing to do here
  83.     }
  84.     Object enhanced;
  85.     if (constructorArgTypes.isEmpty()) {
  86.       enhanced = enhancer.create();
  87.     } else {
  88.       Class<?>[] typesArray = constructorArgTypes.toArray(new Class[constructorArgTypes.size()]);
  89.       Object[] valuesArray = constructorArgs.toArray(new Object[constructorArgs.size()]);
  90.       enhanced = enhancer.create(typesArray, valuesArray);
  91.     }
  92.     return enhanced;
  93.   }

  94.   private static class EnhancedResultObjectProxyImpl implements MethodInterceptor {

  95.     private final Class<?> type;
  96.     private final ResultLoaderMap lazyLoader;
  97.     private final boolean aggressive;
  98.     private final Set<String> lazyLoadTriggerMethods;
  99.     private final ObjectFactory objectFactory;
  100.     private final List<Class<?>> constructorArgTypes;
  101.     private final List<Object> constructorArgs;
  102.     private final ReentrantLock lock = new ReentrantLock();

  103.     private EnhancedResultObjectProxyImpl(Class<?> type, ResultLoaderMap lazyLoader, Configuration configuration,
  104.         ObjectFactory objectFactory, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {
  105.       this.type = type;
  106.       this.lazyLoader = lazyLoader;
  107.       this.aggressive = configuration.isAggressiveLazyLoading();
  108.       this.lazyLoadTriggerMethods = configuration.getLazyLoadTriggerMethods();
  109.       this.objectFactory = objectFactory;
  110.       this.constructorArgTypes = constructorArgTypes;
  111.       this.constructorArgs = constructorArgs;
  112.     }

  113.     public static Object createProxy(Object target, ResultLoaderMap lazyLoader, Configuration configuration,
  114.         ObjectFactory objectFactory, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {
  115.       final Class<?> type = target.getClass();
  116.       EnhancedResultObjectProxyImpl callback = new EnhancedResultObjectProxyImpl(type, lazyLoader, configuration,
  117.           objectFactory, constructorArgTypes, constructorArgs);
  118.       Object enhanced = createStaticProxy(type, callback, constructorArgTypes, constructorArgs);
  119.       PropertyCopier.copyBeanProperties(type, target, enhanced);
  120.       return enhanced;
  121.     }

  122.     @Override
  123.     public Object intercept(Object enhanced, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
  124.       final String methodName = method.getName();
  125.       lock.lock();
  126.       try {
  127.         if (WRITE_REPLACE_METHOD.equals(methodName)) {
  128.           Object original;
  129.           if (constructorArgTypes.isEmpty()) {
  130.             original = objectFactory.create(type);
  131.           } else {
  132.             original = objectFactory.create(type, constructorArgTypes, constructorArgs);
  133.           }
  134.           PropertyCopier.copyBeanProperties(type, enhanced, original);
  135.           if (lazyLoader.size() > 0) {
  136.             return new CglibSerialStateHolder(original, lazyLoader.getProperties(), objectFactory, constructorArgTypes,
  137.                 constructorArgs);
  138.           } else {
  139.             return original;
  140.           }
  141.         }
  142.         if (lazyLoader.size() > 0 && !FINALIZE_METHOD.equals(methodName)) {
  143.           if (aggressive || lazyLoadTriggerMethods.contains(methodName)) {
  144.             lazyLoader.loadAll();
  145.           } else if (PropertyNamer.isSetter(methodName)) {
  146.             final String property = PropertyNamer.methodToProperty(methodName);
  147.             lazyLoader.remove(property);
  148.           } else if (PropertyNamer.isGetter(methodName)) {
  149.             final String property = PropertyNamer.methodToProperty(methodName);
  150.             if (lazyLoader.hasLoader(property)) {
  151.               lazyLoader.load(property);
  152.             }
  153.           }
  154.         }
  155.         return methodProxy.invokeSuper(enhanced, args);
  156.       } catch (Throwable t) {
  157.         throw ExceptionUtil.unwrapThrowable(t);
  158.       } finally {
  159.         lock.unlock();
  160.       }
  161.     }
  162.   }

  163.   private static class EnhancedDeserializationProxyImpl extends AbstractEnhancedDeserializationProxy
  164.       implements MethodInterceptor {

  165.     private EnhancedDeserializationProxyImpl(Class<?> type, Map<String, ResultLoaderMap.LoadPair> unloadedProperties,
  166.         ObjectFactory objectFactory, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {
  167.       super(type, unloadedProperties, objectFactory, constructorArgTypes, constructorArgs);
  168.     }

  169.     public static Object createProxy(Object target, Map<String, ResultLoaderMap.LoadPair> unloadedProperties,
  170.         ObjectFactory objectFactory, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {
  171.       final Class<?> type = target.getClass();
  172.       EnhancedDeserializationProxyImpl callback = new EnhancedDeserializationProxyImpl(type, unloadedProperties,
  173.           objectFactory, constructorArgTypes, constructorArgs);
  174.       Object enhanced = createStaticProxy(type, callback, constructorArgTypes, constructorArgs);
  175.       PropertyCopier.copyBeanProperties(type, target, enhanced);
  176.       return enhanced;
  177.     }

  178.     @Override
  179.     public Object intercept(Object enhanced, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
  180.       final Object o = super.invoke(enhanced, method, args);
  181.       return o instanceof AbstractSerialStateHolder ? o : methodProxy.invokeSuper(o, args);
  182.     }

  183.     @Override
  184.     protected AbstractSerialStateHolder newSerialStateHolder(Object userBean,
  185.         Map<String, ResultLoaderMap.LoadPair> unloadedProperties, ObjectFactory objectFactory,
  186.         List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {
  187.       return new CglibSerialStateHolder(userBean, unloadedProperties, objectFactory, constructorArgTypes,
  188.           constructorArgs);
  189.     }
  190.   }

  191.   private static class LogHolder {
  192.     private static final Log log = LogFactory.getLog(CglibProxyFactory.class);
  193.   }

  194. }