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.landawn.abacus.util.SQLBuilder Maven / Gradle / Ivy
/*
* Copyright (c) 2015, Haiyang Li.
*
* 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.landawn.abacus.util;
import static com.landawn.abacus.util.WD._PARENTHESES_L;
import static com.landawn.abacus.util.WD._PARENTHESES_R;
import static com.landawn.abacus.util.WD._SPACE;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.Collection;
import java.util.EnumMap;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;
import com.landawn.abacus.DirtyMarker;
import com.landawn.abacus.annotation.Beta;
import com.landawn.abacus.annotation.Internal;
import com.landawn.abacus.annotation.NonUpdatable;
import com.landawn.abacus.annotation.NotColumn;
import com.landawn.abacus.annotation.ReadOnly;
import com.landawn.abacus.annotation.ReadOnlyId;
import com.landawn.abacus.condition.Between;
import com.landawn.abacus.condition.Binary;
import com.landawn.abacus.condition.Cell;
import com.landawn.abacus.condition.Clause;
import com.landawn.abacus.condition.Condition;
import com.landawn.abacus.condition.ConditionFactory.CF;
import com.landawn.abacus.condition.Criteria;
import com.landawn.abacus.condition.Expression;
import com.landawn.abacus.condition.Having;
import com.landawn.abacus.condition.In;
import com.landawn.abacus.condition.InSubQuery;
import com.landawn.abacus.condition.Join;
import com.landawn.abacus.condition.Junction;
import com.landawn.abacus.condition.Limit;
import com.landawn.abacus.condition.NotIn;
import com.landawn.abacus.condition.NotInSubQuery;
import com.landawn.abacus.condition.SubQuery;
import com.landawn.abacus.condition.Where;
import com.landawn.abacus.core.DirtyMarkerUtil;
import com.landawn.abacus.logging.Logger;
import com.landawn.abacus.logging.LoggerFactory;
import com.landawn.abacus.parser.ParserUtil;
import com.landawn.abacus.parser.ParserUtil.EntityInfo;
import com.landawn.abacus.parser.ParserUtil.PropInfo;
import com.landawn.abacus.util.Tuple.Tuple2;
import com.landawn.abacus.util.Tuple.Tuple4;
import com.landawn.abacus.util.u.Optional;
/**
* It's easier to write/maintain the sql by SQLBuilder
and more efficient, comparing to write sql in plain text.
* The sql()
or pair()
method must be called to release resources.
* Here is a sample:
*
* String sql = NE.insert("gui", "firstName", "lastName").into("account").sql();
* // SQL: INSERT INTO account (gui, first_name, last_name) VALUES (:gui, :firstName, :lastName)
*
*
* The {@code tableName} will NOT be formalized.
* {@code select(...).from(String tableName).where(...)}
* {@code insert(...).into(String tableName).values(...)}
* {@code update(String tableName).set(...).where(...)}
* {@code deleteFrom(String tableName).where(...)}
*
*
*
* @author Haiyang Li
* @see {@link com.landawn.abacus.annotation.ReadOnly}
* @see {@link com.landawn.abacus.annotation.ReadOnlyId}
* @see {@link com.landawn.abacus.annotation.NonUpdatable}
* @see {@link com.landawn.abacus.annotation.Transient}
* @see {@link com.landawn.abacus.annotation.Table}
* @see {@link com.landawn.abacus.annotation.Column}
* @since 0.8
*/
@SuppressWarnings("deprecation")
public abstract class SQLBuilder {
// TODO performance goal: 80% cases (or maybe sql.length < 1024?) can be composed in 0.1 millisecond. 0.01 millisecond will be fantastic if possible.
private static final Logger logger = LoggerFactory.getLogger(SQLBuilder.class);
public static final String ALL = WD.ALL;
public static final String TOP = WD.TOP;
public static final String UNIQUE = WD.UNIQUE;
public static final String DISTINCT = WD.DISTINCT;
public static final String DISTINCTROW = WD.DISTINCTROW;
public static final String ASTERISK = WD.ASTERISK;
public static final String COUNT_ALL = "count(*)";
public static final String _1 = "1";
public static final List _1_list = ImmutableList.of(_1);
static final char[] _INSERT = WD.INSERT.toCharArray();
static final char[] _SPACE_INSERT_SPACE = (WD.SPACE + WD.INSERT + WD.SPACE).toCharArray();
static final char[] _INTO = WD.INTO.toCharArray();
static final char[] _SPACE_INTO_SPACE = (WD.SPACE + WD.INTO + WD.SPACE).toCharArray();
static final char[] _VALUES = WD.VALUES.toCharArray();
static final char[] _SPACE_VALUES_SPACE = (WD.SPACE + WD.VALUES + WD.SPACE).toCharArray();
static final char[] _SELECT = WD.SELECT.toCharArray();
static final char[] _SPACE_SELECT_SPACE = (WD.SPACE + WD.SELECT + WD.SPACE).toCharArray();
static final char[] _FROM = WD.FROM.toCharArray();
static final char[] _SPACE_FROM_SPACE = (WD.SPACE + WD.FROM + WD.SPACE).toCharArray();
static final char[] _UPDATE = WD.UPDATE.toCharArray();
static final char[] _SPACE_UPDATE_SPACE = (WD.SPACE + WD.UPDATE + WD.SPACE).toCharArray();
static final char[] _SET = WD.SET.toCharArray();
static final char[] _SPACE_SET_SPACE = (WD.SPACE + WD.SET + WD.SPACE).toCharArray();
static final char[] _DELETE = WD.DELETE.toCharArray();
static final char[] _SPACE_DELETE_SPACE = (WD.SPACE + WD.DELETE + WD.SPACE).toCharArray();
static final char[] _JOIN = WD.JOIN.toCharArray();
static final char[] _SPACE_JOIN_SPACE = (WD.SPACE + WD.JOIN + WD.SPACE).toCharArray();
static final char[] _LEFT_JOIN = WD.LEFT_JOIN.toCharArray();
static final char[] _SPACE_LEFT_JOIN_SPACE = (WD.SPACE + WD.LEFT_JOIN + WD.SPACE).toCharArray();
static final char[] _RIGHT_JOIN = WD.RIGHT_JOIN.toCharArray();
static final char[] _SPACE_RIGHT_JOIN_SPACE = (WD.SPACE + WD.RIGHT_JOIN + WD.SPACE).toCharArray();
static final char[] _FULL_JOIN = WD.FULL_JOIN.toCharArray();
static final char[] _SPACE_FULL_JOIN_SPACE = (WD.SPACE + WD.FULL_JOIN + WD.SPACE).toCharArray();
static final char[] _CROSS_JOIN = WD.CROSS_JOIN.toCharArray();
static final char[] _SPACE_CROSS_JOIN_SPACE = (WD.SPACE + WD.CROSS_JOIN + WD.SPACE).toCharArray();
static final char[] _INNER_JOIN = WD.INNER_JOIN.toCharArray();
static final char[] _SPACE_INNER_JOIN_SPACE = (WD.SPACE + WD.INNER_JOIN + WD.SPACE).toCharArray();
static final char[] _NATURAL_JOIN = WD.NATURAL_JOIN.toCharArray();
static final char[] _SPACE_NATURAL_JOIN_SPACE = (WD.SPACE + WD.NATURAL_JOIN + WD.SPACE).toCharArray();
static final char[] _ON = WD.ON.toCharArray();
static final char[] _SPACE_ON_SPACE = (WD.SPACE + WD.ON + WD.SPACE).toCharArray();
static final char[] _USING = WD.USING.toCharArray();
static final char[] _SPACE_USING_SPACE = (WD.SPACE + WD.USING + WD.SPACE).toCharArray();
static final char[] _WHERE = WD.WHERE.toCharArray();
static final char[] _SPACE_WHERE_SPACE = (WD.SPACE + WD.WHERE + WD.SPACE).toCharArray();
static final char[] _GROUP_BY = WD.GROUP_BY.toCharArray();
static final char[] _SPACE_GROUP_BY_SPACE = (WD.SPACE + WD.GROUP_BY + WD.SPACE).toCharArray();
static final char[] _HAVING = WD.HAVING.toCharArray();
static final char[] _SPACE_HAVING_SPACE = (WD.SPACE + WD.HAVING + WD.SPACE).toCharArray();
static final char[] _ORDER_BY = WD.ORDER_BY.toCharArray();
static final char[] _SPACE_ORDER_BY_SPACE = (WD.SPACE + WD.ORDER_BY + WD.SPACE).toCharArray();
static final char[] _LIMIT = (WD.SPACE + WD.LIMIT + WD.SPACE).toCharArray();
static final char[] _SPACE_LIMIT_SPACE = (WD.SPACE + WD.LIMIT + WD.SPACE).toCharArray();
static final char[] _OFFSET = WD.OFFSET.toCharArray();
static final char[] _SPACE_OFFSET_SPACE = (WD.SPACE + WD.OFFSET + WD.SPACE).toCharArray();
static final char[] _AND = WD.AND.toCharArray();
static final char[] _SPACE_AND_SPACE = (WD.SPACE + WD.AND + WD.SPACE).toCharArray();
static final char[] _OR = WD.OR.toCharArray();
static final char[] _SPACE_OR_SPACE = (WD.SPACE + WD.OR + WD.SPACE).toCharArray();
static final char[] _UNION = WD.UNION.toCharArray();
static final char[] _SPACE_UNION_SPACE = (WD.SPACE + WD.UNION + WD.SPACE).toCharArray();
static final char[] _UNION_ALL = WD.UNION_ALL.toCharArray();
static final char[] _SPACE_UNION_ALL_SPACE = (WD.SPACE + WD.UNION_ALL + WD.SPACE).toCharArray();
static final char[] _INTERSECT = WD.INTERSECT.toCharArray();
static final char[] _SPACE_INTERSECT_SPACE = (WD.SPACE + WD.INTERSECT + WD.SPACE).toCharArray();
static final char[] _EXCEPT = WD.EXCEPT.toCharArray();
static final char[] _SPACE_EXCEPT_SPACE = (WD.SPACE + WD.EXCEPT + WD.SPACE).toCharArray();
static final char[] _EXCEPT2 = WD.EXCEPT2.toCharArray();
static final char[] _SPACE_EXCEPT2_SPACE = (WD.SPACE + WD.EXCEPT2 + WD.SPACE).toCharArray();
static final char[] _AS = WD.AS.toCharArray();
static final char[] _SPACE_AS_SPACE = (WD.SPACE + WD.AS + WD.SPACE).toCharArray();
static final char[] _SPACE_EQUAL_SPACE = (WD.SPACE + WD.EQUAL + WD.SPACE).toCharArray();
static final char[] _SPACE_FOR_UPDATE = (WD.SPACE + WD.FOR_UPDATE).toCharArray();
static final char[] _COMMA_SPACE = WD.COMMA_SPACE.toCharArray();
static final String SPACE_AS_SPACE = WD.SPACE + WD.AS + WD.SPACE;
private static final Set sqlKeyWords = N.newHashSet(1024);
static {
final Field[] fields = WD.class.getDeclaredFields();
int m = 0;
for (Field field : fields) {
m = field.getModifiers();
if (Modifier.isPublic(m) && Modifier.isStatic(m) && Modifier.isFinal(m) && field.getType().equals(String.class)) {
try {
final String value = (String) field.get(null);
for (String e : StringUtil.split(value, ' ', true)) {
sqlKeyWords.add(e);
sqlKeyWords.add(e.toUpperCase());
sqlKeyWords.add(e.toLowerCase());
}
} catch (Exception e) {
// ignore, should never happen.
}
}
}
}
private static final Map, ImmutableSet> subEntityPropNamesPool = new ObjectPool<>(N.POOL_SIZE);
private static final Map, ImmutableSet> nonSubEntityPropNamesPool = new ObjectPool<>(N.POOL_SIZE);
private static final Map, Set[]> defaultPropNamesPool = new ObjectPool<>(N.POOL_SIZE);
private static final Map, String>> fullSelectPartsPool = N.newHashMap(NamingPolicy.values().length);
static {
for (NamingPolicy np : NamingPolicy.values()) {
fullSelectPartsPool.put(np, new ConcurrentHashMap, String>());
}
}
private static final Map tableDeleteFrom = new ConcurrentHashMap<>();
private static final Map, String[]> classTableNameMap = new ConcurrentHashMap<>();
private static final AtomicInteger activeStringBuilderCounter = new AtomicInteger();
private final NamingPolicy namingPolicy;
private final SQLPolicy sqlPolicy;
private final List parameters = new ArrayList<>();
private StringBuilder sb;
private Class> entityClass;
private ImmutableMap> propColumnNameMap;
private String alias;
private OperationType op;
private String tableName;
private String preselect;
private Collection columnNames;
private Map columnAliases;
private List, String, String, Set>> multiSelects;
private Map>> aliasPropColumnNameMap;
private Map props;
private Collection> propsList;
private boolean hasFromBeenSet = false;
private boolean isForConditionOnly = false;
SQLBuilder(final NamingPolicy namingPolicy, final SQLPolicy sqlPolicy) {
if (activeStringBuilderCounter.incrementAndGet() > 1024) {
logger.error("Too many(" + activeStringBuilderCounter.get()
+ ") StringBuilder instances are created in SQLBuilder. The method sql()/pair() must be called to release resources and close SQLBuilder");
}
this.sb = Objectory.createStringBuilder();
this.namingPolicy = namingPolicy == null ? NamingPolicy.LOWER_CASE_WITH_UNDERSCORE : namingPolicy;
this.sqlPolicy = sqlPolicy == null ? SQLPolicy.SQL : sqlPolicy;
}
/**
* Gets the table name.
*
* @param entityClass
* @param namingPolicy
* @return
*/
static String getTableName(final Class> entityClass, final NamingPolicy namingPolicy) {
String[] entityTableNames = classTableNameMap.get(entityClass);
if (entityTableNames == null) {
final EntityInfo entityInfo = ParserUtil.getEntityInfo(entityClass);
if (entityInfo.tableName.isPresent()) {
entityTableNames = Array.repeat(entityInfo.tableName.get(), 4);
} else {
final String simpleClassName = ClassUtil.getSimpleClassName(entityClass);
entityTableNames = new String[] { ClassUtil.toLowerCaseWithUnderscore(simpleClassName), ClassUtil.toUpperCaseWithUnderscore(simpleClassName),
ClassUtil.toCamelCase(simpleClassName), simpleClassName };
}
classTableNameMap.put(entityClass, entityTableNames);
}
switch (namingPolicy) {
case LOWER_CASE_WITH_UNDERSCORE:
return entityTableNames[0];
case UPPER_CASE_WITH_UNDERSCORE:
return entityTableNames[1];
case LOWER_CAMEL_CASE:
return entityTableNames[2];
default:
return entityTableNames[3];
}
}
/**
* Gets the insert prop names by class.
*
* @param entity
* @param excludedPropNames
* @return
*/
@Internal
public static Collection getInsertPropNames(final Object entity, final Set excludedPropNames) {
final Class> entityClass = entity.getClass();
final boolean isDirtyMarker = ClassUtil.isDirtyMarker(entityClass);
if (isDirtyMarker) {
final Collection signedPropNames = ((DirtyMarker) entity).signedPropNames();
if (N.isNullOrEmpty(excludedPropNames)) {
return signedPropNames;
} else {
final List tmp = new ArrayList<>(signedPropNames);
tmp.removeAll(excludedPropNames);
return tmp;
}
} else {
final Collection[] val = loadPropNamesByClass(entityClass);
if (N.isNullOrEmpty(excludedPropNames)) {
final Collection idPropNames = ClassUtil.getIdFieldNames(entityClass);
if (N.isNullOrEmpty(idPropNames)) {
return val[2];
} else {
final EntityInfo entityInfo = ParserUtil.getEntityInfo(entityClass);
for (String idPropName : idPropNames) {
if (!isDefaultIdPropValue(entityInfo.getPropInfo(idPropName))) {
return val[2];
}
}
return val[3];
}
} else {
final List tmp = new ArrayList<>(val[2]);
tmp.removeAll(excludedPropNames);
return tmp;
}
}
}
/**
* Gets the insert prop names by class.
*
* @param entityClass
* @param excludedPropNames
* @return
*/
@Internal
public static Collection getInsertPropNames(final Class> entityClass, final Set excludedPropNames) {
final Collection[] val = loadPropNamesByClass(entityClass);
final Collection propNames = val[2];
if (N.isNullOrEmpty(excludedPropNames)) {
return propNames;
} else {
final List tmp = new ArrayList<>(propNames);
tmp.removeAll(excludedPropNames);
return tmp;
}
}
/**
* Gets the select prop names by class.
*
* @param entityClass
* @param includeSubEntityProperties
* @param excludedPropNames
* @return
*/
@Internal
public static Collection getSelectPropNames(final Class> entityClass, final boolean includeSubEntityProperties,
final Set excludedPropNames) {
final Collection[] val = loadPropNamesByClass(entityClass);
final Collection propNames = includeSubEntityProperties ? val[0] : val[1];
if (N.isNullOrEmpty(excludedPropNames)) {
return propNames;
} else {
final List tmp = new ArrayList<>(N.max(0, propNames.size() - excludedPropNames.size()));
int idx = 0;
for (String propName : propNames) {
if (!(excludedPropNames.contains(propName)
|| ((idx = propName.indexOf(WD._PERIOD)) > 0 && excludedPropNames.contains(propName.substring(0, idx))))) {
tmp.add(propName);
}
}
return tmp;
}
}
/**
* Checks if is default id prop value.
*
* @param propValue
* @return true, if is default id prop value
*/
@Internal
static boolean isDefaultIdPropValue(final Object propValue) {
return (propValue == null) || (propValue instanceof Number && (((Number) propValue).longValue() == 0));
}
/**
* Gets the update prop names by class.
*
* @param entityClass
* @param excludedPropNames
* @return
*/
@Internal
public static Collection getUpdatePropNames(final Class> entityClass, final Set excludedPropNames) {
final Collection[] val = loadPropNamesByClass(entityClass);
final Collection propNames = val[4];
if (N.isNullOrEmpty(excludedPropNames)) {
return propNames;
} else {
final List tmp = new ArrayList<>(propNames);
tmp.removeAll(excludedPropNames);
return tmp;
}
}
/**
* Load prop names by class.
*
* @param entityClass
* @return
*/
static Set[] loadPropNamesByClass(final Class> entityClass) {
Set[] val = defaultPropNamesPool.get(entityClass);
if (val == null) {
synchronized (entityClass) {
final Set entityPropNames = N.newLinkedHashSet(ClassUtil.getPropNameList(entityClass));
final Set subEntityPropNames = getSubEntityPropNames(entityClass);
if (N.notNullOrEmpty(subEntityPropNames)) {
entityPropNames.removeAll(subEntityPropNames);
}
val = new Set[5];
val[0] = N.newLinkedHashSet(entityPropNames);
val[1] = N.newLinkedHashSet(entityPropNames);
val[2] = N.newLinkedHashSet(entityPropNames);
val[3] = N.newLinkedHashSet(entityPropNames);
val[4] = N.newLinkedHashSet(entityPropNames);
final EntityInfo entityInfo = ParserUtil.getEntityInfo(entityClass);
Class> subEntityClass = null;
Set subEntityPropNameList = null;
for (String subEntityPropName : subEntityPropNames) {
PropInfo propInfo = entityInfo.getPropInfo(subEntityPropName);
subEntityClass = (propInfo.type.isCollection() ? propInfo.type.getElementType() : propInfo.type).clazz();
subEntityPropNameList = N.newLinkedHashSet(ClassUtil.getPropNameList(subEntityClass));
subEntityPropNameList.removeAll(getSubEntityPropNames(subEntityClass));
for (String pn : subEntityPropNameList) {
val[0].add(StringUtil.concat(subEntityPropName, WD.PERIOD, pn));
}
}
final Set nonUpdatableNonWritablePropNames = N.newHashSet();
final Set nonUpdatablePropNames = N.newHashSet();
final Set transientPropNames = N.newHashSet();
for (PropInfo propInfo : entityInfo.propInfoList) {
if (propInfo.isAnnotationPresent(ReadOnly.class) || propInfo.isAnnotationPresent(ReadOnlyId.class)) {
nonUpdatableNonWritablePropNames.add(propInfo.name);
}
if (propInfo.isAnnotationPresent(NonUpdatable.class)) {
nonUpdatablePropNames.add(propInfo.name);
}
if (propInfo.isTransient || propInfo.isAnnotationPresent(NotColumn.class)) {
nonUpdatableNonWritablePropNames.add(propInfo.name);
transientPropNames.add(propInfo.name);
}
}
nonUpdatablePropNames.addAll(nonUpdatableNonWritablePropNames);
val[0].removeAll(transientPropNames);
val[1].removeAll(transientPropNames);
val[2].removeAll(nonUpdatableNonWritablePropNames);
val[3].removeAll(nonUpdatableNonWritablePropNames);
val[4].removeAll(nonUpdatablePropNames);
for (String idPropName : ClassUtil.getIdFieldNames(entityClass)) {
val[3].remove(idPropName);
val[3].remove(ClassUtil.getPropNameByMethod(ClassUtil.getPropGetMethod(entityClass, idPropName)));
}
val[0] = ImmutableSet.of(val[0]); // for select, including sub entity properties.
val[1] = ImmutableSet.of(val[1]); // for select, no sub entity properties.
val[2] = ImmutableSet.of(val[2]); // for insert with id
val[3] = ImmutableSet.of(val[3]); // for insert without id
val[4] = ImmutableSet.of(val[4]); // for update.
defaultPropNamesPool.put(entityClass, val);
}
}
return val;
}
/**
* Gets the sub entity prop names.
*
* @param entityClass
* @return
*/
static ImmutableSet getSubEntityPropNames(final Class> entityClass) {
synchronized (entityClass) {
ImmutableSet subEntityPropNames = subEntityPropNamesPool.get(entityClass);
if (subEntityPropNames == null) {
final EntityInfo entityInfo = ParserUtil.getEntityInfo(entityClass);
final ImmutableSet nonSubEntityPropNames = nonSubEntityPropNamesPool.get(entityClass);
final Set subEntityPropNameSet = N.newLinkedHashSet();
for (PropInfo propInfo : entityInfo.propInfoList) {
if (!propInfo.isMarkedToColumn && (propInfo.type.isEntity() || (propInfo.type.isCollection() && propInfo.type.getElementType().isEntity()))
&& (nonSubEntityPropNames == null || !nonSubEntityPropNames.contains(propInfo.name))) {
subEntityPropNameSet.add(propInfo.name);
}
}
subEntityPropNames = ImmutableSet.of(subEntityPropNameSet);
subEntityPropNamesPool.put(entityClass, subEntityPropNames);
}
return subEntityPropNames;
}
}
private static List getSelectTableNames(final Class> entityClass, final String alias, final Set excludedPropNames,
final NamingPolicy namingPolicy) {
final Set subEntityPropNames = getSubEntityPropNames(entityClass);
if (N.isNullOrEmpty(subEntityPropNames)) {
return N.emptyList();
}
final List res = new ArrayList<>(subEntityPropNames.size() + 1);
if (N.isNullOrEmpty(alias)) {
res.add(getTableName(entityClass, namingPolicy));
} else {
res.add(getTableName(entityClass, namingPolicy) + " " + alias);
}
final EntityInfo entityInfo = ParserUtil.getEntityInfo(entityClass);
PropInfo propInfo = null;
Class> subEntityClass = null;
for (String subEntityPropName : subEntityPropNames) {
if (excludedPropNames != null && excludedPropNames.contains(subEntityPropName)) {
continue;
}
propInfo = entityInfo.getPropInfo(subEntityPropName);
subEntityClass = (propInfo.type.isCollection() ? propInfo.type.getElementType() : propInfo.type).clazz();
res.add(getTableName(subEntityClass, namingPolicy));
}
return res;
}
// /**
// * Register the irregular column names which can not be converted from property name by naming policy.
// *
// * @param propNameTableInterface the interface generated by com.landawn.abacus.util.CodeGenerator
// */
// public static void registerColumnName(final Class> propNameTableInterface) {
// final String PCM = "_PCM";
//
// try {
// final Map _pcm = (Map) propNameTableInterface.getField(PCM).get(null);
//
// for (Class> cls : propNameTableInterface.getDeclaredClasses()) {
// final String entityName = (String) cls.getField(D.UNDERSCORE).get(null);
// final Map entityPCM = (Map) cls.getField(PCM).get(null);
//
// final Map propColumnNameMap = new HashMap<>(_pcm);
// propColumnNameMap.putAll(entityPCM);
//
// registerColumnName(entityName, propColumnNameMap);
// }
// } catch (Exception e) {
// throw N.toRuntimeException(e);
// }
// }
// /**
// * Returns an immutable list of the property name by the specified entity class.
// *
// * @param entityClass
// * @return
// */
// public static List propNameList(final Class> entityClass) {
// List propNameList = classPropNameListPool.get(entityClass);
//
// if (propNameList == null) {
// synchronized (classPropNameListPool) {
// propNameList = classPropNameListPool.get(entityClass);
//
// if (propNameList == null) {
// propNameList = N.asImmutableList(new ArrayList<>(N.getPropGetMethodList(entityClass).keySet()));
// classPropNameListPool.put(entityClass, propNameList);
// }
// }
// }
//
// return propNameList;
// }
// /**
// * Returns an immutable set of the property name by the specified entity class.
// *
// * @param entityClass
// * @return
// */
// public static Set propNameSet(final Class> entityClass) {
// Set propNameSet = classPropNameSetPool.get(entityClass);
//
// if (propNameSet == null) {
// synchronized (classPropNameSetPool) {
// propNameSet = classPropNameSetPool.get(entityClass);
//
// if (propNameSet == null) {
// propNameSet = N.asImmutableSet(N.newLinkedHashSet(N.getPropGetMethodList(entityClass).keySet()));
// classPropNameSetPool.put(entityClass, propNameSet);
// }
// }
// }
//
// return propNameSet;
// }
/**
*
* @param propNames
* @return
*/
@Beta
static Map named(final String... propNames) {
final Map m = N.newLinkedHashMap(propNames.length);
for (String propName : propNames) {
m.put(propName, CF.QME);
}
return m;
}
/**
*
* @param propNames
* @return
*/
@Beta
static Map named(final Collection propNames) {
final Map m = N.newLinkedHashMap(propNames.size());
for (String propName : propNames) {
m.put(propName, CF.QME);
}
return m;
}
private static final Map QM_CACHE = new HashMap<>();
static {
for (int i = 0; i <= 30; i++) {
QM_CACHE.put(i, StringUtil.repeat("?", i, ", "));
}
QM_CACHE.put(100, StringUtil.repeat("?", 100, ", "));
QM_CACHE.put(200, StringUtil.repeat("?", 200, ", "));
QM_CACHE.put(300, StringUtil.repeat("?", 300, ", "));
QM_CACHE.put(500, StringUtil.repeat("?", 500, ", "));
QM_CACHE.put(1000, StringUtil.repeat("?", 1000, ", "));
}
/**
* Repeat question mark({@code ?}) {@code n} times with delimiter {@code ", "}.
*
* It's designed for batch SQL builder.
*
* @param n
* @return
*/
public static String repeatQM(int n) {
N.checkArgNotNegative(n, "count");
String result = QM_CACHE.get(n);
if (result == null) {
result = StringUtil.repeat("?", n, ", ");
}
return result;
}
static final Map, Map>>> entityTablePropColumnNameMap = new ObjectPool<>(N.POOL_SIZE);
/**
*
* @param entityClass
* @param namingPolicy
* @return
* @deprecated for internal use only.
*/
@Deprecated
@Beta
public static ImmutableMap> getProp2ColumnNameMap(final Class> entityClass, final NamingPolicy namingPolicy) {
Map>> namingPropColumnNameMap = entityTablePropColumnNameMap.get(entityClass);
ImmutableMap> result = null;
if (namingPropColumnNameMap == null || (result = namingPropColumnNameMap.get(namingPolicy)) == null) {
final ImmutableMap prop2ColumnNameMap = ClassUtil.getProp2ColumnNameMap(entityClass, namingPolicy);
final Map> newProp2ColumnNameMap = N.newHashMap(prop2ColumnNameMap.size() * 2);
for (Map.Entry entry : prop2ColumnNameMap.entrySet()) {
newProp2ColumnNameMap.put(entry.getKey(), Tuple.of(entry.getValue(), entry.getKey().indexOf('.') < 0));
if (!prop2ColumnNameMap.containsKey(entry.getValue())) {
newProp2ColumnNameMap.put(entry.getValue(), Tuple.of(entry.getValue(), entry.getValue().indexOf('.') < 0));
}
}
result = ImmutableMap.of(newProp2ColumnNameMap);
if (namingPropColumnNameMap == null) {
namingPropColumnNameMap = new EnumMap<>(NamingPolicy.class);
// TODO not necessary?
// namingPropColumnNameMap = Collections.synchronizedMap(namingPropColumnNameMap)
entityTablePropColumnNameMap.put(entityClass, namingPropColumnNameMap);
}
namingPropColumnNameMap.put(namingPolicy, result);
}
return result;
}
/**
*
* @param tableName
* @return
*/
public SQLBuilder into(final String tableName) {
if (op != OperationType.ADD) {
throw new RuntimeException("Invalid operation: " + op);
}
if (N.isNullOrEmpty(columnNames) && N.isNullOrEmpty(props) && N.isNullOrEmpty(propsList)) {
throw new RuntimeException("Column names or props must be set first by insert");
}
this.tableName = tableName;
sb.append(_INSERT);
sb.append(_SPACE_INTO_SPACE);
sb.append(tableName);
sb.append(_SPACE);
sb.append(WD._PARENTHESES_L);
if (N.notNullOrEmpty(columnNames)) {
int i = 0;
for (String columnName : columnNames) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, columnName);
}
} else {
final Map props = N.isNullOrEmpty(this.props) ? propsList.iterator().next() : this.props;
int i = 0;
for (String columnName : props.keySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, columnName);
}
}
sb.append(WD._PARENTHESES_R);
sb.append(_SPACE_VALUES_SPACE);
sb.append(WD._PARENTHESES_L);
if (N.notNullOrEmpty(columnNames)) {
switch (sqlPolicy) {
case SQL:
case PARAMETERIZED_SQL: {
for (int i = 0, size = columnNames.size(); i < size; i++) {
if (i > 0) {
sb.append(_COMMA_SPACE);
}
sb.append(WD._QUESTION_MARK);
}
break;
}
case NAMED_SQL: {
int i = 0;
for (String columnName : columnNames) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
sb.append(":");
sb.append(columnName);
}
break;
}
case IBATIS_SQL: {
int i = 0;
for (String columnName : columnNames) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
sb.append("#{");
sb.append(columnName);
sb.append('}');
}
break;
}
default:
throw new RuntimeException("Not supported SQL policy: " + sqlPolicy);
}
} else if (N.notNullOrEmpty(props)) {
appendInsertProps(props);
} else {
int i = 0;
for (Map props : propsList) {
if (i++ > 0) {
sb.append(WD._PARENTHESES_R);
sb.append(_COMMA_SPACE);
sb.append(WD._PARENTHESES_L);
}
appendInsertProps(props);
}
}
sb.append(WD._PARENTHESES_R);
return this;
}
/**
*
* @param entityClass
* @return
*/
public SQLBuilder into(final Class> entityClass) {
if (this.entityClass == null) {
this.entityClass = entityClass;
this.propColumnNameMap = getProp2ColumnNameMap(this.entityClass, this.namingPolicy);
}
return into(getTableName(entityClass, namingPolicy));
}
/**
*
* @return
*/
public SQLBuilder distinct() {
return preselect(DISTINCT);
}
/**
*
* @param preselect ALL | DISTINCT | DISTINCTROW...
* @return
*/
public SQLBuilder preselect(final String preselect) {
N.checkArgNotNull(preselect, "preselect");
if (sb.length() > 0) {
throw new IllegalStateException("'distinct|preselect' must be called before 'from' operation");
}
if (N.isNullOrEmpty(this.preselect)) {
this.preselect = preselect;
} else {
this.preselect += preselect;
}
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder from(String expr) {
expr = expr.trim();
final int idx = expr.indexOf(WD._COMMA);
final String tableName = idx > 0 ? expr.substring(0, idx) : expr;
return from(tableName.trim(), expr);
}
/**
*
* @param tableNames
* @return
*/
@SafeVarargs
public final SQLBuilder from(final String... tableNames) {
if (tableNames.length == 1) {
return from(tableNames[0].trim());
}
final String tableName = tableNames[0].trim();
return from(tableName, StringUtil.join(tableNames, WD.COMMA_SPACE));
}
/**
*
* @param tableNames
* @return
*/
public SQLBuilder from(final Collection tableNames) {
if (tableNames.size() == 1) {
return from(tableNames.iterator().next().trim());
}
final String tableName = tableNames.iterator().next().trim();
return from(tableName, StringUtil.join(tableNames, WD.COMMA_SPACE));
}
/**
*
* @param entityClass
* @return
*/
public SQLBuilder from(final Class> entityClass) {
if (this.entityClass == null) {
this.entityClass = entityClass;
this.propColumnNameMap = getProp2ColumnNameMap(this.entityClass, this.namingPolicy);
}
return from(getTableName(entityClass, namingPolicy));
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public SQLBuilder from(final Class> entityClass, final String alias) {
if (this.entityClass == null) {
this.entityClass = entityClass;
this.propColumnNameMap = getProp2ColumnNameMap(this.entityClass, this.namingPolicy);
}
if (N.isNullOrEmpty(alias)) {
return from(getTableName(entityClass, namingPolicy));
} else {
return from(getTableName(entityClass, namingPolicy) + " " + alias);
}
}
private SQLBuilder from(final Class> entityClass, final Collection tableNames) {
if (this.entityClass == null) {
this.entityClass = entityClass;
this.propColumnNameMap = getProp2ColumnNameMap(this.entityClass, this.namingPolicy);
}
return from(tableNames);
}
/**
*
* @param tableName
* @param fromCause
* @return
*/
private SQLBuilder from(final String tableName, final String fromCause) {
if (op != OperationType.QUERY) {
throw new RuntimeException("Invalid operation: " + op);
}
hasFromBeenSet = true;
if (N.isNullOrEmpty(columnNames) && N.isNullOrEmpty(columnAliases) && N.isNullOrEmpty(multiSelects)) {
throw new RuntimeException("Column names or props must be set first by select");
}
int idx = tableName.indexOf(' ');
if (idx > 0) {
this.tableName = tableName.substring(0, idx).trim();
alias = tableName.substring(idx + 1).trim();
} else {
this.tableName = tableName.trim();
}
if (entityClass != null && N.notNullOrEmpty(alias)) {
addPropColumnMapForAlias(entityClass, alias);
}
sb.append(_SELECT);
sb.append(_SPACE);
if (N.notNullOrEmpty(preselect)) {
sb.append(preselect);
sb.append(_SPACE);
}
final boolean withAlias = N.notNullOrEmpty(alias);
final boolean isForSelect = op == OperationType.QUERY;
if (N.notNullOrEmpty(columnNames)) {
if (entityClass != null && withAlias == false && columnNames == getSelectPropNames(entityClass, false, null)) {
String fullSelectParts = fullSelectPartsPool.get(namingPolicy).get(entityClass);
if (N.isNullOrEmpty(fullSelectParts)) {
fullSelectParts = "";
int i = 0;
for (String columnName : columnNames) {
if (i++ > 0) {
fullSelectParts += WD.COMMA_SPACE;
}
fullSelectParts += formalizeColumnName(propColumnNameMap, columnName);
if ((namingPolicy != NamingPolicy.LOWER_CAMEL_CASE && namingPolicy != NamingPolicy.NO_CHANGE) && !WD.ASTERISK.equals(columnName)) {
fullSelectParts += SPACE_AS_SPACE;
fullSelectParts += WD.QUOTATION_D;
fullSelectParts += columnName;
fullSelectParts += WD.QUOTATION_D;
}
}
fullSelectPartsPool.get(namingPolicy).put(entityClass, fullSelectParts);
}
sb.append(fullSelectParts);
} else {
if (entityClass != null) {
final Set subEntityPropNames = getSubEntityPropNames(entityClass);
if (N.notNullOrEmpty(subEntityPropNames) && N.containsAny(subEntityPropNames, columnNames)) {
final EntityInfo entityInfo = ParserUtil.getEntityInfo(entityClass);
final List tmp = new ArrayList<>(columnNames.size() + 16);
PropInfo propInfo = null;
for (String propName : columnNames) {
if (subEntityPropNames.contains(propName)) {
propInfo = entityInfo.getPropInfo(propName);
final Collection subSelectPropNames = getSelectPropNames(
propInfo.type.isCollection() ? propInfo.type.getElementType().clazz() : propInfo.clazz, false, null);
for (String subPropName : subSelectPropNames) {
tmp.add(propName + WD.PERIOD + subPropName);
}
} else {
tmp.add(propName);
}
}
columnNames = tmp;
}
}
int i = 0;
for (String columnName : columnNames) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, columnName, null, false, null, isForSelect);
}
}
} else if (N.notNullOrEmpty(columnAliases)) {
int i = 0;
for (Map.Entry entry : columnAliases.entrySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, entry.getKey(), entry.getValue(), false, null, isForSelect);
}
} else if (N.notNullOrEmpty(multiSelects)) {
aliasPropColumnNameMap = N.newHashMap(multiSelects.size());
for (Tuple4, String, String, Set> tp : multiSelects) {
if (N.notNullOrEmpty(tp._2)) {
aliasPropColumnNameMap.put(tp._2, getProp2ColumnNameMap(tp._1, namingPolicy));
}
}
final String tmp = this.alias;
int i = 0;
for (Tuple4, String, String, Set> tp : multiSelects) {
this.alias = tp._2;
final String classAlias = tp._3;
final boolean withClassAlias = N.notNullOrEmpty(classAlias);
final ImmutableMap> eachPropColumnNameMap = getProp2ColumnNameMap(tp._1, namingPolicy);
for (String propName : getSelectPropNames(tp._1, false, tp._4)) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(eachPropColumnNameMap, propName, null, withClassAlias, classAlias, isForSelect);
}
}
this.alias = tmp;
} else {
throw new UnsupportedOperationException("No select part specified");
}
sb.append(_SPACE_FROM_SPACE);
sb.append(fromCause);
return this;
}
private void addPropColumnMapForAlias(final Class> entityClass, final String alias) {
if (aliasPropColumnNameMap == null) {
aliasPropColumnNameMap = new HashMap<>();
}
if (N.isNullOrEmpty(propColumnNameMap) && entityClass != null && ClassUtil.isEntity(entityClass)) {
propColumnNameMap = getProp2ColumnNameMap(entityClass, namingPolicy);
}
aliasPropColumnNameMap.put(alias, propColumnNameMap);
}
/**
*
* @param expr
* @return
*/
public SQLBuilder join(final String expr) {
sb.append(_SPACE_JOIN_SPACE);
sb.append(expr);
return this;
}
/**
*
* @param entityClass
* @return
*/
public SQLBuilder join(final Class> entityClass) {
sb.append(_SPACE_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy));
return this;
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public SQLBuilder join(final Class> entityClass, final String alias) {
addPropColumnMapForAlias(entityClass, alias);
sb.append(_SPACE_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy) + " " + alias);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder innerJoin(final String expr) {
sb.append(_SPACE_INNER_JOIN_SPACE);
sb.append(expr);
return this;
}
/**
*
* @param entityClass
* @return
*/
public SQLBuilder innerJoin(final Class> entityClass) {
sb.append(_SPACE_INNER_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy));
return this;
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public SQLBuilder innerJoin(final Class> entityClass, final String alias) {
addPropColumnMapForAlias(entityClass, alias);
sb.append(_SPACE_INNER_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy) + " " + alias);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder leftJoin(final String expr) {
sb.append(_SPACE_LEFT_JOIN_SPACE);
sb.append(expr);
return this;
}
/**
*
* @param entityClass
* @return
*/
public SQLBuilder leftJoin(final Class> entityClass) {
sb.append(_SPACE_LEFT_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy));
return this;
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public SQLBuilder leftJoin(final Class> entityClass, final String alias) {
addPropColumnMapForAlias(entityClass, alias);
sb.append(_SPACE_LEFT_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy) + " " + alias);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder rightJoin(final String expr) {
sb.append(_SPACE_RIGHT_JOIN_SPACE);
sb.append(expr);
return this;
}
/**
*
* @param entityClass
* @return
*/
public SQLBuilder rightJoin(final Class> entityClass) {
sb.append(_SPACE_RIGHT_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy));
return this;
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public SQLBuilder rightJoin(final Class> entityClass, final String alias) {
addPropColumnMapForAlias(entityClass, alias);
sb.append(_SPACE_RIGHT_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy) + " " + alias);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder fullJoin(final String expr) {
sb.append(_SPACE_FULL_JOIN_SPACE);
sb.append(expr);
return this;
}
/**
*
* @param entityClass
* @return
*/
public SQLBuilder fullJoin(final Class> entityClass) {
sb.append(_SPACE_FULL_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy));
return this;
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public SQLBuilder fullJoin(final Class> entityClass, final String alias) {
addPropColumnMapForAlias(entityClass, alias);
sb.append(_SPACE_FULL_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy) + " " + alias);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder crossJoin(final String expr) {
sb.append(_SPACE_CROSS_JOIN_SPACE);
sb.append(expr);
return this;
}
/**
*
* @param entityClass
* @return
*/
public SQLBuilder crossJoin(final Class> entityClass) {
sb.append(_SPACE_CROSS_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy));
return this;
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public SQLBuilder crossJoin(final Class> entityClass, final String alias) {
addPropColumnMapForAlias(entityClass, alias);
sb.append(_SPACE_CROSS_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy) + " " + alias);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder naturalJoin(final String expr) {
sb.append(_SPACE_NATURAL_JOIN_SPACE);
sb.append(expr);
return this;
}
/**
*
* @param entityClass
* @return
*/
public SQLBuilder naturalJoin(final Class> entityClass) {
sb.append(_SPACE_NATURAL_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy));
return this;
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public SQLBuilder naturalJoin(final Class> entityClass, final String alias) {
addPropColumnMapForAlias(entityClass, alias);
sb.append(_SPACE_NATURAL_JOIN_SPACE);
sb.append(getTableName(entityClass, namingPolicy) + " " + alias);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder on(final String expr) {
sb.append(_SPACE_ON_SPACE);
appendStringExpr(expr, false);
return this;
}
/**
*
* @param cond any literal written in Expression
condition won't be formalized
* @return
*/
public SQLBuilder on(final Condition cond) {
sb.append(_SPACE_ON_SPACE);
appendCondition(cond);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder using(final String expr) {
sb.append(_SPACE_USING_SPACE);
appendColumnName(expr);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder where(final String expr) {
init(true);
sb.append(_SPACE_WHERE_SPACE);
appendStringExpr(expr, false);
return this;
}
public SQLBuilder where(final Condition cond) {
init(true);
sb.append(_SPACE_WHERE_SPACE);
appendCondition(cond);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder groupBy(final String expr) {
sb.append(_SPACE_GROUP_BY_SPACE);
appendColumnName(expr);
return this;
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public final SQLBuilder groupBy(final String... columnNames) {
sb.append(_SPACE_GROUP_BY_SPACE);
for (int i = 0, len = columnNames.length; i < len; i++) {
if (i > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, columnNames[i]);
}
return this;
}
/**
*
* @param columnName
* @param direction
* @return
*/
public SQLBuilder groupBy(final String columnName, final SortDirection direction) {
groupBy(columnName);
sb.append(_SPACE);
sb.append(direction.toString());
return this;
}
/**
*
* @param columnNames
* @return
*/
public SQLBuilder groupBy(final Collection columnNames) {
sb.append(_SPACE_GROUP_BY_SPACE);
int i = 0;
for (String columnName : columnNames) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, columnName);
}
return this;
}
/**
*
* @param columnNames
* @param direction
* @return
*/
public SQLBuilder groupBy(final Collection columnNames, final SortDirection direction) {
groupBy(columnNames);
sb.append(_SPACE);
sb.append(direction.toString());
return this;
}
/**
*
* @param orders
* @return
*/
public SQLBuilder groupBy(final Map orders) {
sb.append(_SPACE_GROUP_BY_SPACE);
int i = 0;
for (Map.Entry entry : orders.entrySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, entry.getKey());
sb.append(_SPACE);
sb.append(entry.getValue().toString());
}
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder having(final String expr) {
sb.append(_SPACE_HAVING_SPACE);
appendStringExpr(expr, false);
return this;
}
/**
*
* @param cond any literal written in Expression
condition won't be formalized
* @return
*/
public SQLBuilder having(final Condition cond) {
sb.append(_SPACE_HAVING_SPACE);
appendCondition(cond);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder orderBy(final String expr) {
sb.append(_SPACE_ORDER_BY_SPACE);
appendColumnName(expr);
return this;
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public final SQLBuilder orderBy(final String... columnNames) {
sb.append(_SPACE_ORDER_BY_SPACE);
for (int i = 0, len = columnNames.length; i < len; i++) {
if (i > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, columnNames[i]);
}
return this;
}
/**
*
* @param columnName
* @param direction
* @return
*/
public SQLBuilder orderBy(final String columnName, final SortDirection direction) {
orderBy(columnName);
sb.append(_SPACE);
sb.append(direction.toString());
return this;
}
/**
*
* @param columnNames
* @return
*/
public SQLBuilder orderBy(final Collection columnNames) {
sb.append(_SPACE_ORDER_BY_SPACE);
int i = 0;
for (String columnName : columnNames) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, columnName);
}
return this;
}
/**
*
* @param columnNames
* @param direction
* @return
*/
public SQLBuilder orderBy(final Collection columnNames, final SortDirection direction) {
orderBy(columnNames);
sb.append(_SPACE);
sb.append(direction.toString());
return this;
}
/**
*
* @param orders
* @return
*/
public SQLBuilder orderBy(final Map orders) {
sb.append(_SPACE_ORDER_BY_SPACE);
int i = 0;
for (Map.Entry entry : orders.entrySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, entry.getKey());
sb.append(_SPACE);
sb.append(entry.getValue().toString());
}
return this;
}
/**
*
* @param count
* @return
*/
public SQLBuilder limit(final int count) {
sb.append(_SPACE_LIMIT_SPACE);
sb.append(count);
return this;
}
/**
*
* @param offset
* @param count
* @return
*/
public SQLBuilder limit(final int offset, final int count) {
sb.append(_SPACE_LIMIT_SPACE);
sb.append(count);
sb.append(_SPACE_OFFSET_SPACE);
sb.append(offset);
return this;
}
/**
*
* @param offset
* @return
*/
public SQLBuilder offset(final int offset) {
sb.append(_SPACE_OFFSET_SPACE);
sb.append(offset);
return this;
}
public SQLBuilder fetchNextNRowsOnly(final int count) {
sb.append(" FETCH NEXT ").append(count).append(" ROWS ONLY");
return this;
}
public SQLBuilder fetchFirstNRowsOnly(final int count) {
sb.append(" FETCH FIRST ").append(count).append(" ROWS ONLY");
return this;
}
public SQLBuilder append(final Condition cond) {
init(true);
if (cond instanceof Criteria) {
final Criteria criteria = (Criteria) cond;
final Collection joins = criteria.getJoins();
if (N.notNullOrEmpty(joins)) {
for (Join join : joins) {
sb.append(_SPACE).append(join.getOperator()).append(_SPACE);
if (join.getJoinEntities().size() == 1) {
sb.append(join.getJoinEntities().get(0));
} else {
sb.append(WD._PARENTHESES_L);
int idx = 0;
for (String joinTableName : join.getJoinEntities()) {
if (idx++ > 0) {
sb.append(_COMMA_SPACE);
}
sb.append(joinTableName);
}
sb.append(WD._PARENTHESES_R);
}
appendCondition(join.getCondition());
}
}
final Cell where = criteria.getWhere();
if ((where != null)) {
sb.append(_SPACE_WHERE_SPACE);
appendCondition(where.getCondition());
}
final Cell groupBy = criteria.getGroupBy();
if (groupBy != null) {
sb.append(_SPACE_GROUP_BY_SPACE);
appendCondition(groupBy.getCondition());
}
final Cell having = criteria.getHaving();
if (having != null) {
sb.append(_SPACE_HAVING_SPACE);
appendCondition(having.getCondition());
}
List aggregations = criteria.getAggregation();
if (N.notNullOrEmpty(aggregations)) {
for (Cell aggregation : aggregations) {
sb.append(_SPACE).append(aggregation.getOperator()).append(_SPACE);
appendCondition(aggregation.getCondition());
}
}
final Cell orderBy = criteria.getOrderBy();
if (orderBy != null) {
sb.append(_SPACE_ORDER_BY_SPACE);
appendCondition(orderBy.getCondition());
}
final Limit limit = criteria.getLimit();
if (limit != null) {
if (N.notNullOrEmpty(limit.getExpr())) {
sb.append(_SPACE).append(limit.getExpr());
} else {
if (limit.getOffset() > 0) {
limit(limit.getOffset(), limit.getCount());
} else {
limit(limit.getCount());
}
}
}
} else if (cond instanceof Clause) {
sb.append(_SPACE).append(cond.getOperator()).append(_SPACE);
appendCondition(((Clause) cond).getCondition());
} else {
if (!isForConditionOnly) {
sb.append(_SPACE_WHERE_SPACE);
}
appendCondition(cond);
}
return this;
}
public SQLBuilder append(final String expr) {
sb.append(expr);
return this;
}
/**
*
* @param sqlBuilder
* @return
*/
public SQLBuilder union(final SQLBuilder sqlBuilder) {
final String sql = sqlBuilder.sql();
if (N.notNullOrEmpty(sqlBuilder.parameters())) {
parameters.addAll(sqlBuilder.parameters());
}
return union(sql);
}
/**
*
* @param query
* @return
*/
public SQLBuilder union(final String query) {
return union(N.asArray(query));
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public final SQLBuilder union(final String... columnNames) {
op = OperationType.QUERY;
this.columnNames = Array.asList(columnNames);
this.columnAliases = null;
sb.append(_SPACE_UNION_SPACE);
// it's sub query
if (isSubQuery(columnNames)) {
sb.append(columnNames[0]);
this.columnNames = null;
} else {
// build in from method.
}
return this;
}
/**
*
* @param columnNames
* @return
*/
public SQLBuilder union(final Collection columnNames) {
op = OperationType.QUERY;
this.columnNames = columnNames;
this.columnAliases = null;
sb.append(_SPACE_UNION_SPACE);
return this;
}
/**
*
* @param sqlBuilder
* @return
*/
public SQLBuilder unionAll(final SQLBuilder sqlBuilder) {
final String sql = sqlBuilder.sql();
if (N.notNullOrEmpty(sqlBuilder.parameters())) {
parameters.addAll(sqlBuilder.parameters());
}
return unionAll(sql);
}
/**
*
* @param query
* @return
*/
public SQLBuilder unionAll(final String query) {
return unionAll(N.asArray(query));
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public final SQLBuilder unionAll(final String... columnNames) {
op = OperationType.QUERY;
this.columnNames = Array.asList(columnNames);
this.columnAliases = null;
sb.append(_SPACE_UNION_ALL_SPACE);
// it's sub query
if (isSubQuery(columnNames)) {
sb.append(columnNames[0]);
this.columnNames = null;
} else {
// build in from method.
}
return this;
}
/**
*
* @param columnNames
* @return
*/
public SQLBuilder unionAll(final Collection columnNames) {
op = OperationType.QUERY;
this.columnNames = columnNames;
this.columnAliases = null;
sb.append(_SPACE_UNION_ALL_SPACE);
return this;
}
/**
*
* @param sqlBuilder
* @return
*/
public SQLBuilder intersect(final SQLBuilder sqlBuilder) {
final String sql = sqlBuilder.sql();
if (N.notNullOrEmpty(sqlBuilder.parameters())) {
parameters.addAll(sqlBuilder.parameters());
}
return intersect(sql);
}
/**
*
* @param query
* @return
*/
public SQLBuilder intersect(final String query) {
return intersect(N.asArray(query));
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public final SQLBuilder intersect(final String... columnNames) {
op = OperationType.QUERY;
this.columnNames = Array.asList(columnNames);
this.columnAliases = null;
sb.append(_SPACE_INTERSECT_SPACE);
// it's sub query
if (isSubQuery(columnNames)) {
sb.append(columnNames[0]);
this.columnNames = null;
} else {
// build in from method.
}
return this;
}
/**
*
* @param columnNames
* @return
*/
public SQLBuilder intersect(final Collection columnNames) {
op = OperationType.QUERY;
this.columnNames = columnNames;
this.columnAliases = null;
sb.append(_SPACE_INTERSECT_SPACE);
return this;
}
/**
*
* @param sqlBuilder
* @return
*/
public SQLBuilder except(final SQLBuilder sqlBuilder) {
final String sql = sqlBuilder.sql();
if (N.notNullOrEmpty(sqlBuilder.parameters())) {
parameters.addAll(sqlBuilder.parameters());
}
return except(sql);
}
/**
*
* @param query
* @return
*/
public SQLBuilder except(final String query) {
return except(N.asArray(query));
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public final SQLBuilder except(final String... columnNames) {
op = OperationType.QUERY;
this.columnNames = Array.asList(columnNames);
this.columnAliases = null;
sb.append(_SPACE_EXCEPT_SPACE);
// it's sub query
if (isSubQuery(columnNames)) {
sb.append(columnNames[0]);
this.columnNames = null;
} else {
// build in from method.
}
return this;
}
/**
*
* @param columnNames
* @return
*/
public SQLBuilder except(final Collection columnNames) {
op = OperationType.QUERY;
this.columnNames = columnNames;
this.columnAliases = null;
sb.append(_SPACE_EXCEPT_SPACE);
return this;
}
/**
*
* @param sqlBuilder
* @return
*/
public SQLBuilder minus(final SQLBuilder sqlBuilder) {
final String sql = sqlBuilder.sql();
if (N.notNullOrEmpty(sqlBuilder.parameters())) {
parameters.addAll(sqlBuilder.parameters());
}
return minus(sql);
}
/**
*
* @param query
* @return
*/
public SQLBuilder minus(final String query) {
return minus(N.asArray(query));
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public final SQLBuilder minus(final String... columnNames) {
op = OperationType.QUERY;
this.columnNames = Array.asList(columnNames);
this.columnAliases = null;
sb.append(_SPACE_EXCEPT2_SPACE);
// it's sub query
if (isSubQuery(columnNames)) {
sb.append(columnNames[0]);
this.columnNames = null;
} else {
// build in from method.
}
return this;
}
/**
*
* @param columnNames
* @return
*/
public SQLBuilder minus(final Collection columnNames) {
op = OperationType.QUERY;
this.columnNames = columnNames;
this.columnAliases = null;
sb.append(_SPACE_EXCEPT2_SPACE);
return this;
}
public SQLBuilder forUpdate() {
sb.append(_SPACE_FOR_UPDATE);
return this;
}
/**
*
* @param expr
* @return
*/
public SQLBuilder set(final String expr) {
return set(Array.asList(expr));
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public final SQLBuilder set(final String... columnNames) {
return set(Array.asList(columnNames));
}
/**
*
* @param columnNames
* @return
*/
public SQLBuilder set(final Collection columnNames) {
init(false);
switch (sqlPolicy) {
case SQL:
case PARAMETERIZED_SQL: {
int i = 0;
for (String columnName : columnNames) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, columnName);
if (columnName.indexOf('=') < 0) {
sb.append(" = ?");
}
}
break;
}
case NAMED_SQL: {
int i = 0;
for (String columnName : columnNames) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, columnName);
if (columnName.indexOf('=') < 0) {
sb.append(" = :");
sb.append(columnName);
}
}
break;
}
case IBATIS_SQL: {
int i = 0;
for (String columnName : columnNames) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, columnName);
if (columnName.indexOf('=') < 0) {
sb.append(" = #{");
sb.append(columnName);
sb.append('}');
}
}
break;
}
default:
throw new RuntimeException("Not supported SQL policy: " + sqlPolicy);
}
this.columnNames = null;
return this;
}
/**
*
* @param props
* @return
*/
public SQLBuilder set(final Map props) {
init(false);
switch (sqlPolicy) {
case SQL: {
int i = 0;
for (Map.Entry entry : props.entrySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, entry.getKey());
sb.append(_SPACE_EQUAL_SPACE);
setParameterForSQL(entry.getValue());
}
break;
}
case PARAMETERIZED_SQL: {
int i = 0;
for (Map.Entry entry : props.entrySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, entry.getKey());
sb.append(_SPACE_EQUAL_SPACE);
setParameterForRawSQL(entry.getValue());
}
break;
}
case NAMED_SQL: {
int i = 0;
for (Map.Entry entry : props.entrySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, entry.getKey());
sb.append(_SPACE_EQUAL_SPACE);
setParameterForNamedSQL(entry.getKey(), entry.getValue());
}
break;
}
case IBATIS_SQL: {
int i = 0;
for (Map.Entry entry : props.entrySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(propColumnNameMap, entry.getKey());
sb.append(_SPACE_EQUAL_SPACE);
setParameterForIbatisNamedSQL(entry.getKey(), entry.getValue());
}
break;
}
default:
throw new RuntimeException("Not supported SQL policy: " + sqlPolicy);
}
this.columnNames = null;
return this;
}
/**
* Only the dirty properties will be set into the result SQL if the specified entity is a dirty marker entity.
*
* @param entity
* @return
*/
public SQLBuilder set(final Object entity) {
return set(entity, null);
}
/**
* Only the dirty properties will be set into the result SQL if the specified entity is a dirty marker entity.
*
* @param entity
* @param excludedPropNames
* @return
*/
public SQLBuilder set(final Object entity, final Set excludedPropNames) {
if (entity instanceof String) {
return set(N.asArray((String) entity));
} else if (entity instanceof Map) {
if (N.isNullOrEmpty(excludedPropNames)) {
return set((Map) entity);
} else {
final Map props = new LinkedHashMap<>((Map) entity);
Maps.removeKeys(props, excludedPropNames);
return set(props);
}
} else {
final Class> entityClass = entity.getClass();
this.entityClass = entityClass;
this.propColumnNameMap = getProp2ColumnNameMap(this.entityClass, this.namingPolicy);
final Collection propNames = getUpdatePropNames(entityClass, excludedPropNames);
final Set dirtyPropNames = DirtyMarkerUtil.isDirtyMarker(entityClass) ? DirtyMarkerUtil.dirtyPropNames((DirtyMarker) entity) : null;
final boolean isEmptyDirtyPropNames = N.isNullOrEmpty(dirtyPropNames);
final Map props = N.newHashMap(N.isNullOrEmpty(dirtyPropNames) ? propNames.size() : dirtyPropNames.size());
final EntityInfo entityInfo = ParserUtil.getEntityInfo(entityClass);
for (String propName : propNames) {
if (isEmptyDirtyPropNames || dirtyPropNames.contains(propName)) {
props.put(propName, entityInfo.getPropValue(entity, propName));
}
}
return set(props);
}
}
/**
*
* @param entityClass
* @return
*/
public SQLBuilder set(Class> entityClass) {
this.entityClass = entityClass;
this.propColumnNameMap = getProp2ColumnNameMap(this.entityClass, this.namingPolicy);
return set(entityClass, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public SQLBuilder set(Class> entityClass, final Set excludedPropNames) {
this.entityClass = entityClass;
this.propColumnNameMap = getProp2ColumnNameMap(this.entityClass, this.namingPolicy);
return set(getUpdatePropNames(entityClass, excludedPropNames));
}
/**
* This SQLBuilder will be closed after sql() is called.
*
* @return
*/
public String sql() {
if (sb == null) {
throw new RuntimeException("This SQLBuilder has been closed after sql() was called previously");
}
init(true);
String sql = null;
try {
sql = sb.charAt(0) == ' ' ? sb.substring(1) : sb.toString();
} finally {
Objectory.recycle(sb);
sb = null;
activeStringBuilderCounter.decrementAndGet();
}
if (logger.isDebugEnabled()) {
logger.debug(sql);
}
return sql;
}
public List parameters() {
return parameters;
}
/**
* This SQLBuilder will be closed after pair() is called.
*
* @return
*/
public SP pair() {
final String sql = sql();
return new SP(sql, parameters);
}
public T apply(final Throwables.Function super SP, T, E> func) throws E {
return func.apply(this.pair());
}
public T apply(final Throwables.BiFunction super String, ? super List, T, E> func) throws E {
final SP sp = this.pair();
return func.apply(sp.sql, sp.parameters);
}
public void accept(final Throwables.Consumer super SP, E> consumer) throws E {
consumer.accept(this.pair());
}
public void accept(final Throwables.BiConsumer super String, ? super List, E> consumer) throws E {
final SP sp = this.pair();
consumer.accept(sp.sql, sp.parameters);
}
private static final Set> namedSQLBuilderClasses = N.asSet(NSC.class, NAC.class, NLC.class);
@SuppressWarnings("rawtypes")
public com.landawn.abacus.util.AbstractPreparedQuery toPreparedQuery(final javax.sql.DataSource ds) throws java.sql.SQLException {
final SP sp = this.pair();
if (namedSQLBuilderClasses.contains(this.getClass())) {
return com.landawn.abacus.util.JdbcUtil.prepareNamedQuery(ds, sp.sql).setParameters(sp.parameters);
} else {
return com.landawn.abacus.util.JdbcUtil.prepareQuery(ds, sp.sql).setParameters(sp.parameters);
}
}
@SuppressWarnings("rawtypes")
public com.landawn.abacus.util.AbstractPreparedQuery toPreparedQuery(final java.sql.Connection conn) throws java.sql.SQLException {
final SP sp = this.pair();
if (namedSQLBuilderClasses.contains(this.getClass())) {
return com.landawn.abacus.util.JdbcUtil.prepareNamedQuery(conn, sp.sql).setParameters(sp.parameters);
} else {
return com.landawn.abacus.util.JdbcUtil.prepareQuery(conn, sp.sql).setParameters(sp.parameters);
}
}
@SuppressWarnings("rawtypes")
public Q toPreparedQuery(final Throwables.Function func)
throws java.sql.SQLException {
final SP sp = this.pair();
return func.apply(sp);
}
@SuppressWarnings("rawtypes")
public Q toPreparedQuery(
final Throwables.BiFunction super String, ? super List, Q, java.sql.SQLException> func) throws java.sql.SQLException {
final SP sp = this.pair();
return func.apply(sp.sql, sp.parameters);
}
public com.landawn.abacus.util.NamedQuery toNamedQuery(final javax.sql.DataSource ds) throws java.sql.SQLException {
requiresNamedSQLBuilder();
final SP sp = this.pair();
return com.landawn.abacus.util.JdbcUtil.prepareNamedQuery(ds, sp.sql).setParameters(sp.parameters);
}
public com.landawn.abacus.util.NamedQuery toNamedQuery(final java.sql.Connection conn) throws java.sql.SQLException {
requiresNamedSQLBuilder();
final SP sp = this.pair();
return com.landawn.abacus.util.JdbcUtil.prepareNamedQuery(conn, sp.sql).setParameters(sp.parameters);
}
private void requiresNamedSQLBuilder() {
if (!namedSQLBuilderClasses.contains(this.getClass())) {
throw new RuntimeException(
"NamedQuery must be created from named SQLBuilder: NSC/NAC/NLC. But this SQLBuilder: " + this.getClass() + " is not named SQLBuilder");
}
}
/**
*
* @param setForUpdate
*/
void init(boolean setForUpdate) {
// Note: any change, please take a look at: parse(final Class> entityClass, final Condition cond) first.
if (sb.length() > 0) {
return;
}
if (op == OperationType.UPDATE) {
sb.append(_UPDATE);
sb.append(_SPACE);
sb.append(tableName);
sb.append(_SPACE_SET_SPACE);
if (setForUpdate && N.notNullOrEmpty(columnNames)) {
set(columnNames);
}
} else if (op == OperationType.DELETE) {
final String newTableName = tableName;
char[] deleteFromTableChars = tableDeleteFrom.get(newTableName);
if (deleteFromTableChars == null) {
deleteFromTableChars = (WD.DELETE + WD.SPACE + WD.FROM + WD.SPACE + newTableName).toCharArray();
tableDeleteFrom.put(newTableName, deleteFromTableChars);
}
sb.append(deleteFromTableChars);
} else if (op == OperationType.QUERY && hasFromBeenSet == false && isForConditionOnly == false) {
throw new RuntimeException("'from' methods has not been called for query: " + op);
}
}
/**
* Sets the parameter for SQL.
*
* @param propValue the new parameter for SQL
*/
private void setParameterForSQL(final Object propValue) {
if (CF.QME.equals(propValue)) {
sb.append(WD._QUESTION_MARK);
} else if (propValue instanceof Condition) {
appendCondition((Condition) propValue);
} else {
sb.append(Expression.formalize(propValue));
}
}
/**
* Sets the parameter for raw SQL.
*
* @param propValue the new parameter for raw SQL
*/
private void setParameterForRawSQL(final Object propValue) {
if (CF.QME.equals(propValue)) {
sb.append(WD._QUESTION_MARK);
} else if (propValue instanceof Condition) {
appendCondition((Condition) propValue);
} else {
sb.append(WD._QUESTION_MARK);
parameters.add(propValue);
}
}
/**
* Sets the parameter for ibatis named SQL.
*
* @param propName
* @param propValue
*/
private void setParameterForIbatisNamedSQL(final String propName, final Object propValue) {
if (CF.QME.equals(propValue)) {
sb.append("#{");
sb.append(propName);
sb.append('}');
} else if (propValue instanceof Condition) {
appendCondition((Condition) propValue);
} else {
sb.append("#{");
sb.append(propName);
sb.append('}');
parameters.add(propValue);
}
}
/**
* Sets the parameter for named SQL.
*
* @param propName
* @param propValue
*/
private void setParameterForNamedSQL(final String propName, final Object propValue) {
if (CF.QME.equals(propValue)) {
sb.append(":");
sb.append(propName);
} else if (propValue instanceof Condition) {
appendCondition((Condition) propValue);
} else {
sb.append(":");
sb.append(propName);
parameters.add(propValue);
}
}
/**
* Sets the parameter.
*
* @param propName
* @param propValue
*/
private void setParameter(final String propName, final Object propValue) {
switch (sqlPolicy) {
case SQL: {
setParameterForSQL(propValue);
break;
}
case PARAMETERIZED_SQL: {
setParameterForRawSQL(propValue);
break;
}
case NAMED_SQL: {
setParameterForNamedSQL(propName, propValue);
break;
}
case IBATIS_SQL: {
setParameterForIbatisNamedSQL(propName, propValue);
break;
}
default:
throw new RuntimeException("Not supported SQL policy: " + sqlPolicy);
}
}
/**
* Append insert props.
*
* @param props
*/
private void appendInsertProps(final Map props) {
switch (sqlPolicy) {
case SQL: {
int i = 0;
Object propValue = null;
for (String propName : props.keySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
propValue = props.get(propName);
setParameterForSQL(propValue);
}
break;
}
case PARAMETERIZED_SQL: {
int i = 0;
Object propValue = null;
for (String propName : props.keySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
propValue = props.get(propName);
setParameterForRawSQL(propValue);
}
break;
}
case NAMED_SQL: {
int i = 0;
Object propValue = null;
for (String propName : props.keySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
propValue = props.get(propName);
setParameterForNamedSQL(propName, propValue);
}
break;
}
case IBATIS_SQL: {
int i = 0;
Object propValue = null;
for (String propName : props.keySet()) {
if (i++ > 0) {
sb.append(_COMMA_SPACE);
}
propValue = props.get(propName);
setParameterForIbatisNamedSQL(propName, propValue);
}
break;
}
default:
throw new RuntimeException("Not supported SQL policy: " + sqlPolicy);
}
}
/**
*
* @param cond
*/
private void appendCondition(final Condition cond) {
// if (sb.charAt(sb.length() - 1) != _SPACE) {
// sb.append(_SPACE);
// }
if (cond instanceof Binary) {
final Binary binary = (Binary) cond;
final String propName = binary.getPropName();
appendColumnName(propName);
sb.append(_SPACE);
sb.append(binary.getOperator().toString());
sb.append(_SPACE);
Object propValue = binary.getPropValue();
setParameter(propName, propValue);
} else if (cond instanceof Between) {
final Between bt = (Between) cond;
final String propName = bt.getPropName();
appendColumnName(propName);
sb.append(_SPACE);
sb.append(bt.getOperator().toString());
sb.append(_SPACE);
Object minValue = bt.getMinValue();
if (sqlPolicy == SQLPolicy.NAMED_SQL || sqlPolicy == SQLPolicy.IBATIS_SQL) {
setParameter("min" + StringUtil.capitalize(propName), minValue);
} else {
setParameter(propName, minValue);
}
sb.append(_SPACE);
sb.append(WD.AND);
sb.append(_SPACE);
Object maxValue = bt.getMaxValue();
if (sqlPolicy == SQLPolicy.NAMED_SQL || sqlPolicy == SQLPolicy.IBATIS_SQL) {
setParameter("max" + StringUtil.capitalize(propName), maxValue);
} else {
setParameter(propName, maxValue);
}
} else if (cond instanceof In) {
final In in = (In) cond;
final String propName = in.getPropName();
final List parameters = in.getParameters();
appendColumnName(propName);
sb.append(_SPACE);
sb.append(in.getOperator().toString());
sb.append(WD.SPACE_PARENTHESES_L);
for (int i = 0, len = parameters.size(); i < len; i++) {
if (i > 0) {
sb.append(WD.COMMA_SPACE);
}
if (sqlPolicy == SQLPolicy.NAMED_SQL || sqlPolicy == SQLPolicy.IBATIS_SQL) {
setParameter(propName + (i + 1), parameters.get(i));
} else {
setParameter(propName, parameters.get(i));
}
}
sb.append(WD._PARENTHESES_R);
} else if (cond instanceof InSubQuery) {
final InSubQuery inSubQuery = (InSubQuery) cond;
final String propName = inSubQuery.getPropName();
if (N.notNullOrEmpty(propName)) {
appendColumnName(propName);
} else {
sb.append(WD._PARENTHESES_L);
int idx = 0;
for (String e : inSubQuery.getPropNames()) {
if (idx++ > 0) {
sb.append(_COMMA_SPACE);
}
appendColumnName(e);
}
sb.append(WD._PARENTHESES_R);
}
sb.append(_SPACE);
sb.append(inSubQuery.getOperator().toString());
sb.append(WD.SPACE_PARENTHESES_L);
appendCondition(inSubQuery.getSubQuery());
sb.append(WD._PARENTHESES_R);
} else if (cond instanceof NotIn) {
final NotIn notIn = (NotIn) cond;
final String propName = notIn.getPropName();
final List parameters = notIn.getParameters();
appendColumnName(propName);
sb.append(_SPACE);
sb.append(notIn.getOperator().toString());
sb.append(WD.SPACE_PARENTHESES_L);
for (int i = 0, len = parameters.size(); i < len; i++) {
if (i > 0) {
sb.append(WD.COMMA_SPACE);
}
if (sqlPolicy == SQLPolicy.NAMED_SQL || sqlPolicy == SQLPolicy.IBATIS_SQL) {
setParameter(propName + (i + 1), parameters.get(i));
} else {
setParameter(propName, parameters.get(i));
}
}
sb.append(WD._PARENTHESES_R);
} else if (cond instanceof NotInSubQuery) {
final NotInSubQuery notInSubQuery = (NotInSubQuery) cond;
final String propName = notInSubQuery.getPropName();
appendColumnName(propName);
sb.append(_SPACE);
sb.append(notInSubQuery.getOperator().toString());
sb.append(WD.SPACE_PARENTHESES_L);
appendCondition(notInSubQuery.getSubQuery());
sb.append(WD._PARENTHESES_R);
} else if (cond instanceof Where || cond instanceof Having) {
final Cell cell = (Cell) cond;
sb.append(_SPACE);
sb.append(cell.getOperator().toString());
sb.append(_SPACE);
appendCondition(cell.getCondition());
} else if (cond instanceof Cell) {
final Cell cell = (Cell) cond;
sb.append(_SPACE);
sb.append(cell.getOperator().toString());
sb.append(_SPACE);
sb.append(_PARENTHESES_L);
appendCondition(cell.getCondition());
sb.append(_PARENTHESES_R);
} else if (cond instanceof Junction) {
final Junction junction = (Junction) cond;
final List conditionList = junction.getConditions();
if (N.isNullOrEmpty(conditionList)) {
throw new IllegalArgumentException("The junction condition(" + junction.getOperator().toString() + ") doesn't include any element.");
}
if (conditionList.size() == 1) {
appendCondition(conditionList.get(0));
} else {
// TODO ((id = :id) AND (gui = :gui)) is not support in Cassandra.
// only (id = :id) AND (gui = :gui) works.
// sb.append(_PARENTHESES_L);
for (int i = 0, size = conditionList.size(); i < size; i++) {
if (i > 0) {
sb.append(_SPACE);
sb.append(junction.getOperator().toString());
sb.append(_SPACE);
}
sb.append(_PARENTHESES_L);
appendCondition(conditionList.get(i));
sb.append(_PARENTHESES_R);
}
// sb.append(_PARENTHESES_R);
}
} else if (cond instanceof SubQuery) {
final SubQuery subQuery = (SubQuery) cond;
final Condition subCond = subQuery.getCondition();
if (N.notNullOrEmpty(subQuery.getSql())) {
sb.append(subQuery.getSql());
} else {
if (subQuery.getEntityClass() != null) {
if (this instanceof SCSB) {
sb.append(SCSB.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof PSC) {
sb.append(PSC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof MSC) {
sb.append(MSC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof NSC) {
sb.append(NSC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof ACSB) {
sb.append(ACSB.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof PAC) {
sb.append(PAC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof MAC) {
sb.append(MAC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof NAC) {
sb.append(NAC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof LCSB) {
sb.append(LCSB.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof PLC) {
sb.append(PLC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof MLC) {
sb.append(MLC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else if (this instanceof NLC) {
sb.append(NLC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityClass()).append(subCond).sql());
} else {
throw new RuntimeException("Unsupproted subQuery condition: " + cond);
}
} else {
if (this instanceof SCSB) {
sb.append(SCSB.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof PSC) {
sb.append(PSC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof MSC) {
sb.append(MSC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof NSC) {
sb.append(NSC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof ACSB) {
sb.append(ACSB.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof PAC) {
sb.append(PAC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof MAC) {
sb.append(MAC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof NAC) {
sb.append(NAC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof LCSB) {
sb.append(LCSB.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof PLC) {
sb.append(PLC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof MLC) {
sb.append(MLC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else if (this instanceof NLC) {
sb.append(NLC.select(subQuery.getSelectPropNames()).from(subQuery.getEntityName()).append(subCond).sql());
} else {
throw new RuntimeException("Unsupproted subQuery condition: " + cond);
}
}
}
} else if (cond instanceof Expression) {
// ==== version 1
// sb.append(cond.toString());
// ==== version 2
// final List words = SQLParser.parse(((Expression) cond).getLiteral());
// final Map propColumnNameMap = getPropColumnNameMap(entityClass, namingPolicy);
//
// String word = null;
//
// for (int i = 0, size = words.size(); i < size; i++) {
// word = words.get(i);
//
// if ((i > 2) && WD.AS.equalsIgnoreCase(words.get(i - 2))) {
// sb.append(word);
// } else if ((i > 1) && WD.SPACE.equalsIgnoreCase(words.get(i - 1))
// && (propColumnNameMap.containsKey(words.get(i - 2)) || propColumnNameMap.containsValue(words.get(i - 2)))) {
// sb.append(word);
// } else {
// sb.append(formalizeColumnName(propColumnNameMap, word));
// }
// }
// ==== version 3
appendStringExpr(((Expression) cond).getLiteral(), false);
} else {
throw new IllegalArgumentException("Unsupported condtion: " + cond.toString());
}
}
private void appendStringExpr(final String expr, final boolean isFromAppendColumn) {
// TODO performance improvement.
if (expr.length() < 16) {
boolean allChars = true;
char ch = 0;
for (int i = 0, len = expr.length(); i < len; i++) {
ch = expr.charAt(i);
// https://www.sciencebuddies.org/science-fair-projects/references/ascii-table
if (ch < 'A' || (ch > 'Z' && ch < '_') || ch > 'z') {
allChars = false;
break;
}
}
if (allChars) {
if (isFromAppendColumn) {
sb.append(formalizeColumnName(expr, namingPolicy));
} else {
sb.append(formalizeColumnName(propColumnNameMap, expr));
}
return;
}
}
final List words = SQLParser.parse(expr);
String word = null;
for (int i = 0, len = words.size(); i < len; i++) {
word = words.get(i);
if (!StringUtil.isAsciiAlpha(word.charAt(0))) {
sb.append(word);
} else if (SQLParser.isFunctionName(words, len, i)) {
sb.append(word);
} else {
sb.append(formalizeColumnName(propColumnNameMap, word));
}
}
}
private void appendColumnName(final String propName) {
appendColumnName(propColumnNameMap, propName);
}
private void appendColumnName(final ImmutableMap> propColumnNameMap, final String propName) {
appendColumnName(propColumnNameMap, propName, null, false, null, false);
}
private void appendColumnName(final ImmutableMap> propColumnNameMap, final String propName, final String propAlias,
final boolean withClassAlias, final String classAlias, final boolean isForSelect) {
Tuple2 tp = propColumnNameMap == null ? null : propColumnNameMap.get(propName);
if (tp != null) {
if (tp._2.booleanValue() && alias != null && alias.length() > 0) {
sb.append(alias).append(WD._PERIOD);
}
sb.append(tp._1);
if (isForSelect && namingPolicy != NamingPolicy.NO_CHANGE) {
sb.append(_SPACE_AS_SPACE);
sb.append(WD._QUOTATION_D);
if (withClassAlias) {
sb.append(classAlias).append(WD._PERIOD);
}
sb.append(N.notNullOrEmpty(propAlias) ? propAlias : propName);
sb.append(WD._QUOTATION_D);
}
return;
} else {
if (aliasPropColumnNameMap != null && aliasPropColumnNameMap.size() > 0) {
int index = propName.indexOf('.');
if (index > 0) {
final String propTableAlias = propName.substring(0, index);
final Map> newPropColumnNameMap = aliasPropColumnNameMap.get(propTableAlias);
if (newPropColumnNameMap != null) {
final String newPropName = propName.substring(index + 1);
tp = newPropColumnNameMap.get(newPropName);
if (tp != null) {
sb.append(propTableAlias).append('.').append(tp._1);
if (isForSelect && namingPolicy != NamingPolicy.NO_CHANGE) {
sb.append(_SPACE_AS_SPACE);
sb.append(WD._QUOTATION_D);
if (withClassAlias) {
sb.append(classAlias).append(WD._PERIOD);
}
sb.append(N.notNullOrEmpty(propAlias) ? propAlias : propName);
sb.append(WD._QUOTATION_D);
}
return;
}
}
}
}
}
if (N.notNullOrEmpty(propAlias)) {
appendStringExpr(propName, true);
if (isForSelect && namingPolicy != NamingPolicy.NO_CHANGE) {
sb.append(_SPACE_AS_SPACE);
sb.append(WD._QUOTATION_D);
if (withClassAlias) {
sb.append(classAlias).append(WD._PERIOD);
}
sb.append(propAlias);
sb.append(WD._QUOTATION_D);
}
} else {
if (isForSelect && namingPolicy != NamingPolicy.NO_CHANGE) {
int index = propName.indexOf(" AS ");
if (index < 0) {
index = propName.indexOf(" as ");
}
if (index > 0) {
appendColumnName(propColumnNameMap, propName.substring(0, index).trim(), propName.substring(index + 4).trim(), withClassAlias, classAlias,
isForSelect);
} else {
appendStringExpr(propName, true);
if (propName.charAt(propName.length() - 1) != '*') {
sb.append(_SPACE_AS_SPACE);
sb.append(WD._QUOTATION_D);
if (withClassAlias) {
sb.append(classAlias).append(WD._PERIOD);
}
sb.append(propName);
sb.append(WD._QUOTATION_D);
}
}
} else {
appendStringExpr(propName, true);
}
}
}
/**
* Checks if is sub query.
*
* @param columnNames
* @return true, if is sub query
*/
private boolean isSubQuery(final String... columnNames) {
if (columnNames.length == 1) {
int index = SQLParser.indexWord(columnNames[0], WD.SELECT, 0, false);
if (index >= 0) {
index = SQLParser.indexWord(columnNames[0], WD.FROM, index, false);
return index >= 1;
}
}
return false;
}
// @Override
// public int hashCode() {
// return sb.hashCode();
// }
//
// @Override
// public boolean equals(Object obj) {
// if (obj == this) {
// return true;
// }
//
// if (obj instanceof SQLBuilder) {
// final SQLBuilder other = (SQLBuilder) obj;
//
// return N.equals(this.sb, other.sb) && N.equals(this.parameters, other.parameters);
// }
//
// return false;
// }
public void println() {
N.println(sql());
}
@Override
public String toString() {
return sql();
}
static String formalizeColumnName(final String word, final NamingPolicy namingPolicy) {
if (sqlKeyWords.contains(word) || namingPolicy == NamingPolicy.NO_CHANGE) {
return word;
} else if (namingPolicy == NamingPolicy.LOWER_CAMEL_CASE) {
return ClassUtil.formalizePropName(word);
} else {
return namingPolicy.convert(word);
}
}
private String formalizeColumnName(final ImmutableMap> propColumnNameMap, final String propName) {
Tuple2 tp = propColumnNameMap == null ? null : propColumnNameMap.get(propName);
if (tp != null) {
if (tp._2.booleanValue() && alias != null && alias.length() > 0) {
return alias + "." + tp._1;
} else {
return tp._1;
}
}
if (aliasPropColumnNameMap != null && aliasPropColumnNameMap.size() > 0) {
int index = propName.indexOf('.');
if (index > 0) {
final String propTableAlias = propName.substring(0, index);
final Map> newPropColumnNameMap = aliasPropColumnNameMap.get(propTableAlias);
if (newPropColumnNameMap != null) {
final String newPropName = propName.substring(index + 1);
tp = newPropColumnNameMap.get(newPropName);
if (tp != null) {
return propTableAlias + "." + tp._1;
}
}
}
}
return formalizeColumnName(propName, namingPolicy);
}
private static void parseInsertEntity(final SQLBuilder instance, final Object entity, final Set excludedPropNames) {
if (entity instanceof String) {
instance.columnNames = Array.asList((String) entity);
} else if (entity instanceof Map) {
if (N.isNullOrEmpty(excludedPropNames)) {
instance.props = (Map) entity;
} else {
instance.props = new LinkedHashMap<>((Map) entity);
Maps.removeKeys(instance.props, excludedPropNames);
}
} else {
final Collection propNames = getInsertPropNames(entity, excludedPropNames);
final Map map = N.newHashMap(propNames.size());
final EntityInfo entityInfo = ParserUtil.getEntityInfo(entity.getClass());
for (String propName : propNames) {
map.put(propName, entityInfo.getPropValue(entity, propName));
}
instance.props = map;
}
}
private static Collection> toInsertPropsList(final Collection> propsList) {
final Optional> first = N.firstNonNull(propsList);
if (first.isPresent() && first.get() instanceof Map) {
return (List>) propsList;
} else {
final Class> entityClass = first.get().getClass();
final Collection propNames = getInsertPropNames(entityClass, null);
final List> newPropsList = new ArrayList<>(propsList.size());
for (Object entity : propsList) {
final Map props = N.newHashMap(propNames.size());
final EntityInfo entityInfo = ParserUtil.getEntityInfo(entityClass);
for (String propName : propNames) {
props.put(propName, entityInfo.getPropValue(entity, propName));
}
newPropsList.add(props);
}
return newPropsList;
}
}
static void checkMultiSelects(final List, String, String, Set>> multiSelects) {
N.checkArgNotNullOrEmpty(multiSelects, "multiSelects");
for (Tuple4, String, String, Set> tp : multiSelects) {
N.checkArgNotNull(tp._1, "Class can't be null in 'multiSelects'");
}
}
enum SQLPolicy {
SQL, PARAMETERIZED_SQL, NAMED_SQL, IBATIS_SQL;
}
/**
* Un-parameterized SQL builder with snake case (lower case with underscore) field/column naming strategy.
*
* For example:
*
*
* SCSB.select("firstName", "lastName").from("account").where(CF.eq("id", 1)).sql();
* // Output: SELECT first_name AS "firstName", last_name AS "lastName" FROM account WHERE id = 1
*
*
*
* @deprecated {@code PSC or NSC} is preferred.
*/
@Deprecated
public static class SCSB extends SQLBuilder {
SCSB() {
super(NamingPolicy.LOWER_CASE_WITH_UNDERSCORE, SQLPolicy.SQL);
}
static SCSB createInstance() {
return new SCSB();
}
/**
* To generate {@code sql} part for the specified {@code cond} only.
*
* @param cond
* @param entityClass
* @return
*/
public static SQLBuilder parse(final Condition cond, final Class> entityClass) {
N.checkArgNotNull(cond, "cond");
final SQLBuilder instance = createInstance();
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.op = OperationType.QUERY;
instance.isForConditionOnly = true;
instance.append(cond);
return instance;
}
/**
*
* @param expr
* @return
*/
public static SQLBuilder insert(final String expr) {
return insert(N.asArray(expr));
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public static SQLBuilder insert(final String... columnNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.columnNames = Array.asList(columnNames);
return instance;
}
/**
*
* @param columnNames
* @return
*/
public static SQLBuilder insert(final Collection columnNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.columnNames = columnNames;
return instance;
}
/**
*
* @param props
* @return
*/
public static SQLBuilder insert(final Map props) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.props = props;
return instance;
}
/**
*
* @param entity
* @return
*/
public static SQLBuilder insert(final Object entity) {
return insert(entity, null);
}
/**
*
* @param entity
* @param excludedPropNames
* @return
*/
public static SQLBuilder insert(final Object entity, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.entityClass = entity.getClass();
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
parseInsertEntity(instance, entity, excludedPropNames);
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder insert(final Class> entityClass) {
return insert(entityClass, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder insert(final Class> entityClass, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.columnNames = getInsertPropNames(entityClass, excludedPropNames);
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder insertInto(final Class> entityClass) {
return insertInto(entityClass, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder insertInto(final Class> entityClass, final Set excludedPropNames) {
return insert(entityClass, excludedPropNames).into(entityClass);
}
/**
* Generate the MySQL style batch insert sql.
*
* @param propsList list of entity or properties map.
* @return
*/
@Beta
public static SQLBuilder batchInsert(final Collection> propsList) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
final Optional> first = N.firstNonNull(propsList);
if (first.isPresent() && ClassUtil.isEntity(first.get().getClass())) {
instance.entityClass = first.get().getClass();
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
}
instance.propsList = toInsertPropsList(propsList);
return instance;
}
/**
*
* @param tableName
* @return
*/
public static SQLBuilder update(final String tableName) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.UPDATE;
instance.tableName = tableName;
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder update(final Class> entityClass) {
return update(entityClass, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder update(final Class> entityClass, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.UPDATE;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.tableName = getTableName(entityClass, NamingPolicy.LOWER_CASE_WITH_UNDERSCORE);
instance.columnNames = getUpdatePropNames(entityClass, excludedPropNames);
return instance;
}
/**
*
* @param tableName
* @return
*/
public static SQLBuilder deleteFrom(final String tableName) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.DELETE;
instance.tableName = tableName;
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder deleteFrom(final Class> entityClass) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.DELETE;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.tableName = getTableName(entityClass, NamingPolicy.LOWER_CASE_WITH_UNDERSCORE);
return instance;
}
/**
*
* @param selectPart
* @return
*/
public static SQLBuilder select(final String selectPart) {
N.checkArgNotNullOrEmpty(selectPart, "selectPart");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnNames = Array.asList(selectPart);
return instance;
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public static SQLBuilder select(final String... columnNames) {
N.checkArgNotNullOrEmpty(columnNames, "columnNames");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnNames = Array.asList(columnNames);
return instance;
}
/**
*
* @param columnNames
* @return
*/
public static SQLBuilder select(final Collection columnNames) {
N.checkArgNotNullOrEmpty(columnNames, "columnNames");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnNames = columnNames;
return instance;
}
/**
*
* @param columnAliases
* @return
*/
public static SQLBuilder select(final Map columnAliases) {
N.checkArgNotNullOrEmpty(columnAliases, "columnAliases");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnAliases = columnAliases;
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder select(final Class> entityClass) {
return select(entityClass, false);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @return
*/
public static SQLBuilder select(final Class> entityClass, final boolean includeSubEntityProperties) {
return select(entityClass, includeSubEntityProperties, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder select(final Class> entityClass, final Set excludedPropNames) {
return select(entityClass, false, excludedPropNames);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @param excludedPropNames
* @return
*/
public static SQLBuilder select(final Class> entityClass, final boolean includeSubEntityProperties, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.columnNames = getSelectPropNames(entityClass, includeSubEntityProperties, excludedPropNames);
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass) {
return selectFrom(entityClass, false);
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias) {
return selectFrom(entityClass, alias, false);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final boolean includeSubEntityProperties) {
return selectFrom(entityClass, includeSubEntityProperties, null);
}
/**
*
* @param entityClass
* @param alias
* @param includeSubEntityProperties
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias, final boolean includeSubEntityProperties) {
return selectFrom(entityClass, alias, includeSubEntityProperties, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final Set excludedPropNames) {
return selectFrom(entityClass, false, excludedPropNames);
}
/**
*
* @param entityClass
* @param alias
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias, final Set excludedPropNames) {
return selectFrom(entityClass, alias, false, excludedPropNames);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final boolean includeSubEntityProperties, final Set excludedPropNames) {
return selectFrom(entityClass, null, includeSubEntityProperties, excludedPropNames);
}
/**
*
* @param entityClass
* @param alias
* @param includeSubEntityProperties
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias, final boolean includeSubEntityProperties,
final Set excludedPropNames) {
if (includeSubEntityProperties) {
final List selectTableNames = getSelectTableNames(entityClass, alias, excludedPropNames, NamingPolicy.LOWER_CASE_WITH_UNDERSCORE);
return select(entityClass, includeSubEntityProperties, excludedPropNames).from(entityClass, selectTableNames);
} else {
return select(entityClass, includeSubEntityProperties, excludedPropNames).from(entityClass, alias);
}
}
public static SQLBuilder select(final Class> entityClassA, final String tableAliasA, final String classAliasA, final Class> entityClassB,
final String tableAliasB, final String classAliasB) {
return select(entityClassA, tableAliasA, classAliasA, null, entityClassB, tableAliasB, classAliasB, null);
}
public static SQLBuilder select(final Class> entityClassA, final String tableAliasA, final String classAliasA, final Set excludedPropNamesA,
final Class> entityClassB, final String tableAliasB, final String classAliasB, final Set excludedPropNamesB) {
final List, String, String, Set>> multiSelects = N.asList(
Tuple., String, String, Set> of(entityClassA, tableAliasA, classAliasA, excludedPropNamesA),
Tuple., String, String, Set> of(entityClassB, tableAliasB, classAliasB, excludedPropNamesB));
return select(multiSelects);
}
public static SQLBuilder select(final List, String, String, Set>> multiSelects) {
checkMultiSelects(multiSelects);
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.entityClass = multiSelects.get(0)._1;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.multiSelects = multiSelects;
return instance;
}
public static SQLBuilder selectFrom(final Class> entityClassA, final String tableAliasA, final String classAliasA, final Class> entityClassB,
final String tableAliasB, final String classAliasB) {
return selectFrom(entityClassA, tableAliasA, classAliasA, null, entityClassB, tableAliasB, classAliasB, null);
}
public static SQLBuilder selectFrom(final Class> entityClassA, final String tableAliasA, final String classAliasA,
final Set excludedPropNamesA, final Class> entityClassB, final String tableAliasB, final String classAliasB,
final Set excludedPropNamesB) {
final List, String, String, Set>> multiSelects = N.asList(
Tuple., String, String, Set> of(entityClassA, tableAliasA, classAliasA, excludedPropNamesA),
Tuple., String, String, Set> of(entityClassB, tableAliasB, classAliasB, excludedPropNamesB));
return selectFrom(multiSelects);
}
public static SQLBuilder selectFrom(final List, String, String, Set>> multiSelects) {
checkMultiSelects(multiSelects);
final NamingPolicy namingPolicy = NamingPolicy.LOWER_CASE_WITH_UNDERSCORE;
final StringBuilder sb = Objectory.createStringBuilder();
int idx = 0;
for (Tuple4, String, String, Set> tp : multiSelects) {
if (idx++ > 0) {
sb.append(_COMMA_SPACE);
}
sb.append(getTableName(tp._1, namingPolicy));
if (N.notNullOrEmpty(tp._2)) {
sb.append(' ').append(tp._2);
}
}
String fromClause = sb.toString();
Objectory.recycle(sb);
return select(multiSelects).from(fromClause);
}
}
/**
* Un-parameterized SQL builder with all capitals case (upper case with underscore) field/column naming strategy.
*
* For example:
*
*
* N.println(ACSB.select("firstName", "lastName").from("account").where(CF.eq("id", 1)).sql());
* // Output: SELECT FIRST_NAME AS "firstName", LAST_NAME AS "lastName" FROM ACCOUNT WHERE ID = 1
*
*
*
* @deprecated {@code PAC or NAC} is preferred.
*/
@Deprecated
public static class ACSB extends SQLBuilder {
ACSB() {
super(NamingPolicy.UPPER_CASE_WITH_UNDERSCORE, SQLPolicy.SQL);
}
static ACSB createInstance() {
return new ACSB();
}
/**
* To generate {@code sql} part for the specified {@code cond} only.
*
* @param cond
* @param entityClass
* @return
*/
public static SQLBuilder parse(final Condition cond, final Class> entityClass) {
N.checkArgNotNull(cond, "cond");
final SQLBuilder instance = createInstance();
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.op = OperationType.QUERY;
instance.isForConditionOnly = true;
instance.append(cond);
return instance;
}
/**
*
* @param expr
* @return
*/
public static SQLBuilder insert(final String expr) {
return insert(N.asArray(expr));
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public static SQLBuilder insert(final String... columnNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.columnNames = Array.asList(columnNames);
return instance;
}
/**
*
* @param columnNames
* @return
*/
public static SQLBuilder insert(final Collection columnNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.columnNames = columnNames;
return instance;
}
/**
*
* @param props
* @return
*/
public static SQLBuilder insert(final Map props) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.props = props;
return instance;
}
/**
*
* @param entity
* @return
*/
public static SQLBuilder insert(final Object entity) {
return insert(entity, null);
}
/**
*
* @param entity
* @param excludedPropNames
* @return
*/
public static SQLBuilder insert(final Object entity, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.entityClass = entity.getClass();
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
parseInsertEntity(instance, entity, excludedPropNames);
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder insert(final Class> entityClass) {
return insert(entityClass, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder insert(final Class> entityClass, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.columnNames = getInsertPropNames(entityClass, excludedPropNames);
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder insertInto(final Class> entityClass) {
return insertInto(entityClass, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder insertInto(final Class> entityClass, final Set excludedPropNames) {
return insert(entityClass, excludedPropNames).into(entityClass);
}
/**
* Generate the MySQL style batch insert sql.
*
* @param propsList list of entity or properties map.
* @return
*/
@Beta
public static SQLBuilder batchInsert(final Collection> propsList) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
final Optional> first = N.firstNonNull(propsList);
if (first.isPresent() && ClassUtil.isEntity(first.get().getClass())) {
instance.entityClass = first.get().getClass();
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
}
instance.propsList = toInsertPropsList(propsList);
return instance;
}
/**
*
* @param tableName
* @return
*/
public static SQLBuilder update(final String tableName) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.UPDATE;
instance.tableName = tableName;
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder update(final Class> entityClass) {
return update(entityClass, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder update(final Class> entityClass, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.UPDATE;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.tableName = getTableName(entityClass, NamingPolicy.UPPER_CASE_WITH_UNDERSCORE);
instance.columnNames = getUpdatePropNames(entityClass, excludedPropNames);
return instance;
}
/**
*
* @param tableName
* @return
*/
public static SQLBuilder deleteFrom(final String tableName) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.DELETE;
instance.tableName = tableName;
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder deleteFrom(final Class> entityClass) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.DELETE;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.tableName = getTableName(entityClass, NamingPolicy.UPPER_CASE_WITH_UNDERSCORE);
return instance;
}
/**
*
* @param selectPart
* @return
*/
public static SQLBuilder select(final String selectPart) {
N.checkArgNotNullOrEmpty(selectPart, "selectPart");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnNames = Array.asList(selectPart);
return instance;
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public static SQLBuilder select(final String... columnNames) {
N.checkArgNotNullOrEmpty(columnNames, "columnNames");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnNames = Array.asList(columnNames);
return instance;
}
/**
*
* @param columnNames
* @return
*/
public static SQLBuilder select(final Collection columnNames) {
N.checkArgNotNullOrEmpty(columnNames, "columnNames");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnNames = columnNames;
return instance;
}
/**
*
* @param columnAliases
* @return
*/
public static SQLBuilder select(final Map columnAliases) {
N.checkArgNotNullOrEmpty(columnAliases, "columnAliases");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnAliases = columnAliases;
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder select(final Class> entityClass) {
return select(entityClass, false);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @return
*/
public static SQLBuilder select(final Class> entityClass, final boolean includeSubEntityProperties) {
return select(entityClass, includeSubEntityProperties, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder select(final Class> entityClass, final Set excludedPropNames) {
return select(entityClass, false, excludedPropNames);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @param excludedPropNames
* @return
*/
public static SQLBuilder select(final Class> entityClass, final boolean includeSubEntityProperties, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.columnNames = getSelectPropNames(entityClass, includeSubEntityProperties, excludedPropNames);
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass) {
return selectFrom(entityClass, false);
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias) {
return selectFrom(entityClass, alias, false);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final boolean includeSubEntityProperties) {
return selectFrom(entityClass, includeSubEntityProperties, null);
}
/**
*
* @param entityClass
* @param alias
* @param includeSubEntityProperties
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias, final boolean includeSubEntityProperties) {
return selectFrom(entityClass, alias, includeSubEntityProperties, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final Set excludedPropNames) {
return selectFrom(entityClass, false, excludedPropNames);
}
/**
*
* @param entityClass
* @param alias
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias, final Set excludedPropNames) {
return selectFrom(entityClass, alias, false, excludedPropNames);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final boolean includeSubEntityProperties, final Set excludedPropNames) {
return selectFrom(entityClass, null, includeSubEntityProperties, excludedPropNames);
}
/**
*
* @param entityClass
* @param alias
* @param includeSubEntityProperties
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias, final boolean includeSubEntityProperties,
final Set excludedPropNames) {
if (includeSubEntityProperties) {
final List selectTableNames = getSelectTableNames(entityClass, alias, excludedPropNames, NamingPolicy.UPPER_CASE_WITH_UNDERSCORE);
return select(entityClass, includeSubEntityProperties, excludedPropNames).from(entityClass, selectTableNames);
} else {
return select(entityClass, includeSubEntityProperties, excludedPropNames).from(entityClass, alias);
}
}
public static SQLBuilder select(final Class> entityClassA, final String tableAliasA, final String classAliasA, final Class> entityClassB,
final String tableAliasB, final String classAliasB) {
return select(entityClassA, tableAliasA, classAliasA, null, entityClassB, tableAliasB, classAliasB, null);
}
public static SQLBuilder select(final Class> entityClassA, final String tableAliasA, final String classAliasA, final Set excludedPropNamesA,
final Class> entityClassB, final String tableAliasB, final String classAliasB, final Set excludedPropNamesB) {
final List, String, String, Set>> multiSelects = N.asList(
Tuple., String, String, Set> of(entityClassA, tableAliasA, classAliasA, excludedPropNamesA),
Tuple., String, String, Set> of(entityClassB, tableAliasB, classAliasB, excludedPropNamesB));
return select(multiSelects);
}
public static SQLBuilder select(final List, String, String, Set>> multiSelects) {
checkMultiSelects(multiSelects);
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.entityClass = multiSelects.get(0)._1;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.multiSelects = multiSelects;
return instance;
}
public static SQLBuilder selectFrom(final Class> entityClassA, final String tableAliasA, final String classAliasA, final Class> entityClassB,
final String tableAliasB, final String classAliasB) {
return selectFrom(entityClassA, tableAliasA, classAliasA, null, entityClassB, tableAliasB, classAliasB, null);
}
public static SQLBuilder selectFrom(final Class> entityClassA, final String tableAliasA, final String classAliasA,
final Set excludedPropNamesA, final Class> entityClassB, final String tableAliasB, final String classAliasB,
final Set excludedPropNamesB) {
final List, String, String, Set>> multiSelects = N.asList(
Tuple., String, String, Set> of(entityClassA, tableAliasA, classAliasA, excludedPropNamesA),
Tuple., String, String, Set> of(entityClassB, tableAliasB, classAliasB, excludedPropNamesB));
return selectFrom(multiSelects);
}
public static SQLBuilder selectFrom(final List, String, String, Set>> multiSelects) {
checkMultiSelects(multiSelects);
final NamingPolicy namingPolicy = NamingPolicy.UPPER_CASE_WITH_UNDERSCORE;
final StringBuilder sb = Objectory.createStringBuilder();
int idx = 0;
for (Tuple4, String, String, Set> tp : multiSelects) {
if (idx++ > 0) {
sb.append(_COMMA_SPACE);
}
sb.append(getTableName(tp._1, namingPolicy));
if (N.notNullOrEmpty(tp._2)) {
sb.append(' ').append(tp._2);
}
}
String fromClause = sb.toString();
Objectory.recycle(sb);
return select(multiSelects).from(fromClause);
}
}
/**
* Un-parameterized SQL builder with lower camel case field/column naming strategy.
*
* For example:
*
*
* N.println(LCSB.select("firstName", "lastName").from("account").where(CF.eq("id", 1)).sql());
* // SELECT firstName, lastName FROM account WHERE id = 1
*
*
*
* @deprecated {@code PLC or NLC} is preferred.
*/
@Deprecated
public static class LCSB extends SQLBuilder {
LCSB() {
super(NamingPolicy.LOWER_CAMEL_CASE, SQLPolicy.SQL);
}
static LCSB createInstance() {
return new LCSB();
}
/**
* To generate {@code sql} part for the specified {@code cond} only.
*
* @param cond
* @param entityClass
* @return
*/
public static SQLBuilder parse(final Condition cond, final Class> entityClass) {
N.checkArgNotNull(cond, "cond");
final SQLBuilder instance = createInstance();
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.op = OperationType.QUERY;
instance.isForConditionOnly = true;
instance.append(cond);
return instance;
}
/**
*
* @param expr
* @return
*/
public static SQLBuilder insert(final String expr) {
return insert(N.asArray(expr));
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public static SQLBuilder insert(final String... columnNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.columnNames = Array.asList(columnNames);
return instance;
}
/**
*
* @param columnNames
* @return
*/
public static SQLBuilder insert(final Collection columnNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.columnNames = columnNames;
return instance;
}
/**
*
* @param props
* @return
*/
public static SQLBuilder insert(final Map props) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.props = props;
return instance;
}
/**
*
* @param entity
* @return
*/
public static SQLBuilder insert(final Object entity) {
return insert(entity, null);
}
/**
*
* @param entity
* @param excludedPropNames
* @return
*/
public static SQLBuilder insert(final Object entity, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.entityClass = entity.getClass();
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
parseInsertEntity(instance, entity, excludedPropNames);
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder insert(final Class> entityClass) {
return insert(entityClass, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder insert(final Class> entityClass, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.columnNames = getInsertPropNames(entityClass, excludedPropNames);
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder insertInto(final Class> entityClass) {
return insertInto(entityClass, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder insertInto(final Class> entityClass, final Set excludedPropNames) {
return insert(entityClass, excludedPropNames).into(entityClass);
}
/**
* Generate the MySQL style batch insert sql.
*
* @param propsList list of entity or properties map.
* @return
*/
@Beta
public static SQLBuilder batchInsert(final Collection> propsList) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.ADD;
final Optional> first = N.firstNonNull(propsList);
if (first.isPresent() && ClassUtil.isEntity(first.get().getClass())) {
instance.entityClass = first.get().getClass();
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
}
instance.propsList = toInsertPropsList(propsList);
return instance;
}
/**
*
* @param tableName
* @return
*/
public static SQLBuilder update(final String tableName) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.UPDATE;
instance.tableName = tableName;
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder update(final Class> entityClass) {
return update(entityClass, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder update(final Class> entityClass, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.UPDATE;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.tableName = getTableName(entityClass, NamingPolicy.LOWER_CAMEL_CASE);
instance.columnNames = getUpdatePropNames(entityClass, excludedPropNames);
return instance;
}
/**
*
* @param tableName
* @return
*/
public static SQLBuilder deleteFrom(final String tableName) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.DELETE;
instance.tableName = tableName;
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder deleteFrom(final Class> entityClass) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.DELETE;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.tableName = getTableName(entityClass, NamingPolicy.LOWER_CAMEL_CASE);
return instance;
}
/**
*
* @param selectPart
* @return
*/
public static SQLBuilder select(final String selectPart) {
N.checkArgNotNullOrEmpty(selectPart, "selectPart");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnNames = Array.asList(selectPart);
return instance;
}
/**
*
* @param columnNames
* @return
*/
@SafeVarargs
public static SQLBuilder select(final String... columnNames) {
N.checkArgNotNullOrEmpty(columnNames, "columnNames");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnNames = Array.asList(columnNames);
return instance;
}
/**
*
* @param columnNames
* @return
*/
public static SQLBuilder select(final Collection columnNames) {
N.checkArgNotNullOrEmpty(columnNames, "columnNames");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnNames = columnNames;
return instance;
}
/**
*
* @param columnAliases
* @return
*/
public static SQLBuilder select(final Map columnAliases) {
N.checkArgNotNullOrEmpty(columnAliases, "columnAliases");
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.columnAliases = columnAliases;
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder select(final Class> entityClass) {
return select(entityClass, false);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @return
*/
public static SQLBuilder select(final Class> entityClass, final boolean includeSubEntityProperties) {
return select(entityClass, includeSubEntityProperties, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder select(final Class> entityClass, final Set excludedPropNames) {
return select(entityClass, false, excludedPropNames);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @param excludedPropNames
* @return
*/
public static SQLBuilder select(final Class> entityClass, final boolean includeSubEntityProperties, final Set excludedPropNames) {
final SQLBuilder instance = createInstance();
instance.op = OperationType.QUERY;
instance.entityClass = entityClass;
instance.propColumnNameMap = getProp2ColumnNameMap(instance.entityClass, instance.namingPolicy);
instance.columnNames = getSelectPropNames(entityClass, includeSubEntityProperties, excludedPropNames);
return instance;
}
/**
*
* @param entityClass
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass) {
return selectFrom(entityClass, false);
}
/**
*
* @param entityClass
* @param alias
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias) {
return selectFrom(entityClass, alias, false);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final boolean includeSubEntityProperties) {
return selectFrom(entityClass, includeSubEntityProperties, null);
}
/**
*
* @param entityClass
* @param alias
* @param includeSubEntityProperties
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias, final boolean includeSubEntityProperties) {
return selectFrom(entityClass, alias, includeSubEntityProperties, null);
}
/**
*
* @param entityClass
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final Set excludedPropNames) {
return selectFrom(entityClass, false, excludedPropNames);
}
/**
*
* @param entityClass
* @param alias
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias, final Set excludedPropNames) {
return selectFrom(entityClass, alias, false, excludedPropNames);
}
/**
*
* @param entityClass
* @param includeSubEntityProperties
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final boolean includeSubEntityProperties, final Set excludedPropNames) {
return selectFrom(entityClass, null, includeSubEntityProperties, excludedPropNames);
}
/**
*
* @param entityClass
* @param alias
* @param includeSubEntityProperties
* @param excludedPropNames
* @return
*/
public static SQLBuilder selectFrom(final Class> entityClass, final String alias, final boolean includeSubEntityProperties,
final Set |