/* * Copyright (c) 1998-2011 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 javax.el.MethodInfo; import javax.el.PropertyNotFoundException; import javax.el.ValueReference; import java.io.IOException; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; /** * Represents an array reference: * * <pre> * a[b] * </pre> */ public class ArrayResolverExpr extends Expr { private Expr _left; private Expr _right; // cached getter method private transient Class _lastClass; private transient String _lastField; private transient Method _lastMethod; /** * Creates a new array expression. * * @param left the object expression * @param right the index expression. */ public ArrayResolverExpr(Expr left, Expr right) { _left = left; _right = right; } /** * Returns the base expression. */ public Expr getExpr() { return _left; } /** * Returns the index expression. */ public Expr getIndex() { return _right; } /** * Creates a method for constant arrays. */ @Override public Expr createMethod(Expr []args) { if (! (_right instanceof StringLiteral)) return null; StringLiteral literal = (StringLiteral) _right; return new MethodExpr(_left, literal.getValue(), args); } /** * Evaluates the expression as applicable to the provided context, and returns * the most general type that can be accepted by the setValue(javax.el.ELContext, * java.lang.Object) method. * * @param env * @return * @throws PropertyNotFoundException * @throws ELException */ @Override public Class<?> getType(ELContext env) throws PropertyNotFoundException, ELException { Object aObj = _left.getValue(env); if (aObj == null) return null; Object fieldObj = _right.getValue(env); if (fieldObj == null) return null; return env.getELResolver().getType(env, aObj, fieldObj); } /** * Evaluate the expression as an object. * * @param env the variable environment * * @return the evaluated object */ @Override public Object getValue(ELContext env) throws ELException { Object aObj = _left.getValue(env); if (aObj == null) return null; Object fieldObj = _right.getValue(env); if (fieldObj == null) return null; return env.getELResolver().getValue(env, aObj, fieldObj); } /** * Returns the read-only value of the expression. * * @param env the variable environment * * @return true if read-only */ @Override public boolean isReadOnly(ELContext env) throws ELException { Object aObj = _left.getValue(env); if (aObj == null) return true; Object fieldObj = _right.getValue(env); if (fieldObj == null) return true; return env.getELResolver().isReadOnly(env, aObj, fieldObj); } /** * Evaluate the expression as an object. * * @param env the variable environment * * @return the evaluated object */ @Override public void setValue(ELContext env, Object value) throws ELException { Object aObj = _left.getValue(env); if (aObj == null) throw new PropertyNotFoundException(L.l("'{0}' is null in '{1}'", _left.toString(), toString())); Object fieldObj = _right.getValue(env); if (fieldObj == null) throw new PropertyNotFoundException(L.l("'{0}' is null in '{1}'", _right.toString(), toString())); env.getELResolver().setValue(env, aObj, fieldObj, value); } @Override public ValueReference getValueReference(ELContext context) { Object base = _left.getValue(context); Object property = _right.getValue(context); return new ValueReference(base, property); } /** * Evaluates the expression, returning an object. * * @param env the variable environment * * @return the value of the expression as an object */ @Override public MethodInfo getMethodInfo(ELContext env, Class<?> returnType, Class<?> []argTypes) throws ELException { Object base = _left.getValue(env); if (base == null) throw new PropertyNotFoundException(L.l( "'{0}' not found in context '{1}'.", _left.getExpressionString(), env)); String name = _right.evalString(env); try { Method method = base.getClass().getMethod(name, argTypes); return new MethodInfo(_right.evalString(env), method.getReturnType(), argTypes); } catch (NoSuchMethodException e) { throw new javax.el.MethodNotFoundException(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 { Object base = _left.getValue(env); if (base == null) throw new PropertyNotFoundException(L.l( "'{0}' not found in context '{1}'.", _left.getExpressionString(), env)); String name = _right.evalString(env); try { Method method = base.getClass().getMethod(name, argTypes); return method.invoke(base, args); } catch (NoSuchMethodException e) { throw new javax.el.MethodNotFoundException(e); } catch (IllegalAccessException e) { throw new ELException(e); } catch (InvocationTargetException e) { throw new ELException(e.getCause()); } } /** * Prints the code to create an LongLiteral. * * @param os stream to the generated *.java code */ @Override public void printCreate(WriteStream os) throws IOException { os.print("new com.caucho.el.ArrayResolverExpr("); _left.printCreate(os); os.print(", "); _right.printCreate(os); os.print(")"); } /** * Returns true for equal strings. */ public boolean equals(Object o) { if (! (o instanceof ArrayResolverExpr)) return false; ArrayResolverExpr expr = (ArrayResolverExpr) o; return (_left.equals(expr._left) && _right.equals(expr._right)); } /** * Returns a readable representation of the expr. */ public String toString() { return _left + "[" + _right + "]"; } }