/* * 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 com.facebook.presto.bytecode.expression; import com.facebook.presto.bytecode.BytecodeNode; import com.facebook.presto.bytecode.ClassDefinition; import com.facebook.presto.bytecode.ClassInfoLoader; import com.facebook.presto.bytecode.DumpBytecodeVisitor; import com.facebook.presto.bytecode.DynamicClassLoader; import com.facebook.presto.bytecode.MethodDefinition; import com.facebook.presto.bytecode.ParameterizedType; import com.facebook.presto.bytecode.Scope; import com.facebook.presto.bytecode.SmartClassWriter; import com.google.common.collect.ImmutableList; import org.objectweb.asm.ClassWriter; import java.util.Optional; import java.util.function.Function; import static com.facebook.presto.bytecode.Access.FINAL; import static com.facebook.presto.bytecode.Access.PUBLIC; import static com.facebook.presto.bytecode.Access.STATIC; import static com.facebook.presto.bytecode.Access.a; import static com.facebook.presto.bytecode.CompilerUtils.makeClassName; import static com.facebook.presto.bytecode.ParameterizedType.type; import static org.testng.Assert.assertEquals; public final class BytecodeExpressionAssertions { private BytecodeExpressionAssertions() { } private static final boolean DUMP_BYTE_CODE_TREE = false; static void assertBytecodeExpressionType(BytecodeExpression expression, ParameterizedType type) { assertEquals(expression.getType(), type); } public static void assertBytecodeExpression(BytecodeExpression expression, Object expected, String expectedRendering) throws Exception { assertBytecodeExpression(expression, expected, expectedRendering, Optional.empty()); } public static void assertBytecodeExpression(BytecodeExpression expression, Object expected, String expectedRendering, Optional<ClassLoader> parentClassLoader) throws Exception { assertEquals(expression.toString(), expectedRendering); assertBytecodeNode(expression.ret(), expression.getType(), expected, parentClassLoader); } public static void assertBytecodeExpression(BytecodeExpression expression, Object expected, ClassLoader parentClassLoader) throws Exception { assertBytecodeExpression(expression, expected, Optional.of(parentClassLoader)); } public static void assertBytecodeExpression(BytecodeExpression expression, Object expected, Optional<ClassLoader> parentClassLoader) throws Exception { assertBytecodeNode(expression.ret(), expression.getType(), expected, parentClassLoader); } public static void assertBytecodeNode(BytecodeNode node, ParameterizedType returnType, Object expected) throws Exception { assertBytecodeNode(node, returnType, expected, Optional.empty()); } public static void assertBytecodeNode(BytecodeNode node, ParameterizedType returnType, Object expected, Optional<ClassLoader> parentClassLoader) throws Exception { assertEquals(execute(context -> node, returnType, parentClassLoader), expected); } public static void assertBytecodeNode(Function<Scope, BytecodeNode> nodeGenerator, ParameterizedType returnType, Object expected) throws Exception { assertBytecodeNode(nodeGenerator, returnType, expected, Optional.empty()); } public static void assertBytecodeNode(Function<Scope, BytecodeNode> nodeGenerator, ParameterizedType returnType, Object expected, Optional<ClassLoader> parentClassLoader) throws Exception { assertEquals(execute(nodeGenerator, returnType, parentClassLoader), expected); } public static Object execute(Function<Scope, BytecodeNode> nodeGenerator, ParameterizedType returnType, Optional<ClassLoader> parentClassLoader) throws Exception { ClassDefinition classDefinition = new ClassDefinition( a(PUBLIC, FINAL), makeClassName("Test"), type(Object.class)); MethodDefinition method = classDefinition.declareMethod(a(PUBLIC, STATIC), "test", returnType); BytecodeNode node = nodeGenerator.apply(method.getScope()); method.getBody().append(node); if (DUMP_BYTE_CODE_TREE) { DumpBytecodeVisitor dumpBytecode = new DumpBytecodeVisitor(System.out); dumpBytecode.visitClass(classDefinition); } DynamicClassLoader classLoader = new DynamicClassLoader(parentClassLoader.orElse(BytecodeExpressionAssertions.class.getClassLoader())); ClassWriter cw = new SmartClassWriter(ClassInfoLoader.createClassInfoLoader(ImmutableList.of(classDefinition), classLoader)); classDefinition.visit(cw); Class<?> clazz = classLoader.defineClass(classDefinition.getType().getJavaClassName(), cw.toByteArray()); return clazz.getMethod("test").invoke(null); } }