jetbrick.template.parser.ast.AstInvokeField 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.SignatureUtils;
import jetbrick.template.resolver.property.Getter;
import jetbrick.template.runtime.InterpretContext;
import jetbrick.template.runtime.InterpretException;
public final class AstInvokeField extends AstExpression {
private final AstExpression objectExpression;
private final String name;
private final boolean nullSafe;
private Getter last;
private boolean unsafe;
public AstInvokeField(AstExpression objectExpression, String name, boolean nullSafe, Position position) {
super(position);
this.objectExpression = objectExpression;
this.name = name;
this.nullSafe = nullSafe;
this.last = null;
this.unsafe = true;
}
@Override
public Object execute(InterpretContext ctx) throws InterpretException {
Object object = objectExpression.execute(ctx);
if (object == null) {
if (nullSafe || ctx.getTemplate().getOption().isSafecall()) {
return null;
}
throw new InterpretException(Errors.EXPRESSION_OBJECT_IS_NULL).set(position);
} else if (object == ALU.VOID) {
throw new InterpretException(Errors.EXPRESSION_OBJECT_IS_VOID).set(position);
}
return doInvokeGetter(ctx, last, object);
}
private Object doInvokeGetter(InterpretContext ctx, Getter getter, Object object) throws InterpretException {
boolean useLatest = (getter != null);
if (getter == null) {
Class> objectClass = objectExpression.getResultType(ctx.getValueStack(), object);
getter = ctx.getGlobalResolver().resolveGetter(objectClass, name);
if (getter == null) {
String signature = SignatureUtils.getFieldSignature(objectClass, name);
throw new InterpretException(Errors.PROPERTY_NOT_FOUND, signature).set(position);
}
this.last = getter; // 找到一个新的 getter
}
if (unsafe) {
JetSecurityManager securityManager = ctx.getSecurityManager();
if (securityManager != null) {
try {
getter.checkAccess(securityManager);
} catch (RuntimeException e) {
throw new InterpretException(e).set(position);
}
}
unsafe = false;
}
try {
return getter.get(object);
} catch (InterpretException e) {
throw e;
} catch (RuntimeException e) {
if (useLatest && Errors.isReflectIllegalArgument(e)) {
// 重新查找匹配的 Getter
return doInvokeGetter(ctx, null, object);
}
String signature = object.getClass().getName() + "#" + name;
throw new InterpretException(Errors.PROPERTY_GET_ERROR, signature).cause(e).set(position);
}
}
}