org.apache.cxf.jaxrs.ext.search.SimpleSearchCondition Maven / Gradle / Ivy
Show all versions of cxf-rt-rs-extension-search Show documentation
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.cxf.jaxrs.ext.search;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.EnumSet;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.apache.cxf.jaxrs.ext.search.Beanspector.TypeInfo;
import org.apache.cxf.jaxrs.ext.search.collections.CollectionCheckCondition;
import org.apache.cxf.jaxrs.ext.search.collections.CollectionCheckInfo;
/**
* Simple search condition comparing primitive objects or complex object by its getters. For details see
* {@link #isMet(Object)} description.
*
* @param type of search condition.
*
*/
public class SimpleSearchCondition implements SearchCondition {
protected static final Set SUPPORTED_TYPES = EnumSet.noneOf(ConditionType.class);
static {
SUPPORTED_TYPES.add(ConditionType.EQUALS);
SUPPORTED_TYPES.add(ConditionType.NOT_EQUALS);
SUPPORTED_TYPES.add(ConditionType.GREATER_THAN);
SUPPORTED_TYPES.add(ConditionType.GREATER_OR_EQUALS);
SUPPORTED_TYPES.add(ConditionType.LESS_THAN);
SUPPORTED_TYPES.add(ConditionType.LESS_OR_EQUALS);
}
private final ConditionType joiningType = ConditionType.AND;
private T condition;
private List> scts;
/**
* Creates search condition with same operator (equality, inequality) applied in all comparison; see
* {@link #isMet(Object)} for details of comparison.
*
* @param cType shared condition type
* @param condition template object
*/
public SimpleSearchCondition(ConditionType cType, T condition) {
if (cType == null) {
throw new IllegalArgumentException("cType is null");
}
if (condition == null) {
throw new IllegalArgumentException("condition is null");
}
if (!SUPPORTED_TYPES.contains(cType)) {
throw new IllegalArgumentException("unsupported condition type: " + cType.name());
}
this.condition = condition;
scts = createConditions(null, null, null, cType);
}
/**
* Creates search condition with different operators (equality, inequality etc) specified for each getter;
* see {@link #isMet(Object)} for details of comparison. Cannot be used for primitive T type due to
* per-getter comparison strategy.
*
* @param getters2operators getters names and operators to be used with them during comparison
* @param realGetters
* @param propertyTypeInfo
* @param condition template object
*/
public SimpleSearchCondition(Map getters2operators,
Map realGetters,
Map propertyTypeInfo,
T condition) {
if (getters2operators == null) {
throw new IllegalArgumentException("getters2operators is null");
}
if (condition == null) {
throw new IllegalArgumentException("condition is null");
}
if (isBuiltIn(condition)) {
throw new IllegalArgumentException("mapped operators strategy is "
+ "not supported for primitive type "
+ condition.getClass().getName());
}
this.condition = condition;
for (ConditionType ct : getters2operators.values()) {
if (!SUPPORTED_TYPES.contains(ct)) {
throw new IllegalArgumentException("unsupported condition type: " + ct.name());
}
}
scts = createConditions(getters2operators, realGetters, propertyTypeInfo, null);
}
public SimpleSearchCondition(Map getters2operators,
T condition) {
this(getters2operators, null, null, condition);
}
@Override
public T getCondition() {
return condition;
}
/**
* {@inheritDoc}
*
* When constructor with map is used it returns null.
*/
@Override
public ConditionType getConditionType() {
if (scts.size() > 1) {
return joiningType;
}
return scts.get(0).getStatement().getCondition();
}
@Override
public List> getSearchConditions() {
if (scts.size() > 1) {
return Collections.unmodifiableList(scts);
}
return null;
}
private List> createConditions(Map getters2operators,
Map realGetters,
Map propertyTypeInfo,
ConditionType sharedType) {
if (isBuiltIn(condition)) {
return Collections.singletonList(
(SearchCondition)new PrimitiveSearchCondition<>(null, condition, null, sharedType, condition));
}
List> list = new ArrayList<>();
Map get2val = getGettersAndValues();
Set keySet = get2val != null ? get2val.keySet()
: ((SearchBean)condition).getKeySet();
for (String getter : keySet) {
ConditionType ct = getters2operators == null ? sharedType
: getters2operators.get(getter.toLowerCase());
if (ct == null) {
continue;
}
Object rval = get2val != null
? get2val.get(getter) : ((SearchBean)condition).get(getter);
if (rval == null) {
continue;
}
String realGetter = realGetters != null && realGetters.containsKey(getter)
? realGetters.get(getter) : getter;
TypeInfo tInfo = propertyTypeInfo != null ? propertyTypeInfo.get(getter) : null;
Type genType = tInfo != null ? tInfo.getGenericType() : rval.getClass();
CollectionCheckInfo checkInfo = tInfo != null ? tInfo.getCollectionCheckInfo() : null;
PrimitiveSearchCondition pc = checkInfo == null
? new PrimitiveSearchCondition<>(realGetter, rval, genType, ct, condition)
: new CollectionCheckCondition<>(realGetter, rval, genType, ct, condition, checkInfo);
list.add(pc);
}
if (list.isEmpty()) {
throw new IllegalStateException("This search condition is empty and can not be used");
}
return list;
}
/**
* Compares given object against template condition object.
*
* For built-in type T like String, Number (precisely, from type T located in subpackage of "java.lang.*")
* given object is directly compared with template object. Comparison for {@link ConditionType#EQUALS}
* requires correct implementation of {@link Object#equals(Object)}, using inequalities requires type T
* implementing {@link Comparable}.
*
* For other types the comparison of given object against template object is done using its
* getters; Value returned by {@linkplain #isMet(Object)} operation is conjunction ('and'
* operator) of comparisons of each getter accessible in object of type T. Getters of template object
* that return null or throw exception are not used in comparison. Finally, if all getters
* return nulls (are excluded) it is interpreted as no filter (match every pojo).
*
* If {@link #SimpleSearchCondition(ConditionType, Object) constructor with shared operator} was used,
* then getters are compared using the same operator. If {@link #SimpleSearchCondition(Map, Object)
* constructor with map of operators} was used then for every getter specified operator is used (getters
* for missing mapping are ignored). The way that comparison per-getter is done depending on operator type
* per getter - comparison for {@link ConditionType#EQUALS} requires correct implementation of
* {@link Object#equals(Object)}, using inequalities requires that getter type implements
* {@link Comparable}.
*
* For equality comparison and String type in template object (either being built-in or getter from client
* provided type) it is allowed to used asterisk at the beginning or at the end of text as wild card (zero
* or more of any characters) e.g. "foo*", "*foo" or "*foo*". Inner asterisks are not interpreted as wild
* cards.
*
* Example:
*
*
* SimpleSearchCondition<Integer> ssc = new SimpleSearchCondition<Integer>(
* ConditionType.GREATER_THAN, 10);
* ssc.isMet(20);
* // true since 20>10
*
* class Entity {
* public String getName() {...
* public int getLevel() {...
* public String getMessage() {...
* }
*
* Entity template = new Entity("bbb", 10, null);
* ssc = new SimpleSearchCondition<Entity>(
* ConditionType.GREATER_THAN, template);
*
* ssc.isMet(new Entity("aaa", 20, "some mesage"));
* // false: is not met, expression '"aaa">"bbb" and 20>10' is not true
* // since "aaa" is not greater than "bbb"; not that message is null in template hence ingored
*
* ssc.isMet(new Entity("ccc", 30, "other message"));
* // true: is met, expression '"ccc">"bbb" and 30>10' is true
*
* Map<String, ConditionType> map;
* map.put("name", ConditionType.EQUALS);
* map.put("level", ConditionType.GREATER_THAN);
* ssc = new SimpleSearchCondition<Entity>(
* ConditionType.GREATER_THAN, template);
*
* ssc.isMet(new Entity("ccc", 30, "other message"));
* // false due to expression '"aaa"=="ccc" and 30>10"' (note different operators)
*
*
*
* @throws IllegalAccessException when security manager disallows reflective call of getters.
*/
@Override
public boolean isMet(T pojo) {
for (SearchCondition sc : scts) {
if (!sc.isMet(pojo)) {
return false;
}
}
return true;
}
/**
* Creates cache of getters from template (condition) object and its values returned during one-pass
* invocation. Method isMet() will use its keys to introspect getters of passed pojo object, and values
* from map in comparison.
*
* @return template (condition) object getters mapped to their non-null values
*/
private Map getGettersAndValues() {
if (!SearchBean.class.isAssignableFrom(condition.getClass())) {
Map getters2values = new HashMap<>();
Beanspector beanspector = new Beanspector<>(condition);
for (String getter : beanspector.getGettersNames()) {
Object value = getValue(beanspector, getter, condition);
getters2values.put(getter, value);
}
//we do not need compare class objects
getters2values.keySet().remove("class");
return getters2values;
}
return null;
}
private Object getValue(Beanspector beanspector, String getter, T pojo) {
try {
return beanspector.swap(pojo).getValue(getter);
} catch (Throwable e) {
return null;
}
}
private boolean isBuiltIn(T pojo) {
return pojo.getClass().getName().startsWith("java.lang");
}
@Override
public List findAll(Collection pojos) {
List result = new ArrayList<>();
for (T pojo : pojos) {
if (isMet(pojo)) {
result.add(pojo);
}
}
return result;
}
public String toSQL(String table, String... columns) {
return SearchUtils.toSQL(this, table, columns);
}
@Override
public PrimitiveStatement getStatement() {
if (scts.size() == 1) {
return scts.get(0).getStatement();
}
return null;
}
@Override
public void accept(SearchConditionVisitor visitor) {
visitor.visit(this);
}
}