All Downloads are FREE. Search and download functionalities are using the official Maven repository.

gorm.tools.utils.GormMetaUtils.groovy Maven / Gradle / Ivy

/*
* Copyright 2019 Yak.Works - Licensed under the Apache License, Version 2.0 (the "License")
* You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
*/
package gorm.tools.utils

import java.lang.reflect.Modifier

import groovy.transform.CompileDynamic
import groovy.transform.CompileStatic

import org.grails.datastore.gorm.GormEntity
import org.grails.datastore.mapping.model.MappingContext
import org.grails.datastore.mapping.model.PersistentEntity
import org.grails.datastore.mapping.model.PersistentProperty
import org.grails.datastore.mapping.proxy.ProxyHandler
import org.grails.orm.hibernate.cfg.Mapping
import org.springframework.validation.Validator

import gorm.tools.model.Persistable
import gorm.tools.validation.ApiConstraints
import grails.gorm.validation.ConstrainedEntity
import grails.gorm.validation.ConstrainedProperty
import yakworks.commons.lang.ClassUtils
import yakworks.commons.lang.NameUtils
import yakworks.commons.lang.Validate
import yakworks.spring.AppCtx

/**
 * A bunch of static helpers and lookup/finder statics for dealing with domain classes and PersistentEntity.
 * Useful methods to find the PersistentEntity and the mapping and meta fields.
 *
 * @author Joshua Burnett (@basejump)
 * @since 6.1
 */
@CompileStatic
class GormMetaUtils {

    // list of props to exclude in isExcluded used for getProperties
    private static List PROPERTY_EXCLUDES = [
        "properties", "dirty", "newOrDirty", "new"
    ]

    /**
     * Returns a persistent entity using the class.
     *
     * @param cls the domain class
     * @return The entity or null
     */
    static PersistentEntity getPersistentEntity(Class cls) {
        getMappingContext().getPersistentEntity(cls.name)
    }

    /**
     * Returns a persistent entity using a fully qualified name
     * with package such as "com.foo.Product".
     *
     * @param name an entity name which includes package
     * @return The entity or null
     */
    static PersistentEntity getPersistentEntity(String fullName) {
        getMappingContext().getPersistentEntity(fullName)
    }

    /**
     * calls the static getter method getGormPersistentEntity on the instace
     * to get the PersistentEntity
     *
     * @param instance the domain instance
     * @return The entity or null
     */
    static PersistentEntity getPersistentEntity(GormEntity instance) {
        return (PersistentEntity)ClassUtils.getStaticPropertyValue(instance.class, "gormPersistentEntity")
    }

    /**
     * finds domain using either a simple name like "Product" or fully qualified name "com.foo.Product"
     * name can start with either a lower or upper case
     *
     * @param name an entity name to search for
     * @return The entity or null
     */
    static PersistentEntity findPersistentEntity(String name) {
        if (name.indexOf('.') == -1) {
            String propertyName = NameUtils.getPropertyName(name)
            return getMappingContext().persistentEntities.find { PersistentEntity entity ->
                entity.decapitalizedName == propertyName
            }
        }
        return getPersistentEntity(name)
    }

    /**
     * the mapping grailsDomainClassMappingContext. This is the main holder for the persistentEntities
     */
    static MappingContext getMappingContext() {
        AppCtx.get("grailsDomainClassMappingContext", MappingContext)
    }

    static ProxyHandler getProxyHandler() {
        getMappingContext().getProxyHandler()
    }

    /**
     * the mapping grailsDomainClassMappingContext. This is the main holder for the persistentEntities
     */
    static Object unwrap(Object entity) {
        getProxyHandler().unwrap(entity)
    }

    /**
     * Returns the original class if the instance is a hibernate proxy. or returns the class name of the object
     */
    static Class getEntityClass(Object entity) {
        Validate.notNull(entity, "Entity is null")
        return getProxyHandler().getProxiedClass(entity)
    }

    /**
     * Doesnt require proxyHandler and just looks at the name.
     * - If the name has `_$$_` its java assist
     * - if it matches $HibernateProxy$ then its ByteBuddy
     * method then removes the suffixes then returns just the name.
     * if no match then it just returns the name
     */
    static String unwrapIfProxy(String name) {
        final int proxyIndicatorJavaAssist = name.indexOf('_$$_')
        final int proxyIndicatorByteBuddy = name.indexOf('$HibernateProxy$')
        if (proxyIndicatorJavaAssist > -1) {
            name = name.substring(0, proxyIndicatorJavaAssist)
        } else if(proxyIndicatorByteBuddy > -1){
            name = name.substring(0, proxyIndicatorByteBuddy)
        }
        return name
    }

    /**
     * checks if its new or if its dirty
     */
    static boolean isNewOrDirty(GormEntity entity) {
        if(entity == null){
            return false
        }
        // if its a proxy and its not initialized then its can't be new or dirty
        else if(!proxyHandler.isInitialized(entity)){
            return false
        } else {
            return entity.hasChanged() || ((Persistable)entity).isNew()
        }

    }
    /**
     * Returns the mapping for the entity to DB.
     *
     * @param pe the PersistentEntity can found using the loolup statics above
     * @return the mapping for the entity to DB
     */
    @CompileDynamic
    Mapping getMapping(PersistentEntity pe) {
        return getMappingContext().mappingFactory?.entityToMapping?.get(pe)
    }

    @CompileDynamic
    static Map findConstrainedProperties(Class pe) {
        findConstrainedProperties(pe.getGormPersistentEntity())
    }

    static Map findConstrainedProperties(PersistentEntity entity) {
        Validator validator = entity.getMappingContext().getEntityValidator(entity)
        if(validator instanceof ConstrainedEntity) {
            ConstrainedEntity constrainedEntity = (ConstrainedEntity)validator
            Map constrainedProperties = constrainedEntity.getConstrainedProperties()
            return constrainedProperties
        }
        return Collections.emptyMap()
    }

    static Map findNonValidatedProperties(PersistentEntity entity) {
        def apiConstraints = ApiConstraints.findApiConstraints(entity.javaClass)
        return apiConstraints ? apiConstraints.nonValidatedProperties : [:]
    }

    /**
     * returns both non-validated and validated Constrained Properties
     */
    static Map findAllConstrainedProperties(PersistentEntity entity) {
        return findConstrainedProperties(entity) + findNonValidatedProperties(entity)
    }

    // static Map getConstrainedProperties(PersistentEntity entity) {
    //     return new DefaultConstrainedDiscovery().findConstrainedProperties(entity)
    // }

    /**
     * Check if Persistent Entity has property by path
     *
     * @param domain Persistent Entity
     * @param property path for property
     * @return true if there is such property, false othervise
     */
    @CompileDynamic
    static boolean hasProperty(PersistentEntity domain, String property) {
        Closure checkProperty
        checkProperty = { PersistentEntity domainClass, List path ->
            PersistentProperty prop = domainClass?.getPropertyByName(path[0].toString())
            if (path.size() > 1 && prop) {
                checkProperty(prop.associatedEntity, path.tail())
            } else {
                prop as boolean
            }
        }
        checkProperty(domain, property.split("[.]") as List)

    }

    /** get PersistentProperty for path */
    @CompileDynamic
    static PersistentProperty getPersistentProperty(PersistentEntity domain, String property) {
        Closure getPerProperty
        getPerProperty = { PersistentEntity domainClass, List path ->
            PersistentProperty prop = domainClass?.getPropertyByName(path[0].toString())
            if (path.size() > 1 && prop) {
                getPerProperty(prop.associatedEntity, path.tail())
            } else {
                prop
            }
        }
        getPerProperty(domain, property.split("[.]") as List)
    }

    /**
     * Returns persistent properties for persistent entity(finds by name)
     * Adds composite identeties, which are not in persistent properties by default
     *
     * @param className name of the Persistent entity
     * @return list of PersistentProperties, includes composite Identities and identity
     */
    static List getPersistentProperties(String className){
        PersistentEntity domain = findPersistentEntity(className)
        getPersistentProperties(domain)
    }

    /**
     * Returns persistent properties for persistent entity(finds by name)
     * Adds composite identities, which are not in persistent properties by default
     *
     * @param className name of the Persistent entity
     * @return list of PersistentProperties, includes composite Identities and identity
     */
    static List getPersistentProperties(PersistentEntity domain){
        List result = domain.persistentProperties.collect() // collect copies it
        if(domain.compositeIdentity) result.addAll(domain.compositeIdentity)
        result.add(0, domain.getIdentity())
        result.unique()
    }

    /**
     * gets the id on the instance using entity reflector and trying not to init the proxy if its is one
     */
    static Serializable getId(Object instance) {
        GormEntity entity = (GormEntity)instance
        PersistentEntity persistentEntity = getPersistentEntity(entity)
        ProxyHandler proxyHandler = persistentEntity.mappingContext.proxyHandler
        if(proxyHandler.isProxy(entity)) {
            return proxyHandler.getIdentifier(entity)
        }
        else {
            persistentEntity.mappingContext.getEntityReflector(persistentEntity).getIdentifier(entity)
        }
    }

    /**
     * gets a Map representing the id key trying not to init the proxy if its is one.
     * see getId helper here as well.
     * If no defaults changed and id field is named id and is a of Long type
     * its would return '[id: 123]' as an example.
     */
    static Map getIdMap(GormEntity instance) {
        PersistentEntity persistentEntity = getPersistentEntity(instance)
        Serializable idVal = getId(instance)
        String idName = persistentEntity.identity.name
        return  [(idName): idVal]
    }

    /**
     * Get the entity properties for a GormEntity. Filters out the utility props using isExcludedProperty
     */
    static List getMetaProperties(Class entityClass) {
        List metaProps = entityClass.metaClass.properties
        List filteredProps = metaProps.findAll { MetaProperty mp ->
            !isExcludedProperty(mp)
        }
        return filteredProps
    }

    /**
     * default get properties for gorm entity class
     */
    static Map getProperties(Object instance) {
        Map props = [:]
        for (MetaProperty mp : getMetaProperties(instance.class)) {
            props[mp.name] = mp.getProperty(instance)
        }
        return props
    }

    /**
     * used for getProperties to exclude the utility properties that are on a GormEntity.
     */
    static boolean isExcludedProperty(MetaProperty mp) {
        return Modifier.isStatic(mp.getModifiers()) ||
            PROPERTY_EXCLUDES.contains(mp.getName()) ||
            org.grails.datastore.mapping.reflect.NameUtils.isConfigurational(mp.getName()) ||
            (mp instanceof MetaBeanProperty) && (((MetaBeanProperty) mp).getGetter()) == null
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy