| /* |
| * 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 metaprogramming |
| |
| |
| import org.codehaus.groovy.ast.ASTNode |
| import org.codehaus.groovy.ast.ClassHelper |
| import org.codehaus.groovy.ast.ClassNode |
| import org.codehaus.groovy.ast.MethodNode |
| import org.codehaus.groovy.ast.Parameter |
| import org.codehaus.groovy.ast.expr.ConstantExpression |
| import org.codehaus.groovy.ast.stmt.ReturnStatement |
| import org.codehaus.groovy.control.CompilePhase |
| import org.codehaus.groovy.control.SourceUnit |
| import org.codehaus.groovy.transform.AbstractASTTransformation |
| import org.codehaus.groovy.transform.GroovyASTTransformation |
| import org.codehaus.groovy.transform.GroovyASTTransformationClass |
| import org.junit.jupiter.api.Test |
| |
| import java.lang.annotation.ElementType |
| import java.lang.annotation.Retention |
| import java.lang.annotation.RetentionPolicy |
| import java.lang.annotation.Target |
| |
| import static groovy.test.GroovyAssert.assertScript |
| import static org.objectweb.asm.Opcodes.ACC_PUBLIC |
| |
| class MacroStatementTest { |
| |
| @Test |
| void testOlderASTTransformation() { |
| assertScript ''' |
| package metaprogramming |
| |
| @AddMethod |
| class A { |
| |
| } |
| |
| new A().message == '42' |
| ''' |
| } |
| |
| @Test |
| void testOlderASTTransformationWithMacros() { |
| assertScript ''' |
| package metaprogramming |
| |
| @AddMethodWithMacros |
| class A { |
| |
| } |
| |
| new A().message == '42' |
| ''' |
| } |
| } |
| |
| // tag::addmethodannotation[] |
| @Retention(RetentionPolicy.SOURCE) |
| @Target([ElementType.TYPE]) |
| @GroovyASTTransformationClass(["metaprogramming.AddMethodASTTransformation"]) |
| @interface AddMethod { } |
| // end::addmethodannotation[] |
| |
| // tag::addmethodtransformationwithoutmacro[] |
| @GroovyASTTransformation(phase = CompilePhase.INSTRUCTION_SELECTION) |
| class AddMethodASTTransformation extends AbstractASTTransformation { |
| @Override |
| void visit(ASTNode[] nodes, SourceUnit source) { |
| ClassNode classNode = (ClassNode) nodes[1] |
| |
| ReturnStatement code = |
| new ReturnStatement( // <1> |
| new ConstantExpression("42")) // <2> |
| |
| MethodNode methodNode = |
| new MethodNode( |
| "getMessage", |
| ACC_PUBLIC, |
| ClassHelper.make(String), |
| [] as Parameter[], |
| [] as ClassNode[], |
| code) // <3> |
| |
| classNode.addMethod(methodNode) // <4> |
| } |
| } |
| // end::addmethodtransformationwithoutmacro[] |
| |
| @Retention(RetentionPolicy.SOURCE) |
| @Target([ElementType.TYPE]) |
| @GroovyASTTransformationClass(["metaprogramming.AddMethodWithMacrosASTTransformation"]) |
| @interface AddMethodWithMacros { } |
| |
| // tag::basicWithMacro[] |
| @GroovyASTTransformation(phase = CompilePhase.INSTRUCTION_SELECTION) |
| class AddMethodWithMacrosASTTransformation extends AbstractASTTransformation { |
| @Override |
| void visit(ASTNode[] nodes, SourceUnit source) { |
| ClassNode classNode = (ClassNode) nodes[1] |
| |
| ReturnStatement simplestCode = macro { return "42" } // <1> |
| |
| MethodNode methodNode = |
| new MethodNode( |
| "getMessage", |
| ACC_PUBLIC, |
| ClassHelper.make(String), |
| [] as Parameter[], |
| [] as ClassNode[], |
| simplestCode) // <2> |
| |
| classNode.addMethod(methodNode) // <3> |
| } |
| } |
| // end::basicWithMacro[] |