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

org.mvel2.optimizers.impl.refl.nodes.GetterAccessor Maven / Gradle / Ivy

/**
 * MVEL (The MVFLEX Expression Language)
 *
 * Copyright (C) 2007 Christopher Brock, MVFLEX/Valhalla Project and the Codehaus
 *
 * 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 org.mvel2.optimizers.impl.refl.nodes;

import static org.mvel2.MVEL.getProperty;
import static org.mvel2.util.ParseTools.getBestCandidate;
import static org.mvel2.util.ReflectionUtil.getPropertyFromAccessor;

import java.lang.reflect.Method;

import org.mvel2.CompileException;
import org.mvel2.compiler.AccessorNode;
import org.mvel2.integration.VariableResolverFactory;

public class GetterAccessor implements AccessorNode {

    public static final Object[] EMPTY = new Object[0];
    private final Method method;
    private AccessorNode nextNode;

    public GetterAccessor(Method method) {
        this.method = method;
    }

    public Object getValue(Object ctx, Object elCtx, VariableResolverFactory vars) {
        try {
            if (nextNode != null) {
                return nextNode.getValue(method.invoke(ctx, EMPTY), elCtx, vars);
            } else {
                return method.invoke(ctx, EMPTY);
            }
        } catch (IllegalArgumentException e) {
            if (ctx != null && method.getDeclaringClass() != ctx.getClass()) {
                Method o = getBestCandidate(EMPTY, method.getName(), ctx.getClass(), ctx.getClass().getMethods(), true);
                if (o != null) {
                    return executeOverrideTarget(o, ctx, elCtx, vars);
                }
            }

            /**
             * HACK: Try to access this another way.
             */
            if (nextNode != null) {
                return nextNode.getValue(getProperty(getPropertyFromAccessor(method.getName()), ctx), elCtx, vars);
            } else {
                return getProperty(getPropertyFromAccessor(method.getName()), ctx);
            }
        } catch (NullPointerException e) {
            if (ctx == null) {
                throw new RuntimeException("unable to invoke method: " + method.getDeclaringClass().getName() + "." + method.getName()
                        + ": " + "target of method is null", e);
            } else {
                throw new RuntimeException("cannot invoke getter: " + method.getName() + " (see trace)", e);
            }
        } catch (Exception e) {
            throw new RuntimeException(
                    "cannot invoke getter: " + method.getName() + " [declr.class: " + method.getDeclaringClass().getName() + "; act.class: "
                            + (ctx != null ? ctx.getClass().getName() : "null") + "] (see trace)",
                    e);
        }
    }

    public Method getMethod() {
        return method;
    }

    public AccessorNode setNextNode(AccessorNode nextNode) {
        return this.nextNode = nextNode;
    }

    public AccessorNode getNextNode() {
        return nextNode;
    }

    public String toString() {
        return method.getDeclaringClass().getName() + "." + method.getName();
    }

    public Object setValue(Object ctx, Object elCtx, VariableResolverFactory vars, Object value) {
        try {
            if (nextNode != null) {
                return nextNode.setValue(method.invoke(ctx, EMPTY), elCtx, vars, value);
            } else {
                throw new RuntimeException("bad payload");
            }
        } catch (IllegalArgumentException e) {
            /**
             * HACK: Try to access this another way.
             */

            if (nextNode != null) {
                return nextNode.setValue(getProperty(getPropertyFromAccessor(method.getName()), ctx), elCtx, vars, value);
            } else {
                return getProperty(getPropertyFromAccessor(method.getName()), ctx);
            }
        } catch (CompileException e) {
            throw e;
        } catch (Exception e) {
            throw new RuntimeException("error " + method.getName() + ": " + e.getClass().getName() + ":" + e.getMessage(), e);
        }
    }

    public Class getKnownEgressType() {
        return method.getReturnType();
    }

    private Object executeOverrideTarget(Method o, Object ctx, Object elCtx, VariableResolverFactory vars) {
        try {
            if (nextNode != null) {
                return nextNode.getValue(o.invoke(ctx, EMPTY), elCtx, vars);
            } else {
                return o.invoke(ctx, EMPTY);
            }
        } catch (Exception e2) {
            throw new RuntimeException("unable to invoke method", e2);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy