Painless: Fix errors allowing void to be assigned to def. (#27460)

This commit is contained in:
Jack Conradson 2017-11-28 13:44:52 -08:00 committed by GitHub
parent 9e42b77f7e
commit 2d927fabab
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 30 additions and 15 deletions

View File

@ -20,7 +20,6 @@
package org.elasticsearch.painless.node; package org.elasticsearch.painless.node;
import org.elasticsearch.painless.DefBootstrap; import org.elasticsearch.painless.DefBootstrap;
import org.elasticsearch.painless.Definition;
import org.elasticsearch.painless.Definition.Cast; import org.elasticsearch.painless.Definition.Cast;
import org.elasticsearch.painless.Definition.Type; import org.elasticsearch.painless.Definition.Type;
import org.elasticsearch.painless.Globals; import org.elasticsearch.painless.Globals;
@ -213,6 +212,11 @@ public final class EAssignment extends AExpression {
// If the lhs node is a def optimized node we update the actual type to remove the need for a cast. // If the lhs node is a def optimized node we update the actual type to remove the need for a cast.
if (lhs.isDefOptimized()) { if (lhs.isDefOptimized()) {
rhs.analyze(locals); rhs.analyze(locals);
if (rhs.actual.clazz == void.class) {
throw createError(new IllegalArgumentException("Right-hand side cannot be a [void] type for assignment."));
}
rhs.expected = rhs.actual; rhs.expected = rhs.actual;
lhs.updateActual(rhs.actual); lhs.updateActual(rhs.actual);
// Otherwise, we must adapt the rhs type to the lhs type with a cast. // Otherwise, we must adapt the rhs type to the lhs type with a cast.

View File

@ -19,9 +19,7 @@
package org.elasticsearch.painless.node; package org.elasticsearch.painless.node;
import java.util.Collections;
import org.elasticsearch.painless.DefBootstrap; import org.elasticsearch.painless.DefBootstrap;
import org.elasticsearch.painless.Definition;
import org.elasticsearch.painless.Globals; import org.elasticsearch.painless.Globals;
import org.elasticsearch.painless.Locals; import org.elasticsearch.painless.Locals;
import org.elasticsearch.painless.Location; import org.elasticsearch.painless.Location;
@ -29,6 +27,7 @@ import org.elasticsearch.painless.MethodWriter;
import org.objectweb.asm.Type; import org.objectweb.asm.Type;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.Objects; import java.util.Objects;
import java.util.Set; import java.util.Set;
@ -76,6 +75,10 @@ final class PSubDefCall extends AExpression {
totalCaptures += lambda.getCaptureCount(); totalCaptures += lambda.getCaptureCount();
} }
if (expression.actual.clazz == void.class) {
throw createError(new IllegalArgumentException("Argument(s) cannot be of [void] type when calling method [" + name + "]."));
}
expression.expected = expression.actual; expression.expected = expression.actual;
arguments.set(argument, expression.cast(locals)); arguments.set(argument, expression.cast(locals));
} }

View File

@ -20,7 +20,6 @@
package org.elasticsearch.painless.node; package org.elasticsearch.painless.node;
import org.elasticsearch.painless.DefBootstrap; import org.elasticsearch.painless.DefBootstrap;
import org.elasticsearch.painless.Definition;
import org.elasticsearch.painless.Definition.Type; import org.elasticsearch.painless.Definition.Type;
import org.elasticsearch.painless.Globals; import org.elasticsearch.painless.Globals;
import org.elasticsearch.painless.Locals; import org.elasticsearch.painless.Locals;

View File

@ -22,7 +22,7 @@ package org.elasticsearch.painless;
/** Tests for addition operator across all types */ /** Tests for addition operator across all types */
//TODO: NaN/Inf/overflow/... //TODO: NaN/Inf/overflow/...
public class AdditionTests extends ScriptTestCase { public class AdditionTests extends ScriptTestCase {
public void testBasics() throws Exception { public void testBasics() throws Exception {
assertEquals(3.0, exec("double x = 1; byte y = 2; return x + y;")); assertEquals(3.0, exec("double x = 1; byte y = 2; return x + y;"));
} }
@ -195,7 +195,7 @@ public class AdditionTests extends ScriptTestCase {
assertEquals(1.0+0.0, exec("return 1.0+0.0;")); assertEquals(1.0+0.0, exec("return 1.0+0.0;"));
assertEquals(0.0+0.0, exec("return 0.0+0.0;")); assertEquals(0.0+0.0, exec("return 0.0+0.0;"));
} }
public void testDef() { public void testDef() {
assertEquals(2, exec("def x = (byte)1; def y = (byte)1; return x + y")); assertEquals(2, exec("def x = (byte)1; def y = (byte)1; return x + y"));
assertEquals(2, exec("def x = (short)1; def y = (byte)1; return x + y")); assertEquals(2, exec("def x = (short)1; def y = (byte)1; return x + y"));
@ -253,7 +253,7 @@ public class AdditionTests extends ScriptTestCase {
assertEquals(2D, exec("def x = (float)1; def y = (double)1; return x + y")); assertEquals(2D, exec("def x = (float)1; def y = (double)1; return x + y"));
assertEquals(2D, exec("def x = (double)1; def y = (double)1; return x + y")); assertEquals(2D, exec("def x = (double)1; def y = (double)1; return x + y"));
} }
public void testDefTypedLHS() { public void testDefTypedLHS() {
assertEquals(2, exec("byte x = (byte)1; def y = (byte)1; return x + y")); assertEquals(2, exec("byte x = (byte)1; def y = (byte)1; return x + y"));
assertEquals(2, exec("short x = (short)1; def y = (byte)1; return x + y")); assertEquals(2, exec("short x = (short)1; def y = (byte)1; return x + y"));
@ -311,7 +311,7 @@ public class AdditionTests extends ScriptTestCase {
assertEquals(2D, exec("float x = (float)1; def y = (double)1; return x + y")); assertEquals(2D, exec("float x = (float)1; def y = (double)1; return x + y"));
assertEquals(2D, exec("double x = (double)1; def y = (double)1; return x + y")); assertEquals(2D, exec("double x = (double)1; def y = (double)1; return x + y"));
} }
public void testDefTypedRHS() { public void testDefTypedRHS() {
assertEquals(2, exec("def x = (byte)1; byte y = (byte)1; return x + y")); assertEquals(2, exec("def x = (byte)1; byte y = (byte)1; return x + y"));
assertEquals(2, exec("def x = (short)1; byte y = (byte)1; return x + y")); assertEquals(2, exec("def x = (short)1; byte y = (byte)1; return x + y"));
@ -369,19 +369,19 @@ public class AdditionTests extends ScriptTestCase {
assertEquals(2D, exec("def x = (float)1; double y = (double)1; return x + y")); assertEquals(2D, exec("def x = (float)1; double y = (double)1; return x + y"));
assertEquals(2D, exec("def x = (double)1; double y = (double)1; return x + y")); assertEquals(2D, exec("def x = (double)1; double y = (double)1; return x + y"));
} }
public void testDefNulls() { public void testDefNulls() {
expectScriptThrows(NullPointerException.class, () -> { expectScriptThrows(NullPointerException.class, () -> {
exec("def x = null; int y = 1; return x + y"); exec("def x = null; int y = 1; return x + y");
}); });
expectScriptThrows(NullPointerException.class, () -> { expectScriptThrows(NullPointerException.class, () -> {
exec("int x = 1; def y = null; return x + y"); exec("int x = 1; def y = null; return x + y");
}); });
expectScriptThrows(NullPointerException.class, () -> { expectScriptThrows(NullPointerException.class, () -> {
exec("def x = null; def y = 1; return x + y"); exec("def x = null; def y = 1; return x + y");
}); });
} }
public void testCompoundAssignment() { public void testCompoundAssignment() {
// byte // byte
assertEquals((byte) 15, exec("byte x = 5; x += 10; return x;")); assertEquals((byte) 15, exec("byte x = 5; x += 10; return x;"));
@ -406,7 +406,7 @@ public class AdditionTests extends ScriptTestCase {
assertEquals(15D, exec("double x = 5.0; x += 10; return x;")); assertEquals(15D, exec("double x = 5.0; x += 10; return x;"));
assertEquals(-5D, exec("double x = 5.0; x += -10; return x;")); assertEquals(-5D, exec("double x = 5.0; x += -10; return x;"));
} }
public void testDefCompoundAssignmentLHS() { public void testDefCompoundAssignmentLHS() {
// byte // byte
assertEquals((byte) 15, exec("def x = (byte)5; x += 10; return x;")); assertEquals((byte) 15, exec("def x = (byte)5; x += 10; return x;"));
@ -431,7 +431,7 @@ public class AdditionTests extends ScriptTestCase {
assertEquals(15D, exec("def x = 5.0; x += 10; return x;")); assertEquals(15D, exec("def x = 5.0; x += 10; return x;"));
assertEquals(-5D, exec("def x = 5.0; x += -10; return x;")); assertEquals(-5D, exec("def x = 5.0; x += -10; return x;"));
} }
public void testDefCompoundAssignmentRHS() { public void testDefCompoundAssignmentRHS() {
// byte // byte
assertEquals((byte) 15, exec("byte x = 5; def y = 10; x += y; return x;")); assertEquals((byte) 15, exec("byte x = 5; def y = 10; x += y; return x;"));

View File

@ -318,4 +318,13 @@ public class CastTests extends ScriptTestCase {
exec("def x = 5L; boolean y = (boolean) (x + x); return y"); exec("def x = 5L; boolean y = (boolean) (x + x); return y");
}); });
} }
public void testIllegalVoidCasts() {
expectScriptThrows(IllegalArgumentException.class, () -> {
exec("def map = ['a': 1,'b': 2,'c': 3]; map.c = Collections.sort(new ArrayList(map.keySet()));");
});
expectScriptThrows(IllegalArgumentException.class, () -> {
exec("Map map = ['a': 1,'b': 2,'c': 3]; def x = new HashMap(); x.put(1, map.clear());");
});
}
} }