package io.ebeaninternal.server.query;

import io.ebean.CountedValue;
import io.ebean.core.type.ScalarDataReader;
import io.ebean.util.JdbcClose;
import io.ebeaninternal.api.SpiProfileTransactionEvent;
import io.ebeaninternal.api.SpiQuery;
import io.ebeaninternal.api.SpiTransaction;
import io.ebeaninternal.server.core.OrmQueryRequest;
import io.ebeaninternal.server.deploy.BeanDescriptor;
import io.ebeaninternal.server.type.RsetDataReader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;

Base compiled query request for single attribute queries.
/** * Base compiled query request for single attribute queries. */
class CQueryFetchSingleAttribute implements SpiProfileTransactionEvent { private static final Logger logger = LoggerFactory.getLogger(CQueryFetchSingleAttribute.class); private final CQueryPlan queryPlan;
The overall find request wrapper object.
/** * The overall find request wrapper object. */
private final OrmQueryRequest<?> request; private final BeanDescriptor<?> desc; private final SpiQuery<?> query;
Where clause predicates.
/** * Where clause predicates. */
private final CQueryPredicates predicates;
The final sql that is generated.
/** * The final sql that is generated. */
private final String sql; private RsetDataReader dataReader;
The statement used to create the resultSet.
/** * The statement used to create the resultSet. */
private PreparedStatement pstmt; private String bindLog; private long executionTimeMicros; private int rowCount; private final ScalarDataReader<?> reader; private final boolean containsCounts; private long profileOffset;
Create the Sql select based on the request.
/** * Create the Sql select based on the request. */
CQueryFetchSingleAttribute(OrmQueryRequest<?> request, CQueryPredicates predicates, CQueryPlan queryPlan, boolean containsCounts) { this.request = request; this.queryPlan = queryPlan; this.query = request.getQuery(); this.sql = queryPlan.getSql(); this.desc = request.getBeanDescriptor(); this.predicates = predicates; this.containsCounts = containsCounts; this.reader = queryPlan.getSingleAttributeScalarType(); query.setGeneratedSql(sql); }
Return a summary description of this query.
/** * Return a summary description of this query. */
String getSummary() { StringBuilder sb = new StringBuilder(80); sb.append("FindAttr exeMicros[").append(executionTimeMicros) .append("] rows[").append(rowCount) .append("] type[").append(desc.getName()) .append("] predicates[").append(predicates.getLogWhereSql()) .append("] bind[").append(bindLog).append("]"); return sb.toString(); }
Execute the query returning the row count.
/** * Execute the query returning the row count. */
List<Object> findList() throws SQLException { long startNano = System.nanoTime(); try { prepareExecute(); List<Object> result = new ArrayList<>(); while (dataReader.next()) { Object value = reader.read(dataReader); if (containsCounts) { value = new CountedValue<>(value, dataReader.getLong()); } result.add(value); dataReader.resetColumnPosition(); rowCount++; } executionTimeMicros = (System.nanoTime() - startNano) / 1000L; request.slowQueryCheck(executionTimeMicros, rowCount); if (queryPlan.executionTime(executionTimeMicros)) { queryPlan.captureBindForQueryPlan(predicates, executionTimeMicros); } getTransaction().profileEvent(this); return result; } finally { close(); } } private SpiTransaction getTransaction() { return request.getTransaction(); }
Return the bind log.
/** * Return the bind log. */
String getBindLog() { return bindLog; }
Return the generated sql.
/** * Return the generated sql. */
String getGeneratedSql() { return sql; } private void prepareExecute() throws SQLException { SpiTransaction t = getTransaction(); profileOffset = t.profileOffset(); Connection conn = t.getInternalConnection(); pstmt = conn.prepareStatement(sql); if (query.getBufferFetchSizeHint() > 0) { pstmt.setFetchSize(query.getBufferFetchSizeHint()); } if (query.getTimeout() > 0) { pstmt.setQueryTimeout(query.getTimeout()); } bindLog = predicates.bind(pstmt, conn); dataReader = new RsetDataReader(request.getDataTimeZone(), pstmt.executeQuery()); }
Close the resources.

The jdbc resultSet and statement need to be closed. Its important that this method is called.

/** * Close the resources. * <p> * The jdbc resultSet and statement need to be closed. Its important that * this method is called. * </p> */
private void close() { try { if (dataReader != null) { dataReader.close(); dataReader = null; } } catch (SQLException e) { logger.error("Error closing DataReader", e); } JdbcClose.close(pstmt); pstmt = null; } @Override public void profile() { getTransaction() .profileStream() .addQueryEvent(query.profileEventId(), profileOffset, desc.getName(), rowCount, query.getProfileId()); } Set<String> getDependentTables() { return queryPlan.getDependentTables(); } }