com.scalar.db.sql.statement.StatementUtils Maven / Gradle / Ivy
package com.scalar.db.sql.statement;
import com.google.common.collect.ImmutableList;
import com.scalar.db.sql.AndPredicateList;
import com.scalar.db.sql.BindMarker;
import com.scalar.db.sql.ColumnRef;
import com.scalar.db.sql.JoinPredicate;
import com.scalar.db.sql.LikePredicate;
import com.scalar.db.sql.Literal;
import com.scalar.db.sql.NamedBindMarker;
import com.scalar.db.sql.OrPredicateList;
import com.scalar.db.sql.PositionalBindMarker;
import com.scalar.db.sql.Predicate;
import com.scalar.db.sql.PredicateList;
import com.scalar.db.sql.TableRef;
import com.scalar.db.sql.Term;
import com.scalar.db.sql.Value;
import com.scalar.db.sql.common.SqlError;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
public final class StatementUtils {
private StatementUtils() {}
@SuppressWarnings("unchecked")
public static ImmutableList bindPredicateLists(
ImmutableList predicateLists, Iterator positionalValueIterator) {
ImmutableList.Builder builder = ImmutableList.builder();
predicateLists.forEach(
predicateList -> {
ImmutableList.Builder predicateBuilder = ImmutableList.builder();
predicateList
.getPredicates()
.forEach(p -> predicateBuilder.add(bindPredicate(p, positionalValueIterator)));
if (predicateList instanceof AndPredicateList) {
((ImmutableList.Builder) builder)
.add(AndPredicateList.predicates(predicateBuilder.build()).build());
} else {
assert predicateList instanceof OrPredicateList;
((ImmutableList.Builder) builder)
.add(OrPredicateList.predicates(predicateBuilder.build()).build());
}
});
return builder.build();
}
private static Predicate bindPredicate(
Predicate predicate, Iterator positionalValueIterator) {
if (predicate instanceof LikePredicate) {
return bindLikePredicate((LikePredicate) predicate, positionalValueIterator);
}
if (positionalValueIterator.hasNext() && predicate.value instanceof BindMarker) {
if (predicate.value instanceof NamedBindMarker) {
throw new IllegalArgumentException(SqlError.NAMED_BIND_MARKER_NOT_ALLOWED.buildMessage());
}
return predicate.replaceValue(positionalValueIterator.next());
} else {
return predicate;
}
}
private static LikePredicate bindLikePredicate(
LikePredicate predicate, Iterator positionalValueIterator) {
if (positionalValueIterator.hasNext() && predicate.value instanceof BindMarker) {
if (predicate.value instanceof NamedBindMarker) {
throw new IllegalArgumentException(SqlError.NAMED_BIND_MARKER_NOT_ALLOWED.buildMessage());
}
Value value = positionalValueIterator.next();
if (positionalValueIterator.hasNext() && predicate.escape instanceof BindMarker) {
if (predicate.escape instanceof NamedBindMarker) {
throw new IllegalArgumentException(SqlError.NAMED_BIND_MARKER_NOT_ALLOWED.buildMessage());
}
Value escape = positionalValueIterator.next();
return predicate.replaceValueAndEscape(value, escape);
}
return predicate.replaceValue(value);
} else if (positionalValueIterator.hasNext() && predicate.escape instanceof BindMarker) {
if (predicate.escape instanceof NamedBindMarker) {
throw new IllegalArgumentException(SqlError.NAMED_BIND_MARKER_NOT_ALLOWED.buildMessage());
}
return predicate.replaceEscape(positionalValueIterator.next());
} else {
return predicate;
}
}
@SuppressWarnings("unchecked")
public static ImmutableList bindPredicateLists(
ImmutableList predicateLists, Map namedValues) {
ImmutableList.Builder builder = ImmutableList.builder();
predicateLists.forEach(
predicateList -> {
ImmutableList.Builder predicateBuilder = ImmutableList.builder();
predicateList
.getPredicates()
.forEach(p -> predicateBuilder.add(bindPredicate(p, namedValues)));
if (predicateList instanceof AndPredicateList) {
((ImmutableList.Builder) builder)
.add(AndPredicateList.predicates(predicateBuilder.build()).build());
} else {
assert predicateList instanceof OrPredicateList;
((ImmutableList.Builder) builder)
.add(OrPredicateList.predicates(predicateBuilder.build()).build());
}
});
return builder.build();
}
private static Predicate bindPredicate(Predicate predicate, Map namedValues) {
if (predicate instanceof LikePredicate) {
return bindLikePredicate((LikePredicate) predicate, namedValues);
}
if (predicate.value instanceof BindMarker) {
if (predicate.value instanceof PositionalBindMarker) {
throw new IllegalArgumentException(
SqlError.POSITIONAL_BIND_MARKER_NOT_ALLOWED.buildMessage());
}
String name = ((NamedBindMarker) predicate.value).name;
return namedValues.containsKey(name)
? predicate.replaceValue(namedValues.get(name))
: predicate;
} else {
return predicate;
}
}
private static LikePredicate bindLikePredicate(
LikePredicate predicate, Map namedValues) {
if (predicate.value instanceof BindMarker) {
if (predicate.value instanceof PositionalBindMarker) {
throw new IllegalArgumentException(
SqlError.POSITIONAL_BIND_MARKER_NOT_ALLOWED.buildMessage());
}
String name = ((NamedBindMarker) predicate.value).name;
if (predicate.escape instanceof BindMarker) {
if (predicate.escape instanceof PositionalBindMarker) {
throw new IllegalArgumentException(
SqlError.POSITIONAL_BIND_MARKER_NOT_ALLOWED.buildMessage());
}
String escapeName = ((NamedBindMarker) predicate.escape).name;
return predicate.replaceValueAndEscape(
namedValues.containsKey(name) ? namedValues.get(name) : predicate.value,
namedValues.containsKey(escapeName) ? namedValues.get(escapeName) : predicate.escape);
}
return namedValues.containsKey(name)
? predicate.replaceValue(namedValues.get(name))
: predicate;
} else if (predicate.escape instanceof BindMarker) {
if (predicate.escape instanceof PositionalBindMarker) {
throw new IllegalArgumentException(
SqlError.POSITIONAL_BIND_MARKER_NOT_ALLOWED.buildMessage());
}
String escapeName = ((NamedBindMarker) predicate.escape).name;
return namedValues.containsKey(escapeName)
? predicate.replaceEscape(namedValues.get(escapeName))
: predicate;
} else {
return predicate;
}
}
public static void appendOptions(StringBuilder builder, Map options) {
boolean first = true;
for (Entry entry : options.entrySet()) {
if (!first) {
builder.append(" AND ");
} else {
first = false;
}
appendStringLiteral(builder, entry.getKey());
builder.append('=');
appendStringLiteral(builder, entry.getValue());
}
}
public static void appendStringLiteral(StringBuilder builder, String string) {
builder.append('\'').append(string).append('\'');
}
public static void appendObjectName(StringBuilder builder, String object) {
appendObjectName(builder, object, true);
}
public static void appendObjectName(StringBuilder builder, String object, boolean doubleQuote) {
if (doubleQuote) {
builder.append('"').append(object).append('"');
} else {
builder.append(object);
}
}
public static void appendObjectNames(StringBuilder builder, Collection object) {
appendObjectNames(builder, object, true);
}
public static void appendObjectNames(
StringBuilder builder, Collection object, boolean doubleQuoteForObjectName) {
boolean first = true;
for (String o : object) {
if (!first) {
builder.append(',');
} else {
first = false;
}
StatementUtils.appendObjectName(builder, o, doubleQuoteForObjectName);
}
}
public static void appendTable(StringBuilder builder, TableRef table) {
appendTable(builder, table, true);
}
public static void appendTable(
StringBuilder builder, TableRef table, boolean doubleQuoteForObjectName) {
if (table.namespaceName != null) {
StatementUtils.appendObjectName(builder, table.namespaceName, doubleQuoteForObjectName);
builder.append('.');
}
StatementUtils.appendObjectName(builder, table.tableName, doubleQuoteForObjectName);
}
public static void appendTables(StringBuilder builder, Collection tables) {
appendTables(builder, tables, true);
}
public static void appendTables(
StringBuilder builder, Collection tables, boolean doubleQuoteForObjectName) {
boolean first = true;
for (TableRef table : tables) {
if (!first) {
builder.append(',');
} else {
first = false;
}
StatementUtils.appendTable(builder, table, doubleQuoteForObjectName);
}
}
public static void appendColumn(StringBuilder builder, ColumnRef column) {
appendColumn(builder, column, true);
}
public static void appendColumn(
StringBuilder builder, ColumnRef column, boolean doubleQuoteForObjectName) {
if (column.table != null) {
appendTable(builder, column.table, doubleQuoteForObjectName);
builder.append('.');
}
StatementUtils.appendObjectName(builder, column.columnName, doubleQuoteForObjectName);
}
public static void appendPredicate(StringBuilder builder, Predicate predicate) {
appendPredicate(builder, predicate, true);
}
public static void appendPredicate(
StringBuilder builder, Predicate predicate, boolean doubleQuoteForObjectName) {
StatementUtils.appendColumn(builder, predicate.column, doubleQuoteForObjectName);
switch (predicate.operator) {
case EQUAL_TO:
builder.append('=');
StatementUtils.appendTerm(builder, predicate.value, doubleQuoteForObjectName);
break;
case NOT_EQUAL_TO:
builder.append("<>");
StatementUtils.appendTerm(builder, predicate.value, doubleQuoteForObjectName);
break;
case GREATER_THAN:
builder.append('>');
StatementUtils.appendTerm(builder, predicate.value, doubleQuoteForObjectName);
break;
case GREATER_THAN_OR_EQUAL_TO:
builder.append(">=");
StatementUtils.appendTerm(builder, predicate.value, doubleQuoteForObjectName);
break;
case LESS_THAN:
builder.append('<');
StatementUtils.appendTerm(builder, predicate.value, doubleQuoteForObjectName);
break;
case LESS_THAN_OR_EQUAL_TO:
builder.append("<=");
StatementUtils.appendTerm(builder, predicate.value, doubleQuoteForObjectName);
break;
case IS_NULL:
builder.append(" IS NULL");
break;
case IS_NOT_NULL:
builder.append(" IS NOT NULL");
break;
case LIKE:
case NOT_LIKE:
assert predicate instanceof LikePredicate;
appendLikePredicate(builder, (LikePredicate) predicate, doubleQuoteForObjectName);
break;
default:
throw new AssertionError();
}
}
private static void appendLikePredicate(
StringBuilder builder, LikePredicate predicate, boolean doubleQuoteForObjectName) {
assert predicate.operator == Predicate.Operator.LIKE
|| predicate.operator == Predicate.Operator.NOT_LIKE;
if (predicate.operator == Predicate.Operator.NOT_LIKE) {
builder.append(" NOT");
}
builder.append(" LIKE ");
StatementUtils.appendTerm(builder, predicate.value, doubleQuoteForObjectName);
builder.append(" ESCAPE ");
StatementUtils.appendTerm(builder, predicate.escape, doubleQuoteForObjectName);
}
public static void appendTerm(StringBuilder builder, Term term) {
appendTerm(builder, term, true);
}
public static void appendTerm(
StringBuilder builder, Term term, boolean doubleQuoteForObjectName) {
if (term instanceof PositionalBindMarker) {
builder.append('?');
} else if (term instanceof NamedBindMarker) {
builder.append(':');
appendObjectName(builder, ((NamedBindMarker) term).name, doubleQuoteForObjectName);
} else if (term instanceof Value) {
Value value = (Value) term;
switch (value.type) {
case BOOLEAN:
assert value.value != null;
builder.append(((boolean) value.value) ? "TRUE" : "FALSE");
break;
case INT:
assert value.value != null;
builder.append((int) value.value);
break;
case BIGINT:
assert value.value != null;
builder.append((long) value.value);
break;
case FLOAT:
assert value.value != null;
builder.append((float) value.value);
break;
case DOUBLE:
assert value.value != null;
builder.append((double) value.value);
break;
case TEXT:
assert value.value != null;
appendStringLiteral(builder, (String) value.value);
break;
case BLOB_BYTES:
case BLOB_BYTE_BUFFER:
throw new IllegalStateException(SqlError.CANNOT_CONVERT_BLOB_TO_SQL.buildMessage());
case NULL:
builder.append("NULL");
break;
default:
throw new AssertionError();
}
} else {
assert term instanceof Literal;
Literal literal = (Literal) term;
switch (literal.type) {
case BOOLEAN:
assert literal.value != null;
builder.append(((boolean) literal.value) ? "TRUE" : "FALSE");
break;
case DECIMAL:
case FLOAT:
assert literal.value != null;
builder.append((String) literal.value);
break;
case STRING:
assert literal.value != null;
appendStringLiteral(builder, (String) literal.value);
break;
case NULL:
builder.append("NULL");
break;
default:
throw new AssertionError();
}
}
}
public static void appendWhere(
StringBuilder builder,
List andPredicateLists,
List orPredicateLists) {
appendWhere(builder, andPredicateLists, orPredicateLists, true);
}
public static void appendWhere(
StringBuilder builder,
List andPredicateLists,
List orPredicateLists,
boolean doubleQuoteForObjectName) {
StringBuilder predicateBuilder = new StringBuilder();
boolean isCnf = andPredicateLists.isEmpty();
Iterator extends PredicateList> iterator =
isCnf ? orPredicateLists.iterator() : andPredicateLists.iterator();
String outerConjunction = isCnf ? " AND " : " OR ";
String innerConjunction = isCnf ? " OR " : " AND ";
boolean firstInClause = true;
while (iterator.hasNext()) {
PredicateList predicateList = iterator.next();
if (!firstInClause) {
predicateBuilder.append(outerConjunction);
} else {
firstInClause = false;
}
predicateBuilder.append("(");
boolean firstInBrackets = true;
for (Predicate predicate : predicateList.getPredicates()) {
if (!firstInBrackets) {
predicateBuilder.append(innerConjunction);
} else {
firstInBrackets = false;
}
StatementUtils.appendPredicate(predicateBuilder, predicate, doubleQuoteForObjectName);
}
predicateBuilder.append(")");
}
if (predicateBuilder.length() > 0) {
builder.append(" WHERE ").append(predicateBuilder);
}
}
public static void appendJoinPredicates(
StringBuilder builder, List joinPredicates) {
appendJoinPredicates(builder, joinPredicates, true);
}
public static void appendJoinPredicates(
StringBuilder builder, List joinPredicates, boolean doubleQuoteForObjectName) {
boolean first = true;
for (JoinPredicate joinPredicate : joinPredicates) {
if (!first) {
builder.append(" AND ");
} else {
first = false;
}
appendJoinPredicate(builder, joinPredicate, doubleQuoteForObjectName);
}
}
public static void appendJoinPredicate(StringBuilder builder, JoinPredicate joinPredicate) {
appendJoinPredicate(builder, joinPredicate, true);
}
public static void appendJoinPredicate(
StringBuilder builder, JoinPredicate joinPredicate, boolean doubleQuoteForObjectName) {
StatementUtils.appendColumn(builder, joinPredicate.leftColumn, doubleQuoteForObjectName);
builder.append('=');
StatementUtils.appendColumn(builder, joinPredicate.rightColumn, doubleQuoteForObjectName);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy