MyBatisPagingItemReader.java

  1. /*
  2.  * Copyright 2010-2022 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.batch;

  17. import static org.springframework.util.Assert.notNull;
  18. import static org.springframework.util.ClassUtils.getShortName;

  19. import java.util.HashMap;
  20. import java.util.Map;
  21. import java.util.Optional;
  22. import java.util.concurrent.CopyOnWriteArrayList;
  23. import java.util.function.Supplier;

  24. import org.apache.ibatis.session.ExecutorType;
  25. import org.apache.ibatis.session.SqlSession;
  26. import org.apache.ibatis.session.SqlSessionFactory;
  27. import org.mybatis.spring.SqlSessionTemplate;
  28. import org.springframework.batch.item.database.AbstractPagingItemReader;

  29. /**
  30.  * {@code org.springframework.batch.item.ItemReader} for reading database records using MyBatis in a paging fashion.
  31.  * <p>
  32.  * Provided to facilitate the migration from Spring-Batch iBATIS 2 page item readers to MyBatis 3.
  33.  *
  34.  * @author Eduardo Macarron
  35.  *
  36.  * @since 1.1.0
  37.  */
  38. public class MyBatisPagingItemReader<T> extends AbstractPagingItemReader<T> {

  39.   private String queryId;

  40.   private SqlSessionFactory sqlSessionFactory;

  41.   private SqlSessionTemplate sqlSessionTemplate;

  42.   private Map<String, Object> parameterValues;

  43.   private Supplier<Map<String, Object>> parameterValuesSupplier;

  44.   public MyBatisPagingItemReader() {
  45.     setName(getShortName(MyBatisPagingItemReader.class));
  46.   }

  47.   /**
  48.    * Public setter for {@link SqlSessionFactory} for injection purposes.
  49.    *
  50.    * @param sqlSessionFactory
  51.    *          a factory object for the {@link SqlSession}.
  52.    */
  53.   public void setSqlSessionFactory(SqlSessionFactory sqlSessionFactory) {
  54.     this.sqlSessionFactory = sqlSessionFactory;
  55.   }

  56.   /**
  57.    * Public setter for the statement id identifying the statement in the SqlMap configuration file.
  58.    *
  59.    * @param queryId
  60.    *          the id for the statement
  61.    */
  62.   public void setQueryId(String queryId) {
  63.     this.queryId = queryId;
  64.   }

  65.   /**
  66.    * The parameter values to be used for the query execution.
  67.    *
  68.    * @param parameterValues
  69.    *          the values keyed by the parameter named used in the query string.
  70.    */
  71.   public void setParameterValues(Map<String, Object> parameterValues) {
  72.     this.parameterValues = parameterValues;
  73.   }

  74.   /**
  75.    * The parameter supplier used to get parameter values for the query execution.
  76.    *
  77.    * @param parameterValuesSupplier
  78.    *          the supplier used to get values keyed by the parameter named used in the query string.
  79.    *
  80.    * @since 2.1.0
  81.    */
  82.   public void setParameterValuesSupplier(Supplier<Map<String, Object>> parameterValuesSupplier) {
  83.     this.parameterValuesSupplier = parameterValuesSupplier;
  84.   }

  85.   /**
  86.    * Check mandatory properties.
  87.    *
  88.    * @see org.springframework.beans.factory.InitializingBean#afterPropertiesSet()
  89.    */
  90.   @Override
  91.   public void afterPropertiesSet() throws Exception {
  92.     super.afterPropertiesSet();
  93.     notNull(sqlSessionFactory, "A SqlSessionFactory is required.");
  94.     notNull(queryId, "A queryId is required.");
  95.   }

  96.   @Override
  97.   protected void doReadPage() {
  98.     if (sqlSessionTemplate == null) {
  99.       sqlSessionTemplate = new SqlSessionTemplate(sqlSessionFactory, ExecutorType.BATCH);
  100.     }
  101.     Map<String, Object> parameters = new HashMap<>();
  102.     if (parameterValues != null) {
  103.       parameters.putAll(parameterValues);
  104.     }
  105.     Optional.ofNullable(parameterValuesSupplier).map(Supplier::get).ifPresent(parameters::putAll);
  106.     parameters.put("_page", getPage());
  107.     parameters.put("_pagesize", getPageSize());
  108.     parameters.put("_skiprows", getPage() * getPageSize());
  109.     if (results == null) {
  110.       results = new CopyOnWriteArrayList<>();
  111.     } else {
  112.       results.clear();
  113.     }
  114.     results.addAll(sqlSessionTemplate.selectList(queryId, parameters));
  115.   }

  116. }