com.hazelcast.org.apache.calcite.interpreter.JaninoRexCompiler Maven / Gradle / Ivy
/*
* 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 com.hazelcast.org.apache.calcite.interpreter;
import com.hazelcast.org.apache.calcite.DataContext;
import com.hazelcast.org.apache.calcite.adapter.enumerable.JavaRowFormat;
import com.hazelcast.org.apache.calcite.adapter.enumerable.PhysTypeImpl;
import com.hazelcast.org.apache.calcite.adapter.enumerable.RexToLixTranslator;
import com.hazelcast.org.apache.calcite.config.CalciteSystemProperty;
import com.hazelcast.org.apache.calcite.jdbc.JavaTypeFactoryImpl;
import com.hazelcast.org.apache.calcite.linq4j.Ord;
import com.hazelcast.org.apache.calcite.linq4j.function.Function1;
import com.hazelcast.org.apache.calcite.linq4j.tree.BlockBuilder;
import com.hazelcast.org.apache.calcite.linq4j.tree.BlockStatement;
import com.hazelcast.org.apache.calcite.linq4j.tree.ClassDeclaration;
import com.hazelcast.org.apache.calcite.linq4j.tree.Expression;
import com.hazelcast.org.apache.calcite.linq4j.tree.Expressions;
import com.hazelcast.org.apache.calcite.linq4j.tree.MemberDeclaration;
import com.hazelcast.org.apache.calcite.linq4j.tree.ParameterExpression;
import com.hazelcast.org.apache.calcite.linq4j.tree.Statement;
import com.hazelcast.org.apache.calcite.rel.type.RelDataType;
import com.hazelcast.org.apache.calcite.rex.RexBuilder;
import com.hazelcast.org.apache.calcite.rex.RexNode;
import com.hazelcast.org.apache.calcite.rex.RexProgram;
import com.hazelcast.org.apache.calcite.rex.RexProgramBuilder;
import com.hazelcast.org.apache.calcite.sql.validate.SqlConformance;
import com.hazelcast.org.apache.calcite.sql.validate.SqlConformanceEnum;
import com.hazelcast.org.apache.calcite.util.BuiltInMethod;
import com.hazelcast.org.apache.calcite.util.Util;
import com.hazelcast.com.google.common.collect.ImmutableList;
import com.hazelcast.org.codehaus.commons.compiler.CompileException;
import com.hazelcast.org.codehaus.commons.compiler.CompilerFactoryFactory;
import com.hazelcast.org.codehaus.commons.compiler.IClassBodyEvaluator;
import com.hazelcast.org.codehaus.commons.compiler.ICompilerFactory;
import java.io.IOException;
import java.io.StringReader;
import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
/**
* Compiles a scalar expression ({@link RexNode}) to an expression that
* can be evaluated ({@link Scalar}) by generating a Java AST and compiling it
* to a class using Janino.
*/
public class JaninoRexCompiler implements Interpreter.ScalarCompiler {
private final RexBuilder rexBuilder;
public JaninoRexCompiler(RexBuilder rexBuilder) {
this.rexBuilder = rexBuilder;
}
@Override public Scalar.Producer compile(List nodes,
RelDataType inputRowType) {
final RexProgramBuilder programBuilder =
new RexProgramBuilder(inputRowType, rexBuilder);
for (RexNode node : nodes) {
programBuilder.addProject(node, null);
}
final RexProgram program = programBuilder.getProgram();
final BlockBuilder list = new BlockBuilder();
final BlockBuilder staticList = new BlockBuilder().withRemoveUnused(false);
final ParameterExpression context_ =
Expressions.parameter(Context.class, "context");
final ParameterExpression outputValues_ =
Expressions.parameter(Object[].class, "outputValues");
final JavaTypeFactoryImpl javaTypeFactory =
new JavaTypeFactoryImpl(rexBuilder.getTypeFactory().getTypeSystem());
// public void execute(Context, Object[] outputValues)
final RexToLixTranslator.InputGetter inputGetter =
new RexToLixTranslator.InputGetterImpl(
Expressions.field(context_,
BuiltInMethod.CONTEXT_VALUES.field),
PhysTypeImpl.of(javaTypeFactory, inputRowType,
JavaRowFormat.ARRAY, false));
final Function1 correlates = a0 -> {
throw new UnsupportedOperationException();
};
final Expression root =
Expressions.field(context_, BuiltInMethod.CONTEXT_ROOT.field);
final SqlConformance conformance =
SqlConformanceEnum.DEFAULT; // TODO: get this from implementor
final List expressionList =
RexToLixTranslator.translateProjects(program, javaTypeFactory,
conformance, list, staticList, null, root, inputGetter, correlates);
Ord.forEach(expressionList, (expression, i) ->
list.add(
Expressions.statement(
Expressions.assign(
Expressions.arrayIndex(outputValues_,
Expressions.constant(i)),
expression))));
return baz(context_, outputValues_, list.toBlock(),
staticList.toBlock().statements);
}
/** Given a method that implements {@link Scalar#execute(Context, Object[])},
* adds a bridge method that implements {@link Scalar#execute(Context)}, and
* compiles. */
static Scalar.Producer baz(ParameterExpression context_,
ParameterExpression outputValues_, BlockStatement block,
List declList) {
final List declarations = new ArrayList<>();
final List innerDeclarations = new ArrayList<>();
// public Scalar apply(DataContext root) {
// <>
// return new Scalar() {
// <>
// };
// }
final List statements = new ArrayList<>(declList);
statements.add(
Expressions.return_(null,
Expressions.new_(Scalar.class, ImmutableList.of(),
innerDeclarations)));
declarations.add(
Expressions.methodDecl(Modifier.PUBLIC, Scalar.class,
BuiltInMethod.FUNCTION_APPLY.method.getName(),
ImmutableList.of(DataContext.ROOT),
Expressions.block(statements)));
// (bridge method)
// public Object apply(Object root) {
// return this.apply((DataContext) root);
// }
final ParameterExpression objectRoot =
Expressions.parameter(Object.class, "root");
declarations.add(
Expressions.methodDecl(Modifier.PUBLIC, Object.class,
BuiltInMethod.FUNCTION_APPLY.method.getName(),
ImmutableList.of(
objectRoot),
Expressions.block(
Expressions.return_(null,
Expressions.call(
Expressions.parameter(Scalar.Producer.class, "this"),
BuiltInMethod.FUNCTION_APPLY.method,
Expressions.convert_(objectRoot,
DataContext.class))))));
// public void execute(Context, Object[] outputValues)
innerDeclarations.add(
Expressions.methodDecl(Modifier.PUBLIC, void.class,
BuiltInMethod.SCALAR_EXECUTE2.method.getName(),
ImmutableList.of(context_, outputValues_), block));
// public Object execute(Context)
final BlockBuilder builder = new BlockBuilder();
final Expression values_ = builder.append("values",
Expressions.newArrayBounds(Object.class, 1,
Expressions.constant(1)));
builder.add(
Expressions.statement(
Expressions.call(
Expressions.parameter(Scalar.class, "this"),
BuiltInMethod.SCALAR_EXECUTE2.method, context_, values_)));
builder.add(
Expressions.return_(null,
Expressions.arrayIndex(values_, Expressions.constant(0))));
innerDeclarations.add(
Expressions.methodDecl(Modifier.PUBLIC, Object.class,
BuiltInMethod.SCALAR_EXECUTE1.method.getName(),
ImmutableList.of(context_), builder.toBlock()));
final ClassDeclaration classDeclaration =
Expressions.classDecl(Modifier.PUBLIC, "Buzz", null,
ImmutableList.of(Scalar.Producer.class), declarations);
String s = Expressions.toString(declarations, "\n", false);
if (CalciteSystemProperty.DEBUG.value()) {
Util.debugCode(System.out, s);
}
try {
return getScalar(classDeclaration, s);
} catch (CompileException | IOException e) {
throw new RuntimeException(e);
}
}
static Scalar.Producer getScalar(ClassDeclaration expr, String s)
throws CompileException, IOException {
ICompilerFactory compilerFactory;
ClassLoader classLoader =
Objects.requireNonNull(JaninoRexCompiler.class.getClassLoader(), "classLoader");
try {
compilerFactory = CompilerFactoryFactory.getDefaultCompilerFactory(classLoader);
} catch (Exception e) {
throw new IllegalStateException(
"Unable to instantiate java compiler", e);
}
IClassBodyEvaluator cbe = compilerFactory.newClassBodyEvaluator();
cbe.setClassName(expr.name);
cbe.setImplementedInterfaces(new Class[] {Scalar.Producer.class});
cbe.setParentClassLoader(classLoader);
if (CalciteSystemProperty.DEBUG.value()) {
// Add line numbers to the generated janino class
cbe.setDebuggingInformation(true, true, true);
}
return (Scalar.Producer) cbe.createInstance(new StringReader(s));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy