/* * Hibernate, Relational Persistence for Idiomatic Java * * License: GNU Lesser General Public License (LGPL), version 2.1 or later. * See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>. */ package org.hibernate.query.criteria.internal.path; import java.io.Serializable; import java.lang.reflect.Member; import java.util.Map; import javax.persistence.metamodel.Attribute; import javax.persistence.metamodel.Bindable; import javax.persistence.metamodel.ManagedType; import javax.persistence.metamodel.MapAttribute; import javax.persistence.metamodel.SingularAttribute; import javax.persistence.metamodel.Type; import org.hibernate.engine.spi.SessionFactoryImplementor; import org.hibernate.query.criteria.internal.CriteriaBuilderImpl; import org.hibernate.query.criteria.internal.MapJoinImplementor; import org.hibernate.query.criteria.internal.PathImplementor; import org.hibernate.query.criteria.internal.PathSource; import org.hibernate.query.criteria.internal.compile.RenderingContext; import org.hibernate.persister.collection.CollectionPersister; /** * {@link javax.persistence.criteria.MapJoin#key} poses a number of implementation difficulties in terms of the * type signatures amongst the {@link javax.persistence.criteria.Path}, {@link javax.persistence.criteria.Join} and * {@link Attribute}. The implementations found here provide that bridge. * * @author Steve Ebersole */ public class MapKeyHelpers { /** * Disallow instantiation */ private MapKeyHelpers() { } /** * Models a path to a map key. This is the actual return used from {@link javax.persistence.criteria.MapJoin#key} * * @param <K> The type of the map key. */ public static class MapKeyPath<K> extends AbstractPathImpl<K> implements PathImplementor<K>, Serializable { private final MapKeyAttribute<K> mapKeyAttribute; public MapKeyPath( CriteriaBuilderImpl criteriaBuilder, MapKeySource<K,?> source, MapKeyAttribute<K> mapKeyAttribute) { super( criteriaBuilder, mapKeyAttribute.getJavaType(), source ); this.mapKeyAttribute = mapKeyAttribute; } @Override public MapKeySource getPathSource() { return (MapKeySource) super.getPathSource(); } public MapKeyAttribute<K> getAttribute() { return mapKeyAttribute; } private boolean isBasicTypeKey() { return Attribute.PersistentAttributeType.BASIC == mapKeyAttribute.getPersistentAttributeType(); } @Override protected boolean canBeDereferenced() { return ! isBasicTypeKey(); } @Override protected Attribute locateAttributeInternal(String attributeName) { if ( ! canBeDereferenced() ) { throw new IllegalArgumentException( "Map key [" + getPathSource().getPathIdentifier() + "] cannot be dereferenced" ); } throw new UnsupportedOperationException( "Not yet supported!" ); } public Bindable<K> getModel() { return mapKeyAttribute; } @Override @SuppressWarnings("unchecked") public <T extends K> MapKeyPath<T> treatAs(Class<T> treatAsType) { // todo : if key is an entity, this is probably not enough return (MapKeyPath<T>) this; } @Override public String render(RenderingContext renderingContext) { PathSource<?> source = getPathSource(); String name; if ( source != null ) { source.prepareAlias( renderingContext ); name = source.getPathIdentifier(); } else { name = getAttribute().getName(); } return "key(" + name + ")"; } } /** * Defines a path for the map which can then be used to represent the source of the * map key "attribute". * * @param <K> The map key type * @param <V> The map value type */ public static class MapKeySource<K,V> extends AbstractPathImpl<Map<K, V>> implements PathImplementor<Map<K, V>>, Serializable { private final MapAttribute<?,K,V> mapAttribute; private final MapJoinImplementor<?,K,V> mapJoin; public MapKeySource( CriteriaBuilderImpl criteriaBuilder, Class<Map<K, V>> javaType, MapJoinImplementor<?,K,V> mapJoin, MapAttribute<?,K,V> attribute) { super( criteriaBuilder, javaType, null ); this.mapJoin = mapJoin; this.mapAttribute = attribute; } public MapAttribute<?,K,V> getAttribute() { return mapAttribute; } @SuppressWarnings({ "unchecked" }) public Bindable<Map<K, V>> getModel() { // TODO : ok??? the attribute is in fact bindable, but its type signature is different return (Bindable<Map<K, V>>) mapAttribute; } @Override public PathImplementor<?> getParentPath() { return (PathImplementor<?>) mapJoin.getParentPath(); } @Override protected boolean canBeDereferenced() { return false; } @Override protected Attribute locateAttributeInternal(String attributeName) { throw new IllegalArgumentException( "Map [" + mapJoin.getPathIdentifier() + "] cannot be dereferenced" ); } @Override public <T extends Map<K, V>> PathImplementor<T> treatAs(Class<T> treatAsType) { throw new UnsupportedOperationException(); } @Override public String getPathIdentifier() { return mapJoin.getPathIdentifier(); } @Override public void prepareAlias(RenderingContext renderingContext) { mapJoin.prepareAlias( renderingContext ); } } /** * Defines an {@link javax.persistence.metamodel.Attribute} modelling of a map-key. * * @param <K> The type of the map key */ public static class MapKeyAttribute<K> implements SingularAttribute<Map<K,?>,K>, Bindable<K>, Serializable { private final MapAttribute<?,K,?> attribute; private final CollectionPersister mapPersister; private final org.hibernate.type.Type mapKeyType; private final Type<K> jpaType; private final BindableType jpaBindableType; private final Class<K> jpaBinableJavaType; private final PersistentAttributeType persistentAttributeType; public MapKeyAttribute(CriteriaBuilderImpl criteriaBuilder, MapAttribute<?, K, ?> attribute) { this.attribute = attribute; this.jpaType = attribute.getKeyType(); this.jpaBinableJavaType = attribute.getKeyJavaType(); this.jpaBindableType = Type.PersistenceType .ENTITY.equals( jpaType.getPersistenceType() ) ? BindableType.ENTITY_TYPE : BindableType.SINGULAR_ATTRIBUTE; String guessedRoleName = determineRole( attribute ); SessionFactoryImplementor sfi = criteriaBuilder.getEntityManagerFactory().getSessionFactory(); mapPersister = sfi.getCollectionPersister( guessedRoleName ); if ( mapPersister == null ) { throw new IllegalStateException( "Could not locate collection persister [" + guessedRoleName + "]" ); } mapKeyType = mapPersister.getIndexType(); if ( mapKeyType == null ) { throw new IllegalStateException( "Could not determine map-key type [" + guessedRoleName + "]" ); } this.persistentAttributeType = mapKeyType.isEntityType() ? PersistentAttributeType.MANY_TO_ONE : mapKeyType.isComponentType() ? PersistentAttributeType.EMBEDDED : PersistentAttributeType.BASIC; } private String determineRole(MapAttribute<?,K,?> attribute) { return attribute.getDeclaringType().getJavaType().getName() + '.' + attribute.getName(); } @Override public String getName() { // TODO : ??? return "map-key"; } @Override public PersistentAttributeType getPersistentAttributeType() { return persistentAttributeType; } @Override public ManagedType<Map<K, ?>> getDeclaringType() { // TODO : ??? return null; } @Override public Class<K> getJavaType() { return attribute.getKeyJavaType(); } @Override public Member getJavaMember() { // TODO : ??? return null; } @Override public boolean isAssociation() { return mapKeyType.isEntityType(); } @Override public boolean isCollection() { return false; } @Override public boolean isId() { return false; } @Override public boolean isVersion() { return false; } @Override public boolean isOptional() { return false; } @Override public Type<K> getType() { return jpaType; } @Override public BindableType getBindableType() { return jpaBindableType; } @Override public Class<K> getBindableJavaType() { return jpaBinableJavaType; } } }