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

com.caucho.amber.field.OneToManyField 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.amber.field;

import com.caucho.amber.expr.AmberExpr;
import com.caucho.amber.expr.OneToManyExpr;
import com.caucho.amber.expr.PathExpr;
import com.caucho.amber.query.QueryParser;
import com.caucho.amber.table.LinkColumns;
import com.caucho.amber.table.AmberTable;
import com.caucho.amber.type.EntityType;
import com.caucho.amber.type.AmberType;
import com.caucho.bytecode.JType;
import com.caucho.bytecode.JTypeWrapper;
import com.caucho.config.ConfigException;
import com.caucho.java.JavaWriter;
import com.caucho.util.CharBuffer;
import com.caucho.util.L10N;

import javax.persistence.CascadeType;
import java.io.IOException;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.Map;
import java.util.Set;
import java.util.logging.Logger;

/**
 * Represents a field to a collection of objects where the target
 * hold a back-link to the source entity.
 */
public class OneToManyField extends CollectionField {
  private static final L10N L = new L10N(OneToManyField.class);
  private static final Logger log 
    = Logger.getLogger(OneToManyField.class.getName());

  private String _mapKey;

  private ArrayList _orderByFields;
  private ArrayList _orderByAscending;

  private ManyToOneField _sourceField;

  public OneToManyField(EntityType entityType,
                              String name,
                              CascadeType[] cascadeTypes)
    throws ConfigException
  {
    super(entityType, name, cascadeTypes);
  }

  public OneToManyField(EntityType entityType,
                              String name)
    throws ConfigException
  {
    this(entityType, name, null);
  }

  public OneToManyField(EntityType entityType)
  {
    super(entityType);
  }

  /**
   * Sets the order by.
   */
  public void setOrderBy(ArrayList orderByFields,
                         ArrayList orderByAscending)
  {
    _orderByFields = orderByFields;
    _orderByAscending = orderByAscending;
  }

  /**
   * Returns the source type as
   * entity or mapped-superclass.
   */
  @Override
  public EntityType getEntitySourceType()
  {
    return (EntityType) getSourceType();
  }

  /**
   * Returns the target type as
   * entity or mapped-superclass.
   */
  public EntityType getEntityTargetType()
  {
    return (EntityType) getTargetType();
  }

  /**
   * Returns the target type as entity.
   */
  @Override
  public AmberType getTargetType()
  {
    return _sourceField.getSourceType();
  }

  /**
   * Gets the source field.
   */
  public ManyToOneField getSourceField()
  {
    return _sourceField;
  }

  /**
   * Sets the source field.
   */
  public void setSourceField(ManyToOneField sourceField)
  {
    _sourceField = sourceField;
  }

  /**
   * Returns the link.
   */
  @Override
  public LinkColumns getLinkColumns()
  {
    return _sourceField.getLinkColumns();
  }

  /**
   * Gets the map key.
   */
  public String getMapKey()
  {
    return _mapKey;
  }

  /**
   * Sets the map key.
   */
  public void setMapKey(String mapKey)
  {
    _mapKey = mapKey;
  }

  /**
   * Initialize.
   */
  @Override
  public void init()
  {
    // jpa/0gg2
    if (_sourceField == null) // || getLinkColumns() == null)
      throw new IllegalStateException();
  }

  /**
   * Creates the expression for the field.
   */
  @Override
  public AmberExpr createExpr(QueryParser parser, PathExpr parent)
  {
    return new OneToManyExpr(parser, parent, getLinkColumns());
  }
  /**
   * Generates the (pre) cascade operation from
   * parent to this child. This field will only
   * be cascaded first if the operation can be
   * performed with no risk to break FK constraints.
   */
  @Override
  public void generatePreCascade(JavaWriter out,
                                 String aConn,
                                 CascadeType cascadeType)
    throws IOException
  {
    if (cascadeType == CascadeType.PERSIST)
      return;

    generateInternalCascade(out, aConn, cascadeType);
  }

  /**
   * Generates the (post) cascade operation from
   * parent to this child. This field will only
   * be cascaded first if the operation can be
   * performed with no risk to break FK constraints.
   */
  @Override
  public void generatePostCascade(JavaWriter out,
                                  String aConn,
                                  CascadeType cascadeType)
    throws IOException
  {
    if (cascadeType != CascadeType.PERSIST)
      return;

    generateInternalCascade(out, aConn, cascadeType);
  }

  @Override
  protected void generateInternalCascade(JavaWriter out,
                                       String aConn,
                                       CascadeType cascadeType)
    throws IOException
  {
    if (isCascade(cascadeType)) {

      String getter = "_caucho_field_" + getGetterName(); // generateSuperGetterMethod();

      out.println("if (" + getter + " == null && " + generateSuperGetter("this") + " != null)");
      out.pushDepth();
      out.println(getSetterName() + "(" + generateSuperGetter("this") + ");");
      out.popDepth();

      out.println();
      out.println("if (" + getter + " != null) {");
      out.pushDepth();

      out.print("for (Object o : " + getter);

      // jpa/0v04
      if (Map.class.isAssignableFrom(getJavaClass()))
        out.print(".values()");

      out.println(") {");
      out.pushDepth();

      // XXX
      out.println("if (o == null)");
      out.println("  continue;");

      if (_sourceField != null) {
        String typeName = getEntityTargetType().getJavaTypeName();
        String setter = _sourceField.getSetterName();
        out.println("((" + typeName + ") o)." + setter + "(this);");
      }

      out.print(aConn + ".");

      switch (cascadeType) {
      case PERSIST:
        out.print("persistFromCascade");
        break;

      case MERGE:
        out.print("merge");
        break;

      case REMOVE:
        out.print("remove");
        break;

      case REFRESH:
        out.print("refresh");
        break;
      }

      out.println("(o);");

      out.popDepth();
      out.println("}");

      out.popDepth();
      out.println("}");
    }
  }

  /**
   * Generates the set clause.
   */
  @Override
  public void generateStatementSet(JavaWriter out, String pstmt,
                          String obj, String index)
    throws IOException
  {
  }

  /**
   * Generates the select clause.
   */
  @Override
  public String generateLoadSelect(String id)
  {
    return null;
  }

  /**
   * Generates loading code after the basic fields.
   */
  @Override
  public int generatePostLoadSelect(JavaWriter out, int index)
    throws IOException
  {
    if (! isLazy()) {
      out.println(getGetterName() + "();");
      
      return ++index;
    }
    else
      return index;
  }

  /**
   * Updates from the cached copy.
   */
  @Override
  public void generateCopyLoadObject(JavaWriter out,
                                     String dst, String src,
                                     int loadIndex)
    throws IOException
  {
  }

  /**
   * Generates the target select.
   */
  @Override
  public String generateTargetSelect(String id)
  {
    CharBuffer cb = CharBuffer.allocate();

    Id key = getEntityTargetType().getId();

    cb.append(key.generateSelect(id));

    String value = getEntityTargetType().generateLoadSelect(id);

    if (cb.length() > 0 && value.length() > 0)
      cb.append(", ");

    cb.append(value);

    return cb.close();
  }

  /**
   * Generates the set property.
   */
  @Override
  public void generateGetterMethod(JavaWriter out)
    throws IOException
  {
    String var = "_caucho_field_" + getGetterName();

    boolean isSet = Set.class.isAssignableFrom(getJavaClass());
    boolean isMap = false;
    if (!isSet) {
      isMap = Map.class.isAssignableFrom(getJavaClass());
    }

    JType type = getJavaType();
    JType []paramArgs = type.getActualTypeArguments();
    JType param = paramArgs.length > 0 ? paramArgs[0] : null;
    JType param2 = paramArgs.length > 1 ? paramArgs[1] : null;

    out.print("protected transient ");

    String collectionImpl;

    if (isSet)
      collectionImpl = "com.caucho.amber.collection.SetImpl";
    else if (isMap)
      collectionImpl = "com.caucho.amber.collection.MapImpl";
    else
      collectionImpl = "com.caucho.amber.collection.CollectionImpl";

    out.print(collectionImpl);

    if (param != null) {
      out.print("<");
      out.print(param.getPrintName());
      if (isMap) {
        if (param2 != null) {
          out.print(", ");
          out.print(param2.getPrintName());
        }
      }
      out.print(">");
    }

    out.println(" " + var + ";");

    out.println();
    out.println("public " + getJavaTypeName() + " " + getGetterName() + "()");
    out.println("{");
    out.pushDepth();

    out.println("if (" + var + " != null) {");
    out.pushDepth();

    out.println("if (__caucho_state.isPersist()) {");
    out.pushDepth();

    out.println(var + ".setSession(__caucho_session);");
    out.println("return " + var + ";");

    out.popDepth();
    out.println("}");
    out.println();

    // jpa/1621
    out.println("if (" + var + ".getSession() != null");
    out.println("    && " + var + ".getSession() == __caucho_session)");
    out.println("  return " + var + ";");

    out.popDepth();
    out.println("}");

    out.println();
    out.println("com.caucho.amber.AmberQuery query = null;");
    
    String newEmptyCollection = "new " + collectionImpl;

    if (param != null) {
      newEmptyCollection += "<" + param.getPrintName();
      if (isMap) {
        newEmptyCollection += ", ";
        newEmptyCollection += param2.getPrintName();
      }
      newEmptyCollection += ">";
    }

    newEmptyCollection += "(query";
    if (isMap) {
      // jpa/0v00
      newEmptyCollection += "," + getEntityTargetType().getBeanClass().getName();
      newEmptyCollection += ".class.getDeclaredMethod(\"";

      String getterMapKey = getMapKey();

      // jpa/0j63
      if (getterMapKey == null) {
        getterMapKey = getEntityTargetType().getId().generateGet("this");
      }
      else {
        getterMapKey = "get" + Character.toUpperCase(getterMapKey.charAt(0))
          + getterMapKey.substring(1);
      }

      newEmptyCollection += getterMapKey; // "getId");
      newEmptyCollection += "\", (Class []) null)";
    }
    newEmptyCollection += ")";

    out.println();
    out.println("try {");
    out.pushDepth();

    out.println("if (__caucho_session == null) {");
    out.pushDepth();

    /*
    out.println("if (" + var + " == null)");
    out.println("  " + var + " = " + newEmptyCollection + ";");

    // if (! isAbstract())
    out.println();
    out.println("return " + var + ";");
    */
    
    out.println("return " + generateSuperGetter("this") + ";");

    out.popDepth();
    out.println("}");

    out.println();
    out.print("String sql=\"");

    out.print("SELECT c");
    out.print(" FROM " + getEntitySourceType().getName() + " o,");
    out.print("      o." + getName() + " c");
    out.print(" WHERE ");
    out.print(getEntitySourceType().getId().generateRawWhere("o"));

    if (_orderByFields != null) {
      out.print(" ORDER BY ");

      for (int i = 0; i < _orderByFields.size(); i++) {
        if (i != 0)
          out.print(", ");

        out.print("c." + _orderByFields.get(i));
        if (Boolean.FALSE.equals(_orderByAscending.get(i)))
          out.print(" DESC");
      }
    }

    out.println("\";");
    out.println("query = __caucho_session.prepareQuery(sql);");

    out.println("int index = 1;");
    getEntitySourceType().getId().generateSet(out, "query", "index", "this");

    // Ex: _caucho_getChildren = new com.caucho.amber.collection.CollectionImpl
    out.print(var);
    out.print(" = " + newEmptyCollection + ";");

    /*
      out.pushDepth();

      //generateAdd(out);
      //generateRemove(out);
      //generateClear(out);
      // generateSize(out);

      out.popDepth();
      out.println("};");
    */

    // ejb/0aj2
    if (getEntitySourceType().getPersistenceUnit().isJPA()) {
      // jpa/0l43
      out.println();
      out.print("for (Object o : " + var);

      // jpa/0v04
      if (getJavaType().isAssignableTo(Map.class))
        out.print(".values()");

      out.println(")");
      //out.println("  __caucho_session.makeTransactional((com.caucho.amber.entity.Entity) o);");
    }

    // jpa/0j70
    out.println(generateSuperSetter("this", var) + ";");

    out.println();
    out.println("return " + var + ";");

    out.popDepth();
    out.println("} catch (Exception e) {");
    out.println("  throw com.caucho.amber.AmberRuntimeException.create(e);");
    out.println("}");

    out.popDepth();
    out.println("}");
  }

  /**
   * Generates the size method.
   */
  private void generateSize(JavaWriter out)
    throws IOException
  {
    out.println("public int size()");
    out.println("{");
    out.pushDepth();

    out.println("if (__caucho_session == null || isValid())");
    out.println("  return super.size();");

    out.println("try {");
    out.pushDepth();

    out.println("__caucho_session.flushNoChecks();");

    out.print("String sql=\"");

    out.print("SELECT count(*) FROM ");
    out.print(getEntitySourceType().getName());
    out.print(" AS o ");

    out.print(" WHERE ");

    // getKeyColumn().generateRawMatchArgWhere("o");

    ArrayList keys = getEntitySourceType().getId().getKeys();
    for (int i = 0; i < keys.size(); i++) {
      if (i != 0)
        out.print(" AND ");

      out.print("o." + keys.get(i).getName());
      out.print("=?");
    }

    out.println("\";");
    out.println("com.caucho.amber.AmberQuery query;");
    out.println("query = __caucho_session.prepareQuery(sql);");

    out.println("int index = 1;");

    // ejb/06h0
    getEntitySourceType().getId().generateSet(out, "query", "index", getEntitySourceType().getInstanceClassName() + ".this"); // "__ResinExt.this");

    out.println("java.sql.ResultSet rs = query.executeQuery();");

    out.println("if (rs.next())");
    out.println("  return rs.getInt(1);");
    out.println("else");
    out.println("  return 0;");

    out.popDepth();
    out.println("} catch (java.sql.SQLException e) {");
    out.println("  throw com.caucho.amber.AmberRuntimeException.create(e);");
    out.println("}");

    out.popDepth();
    out.println("}");
  }

  /**
   * Generates the set property.
   */
  @Override
  public void generateSetterMethod(JavaWriter out)
    throws IOException
  {
    // commented out: jpa/0s2d
    // JMethod setter = getSetterMethod();
    //
    // if (setter == null)
    //   return;
    //
    // JClass []paramTypes = setter.getParameterTypes();

    JType type;

    ClassLoader loader = getSourceType().getPersistenceUnit().getTempClassLoader();
    
    if (! getEntitySourceType().isFieldAccess()) {
      type = JTypeWrapper.create(getGetterMethod().getGenericReturnType(),
                                 loader);
    }
    else {
      Field field = EntityType.getField(getBeanClass(), getName());
      type = JTypeWrapper.create(field.getGenericType(),
                                 loader);
    }

    out.println();
    // commented out: jpa/0s2d
    // out.print("public void " + setter.getName() + "(");
    // out.print(getJavaTypeName() + " value)");
    out.print("public void " + getSetterName() + "(");
    out.print(type.getName() + " value)");
    out.println("{");
    out.pushDepth();

    // out.println("if (" + generateSuperGetterMethod() + " == value)");
    // out.println("  return;");
    // out.println();

    //
    // jpa/0j57 needs to generate the following snippet:
    //
    // _caucho___caucho_get_xAnnualReviews
    //   = new com.caucho.amber.collection.CollectionImpl(__caucho_session, null);
    // _caucho___caucho_get_xAnnualReviews.addAll(0, value);
    //
    // jpa/0j57:

    out.println("if (__caucho_session == null) {");
    out.pushDepth();
    out.println(generateSuperSetter("this", "value") + ";");
    out.popDepth();
    out.println("} else {");
    out.pushDepth();
    
    out.println("try {");
    out.pushDepth();

    String var = "_caucho_field_" + getGetterName();

    out.print(var + " = new ");

    type = getJavaType();

    boolean isSet = type.isAssignableTo(Set.class);
    boolean isMap = false;
    if (!isSet) {
      isMap = type.isAssignableTo(Map.class);
    }

    JType []paramArgs = type.getActualTypeArguments();
    JType param = paramArgs.length > 0 ? paramArgs[0] : null;
    JType param2 = paramArgs.length > 1 ? paramArgs[1] : null;

    String collectionImpl;

    if (isSet)
      collectionImpl = "com.caucho.amber.collection.SetImpl";
    else if (isMap)
      collectionImpl = "com.caucho.amber.collection.MapImpl";
    else
      collectionImpl = "com.caucho.amber.collection.CollectionImpl";

    out.print(collectionImpl);

    if (param != null) {
      out.print("<");
      out.print(param.getPrintName());
      if (isMap) {
        if (param2 != null) {
          out.print(", ");
          out.print(param2.getPrintName());
        }
      }
      out.print(">");
    }

    out.print("(__caucho_session, null");
    if (isMap) {
      out.print(", ");
      out.print(getEntityTargetType().getBeanClass().getName());
      out.print(".class.getDeclaredMethod(\"");

      String getterMapKey = getMapKey();

      // jpa/0j63
      if (getterMapKey == null) {
        getterMapKey = getEntityTargetType().getId().getKey().getGetterName();
      }
      else {
        getterMapKey = "get" + Character.toUpperCase(getterMapKey.charAt(0))
          + getterMapKey.substring(1);
      }

      out.print(getterMapKey); // "getId");
      out.print("\")");
    }
    out.println(");");

    out.print(var + ".");

    if (isMap)
      out.println("putAll(value);");
    else if (isSet)
      out.println("addAll(value);");
    else
      out.println("addAll(0, value);");

    out.popDepth();
    out.println("} catch(Exception e) {");
    out.println("  throw com.caucho.amber.AmberRuntimeException.create(e);");
    out.println("}");

    out.popDepth();
    out.println("}");

    out.popDepth();
    out.println("}");
  }

  /**
   * Generates code for foreign entity create/delete
   */
  @Override
  public void generateInvalidateForeign(JavaWriter out)
    throws IOException
  {
    // XXX: jpa/0gg2
    if (getEntitySourceType().getPersistenceUnit().isJPA())
      return;

    AmberTable table = getLinkColumns().getSourceTable();

    out.println("if (\"" + table.getName() + "\".equals(table)) {");
    out.pushDepth();

    String var = "_caucho_field_" + getGetterName();

    out.println("if (" + var + " != null)");
    out.println("  " + var + ".update();");
    out.popDepth();
    out.println("}");
  }

  /**
   * Generates the expire code
   *
   * ejb/06hi
   */
  @Override
  public void generateExpire(JavaWriter out)
    throws IOException
  {
    String var = "_caucho_field_" + getGetterName();

    out.println(var + " = null;");
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy