com.querydsl.sql.SQLSerializer Maven / Gradle / Ivy
/*
* Copyright 2015, The Querydsl Team (http://www.querydsl.com/team)
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
* http://www.apache.org/licenses/LICENSE-2.0
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.querydsl.sql;
import com.querydsl.core.JoinExpression;
import com.querydsl.core.JoinFlag;
import com.querydsl.core.QueryFlag;
import com.querydsl.core.QueryFlag.Position;
import com.querydsl.core.QueryMetadata;
import com.querydsl.core.support.SerializerBase;
import com.querydsl.core.types.Constant;
import com.querydsl.core.types.ConstantImpl;
import com.querydsl.core.types.Expression;
import com.querydsl.core.types.ExpressionUtils;
import com.querydsl.core.types.FactoryExpression;
import com.querydsl.core.types.Operation;
import com.querydsl.core.types.Operator;
import com.querydsl.core.types.Ops;
import com.querydsl.core.types.Order;
import com.querydsl.core.types.OrderSpecifier;
import com.querydsl.core.types.ParamExpression;
import com.querydsl.core.types.Path;
import com.querydsl.core.types.Predicate;
import com.querydsl.core.types.SubQueryExpression;
import com.querydsl.core.types.Template;
import com.querydsl.core.types.Template.Element;
import com.querydsl.core.types.TemplateExpression;
import com.querydsl.core.types.TemplateFactory;
import com.querydsl.core.types.dsl.Expressions;
import com.querydsl.core.types.dsl.SimpleExpression;
import com.querydsl.core.util.CollectionUtils;
import com.querydsl.core.util.StringUtils;
import com.querydsl.sql.dml.SQLInsertBatch;
import com.querydsl.sql.dml.SQLMergeUsingCase;
import com.querydsl.sql.types.Null;
import java.sql.Types;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;
import org.jetbrains.annotations.Nullable;
/**
* {@code SqlSerializer} serializes SQL clauses into SQL
*
* @author tiwe
*/
public class SQLSerializer extends SerializerBase {
protected enum Stage {
SELECT,
FROM,
WHERE,
GROUP_BY,
HAVING,
ORDER_BY,
MODIFIERS
}
protected static final Expression> Q = Expressions.template(Object.class, "?");
protected static final String COMMA = ", ";
protected final LinkedList> constantPaths = new LinkedList<>();
protected final Set> withAliases = new HashSet<>();
protected final boolean dml;
protected Stage stage = Stage.SELECT;
protected boolean skipParent;
protected boolean dmlWithSchema;
protected RelationalPath> entity;
protected final Configuration configuration;
protected final SQLTemplates templates;
protected boolean inUnion = false;
protected boolean inJoin = false;
protected boolean inSubquery = false;
protected boolean useLiterals = false;
public SQLSerializer(Configuration conf) {
this(conf, false);
}
public SQLSerializer(Configuration conf, boolean dml) {
super(conf.getTemplates());
this.configuration = conf;
this.templates = conf.getTemplates();
this.dml = dml;
}
protected void appendAsColumnName(Path> path, boolean precededByDot) {
var column = ColumnMetadata.getName(path);
if (path.getMetadata().getParent() instanceof RelationalPath) {
RelationalPath> parent = (RelationalPath>) path.getMetadata().getParent();
column = configuration.getColumnOverride(parent.getSchemaAndTable(), column);
}
append(templates.quoteIdentifier(column, precededByDot));
}
protected SchemaAndTable getSchemaAndTable(RelationalPath> path) {
return configuration.getOverride(path.getSchemaAndTable());
}
protected void appendSchemaName(String schema) {
append(templates.quoteIdentifier(schema));
}
protected void appendTableName(String table, boolean precededByDot) {
append(templates.quoteIdentifier(table, precededByDot));
}
public List> getConstantPaths() {
return constantPaths;
}
/**
* Return a list of expressions that can be used to uniquely define the query sources
*
* @param joins
* @return identifier columns
*/
@SuppressWarnings("unchecked")
protected List> getIdentifierColumns(List joins, boolean alias) {
if (joins.size() == 1) {
var join = joins.get(0);
if (join.getTarget() instanceof RelationalPath) {
return ((RelationalPath) join.getTarget()).getColumns();
} else {
return Collections.emptyList();
}
} else {
List> rv = new ArrayList<>();
var counter = 0;
for (JoinExpression join : joins) {
if (join.getTarget() instanceof RelationalPath) {
var path = (RelationalPath) join.getTarget();
List> columns;
if (path.getPrimaryKey() != null) {
columns = path.getPrimaryKey().getLocalColumns();
} else {
columns = path.getColumns();
}
if (alias) {
for (Expression> column : columns) {
rv.add(ExpressionUtils.as(column, "col" + (++counter)));
}
} else {
rv.addAll(columns);
}
} else {
// not able to provide a distinct list of columns
return Collections.emptyList();
}
}
return rv;
}
}
protected SQLTemplates getTemplates() {
return templates;
}
public void handle(String template, Object... args) {
handleTemplate(TemplateFactory.DEFAULT.create(template), Arrays.asList(args));
}
public final SQLSerializer handleSelect(
final String sep, final List extends Expression>> expressions) {
if (inSubquery) {
Set names = new HashSet<>();
List> replacements = new ArrayList<>();
for (Expression> expr : expressions) {
if (expr instanceof Path) {
var name = ColumnMetadata.getName((Path>) expr);
if (!names.add(name.toLowerCase())) {
expr = ExpressionUtils.as(expr, "col__" + name + replacements.size());
}
}
replacements.add(expr);
}
return handle(sep, replacements);
} else {
return handle(sep, expressions);
}
}
protected void handleJoinTarget(JoinExpression je) {
// type specifier
if (je.getTarget() instanceof RelationalPath && templates.isSupportsAlias()) {
final RelationalPath> pe = (RelationalPath>) je.getTarget();
if (pe.getMetadata().getParent() == null) {
if (withAliases.contains(pe)) {
appendTableName(pe.getMetadata().getName(), false);
append(templates.getTableAlias());
} else {
var schemaAndTable = getSchemaAndTable(pe);
boolean precededByDot;
if (templates.isPrintSchema()) {
appendSchemaName(schemaAndTable.getSchema());
append(".");
precededByDot = true;
} else {
precededByDot = false;
}
appendTableName(schemaAndTable.getTable(), precededByDot);
append(templates.getTableAlias());
}
}
}
inJoin = true;
handle(je.getTarget());
inJoin = false;
}
public void serialize(QueryMetadata metadata, boolean forCountRow) {
templates.serialize(metadata, forCountRow, this);
}
protected void serializeForQuery(QueryMetadata metadata, boolean forCountRow) {
var oldInSubquery = inSubquery;
inSubquery = inSubquery || getLength() > 0;
var oldSkipParent = skipParent;
skipParent = false;
final Expression> select = metadata.getProjection();
final List joins = metadata.getJoins();
final Predicate where = metadata.getWhere();
final List extends Expression>> groupBy = metadata.getGroupBy();
final Predicate having = metadata.getHaving();
final List> orderBy = metadata.getOrderBy();
final Set flags = metadata.getFlags();
final var hasFlags = !flags.isEmpty();
String suffix = null;
List extends Expression>> sqlSelect;
if (select instanceof FactoryExpression) {
sqlSelect = ((FactoryExpression>) select).getArgs();
} else if (select != null) {
sqlSelect = Collections.singletonList(select);
} else {
sqlSelect = Collections.emptyList();
}
// with
if (hasFlags) {
List> withFlags = new ArrayList<>();
var recursive = false;
for (QueryFlag flag : flags) {
if (flag.getPosition() == Position.WITH) {
if (flag.getFlag() == SQLTemplates.RECURSIVE) {
recursive = true;
continue;
}
withFlags.add(flag.getFlag());
}
}
if (!withFlags.isEmpty()) {
if (recursive) {
append(templates.getWithRecursive());
} else {
append(templates.getWith());
}
handle(", ", withFlags);
append("\n");
}
}
// start
if (hasFlags) {
serialize(Position.START, flags);
}
// select
var oldStage = stage;
stage = Stage.SELECT;
if (forCountRow) {
append(templates.getSelect());
if (hasFlags) {
serialize(Position.AFTER_SELECT, flags);
}
if (!metadata.isDistinct()) {
append(templates.getCountStar());
if (!groupBy.isEmpty()) {
append(templates.getFrom());
append("(");
append(templates.getSelect());
append("1 as one ");
suffix = ") internal";
}
} else {
List extends Expression>> columns;
if (sqlSelect.isEmpty()) {
columns = getIdentifierColumns(joins, !templates.isCountDistinctMultipleColumns());
} else {
columns = sqlSelect;
}
if (!groupBy.isEmpty()) {
// select count(*) from (select distinct ...)
append(templates.getCountStar());
append(templates.getFrom());
append("(");
append(templates.getSelectDistinct());
handleSelect(COMMA, columns);
suffix = ") internal";
} else if (columns.size() == 1) {
append(templates.getDistinctCountStart());
handle(columns.get(0));
append(templates.getDistinctCountEnd());
} else if (templates.isCountDistinctMultipleColumns()) {
append(templates.getDistinctCountStart());
append("(").handleSelect(COMMA, columns).append(")");
append(templates.getDistinctCountEnd());
} else {
// select count(*) from (select distinct ...)
append(templates.getCountStar());
append(templates.getFrom());
append("(");
append(templates.getSelectDistinct());
handleSelect(COMMA, columns);
suffix = ") internal";
}
}
} else if (!sqlSelect.isEmpty()) {
if (!metadata.isDistinct()) {
append(templates.getSelect());
} else {
append(templates.getSelectDistinct());
}
if (hasFlags) {
serialize(Position.AFTER_SELECT, flags);
}
handleSelect(COMMA, sqlSelect);
}
if (hasFlags) {
serialize(Position.AFTER_PROJECTION, flags);
}
// from
stage = Stage.FROM;
serializeSources(joins);
// where
if (hasFlags) {
serialize(Position.BEFORE_FILTERS, flags);
}
if (where != null) {
stage = Stage.WHERE;
append(templates.getWhere()).handle(where);
}
if (hasFlags) {
serialize(Position.AFTER_FILTERS, flags);
}
// group by
if (hasFlags) {
serialize(Position.BEFORE_GROUP_BY, flags);
}
if (!groupBy.isEmpty()) {
stage = Stage.GROUP_BY;
append(templates.getGroupBy()).handle(COMMA, groupBy);
}
if (hasFlags) {
serialize(Position.AFTER_GROUP_BY, flags);
}
// having
if (hasFlags) {
serialize(Position.BEFORE_HAVING, flags);
}
if (having != null) {
stage = Stage.HAVING;
append(templates.getHaving()).handle(having);
}
if (hasFlags) {
serialize(Position.AFTER_HAVING, flags);
}
// order by
if (hasFlags) {
serialize(Position.BEFORE_ORDER, flags);
}
if (!orderBy.isEmpty() && !forCountRow) {
stage = Stage.ORDER_BY;
append(templates.getOrderBy());
handleOrderBy(orderBy);
}
if (hasFlags) {
serialize(Position.AFTER_ORDER, flags);
}
// modifiers
if (!forCountRow && metadata.getModifiers().isRestricting() && !joins.isEmpty()) {
stage = Stage.MODIFIERS;
templates.serializeModifiers(metadata, this);
}
if (suffix != null) {
append(suffix);
}
// reset stage
stage = oldStage;
skipParent = oldSkipParent;
inSubquery = oldInSubquery;
}
protected void handleOrderBy(List> orderBy) {
var first = true;
for (final OrderSpecifier> os : orderBy) {
if (!first) {
append(COMMA);
}
var order = os.getOrder() == Order.ASC ? templates.getAsc() : templates.getDesc();
if (os.getNullHandling() == OrderSpecifier.NullHandling.NullsFirst) {
if (templates.getNullsFirst() != null) {
handle(os.getTarget());
append(order);
append(templates.getNullsFirst());
} else {
append("(case when ");
handle(os.getTarget());
append(" is null then 0 else 1 end), ");
handle(os.getTarget());
append(order);
}
} else if (os.getNullHandling() == OrderSpecifier.NullHandling.NullsLast) {
if (templates.getNullsLast() != null) {
handle(os.getTarget());
append(order);
append(templates.getNullsLast());
} else {
append("(case when ");
handle(os.getTarget());
append(" is null then 1 else 0 end), ");
handle(os.getTarget());
append(order);
}
} else {
handle(os.getTarget());
append(order);
}
first = false;
}
}
public void serializeDelete(QueryMetadata metadata, RelationalPath> entity) {
this.entity = entity;
templates.serializeDelete(metadata, entity, this);
}
protected void serializeForDelete(QueryMetadata metadata, RelationalPath> entity) {
serialize(Position.START, metadata.getFlags());
if (!serialize(Position.START_OVERRIDE, metadata.getFlags())) {
append(templates.getDelete());
}
serialize(Position.AFTER_SELECT, metadata.getFlags());
append("from ");
var originalDmlWithSchema = dmlWithSchema;
dmlWithSchema = true;
handle(entity);
dmlWithSchema = originalDmlWithSchema;
if (metadata.getWhere() != null) {
serializeForWhere(metadata);
}
}
protected void serializeForWhere(QueryMetadata metadata) {
var requireSchemaInWhere = templates.isRequiresSchemaInWhere();
var originalDmlWithSchema = dmlWithSchema;
if (requireSchemaInWhere) {
dmlWithSchema = true;
}
append(templates.getWhere()).handle(metadata.getWhere());
if (requireSchemaInWhere) {
dmlWithSchema = originalDmlWithSchema;
}
}
public void serializeMerge(
QueryMetadata metadata,
RelationalPath> entity,
List> keys,
List> columns,
List> values,
@Nullable SubQueryExpression> subQuery) {
this.entity = entity;
templates.serializeMerge(metadata, entity, keys, columns, values, subQuery, this);
}
protected void serializeForMerge(
QueryMetadata metadata,
RelationalPath> entity,
List> keys,
List> columns,
List> values,
@Nullable SubQueryExpression> subQuery) {
serialize(Position.START, metadata.getFlags());
if (!serialize(Position.START_OVERRIDE, metadata.getFlags())) {
append(templates.getMergeInto());
}
serialize(Position.AFTER_SELECT, metadata.getFlags());
var originalDmlWithSchema = dmlWithSchema;
dmlWithSchema = true;
handle(entity);
dmlWithSchema = originalDmlWithSchema;
append(" ");
// columns
if (!columns.isEmpty()) {
skipParent = true;
append("(").handle(COMMA, columns).append(") ");
skipParent = false;
}
// keys
if (!keys.isEmpty()) {
append(templates.getKey());
skipParent = true;
append("(").handle(COMMA, keys).append(") ");
skipParent = false;
}
if (subQuery != null) {
// subquery
append("\n");
serialize(subQuery.getMetadata(), false);
} else {
if (!useLiterals) {
for (var i = 0; i < columns.size(); i++) {
if (values.get(i) instanceof Constant>) {
constantPaths.add(columns.get(i));
}
}
}
// values
append(templates.getValues());
append("(").handle(COMMA, values).append(") ");
}
}
public void serializeMergeUsing(
QueryMetadata metadata,
RelationalPath> entity,
SimpleExpression> usingExpression,
Predicate usingOn,
List whens) {
this.entity = entity;
templates.serializeMergeUsing(metadata, entity, usingExpression, usingOn, whens, this);
}
public void serializeForMergeUsing(
QueryMetadata metadata,
RelationalPath> entity,
SimpleExpression> usingExpression,
Predicate usingOn,
List whens) {
serialize(Position.START, metadata.getFlags());
if (!serialize(Position.START_OVERRIDE, metadata.getFlags())) {
append(templates.getMergeInto());
}
serialize(Position.AFTER_SELECT, metadata.getFlags());
var originalDmlWithSchema = dmlWithSchema;
dmlWithSchema = true;
handle(entity);
dmlWithSchema = originalDmlWithSchema;
append("\nusing ");
// A hacky way to allow merging table to table directly
if (usingExpression instanceof RelationalPath) {
dmlWithSchema = true;
// If table has an alias, handle both original table name and alias
if (!((RelationalPath>) usingExpression)
.getTableName()
.equals(ColumnMetadata.getName((RelationalPath>) usingExpression))) {
RelationalPath> originalEntity = this.entity;
this.entity = (RelationalPath>) usingExpression;
handle(usingExpression);
append(" ");
this.entity = originalEntity;
dmlWithSchema = originalDmlWithSchema;
}
}
handle(usingExpression);
dmlWithSchema = originalDmlWithSchema;
append(templates.getOn());
dmlWithSchema = true;
handle(usingOn);
dmlWithSchema = originalDmlWithSchema;
for (final SQLMergeUsingCase when : whens) {
append("\nwhen ");
if (!when.getMatched()) {
append("not ");
}
append("matched ");
for (final Predicate matchAnd : when.getMatchAnds()) {
append("and ");
handle(matchAnd);
}
append("\nthen ");
if (when.getMergeOperation() == SQLMergeUsingCase.MergeOperation.INSERT) {
var columns = new ArrayList>(when.getUpdates().keySet());
List> values =
columns.stream().map(when.getUpdates()::get).collect(Collectors.toList());
append("insert (");
skipParent = true;
handle(COMMA, columns);
skipParent = false;
append(")");
if (!useLiterals) {
for (var i = 0; i < columns.size(); i++) {
if (values.get(i) instanceof Constant>) {
constantPaths.add(columns.get(i));
}
}
}
// values
append(templates.getValues());
append("(");
handle(COMMA, values);
append(")");
} else if (when.getMergeOperation() == SQLMergeUsingCase.MergeOperation.UPDATE) {
append(templates.getUpdate());
append("\n");
append(templates.getSet());
var first = true;
for (final Map.Entry, Expression>> update : when.getUpdates().entrySet()) {
if (!first) {
append(COMMA);
}
skipParent = true;
handle(update.getKey());
skipParent = false;
append(" = ");
if (!useLiterals && update.getValue() instanceof Constant>) {
constantPaths.add(update.getKey());
}
handle(update.getValue());
first = false;
}
} else if (when.getMergeOperation() == SQLMergeUsingCase.MergeOperation.DELETE) {
append(templates.getDelete());
}
}
append(";");
}
public void serializeInsert(
QueryMetadata metadata,
RelationalPath> entity,
List> columns,
List> values,
@Nullable SubQueryExpression> subQuery) {
this.entity = entity;
templates.serializeInsert(metadata, entity, columns, values, subQuery, this);
}
public void serializeInsert(
QueryMetadata metadata, RelationalPath> entity, List batches) {
this.entity = entity;
templates.serializeInsert(metadata, entity, batches, this);
}
protected void serializeForInsert(
QueryMetadata metadata, RelationalPath> entity, List batches) {
serializeForInsert(
metadata, entity, batches.get(0).getColumns(), batches.get(0).getValues(), null);
for (var i = 1; i < batches.size(); i++) {
append(COMMA);
append("(");
handle(COMMA, batches.get(i).getValues());
append(")");
}
}
protected void serializeForInsert(
QueryMetadata metadata,
RelationalPath> entity,
List> columns,
List> values,
@Nullable SubQueryExpression> subQuery) {
serialize(Position.START, metadata.getFlags());
if (!serialize(Position.START_OVERRIDE, metadata.getFlags())) {
append(templates.getInsertInto());
}
serialize(Position.AFTER_SELECT, metadata.getFlags());
var originalDmlWithSchema = dmlWithSchema;
dmlWithSchema = true;
handle(entity);
dmlWithSchema = originalDmlWithSchema;
// columns
if (!columns.isEmpty()) {
append(" (");
skipParent = true;
handle(COMMA, columns);
skipParent = false;
append(")");
}
if (subQuery != null) {
append("\n");
serialize(subQuery.getMetadata(), false);
} else {
if (!useLiterals) {
for (var i = 0; i < columns.size(); i++) {
if (values.get(i) instanceof Constant>) {
constantPaths.add(columns.get(i));
}
}
}
if (!values.isEmpty()) {
// values
append(templates.getValues());
append("(");
handle(COMMA, values);
append(")");
} else {
append(templates.getDefaultValues());
}
}
}
public void serializeUpdate(
QueryMetadata metadata, RelationalPath> entity, Map, Expression>> updates) {
templates.serializeUpdate(metadata, entity, updates, this);
}
protected void serializeForUpdate(
QueryMetadata metadata, RelationalPath> entity, Map, Expression>> updates) {
this.entity = entity;
serialize(Position.START, metadata.getFlags());
if (!serialize(Position.START_OVERRIDE, metadata.getFlags())) {
append(templates.getUpdate());
}
serialize(Position.AFTER_SELECT, metadata.getFlags());
var originalDmlWithSchema = dmlWithSchema;
dmlWithSchema = true;
handle(entity);
dmlWithSchema = originalDmlWithSchema;
append("\n");
append(templates.getSet());
var first = true;
skipParent = true;
for (final Map.Entry, Expression>> update : updates.entrySet()) {
if (!first) {
append(COMMA);
}
handle(update.getKey());
append(" = ");
if (!useLiterals && update.getValue() instanceof Constant>) {
constantPaths.add(update.getKey());
}
handle(update.getValue());
first = false;
}
skipParent = false;
serialize(Position.BEFORE_FILTERS, metadata.getFlags());
if (metadata.getWhere() != null) {
serializeForWhere(metadata);
}
}
protected void serializeSources(List joins) {
if (joins.isEmpty()) {
var dummyTable = templates.getDummyTable();
if (!StringUtils.isNullOrEmpty(dummyTable)) {
append(templates.getFrom());
append(dummyTable);
}
} else {
append(templates.getFrom());
for (var i = 0; i < joins.size(); i++) {
final var je = joins.get(i);
if (je.getFlags().isEmpty()) {
if (i > 0) {
append(templates.getJoinSymbol(je.getType()));
}
handleJoinTarget(je);
if (je.getCondition() != null) {
append(templates.getOn()).handle(je.getCondition());
}
} else {
serialize(JoinFlag.Position.START, je.getFlags());
if (!serialize(JoinFlag.Position.OVERRIDE, je.getFlags()) && i > 0) {
append(templates.getJoinSymbol(je.getType()));
}
serialize(JoinFlag.Position.BEFORE_TARGET, je.getFlags());
handleJoinTarget(je);
serialize(JoinFlag.Position.BEFORE_CONDITION, je.getFlags());
if (je.getCondition() != null) {
append(templates.getOn()).handle(je.getCondition());
}
serialize(JoinFlag.Position.END, je.getFlags());
}
}
}
}
public void serializeUnion(Expression> union, QueryMetadata metadata, boolean unionAll) {
final List extends Expression>> groupBy = metadata.getGroupBy();
final Predicate having = metadata.getHaving();
final List> orderBy = metadata.getOrderBy();
final Set flags = metadata.getFlags();
final var hasFlags = !flags.isEmpty();
// with
if (hasFlags) {
var handled = false;
var recursive = false;
for (QueryFlag flag : flags) {
if (flag.getPosition() == Position.WITH) {
if (flag.getFlag() == SQLTemplates.RECURSIVE) {
recursive = true;
continue;
}
if (handled) {
append(", ");
}
handle(flag.getFlag());
handled = true;
}
}
if (handled) {
if (recursive) {
prepend(templates.getWithRecursive());
} else {
prepend(templates.getWith());
}
append("\n");
}
}
// union
var oldStage = stage;
handle(union);
// group by
if (hasFlags) {
serialize(Position.BEFORE_GROUP_BY, flags);
}
if (!groupBy.isEmpty()) {
stage = Stage.GROUP_BY;
append(templates.getGroupBy()).handle(COMMA, groupBy);
}
if (hasFlags) {
serialize(Position.AFTER_GROUP_BY, flags);
}
// having
if (hasFlags) {
serialize(Position.BEFORE_HAVING, flags);
}
if (having != null) {
stage = Stage.HAVING;
append(templates.getHaving()).handle(having);
}
if (hasFlags) {
serialize(Position.AFTER_HAVING, flags);
}
// order by
if (hasFlags) {
serialize(Position.BEFORE_ORDER, flags);
}
if (!orderBy.isEmpty()) {
stage = Stage.ORDER_BY;
append(templates.getOrderBy());
skipParent = true;
handleOrderBy(orderBy);
skipParent = false;
}
if (hasFlags) {
serialize(Position.AFTER_ORDER, flags);
}
// end
if (hasFlags) {
serialize(Position.END, flags);
}
// reset stage
stage = oldStage;
}
@SuppressWarnings("unchecked")
@Override
public void visitConstant(Object constant) {
if (useLiterals) {
if (constant instanceof Collection) {
append("(");
var first = true;
for (Object o : ((Collection) constant)) {
if (!first) {
append(COMMA);
}
append(configuration.asLiteral(o));
first = false;
}
append(")");
} else {
append(configuration.asLiteral(constant));
}
} else if (constant instanceof Collection) {
append("(");
var first = true;
for (Object o : ((Collection) constant)) {
if (!first) {
append(COMMA);
}
serializeConstant(constants.size() + 1, null);
constants.add(o);
if (first && (constantPaths.size() < constants.size())) {
constantPaths.add(null);
}
first = false;
}
append(")");
var size = ((Collection) constant).size() - 1;
Path> lastPath = constantPaths.peekLast();
for (var i = 0; i < size; i++) {
constantPaths.add(lastPath);
}
} else {
if (stage == Stage.SELECT
&& !Null.class.isInstance(constant)
&& configuration.getTemplates().isWrapSelectParameters()) {
var typeName = configuration.getTypeNameForCast(constant.getClass());
Expression type = Expressions.constant(typeName);
super.visitOperation(
constant.getClass(), SQLOps.CAST, Arrays.>asList(Q, type));
} else {
serializeConstant(constants.size() + 1, null);
}
constants.add(constant);
if (constantPaths.size() < constants.size()) {
constantPaths.add(null);
}
}
}
@Override
public Void visit(ParamExpression> param, Void context) {
constants.add(param);
serializeConstant(constants.size(), null);
if (constantPaths.size() < constants.size()) {
constantPaths.add(null);
}
return null;
}
@Override
protected void serializeConstant(int parameterIndex, String constantLabel) {
append("?");
}
@Override
public Void visit(Path> path, Void context) {
if (dml) {
if (path.equals(entity) && path instanceof RelationalPath>) {
var schemaAndTable = getSchemaAndTable((RelationalPath>) path);
boolean precededByDot;
if (dmlWithSchema && templates.isPrintSchema()) {
appendSchemaName(schemaAndTable.getSchema());
append(".");
precededByDot = true;
} else {
precededByDot = false;
}
appendTableName(schemaAndTable.getTable(), precededByDot);
return null;
} else if (entity.equals(path.getMetadata().getParent()) && skipParent) {
appendAsColumnName(path, false);
return null;
}
}
final var metadata = path.getMetadata();
boolean precededByDot;
if (metadata.getParent() != null && (!skipParent || dml)) {
visit(metadata.getParent(), context);
append(".");
precededByDot = true;
} else {
precededByDot = false;
}
appendAsColumnName(path, precededByDot);
return null;
}
@Override
public Void visit(SubQueryExpression> query, Void context) {
var oldInSubsuery = inSubquery;
inSubquery = true;
if (inUnion && !templates.isUnionsWrapped()) {
serialize(query.getMetadata(), false);
} else {
append("(");
serialize(query.getMetadata(), false);
append(")");
}
inSubquery = oldInSubsuery;
return null;
}
@Override
public Void visit(TemplateExpression> expr, Void context) {
if (expr.equals(Expressions.TRUE)) {
append(templates.serialize("1", Types.BOOLEAN));
} else if (expr.equals(Expressions.FALSE)) {
append(templates.serialize("0", Types.BOOLEAN));
} else if (inJoin
&& expr instanceof RelationalFunctionCall
&& templates.isFunctionJoinsWrapped()) {
append("table(");
super.visit(expr, context);
append(")");
} else {
super.visit(expr, context);
}
return null;
}
@SuppressWarnings("unchecked")
@Override
protected void visitOperation(
Class> type, Operator operator, List extends Expression>> args) {
var pathAdded = false;
if (args.size() == 2
&& !useLiterals
&& args.get(0) instanceof Path>
&& args.get(1) instanceof Constant>
&& operator != Ops.NUMCAST) {
Object constant = ((Constant>) args.get(1)).getConstant();
if (!Collection.class.isInstance(constant) || !((Collection) constant).isEmpty()) {
for (Element element : templates.getTemplate(operator).getElements()) {
if (element instanceof Template.ByIndex && ((Template.ByIndex) element).getIndex() == 1) {
constantPaths.add((Path>) args.get(0));
pathAdded = true;
break;
}
}
}
}
if (operator == Ops.SET && args.get(0) instanceof SubQueryExpression) {
var oldUnion = inUnion;
inUnion = true;
super.visitOperation(type, SQLOps.UNION, args);
inUnion = oldUnion;
} else if (operator == SQLOps.UNION || operator == SQLOps.UNION_ALL) {
var oldUnion = inUnion;
inUnion = true;
super.visitOperation(type, operator, args);
inUnion = oldUnion;
} else if (operator == Ops.LIKE && args.get(1) instanceof Constant>) {
final var escape = String.valueOf(templates.getEscapeChar());
final var escaped = args.get(1).toString().replace(escape, escape + escape);
super.visitOperation(
String.class, Ops.LIKE, Arrays.asList(args.get(0), ConstantImpl.create(escaped)));
} else if (operator == Ops.STRING_CAST) {
final var typeName = configuration.getTypeNameForCast(String.class);
super.visitOperation(
String.class, SQLOps.CAST, Arrays.asList(args.get(0), ConstantImpl.create(typeName)));
} else if (operator == Ops.NUMCAST) {
@SuppressWarnings("unchecked") // this is the second argument's type
var expectedConstant = (Constant>) args.get(1);
final Class> targetType = expectedConstant.getConstant();
final var typeName = configuration.getTypeNameForCast(targetType);
super.visitOperation(
targetType, SQLOps.CAST, Arrays.asList(args.get(0), ConstantImpl.create(typeName)));
} else if (operator == Ops.ALIAS) {
if (stage == Stage.SELECT || stage == Stage.FROM) {
if (args.get(1) instanceof Path && !((Path>) args.get(1)).getMetadata().isRoot()) {
Path> path = (Path>) args.get(1);
args =
Arrays.asList(
args.get(0), ExpressionUtils.path(path.getType(), path.getMetadata().getName()));
}
super.visitOperation(type, operator, args);
} else {
// handle only target
handle(args.get(1));
}
} else if ((operator == Ops.IN || operator == Ops.NOT_IN)
&& args.get(0) instanceof Path>
&& args.get(1) instanceof Constant>) {
// The type of the constant expression is compatible with the left
// expression, since the compile time checking mandates it to be.
@SuppressWarnings("unchecked")
var coll = ((Constant>) args.get(1)).getConstant();
if (coll.isEmpty()) {
super.visitOperation(
type,
operator == Ops.IN ? Ops.EQ : Ops.NE,
Arrays.asList(Expressions.ONE, Expressions.TWO));
} else {
if (templates.getListMaxSize() == 0 || coll.size() <= templates.getListMaxSize()) {
super.visitOperation(type, operator, args);
} else {
// The type of the path is compatible with the constant
// expression, since the compile time checking mandates it to be
@SuppressWarnings("unchecked")
var path = (Expression
© 2015 - 2025 Weber Informatics LLC | Privacy Policy