jetbrick.template.parser.ast.AstInvokeFunction Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jetbrick-template Show documentation
Show all versions of jetbrick-template Show documentation
Next generation template engine for Java
/**
* Copyright 2013-2016 Guoqiang Chen, Shanghai, China. All rights reserved.
*
* Author: Guoqiang Chen
* Email: [email protected]
* WebURL: https://github.com/subchen
*
* 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 jetbrick.template.parser.ast;
import jetbrick.template.Errors;
import jetbrick.template.JetSecurityManager;
import jetbrick.template.resolver.ParameterUtils;
import jetbrick.template.resolver.SignatureUtils;
import jetbrick.template.resolver.function.FunctionInvoker;
import jetbrick.template.runtime.InterpretContext;
import jetbrick.template.runtime.InterpretException;
import jetbrick.util.ArrayUtils;
public final class AstInvokeFunction extends AstExpression {
private final String name;
private final AstExpressionList argumentList;
private FunctionInvoker last;
private boolean unsafe;
public AstInvokeFunction(String name, AstExpressionList argumentList, Position position) {
super(position);
this.name = name;
this.argumentList = argumentList;
this.last = null;
this.unsafe = true;
}
@Override
public Object execute(InterpretContext ctx) throws InterpretException {
Object[] arguments;
if (argumentList == null) {
arguments = ArrayUtils.EMPTY_OBJECT_ARRAY;
} else {
arguments = argumentList.execute(ctx);
}
return doInvoke(ctx, last, arguments);
}
private Object doInvoke(InterpretContext ctx, FunctionInvoker fn, Object[] arguments) throws InterpretException {
boolean useLatest = (fn != null);
if (fn == null) {
Class>[] argumentTypes = ParameterUtils.getParameterTypes(arguments);
fn = ctx.getGlobalResolver().resolveFunction(name, argumentTypes);
if (fn == null) {
String signature = SignatureUtils.getFunctionSignature(name, argumentTypes);
throw new InterpretException(Errors.FUNCTION_NOT_FOUND, signature).set(position);
}
this.last = fn; // 找到一个新的 invoker
}
if (unsafe) {
JetSecurityManager securityManager = ctx.getSecurityManager();
if (securityManager != null) {
try {
fn.checkAccess(securityManager);
} catch (RuntimeException e) {
throw new InterpretException(e).set(position);
}
}
unsafe = false;
}
try {
return fn.invoke(arguments);
} catch (InterpretException e) {
throw e;
} catch (RuntimeException e) {
if (useLatest && Errors.isReflectIllegalArgument(e)) {
// 重新查找匹配的 Invoker
return doInvoke(ctx, null, arguments);
}
throw new InterpretException(Errors.FUNCTION_INVOKE_ERROR, fn.getSignature()).cause(e).set(position);
}
}
}