Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.rapleaf.jack.queries.GenericQuery Maven / Gradle / Ivy
package com.rapleaf.jack.queries;
import java.io.IOException;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.sql.SQLRecoverableException;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Stream;
import java.util.stream.StreamSupport;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.rapleaf.jack.BaseDatabaseConnection;
import com.rapleaf.jack.tracking.NoOpAction;
import com.rapleaf.jack.tracking.PostQueryAction;
import com.rapleaf.jack.tracking.QueryStatistics;
public class GenericQuery extends AbstractExecution {
private static final Logger LOG = LoggerFactory.getLogger(GenericQuery.class);
protected static int MAX_CONNECTION_RETRIES = 1;
private final List includedTables;
private Set selectedColumns;
private final PostQueryAction postQueryAction;
private final List joinConditions;
private final List whereConstraints;
private final List parameters;
private final List orderCriteria;
private final Set groupByColumns;
private Optional limitCriteria;
private boolean selectDistinct;
private GenericQuery(BaseDatabaseConnection dbConnection, Table table, PostQueryAction postQueryAction) {
super(dbConnection);
this.includedTables = Lists.newArrayList(table);
this.selectedColumns = Sets.newHashSet();
this.postQueryAction = postQueryAction;
this.joinConditions = Lists.newArrayList();
this.whereConstraints = Lists.newArrayList();
this.parameters = Lists.newArrayList();
parameters.addAll(table.getParameters());
this.orderCriteria = Lists.newArrayList();
this.groupByColumns = Sets.newHashSet();
this.limitCriteria = Optional.empty();
this.selectDistinct = false;
}
public static Builder create(BaseDatabaseConnection dbConnection) {
return new Builder(dbConnection);
}
public static class Builder {
private BaseDatabaseConnection dbConnection;
private PostQueryAction postQueryAction = new NoOpAction();
public Builder(BaseDatabaseConnection dbConnection) {
this.dbConnection = dbConnection;
}
public Builder setConnection(BaseDatabaseConnection dbConnection) {
this.dbConnection = dbConnection;
return this;
}
public Builder setPostQueryAction(PostQueryAction action) {
this.postQueryAction = action;
return this;
}
public GenericQuery from(Table table) {
return new GenericQuery(dbConnection, table, postQueryAction);
}
}
public JoinConditionBuilder leftJoin(Table table) {
this.parameters.addAll(table.getParameters());
return new JoinConditionBuilder(this, JoinType.LEFT_JOIN, table);
}
public JoinConditionBuilder rightJoin(Table table) {
this.parameters.addAll(table.getParameters());
return new JoinConditionBuilder(this, JoinType.RIGHT_JOIN, table);
}
public JoinConditionBuilder innerJoin(Table table) {
this.parameters.addAll(table.getParameters());
return new JoinConditionBuilder(this, JoinType.INNER_JOIN, table);
}
void addJoinCondition(JoinCondition joinCondition) {
this.includedTables.add(joinCondition.getTable());
this.joinConditions.add(joinCondition);
}
void addParameters(List parameters) {
this.parameters.addAll(parameters);
}
public GenericQuery where(GenericConstraint constraint, GenericConstraint... constraints) {
this.whereConstraints.add(constraint);
this.parameters.addAll(constraint.getParameters());
for (GenericConstraint genericConstraint : constraints) {
this.whereConstraints.add(genericConstraint);
this.parameters.addAll(genericConstraint.getParameters());
}
return this;
}
public GenericQuery orderBy(Column column, QueryOrder queryOrder) {
this.orderCriteria.add(new OrderCriterion(column, queryOrder));
return this;
}
public GenericQuery orderBy(Column column) {
this.orderCriteria.add(new OrderCriterion(column, QueryOrder.ASC));
return this;
}
public GenericQuery limit(int offset, int limit) {
this.limitCriteria = Optional.of(new LimitCriterion(offset, limit));
return this;
}
public GenericQuery limit(int limit) {
this.limitCriteria = Optional.of(new LimitCriterion(limit));
return this;
}
public GenericQuery groupBy(Collection columns) {
this.groupByColumns.addAll(columns);
return this;
}
public GenericQuery groupBy(Column column, Column... columns) {
this.groupByColumns.add(column);
this.groupByColumns.addAll(Arrays.asList(columns));
return this;
}
public GenericQuery select(Collection columns) {
this.selectedColumns.addAll(columns);
return this;
}
public GenericQuery select(Column column, Column... columns) {
this.selectedColumns.add(column);
this.selectedColumns.addAll(Arrays.asList(columns));
return this;
}
public GenericQuery select(AbstractTable table, AbstractTable... tbls) {
this.selectedColumns.addAll(table.getAllColumns());
for (AbstractTable tbl : tbls) {
this.selectedColumns.addAll(tbl.getAllColumns());
}
return this;
}
public GenericQuery distinct() {
this.selectDistinct = true;
return this;
}
public SubTable asSubTable(String alias) {
return new SubTable(this, alias);
}
public SingleValue asSingleValue(Column column) {
selectedColumns = Collections.singleton(column);
return new SingleValue<>(getQueryStatement(), getParameters());
}
public MultiValue asMultiValue(Column column) {
selectedColumns = Collections.singleton(column);
return new MultiValue<>(getQueryStatement(), getParameters());
}
public Records fetch() throws IOException {
int retryCount = 0;
final QueryStatistics.Measurer statTracker = new QueryStatistics.Measurer();
statTracker.recordQueryPrepStart();
PreparedStatement preparedStatement = getPreparedStatement(Optional.empty());
statTracker.recordQueryPrepEnd();
while (true) {
try {
statTracker.recordAttempt();
statTracker.recordQueryExecStart();
final Records queryResults = QueryFetcher.getQueryResults(preparedStatement, selectedColumns, dbConnection);
statTracker.recordQueryExecEnd();
final QueryStatistics statistics = statTracker.calculate();
queryResults.addStatistics(statistics);
try {
postQueryAction.perform(statistics);
} catch (Exception ignoredException) {
LOG.error(String.format("Error occurred running post-query action %s", postQueryAction), ignoredException);
}
return queryResults;
} catch (SQLRecoverableException e) {
LOG.error(e.toString());
if (++retryCount > MAX_CONNECTION_RETRIES) {
throw new IOException(e);
}
} catch (SQLException e) {
throw new IOException(e);
}
}
}
/***
* For cases where you don't want sql to send all the records at once. This is NOT just a convenience method to use
* java streams, this is using JDBC built-in streaming
* https://dev.mysql.com/doc/connector-j/8.0/en/connector-j-reference-implementation-notes.html
*/
public Stream fetchAsStream() throws IOException {
int retryCount = 0;
final QueryStatistics.Measurer statTracker = new QueryStatistics.Measurer();
statTracker.recordQueryPrepStart();
PreparedStatement preparedStatement = getPreparedStatementStreaming();
statTracker.recordQueryPrepEnd();
while (true) {
try {
statTracker.recordQueryExecStart();
RecordIterator itr =
QueryFetcher.getQueryResultsStream(preparedStatement, selectedColumns, dbConnection);
itr.addStatisticsMeasurer(statTracker);
Iterable i = () -> itr;
Stream stream = StreamSupport.stream(i.spliterator(), false);
stream = stream.onClose(() -> itr.close());
return stream;
} catch (SQLRecoverableException e) {
LOG.error(e.toString());
if (++retryCount > MAX_CONNECTION_RETRIES) {
throw new IOException(e);
}
} catch (SQLException e) {
throw new IOException(e);
}
}
}
@Override
public String getQueryStatement() {
return getSelectClause()
+ getFromClause()
+ getJoinClause()
+ getWhereClause()
+ getGroupByClause()
+ getOrderClause()
+ getLimitClause();
}
@Override
public List getParameters() {
return parameters;
}
List getIncludedTables() {
return includedTables;
}
Set getSelectedColumns() {
return selectedColumns;
}
private String getSelectClause() {
if (!groupByColumns.isEmpty()) {
if (selectedColumns.isEmpty()) {
throw new RuntimeException("The SELECT list cannot be empty when the GROUP BY clause is specified.");
}
for (Column column : selectedColumns) {
if (!groupByColumns.contains(column) && !(column instanceof AggregatedColumn)) {
throw new RuntimeException("The non-aggregated column " + column.getSqlKeyword() +
" not named in the GROUP BY clause cannot be included in the SELECT list.");
}
}
}
if (selectedColumns.isEmpty()) {
for (Table table : includedTables) {
selectedColumns.addAll(table.getAllColumns());
}
}
String initialKeyword = selectDistinct ? "SELECT DISTINCT " : "SELECT ";
return getClauseFromColumns(selectedColumns, Column::getSelectKeyword, initialKeyword, ", ", " ");
}
private String getFromClause() {
return "FROM " + includedTables.get(0).getSqlKeyword() + " ";
}
private String getJoinClause() {
return getClauseFromQueryConditions(joinConditions, "", "", " ");
}
private String getWhereClause() {
return getClauseFromQueryConditions(whereConstraints, "WHERE ", " AND ", " ");
}
private String getGroupByClause() {
return getClauseFromColumns(groupByColumns, Column::getSqlKeyword, "GROUP BY ", ", ", " ");
}
private String getOrderClause() {
if (orderCriteria.isEmpty()) {
return "";
} else {
return getClauseFromQueryConditions(orderCriteria, "ORDER BY ", ", ", " ");
}
}
private String getLimitClause() {
if (limitCriteria.isPresent()) {
return limitCriteria.get().getSqlStatement() + " ";
} else {
return "";
}
}
}