From 896e968cb10bea8b622d1f657b62458fee9c67f4 Mon Sep 17 00:00:00 2001 From: Stuart Tettemer Date: Tue, 4 Jan 2022 15:14:51 -0600 Subject: [PATCH 1/2] Script: track this pointer capture from blocks within lambdas (#82228) * Script: track this pointer capture from blocks within lambdas If a lambda contains a block that calls into a user function, the painless compiler was not tracking that the lambda needs to capture the `this` pointer. Scripts that attempted to use a lambda that calls a user function from inside a block would trigger an `illegal_state_exception`: > no 'this' pointer within static method `BlockScope` now forwards `setUsesInstanceMethod` calls to it's parent, which may be a `LambdaScope`. `LambdaScope` will also forward `setUsesInstanceMethod` to it's parent after setting it's own `usesInstanceMethod` flag, propagating `this` pointer capture in the case of nested lambdas. Fixes: #82224 --- docs/changelog/82228.yaml | 5 +++ .../painless/symbol/SemanticScope.java | 12 ++++++- .../painless/UserFunctionTests.java | 36 +++++++++++++++++++ 3 files changed, 52 insertions(+), 1 deletion(-) create mode 100644 docs/changelog/82228.yaml diff --git a/docs/changelog/82228.yaml b/docs/changelog/82228.yaml new file mode 100644 index 0000000000000..5bf707b6dddde --- /dev/null +++ b/docs/changelog/82228.yaml @@ -0,0 +1,5 @@ +pr: 82228 +summary: "Script: track this pointer capture from blocks within lambdas" +area: Infra/Scripting +type: bug +issues: [] diff --git a/modules/lang-painless/src/main/java/org/elasticsearch/painless/symbol/SemanticScope.java b/modules/lang-painless/src/main/java/org/elasticsearch/painless/symbol/SemanticScope.java index 4f05606dffa9d..3e3c4b9e8ad46 100644 --- a/modules/lang-painless/src/main/java/org/elasticsearch/painless/symbol/SemanticScope.java +++ b/modules/lang-painless/src/main/java/org/elasticsearch/painless/symbol/SemanticScope.java @@ -198,6 +198,9 @@ public void setUsesInstanceMethod() { return; } usesInstanceMethod = true; + if (parent != null) { + parent.setUsesInstanceMethod(); + } } @Override @@ -261,6 +264,12 @@ public Class getReturnType() { public String getReturnCanonicalTypeName() { return parent.getReturnCanonicalTypeName(); } + + @Override + // If the parent scope is a lambda, we want to track this usage, so forward call to parent. + public void setUsesInstanceMethod() { + parent.setUsesInstanceMethod(); + } } /** @@ -356,7 +365,8 @@ public Variable defineVariable(Location location, Class type, String name, bo public abstract Variable getVariable(Location location, String name); - // We only want to track instance method use inside of lambdas for "this" injection. It's a noop for other scopes. + // We only want to track instance method use inside of lambdas (and blocks inside lambdas) for "this" injection. + // It's a noop for other scopes. public void setUsesInstanceMethod() {} public boolean usesInstanceMethod() { diff --git a/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java b/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java index 757b5a4178390..f3dfaef1b8052 100644 --- a/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java +++ b/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java @@ -140,4 +140,40 @@ public void testLambdaCapture() { assertEquals(org.elasticsearch.core.List.of(100, 1, -100), exec(source, org.elasticsearch.core.Map.of("a", 1), false)); assertBytecodeExists(source, "public static synthetic lambda$synthetic$0(ILjava/lang/Object;Ljava/lang/Object;)I"); } + + public void testCallUserMethodFromStatementWithinLambda() { + String source = "" + + "int test1() { return 1; }" + + "void test(Map params) { " + + " int i = 0;" + + " params.forEach(" + + " (k, v) -> { if (i == 0) { test1() } else { 20 } }" + + " );" + + "}" + + "test(params)"; + assertNull(exec(source, Map.of("a", 5), false)); + assertBytecodeExists(source, "public synthetic lambda$synthetic$0(ILjava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;"); + } + + public void testCallUserMethodFromStatementWithinNestedLambda() { + String source = "" + + "int test1() { return 1; }" + + "void test(Map params) { " + + " int i = 0;" + + " int j = 5;" + + " params.replaceAll( " + + " (n, m) -> {" + + " m.forEach(" + + " (k, v) -> { if (i == 0) { test1() } else { 20 } }" + + " );" + + " return ['aaa': j];" + + " }" + + " );" + + "}" + + "Map myParams = new HashMap(params);" + + "test(myParams);" + + "myParams['a']['aaa']"; + assertEquals(5, exec(source, Map.of("a", Map.of("b", 1)), false)); + assertBytecodeExists(source, "public synthetic lambda$synthetic$1(IILjava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;"); + } } From fc6875f022aacde5c67cebf4c711881337e0fb10 Mon Sep 17 00:00:00 2001 From: Stuart Tettemer Date: Tue, 4 Jan 2022 15:35:06 -0600 Subject: [PATCH 2/2] org.elasticsearch.core.Map.of --- .../java/org/elasticsearch/painless/UserFunctionTests.java | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java b/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java index f3dfaef1b8052..c336794d28e4e 100644 --- a/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java +++ b/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java @@ -151,7 +151,7 @@ public void testCallUserMethodFromStatementWithinLambda() { + " );" + "}" + "test(params)"; - assertNull(exec(source, Map.of("a", 5), false)); + assertNull(exec(source, org.elasticsearch.core.Map.of("a", 5), false)); assertBytecodeExists(source, "public synthetic lambda$synthetic$0(ILjava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;"); } @@ -173,7 +173,7 @@ public void testCallUserMethodFromStatementWithinNestedLambda() { + "Map myParams = new HashMap(params);" + "test(myParams);" + "myParams['a']['aaa']"; - assertEquals(5, exec(source, Map.of("a", Map.of("b", 1)), false)); + assertEquals(5, exec(source, org.elasticsearch.core.Map.of("a", org.elasticsearch.core.Map.of("b", 1)), false)); assertBytecodeExists(source, "public synthetic lambda$synthetic$1(IILjava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;"); } }