All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.hazelcast.org.codehaus.commons.compiler.IExpressionEvaluator Maven / Gradle / Ivy

There is a newer version: 5.5.0
Show newest version

/*
 * Janino - An embedded Java[TM] compiler
 *
 * Copyright (c) 2001-2010 Arno Unkrig. All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
 * following conditions are met:
 *
 *    1. Redistributions of source code must retain the above copyright notice, this list of conditions and the
 *       following disclaimer.
 *    2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
 *       following disclaimer in the documentation and/or other materials provided with the distribution.
 *    3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote
 *       products derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package com.hazelcast.org.codehaus.commons.compiler;

import java.io.IOException;
import java.io.Reader;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import com.hazelcast.org.codehaus.commons.compiler.lang.ClassLoaders;
import com.hazelcast.org.codehaus.commons.nullanalysis.Nullable;

/**
 * An engine that evaluates expressions in JVM bytecode.
 * 

* The syntax of the expression to compile is that of a Java expression, as defined in JLS7, section 15. Notice * that a Java expression does not have a concluding semicolon. *

*

* Example: *

*
 *   a + 7 * b
 * 
*

* (Notice that this expression refers to two parameters "a" and "b", as explained below.) *

*

* The expression may optionally be preceeded with a sequence of import directives like *

*
 *   import java.text.*;
 *   new DecimalFormat("####,###.##").format(10200020.345345)
 * 
*

* (Notice that the import directive is concluded with a semicolon, while the expression is not.) This feature is not * available if you compile many expressions at a time (see below). *

*

* To set up an {@link IExpressionEvaluator} object, proceed as follows: *

*
    *
  1. Create an {@link IExpressionEvaluator}-derived class
  2. *
  3. Configure the {@link IExpressionEvaluator} by calling any of the following methods:
  4. *
      *
    • {@link #setExpressionType(Class)} *
    • {@link #setParameters(String[], Class[])} *
    • {@link #setThrownExceptions(Class[])} *
    • {@link #setParentClassLoader(ClassLoader)} *
    • {@link #setDefaultImports(String[])} *
    *
  5. * Call any of the {@link #cook(String, java.io.Reader)} methods to scan, parse, compile and load the expression * into the JVM. *
  6. *
*

* After the {@link IExpressionEvaluator} object is set up, the expression can be evaluated as often with different * parameter values (see {@link #evaluate(Object[])}). This evaluation is very fast, compared to the compilation. *

*

* Less common methods exist that allow for the specification of the name of the generated class, the class it * extends, the interfaces it implements, the name of the method that executes the expression, the exceptions that * this method (i.e. the expression) is allowed to throw, and the {@link ClassLoader} that is used to define the * generated class and to load classes referenced by the expression. *

*

* If you want to compile many expressions at the same time, you have the option to cook an array of * expressions in one {@link IExpressionEvaluator} by using the following methods: *

*
    *
  • {@link #setMethodNames(String[])} *
  • {@link #setParameters(String[][], Class[][])} *
  • {@link #setExpressionTypes(Class[])} *
  • {@link #setStaticMethod(boolean[])} *
  • {@link #setThrownExceptions(Class[][])} *
  • {@link #cook(String[], Reader[])} *
  • {@link #evaluate(int, Object[])} *
*

* Notice that these methods have array parameters in contrast to their one-expression brethren. *

*

* Notice that for functionally identical {@link IExpressionEvaluator}s, {@link * java.lang.Object#equals(java.lang.Object)} will return {@code true}. E.g. "a+b" and "c + d" are functionally * identical if "a" and "c" have the same type, and so do "b" and "d". *

*

* 'JLS7' refers to the Java Language Specification, Java SE 7 * Edition. *

*/ public interface IExpressionEvaluator extends ICookable, IMultiCookable { /** * The fully qualified name of the generated class, iff not reconfigured by {@link #setClassName(String)}. */ String DEFAULT_CLASS_NAME = "SC"; /** * The type of all expressions that were not reconfigured with {@link #setExpressionTypes(Class[])}. */ Class DEFAULT_EXPRESSION_TYPE = Object.class; /** * Special value for {@link #setExpressionType(Class)} that indicates that the expression may have any type. * * @deprecated Since autoboxing was introduced in JANINO, this feature is no longer necessary because you can use * expression type {@link Object}{@code .class} */ @Deprecated Class ANY_TYPE = Object.class; /** * The "parent class loader" is used to load referenced classes. Useful values are: * * * * * * * * * *
{@code System.getSystemClassLoader()}The running JVM's class path
{@code Thread.currentThread().getContextClassLoader()} or {@code null}The class loader effective for the invoking thread
{@link ClassLoaders#BOOTCLASSPATH_CLASS_LOADER}The running JVM's boot class path
*

* The parent class loader defaults to the current thread's context class loader. *

*/ void setParentClassLoader(@Nullable ClassLoader parentClassLoader); /** * Determines what kind of debugging information is included in the generates classes. The default is typically * "{@code -g:none}". */ void setDebuggingInformation(boolean debugSource, boolean debugLines, boolean debugVars); /** * By default, {@link CompileException}s are thrown on compile errors, but an application my install its own * {@link ErrorHandler}. *

* Be aware that a single problem during compilation often causes a bunch of compile errors, so a good {@link * ErrorHandler} counts errors and throws a {@link CompileException} when a limit is reached. *

*

* If the given {@link ErrorHandler} throws {@link CompileException}s, then the compilation is terminated and * the exception is propagated. *

*

* If the given {@link ErrorHandler} does not throw {@link CompileException}s, then the compiler may or may not * continue compilation, but must eventually throw a {@link CompileException}. *

*

* In other words: The {@link ErrorHandler} may throw a {@link CompileException} or not, but the compiler must * definitely throw a {@link CompileException} if one or more compile errors have occurred. *

* * @param compileErrorHandler {@code null} to restore the default behavior (throwing a {@link CompileException} */ void setCompileErrorHandler(@Nullable ErrorHandler compileErrorHandler); /** * By default, warnings are discarded, but an application my install a custom {@link WarningHandler}. * * @param warningHandler {@code null} to indicate that no warnings be issued */ void setWarningHandler(@Nullable WarningHandler warningHandler); /** * Evaluates the expression with concrete parameter values. *

* Each argument value must have the same type as specified through the "parameterTypes" parameter of {@link * #setParameters(String[], Class[])}. *

*

* Arguments of primitive type must passed with their wrapper class objects. *

*

* The object returned has the class as specified through {@link #setExpressionType(Class)}. *

*

* This method is thread-safe. *

*

* {@code Null} arguments is equivalent with {@code new Object[0]}. *

* * @param arguments The actual parameter values */ @Nullable Object evaluate(@Nullable Object... arguments) throws InvocationTargetException; /** * Reconfigures the "default expression type"; if no expression type is configured for an expression, then, when * cooking this {@link IExpressionEvaluator}, the "default expression type" is used for the expression */ void setDefaultExpressionType(Class defaultExpressionType); /** * @return The currently configured "default expression type" * @see #setDefaultExpressionType(Class) */ Class getDefaultExpressionType(); /** * Configures the interfaces that the generated class implements. */ void setImplementedInterfaces(Class[] implementedTypes); /** * @deprecated Use {@link #setExpressionType(Class)} instead */ @Deprecated void setReturnType(@Deprecated Class returnType); /** * Defines the type of the expression. *

* Defaults to {@link Object}{@code .class}, which, thanks to autoboxing, allows for any expression type * (including primitive types). *

*

* If {@code expressionType} is {@code void.class}, then the expression must be an invocation of a {@code void} * method. *

*/ void setExpressionType(Class expressionType); /** * Configures the types of the expressions. *

* Unless this method is called, all expressions have type {@link Object}{@code .class}. *

*

* If any expression has type {@code void.class}, then it must be an invocation of a {@code void} method. *

*/ void setExpressionTypes(Class[] expressionTypes); /** @see IScriptEvaluator#setOverrideMethod(boolean) */ void setOverrideMethod(boolean overrideMethod); /** @see IScriptEvaluator#setOverrideMethod(boolean[]) */ void setOverrideMethod(boolean[] overrideMethod); /** @see IScriptEvaluator#setParameters(String[], Class[]) */ void setParameters(String[] parameterNames, Class[] parameterTypes); /** @see IScriptEvaluator#setParameters(String[][], Class[][]) */ void setParameters(String[][] parameterNames, Class[][] parameterTypes); /** @see IClassBodyEvaluator#setClassName(String) */ void setClassName(String className); /** @see IClassBodyEvaluator#setExtendedClass(Class) */ void setExtendedClass(Class extendedType); /** @see IClassBodyEvaluator#setDefaultImports(String...) */ void setDefaultImports(String... defaultImports); /** @see IClassBodyEvaluator#getDefaultImports() */ String[] getDefaultImports(); /** @see IScriptEvaluator#setStaticMethod(boolean) */ void setStaticMethod(boolean staticMethod); /** @see IScriptEvaluator#setStaticMethod(boolean[]) */ void setStaticMethod(boolean[] staticMethod); /** @see IScriptEvaluator#setMethodName(String) */ void setMethodName(String methodName); /** @see IScriptEvaluator#setMethodNames(String[]) */ void setMethodNames(String[] methodNames); /** @see IScriptEvaluator#setThrownExceptions(Class[]) */ void setThrownExceptions(Class[] thrownExceptions); /** @see IScriptEvaluator#setThrownExceptions(Class[][]) */ void setThrownExceptions(Class[][] thrownExceptions); /** * {@code Null} arguments is equivalent with {@code new Object[0]}. */ @Nullable Object evaluate(int idx, @Nullable Object... arguments) throws InvocationTargetException; /** * If the parameter and return types of the expression are known at compile time, then a "fast" expression evaluator * can be instantiated through {@link #createFastEvaluator(String, Class, String[])}. Expression evaluation is * faster than through {@link #evaluate(Object[])}, because it is not done through reflection but through direct * method invocation. *

* Example: *

*
     * public interface Foo {
     *     int bar(int a, int b);
     * }
     * ...
     * ExpressionEvaluator ee = CompilerFactoryFactory.getDefaultCompilerFactory().newExpressionEvaluator();
     *
     * // Optionally configure the EE here...
     * ee.{@link #setClassName(String) setClassName}("Bar");
     * ee.{@link #setDefaultImports(String[]) setDefaultImports}(new String[] { "java.util.*" });
     * ee.{@link #setExtendedClass(Class) setExtendedClass}(SomeOtherClass.class);
     * ee.{@link #setParentClassLoader(ClassLoader) setParentClassLoader}(someClassLoader);
     *
     * // Optionally configure the EE here...
     * Foo f = (Foo) ee.createFastEvaluator(
     *     "a + b",                    // expression to evaluate
     *     Foo.class,                  // interface that describes the expression's signature
     *     new String[] { "a", "b" }   // the parameters' names
     * );
     * System.out.println("1 + 2 = " + f.bar(1, 2)); // Evaluate the expression
     * 
*

* All other configuration (implemented type, static method, return type, method name, parameter names and types, * thrown exceptions) are predetermined by the interfaceToImplement. *

*

* Notice: The {@code interfaceToImplement} must be accessible by the compiled class, i.e. either be declared * {@code public}, or with {@code protected} or default access in the package of the compiled class (see {@link * #setClassName(String)}. *

*/ T createFastEvaluator(String expression, Class interfaceToImplement, String... parameterNames) throws CompileException; /** @see #createFastEvaluator(String, Class, String[]) */ T createFastEvaluator(Reader reader, Class interfaceToImplement, String... parameterNames) throws CompileException, IOException; /** @see IScriptEvaluator#getMethod() */ Method getMethod(); /** @see IScriptEvaluator#getMethod(int) */ Method getMethod(int idx); /** @see IClassBodyEvaluator#getClazz() */ Class getClazz(); /** @see IScriptEvaluator#getResult() */ Method[] getResult(); }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy