org.dbflute.cbean.coption.LikeSearchOption Maven / Gradle / Ivy
/*
* Copyright 2014-2020 the original author or authors.
*
* 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 org.dbflute.cbean.coption;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import org.dbflute.cbean.cipher.GearedCipherManager;
import org.dbflute.cbean.dream.SpecifiedColumn;
import org.dbflute.cbean.sqlclause.query.QueryClauseArranger;
import org.dbflute.dbway.ExtensionOperand;
import org.dbflute.dbway.OnQueryStringConnector;
import org.dbflute.twowaysql.node.FilteringBindOption;
import org.dbflute.util.DfTypeUtil;
/**
* The option of like search.
*
* e.g.
* new LikeSearchOption().likePrefix() : PrefixSearch
* new LikeSearchOption().likeContain() : ContainSearch
* new LikeSearchOption().likeSuffix() : SuffixSearch
*
* @author jflute
* @author h-funaki added setCompoundColumnNullAsEmpty()
*/
public class LikeSearchOption extends SimpleStringOption implements FilteringBindOption {
// ===================================================================================
// Definition
// ==========
protected static final String LIKE_PREFIX = "prefix";
protected static final String LIKE_SUFFIX = "suffix";
protected static final String LIKE_CONTAIN = "contain";
// ===================================================================================
// Attribute
// =========
protected String _like;
protected String _escape;
protected boolean _asOrSplit;
protected List _originalWildCardList;
protected List _compoundColumnList;
protected List _compoundColumnSizeList;
protected boolean _nullCompoundedAsEmpty;
protected OnQueryStringConnector _stringConnector;
protected GearedCipherManager _cipherManager;;
// ===================================================================================
// Constructor
// ===========
/**
* Construct the option of like search normally.
*
* e.g.
* new LikeSearchOption().likePrefix() : PrefixSearch
* new LikeSearchOption().likeContain() : ContainSearch
* new LikeSearchOption().likeSuffix() : SuffixSearch
*
* new LikeSearchOption().likeContain().splitByBlank()
* new LikeSearchOption().likeContain().splitByBlank().asOrSplit()
*
*/
public LikeSearchOption() {
}
// ===================================================================================
// Rear Option
// ===========
@Override
public String getRearOption() {
if (_escape == null || _escape.trim().length() == 0) {
return "";
}
return " escape '" + _escape + "'";
}
// ===================================================================================
// Like
// ====
/**
* Set up prefix-search. e.g. like 'foo%' escape '|'
* @return this. (NotNull)
*/
public LikeSearchOption likePrefix() {
_like = LIKE_PREFIX;
doLikeAutoEscape();
return this;
}
/**
* Set up suffix-search. e.g. like '%foo' escape '|'
* @return this. (NotNull)
*/
public LikeSearchOption likeSuffix() {
_like = LIKE_SUFFIX;
doLikeAutoEscape();
return this;
}
/**
* Set up contain-search. e.g. like '%foo%' escape '|'
* @return this. (NotNull)
*/
public LikeSearchOption likeContain() {
_like = LIKE_CONTAIN;
doLikeAutoEscape();
return this;
}
protected void doLikeAutoEscape() {
escape();
}
public boolean isLikePrefix() {
return LIKE_PREFIX.equals(_like);
}
public boolean isLikeSuffix() {
return LIKE_SUFFIX.equals(_like);
}
public boolean isLikeContain() {
return LIKE_CONTAIN.equals(_like);
}
// ===================================================================================
// Escape
// ======
/**
* Escape like search by pipe-line '|'.
* @return The option of like search. (NotNull)
*/
public LikeSearchOption escape() {
return escapeByPipeLine();
}
public LikeSearchOption escapeByPipeLine() {
_escape = "|";
return this;
}
public LikeSearchOption escapeByAtMark() {
_escape = "@";
return this;
}
public LikeSearchOption escapeBySlash() {
_escape = "/";
return this;
}
public LikeSearchOption escapeByBackSlash() {
_escape = "\\";
return this;
}
public LikeSearchOption notEscape() {
_escape = null;
return this;
}
// ===================================================================================
// Split
// =====
/**
* Split a value as several condition by blank (space, full-width space, tab, CR, LF).
*
* e.g. 'and' condition
* LikeSearchOption option = new LikeSearchOption();
* option.likeContain().splitByBlank();
* cb.query().setFoo_ContainSearch("ab g kl", option);
* // FOO like '%ab%' and FOO like '%g%' and FOO like '%kl%'
* // (all conditions have escape statements)
*
* e.g. 'or' condition
* LikeSearchOption option = new LikeSearchOption();
* option.likeContain().splitByBlank().asOrSplit();
* cb.query().setFoo_ContainSearch("ab g kl", option);
* // FOO like '%ab%' or FOO like '%g%' or FOO like '%kl%'
* // (conditions have escape statements)
*
* @return this. (NotNull)
*/
public LikeSearchOption splitByBlank() {
assertSplitByPrecondition();
return (LikeSearchOption) doSplitByBlank();
}
// memorable codes...
/**
* Split a value as several condition by half-size space.
* @return this. (NotNull)
*/
public LikeSearchOption splitBySpace() {
assertSplitByPrecondition();
return (LikeSearchOption) doSplitBySpace();
}
/**
* Split a value as several condition by space that contains full-width space.
* @return this. (NotNull)
*/
public LikeSearchOption splitBySpaceContainsDoubleByte() {
assertSplitByPrecondition();
return (LikeSearchOption) doSplitBySpaceContainsDoubleByte();
}
/**
* Split a value as several condition by pipeline.
* @return this. (NotNull)
*/
public LikeSearchOption splitByPipeLine() {
assertSplitByPrecondition();
return (LikeSearchOption) doSplitByPipeLine();
}
/**
* Split a value as several condition by specified various delimiters.
* @param delimiterList The list of delimiter for split. (NotNull, NotEmpty)
* @return this. (NotNull)
*/
public LikeSearchOption splitByVarious(List delimiterList) {
assertSplitByPrecondition();
return (LikeSearchOption) doSplitByVarious(delimiterList);
}
protected void assertSplitByPrecondition() {
if (hasCompoundColumn()) {
String msg = "The Split of LikeSearch is unsupported with CompoundColumn.";
throw new IllegalStateException(msg);
}
}
/**
* Split as OR condition.
* You should call this with a splitByXxx method.
* @return this. (NotNull)
*/
public LikeSearchOption asOrSplit() {
_asOrSplit = true;
return this;
}
public boolean isAsOrSplit() {
return _asOrSplit;
}
/**
* Cut the large split (set limit count for split).
* @param splitLimitCount The limit count of split. (NotZero, NotMinus)
* @return this.
*/
public LikeSearchOption cutSplit(int splitLimitCount) {
doCutSplit(splitLimitCount);
return this;
}
// ===================================================================================
// Compound Column
// ===============
// -----------------------------------------------------
// Dream Cruise
// ------------
/**
* Add compound column connected to main column. {Dream Cruise}
*
* e.g. LikeSearch: MEMBER_NAME || MEMBER_ACCOUNT like ...
* MemberCB cb = new MemberCB();
* LikeSearchOption option = new LikeSearchOption().likeContain()
* .addCompoundColumn(cb.dreamCruiseCB().specify().columnMemberAccount());
* cb.query().setMemberName_LikeSearch("S", option);
*
* If any selected value is null, the compounded value is also null as default (even others are not null).
* If some of compounded columns are allowed to be null,
* setCompoundColumnNullAsEmpty() is recommended as LikeSearchOption.
*
* @param compoundColumn The compound column specified by Dream Cruise. (NotNull)
* @return this. (NotNull)
*/
public LikeSearchOption addCompoundColumn(SpecifiedColumn compoundColumn) {
assertCompoundColumnPrecondition(compoundColumn);
assertCompoundColumnSpecifiedColumn(compoundColumn);
if (_compoundColumnList == null) {
_compoundColumnList = new ArrayList();
}
_compoundColumnList.add(compoundColumn);
return this;
}
protected void assertCompoundColumnPrecondition(SpecifiedColumn compoundColumn) {
if (isSplit()) {
String msg = "The CompoundColumn of LikeSearch is unsupported with Split: " + compoundColumn;
throw new IllegalStateException(msg);
}
}
protected void assertCompoundColumnSpecifiedColumn(SpecifiedColumn compoundColumn) {
if (compoundColumn == null) {
String msg = "The argument 'compoundColumn' should not be null.";
throw new IllegalArgumentException(msg);
}
if (!compoundColumn.getColumnInfo().isObjectNativeTypeString()) {
String msg = "The type of the compound column should be String: " + compoundColumn;
throw new IllegalArgumentException(msg);
}
}
@Override
public boolean hasCompoundColumn() {
return _compoundColumnList != null && !_compoundColumnList.isEmpty();
}
@Override
public List getCompoundColumnList() {
return _compoundColumnList;
}
public void compoundsNullAsEmpty() {
_nullCompoundedAsEmpty = true;
}
@Override
public boolean isNullCompoundedAsEmpty() {
return _nullCompoundedAsEmpty;
}
public void clearCompoundColumn() {
if (_compoundColumnList != null) {
_compoundColumnList.clear();
}
if (_compoundColumnSizeList != null) {
_compoundColumnSizeList.clear();
}
}
// -----------------------------------------------------
// Optimization
// ------------
/**
* Optimize compound columns by fixed size.
* The columns specified their sizes should be fixed-size string type 'char'. (but no check so attention)
* @param sizes The array of column size for main column and compound columns. (NotNull)
* @return this. (NotNull)
*/
public LikeSearchOption optimizeCompoundColumnByFixedSize(Integer... sizes) {
if (!hasCompoundColumn()) {
String msg = "The CompoundColumnOptimization needs CompoundColumn.";
throw new IllegalStateException(msg);
}
if (sizes.length > (_compoundColumnList.size() + 1)) {
String msg = "The length of argument 'sizes' should be less or equal count of compound columns + 1.";
msg = msg + " sizes.length=" + sizes.length;
msg = msg + " compoundColumnList.size()=" + _compoundColumnList.size();
throw new IllegalArgumentException(msg);
}
_compoundColumnSizeList = Arrays.asList(sizes);
return this;
}
public boolean canOptimizeCompoundColumnLikePrefix() {
return hasCompoundColumn() && hasCompoundColumnOptimization() && isLikePrefix();
}
public boolean hasCompoundColumnOptimization() {
return _compoundColumnSizeList != null && !_compoundColumnSizeList.isEmpty();
}
public List getCompoundColumnSizeList() {
return _compoundColumnSizeList;
}
// -----------------------------------------------------
// StringConnector
// ---------------
// called after being set to condition-query or parameter-bean
// for DBMS that has original string connection way
// e.g. MySQL, SQLServer
public LikeSearchOption acceptStringConnector(OnQueryStringConnector stringConnector) {
_stringConnector = stringConnector;
return this;
}
public boolean hasStringConnector() {
return _stringConnector != null;
}
public OnQueryStringConnector getStringConnector() {
return _stringConnector;
}
// ===================================================================================
// Real Value
// ==========
@Override
public String generateRealValue(String value) {
value = super.generateRealValue(value);
// escape wild-cards
if (_escape != null && _escape.trim().length() != 0) {
String tmp = replace(value, _escape, _escape + _escape);
// basic wild-cards
tmp = filterEscape(tmp, "%");
tmp = filterEscape(tmp, "_");
if (_originalWildCardList != null) {
for (String wildCard : _originalWildCardList) {
tmp = filterEscape(tmp, wildCard);
}
}
value = tmp;
}
final String wildCard = "%";
if (_like == null || _like.trim().length() == 0) {
return value;
} else if (_like.equals(LIKE_PREFIX)) {
return value + wildCard;
} else if (_like.equals(LIKE_SUFFIX)) {
return wildCard + value;
} else if (_like.equals(LIKE_CONTAIN)) {
return wildCard + value + wildCard;
} else {
String msg = "The like was wrong string: " + _like;
throw new IllegalStateException(msg);
}
}
protected String filterEscape(String target, String wildCard) {
return replace(target, wildCard, _escape + wildCard);
}
// called after being set to condition-query or parameter-bean
// for DBMS that does not ignore an unused escape character
// e.g. Oracle, Apache Derby
public LikeSearchOption acceptOriginalWildCardList(List originalWildCardList) {
_originalWildCardList = originalWildCardList;
return this;
}
// ===================================================================================
// Deep Copy
// =========
@Override
public LikeSearchOption createDeepCopy() {
final LikeSearchOption copy = (LikeSearchOption) super.createDeepCopy();
copy._like = _like;
copy._escape = _escape;
copy._asOrSplit = _asOrSplit;
if (_originalWildCardList != null) {
copy._originalWildCardList = new ArrayList(_originalWildCardList);
}
if (_compoundColumnList != null) {
copy._compoundColumnList = new ArrayList(_compoundColumnList);
}
if (_compoundColumnSizeList != null) {
copy._compoundColumnSizeList = new ArrayList(_compoundColumnSizeList);
}
copy._stringConnector = _stringConnector;
return copy;
}
@Override
protected LikeSearchOption newDeepCopyInstance() {
return new LikeSearchOption();
}
// ===================================================================================
// Extension Operand
// =================
/**
* Get the operand for extension.
* @return The operand for extension. (NullAllowed: If the value is null, it means no extension)
*/
@Override
public ExtensionOperand getExtensionOperand() { // for application extension
return null; // as default
}
// ===================================================================================
// Where Clause Arranger
// =====================
/**
* Get the arranger of where clause.
* @return The arranger of where clause. (NullAllowed: If the value is null, it means no arrangement)
*/
@Override
public QueryClauseArranger getWhereClauseArranger() { // for application extension
return null; // as default
}
// ===================================================================================
// Geared Cipher
// =============
/**
* Accept the manager of geared cipher. (basically for compound columns)
* @param cipherManager The manager of geared cipher. (NullAllowed)
*/
public void acceptGearedCipherManager(GearedCipherManager cipherManager) {
_cipherManager = cipherManager;
}
@Override
public GearedCipherManager getGearedCipherManager() {
return _cipherManager;
}
// ===================================================================================
// Basic Override
// ==============
@Override
public String toString() {
final String title = DfTypeUtil.toClassTitle(this);
final String split = (isSplit() ? (_asOrSplit ? "true(or)" : "true(and)") : "false");
return title + ":{like=" + _like + ", escape=" + _escape + ", split=" + split + "}";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy