org.beangle.orm.hibernate.internal.HibernateEntityContext.java Source code

Java tutorial

Introduction

Here is the source code for org.beangle.orm.hibernate.internal.HibernateEntityContext.java

Source

/*
 * Beangle, Agile Java/Scala Development Scaffold and Toolkit
 *
 * Copyright (c) 2005-2013, Beangle Software.
 *
 * Beangle is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * Beangle 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 Beangle.  If not, see <http://www.gnu.org/licenses/>.
 */
package org.beangle.orm.hibernate.internal;

import java.util.*;

import org.beangle.commons.collection.CollectUtils;
import org.beangle.commons.entity.metadata.CollectionType;
import org.beangle.commons.entity.metadata.ComponentType;
import org.beangle.commons.entity.metadata.EntityType;
import org.beangle.commons.entity.metadata.IdentifierType;
import org.beangle.commons.entity.metadata.Type;
import org.beangle.commons.entity.metadata.impl.AbstractEntityContext;
import org.beangle.commons.lang.time.Stopwatch;
import org.hibernate.SessionFactory;
import org.hibernate.metadata.ClassMetadata;
import org.hibernate.metadata.CollectionMetadata;
import org.hibernate.proxy.HibernateProxy;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.Assert;

public class HibernateEntityContext extends AbstractEntityContext {

    private final Logger logger = LoggerFactory.getLogger(HibernateEntityContext.class);

    private final Map<String, CollectionType> collectionTypes = CollectUtils.newHashMap();

    @Override
    public String getEntityName(Object target) {
        if (target instanceof HibernateProxy) {
            return ((HibernateProxy) target).getHibernateLazyInitializer().getEntityName();
        } else {
            return getEntityType(target.getClass()).getEntityName();
        }
    }

    /**
     * Build context from session factory
     */
    public void initFrom(SessionFactory sessionFactory) {
        Assert.notNull(sessionFactory);
        Stopwatch watch = new Stopwatch().start();
        Map<String, ClassMetadata> classMetadatas = sessionFactory.getAllClassMetadata();
        for (Iterator<ClassMetadata> iter = classMetadatas.values().iterator(); iter.hasNext();) {
            ClassMetadata cm = (ClassMetadata) iter.next();
            buildEntityType(sessionFactory, cm.getEntityName());
        }
        logger.info("Find {} entities,{} collections from hibernate in {}",
                new Object[] { entityTypes.size(), collectionTypes.size(), watch });
        if (logger.isDebugEnabled())
            loggerTypeInfo();
        collectionTypes.clear();
    }

    private void loggerTypeInfo() {
        List<String> names = CollectUtils.newArrayList(entityTypes.keySet());
        Collections.sort(names);
        for (final String entityName : names) {
            EntityType entityType = (EntityType) entityTypes.get(entityName);
            logger.debug("entity:{}-->{}", entityType.getEntityName(), entityType.getEntityClass().getName());
            logger.debug("propertyType size:{}", Integer.valueOf(entityType.getPropertyTypes().size()));
        }
        names.clear();
        names.addAll(collectionTypes.keySet());
        Collections.sort(names);
        for (final String collectionName : names) {
            CollectionType collectionType = (CollectionType) collectionTypes.get(collectionName);
            logger.debug("collection:{}", collectionType.getName());
            logger.debug("class:{}", collectionType.getCollectionClass());
            logger.debug("elementType:{}", collectionType.getElementType().getReturnedClass());
        }
    }

    /**
     * ???.<br>
     * ?entityTypes.
     * 
     * @param entityName
     */
    private EntityType buildEntityType(SessionFactory sessionFactory, String entityName) {
        EntityType entityType = (EntityType) entityTypes.get(entityName);
        if (null == entityType) {
            ClassMetadata cm = sessionFactory.getClassMetadata(entityName);
            if (null == cm) {
                logger.error("Cannot find ClassMetadata for {}", entityName);
                return null;
            }
            entityType = new EntityType(cm.getEntityName(), cm.getMappedClass(), cm.getIdentifierPropertyName(),
                    new IdentifierType(cm.getIdentifierType().getReturnedClass()));
            entityTypes.put(cm.getEntityName(), entityType);

            Map<String, Type> propertyTypes = entityType.getPropertyTypes();
            String[] ps = cm.getPropertyNames();
            for (int i = 0; i < ps.length; i++) {
                org.hibernate.type.Type type = cm.getPropertyType(ps[i]);
                if (type.isEntityType()) {
                    propertyTypes.put(ps[i], buildEntityType(sessionFactory, type.getName()));
                } else if (type.isComponentType()) {
                    propertyTypes.put(ps[i], buildComponentType(sessionFactory, entityName, ps[i]));
                } else if (type.isCollectionType()) {
                    propertyTypes.put(ps[i], buildCollectionType(sessionFactory, defaultCollectionClass(type),
                            entityName + "." + ps[i]));
                }
            }
        }
        return entityType;
    }

    private CollectionType buildCollectionType(SessionFactory sessionFactory, Class<?> collectionClass,
            String role) {
        CollectionMetadata cm = sessionFactory.getCollectionMetadata(role);
        // FIXME buildCollectionType in class hierarchy
        if (null == cm)
            return null;
        org.hibernate.type.Type type = cm.getElementType();
        EntityType elementType = null;
        if (type.isEntityType()) {
            elementType = (EntityType) entityTypes.get(type.getName());
            if (null == elementType)
                elementType = buildEntityType(sessionFactory, type.getName());
        } else {
            elementType = new EntityType(type.getReturnedClass());
        }

        CollectionType collectionType = new CollectionType();
        collectionType.setElementType(elementType);
        collectionType.setArray(cm.isArray());
        collectionType.setCollectionClass(collectionClass);
        if (!collectionTypes.containsKey(collectionType.getName())) {
            collectionTypes.put(collectionType.getName(), collectionType);
        }
        return collectionType;
    }

    private ComponentType buildComponentType(SessionFactory sessionFactory, String entityName,
            String propertyName) {
        EntityType entityType = (EntityType) entityTypes.get(entityName);
        if (null != entityType) {
            Type propertyType = (Type) entityType.getPropertyTypes().get(propertyName);
            if (null != propertyType) {
                return (ComponentType) propertyType;
            }
        }

        ClassMetadata cm = sessionFactory.getClassMetadata(entityName);
        org.hibernate.type.ComponentType hcType = (org.hibernate.type.ComponentType) cm
                .getPropertyType(propertyName);
        String[] propertyNames = hcType.getPropertyNames();

        ComponentType cType = new ComponentType(hcType.getReturnedClass());
        Map<String, Type> propertyTypes = cType.getPropertyTypes();
        for (int j = 0; j < propertyNames.length; j++) {
            org.hibernate.type.Type type = cm.getPropertyType(propertyName + "." + propertyNames[j]);
            if (type.isEntityType()) {
                propertyTypes.put(propertyNames[j], buildEntityType(sessionFactory, type.getName()));
            } else if (type.isComponentType()) {
                propertyTypes.put(propertyNames[j],
                        buildComponentType(sessionFactory, entityName, propertyName + "." + propertyNames[j]));
            } else if (type.isCollectionType()) {
                propertyTypes.put(propertyNames[j], buildCollectionType(sessionFactory,
                        defaultCollectionClass(type), entityName + "." + propertyName + "." + propertyNames[j]));
            }
        }
        return cType;
    }

    private Class<?> defaultCollectionClass(org.hibernate.type.Type collectionType) {
        if (collectionType.isAnyType()) {
            return null;
        } else if (org.hibernate.type.SetType.class.isAssignableFrom(collectionType.getClass())) {
            return HashSet.class;
        } else if (org.hibernate.type.MapType.class.isAssignableFrom(collectionType.getClass())) {
            return HashMap.class;
        } else {
            return ArrayList.class;
        }
    }

}