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

com.caucho.el.MethodExpr Maven / Gradle / Ivy

/*
 * Copyright (c) 1998-2018 Caucho Technology -- all rights reserved
 *
 * This file is part of Resin(R) Open Source
 *
 * Each copy or derived work must preserve the copyright notice and this
 * notice unmodified.
 *
 * Resin Open Source is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * Resin Open Source is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
 * of NON-INFRINGEMENT.  See the GNU General Public License for more
 * details.
 *
 * You should have received a copy of the GNU General Public License
 * along with Resin Open Source; if not, write to the
 *
 *   Free Software Foundation, Inc.
 *   59 Temple Place, Suite 330
 *   Boston, MA 02111-1307  USA
 *
 * @author Scott Ferguson
 */

package com.caucho.el;

import com.caucho.vfs.WriteStream;

import javax.el.ELContext;
import javax.el.ELException;
import java.io.IOException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * Represents a named method call on an object.
 */
@SuppressWarnings("serial")
public class MethodExpr extends Expr {
  private transient ConcurrentHashMap,MethodCall> _methodMap
    = new ConcurrentHashMap,MethodCall>();
  
  private Expr _expr;
  private String _methodName;
  
  private Expr []_args;

  /**
   * Creates a new method expression.
   *
   * @param expr the expression generating the object on which the method 
   *        is to be called
   * @param methodName the name of the method to call
   * @param args the arguments for the method
   */
  public MethodExpr(Expr expr, String methodName, Expr []args)
  {
    _expr = expr;
    _methodName = methodName;
    _args = args;
  }
  
  /**
   * Evaluate the expression as an object.
   *
   * @param env the variable environment
   */
  @Override
  public Object getValue(ELContext env)
    throws ELException
  {
    Object aObj = _expr.getValue(env);

    if (aObj == null)
      return null;

    Object []objs = new Object[_args.length];

    try {
      MethodCall methodCall = findMethod(aObj.getClass());

      if (methodCall != null) {
        Marshall []marshall = methodCall.getMarshall();

        for (int j = 0; j < marshall.length; j++) {
          objs[j] = marshall[j].marshall(_args[j], env);
        }
        
        return methodCall.getMethod().invoke(aObj, objs);
      }
      
      return null;
    } catch (Exception e) {
      return invocationError(e);
    }
  }

  /**
   * Evaluates the expression, returning an object.
   *
   * @param env the variable environment
   *
   * @return the value of the expression as an object
   */
  @Override
  public Object invoke(ELContext env, Class []argTypes, Object []args)
    throws ELException
  {
    if (args != null && args.length != 0)
      throw new ELException(L.l("'{0}' is an illegal method invocation on {1}",
                                toString(), getClass().getName()));
    
    return getValue(env);
  }
  
  private MethodCall findMethod(Class type)
  {
    if (type == null)
      return null;
    
    MethodCall method = getMethodMap().get(type);
    
    if (method == null) {
      method = findMethodImpl(type);

      if (method != null)
        getMethodMap().put(type, method);
    }
    
    return method;
  }
  
    
  private MethodCall findMethodImpl(Class type)
  {
    if (type == null)
      return null;
    
    if (Modifier.isPublic(type.getModifiers())) {
      Method []methods = type.getDeclaredMethods();
      
      for (int i = 0; i < methods.length; i++) {
        Method method = methods[i];
        
        if (! Modifier.isPublic(method.getModifiers())) {
          continue;
        }
        
        Class []params = method.getParameterTypes();
        
        if (method.getName().equals(_methodName)
            && params.length == _args.length) {
          return new MethodCall(method);
        }
      }
    }
    
    Class []interfaces = type.getInterfaces();
    for (int i = 0; i < interfaces.length; i++) {
      MethodCall method = findMethod(interfaces[i]);
      
      if (method != null) {
        return method;
      }
    }
    
    return findMethod(type.getSuperclass());
  }
  
  private Map,MethodCall> getMethodMap()
  {
    if (_methodMap == null) {
      _methodMap = new ConcurrentHashMap,MethodCall>();
    }
      
    return _methodMap;
  }
  
  static Object evalArg(Class cl, Expr expr, ELContext env)
    throws ELException
  {
    Marshall marshall = Marshall.create(cl);

    return marshall.marshall(expr, env);
  }
  
  /**
   * Prints the code to create an LongLiteral.
   */
  @Override
  public void printCreate(WriteStream os)
    throws IOException
  {
    os.print("new com.caucho.el.MethodExpr(");
    _expr.printCreate(os);
    os.print(", \"");
    os.print(_methodName);
    os.print("\", new com.caucho.el.Expr[] {");
    
    for (int i = 0; i < _args.length; i++) {
      if (i != 0)
        os.print(", ");
      _args[i].printCreate(os);
    }
    os.println("})");
  }
  
  /**
   * Returns true for equal strings.
   */
  public boolean equals(Object o)
  {
    if (! (o instanceof MethodExpr))
      return false;
    
    MethodExpr expr = (MethodExpr) o;
    
    if (! _expr.equals(expr._expr))
      return false;
    
    if (! _methodName.equals(expr._methodName))
      return false;
    
    if (_args.length != expr._args.length)
      return false;
    
    for (int i = 0; i < _args.length; i++) {
      if (! _args[i].equals(expr._args[i]))
        return false;
    }
    
    return true;
  }
  
  /**
   * Returns the printed version.
   */
  public String toString()
  {
    return "MethodExpr[" + _expr + "," + _methodName + "]";
  }
  
  private static class MethodCall {
    private final Method _method;
    private final Marshall []_marshall;
    
    MethodCall(Method method)
    {
      _method = method;
      
      try {
        method.setAccessible(true);
      } catch (Throwable e) {
      }
      
      Class []paramTypes = method.getParameterTypes();
      
      _marshall = new Marshall[paramTypes.length];
      
      for (int i = 0; i < paramTypes.length; i++) {
        _marshall[i] = Marshall.create(paramTypes[i]);
      }
    }
    
    public Method getMethod()
    {
      return _method;
    }
    
    public Marshall []getMarshall()
    {
      return _marshall;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy