org.springmodules.validation.valang.predicates.GenericTestPredicate Maven / Gradle / Ivy
/*
* Copyright 2004-2005 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.springmodules.validation.valang.predicates;
import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Date;
import java.util.Iterator;
import org.apache.commons.collections.Predicate;
import org.apache.commons.collections.functors.AndPredicate;
import org.apache.commons.collections.functors.AnyPredicate;
import org.apache.commons.collections.functors.FalsePredicate;
import org.springframework.util.StringUtils;
import org.springmodules.validation.valang.ValangException;
import org.springmodules.validation.valang.functions.Function;
import org.springmodules.validation.valang.functions.LiteralFunction;
/**
* GenericTestPredicate can test if a property value is null or not null.
*
* @author Steven Devijver
* @since 23-04-2005
*/
public class GenericTestPredicate extends AbstractPropertyPredicate {
/**
*
Constructor taking two functions and an operator.
*
* @param leftFunction the left function
* @param operator the operator.
*/
public GenericTestPredicate(Function leftFunction, Operator operator, Function rightFunction, int line, int column) {
super(leftFunction, operator, rightFunction, line, column);
}
/**
*
The evaluate method takes the result of both functions and tests with the operator.
*
* @param target the target bean
* @return the result of the test
*/
public boolean evaluate(Object target) {
Object leftValue = getLeftFunction().getResult(target);
Object rightValue = null;
boolean dates = false;
boolean numbers = false;
if (getRightFunction() != null) {
rightValue = getRightFunction().getResult(target);
}
if (leftValue instanceof Number) {
leftValue = new BigDecimal(leftValue.toString());
}
if (rightValue instanceof Number) {
rightValue = new BigDecimal(rightValue.toString());
}
dates = leftValue instanceof Date && rightValue instanceof Date;
numbers = leftValue instanceof BigDecimal && rightValue instanceof BigDecimal;
if (getOperator() instanceof Operator.NullOperator) {
return leftValue == null;
} else if (getOperator() instanceof Operator.NotNullOperator) {
return leftValue != null;
} else if (getOperator() instanceof Operator.EqualsOperator) {
if (leftValue instanceof BigDecimal && rightValue instanceof BigDecimal) {
return ((BigDecimal)leftValue).compareTo((BigDecimal)rightValue) == 0;
} else if (dates) {
return ((Date)leftValue).getTime() == ((Date)rightValue).getTime();
} else {
return leftValue.equals(rightValue);
}
} else if (getOperator() instanceof Operator.NotEqualsOperator) {
if (leftValue instanceof BigDecimal && rightValue instanceof BigDecimal) {
return ((BigDecimal)leftValue).compareTo((BigDecimal)rightValue) != 0;
} else if (dates) {
return ((Date)leftValue).getTime() != ((Date)rightValue).getTime();
} else {
return !leftValue.equals(rightValue);
}
} else if (getOperator() instanceof Operator.LessThanOperator) {
if (dates) {
return ((Date)leftValue).getTime() < ((Date)rightValue).getTime();
} else if (numbers) {
return ((BigDecimal)leftValue).compareTo((BigDecimal)rightValue) < 0;
} else {
throw new ValangException("< operator only supports two date or two number values!", getLine(), getColumn());
}
} else if (getOperator() instanceof Operator.LessThanOrEqualOperator) {
if (dates) {
return ((Date)leftValue).getTime() <= ((Date)rightValue).getTime();
} else if (numbers) {
return ((BigDecimal)leftValue).compareTo((BigDecimal)rightValue) <= 0;
} else {
throw new ValangException("<= operator only supports two date or two number values!", getLine(), getColumn());
}
} else if (getOperator() instanceof Operator.MoreThanOperator) {
if (dates) {
return ((Date)leftValue).getTime() > ((Date)rightValue).getTime();
} else if (numbers) {
return ((BigDecimal)leftValue).compareTo((BigDecimal)rightValue) > 0;
} else {
throw new ValangException("> operator only supports two date or two number values!", getLine(), getColumn());
}
} else if (getOperator() instanceof Operator.MoreThanOrEqualOperator) {
if (dates) {
return ((Date)leftValue).getTime() >= ((Date)rightValue).getTime();
} else if (numbers) {
return ((BigDecimal)leftValue).compareTo((BigDecimal)rightValue) >= 0;
} else {
throw new IllegalArgumentException(">= operator only supports two date or two number values!");
}
} else if (getOperator() instanceof Operator.InOperator) {
Collection predicates = new ArrayList();
for (Iterator iter = getIterator(rightValue); iter.hasNext();) {
Object o = iter.next();
if (o instanceof Function) {
predicates.add(getPredicate(new LiteralFunction(leftValue), OperatorConstants.EQUALS_OPERATOR, (Function)o, getLine(), getColumn()));
} else {
predicates.add(getPredicate(new LiteralFunction(leftValue), OperatorConstants.EQUALS_OPERATOR, new LiteralFunction(o), getLine(), getColumn()));
}
}
if (predicates.isEmpty()) {
throw new IllegalStateException("IN expression contains no elements!");
} else if (predicates.size() == 1) {
predicates.add(FalsePredicate.getInstance());
}
return AnyPredicate.getInstance(predicates).evaluate(target);
} else if (getOperator() instanceof Operator.NotInOperator) {
Collection predicates = new ArrayList();
for (Iterator iter = getIterator(rightValue); iter.hasNext();) {
Object o = iter.next();
if (o instanceof Function) {
predicates.add(getPredicate(new LiteralFunction(leftValue), OperatorConstants.EQUALS_OPERATOR, (Function)o, getLine(), getColumn()));
} else {
predicates.add(getPredicate(new LiteralFunction(leftValue), OperatorConstants.EQUALS_OPERATOR, new LiteralFunction(o), getLine(), getColumn()));
}
}
if (predicates.isEmpty()) {
throw new IllegalStateException("NOT IN expression contains no elements!");
} else if (predicates.size() == 1) {
predicates.add(FalsePredicate.getInstance());
}
return !AnyPredicate.getInstance(predicates).evaluate(target);
} else if (getOperator() instanceof Operator.BetweenOperator) {
Object[] array = getArray(rightValue);
Predicate predicate1 = getPredicate(new LiteralFunction(leftValue), OperatorConstants.MORE_THAN_OR_EQUAL_OPERATOR, (Function)array[0], getLine(), getColumn());
Predicate predicate2 = getPredicate(new LiteralFunction(leftValue), OperatorConstants.LESS_THAN_OR_EQUAL_OPERATOR, (Function)array[1], getLine(), getColumn());
return AndPredicate.getInstance(predicate1, predicate2).evaluate(target);
} else if (getOperator() instanceof Operator.NotBetweenOperator) {
Object[] array = getArray(rightValue);
Predicate predicate1 = getPredicate(new LiteralFunction(leftValue), OperatorConstants.MORE_THAN_OR_EQUAL_OPERATOR, (Function)array[0], getLine(), getColumn());
Predicate predicate2 = getPredicate(new LiteralFunction(leftValue), OperatorConstants.LESS_THAN_OR_EQUAL_OPERATOR, (Function)array[1], getLine(), getColumn());
return !AndPredicate.getInstance(predicate1, predicate2).evaluate(target);
} else if (getOperator() instanceof Operator.HasLengthOperator) {
return StringUtils.hasLength(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.HasNoLengthOperator) {
return !StringUtils.hasLength(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.HasTextOperator) {
return StringUtils.hasText(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.HasNoTextOperator) {
return !StringUtils.hasText(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.IsBlankOperator) {
return isBlank(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.IsNotBlankOperator) {
return !isBlank(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.IsWordOperator) {
return isWord(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.IsNotWordOperator) {
return !isWord(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.IsLowerCaseOperator) {
return isLowerCase(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.IsNotLowerCaseOperator) {
return !isLowerCase(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.IsUpperCaseOperator) {
return isUpperCase(leftValue != null ? leftValue.toString() : null);
} else if (getOperator() instanceof Operator.IsNotUpperCaseOperator) {
return !isUpperCase(leftValue != null ? leftValue.toString() : null);
}
throw new IllegalStateException("Operator class [" + getOperator().getClass().getName() + "] not supported!");
}
protected Predicate getPredicate(Function leftFunction, Operator operator, Function rightFunction, int line, int column) {
return new GenericTestPredicate(leftFunction, operator, rightFunction, line, column);
}
private boolean isWord(String s) {
return s != null && s.length() > 0 &&
deleteWhitespace(s).equals(s);
}
private boolean isLowerCase(String s) {
return s.length() > 0 &&
s.toLowerCase().equals(s);
}
private boolean isUpperCase(String s) {
return s.length() > 0 &&
s.toUpperCase().equals(s);
}
private boolean isBlank(String s) {
return s == null || s.length() == 0;
}
private String deleteWhitespace(String s) {
return StringUtils.deleteAny(s, " \t\r\n\b");
}
}