Package com.caucho.quercus.env

Source Code of com.caucho.quercus.env.CompiledObjectValue

/*
* Copyright (c) 1998-2008 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.quercus.env;

import com.caucho.quercus.expr.Expr;
import com.caucho.quercus.function.AbstractFunction;

import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.IdentityHashMap;
import java.util.Map;
import java.util.Set;

/**
* Represents a compiled object value.
*/
public class CompiledObjectValue extends ObjectValue
  implements Serializable
{
  /**
   *
   */
  private static final long serialVersionUID = 1L;
  private static final Value []NULL_FIELDS = new Value[0];

  public Value []_fields;

  private ObjectExtValue _object;

  public CompiledObjectValue(QuercusClass cl)
  {
    super(cl);

    int size = cl.getFieldSize();
    if (size != 0)
      _fields = new Value[cl.getFieldSize()];
    else
      _fields = NULL_FIELDS;
  }

  /**
   * Returns the number of entries.
   */
  @Override
  public int getSize()
  {
    int size = 0;

    for (int i = 0; i < _fields.length; i++) {
      if (_fields[i] != UnsetValue.UNSET)
        size++;
    }

    if (_object != null)
      size += _object.getSize();

    return size;
  }

  /**
   * Gets a new value.
   */
  @Override
  public Value getField(Env env, StringValue key)
  {
    if (_fields.length > 0) {
      int index = _quercusClass.findFieldIndex(key);

      if (index >= 0)
        return _fields[index].toValue();
    }
   
    if (_object != null) {
      return _object.getField(env, key);
    }
    else
      return UnsetValue.UNSET;
  }

  /**
   * Returns the array ref.
   */
  @Override
  public Var getFieldRef(Env env, StringValue key)
  {
    if (_fields.length > 0) {
      int index = _quercusClass.findFieldIndex(key);

      if (index >= 0) {
        Var var = _fields[index].toRefVar();
       
        _fields[index] = var;

        return var;
      }
    }

    if (_object == null)
      _object = new ObjectExtValue(_quercusClass);
   
    return _object.getFieldRef(env, key);
  }

  /**
   * Returns the value as an argument which may be a reference.
   */
  @Override
  public Value getFieldArg(Env env, StringValue key, boolean isTop)
  {
    if (_fields.length > 0) {
      int index = _quercusClass.findFieldIndex(key);

      if (index >= 0) {
        Var var = _fields[index].toRefVar();
       
        _fields[index] = var;

        return var;
      }
    }

    if (_object == null)
      _object = new ObjectExtValue(_quercusClass);
   
    return _object.getFieldArg(env, key, isTop);
  }

  /**
   * Returns the value as an argument which may be a reference.
   */
  @Override
  public Value getFieldArgRef(Env env, StringValue key)
  {
    if (_fields.length > 0) {
      int index = _quercusClass.findFieldIndex(key);

      if (index >= 0) {
        Var var = _fields[index].toRefVar();
       
        _fields[index] = var;

        return var;
      }
    }

    if (_object == null)
      _object = new ObjectExtValue(_quercusClass);
   
    return _object.getFieldArgRef(env, key);
  }

  /**
   * Returns field as an array.
   */
  @Override
  public Value getFieldArray(Env env, StringValue key)
  {
    if (_fields.length > 0) {
      int index = _quercusClass.findFieldIndex(key);

      if (index >= 0) {
        _fields[index] = _fields[index].toAutoArray();
       
        return _fields[index];
      }
    }

    if (_object == null)
      _object = new ObjectExtValue(_quercusClass);
   
    return _object.getFieldArray(env, key);
  }

  /**
   * Returns field as an object.
   */
  @Override
  public Value getFieldObject(Env env, StringValue key)
  {
    if (_fields.length > 0) {
      int index = _quercusClass.findFieldIndex(key);

      if (index >= 0) {
        _fields[index] = _fields[index].toAutoObject(env);
       
        return _fields[index];
      }
    }

    if (_object == null)
      _object = new ObjectExtValue(_quercusClass);
   
    return _object.getFieldObject(env, key);
  }

  /**
   * Adds a new value.
   */
  @Override
  public Value putField(Env env, StringValue key, Value value)
  {
    if (_fields.length > 0) {
      int index = _quercusClass.findFieldIndex(key);

      if (index >= 0) {
        _fields[index] = _fields[index].set(value);

        return value;
      }
    }
   
    if (_object == null)
      _object = new ObjectExtValue(_quercusClass);

    return _object.putField(env, key, value);
  }

  /**
   * Removes a value.
   */
  @Override
  public void unsetField(StringValue key)
  {
    if (_fields.length > 0) {
      int index = _quercusClass.findFieldIndex(key);

      if (index >= 0) {
        _fields[index] = UnsetValue.UNSET;

        return;
      }
    }
   
    if (_object != null)
      _object.unsetField(key);
  }

  /**
   * Finds the method name.
   */
  @Override
  public AbstractFunction findFunction(String methodName)
  {
    return _quercusClass.findFunction(methodName);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethod(Env env, String methodName, Expr []args)
  {
    return _quercusClass.getFunction(methodName).callMethod(env, this, args);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethod(Env env, String methodName, Value []args)
  {
    AbstractFunction fun = _quercusClass.findFunction(methodName);

    if (fun != null)
      return fun.callMethod(env, this, args);
    else
      return env.error(L.l("Call to undefined method {0}::{1}()",
                           _quercusClass.getName(), methodName));
  }

  /**
   * Evaluates a method.
   */
  public Value callMethod(Env env, String methodName)
  {
    return _quercusClass.getFunction(methodName).callMethod(env, this);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethod(Env env, String methodName, Value a0)
  {
    return _quercusClass.getFunction(methodName).callMethod(env, this, a0);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethod(Env env, String methodName,
                          Value a0, Value a1)
  {
    return _quercusClass.getFunction(methodName).callMethod(env, this, a0, a1);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethod(Env env, String methodName,
                          Value a0, Value a1, Value a2)
  {
    return _quercusClass.getFunction(methodName).callMethod(env, this,
                                                            a0, a1, a2);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethod(Env env, String methodName,
                          Value a0, Value a1, Value a2, Value a3)
  {
    return _quercusClass.getFunction(methodName).callMethod(env, this,
                                                            a0, a1, a2, a3);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethod(Env env, String methodName,
                          Value a0, Value a1, Value a2, Value a3, Value a4)
  {
    return _quercusClass.getFunction(methodName).callMethod(env, this,
                                                            a0, a1, a2, a3, a4);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethodRef(Env env, String methodName, Expr []args)
  {
    return _quercusClass.getFunction(methodName).callMethodRef(env, this, args);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethodRef(Env env, String methodName, Value []args)
  {
    return _quercusClass.getFunction(methodName).callMethodRef(env, this, args);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethodRef(Env env, String methodName)
  {
    return _quercusClass.getFunction(methodName).callMethodRef(env, this);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethodRef(Env env, String methodName, Value a0)
  {
    return _quercusClass.getFunction(methodName).callMethodRef(env, this, a0);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethodRef(Env env, String methodName,
                             Value a0, Value a1)
  {
    return _quercusClass.getFunction(methodName).callMethodRef(env, this, a0, a1);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethodRef(Env env, String methodName,
                             Value a0, Value a1, Value a2)
  {
    return _quercusClass.getFunction(methodName).callMethodRef(env, this,
                                                     a0, a1, a2);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethodRef(Env env, String methodName,
                             Value a0, Value a1, Value a2, Value a3)
  {
    return _quercusClass.getFunction(methodName).callMethodRef(env, this,
                                                     a0, a1, a2, a3);
  }

  /**
   * Evaluates a method.
   */
  public Value callMethodRef(Env env, String methodName,
                             Value a0, Value a1, Value a2, Value a3, Value a4)
  {
    return _quercusClass.getFunction(methodName).callMethodRef(env, this,
                                                     a0, a1, a2, a3, a4);
  }

  /**
   * Evaluates a method.
   */
  @Override
  public Value callClassMethod(Env env, AbstractFunction fun, Value []args)
  {
    Value oldThis = env.getThis();

    try {
      env.setThis(this);

      return fun.call(env, args);
    } finally {
      env.setThis(oldThis);
    }
  }

  /**
   * Returns the value for the variable, creating an object if the var
   * is unset.
   */
  @Override
  public Value getObject(Env env)
  {
    return this;
  }

  /**
   * Copy for assignment.
   */
  @Override
  public Value copy()
  {
    return this;
  }

  /**
   * Copy for serialization
   */
  @Override
  public Value copy(Env env, IdentityHashMap<Value,Value> map)
  {
    Value oldValue = map.get(this);

    if (oldValue != null)
      return oldValue;

    // XXX:
    // return new ObjectExtValue(env, map, _cl, getArray());

    return this;
  }

  /**
   * Clone the object
   */
  @Override
  public Value clone()
  {
    throw new UnsupportedOperationException();
  }

  // XXX: need to check the other copy, e.g. for sessions

  /**
   * Serializes the value.
   */
  @Override
  public void serialize(Env env, StringBuilder sb, SerializeMap map)
  {
    sb.append("O:");
    sb.append(_quercusClass.getName().length());
    sb.append(":\"");
    sb.append(_quercusClass.getName());
    sb.append("\":");
    sb.append(getSize());
    sb.append(":{");

    ArrayList<StringValue> names = _quercusClass.getFieldNames();
   
    if (names != null) {
      for (int i = 0; i < names.size(); i++) {
        StringValue key = names.get(i);
       
        if (_fields[i] == UnsetValue.UNSET)
          continue;
       
        sb.append("s:");
        sb.append(key.length());
        sb.append(":\"");
        sb.append(key);
        sb.append("\";");

        _fields[i].serialize(env, sb, map);
      }
    }

    if (_object != null) {
      for (Map.Entry<Value,Value> mapEntry : _object.sortedEntrySet()) {
        ObjectExtValue.Entry entry = (ObjectExtValue.Entry) mapEntry;

        StringValue key = entry.getKey().toStringValue(env);
       
        sb.append("s:");
        sb.append(key.length());
        sb.append(":\"");
        sb.append(key);
        sb.append("\";");

        entry.getValue().serialize(env, sb, map);
      }
    }

    sb.append("}");
  }

  /**
   * Converts to a string.
   * @param env
   */
  @Override
  public StringValue toReprString(Env env)
  {
    AbstractFunction fun = _quercusClass.findFunction("__toString");

    if (fun != null)
      return fun.callMethod(env, this, new Expr[0]).toReprString(env);
    else
      return env.createUnicodeBuilder().append(_quercusClass.getName()).append("[]");
  }

  /**
   * Converts to a string.
   * @param env
   */
  @Override
  public void print(Env env)
  {
    env.print(toReprString(env));
  }

  /**
   * Converts to an array.
   */
  @Override
  public Value toArray()
  {
    ArrayValue array = new ArrayValueImpl();

    for (Map.Entry<Value,Value> entry : entrySet()) {
      array.put(entry.getKey().toStringValue(), entry.getValue());
    }

    return array;
  }

  /**
   * Converts to an object.
   */
  @Override
  public Value toObject(Env env)
  {
    return this;
  }

  /**
   * Converts to an object.
   */
  @Override
  public Object toJavaObject()
  {
    return this;
  }

  @Override
  public Set<? extends Map.Entry<Value,Value>> entrySet()
  {
    throw new UnsupportedOperationException();
    // return new EntrySet();
  }

  /**
   * Returns a Set of entries, sorted by key.
   */
  public Set<? extends Map.Entry<Value,Value>> sortedEntrySet()
  {
    throw new UnsupportedOperationException();
    //return new TreeSet<Map.Entry<String, Value>>(entrySet());
  }

  @Override
  public String toString()
  {
    return "CompiledObjectValue@" + System.identityHashCode(this) "[" + _quercusClass.getName() + "]";
  }

  //
  // Java Serialization
  //

  private void writeObject(ObjectOutputStream out)
    throws IOException
  {
    out.writeObject(_fields);
    out.writeObject(_object);
    out.writeObject(_quercusClass.getName());
  }
 
  private void readObject(ObjectInputStream in)
    throws ClassNotFoundException, IOException
  {
    _fields = (Value []) in.readObject();
    _object = (ObjectExtValue) in.readObject();
   
    Env env = Env.getInstance();
    String name = (String) in.readObject();

    QuercusClass cl = env.findClass(name);

    if (cl != null) {
      setQuercusClass(cl);
    }
    else {
      cl = env.getQuercus().getStdClass();
     
      setQuercusClass(cl);
     
      setIncompleteObjectName(name);
    }
  }
}
TOP

Related Classes of com.caucho.quercus.env.CompiledObjectValue

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.