Package org.hibernate.engine.query

Source Code of org.hibernate.engine.query.QueryPlanCache$DynamicFilterKey

/*
* Hibernate, Relational Persistence for Idiomatic Java
*
* Copyright (c) 2010, Red Hat Inc. or third-party contributors as
* indicated by the @author tags or express copyright attribution
* statements applied by the authors.  All third-party contributions are
* distributed under license by Red Hat Inc.
*
* This copyrighted material is made available to anyone wishing to use, modify,
* copy, or redistribute it subject to the terms and conditions of the GNU
* Lesser General Public License, as published by the Free Software Foundation.
*
* This program 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.  See the GNU Lesser General Public License
* for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this distribution; if not, write to:
* Free Software Foundation, Inc.
* 51 Franklin Street, Fifth Floor
* Boston, MA  02110-1301  USA
*/
package org.hibernate.engine.query;

import org.hibernate.util.PropertiesHelper;
import org.hibernate.util.SimpleMRUCache;
import org.hibernate.util.SoftLimitMRUCache;
import org.hibernate.util.CollectionHelper;
import org.hibernate.cfg.Environment;
import org.hibernate.engine.SessionFactoryImplementor;
import org.hibernate.engine.query.sql.NativeSQLQuerySpecification;
import org.hibernate.QueryException;
import org.hibernate.MappingException;
import org.hibernate.impl.FilterImpl;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

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

/**
* Acts as a cache for compiled query plans, as well as query-parameter metadata.
*
* @see Environment#QUERY_PLAN_CACHE_MAX_STRONG_REFERENCES
* @see Environment#QUERY_PLAN_CACHE_MAX_SOFT_REFERENCES
*
* @author Steve Ebersole
*/
public class QueryPlanCache implements Serializable {

  private static final Logger log = LoggerFactory.getLogger( QueryPlanCache.class );

  private SessionFactoryImplementor factory;

  public QueryPlanCache(SessionFactoryImplementor factory) {
    int maxStrongReferenceCount = PropertiesHelper.getInt(
        Environment.QUERY_PLAN_CACHE_MAX_STRONG_REFERENCES,
        factory.getProperties(),
        SoftLimitMRUCache.DEFAULT_STRONG_REF_COUNT
    );
    int maxSoftReferenceCount = PropertiesHelper.getInt(
        Environment.QUERY_PLAN_CACHE_MAX_SOFT_REFERENCES,
        factory.getProperties(),
        SoftLimitMRUCache.DEFAULT_SOFT_REF_COUNT
    );

    this.factory = factory;
    this.sqlParamMetadataCache = new SimpleMRUCache( maxStrongReferenceCount );
    this.planCache = new SoftLimitMRUCache( maxStrongReferenceCount, maxSoftReferenceCount );
  }

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

  /**
   * the cache of the actual plans...
   */
  private final SoftLimitMRUCache planCache;


  /**
   * Obtain the parameter metadata for given native-sql query.
   * <p/>
   * for native-sql queries, the param metadata is determined outside any relation to a query plan, because
   * query plan creation and/or retrieval 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
   *
   * @param query The query
   * @return The parameter metadata
   */
  public ParameterMetadata getSQLParameterMetadata(String query) {
    ParameterMetadata metadata = ( ParameterMetadata ) sqlParamMetadataCache.get( query );
    if ( metadata == null ) {
      metadata = buildNativeSQLParameterMetadata( query );
      sqlParamMetadataCache.put( query, metadata );
    }
    return metadata;
  }

  public HQLQueryPlan getHQLQueryPlan(String queryString, boolean shallow, Map enabledFilters)
      throws QueryException, MappingException {
    HQLQueryPlanKey key = new HQLQueryPlanKey( queryString, 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.put( key, plan );

    return plan;
  }

  public FilterQueryPlan getFilterQueryPlan(String filterString, String collectionRole, boolean shallow, Map enabledFilters)
      throws QueryException, MappingException {
    FilterQueryPlanKey key = new FilterQueryPlanKey( filterString, 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 FilterQueryPlan( filterString, collectionRole, shallow, enabledFilters, factory );
    }
    else {
      if ( log.isTraceEnabled() ) {
        log.trace( "located collection-filter query plan in cache (" + collectionRole + " : " + filterString + ")" );
      }
    }

    planCache.put( key, plan );

    return plan;
  }

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

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

    planCache.put( spec, plan );
    return plan;
  }

  @SuppressWarnings({ "UnnecessaryUnboxing" })
  private ParameterMetadata buildNativeSQLParameterMetadata(String sqlString) {
    ParamLocationRecognizer recognizer = ParamLocationRecognizer.parseLocations( sqlString );

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

    Iterator itr = recognizer.getNamedParameterDescriptionMap().entrySet().iterator();
    Map<String,NamedParameterDescriptor> namedParamDescriptorMap = new HashMap<String,NamedParameterDescriptor>();
    while( itr.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 NamedParameterDescriptor( name, null, description.buildPositionsArray(), description.isJpaStyle() )
      );
    }

    return new ParameterMetadata( ordinalDescriptors, namedParamDescriptorMap );
  }

  private static class HQLQueryPlanKey implements Serializable {
    private final String query;
    private final boolean shallow;
    private final Set<DynamicFilterKey> filterKeys;
    private final int hashCode;

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

      if ( enabledFilters == null || enabledFilters.isEmpty() ) {
        filterKeys = Collections.emptySet();
      }
      else {
        Set<DynamicFilterKey> tmp = new HashSet<DynamicFilterKey>(
            CollectionHelper.determineProperSizing( enabledFilters ),
            CollectionHelper.LOAD_FACTOR
        );
        for ( Object o : enabledFilters.values() ) {
          tmp.add( new DynamicFilterKey( (FilterImpl) o ) );
        }
        this.filterKeys = Collections.unmodifiableSet( tmp );
      }

      int hash = query.hashCode();
      hash = 29 * hash + ( shallow ? 1 : 0 );
      hash = 29 * hash + filterKeys.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;

      return shallow == that.shallow
          && filterKeys.equals( that.filterKeys )
          && query.equals( that.query );

    }

    public int hashCode() {
      return hashCode;
    }
  }

  private static class DynamicFilterKey implements Serializable {
    private final String filterName;
    private final Map<String,Integer> parameterMetadata;
    private final int hashCode;

    @SuppressWarnings({ "UnnecessaryBoxing" })
    private DynamicFilterKey(FilterImpl filter) {
      this.filterName = filter.getName();
      if ( filter.getParameters().isEmpty() ) {
        parameterMetadata = Collections.emptyMap();
      }
      else {
        parameterMetadata = new HashMap<String,Integer>(
            CollectionHelper.determineProperSizing( filter.getParameters() ),
            CollectionHelper.LOAD_FACTOR
        );
        for ( Object o : filter.getParameters().entrySet() ) {
          final Map.Entry entry = (Map.Entry) o;
          final String key = (String) entry.getKey();
          final Integer valueCount;
          if ( Collection.class.isInstance( entry.getValue() ) ) {
            valueCount =  ( (Collection) entry.getValue() ).size();
          }
          else {
            valueCount = 1;
          }
          parameterMetadata.put( key, valueCount );
        }
      }

      int hash = filterName.hashCode();
      hash = 31 * hash + parameterMetadata.hashCode();
      this.hashCode = hash;
    }

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

      DynamicFilterKey that = ( DynamicFilterKey ) o;

      return filterName.equals( that.filterName )
          && parameterMetadata.equals( that.parameterMetadata );

    }

    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<String> filterNames;
    private final int hashCode;

    @SuppressWarnings({ "unchecked" })
    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.emptySet();
      }
      else {
        Set<String> tmp = new HashSet<String>();
        tmp.addAll( enabledFilters.keySet() );
        this.filterNames = Collections.unmodifiableSet( tmp );
      }

      int hash = query.hashCode();
      hash = 29 * hash + collectionRole.hashCode();
      hash = 29 * hash + ( shallow ? 1 : 0 );
      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;

      return shallow == that.shallow
          && filterNames.equals( that.filterNames )
          && query.equals( that.query )
          && collectionRole.equals( that.collectionRole );

    }

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

Related Classes of org.hibernate.engine.query.QueryPlanCache$DynamicFilterKey

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.