com.landawn.abacus.util.DynamicSQLBuilder Maven / Gradle / Ivy
/*
* Copyright (C) 2018 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 java.util.Collection;
import java.util.Map;
import com.landawn.abacus.logging.Logger;
import com.landawn.abacus.logging.LoggerFactory;
/**
* Dynamic SQL builder. Must remember to call {@code build()} to generate target sql and release resources.
*
* @author haiyangl
*
*/
@SuppressWarnings("java:S1192")
public class DynamicSQLBuilder {
static final Logger logger = LoggerFactory.getLogger(DynamicSQLBuilder.class);
private Select select = new Select(Objectory.createStringBuilder());
private From from = new From(Objectory.createStringBuilder());
private Where where;
private GroupBy groupBy;
private Having having;
private OrderBy orderBy;
private StringBuilder moreParts = null;
private DynamicSQLBuilder() {
}
/**
*
*
* @return
*/
public static DynamicSQLBuilder create() {
return new DynamicSQLBuilder();
}
/**
*
*
* @return
*/
public Select select() {
return select;
}
/**
*
*
* @return
*/
public From from() {
return from;
}
/**
*
*
* @return
*/
public Where where() {
if (where == null) {
where = new Where(Objectory.createStringBuilder());
}
return where;
}
/**
*
*
* @return
*/
public GroupBy groupBy() {
if (groupBy == null) {
groupBy = new GroupBy(Objectory.createStringBuilder());
}
return groupBy;
}
/**
*
*
* @return
*/
public Having having() {
if (having == null) {
having = new Having(Objectory.createStringBuilder());
}
return having;
}
/**
*
*
* @return
*/
public OrderBy orderBy() {
if (orderBy == null) {
orderBy = new OrderBy(Objectory.createStringBuilder());
}
return orderBy;
}
/**
*
* @param limitCond
* @return
*/
public DynamicSQLBuilder limit(String limitCond) {
getStringBuilderForMoreParts().append(" ").append(limitCond);
return this;
}
/**
*
* @param count
* @return
*/
public DynamicSQLBuilder limit(final int count) {
getStringBuilderForMoreParts().append(" LIMIT ").append(count);
return this;
}
/**
*
* @param offset
* @param count
* @return
*/
public DynamicSQLBuilder limit(final int offset, final int count) {
getStringBuilderForMoreParts().append(" LIMIT ").append(offset).append(", ").append(count);
return this;
}
/**
* Limit by row num.
*
* @param count
* @return
*/
public DynamicSQLBuilder limitByRowNum(int count) {
getStringBuilderForMoreParts().append(" ROWNUM < ").append(count);
return this;
}
/**
*
* @param offset
* @return
*/
public DynamicSQLBuilder offsetRows(final int offset) {
getStringBuilderForMoreParts().append(" OFFSET ").append(offset).append(" ROWS");
return this;
}
/**
*
*
* @param n
* @return
*/
public DynamicSQLBuilder fetchNextNRowsOnly(final int n) {
getStringBuilderForMoreParts().append(" FETCH NEXT ").append(n).append(" ROWS ONLY");
return this;
}
/**
*
*
* @param n
* @return
*/
public DynamicSQLBuilder fetchFirstNRowsOnly(final int n) {
getStringBuilderForMoreParts().append(" FETCH FIRST ").append(n).append(" ROWS ONLY");
return this;
}
private StringBuilder getStringBuilderForMoreParts() {
if (moreParts == null) {
moreParts = Objectory.createStringBuilder();
}
return moreParts;
}
/**
*
* @param query
* @return
*/
public DynamicSQLBuilder union(final String query) {
getStringBuilderForMoreParts().append(" UNION ").append(query);
return this;
}
/**
*
* @param query
* @return
*/
public DynamicSQLBuilder unionAll(final String query) {
getStringBuilderForMoreParts().append(" UNION ALL ").append(query);
return this;
}
/**
*
* @param query
* @return
*/
public DynamicSQLBuilder intersect(final String query) {
getStringBuilderForMoreParts().append(" INTERSECT ").append(query);
return this;
}
/**
*
* @param query
* @return
*/
public DynamicSQLBuilder except(final String query) {
getStringBuilderForMoreParts().append(" EXCEPT ").append(query);
return this;
}
/**
*
* @param query
* @return
*/
public DynamicSQLBuilder minus(final String query) {
getStringBuilderForMoreParts().append(" MINUS ").append(query);
return this;
}
// /**
// *
// * @param b
// * @param actionForTrue
// * @return
// */
// @Beta
// public DynamicSQLBuilder iF(final boolean b, final Consumer actionForTrue) {
// if (b) {
// actionForTrue.accept(this);
// }
//
// return this;
// }
//
// /**
// *
// * @param b
// * @param actionForTrue
// * @param actionForFalse
// * @return
// */
// @Beta
// public DynamicSQLBuilder ifOrElse(final boolean b, final Consumer actionForTrue, final Consumer actionForFalse) {
// if (b) {
// actionForTrue.accept(this);
// } else {
// actionForFalse.accept(this);
// }
//
// return this;
// }
/**
*
*
* @return
*/
public String build() {
select.sb.append(" ").append(from.sb);
if (where != null) {
select.sb.append(" ").append(where.sb);
Objectory.recycle(where.sb);
where = null;
}
if (groupBy != null) {
select.sb.append(" ").append(groupBy.sb);
Objectory.recycle(groupBy.sb);
groupBy = null;
}
if (having != null) {
select.sb.append(" ").append(having.sb);
Objectory.recycle(having.sb);
having = null;
}
if (orderBy != null) {
select.sb.append(" ").append(orderBy.sb);
Objectory.recycle(orderBy.sb);
orderBy = null;
}
if (moreParts != null) {
select.sb.append(moreParts);
Objectory.recycle(moreParts);
}
final String sql = select.sb.toString();
Objectory.recycle(from.sb);
Objectory.recycle(select.sb);
select = null;
from = null;
// if (logger.isDebugEnabled()) {
// logger.debug(sql);
// }
return sql;
}
/**
* The Class Select.
*/
public static class Select {
/** The sb. */
final StringBuilder sb;
/**
* Instantiates a new select.
*
* @param sb
*/
Select(final StringBuilder sb) {
this.sb = sb;
}
/**
*
* @param column
* @return
*/
public Select append(String column) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("SELECT ");
}
sb.append(column);
return this;
}
/**
*
* @param column
* @param alias
* @return
*/
public Select append(String column, String alias) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("SELECT ");
}
sb.append(column).append(" AS ").append(alias);
return this;
}
/**
*
* @param columns
* @return
*/
public Select append(Collection columns) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("SELECT ");
}
sb.append(Strings.join(columns, ", "));
return this;
}
/**
*
* @param column
* @return
*/
public Select append(final Map columnsAndAliasMap) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("SELECT ");
}
sb.append(Strings.joinEntries(columnsAndAliasMap, " AS ", ", "));
return this;
}
/**
*
*
* @param b
* @param str
* @return
*/
public Select appendIf(final boolean b, final String str) {
if (b) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("SELECT ");
}
sb.append(str);
}
return this;
}
/**
*
*
* @param b
* @param strToAppendForTrue
* @param strToAppendForFalse
* @return
*/
public Select appendIfOrElse(final boolean b, final String strToAppendForTrue, final String strToAppendForFalse) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("SELECT ");
}
if (b) {
sb.append(strToAppendForTrue);
} else {
sb.append(strToAppendForFalse);
}
return this;
}
}
/**
* The Class From.
*/
public static class From {
/** The sb. */
final StringBuilder sb;
/**
* Instantiates a new from.
*
* @param sb
*/
From(final StringBuilder sb) {
this.sb = sb;
}
/**
*
* @param table
* @return
*/
public From append(String table) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("FROM ");
}
sb.append(table);
return this;
}
/**
*
* @param table
* @param alias
* @return
*/
public From append(String table, String alias) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("FROM");
}
sb.append(table).append(" ").append(alias);
return this;
}
/**
*
* @param table
* @param on
* @return
*/
public From join(String table, String on) {
sb.append(" JOIN ").append(table).append(" ON ").append(on);
return this;
}
/**
*
* @param table
* @param on
* @return
*/
public From innerJoin(String table, String on) {
sb.append(" INNER JOIN ").append(table).append(" ON ").append(on);
return this;
}
/**
*
* @param table
* @param on
* @return
*/
public From leftJoin(String table, String on) {
sb.append(" LEFT JOIN ").append(table).append(" ON ").append(on);
return this;
}
/**
*
* @param table
* @param on
* @return
*/
public From rightJoin(String table, String on) {
sb.append(" RIGHT JOIN ").append(table).append(" ON ").append(on);
return this;
}
/**
*
* @param table
* @param on
* @return
*/
public From fullJoin(String table, String on) {
sb.append(" FULL JOIN ").append(table).append(" ON ").append(on);
return this;
}
/**
*
*
* @param b
* @param str
* @return
*/
public From appendIf(final boolean b, final String str) {
if (b) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("FROM ");
}
sb.append(str);
}
return this;
}
/**
*
*
* @param b
* @param strToAppendForTrue
* @param strToAppendForFalse
* @return
*/
public From appendIfOrElse(final boolean b, final String strToAppendForTrue, final String strToAppendForFalse) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("FROM ");
}
if (b) {
sb.append(strToAppendForTrue);
} else {
sb.append(strToAppendForFalse);
}
return this;
}
}
/**
* The Class Where.
*/
public static class Where {
/** The sb. */
final StringBuilder sb;
/**
* Instantiates a new where.
*
* @param sb
*/
Where(final StringBuilder sb) {
this.sb = sb;
}
/**
*
* @param cond
* @return
*/
public Where append(String cond) {
if (sb.length() > 0) {
sb.append(" ");
} else {
sb.append("WHERE ");
}
sb.append(cond);
return this;
}
/**
* Append question mark {@code ?} {@code n} times.
*
* @param n
* @return
*/
public Where repeatQM(int n) {
N.checkArgNotNegative(n, "n");
for (int i = 0; i < n; i++) {
if (i > 0) {
sb.append(", ?");
} else {
sb.append('?');
}
}
return this;
}
/**
* Append question mark {@code ?} {@code n} times.
*
* @param n
* @param prefix
* @param postfix
* @return
*/
public Where repeatQM(int n, String prefix, String postfix) {
N.checkArgNotNegative(n, "n");
sb.append(prefix);
for (int i = 0; i < n; i++) {
if (i > 0) {
sb.append(", ?");
} else {
sb.append('?');
}
}
sb.append(postfix);
return this;
}
/**
*
* @param cond
* @return
*/
public Where and(String cond) {
sb.append(" AND ").append(cond);
return this;
}
/**
*
* @param cond
* @return
*/
public Where or(String cond) {
sb.append(" OR ").append(cond);
return this;
}
/**
*
*
* @param b
* @param str
* @return
*/
public Where appendIf(final boolean b, final String str) {
if (b) {
if (sb.length() > 0) {
sb.append(" ");
} else {
sb.append("WHERE ");
}
sb.append(str);
}
return this;
}
/**
*
*
* @param b
* @param strToAppendForTrue
* @param strToAppendForFalse
* @return
*/
public Where appendIfOrElse(final boolean b, final String strToAppendForTrue, final String strToAppendForFalse) {
if (sb.length() > 0) {
sb.append(" ");
} else {
sb.append("WHERE ");
}
if (b) {
sb.append(strToAppendForTrue);
} else {
sb.append(strToAppendForFalse);
}
return this;
}
}
/**
* The Class GroupBy.
*/
public static class GroupBy {
/** The sb. */
final StringBuilder sb;
/**
* Instantiates a new group by.
*
* @param sb
*/
GroupBy(final StringBuilder sb) {
this.sb = sb;
}
/**
*
* @param column
* @return
*/
public GroupBy append(String column) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("GROUP BY ");
}
sb.append(column);
return this;
}
/**
*
* @param columns
* @return
*/
public GroupBy append(Collection columns) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("GROUP BY ");
}
sb.append(Strings.join(columns, ", "));
return this;
}
/**
*
*
* @param b
* @param str
* @return
*/
public GroupBy appendIf(final boolean b, final String str) {
if (b) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("GROUP BY ");
}
sb.append(str);
}
return this;
}
/**
*
*
* @param b
* @param strToAppendForTrue
* @param strToAppendForFalse
* @return
*/
public GroupBy appendIfOrElse(final boolean b, final String strToAppendForTrue, final String strToAppendForFalse) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("GROUP BY ");
}
if (b) {
sb.append(strToAppendForTrue);
} else {
sb.append(strToAppendForFalse);
}
return this;
}
}
/**
* The Class Having.
*/
public static class Having {
/** The sb. */
final StringBuilder sb;
/**
* Instantiates a new having.
*
* @param sb
*/
Having(final StringBuilder sb) {
this.sb = sb;
}
/**
*
* @param cond
* @return
*/
public Having append(String cond) {
if (sb.length() > 0) {
sb.append(" ");
} else {
sb.append("HAVING ");
}
sb.append(cond);
return this;
}
/**
*
* @param cond
* @return
*/
public Having and(String cond) {
sb.append(" AND ").append(cond);
return this;
}
/**
*
* @param cond
* @return
*/
public Having or(String cond) {
sb.append(" OR ").append(cond);
return this;
}
/**
*
*
* @param b
* @param str
* @return
*/
public Having appendIf(final boolean b, final String str) {
if (b) {
if (sb.length() > 0) {
sb.append(" ");
} else {
sb.append("HAVING ");
}
sb.append(str);
}
return this;
}
/**
*
*
* @param b
* @param strToAppendForTrue
* @param strToAppendForFalse
* @return
*/
public Having appendIfOrElse(final boolean b, final String strToAppendForTrue, final String strToAppendForFalse) {
if (sb.length() > 0) {
sb.append(" ");
} else {
sb.append("HAVING ");
}
if (b) {
sb.append(strToAppendForTrue);
} else {
sb.append(strToAppendForFalse);
}
return this;
}
}
/**
* The Class OrderBy.
*/
public static class OrderBy {
/** The sb. */
final StringBuilder sb;
/**
* Instantiates a new order by.
*
* @param sb
*/
OrderBy(final StringBuilder sb) {
this.sb = sb;
}
/**
*
* @param column
* @return
*/
public OrderBy append(String column) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("ORDER BY ");
}
sb.append(column);
return this;
}
/**
*
* @param columns
* @return
*/
public OrderBy append(Collection columns) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("ORDER BY ");
}
sb.append(Strings.join(columns, ", "));
return this;
}
/**
*
*
* @param b
* @param str
* @return
*/
public OrderBy appendIf(final boolean b, final String str) {
if (b) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("ORDER BY ");
}
sb.append(str);
}
return this;
}
/**
*
*
* @param b
* @param strToAppendForTrue
* @param strToAppendForFalse
* @return
*/
public OrderBy appendIfOrElse(final boolean b, final String strToAppendForTrue, final String strToAppendForFalse) {
if (sb.length() > 0) {
sb.append(", ");
} else {
sb.append("ORDER BY ");
}
if (b) {
sb.append(strToAppendForTrue);
} else {
sb.append(strToAppendForFalse);
}
return this;
}
}
/**
* The Class DSB.
*/
public static final class DSB extends DynamicSQLBuilder {
/**
* Instantiates a new dsb.
*/
private DSB() {
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy