Open Source Repository

Home /hibernate/hibernate-3.2.7.ga | Repository Home



org/hibernate/engine/query/QueryPlanCache.java
package org.hibernate.engine.query;

import org.hibernate.util.ArrayHelper;
import org.hibernate.util.SimpleMRUCache;
import org.hibernate.util.SoftLimitMRUCache;
import org.hibernate.engine.SessionFactoryImplementor;
import org.hibernate.engine.query.sql.NativeSQLQuerySpecification;
import org.hibernate.QueryException;
import org.hibernate.MappingException;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import java.io.Serializable;
import java.util.Map;
import java.util.HashMap;
import java.util.List;
import java.util.Iterator;
import java.util.Set;
import java.util.HashSet;
import java.util.Collections;

/**
 * Acts as a cache for compiled query plans, as well as query-parameter metadata.
 *
 @author <a href="mailto:[email protected]">Steve Ebersole </a>
 */
public class QueryPlanCache implements Serializable {

  private static final Log log = LogFactory.getLogQueryPlanCache.class );

  private SessionFactoryImplementor factory;

  public QueryPlanCache(SessionFactoryImplementor factory) {
    this.factory = factory;
  }

  // simple cache of param metadata based on query string.  Ideally, the
  // original "user-supplied query" string should be used to retreive this
  // metadata (i.e., not the para-list-expanded query string) to avoid
  // unnecessary cache entries.
  // Used solely for caching param metadata for native-sql queries, see
  // getSQLParameterMetadata() for a discussion as to why...
  private final SimpleMRUCache sqlParamMetadataCache = new SimpleMRUCache();

  // the cache of the actual plans...
  private final SoftLimitMRUCache planCache = new SoftLimitMRUCache128 );


  public ParameterMetadata getSQLParameterMetadata(String query) {
    ParameterMetadata metadata = ParameterMetadata sqlParamMetadataCache.getquery );
    if metadata == null ) {
      // for native-sql queries, the param metadata is determined outside
      // any relation to a query plan, because query plan creation and/or
      // retreival for a native-sql query depends on all of the return
      // types having been set, which might not be the case up-front when
      // param metadata would be most useful
      metadata = buildNativeSQLParameterMetadataquery );
      sqlParamMetadataCache.putquery, metadata );
    }
    return metadata;
  }

  public HQLQueryPlan getHQLQueryPlan(String queryString, boolean shallow, Map enabledFilters)
      throws QueryException, MappingException {
    HQLQueryPlanKey key = new HQLQueryPlanKeyqueryString, shallow, enabledFilters );
    HQLQueryPlan plan = HQLQueryPlan planCache.get key );

    if plan == null ) {
      if log.isTraceEnabled() ) {
        log.trace"unable to locate HQL query plan in cache; generating (" + queryString + ")" );
      }
      plan = new HQLQueryPlan(queryString, shallow, enabledFilters, factory );
    }
    else {
      if log.isTraceEnabled() ) {
        log.trace"located HQL query plan in cache (" + queryString + ")" );
      }
    }

    planCache.putkey, plan );

    return plan;
  }

  public FilterQueryPlan getFilterQueryPlan(String filterString, String collectionRole, boolean shallow, Map enabledFilters)
      throws QueryException, MappingException {
    FilterQueryPlanKey key = new FilterQueryPlanKeyfilterString, collectionRole, shallow, enabledFilters );
    FilterQueryPlan plan = FilterQueryPlan planCache.get key );

    if plan == null ) {
      if log.isTraceEnabled() ) {
        log.trace"unable to locate collection-filter query plan in cache; generating (" + collectionRole + " : " + filterString + ")" );
      }
      plan = new FilterQueryPlanfilterString, collectionRole, shallow, enabledFilters, factory );
    }
    else {
      if log.isTraceEnabled() ) {
        log.trace"located collection-filter query plan in cache (" + collectionRole + " : " + filterString + ")" );
      }
    }

    planCache.putkey, plan );

    return plan;
  }

  public NativeSQLQueryPlan getNativeSQLQueryPlan(NativeSQLQuerySpecification spec) {
    NativeSQLQueryPlan plan = NativeSQLQueryPlan planCache.getspec );

    if plan == null ) {
      if log.isTraceEnabled() ) {
        log.trace"unable to locate native-sql query plan in cache; generating (" + spec.getQueryString() ")" );
      }
      plan = new NativeSQLQueryPlanspec, factory );
    }
    else {
      if log.isTraceEnabled() ) {
        log.trace"located native-sql query plan in cache (" + spec.getQueryString() ")" );
      }
    }

    planCache.putspec, plan );
    return plan;
  }

  private ParameterMetadata buildNativeSQLParameterMetadata(String sqlString) {
    ParamLocationRecognizer recognizer = ParamLocationRecognizer.parseLocationssqlString );

    OrdinalParameterDescriptor[] ordinalDescriptors =
        new OrdinalParameterDescriptorrecognizer.getOrdinalParameterLocationList().size() ];
    for int i = 0; i < recognizer.getOrdinalParameterLocationList().size(); i++ ) {
      final Integer position = Integer recognizer.getOrdinalParameterLocationList().get);
      ordinalDescriptors[inew OrdinalParameterDescriptori, null, position.intValue() );
    }

    Iterator itr = recognizer.getNamedParameterDescriptionMap().entrySet().iterator();
    Map namedParamDescriptorMap = new HashMap();
    whileitr.hasNext() ) {
      final Map.Entry entry = Map.Entry itr.next();
      final String name = String entry.getKey();
      final ParamLocationRecognizer.NamedParameterDescription description =
          ParamLocationRecognizer.NamedParameterDescription entry.getValue();
      namedParamDescriptorMap.put(
          name ,
              new NamedParameterDescriptorname, null, description.buildPositionsArray(), description.isJpaStyle() )
      );
    }

    return new ParameterMetadataordinalDescriptors, namedParamDescriptorMap );
  }

  private static class HQLQueryPlanKey implements Serializable {
    private final String query;
    private final boolean shallow;
    private final Set filterNames;
    private final int hashCode;

    public HQLQueryPlanKey(String query, boolean shallow, Map enabledFilters) {
      this.query = query;
      this.shallow = shallow;

      if enabledFilters == null || enabledFilters.isEmpty() ) {
        filterNames = Collections.EMPTY_SET;
      }
      else {
        Set tmp = new HashSet();
        tmp.addAllenabledFilters.keySet() );
        this.filterNames = Collections.unmodifiableSettmp );
      }

      int hash = query.hashCode();
      hash = 29 * hash + shallow ? );
      hash = 29 * hash + filterNames.hashCode();
      this.hashCode = hash;
    }

    public boolean equals(Object o) {
      if this == o ) {
        return true;
      }
      if o == null || getClass() != o.getClass() ) {
        return false;
      }

      final HQLQueryPlanKey that = HQLQueryPlanKey o;

      if shallow != that.shallow ) {
        return false;
      }
      if !filterNames.equalsthat.filterNames ) ) {
        return false;
      }
      if !query.equalsthat.query ) ) {
        return false;
      }

      return true;
    }

    public int hashCode() {
      return hashCode;
    }
  }

  private static class FilterQueryPlanKey implements Serializable {
    private final String query;
    private final String collectionRole;
    private final boolean shallow;
    private final Set filterNames;
    private final int hashCode;

    public FilterQueryPlanKey(String query, String collectionRole, boolean shallow, Map enabledFilters) {
      this.query = query;
      this.collectionRole = collectionRole;
      this.shallow = shallow;

      if enabledFilters == null || enabledFilters.isEmpty() ) {
        filterNames = Collections.EMPTY_SET;
      }
      else {
        Set tmp = new HashSet();
        tmp.addAllenabledFilters.keySet() );
        this.filterNames = Collections.unmodifiableSettmp );
      }

      int hash = query.hashCode();
      hash = 29 * hash + collectionRole.hashCode();
      hash = 29 * hash + shallow ? );
      hash = 29 * hash + filterNames.hashCode();
      this.hashCode = hash;
    }

    public boolean equals(Object o) {
      if this == o ) {
        return true;
      }
      if o == null || getClass() != o.getClass() ) {
        return false;
      }

      final FilterQueryPlanKey that = FilterQueryPlanKey o;

      if shallow != that.shallow ) {
        return false;
      }
      if !filterNames.equalsthat.filterNames ) ) {
        return false;
      }
      if !query.equalsthat.query ) ) {
        return false;
      }
      if !collectionRole.equalsthat.collectionRole ) ) {
        return false;
      }

      return true;
    }

    public int hashCode() {
      return hashCode;
    }
  }
}