/* * Copyright 2004-2009 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.compass.gps.device.hibernate.entities; import java.util.ArrayList; import java.util.Iterator; import java.util.Map; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.compass.core.mapping.ResourceMapping; import org.compass.gps.device.hibernate.HibernateGpsDevice; import org.compass.gps.device.hibernate.HibernateGpsDeviceException; import org.compass.gps.spi.CompassGpsInterfaceDevice; import org.hibernate.EntityMode; import org.hibernate.SessionFactory; import org.hibernate.metadata.ClassMetadata; import org.hibernate.persister.entity.AbstractEntityPersister; /** * A default implemenation that automatically locates entities to index based * on Hibernate current mapped objects and mappings for the objects that exists * with Compass. * * @author kimchy */ public class DefaultHibernateEntitiesLocator implements HibernateEntitiesLocator { protected Log log = LogFactory.getLog(getClass()); public EntityInformation[] locate(SessionFactory sessionFactory, HibernateGpsDevice device) throws HibernateGpsDeviceException { CompassGpsInterfaceDevice gps = (CompassGpsInterfaceDevice) device.getGps(); ArrayList<EntityInformation> entitiesList = new ArrayList<EntityInformation>(); Map allClassMetaData = sessionFactory.getAllClassMetadata(); for (Iterator it = allClassMetaData.keySet().iterator(); it.hasNext();) { String entityname = (String) it.next(); if (!gps.hasMappingForEntityForIndex((entityname))) { if (log.isDebugEnabled()) { log.debug("Entity [" + entityname + "] does not have compass mapping, filtering it out"); } continue; } ClassMetadata classMetadata = (ClassMetadata) allClassMetaData.get(entityname); if (shouldFilter(entityname, classMetadata, allClassMetaData, device)) { continue; } Class clazz = classMetadata.getMappedClass(EntityMode.POJO); ResourceMapping resourceMapping = gps.getMappingForEntityForIndex(entityname); EntityInformation entityInformation = new EntityInformation(clazz, entityname, resourceMapping.getSubIndexHash().getSubIndexes()); entitiesList.add(entityInformation); if (log.isDebugEnabled()) { log.debug("Entity [" + entityname + "] will be indexed"); } } return entitiesList.toArray(new EntityInformation[entitiesList.size()]); } /** * Returns <code>true</code> if the entity name needs to be filtered. * * <p>Implementation filteres out inherited hibernate mappings, since the select query * for the base class will cover any inherited classes as well. * * <p>Note, that this method is called after it has been verified that the class has * Compass mappings (either directly, or indirectly by an interface or a super class). * * @param entityname The name of the entity * @param classMetadata The Hibernate class meta data. * @param device The Hibernate Gps device * @return <code>true</code> if the entity should be filtered out, <code>false</code> if not. */ protected boolean shouldFilter(String entityname, ClassMetadata classMetadata, Map allClassMetaData, HibernateGpsDevice device) { Class clazz = classMetadata.getMappedClass(EntityMode.POJO); // if it is inherited, do not add it to the classes to index, since the "from [entity]" // query for the base class will return results for this class as well if (classMetadata.isInherited()) { String superClassEntityName = ((AbstractEntityPersister) classMetadata).getMappedSuperclass(); ClassMetadata superClassMetadata = (ClassMetadata) allClassMetaData.get(superClassEntityName); Class superClass = superClassMetadata.getMappedClass(EntityMode.POJO); // only filter out classes that their super class has compass mappings if (superClass != null && ((CompassGpsInterfaceDevice) device.getGps()).hasMappingForEntityForIndex(superClass)) { if (log.isDebugEnabled()) { log.debug("Entity [" + entityname + "] is inherited and super class [" + superClass + "] has compass mapping, filtering it out"); } return true; } } return false; } }