[Git][java-team/mvel][upstream] New upstream version 2.4.4
Markus Koschany
gitlab at salsa.debian.org
Mon Aug 26 22:50:23 BST 2019
Markus Koschany pushed to branch upstream at Debian Java Maintainers / mvel
Commits:
4856310a by Markus Koschany at 2019-08-26T19:51:49Z
New upstream version 2.4.4
- - - - -
21 changed files:
- pom.xml
- src/main/java/org/mvel2/ast/BinaryOperation.java
- src/main/java/org/mvel2/compiler/AbstractParser.java
- src/main/java/org/mvel2/compiler/ExpressionCompiler.java
- src/main/java/org/mvel2/jsr223/MvelScriptEngineFactory.java
- src/main/java/org/mvel2/optimizers/impl/asm/ASMAccessorOptimizer.java
- src/main/java/org/mvel2/optimizers/impl/refl/ReflectiveAccessorOptimizer.java
- src/main/java/org/mvel2/optimizers/impl/refl/nodes/DynamicFunctionAccessor.java
- src/main/java/org/mvel2/util/CompilerTools.java
- src/main/java/org/mvel2/util/FastList.java
- src/main/java/org/mvel2/util/ParseTools.java
- src/main/java/org/mvel2/util/PropertyTools.java
- + src/test/java/org/mvel2/compiler/CollectionElementByIndexTest.java
- + src/test/java/org/mvel2/compiler/MvelCompileExpNullSafeTest.java
- src/test/java/org/mvel2/tests/core/CoreConfidenceTests.java
- src/test/java/org/mvel2/tests/core/FunctionsTest.java
- + src/test/java/org/mvel2/tests/core/MvelArrayTest.java
- + src/test/java/org/mvel2/tests/core/PrimitiveTypesTest.java
- src/test/java/org/mvel2/tests/core/PropertyAccessTests.java
- + src/test/java/org/mvel2/tests/core/res/Member.java
- + src/test/java/org/mvel2/tests/core/res/SharedFuncLib.java
Changes:
=====================================
pom.xml
=====================================
@@ -5,7 +5,7 @@
<groupId>org.mvel</groupId>
<artifactId>mvel2</artifactId>
<packaging>jar</packaging>
- <version>2.4.0.Final</version>
+ <version>2.4.4.Final</version>
<name>mvel</name>
<url>http://mvel.codehaus.org/</url>
@@ -31,7 +31,7 @@
<connection>scm:git:git at github.com:mvel/mvel.git</connection>
<url>scm:git:git at github.com:mvel/mvel.git</url>
<developerConnection>scm:git:git at github.com:mvel/mvel.git</developerConnection>
- <tag>mvel2-2.4.0.Final</tag>
+ <tag>mvel2-2.4.4.Final</tag>
</scm>
<issueManagement>
<system>jira</system>
@@ -229,6 +229,9 @@
<include>build.properties</include>
</includes>
</resource>
+ <resource>
+ <directory>src/main/resources</directory>
+ </resource>
</resources>
</build>
=====================================
src/main/java/org/mvel2/ast/BinaryOperation.java
=====================================
@@ -73,8 +73,16 @@ public class BinaryOperation extends BooleanNode {
egressType = getReturnTypeFromOp(operation, this.left.egressType, this.right.egressType);
if (!ctx.isStrongTyping()) break;
- if (!left.getEgressType().isAssignableFrom(right.getEgressType()) && !right.getEgressType().isAssignableFrom(left.getEgressType())) {
- if (right.isLiteral() && canConvert(left.getEgressType(), right.getEgressType())) {
+ final boolean leftIsAssignableFromRight = left.getEgressType().isAssignableFrom(right.getEgressType());
+ final boolean rightIsAssignableFromLeft = right.getEgressType().isAssignableFrom(left.getEgressType());
+
+ if (!leftIsAssignableFromRight && !rightIsAssignableFromLeft) {
+
+ // Convert literals only when passing from String to Character or from Float to Double
+ final boolean requiresConversion = right.getEgressType() == String.class || (
+ right.getEgressType() == Double.class && ( left.getEgressType() == Float.class || left.getEgressType() == float.class ) );
+
+ if (right.isLiteral() && requiresConversion && canConvert(left.getEgressType(), right.getEgressType())) {
Class targetType = isAritmeticOperation(operation) ? egressType : left.getEgressType();
this.right = new LiteralNode(convert(right.getReducedValueAccelerated(null, null, null), targetType), pCtx);
} else if ( !(areCompatible(left.getEgressType(), right.getEgressType()) ||
=====================================
src/main/java/org/mvel2/compiler/AbstractParser.java
=====================================
@@ -113,6 +113,7 @@ public class AbstractParser implements Parser, Serializable {
protected int fields;
+ protected static final int OP_NOT_LITERAL = -3;
protected static final int OP_OVERFLOW = -2;
protected static final int OP_TERMINATE = -1;
protected static final int OP_RESET_FRAME = 0;
@@ -2407,8 +2408,6 @@ public class AbstractParser implements Parser, Serializable {
* need to stop if this is not a literal.
*/
if (compileMode && !tk.isLiteral()) {
-
-
splitAccumulator.push(tk, new OperatorNode(operator2, expr, st, pCtx));
return OP_OVERFLOW;
}
@@ -2416,6 +2415,7 @@ public class AbstractParser implements Parser, Serializable {
dStack.push(operator = operator2, tk.getReducedValue(ctx, ctx, variableFactory));
while (true) {
+ ASTNode previousToken = tk;
// look ahead again
if ((tk = nextToken()) != null && (operator2 = tk.getOperator()) != -1
&& operator2 != END_OF_STMT && PTABLE[operator2] > PTABLE[operator]) {
@@ -2428,7 +2428,12 @@ public class AbstractParser implements Parser, Serializable {
/**
* This operator is of higher precedence, or the same level precedence. push to the RHS.
*/
- dStack.push(operator = operator2, nextToken().getReducedValue(ctx, ctx, variableFactory));
+ ASTNode nextToken = nextToken();
+ if (compileMode && !nextToken.isLiteral()) {
+ splitAccumulator.push(nextToken, new OperatorNode(operator2, expr, st, pCtx));
+ return OP_OVERFLOW;
+ }
+ dStack.push(operator = operator2, nextToken.getReducedValue(ctx, ctx, variableFactory));
continue;
}
@@ -2498,6 +2503,10 @@ public class AbstractParser implements Parser, Serializable {
}
default:
+ if (compileMode && !tk.isLiteral()) {
+ stk.push(operator, tk);
+ return OP_NOT_LITERAL;
+ }
stk.push(operator, tk.getReducedValue(ctx, ctx, variableFactory));
}
}
=====================================
src/main/java/org/mvel2/compiler/ExpressionCompiler.java
=====================================
@@ -123,7 +123,7 @@ public class ExpressionCompiler extends AbstractParser {
fields |= COMPILE_IMMEDIATE;
- while ((tk = nextToken()) != null) {
+ main_loop: while ((tk = nextToken()) != null) {
/**
* If this is a debug symbol, just add it and continue.
*/
@@ -199,7 +199,7 @@ public class ExpressionCompiler extends AbstractParser {
stk.push(tkLA2.getLiteralValue(), op = tkOp2.getOperator());
if (isArithmeticOperator(op)) {
- compileReduce(op, astBuild);
+ if (!compileReduce(op, astBuild)) continue main_loop;
}
else {
reduce();
@@ -327,7 +327,7 @@ public class ExpressionCompiler extends AbstractParser {
private boolean compileReduce(int opCode, ASTLinkedList astBuild) {
switch (arithmeticFunctionReduction(opCode)) {
- case -1:
+ case OP_TERMINATE:
/**
* The reduction failed because we encountered a non-literal,
* so we must now back out and cleanup.
@@ -341,7 +341,7 @@ public class ExpressionCompiler extends AbstractParser {
verify(pCtx, (ASTNode) splitAccumulator.pop())
);
return false;
- case -2:
+ case OP_OVERFLOW:
/**
* Back out completely, pull everything back off the stack and add the instructions
* to the output payload as they are.
@@ -353,6 +353,13 @@ public class ExpressionCompiler extends AbstractParser {
astBuild.addTokenNode(new LiteralNode(stk.pop(), pCtx), operator);
astBuild.addTokenNode(rightValue, (OperatorNode) splitAccumulator.pop());
astBuild.addTokenNode(verify(pCtx, (ASTNode) splitAccumulator.pop()));
+ return false;
+ case OP_NOT_LITERAL:
+ ASTNode tkLA2 = (ASTNode) stk.pop();
+ Integer tkOp2 = (Integer) stk.pop();
+ astBuild.addTokenNode(new LiteralNode(getStackValueResult(), pCtx));
+ astBuild.addTokenNode(new OperatorNode(tkOp2, expr, st, pCtx), verify(pCtx, tkLA2));
+ return false;
}
return true;
}
=====================================
src/main/java/org/mvel2/jsr223/MvelScriptEngineFactory.java
=====================================
@@ -1,6 +1,7 @@
package org.mvel2.jsr223;
import java.util.ArrayList;
+import java.util.Collections;
import java.util.List;
import javax.script.ScriptEngine;
@@ -14,14 +15,25 @@ public class MvelScriptEngineFactory implements ScriptEngineFactory {
private static final String LANGUAGE_NAME = "mvel";
private static final String LANGUAGE_VERSION = MVEL.VERSION;
- private static final List<String> NAMES = new ArrayList<String>();
- private static final List<String> EXTENSIONS = new ArrayList<String>();
- private static final List<String> MIME_TYPES = new ArrayList<String>();
+ private static final List<String> NAMES;
+ private static final List<String> EXTENSIONS;
+ private static final List<String> MIME_TYPES;
+
+ static {
+ List<String> n = new ArrayList<String>(1);
+ n.add(LANGUAGE_NAME);
+ NAMES = Collections.unmodifiableList(n);
+
+ EXTENSIONS = NAMES;
+
+ n = new ArrayList<String>(1);
+ n.add("application/x-"+LANGUAGE_NAME);
+ MIME_TYPES = Collections.unmodifiableList(n);
+ }
private static final MvelScriptEngine MVEL_SCRIPT_ENGINE = new MvelScriptEngine();
public MvelScriptEngineFactory() {
- NAMES.add(LANGUAGE_NAME);
}
@Override
=====================================
src/main/java/org/mvel2/optimizers/impl/asm/ASMAccessorOptimizer.java
=====================================
@@ -121,7 +121,9 @@ public class ASMAccessorOptimizer extends AbstractOptimizer implements AccessorO
} else if (javaVersion.startsWith("1.6")
|| javaVersion.startsWith("1.7")
|| javaVersion.startsWith("1.8")
- || javaVersion.startsWith("9")) {
+ || javaVersion.startsWith("9")
+ || javaVersion.startsWith("10")
+ || javaVersion.startsWith("11")) {
OPCODES_VERSION = Opcodes.V1_6;
} else {
OPCODES_VERSION = Opcodes.V1_2;
=====================================
src/main/java/org/mvel2/optimizers/impl/refl/ReflectiveAccessorOptimizer.java
=====================================
@@ -377,7 +377,7 @@ public class ReflectiveAccessorOptimizer extends AbstractOptimizer implements Ac
if (cursor < end) {
if (nullSafe) {
int os = expr[cursor] == '.' ? 1 : 0;
- addAccessorNode(new NullSafe(expr, cursor + os, length - cursor - os, pCtx));
+ addAccessorNode(new NullSafe(expr, cursor + os, end - cursor - os, pCtx));
if (curr == null) break;
}
if (curr == null) throw new NullPointerException();
@@ -806,17 +806,16 @@ public class ReflectiveAccessorOptimizer extends AbstractOptimizer implements Ac
return ((CharSequence) ctx).charAt((Integer) idx);
}
- else {
+ else if (ctx instanceof Class) {
TypeDescriptor tDescr = new TypeDescriptor(expr, this.start, length, 0);
if (tDescr.isArray()) {
Class cls = getClassReference((Class) ctx, tDescr, variableFactory, pCtx);
rootNode = new StaticReferenceAccessor(cls);
return cls;
}
-
- throw new CompileException("illegal use of []: unknown type: "
- + ctx.getClass().getName(), this.expr, this.start);
}
+ throw new CompileException("illegal use of []: unknown type: "
+ + ctx.getClass().getName(), this.expr, this.start);
}
=====================================
src/main/java/org/mvel2/optimizers/impl/refl/nodes/DynamicFunctionAccessor.java
=====================================
@@ -19,6 +19,7 @@
package org.mvel2.optimizers.impl.refl.nodes;
import org.mvel2.ast.Function;
+import org.mvel2.ast.FunctionInstance;
import org.mvel2.compiler.Accessor;
import org.mvel2.integration.VariableResolverFactory;
@@ -34,7 +35,12 @@ public class DynamicFunctionAccessor extends BaseAccessor {
public Object getValue(Object ctx, Object elCtx, VariableResolverFactory variableFactory) {
Object[] parms = null;
- Function function = (Function) ctx;
+ Function function;
+ if (ctx instanceof FunctionInstance){
+ function = ((FunctionInstance)ctx).getFunction();
+ } else {
+ function = (Function)ctx;
+ }
if (parameters != null && parameters.length != 0) {
parms = new Object[parameters.length];
=====================================
src/main/java/org/mvel2/util/CompilerTools.java
=====================================
@@ -103,7 +103,7 @@ public class CompilerTools {
bo = null;
boolean inv = tkOp.isOperator(Operator.SUB);
- boolean reduc = isReductionOpportunity(tkOp, tk2);
+ boolean reduc = tk.isLiteral() && isReductionOpportunity(tkOp, tk2);
boolean p_inv = false;
while (reduc) {
=====================================
src/main/java/org/mvel2/util/FastList.java
=====================================
@@ -18,13 +18,17 @@
package org.mvel2.util;
-import org.mvel2.ImmutableElementException;
-
import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
-import java.util.*;
+import java.util.AbstractList;
+import java.util.Collection;
+import java.util.Iterator;
+import java.util.List;
+import java.util.ListIterator;
+
+import org.mvel2.ImmutableElementException;
public class FastList<E> extends AbstractList<E> implements Externalizable {
private E[] elements;
@@ -168,14 +172,14 @@ public class FastList<E> extends AbstractList<E> implements Externalizable {
public ListIterator<E> listIterator() {
return new ListIterator<E>() {
- private int i = 0;
+ private int i = -1;
public boolean hasNext() {
- return i < size;
+ return i < size-1;
}
public E next() {
- return elements[i++];
+ return elements[++i];
}
public boolean hasPrevious() {
@@ -183,7 +187,7 @@ public class FastList<E> extends AbstractList<E> implements Externalizable {
}
public E previous() {
- return elements[i--];
+ return elements[--i];
}
public int nextIndex() {
=====================================
src/main/java/org/mvel2/util/ParseTools.java
=====================================
@@ -320,7 +320,7 @@ public class ParseTools {
if (arguments[i] == null) {
if (!actualParamType.isPrimitive()) {
- score += 6;
+ score += 7;
}
else {
score = 0;
@@ -328,15 +328,18 @@ public class ParseTools {
}
}
else if (actualParamType == arguments[i]) {
- score += 7;
+ score += 8;
}
else if (actualParamType.isPrimitive() && boxPrimitive(actualParamType) == arguments[i]) {
- score += 6;
+ score += 7;
}
else if (arguments[i].isPrimitive() && unboxPrimitive(arguments[i]) == actualParamType) {
- score += 6;
+ score += 7;
}
else if (actualParamType.isAssignableFrom(arguments[i])) {
+ score += 6;
+ }
+ else if (isPrimitiveSubtype(arguments[i], actualParamType)) {
score += 5;
}
else if (isNumericallyCoercible(arguments[i], actualParamType)) {
@@ -1065,6 +1068,22 @@ public class ParseTools {
return code;
}
+ private static boolean isPrimitiveSubtype( Class argument, Class<?> actualParamType ) {
+ if (!actualParamType.isPrimitive()) {
+ return false;
+ }
+ Class<?> primitiveArgument = unboxPrimitive(argument);
+ if (!primitiveArgument.isPrimitive()) {
+ return false;
+ }
+ return ( actualParamType == double.class && primitiveArgument == float.class ) ||
+ ( actualParamType == float.class && primitiveArgument == long.class ) ||
+ ( actualParamType == long.class && primitiveArgument == int.class ) ||
+ ( actualParamType == int.class && primitiveArgument == char.class ) ||
+ ( actualParamType == int.class && primitiveArgument == short.class ) ||
+ ( actualParamType == short.class && primitiveArgument == byte.class );
+ }
+
public static boolean isNumericallyCoercible(Class target, Class parm) {
Class boxedTarget = target.isPrimitive() ? boxPrimitive(target) : target;
@@ -2175,7 +2194,7 @@ public class ParseTools {
public static Class forNameWithInner(String className, ClassLoader classLoader) throws ClassNotFoundException {
try {
- return Class.forName(className, true, classLoader);
+ return classLoader.loadClass( className );
} catch (ClassNotFoundException cnfe) {
return findInnerClass( className, classLoader, cnfe );
}
@@ -2185,7 +2204,7 @@ public class ParseTools {
for (int lastDotPos = className.lastIndexOf('.'); lastDotPos > 0; lastDotPos = className.lastIndexOf('.')) {
className = className.substring(0, lastDotPos) + "$" + className.substring(lastDotPos+1);
try {
- return Class.forName(className, true, classLoader);
+ return classLoader.loadClass( className );
} catch (ClassNotFoundException e) { /* ignore */ }
}
throw cnfe;
=====================================
src/main/java/org/mvel2/util/PropertyTools.java
=====================================
@@ -97,6 +97,13 @@ public class PropertyTools {
String getter = ReflectionUtil.getGetter(property);
Method candidate = null;
+
+ if (Collection.class.isAssignableFrom(clazz) && "isEmpty".equals(isGet)) {
+ try {
+ return Collection.class.getMethod("isEmpty");
+ } catch (NoSuchMethodException ignore) {}
+ }
+
for (Method meth : clazz.getMethods()) {
if ((meth.getModifiers() & PUBLIC) != 0 && (meth.getModifiers() & STATIC) == 0 && meth.getParameterTypes().length == 0
&& (getter.equals(meth.getName()) || property.equals(meth.getName()) || ((isGet.equals(meth.getName()) || simpleIsGet.equals(meth.getName())) && meth.getReturnType() == boolean.class)
=====================================
src/test/java/org/mvel2/compiler/CollectionElementByIndexTest.java
=====================================
@@ -0,0 +1,31 @@
+package org.mvel2.compiler;
+
+import java.io.Serializable;
+import java.util.Collections;
+import java.util.Map;
+import java.util.Set;
+
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+import org.mvel2.CompileException;
+import org.mvel2.MVEL;
+
+import static org.junit.internal.matchers.StringContains.containsString;
+
+/**
+ * @author Anton Rybochkin (anton.rybochkin at axibase.com)
+ */
+public class CollectionElementByIndexTest {
+ @Rule
+ public ExpectedException expectException = ExpectedException.none();
+
+ @Test
+ public void testGetElementByIndexInSet() {
+ expectException.expect(CompileException.class);
+ expectException.expectMessage(containsString("unknown type: java.util.Collections$SingletonSet"));
+ final Map<String, Set<String>> map = Collections.singletonMap("set", Collections.singleton("test"));
+ final Serializable ser = MVEL.compileExpression("set[0]");
+ MVEL.executeExpression(ser, map);
+ }
+}
=====================================
src/test/java/org/mvel2/compiler/MvelCompileExpNullSafeTest.java
=====================================
@@ -0,0 +1,133 @@
+package org.mvel2.compiler;
+
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.List;
+
+import org.junit.Test;
+import org.mvel2.MVEL;
+import org.mvel2.compiler.CompiledExpression;
+import org.mvel2.compiler.ExpressionCompiler;
+
+/**
+ * @author Viswa Ramamoorthy (viswaramamoorthy at yahoo.com)
+ */
+public class MvelCompileExpNullSafeTest
+{
+ @Test
+ public void testMvelCompileNullSafeNullFirst() {
+ String expression = "((parentGroup != null) && ($.?child.firstName in parentGroup.parentList if $.?child.firstName != null).size() > 0)";
+ ExpressionCompiler compiler = new ExpressionCompiler(expression, true);
+ CompiledExpression compiledExpression = compiler.compile();
+
+ ParentGroup pGroup = new ParentGroup();
+ List<Parent> list = new ArrayList<Parent>();
+ Parent parent = new Parent();
+ parent.setChild(null);
+ list.add(parent);
+ pGroup.setParentList(list);
+
+ Boolean result = (Boolean)MVEL.executeExpression(compiledExpression, Collections.<String, Object>singletonMap("parentGroup", pGroup));
+ assert result == false;
+
+ Child child = new Child();
+ child.setFirstName("vlaa");
+
+ pGroup = new ParentGroup();
+ list = new ArrayList<Parent>();
+ parent = new Parent();
+ parent.setChild(child);
+ list.add(parent);
+ child = new Child();
+ child.setFirstName(null);
+ parent = new Parent();
+ parent.setChild(child);
+ list.add(parent);
+ pGroup.setParentList(list);
+
+ result = (Boolean)MVEL.executeExpression(compiledExpression, Collections.<String, Object>singletonMap("parentGroup", pGroup));
+ assert result == true;
+ }
+
+ @Test
+ public void testMvelCompileNullSafeNullSecond() {
+ String expression = "((parentGroup != null) && ($.?child.firstName in parentGroup.parentList if $.?child.firstName != null).size() > 0)";
+ ExpressionCompiler compiler = new ExpressionCompiler(expression, true);
+ CompiledExpression compiledExpression = compiler.compile();
+
+ Child child = new Child();
+ child.setFirstName("vlaa");
+
+ ParentGroup pGroup = new ParentGroup();
+ List<Parent> list = new ArrayList<Parent>();
+ Parent parent = new Parent();
+ parent.setChild(child);
+ list.add(parent);
+ child = new Child();
+ child.setFirstName(null);
+ parent = new Parent();
+ parent.setChild(child);
+ list.add(parent);
+ pGroup.setParentList(list);
+
+ Boolean result = (Boolean)MVEL.executeExpression(compiledExpression, Collections.<String, Object>singletonMap("parentGroup", pGroup));
+ assert result == true;
+
+ pGroup = new ParentGroup();
+ list = new ArrayList<Parent>();
+ parent = new Parent();
+ parent.setChild(null);
+ list.add(parent);
+ pGroup.setParentList(list);
+
+ result = (Boolean)MVEL.executeExpression(compiledExpression, Collections.<String, Object>singletonMap("parentGroup", pGroup));
+ assert result == false;
+
+ }
+
+ public class Child {
+ private String firstName;
+ private String lastName;
+
+ public String getFirstName() {
+ return firstName;
+ }
+
+ public void setFirstName(String firstName) {
+ this.firstName = firstName;
+ }
+
+ public String getLastName() {
+ return lastName;
+ }
+
+ public void setLastName(String lastName) {
+ this.lastName = lastName;
+ }
+ }
+
+ public class Parent {
+ private Child child;
+
+ public Child getChild() {
+ return child;
+ }
+
+ public void setChild(Child child) {
+ this.child = child;
+ }
+ }
+
+ public class ParentGroup {
+ private List<Parent> parentList;
+
+ public List<Parent> getParentList() {
+ return parentList;
+ }
+
+ public void setParentList(List<Parent> parentList) {
+ this.parentList = parentList;
+ }
+
+ }
+}
=====================================
src/test/java/org/mvel2/tests/core/CoreConfidenceTests.java
=====================================
@@ -27,6 +27,7 @@ import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.Vector;
+import java.util.concurrent.atomic.AtomicInteger;
import junit.framework.TestCase;
import org.mvel2.CompileException;
@@ -82,7 +83,15 @@ import org.mvel2.util.ParseTools;
import org.mvel2.util.ReflectionUtil;
import static java.util.Collections.unmodifiableCollection;
-import static org.mvel2.MVEL.*;
+
+import static org.mvel2.MVEL.compileExpression;
+import static org.mvel2.MVEL.compileSetExpression;
+import static org.mvel2.MVEL.eval;
+import static org.mvel2.MVEL.evalToBoolean;
+import static org.mvel2.MVEL.executeExpression;
+import static org.mvel2.MVEL.executeSetExpression;
+import static org.mvel2.MVEL.parseMacros;
+import static org.mvel2.MVEL.setProperty;
import static org.mvel2.util.ParseTools.loadFromFile;
@SuppressWarnings({"ALL"})
@@ -4018,17 +4027,6 @@ public class CoreConfidenceTests extends AbstractTest {
assertFalse(result);
}
- public void testDhanji2() {
- String ex =
- "name = null;\n" +
- "$_1 = ?name.?toLowerCase();\n";
-
-
- Serializable s = MVEL.compileExpression(ex);
-
- MVEL.executeExpression(s, new HashMap());
- }
-
public void testPrimitiveNumberCoercion() {
final ParserContext parserContext = new ParserContext();
parserContext.setStrictTypeEnforcement(true);
@@ -4158,6 +4156,44 @@ public class CoreConfidenceTests extends AbstractTest {
System.out.println(result);
}
+ public void testPrimitiveNumberCoercionDuringDivisionShouldWorkOnBothSide() {
+ final ParserContext parserContext = new ParserContext();
+ parserContext.setStrictTypeEnforcement(true);
+ parserContext.setStrongTyping(true);
+ parserContext.addInput("a", int.class);
+ parserContext.addInput("b", int.class);
+
+ int a = 1;
+ int b = 2;
+ Object res = a / b;
+ System.out.printf("Result class Java: %s\nResult value: %s\n", res.getClass(), res);
+ Object resBoolean = a / b < 0.99;
+ System.out.println("Result Boolean: " + resBoolean);
+
+ Serializable constantDoubleLeft = MVEL.compileExpression("0.99 >= a / b", parserContext);
+ Object resultLeft = MVEL.executeExpression(constantDoubleLeft, new HashMap() {{
+ put("a", 1);
+ put("b", 2);
+ }});
+ assertEquals(true, resultLeft);
+ Serializable constantDoubleRight = MVEL.compileExpression("a / b < 0.99", parserContext);
+ Object resultRight = MVEL.executeExpression(constantDoubleRight, new HashMap() {{
+ put("a", 1);
+ put("b", 2);
+ }});
+ assertEquals(true, resultRight);
+
+ parserContext.addInput("c", double.class);
+ parserContext.addInput("d", double.class);
+ Serializable constantIntRight = MVEL.compileExpression("c / d > 0", parserContext);
+ Object resultRightInt = MVEL.executeExpression(constantIntRight, new HashMap() {{
+ put("c", 1);
+ put("d", 2);
+ }});
+ assertEquals(true, resultRightInt);
+
+ }
+
public void testUntypedClone() {
String expression = "obj.clone();";
ParserContext context = new ParserContext();
@@ -4325,6 +4361,9 @@ public class CoreConfidenceTests extends AbstractTest {
Map vars = new HashMap() {{ put("ch", 'a'); }};
assertEquals(true, MVEL.executeExpression(MVEL.compileExpression("ch == \"a\"", pctx), vars));
assertEquals(false, MVEL.executeExpression(MVEL.compileExpression("ch == \"b\"", pctx), vars));
+
+// assertEquals(true, MVEL.executeExpression(MVEL.compileExpression("\"a\" == ch", pctx), vars));
+// assertEquals(false, MVEL.executeExpression(MVEL.compileExpression("\"b\" == ch", pctx), vars));
}
public void testFieldNameWithUnderscore() {
@@ -4616,4 +4655,45 @@ public class CoreConfidenceTests extends AbstractTest {
fail("parse of this expression should raise an error");
} catch (Exception e) { }
}
+
+ public void testPrimitiveSubtyping() {
+ ParserConfiguration conf = new ParserConfiguration();
+ ParserContext pctx = new ParserContext( conf );
+ pctx.setStrictTypeEnforcement(true);
+ pctx.setStrongTyping(true);
+ BigDecimal result = (BigDecimal)MVEL.executeExpression(MVEL.compileExpression("java.math.BigDecimal.valueOf(100)", pctx), new HashMap());
+ assertEquals("100", result.toString());
+ }
+
+ public void testUseVariableFactoryWithArithmeticOperation() {
+ checkOperation("3 + 4 * i.get()", 43);
+ checkOperation("2 * 3 + 4 * i.get()", 46);
+ checkOperation("1 + 2 * 3 + i.get()", 17);
+ checkOperation("2 * 3 + 4 * i.get()", 46);
+ checkOperation("1 + 2 * 3 + 4 * i.get()", 47);
+ checkOperation("1 + 2 * 3 + i.get() * 4", 47);
+ checkOperation("1 + 2 * 3 + i.get() + 4", 21);
+ checkOperation("4 * i.get() + 5", 45);
+ checkOperation("3 + 4 * i.get() + 5", 48);
+ checkOperation("2 * 3 + 4 * i.get() + 5", 51);
+ checkOperation("1 + 2 * 3 + 4 * i.get() + 5", 52);
+ checkOperation("1 + 2 * 3 + 4 * i.get() * 5", 207);
+ checkOperation("i.get() + 1 + 2 * 3 + 4 * i.get()", 57);
+ }
+
+ private void checkOperation(String expression, int expectedResult) {
+ AtomicInteger i = new AtomicInteger( 10 );
+ VariableResolverFactory factory = new MapVariableResolverFactory(new HashMap<String, Object>());
+ factory.createVariable("i", i);
+
+ ParserConfiguration pconf = new ParserConfiguration();
+ ParserContext pctx = new ParserContext(pconf);
+ pctx.setStrictTypeEnforcement(true);
+ pctx.setStrongTyping(true);
+ pctx.addInput("i", AtomicInteger.class);
+
+ Serializable compiledExpr = MVEL.compileExpression(expression, pctx);
+ int result = (Integer)MVEL.executeExpression(compiledExpr, null, factory);
+ assertEquals(expectedResult, result);
+ }
}
\ No newline at end of file
=====================================
src/test/java/org/mvel2/tests/core/FunctionsTest.java
=====================================
@@ -10,13 +10,15 @@ import org.mvel2.compiler.ExpressionCompiler;
import org.mvel2.integration.VariableResolverFactory;
import org.mvel2.integration.impl.MapVariableResolverFactory;
import org.mvel2.optimizers.OptimizerFactory;
-import org.mvel2.util.CompilerTools;
-import org.mvel2.util.MVELClassLoader;
+import org.mvel2.tests.core.res.Member;
+import org.mvel2.tests.core.res.SharedFuncLib;
import static org.mvel2.util.CompilerTools.extractAllDeclaredFunctions;
import java.io.Serializable;
+import java.math.BigDecimal;
import java.util.*;
+import java.util.concurrent.*;
/**
* @author Dhanji R. Prasanna (dhanji at gmail com)
@@ -200,4 +202,37 @@ public class FunctionsTest extends AbstractTest {
assertEquals("foobar", MVEL.executeExpression(s, myVarFactory));
}
+
+
+ public void testFunctionReuseMultiThread(){
+ ExecutorService executor = Executors.newFixedThreadPool(4);
+ List<Callable<BigDecimal>> tasks = new ArrayList<Callable<BigDecimal>>();
+ for (int i=0;i<30;i++){
+ tasks.add(new Callable<BigDecimal>() {
+ @Override
+ public BigDecimal call() throws Exception {
+ List<Member> lst = new ArrayList<Member>();
+ lst.add(new Member("a", 18));
+ lst.add(new Member("b", 12));
+ lst.add(new Member("c", 40));
+ lst.add(new Member("d", 66));
+ lst.add(new Member("e", 72));
+ HashMap<String, Object> map = new HashMap<String,Object>();
+ map.put("members",lst);
+ return new SharedFuncLib().eval("round( sum(members,0B, def(p){ return 2B*p.age; }) ,2)", map, BigDecimal.class);
+ }
+ });
+ }
+
+ try {
+ List<Future<BigDecimal>> futures = executor.invokeAll(tasks);
+ for (Future<BigDecimal> future : futures){
+ System.out.println("res=" + future.get().toString());
+ }
+ } catch (InterruptedException ie){
+ throw new RuntimeException(ie);
+ } catch (ExecutionException ee) {
+ throw new RuntimeException(ee);
+ }
+ }
}
=====================================
src/test/java/org/mvel2/tests/core/MvelArrayTest.java
=====================================
@@ -0,0 +1,99 @@
+/*
+ * Copyright 2005 JBoss Inc
+ *
+ * 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.mvel2.tests.core;
+
+import java.io.Serializable;
+import java.util.HashMap;
+
+import junit.framework.TestCase;
+import org.mvel2.MVEL;
+import org.mvel2.integration.VariableResolverFactory;
+import org.mvel2.integration.impl.MapVariableResolverFactory;
+import org.mvel2.optimizers.OptimizerFactory;
+
+import static org.mvel2.MVEL.executeExpression;
+
+public class MvelArrayTest extends TestCase {
+
+ private final String biglistTestScript =
+ "list = [];\n" +
+ "list.add(1);\n" +
+ "list.add(2);\n" +
+ "list.add(3);\n" +
+ "list.add(null);\n" + // null!
+ "list.add(5);\n" +
+ "list.add(6);\n" +
+ "list.add(7);\n" +
+ "list.add(8);\n" +
+ "list.add(9);\n" +
+ "list.add(10);\n" +
+ "list.add(11);\n" +
+ "list.add(12);\n" +
+ "java.util.Collections.replaceAll( list, null, 25 );\n" + // replace nulls, with a big value
+ "java.util.Collections.max( list );\n"; // return
+
+ public void testAsmOptimizerDiesWithBigList() {
+
+ OptimizerFactory.setDefaultOptimizer("ASM");
+ Serializable compileExpression = MVEL.compileExpression( biglistTestScript );
+ VariableResolverFactory factory = new MapVariableResolverFactory(new HashMap());
+
+ int actual = (Integer) executeExpression(compileExpression, factory);
+ assertEquals( actual, 25);
+ }
+
+ public void testReflectiveOptimizerWorksFine() {
+
+ OptimizerFactory.setDefaultOptimizer("reflective");
+ Serializable compileExpression = MVEL.compileExpression( biglistTestScript );
+ VariableResolverFactory factory = new MapVariableResolverFactory(new HashMap());
+
+ int actual = (Integer) executeExpression(compileExpression, factory);
+ assertEquals( actual, 25);
+ }
+
+ public void testDynamicOptimizerWorksFine() {
+
+ OptimizerFactory.setDefaultOptimizer("dynamic");
+ Serializable compileExpression = MVEL.compileExpression( biglistTestScript );
+ VariableResolverFactory factory = new MapVariableResolverFactory(new HashMap());
+
+ int actual = (Integer) executeExpression(compileExpression, factory);
+ assertEquals( actual, 25);
+ }
+
+ private final String smallListTestScript =
+ "list = [];\n" +
+ "list.add(1);\n" +
+ "list.add(2);\n" +
+ "list.add(3);\n" +
+ "list.add(null);\n" + // null!
+ "list.add(5);\n" +
+ "list.add(6);\n" +
+ "java.util.Collections.replaceAll( list, null, 25 );\n" + // replace nulls, with a big value
+ "java.util.Collections.max( list );\n"; // return
+
+ public void testAsmOptimizerWithSmallListIsFine() {
+
+ OptimizerFactory.setDefaultOptimizer("ASM");
+ Serializable compileExpression = MVEL.compileExpression( smallListTestScript );
+ VariableResolverFactory factory = new MapVariableResolverFactory(new HashMap());
+
+ int actual = (Integer) executeExpression(compileExpression, factory);
+ assertEquals( actual, 25);
+ }
+}
\ No newline at end of file
=====================================
src/test/java/org/mvel2/tests/core/PrimitiveTypesTest.java
=====================================
@@ -0,0 +1,64 @@
+/*
+ * Copyright 2005 JBoss Inc
+ *
+ * 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.mvel2.tests.core;
+
+import org.mvel2.MVEL;
+import org.mvel2.ParserConfiguration;
+import org.mvel2.ParserContext;
+
+public class PrimitiveTypesTest extends AbstractTest {
+ public static class FactWithFloat {
+
+ private final float floatValue;
+ private final Float floatObjectValue;
+
+ public FactWithFloat(final float floatValue) {
+ this.floatValue = floatValue;
+ this.floatObjectValue = floatValue;
+ }
+
+ public float getFloatValue() {
+ return floatValue;
+ }
+
+ public Float getFloatObjectValue() {
+ return floatObjectValue;
+ }
+ }
+
+ public void testFloatPrimitive() {
+ ParserConfiguration conf = new ParserConfiguration();
+ conf.addImport( FactWithFloat.class );
+ ParserContext pctx = new ParserContext( conf );
+ pctx.setStrictTypeEnforcement(true);
+ pctx.setStrongTyping(true);
+ pctx.addInput("this", FactWithFloat.class);
+ boolean result = ( Boolean ) MVEL.executeExpression(MVEL.compileExpression("floatValue == 15.1", pctx), new FactWithFloat(15.1f));
+ assertTrue( result );
+ }
+
+ public void testFloat() {
+ ParserConfiguration conf = new ParserConfiguration();
+ conf.addImport( FactWithFloat.class );
+ ParserContext pctx = new ParserContext( conf );
+ pctx.setStrictTypeEnforcement(true);
+ pctx.setStrongTyping(true);
+ pctx.addInput("this", FactWithFloat.class);
+ boolean result = ( Boolean ) MVEL.executeExpression(MVEL.compileExpression("floatObjectValue == 15.1", pctx), new FactWithFloat(15.1f));
+ assertTrue( result );
+ }
+}
=====================================
src/test/java/org/mvel2/tests/core/PropertyAccessTests.java
=====================================
@@ -468,6 +468,41 @@ public class PropertyAccessTests extends AbstractTest {
}
}
+ public final class Data {
+
+ private final Collection<Object> list;
+
+ Data(List<Object> list) {
+ this.list = list;
+ }
+
+ public Collection<Object> getList() {
+ return list;
+ }
+ }
+
+ public void testStaleReflectiveCollectionIsEmptyAccessor() {
+ try
+ {
+ OptimizerFactory.setDefaultOptimizer(OptimizerFactory.SAFE_REFLECTIVE);
+ Serializable getFooExpression = MVEL.compileExpression("list.empty");
+ Map vars = new HashMap();
+
+ // ArrayList -> Colletions.EmptyList
+ assertEquals(true, MVEL.executeExpression(getFooExpression, new Data(new ArrayList<Object>())));
+ assertEquals(true, MVEL.executeExpression(getFooExpression, new Data(Collections.emptyList())));
+
+ // Colletions.EmptyList -> ArrayList
+ assertEquals(true, MVEL.executeExpression(getFooExpression, new Data(Collections.emptyList())));
+ assertEquals(true, MVEL.executeExpression(getFooExpression, new Data(new ArrayList<Object>())));
+ OptimizerFactory.setDefaultOptimizer(OptimizerFactory.DYNAMIC);
+ }
+ finally
+ {
+ OptimizerFactory.setDefaultOptimizer(OptimizerFactory.DYNAMIC);
+ }
+ }
+
public void testMVEL308() {
String expression = "foreach(field: updates.entrySet()) { ctx._target[field.key] = field.value; }";
Serializable compiled = MVEL.compileExpression(expression);
=====================================
src/test/java/org/mvel2/tests/core/res/Member.java
=====================================
@@ -0,0 +1,20 @@
+package org.mvel2.tests.core.res;
+
+public class Member {
+ private String name;
+ private Integer age;
+
+ public Member(String name, Integer age) {
+ super();
+ this.name = name;
+ this.age = age;
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public Integer getAge() {
+ return age;
+ }
+}
=====================================
src/test/java/org/mvel2/tests/core/res/SharedFuncLib.java
=====================================
@@ -0,0 +1,26 @@
+package org.mvel2.tests.core.res;
+
+import org.mvel2.MVEL;
+import org.mvel2.integration.VariableResolverFactory;
+import org.mvel2.integration.impl.MapVariableResolverFactory;
+
+import java.util.Map;
+
+public class SharedFuncLib {
+
+ final static VariableResolverFactory functionFactory = new MapVariableResolverFactory();
+
+ static {
+ MVEL.eval(
+ "def round(numValue, decPlaces) { return numValue.setScale(decPlaces,java.math.RoundingMode.HALF_UP) };"
+ + "\n"
+ + "def sum(lst, startValue, accFunc){ s = startValue; foreach (i : lst){ s = s + accFunc(i); } return s; };",
+ functionFactory);
+ }
+
+ public <T> T eval(String formula, Map<String, Object> context, Class<T> toType) {
+ VariableResolverFactory myVariableResolverFactory = new MapVariableResolverFactory();
+ myVariableResolverFactory.setNextFactory(functionFactory);
+ return MVEL.eval(formula, context, myVariableResolverFactory, toType);
+ }
+}
View it on GitLab: https://salsa.debian.org/java-team/mvel/commit/4856310a55c88d451874f5cde768766e42089044
--
View it on GitLab: https://salsa.debian.org/java-team/mvel/commit/4856310a55c88d451874f5cde768766e42089044
You're receiving this email because of your account on salsa.debian.org.
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://alioth-lists.debian.net/pipermail/pkg-java-commits/attachments/20190826/e0b11ccb/attachment.html>
More information about the pkg-java-commits
mailing list