com.landawn.abacus.util.NamedSQL 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 java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import com.landawn.abacus.exception.AbacusException;
import com.landawn.abacus.pool.KeyedObjectPool;
import com.landawn.abacus.pool.PoolFactory;
import com.landawn.abacus.pool.PoolableWrapper;
/**
*
* @since 0.8
*
* @author Haiyang Li
*/
public final class NamedSQL {
private static final int EVICT_TIME = 60 * 1000;
private static final int LIVE_TIME = 24 * 60 * 60 * 1000;
private static final int MAX_IDLE_TIME = 24 * 60 * 60 * 1000;
private static final Set namedSQLPrefixSet = N.asSet(WD.INSERT, WD.SELECT, WD.UPDATE, WD.DELETE, WD.WITH);
private static final int factor = Math.min(Math.max(1, IOUtil.MAX_MEMORY_IN_MB / 1024), 8);
private static final KeyedObjectPool> pool = PoolFactory.createKeyedObjectPool(1000 * factor, EVICT_TIME);
private static final String PREFIX_OF_NAMED_PARAMETER = ":";
private static final String LEFT_OF_IBATIS_NAMED_PARAMETER = "#{";
private static final String RIGHT_OF_IBATIS_NAMED_PARAMETER = "}";
private static final String PREFIX_OF_COUCHBASE_NAMED_PARAMETER = "$";
private final String namedSQL;
private final String pureSQL;
private String couchbasePureSQL;
private final Map namedParameters;
private Map couchbaseNamedParameters;
private final int parameterCount;
private int couchbaseParameterCount;
private final Map attrs;
@SuppressWarnings({ "unchecked", "rawtypes" })
private NamedSQL(String sql, Map attrs) {
this.namedSQL = sql.trim();
this.namedParameters = new HashMap<>();
this.attrs = N.isNullOrEmpty(attrs) ? (Map) new HashMap<>() : new HashMap<>(attrs);
final List words = SQLParser.parse(namedSQL);
boolean isNamedSQLPrefix = false;
for (String word : words) {
if (N.notNullOrEmpty(word)) {
isNamedSQLPrefix = namedSQLPrefixSet.contains(word.toUpperCase());
break;
}
}
if (isNamedSQLPrefix) {
final StringBuilder sb = ObjectFactory.createStringBuilder();
int countOfParameter = 0;
for (String word : words) {
if (word.equals(WD.QUESTION_MARK)) {
if (namedParameters.size() > 0) {
throw new AbacusException("can't mix '?' and '#{propName}' in the same sql script");
}
countOfParameter++;
} else if (word.startsWith(LEFT_OF_IBATIS_NAMED_PARAMETER) && word.endsWith(RIGHT_OF_IBATIS_NAMED_PARAMETER)) {
namedParameters.put(countOfParameter++, word.substring(2, word.length() - 1));
word = WD.QUESTION_MARK;
} else if (word.startsWith(PREFIX_OF_NAMED_PARAMETER)) {
namedParameters.put(countOfParameter++, word.substring(1));
word = WD.QUESTION_MARK;
}
sb.append(word);
}
parameterCount = countOfParameter;
pureSQL = sb.toString();
ObjectFactory.recycle(sb);
} else {
pureSQL = sql;
parameterCount = 0;
}
}
public static NamedSQL parse(String sql) {
return parse(sql, null);
}
public static NamedSQL parse(String sql, Map attrs) {
NamedSQL result = null;
PoolableWrapper w = pool.get(sql);
if ((w == null) || (w.value() == null)) {
synchronized (pool) {
result = new NamedSQL(sql, attrs);
pool.put(sql, PoolableWrapper.of(result, LIVE_TIME, MAX_IDLE_TIME));
}
} else {
result = w.value();
}
return result;
}
public String getNamedSQL() {
return namedSQL;
}
public String getPureSQL() {
return pureSQL;
}
public String getPureSQL(boolean isForCouchbase) {
if (isForCouchbase) {
if (N.isNullOrEmpty(couchbasePureSQL)) {
parseForCouchbase();
}
return couchbasePureSQL;
} else {
return pureSQL;
}
}
public Map getNamedParameters() {
return namedParameters;
}
public Map getNamedParameters(boolean isForCouchbase) {
if (isForCouchbase) {
if (N.isNullOrEmpty(couchbasePureSQL)) {
parseForCouchbase();
}
return couchbaseNamedParameters;
} else {
return namedParameters;
}
}
public Map getAttribes() {
return attrs;
}
public int getParameterCount() {
return parameterCount;
}
public int getParameterCount(boolean isForCouchbase) {
if (isForCouchbase) {
if (N.isNullOrEmpty(couchbasePureSQL)) {
parseForCouchbase();
}
return couchbaseParameterCount;
} else {
return parameterCount;
}
}
private void parseForCouchbase() {
this.couchbaseNamedParameters = new HashMap<>();
final List words = SQLParser.parse(namedSQL);
boolean isNamedSQLPrefix = false;
for (String word : words) {
if (N.notNullOrEmpty(word)) {
isNamedSQLPrefix = namedSQLPrefixSet.contains(word.toUpperCase());
break;
}
}
if (isNamedSQLPrefix) {
final StringBuilder sb = ObjectFactory.createStringBuilder();
int countOfParameter = 0;
for (String word : words) {
if (word.equals(WD.QUESTION_MARK)) {
if (couchbaseNamedParameters.size() > 0) {
throw new AbacusException("can't mix '?' and '#{propName}' in the same sql script");
}
countOfParameter++;
word = PREFIX_OF_COUCHBASE_NAMED_PARAMETER + countOfParameter;
} else if (word.startsWith(LEFT_OF_IBATIS_NAMED_PARAMETER) && word.endsWith(RIGHT_OF_IBATIS_NAMED_PARAMETER)) {
couchbaseNamedParameters.put(countOfParameter++, word.substring(2, word.length() - 1));
word = PREFIX_OF_COUCHBASE_NAMED_PARAMETER + countOfParameter;
} else if (word.startsWith(PREFIX_OF_NAMED_PARAMETER) || word.startsWith(PREFIX_OF_COUCHBASE_NAMED_PARAMETER)) {
couchbaseNamedParameters.put(countOfParameter++, word.substring(1));
word = PREFIX_OF_COUCHBASE_NAMED_PARAMETER + countOfParameter;
}
sb.append(word);
}
boolean isNamedParametersByNum = true;
for (int i = 0; i < countOfParameter; i++) {
try {
if (N.asInt(couchbaseNamedParameters.get(i)) != i + 1) {
isNamedParametersByNum = false;
break;
}
} catch (Exception e) {
// ignore;
isNamedParametersByNum = false;
break;
}
}
if (isNamedParametersByNum) {
couchbaseNamedParameters.clear();
}
couchbaseParameterCount = countOfParameter;
couchbasePureSQL = sb.toString();
ObjectFactory.recycle(sb);
} else {
couchbaseParameterCount = 0;
couchbasePureSQL = namedSQL;
}
}
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = (prime * result) + ((namedSQL == null) ? 0 : namedSQL.hashCode());
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj) {
return true;
}
if (obj instanceof NamedSQL) {
NamedSQL other = (NamedSQL) obj;
return N.equals(namedSQL, other.namedSQL);
}
return false;
}
@Override
public String toString() {
return "[NamedSQL] " + namedSQL + " [PureSQL] " + pureSQL + " [Attribues] " + attrs;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy