MyBatisCursorItemReader.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.Iterator;
  21. import java.util.Map;
  22. import java.util.Optional;
  23. import java.util.function.Supplier;

  24. import org.apache.ibatis.cursor.Cursor;
  25. import org.apache.ibatis.session.ExecutorType;
  26. import org.apache.ibatis.session.SqlSession;
  27. import org.apache.ibatis.session.SqlSessionFactory;
  28. import org.springframework.batch.item.support.AbstractItemCountingItemStreamItemReader;
  29. import org.springframework.beans.factory.InitializingBean;

  30. /**
  31.  * @author Guillaume Darmont / guillaume@dropinocean.com
  32.  */
  33. public class MyBatisCursorItemReader<T> extends AbstractItemCountingItemStreamItemReader<T>
  34.     implements InitializingBean {

  35.   private String queryId;

  36.   private SqlSessionFactory sqlSessionFactory;
  37.   private SqlSession sqlSession;

  38.   private Map<String, Object> parameterValues;
  39.   private Supplier<Map<String, Object>> parameterValuesSupplier;

  40.   private Cursor<T> cursor;
  41.   private Iterator<T> cursorIterator;

  42.   public MyBatisCursorItemReader() {
  43.     setName(getShortName(MyBatisCursorItemReader.class));
  44.   }

  45.   @Override
  46.   protected T doRead() throws Exception {
  47.     T next = null;
  48.     if (cursorIterator.hasNext()) {
  49.       next = cursorIterator.next();
  50.     }
  51.     return next;
  52.   }

  53.   @Override
  54.   protected void doOpen() throws Exception {
  55.     Map<String, Object> parameters = new HashMap<>();
  56.     if (parameterValues != null) {
  57.       parameters.putAll(parameterValues);
  58.     }

  59.     Optional.ofNullable(parameterValuesSupplier).map(Supplier::get).ifPresent(parameters::putAll);

  60.     sqlSession = sqlSessionFactory.openSession(ExecutorType.SIMPLE);
  61.     cursor = sqlSession.selectCursor(queryId, parameters);
  62.     cursorIterator = cursor.iterator();
  63.   }

  64.   @Override
  65.   protected void doClose() throws Exception {
  66.     if (cursor != null) {
  67.       cursor.close();
  68.     }
  69.     if (sqlSession != null) {
  70.       sqlSession.close();
  71.     }
  72.     cursorIterator = null;
  73.   }

  74.   /**
  75.    * Check mandatory properties.
  76.    *
  77.    * @see org.springframework.beans.factory.InitializingBean#afterPropertiesSet()
  78.    */
  79.   @Override
  80.   public void afterPropertiesSet() throws Exception {
  81.     notNull(sqlSessionFactory, "A SqlSessionFactory is required.");
  82.     notNull(queryId, "A queryId is required.");
  83.   }

  84.   /**
  85.    * Public setter for {@link SqlSessionFactory} for injection purposes.
  86.    *
  87.    * @param sqlSessionFactory
  88.    *          a factory object for the {@link SqlSession}.
  89.    */
  90.   public void setSqlSessionFactory(SqlSessionFactory sqlSessionFactory) {
  91.     this.sqlSessionFactory = sqlSessionFactory;
  92.   }

  93.   /**
  94.    * Public setter for the statement id identifying the statement in the SqlMap configuration file.
  95.    *
  96.    * @param queryId
  97.    *          the id for the statement
  98.    */
  99.   public void setQueryId(String queryId) {
  100.     this.queryId = queryId;
  101.   }

  102.   /**
  103.    * The parameter values to be used for the query execution.
  104.    *
  105.    * @param parameterValues
  106.    *          the values keyed by the parameter named used in the query string.
  107.    */
  108.   public void setParameterValues(Map<String, Object> parameterValues) {
  109.     this.parameterValues = parameterValues;
  110.   }

  111.   /**
  112.    * The parameter supplier used to get parameter values for the query execution.
  113.    *
  114.    * @param parameterValuesSupplier
  115.    *          the supplier used to get values keyed by the parameter named used in the query string.
  116.    *
  117.    * @since 2.1.0
  118.    */
  119.   public void setParameterValuesSupplier(Supplier<Map<String, Object>> parameterValuesSupplier) {
  120.     this.parameterValuesSupplier = parameterValuesSupplier;
  121.   }
  122. }