001 /** 002 * Copyright 2005-2014 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 package org.kuali.rice.kew.docsearch.dao.impl; 017 018 import org.apache.commons.lang.StringUtils; 019 import org.kuali.rice.core.api.uif.RemotableAttributeField; 020 import org.kuali.rice.coreservice.framework.CoreFrameworkServiceLocator; 021 import org.kuali.rice.kew.api.document.search.DocumentSearchCriteria; 022 import org.kuali.rice.kew.api.document.search.DocumentSearchResults; 023 import org.kuali.rice.kew.impl.document.search.DocumentSearchGenerator; 024 import org.kuali.rice.kew.docsearch.dao.DocumentSearchDAO; 025 import org.kuali.rice.kew.api.KewApiConstants; 026 import org.kuali.rice.kew.util.PerformanceLogger; 027 import org.kuali.rice.krad.util.KRADConstants; 028 import org.springframework.dao.DataAccessException; 029 import org.springframework.jdbc.core.ConnectionCallback; 030 import org.springframework.jdbc.core.JdbcTemplate; 031 import org.springframework.jdbc.datasource.TransactionAwareDataSourceProxy; 032 033 import javax.sql.DataSource; 034 import java.sql.Connection; 035 import java.sql.ResultSet; 036 import java.sql.SQLException; 037 import java.sql.Statement; 038 import java.util.List; 039 040 /** 041 * Spring JdbcTemplate implementation of DocumentSearchDAO 042 * 043 * @author Kuali Rice Team (rice.collab@kuali.org) 044 * 045 */ 046 public class DocumentSearchDAOJdbcImpl implements DocumentSearchDAO { 047 048 public static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(DocumentSearchDAOJdbcImpl.class); 049 private static final int DEFAULT_FETCH_MORE_ITERATION_LIMIT = 10; 050 051 private DataSource dataSource; 052 053 public void setDataSource(DataSource dataSource) { 054 this.dataSource = new TransactionAwareDataSourceProxy(dataSource); 055 } 056 057 @Override 058 public DocumentSearchResults.Builder findDocuments(final DocumentSearchGenerator documentSearchGenerator, final DocumentSearchCriteria criteria, final boolean criteriaModified, final List<RemotableAttributeField> searchFields) { 059 final int maxResultCap = getMaxResultCap(criteria); 060 try { 061 final JdbcTemplate template = new JdbcTemplate(dataSource); 062 063 return template.execute(new ConnectionCallback<DocumentSearchResults.Builder>() { 064 @Override 065 public DocumentSearchResults.Builder doInConnection(final Connection con) throws SQLException { 066 final Statement statement = con.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); 067 try { 068 final int fetchIterationLimit = getFetchMoreIterationLimit(); 069 final int fetchLimit = fetchIterationLimit * maxResultCap; 070 statement.setFetchSize(maxResultCap + 1); 071 statement.setMaxRows(fetchLimit + 1); 072 073 PerformanceLogger perfLog = new PerformanceLogger(); 074 String sql = documentSearchGenerator.generateSearchSql(criteria, searchFields); 075 perfLog.log("Time to generate search sql from documentSearchGenerator class: " + documentSearchGenerator 076 .getClass().getName(), true); 077 LOG.info("Executing document search with statement max rows: " + statement.getMaxRows()); 078 LOG.info("Executing document search with statement fetch size: " + statement.getFetchSize()); 079 perfLog = new PerformanceLogger(); 080 final ResultSet rs = statement.executeQuery(sql); 081 try { 082 perfLog.log("Time to execute doc search database query.", true); 083 final Statement searchAttributeStatement = con.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); 084 try { 085 return documentSearchGenerator.processResultSet(criteria, criteriaModified, searchAttributeStatement, rs, maxResultCap, fetchLimit); 086 } finally { 087 try { 088 searchAttributeStatement.close(); 089 } catch (SQLException e) { 090 LOG.warn("Could not close search attribute statement."); 091 } 092 } 093 } finally { 094 try { 095 rs.close(); 096 } catch (SQLException e) { 097 LOG.warn("Could not close result set."); 098 } 099 } 100 } finally { 101 try { 102 statement.close(); 103 } catch (SQLException e) { 104 LOG.warn("Could not close statement."); 105 } 106 } 107 } 108 }); 109 110 } catch (DataAccessException dae) { 111 String errorMsg = "DataAccessException: " + dae.getMessage(); 112 LOG.error("getList() " + errorMsg, dae); 113 throw new RuntimeException(errorMsg, dae); 114 } catch (Exception e) { 115 String errorMsg = "LookupException: " + e.getMessage(); 116 LOG.error("getList() " + errorMsg, e); 117 throw new RuntimeException(errorMsg, e); 118 } 119 } 120 121 /** 122 * Returns the maximum number of results that should be returned from the document search. 123 * 124 * @param criteria the criteria in which to check for a max results value 125 * @return the maximum number of results that should be returned from a document search 126 */ 127 public int getMaxResultCap(DocumentSearchCriteria criteria) { 128 int systemLimit = KewApiConstants.DOCUMENT_LOOKUP_DEFAULT_RESULT_CAP; 129 String resultCapValue = CoreFrameworkServiceLocator.getParameterService().getParameterValueAsString(KewApiConstants.KEW_NAMESPACE, KRADConstants.DetailTypes.DOCUMENT_SEARCH_DETAIL_TYPE, KewApiConstants.DOC_SEARCH_RESULT_CAP); 130 if (StringUtils.isNotBlank(resultCapValue)) { 131 try { 132 int configuredLimit = Integer.parseInt(resultCapValue); 133 if (configuredLimit <= 0) { 134 LOG.warn(KewApiConstants.DOC_SEARCH_RESULT_CAP + " was less than or equal to zero. Please use a positive integer."); 135 } else { 136 systemLimit = configuredLimit; 137 } 138 } catch (NumberFormatException e) { 139 LOG.warn(KewApiConstants.DOC_SEARCH_RESULT_CAP + " is not a valid number. Value was " + resultCapValue + ". Using default: " + KewApiConstants.DOCUMENT_LOOKUP_DEFAULT_RESULT_CAP); 140 } 141 } 142 int maxResults = systemLimit; 143 if (criteria.getMaxResults() != null) { 144 int criteriaLimit = criteria.getMaxResults().intValue(); 145 if (criteriaLimit > systemLimit) { 146 LOG.warn("Result set cap of " + criteriaLimit + " is greater than system value of " + systemLimit); 147 } else { 148 if (criteriaLimit < 0) { 149 LOG.warn("Criteria results limit was less than zero."); 150 criteriaLimit = 0; 151 } 152 maxResults = criteriaLimit; 153 } 154 } 155 return maxResults; 156 } 157 158 public int getFetchMoreIterationLimit() { 159 int fetchMoreLimit = DEFAULT_FETCH_MORE_ITERATION_LIMIT; 160 String fetchMoreLimitValue = CoreFrameworkServiceLocator.getParameterService().getParameterValueAsString(KewApiConstants.KEW_NAMESPACE, KRADConstants.DetailTypes.DOCUMENT_SEARCH_DETAIL_TYPE, KewApiConstants.DOC_SEARCH_FETCH_MORE_ITERATION_LIMIT); 161 if (!StringUtils.isBlank(fetchMoreLimitValue)) { 162 try { 163 fetchMoreLimit = Integer.parseInt(fetchMoreLimitValue); 164 if (fetchMoreLimit < 0) { 165 LOG.warn(KewApiConstants.DOC_SEARCH_FETCH_MORE_ITERATION_LIMIT + " was less than zero. Please use a value greater than or equal to zero."); 166 fetchMoreLimit = DEFAULT_FETCH_MORE_ITERATION_LIMIT; 167 } 168 } catch (NumberFormatException e) { 169 LOG.warn(KewApiConstants.DOC_SEARCH_FETCH_MORE_ITERATION_LIMIT + " is not a valid number. Value was " + fetchMoreLimitValue); 170 } 171 } 172 return fetchMoreLimit; 173 } 174 175 }