ClassPathMapperScanner.java

  1. /*
  2.  * Copyright 2010-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.mybatis.spring.mapper;

  17. import java.lang.annotation.Annotation;
  18. import java.util.Arrays;
  19. import java.util.List;
  20. import java.util.Optional;
  21. import java.util.Set;

  22. import org.apache.ibatis.io.Resources;
  23. import org.apache.ibatis.session.SqlSessionFactory;
  24. import org.mybatis.logging.Logger;
  25. import org.mybatis.logging.LoggerFactory;
  26. import org.mybatis.spring.SqlSessionTemplate;
  27. import org.springframework.aop.scope.ScopedProxyFactoryBean;
  28. import org.springframework.aop.scope.ScopedProxyUtils;
  29. import org.springframework.aot.AotDetector;
  30. import org.springframework.beans.factory.annotation.AnnotatedBeanDefinition;
  31. import org.springframework.beans.factory.config.BeanDefinition;
  32. import org.springframework.beans.factory.config.BeanDefinitionHolder;
  33. import org.springframework.beans.factory.config.ConfigurableBeanFactory;
  34. import org.springframework.beans.factory.config.RuntimeBeanReference;
  35. import org.springframework.beans.factory.support.AbstractBeanDefinition;
  36. import org.springframework.beans.factory.support.BeanDefinitionRegistry;
  37. import org.springframework.beans.factory.support.RootBeanDefinition;
  38. import org.springframework.context.annotation.ClassPathBeanDefinitionScanner;
  39. import org.springframework.core.NativeDetector;
  40. import org.springframework.core.env.Environment;
  41. import org.springframework.core.type.filter.AnnotationTypeFilter;
  42. import org.springframework.core.type.filter.AssignableTypeFilter;
  43. import org.springframework.core.type.filter.TypeFilter;
  44. import org.springframework.util.StringUtils;

  45. /**
  46.  * A {@link ClassPathBeanDefinitionScanner} that registers Mappers by {@code basePackage}, {@code annotationClass}, or
  47.  * {@code markerInterface}. If an {@code annotationClass} and/or {@code markerInterface} is specified, only the
  48.  * specified types will be searched (searching for all interfaces will be disabled).
  49.  * <p>
  50.  * This functionality was previously a private class of {@link MapperScannerConfigurer}, but was broken out in version
  51.  * 1.2.0.
  52.  *
  53.  * @author Hunter Presnall
  54.  * @author Eduardo Macarron
  55.  *
  56.  * @see MapperFactoryBean
  57.  *
  58.  * @since 1.2.0
  59.  */
  60. public class ClassPathMapperScanner extends ClassPathBeanDefinitionScanner {

  61.   private static final Logger LOGGER = LoggerFactory.getLogger(ClassPathMapperScanner.class);

  62.   // Copy of FactoryBean#OBJECT_TYPE_ATTRIBUTE which was added in Spring 5.2
  63.   static final String FACTORY_BEAN_OBJECT_TYPE = "factoryBeanObjectType";

  64.   private boolean addToConfig = true;

  65.   private boolean lazyInitialization;

  66.   private boolean printWarnLogIfNotFoundMappers = true;

  67.   private SqlSessionFactory sqlSessionFactory;

  68.   private SqlSessionTemplate sqlSessionTemplate;

  69.   private String sqlSessionTemplateBeanName;

  70.   private String sqlSessionFactoryBeanName;

  71.   private Class<? extends Annotation> annotationClass;

  72.   private Class<?> markerInterface;

  73.   private Class<? extends MapperFactoryBean> mapperFactoryBeanClass = MapperFactoryBean.class;

  74.   private String defaultScope;
  75.   private List<TypeFilter> excludeFilters;

  76.   public ClassPathMapperScanner(BeanDefinitionRegistry registry, Environment environment) {
  77.     super(registry, false, environment);
  78.     setIncludeAnnotationConfig(!AotDetector.useGeneratedArtifacts());
  79.     setPrintWarnLogIfNotFoundMappers(!NativeDetector.inNativeImage());
  80.   }

  81.   /**
  82.    * @deprecated Please use the {@link #ClassPathMapperScanner(BeanDefinitionRegistry, Environment)}.
  83.    */
  84.   @Deprecated(since = "3.0.4", forRemoval = true)
  85.   public ClassPathMapperScanner(BeanDefinitionRegistry registry) {
  86.     super(registry, false);
  87.     setIncludeAnnotationConfig(!AotDetector.useGeneratedArtifacts());
  88.     setPrintWarnLogIfNotFoundMappers(!NativeDetector.inNativeImage());
  89.   }

  90.   public void setAddToConfig(boolean addToConfig) {
  91.     this.addToConfig = addToConfig;
  92.   }

  93.   public void setAnnotationClass(Class<? extends Annotation> annotationClass) {
  94.     this.annotationClass = annotationClass;
  95.   }

  96.   /**
  97.    * Set whether enable lazy initialization for mapper bean.
  98.    * <p>
  99.    * Default is {@code false}.
  100.    * </p>
  101.    *
  102.    * @param lazyInitialization
  103.    *          Set the @{code true} to enable
  104.    *
  105.    * @since 2.0.2
  106.    */
  107.   public void setLazyInitialization(boolean lazyInitialization) {
  108.     this.lazyInitialization = lazyInitialization;
  109.   }

  110.   /**
  111.    * Set whether print warning log if not found mappers that matches conditions.
  112.    * <p>
  113.    * Default is {@code true}. But {@code false} when running in native image.
  114.    * </p>
  115.    *
  116.    * @param printWarnLogIfNotFoundMappers
  117.    *          Set the @{code true} to print
  118.    *
  119.    * @since 3.0.1
  120.    */
  121.   public void setPrintWarnLogIfNotFoundMappers(boolean printWarnLogIfNotFoundMappers) {
  122.     this.printWarnLogIfNotFoundMappers = printWarnLogIfNotFoundMappers;
  123.   }

  124.   public void setMarkerInterface(Class<?> markerInterface) {
  125.     this.markerInterface = markerInterface;
  126.   }

  127.   public void setExcludeFilters(List<TypeFilter> excludeFilters) {
  128.     this.excludeFilters = excludeFilters;
  129.   }

  130.   public void setSqlSessionFactory(SqlSessionFactory sqlSessionFactory) {
  131.     this.sqlSessionFactory = sqlSessionFactory;
  132.   }

  133.   public void setSqlSessionTemplate(SqlSessionTemplate sqlSessionTemplate) {
  134.     this.sqlSessionTemplate = sqlSessionTemplate;
  135.   }

  136.   public void setSqlSessionTemplateBeanName(String sqlSessionTemplateBeanName) {
  137.     this.sqlSessionTemplateBeanName = sqlSessionTemplateBeanName;
  138.   }

  139.   public void setSqlSessionFactoryBeanName(String sqlSessionFactoryBeanName) {
  140.     this.sqlSessionFactoryBeanName = sqlSessionFactoryBeanName;
  141.   }

  142.   /**
  143.    * @deprecated Since 2.0.1, Please use the {@link #setMapperFactoryBeanClass(Class)}.
  144.    */
  145.   @Deprecated
  146.   public void setMapperFactoryBean(MapperFactoryBean<?> mapperFactoryBean) {
  147.     this.mapperFactoryBeanClass = mapperFactoryBean == null ? MapperFactoryBean.class : mapperFactoryBean.getClass();
  148.   }

  149.   /**
  150.    * Set the {@code MapperFactoryBean} class.
  151.    *
  152.    * @param mapperFactoryBeanClass
  153.    *          the {@code MapperFactoryBean} class
  154.    *
  155.    * @since 2.0.1
  156.    */
  157.   public void setMapperFactoryBeanClass(Class<? extends MapperFactoryBean> mapperFactoryBeanClass) {
  158.     this.mapperFactoryBeanClass = mapperFactoryBeanClass == null ? MapperFactoryBean.class : mapperFactoryBeanClass;
  159.   }

  160.   /**
  161.    * Set the default scope of scanned mappers.
  162.    * <p>
  163.    * Default is {@code null} (equiv to singleton).
  164.    * </p>
  165.    *
  166.    * @param defaultScope
  167.    *          the scope
  168.    *
  169.    * @since 2.0.6
  170.    */
  171.   public void setDefaultScope(String defaultScope) {
  172.     this.defaultScope = defaultScope;
  173.   }

  174.   /**
  175.    * Configures parent scanner to search for the right interfaces. It can search for all interfaces or just for those
  176.    * that extends a markerInterface or/and those annotated with the annotationClass
  177.    */
  178.   public void registerFilters() {
  179.     var acceptAllInterfaces = true;

  180.     // if specified, use the given annotation and / or marker interface
  181.     if (this.annotationClass != null) {
  182.       addIncludeFilter(new AnnotationTypeFilter(this.annotationClass));
  183.       acceptAllInterfaces = false;
  184.     }

  185.     // override AssignableTypeFilter to ignore matches on the actual marker interface
  186.     if (this.markerInterface != null) {
  187.       addIncludeFilter(new AssignableTypeFilter(this.markerInterface) {
  188.         @Override
  189.         protected boolean matchClassName(String className) {
  190.           return false;
  191.         }
  192.       });
  193.       acceptAllInterfaces = false;
  194.     }

  195.     if (acceptAllInterfaces) {
  196.       // default include filter that accepts all classes
  197.       addIncludeFilter((metadataReader, metadataReaderFactory) -> true);
  198.     }

  199.     // exclude package-info.java
  200.     addExcludeFilter((metadataReader, metadataReaderFactory) -> {
  201.       var className = metadataReader.getClassMetadata().getClassName();
  202.       return className.endsWith("package-info");
  203.     });

  204.     // exclude types declared by MapperScan.excludeFilters
  205.     if (excludeFilters != null && excludeFilters.size() > 0) {
  206.       for (TypeFilter excludeFilter : excludeFilters) {
  207.         addExcludeFilter(excludeFilter);
  208.       }
  209.     }
  210.   }

  211.   /**
  212.    * Calls the parent search that will search and register all the candidates. Then the registered objects are post
  213.    * processed to set them as MapperFactoryBeans
  214.    */
  215.   @Override
  216.   public Set<BeanDefinitionHolder> doScan(String... basePackages) {
  217.     var beanDefinitions = super.doScan(basePackages);

  218.     if (beanDefinitions.isEmpty()) {
  219.       if (printWarnLogIfNotFoundMappers) {
  220.         LOGGER.warn(() -> "No MyBatis mapper was found in '" + Arrays.toString(basePackages)
  221.             + "' package. Please check your configuration.");
  222.       }
  223.     } else {
  224.       processBeanDefinitions(beanDefinitions);
  225.     }

  226.     return beanDefinitions;
  227.   }

  228.   private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitions) {
  229.     AbstractBeanDefinition definition;
  230.     var registry = getRegistry();
  231.     for (BeanDefinitionHolder holder : beanDefinitions) {
  232.       definition = (AbstractBeanDefinition) holder.getBeanDefinition();
  233.       var scopedProxy = false;
  234.       if (ScopedProxyFactoryBean.class.getName().equals(definition.getBeanClassName())) {
  235.         definition = (AbstractBeanDefinition) Optional
  236.             .ofNullable(((RootBeanDefinition) definition).getDecoratedDefinition())
  237.             .map(BeanDefinitionHolder::getBeanDefinition).orElseThrow(() -> new IllegalStateException(
  238.                 "The target bean definition of scoped proxy bean not found. Root bean definition[" + holder + "]"));
  239.         scopedProxy = true;
  240.       }
  241.       var beanClassName = definition.getBeanClassName();
  242.       LOGGER.debug(() -> "Creating MapperFactoryBean with name '" + holder.getBeanName() + "' and '" + beanClassName
  243.           + "' mapperInterface");

  244.       // the mapper interface is the original class of the bean
  245.       // but, the actual class of the bean is MapperFactoryBean
  246.       definition.getConstructorArgumentValues().addGenericArgumentValue(beanClassName); // issue #59
  247.       try {
  248.         Class<?> beanClass = Resources.classForName(beanClassName);
  249.         // Attribute for MockitoPostProcessor
  250.         // https://github.com/mybatis/spring-boot-starter/issues/475
  251.         definition.setAttribute(FACTORY_BEAN_OBJECT_TYPE, beanClass);
  252.         // for spring-native
  253.         definition.getPropertyValues().add("mapperInterface", beanClass);
  254.       } catch (ClassNotFoundException ignore) {
  255.         // ignore
  256.       }

  257.       definition.setBeanClass(this.mapperFactoryBeanClass);

  258.       definition.getPropertyValues().add("addToConfig", this.addToConfig);

  259.       var explicitFactoryUsed = false;
  260.       if (StringUtils.hasText(this.sqlSessionFactoryBeanName)) {
  261.         definition.getPropertyValues().add("sqlSessionFactory",
  262.             new RuntimeBeanReference(this.sqlSessionFactoryBeanName));
  263.         explicitFactoryUsed = true;
  264.       } else if (this.sqlSessionFactory != null) {
  265.         definition.getPropertyValues().add("sqlSessionFactory", this.sqlSessionFactory);
  266.         explicitFactoryUsed = true;
  267.       }

  268.       if (StringUtils.hasText(this.sqlSessionTemplateBeanName)) {
  269.         if (explicitFactoryUsed) {
  270.           LOGGER.warn(
  271.               () -> "Cannot use both: sqlSessionTemplate and sqlSessionFactory together. sqlSessionFactory is ignored.");
  272.         }
  273.         definition.getPropertyValues().add("sqlSessionTemplate",
  274.             new RuntimeBeanReference(this.sqlSessionTemplateBeanName));
  275.         explicitFactoryUsed = true;
  276.       } else if (this.sqlSessionTemplate != null) {
  277.         if (explicitFactoryUsed) {
  278.           LOGGER.warn(
  279.               () -> "Cannot use both: sqlSessionTemplate and sqlSessionFactory together. sqlSessionFactory is ignored.");
  280.         }
  281.         definition.getPropertyValues().add("sqlSessionTemplate", this.sqlSessionTemplate);
  282.         explicitFactoryUsed = true;
  283.       }

  284.       if (!explicitFactoryUsed) {
  285.         LOGGER.debug(() -> "Enabling autowire by type for MapperFactoryBean with name '" + holder.getBeanName() + "'.");
  286.         definition.setAutowireMode(AbstractBeanDefinition.AUTOWIRE_BY_TYPE);
  287.       }

  288.       definition.setLazyInit(lazyInitialization);

  289.       if (scopedProxy) {
  290.         continue;
  291.       }

  292.       if (ConfigurableBeanFactory.SCOPE_SINGLETON.equals(definition.getScope()) && defaultScope != null) {
  293.         definition.setScope(defaultScope);
  294.       }

  295.       if (!definition.isSingleton()) {
  296.         var proxyHolder = ScopedProxyUtils.createScopedProxy(holder, registry, true);
  297.         if (registry.containsBeanDefinition(proxyHolder.getBeanName())) {
  298.           registry.removeBeanDefinition(proxyHolder.getBeanName());
  299.         }
  300.         registry.registerBeanDefinition(proxyHolder.getBeanName(), proxyHolder.getBeanDefinition());
  301.       }

  302.     }
  303.   }

  304.   @Override
  305.   protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {
  306.     return beanDefinition.getMetadata().isInterface() && beanDefinition.getMetadata().isIndependent();
  307.   }

  308.   @Override
  309.   protected boolean checkCandidate(String beanName, BeanDefinition beanDefinition) {
  310.     if (super.checkCandidate(beanName, beanDefinition)) {
  311.       return true;
  312.     }
  313.     LOGGER.warn(() -> "Skipping MapperFactoryBean with name '" + beanName + "' and '"
  314.         + beanDefinition.getBeanClassName() + "' mapperInterface" + ". Bean already defined with the same name!");
  315.     return false;
  316.   }

  317. }