BytecodeExpressionAssertions.java
/*
* 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.MethodDefinition;
import com.facebook.presto.bytecode.ParameterizedType;
import com.facebook.presto.bytecode.Scope;
import java.util.Optional;
import java.util.concurrent.atomic.AtomicBoolean;
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.BytecodeUtils.dumpBytecodeTree;
import static com.facebook.presto.bytecode.BytecodeUtils.makeClassName;
import static com.facebook.presto.bytecode.ClassGenerator.classGenerator;
import static com.facebook.presto.bytecode.ParameterizedType.type;
import static org.testng.Assert.assertEquals;
public final class BytecodeExpressionAssertions
{
private BytecodeExpressionAssertions()
{
}
public static final AtomicBoolean DUMP_BYTECODE_TREE = new AtomicBoolean();
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);
String tree = dumpBytecodeTree(classDefinition);
if (DUMP_BYTECODE_TREE.get()) {
System.out.println(tree);
}
ClassLoader classLoader = parentClassLoader.orElseGet(BytecodeExpressionAssertions.class::getClassLoader);
return classGenerator(classLoader)
.defineClass(classDefinition, Object.class)
.getMethod("test")
.invoke(null);
}
}