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

net.hasor.db.ognl.ASTCtor Maven / Gradle / Ivy

The newest version!
// --------------------------------------------------------------------------
// Copyright (c) 1998-2004, Drew Davidson and Luke Blanshard
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
// Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
// Neither the name of the Drew Davidson nor the names of its contributors
// may be used to endorse or promote products derived from this software
// without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
// FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
// COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
// INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
// OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
// AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
// DAMAGE.
// --------------------------------------------------------------------------
package ognl;

import ognl.enhance.ExpressionCompiler;

import java.lang.reflect.Array;
import java.lang.reflect.Constructor;
import java.util.List;

/**
 * @author Luke Blanshard ([email protected])
 * @author Drew Davidson ([email protected])
 */
public class ASTCtor extends SimpleNode
{

    private String className;
    private boolean isArray;

    public ASTCtor(int id)
    {
        super(id);
    }

    public ASTCtor(OgnlParser p, int id)
    {
        super(p, id);
    }

    /** Called from parser action. */
    void setClassName(String className)
    {
        this.className = className;
    }

    Class getCreatedClass(OgnlContext context) throws ClassNotFoundException {
        return OgnlRuntime.classForName(context, className);
    }


    void setArray(boolean value)
    {
        isArray = value;
    }

    public boolean isArray()
    {
        return isArray;
    }

    protected Object getValueBody(OgnlContext context, Object source)
            throws OgnlException
    {
        Object result, root = context.getRoot();
        int count = jjtGetNumChildren();
        Object[] args = OgnlRuntime.getObjectArrayPool().create(count);

        try {
            for(int i = 0; i < count; ++i) {
                args[i] = _children[i].getValue(context, root);
            }
            if (isArray) {
                if (args.length == 1) {
                    try {
                        Class componentClass = OgnlRuntime.classForName(context, className);
                        List sourceList = null;
                        int size;

                        if (args[0] instanceof List) {
                            sourceList = (List) args[0];
                            size = sourceList.size();
                        } else {
                            size = (int) OgnlOps.longValue(args[0]);
                        }
                        result = Array.newInstance(componentClass, size);
                        if (sourceList != null) {
                            TypeConverter converter = context.getTypeConverter();

                            for(int i = 0, icount = sourceList.size(); i < icount; i++) {
                                Object o = sourceList.get(i);

                                if ((o == null) || componentClass.isInstance(o)) {
                                    Array.set(result, i, o);
                                } else {
                                    Array.set(result, i, converter.convertValue(context, null, null, null, o,
                                            componentClass));
                                }
                            }
                        }
                    } catch (ClassNotFoundException ex) {
                        throw new OgnlException("array component class '" + className + "' not found", ex);
                    }
                } else {
                    throw new OgnlException("only expect array size or fixed initializer list");
                }
            } else {
                result = OgnlRuntime.callConstructor(context, className, args);
            }

            return result;
        } finally {
            OgnlRuntime.getObjectArrayPool().recycle(args);
        }
    }

    public String toString()
    {
        String result = "new " + className;

        if (isArray) {
            if (_children[0] instanceof ASTConst) {
                result = result + "[" + _children[0] + "]";
            } else {
                result = result + "[] " + _children[0];
            }
        } else {
            result = result + "(";
            if ((_children != null) && (_children.length > 0)) {
                for(int i = 0; i < _children.length; i++) {
                    if (i > 0) {
                        result = result + ", ";
                    }
                    result = result + _children[i];
                }
            }
            result = result + ")";
        }
        return result;
    }

    public String toGetSourceString(OgnlContext context, Object target)
    {
        String result = "new " + className;

        Class clazz = null;
        Object ctorValue = null;
        try {

            clazz = OgnlRuntime.classForName(context, className);
            
            ctorValue = this.getValueBody(context, target);
            context.setCurrentObject(ctorValue);
            
            if (clazz != null && ctorValue != null) {
                
                context.setCurrentType(ctorValue.getClass());
                context.setCurrentAccessor(ctorValue.getClass());
            }

            if (isArray)
                context.put("_ctorClass", clazz);

        } catch (Throwable t)
        {
            throw OgnlOps.castToRuntime(t);
        }

        try {

            if (isArray) {
                if (_children[0] instanceof ASTConst) {
                    
                    result = result + "[" + _children[0].toGetSourceString(context, target) + "]";
                } else if (ASTProperty.class.isInstance(_children[0])) {

                    result = result + "["
                            + ExpressionCompiler.getRootExpression(_children[0], target, context)
                            + _children[0].toGetSourceString(context, target)
                            + "]";
                } else if (ASTChain.class.isInstance(_children[0])) {

                    result = result + "[" + _children[0].toGetSourceString(context, target) + "]";
                } else {

                    result = result + "[] "+ _children[0].toGetSourceString(context, target);
                }

            } else {
                result = result + "(";

                if ((_children != null) && (_children.length > 0)) {

                    Object[] values = new Object[_children.length];
                    String[] expressions = new String[_children.length];
                    Class[] types = new Class[_children.length];

                    // first populate arrays with child values
                    
                    for(int i = 0; i < _children.length; i++) {
                        
                        Object objValue = _children[i].getValue(context, context.getRoot());
                        String value = _children[i].toGetSourceString(context, target);

                        if (!ASTRootVarRef.class.isInstance(_children[i]))
                        {
                            value = ExpressionCompiler.getRootExpression(_children[i], target, context) + value;
                        }

                        String cast = "";
                        if (ExpressionCompiler.shouldCast(_children[i])) {

                            cast = (String)context.remove(ExpressionCompiler.PRE_CAST);
                        }
                        if (cast == null)
                            cast = "";

                        if (!ASTConst.class.isInstance(_children[i]))
                            value = cast + value;

                        values[i] = objValue;
                        expressions[i] = value;
                        types[i] = context.getCurrentType();
                    }

                    // now try and find a matching constructor
                    
                    Constructor[] cons = clazz.getConstructors();
                    Constructor ctor = null;
                    Class[] ctorParamTypes = null;
                    
                    for (int i=0; i < cons.length; i++)
                    {
                        Class[] ctorTypes = cons[i].getParameterTypes();

                        if (OgnlRuntime.areArgsCompatible(values, ctorTypes)
                            && (ctor == null || OgnlRuntime.isMoreSpecific(ctorTypes, ctorParamTypes))) {
                            ctor = cons[i];
                            ctorParamTypes = ctorTypes;
                        }
                    }

                    if (ctor == null)
                        ctor = OgnlRuntime.getConvertedConstructorAndArgs(context, clazz, OgnlRuntime.getConstructors(clazz), values, new Object[values.length]);

                    if (ctor == null)
                        throw new NoSuchMethodException("Unable to find constructor appropriate for arguments in class: " + clazz);

                    ctorParamTypes = ctor.getParameterTypes();

                    // now loop over child values again and build up the actual source string

                    for(int i = 0; i < _children.length; i++) {
                        if (i > 0) {
                            result = result + ", ";
                        }

                        String value = expressions[i];

                        if (types[i].isPrimitive()) {

                            String literal = OgnlRuntime.getNumericLiteral(types[i]);
                            if (literal != null)
                                value += literal;
                        }

                        if (ctorParamTypes[i] != types[i]) {

                            if (values[i] != null && !types[i].isPrimitive()
                                && !values[i].getClass().isArray() && !ASTConst.class.isInstance(_children[i])) {
                                
                                value = "(" + OgnlRuntime.getCompiler().getInterfaceClass(values[i].getClass()).getName() + ")" + value;
                            } else if (!ASTConst.class.isInstance(_children[i])
                                       || (ASTConst.class.isInstance(_children[i]) && !types[i].isPrimitive())) {
                                
                                if (!types[i].isArray()
                                    && types[i].isPrimitive() && !ctorParamTypes[i].isPrimitive())
                                    value = "new " + ExpressionCompiler.getCastString(OgnlRuntime.getPrimitiveWrapperClass(types[i])) + "(" + value + ")";
                                else
                                    value = " ($w) " + value;
                            }
                        }

                        result += value;
                    }

                }
                result = result + ")";
            }

            context.setCurrentType(ctorValue != null ? ctorValue.getClass() : clazz);
            context.setCurrentAccessor(clazz);
            context.setCurrentObject(ctorValue);

        } catch (Throwable t)
        {
            throw OgnlOps.castToRuntime(t);
        }

        context.remove("_ctorClass");

        return result;
    }

    public String toSetSourceString(OgnlContext context, Object target)
    {
        return "";
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy