org.web4thejob.orm.PropertyMetadataImpl.java Source code

Java tutorial

Introduction

Here is the source code for org.web4thejob.orm.PropertyMetadataImpl.java

Source

/*
 * Copyright (c) 2012-2014 Veniamin Isaias.
 *
 * This file is part of web4thejob.
 *
 * Web4thejob 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 any later version.
 *
 * Web4thejob 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 web4thejob.  If not, see <http://www.gnu.org/licenses/>.
 */

package org.web4thejob.orm;

import org.hibernate.engine.spi.SessionFactoryImplementor;
import org.hibernate.mapping.*;
import org.hibernate.persister.entity.Joinable;
import org.hibernate.proxy.HibernateProxy;
import org.hibernate.tuple.GenerationTiming;
import org.hibernate.type.*;
import org.hibernate.validator.constraints.NotBlank;
import org.hibernate.validator.constraints.NotEmpty;
import org.springframework.beans.PropertyAccessorFactory;
import org.web4thejob.context.ContextUtil;

import javax.validation.Validator;
import javax.validation.constraints.NotNull;
import javax.validation.groups.Default;
import javax.validation.metadata.BeanDescriptor;
import javax.validation.metadata.ConstraintDescriptor;
import javax.validation.metadata.PropertyDescriptor;
import java.lang.annotation.Annotation;
import java.lang.annotation.ElementType;
import java.util.*;
import java.util.Map;
import java.util.Set;

/**
 * @author Veniamin Isaias
 * @since 1.0.0
 */

/*package*/class PropertyMetadataImpl implements PropertyMetadata, Comparable<PropertyMetadata> {
    // ------------------------------ FIELDS ------------------------------

    private static final String META_FRIENDLY_NAME = "friendlyName";
    private static final String META_INDEX = "index";
    private static final String META_DISABLE_USER_INSERT = "disable-user-insert";
    private static final String META_DISABLE_USER_UPDATE = "disable-user-update";
    private static final String META_DEFAULT_FORMAT = "default-format";
    private static final String META_DEFAULT_STYLE = "default-style";
    private static final String META_DEFAULT_ALIGN = "default-align";
    private static final String META_DEFAULT_WIDTH = "default-width";
    private static final String META_DEFAULT_HEIGHT = "default-height";
    private final EntityMetadataImpl entityMetadata;
    private final String associatedEntityName;
    private final Property property;
    private final int maxLength;
    private final int index;
    private final boolean disableUserInsert;
    private final boolean disableUserUpdate;
    private final boolean identifier;
    private final boolean identityIdentifier;
    private final boolean formula;
    private final String friendlyName;
    private final String format;
    private final String align;
    private final String style;
    private final String width;
    private final String height;
    private final Map<String, AnnotationMetadata<? extends Annotation>> annotations;

    // --------------------------- CONSTRUCTORS ---------------------------

    public PropertyMetadataImpl(EntityMetadataImpl entityMetadata, String propertyName) {
        this(entityMetadata, propertyName, null);
    }

    @SuppressWarnings("unchecked")
    protected PropertyMetadataImpl(EntityMetadataImpl entityMetadata, String propertyName,
            Class<? extends Entity> subclassAssociation) {
        this.entityMetadata = entityMetadata;
        this.property = entityMetadata.getPersistentClass().getProperty(propertyName);

        if (property.getType() instanceof AssociationType) {
            String tempName = ((AssociationType) property.getType())
                    .getAssociatedEntityName(ContextUtil.getBean(SessionFactoryImplementor.class));

            if (subclassAssociation != null) {
                tempName = subclassAssociation.getCanonicalName();
            }
            associatedEntityName = tempName;
        } else {
            associatedEntityName = null;
        }

        int maxLength = 0;
        boolean hasFormula = false;
        for (final Iterator<?> iter = property.getColumnIterator(); iter.hasNext();) {
            final Object item = iter.next();
            if (item instanceof Column && ((Column) item).getLength() > 0) {
                maxLength += ((Column) item).getLength();
            } else if (item instanceof Formula) {
                hasFormula = true;
            }
        }
        this.formula = hasFormula;
        this.maxLength = maxLength;
        this.identifier = property.getName().equals(entityMetadata.getIdentifierName());
        this.annotations = getAnnotations();
        this.identityIdentifier = this.identifier
                && MetaUtil.isIdentityKey(entityMetadata.getPersistentClass().getIdentifier());

        if (MetaUtil.hasMetaAttribute(property, META_FRIENDLY_NAME)) {
            friendlyName = MetaUtil.getMetaAttribute(property, META_FRIENDLY_NAME);
        } else {
            friendlyName = property.getName();
        }
        if (MetaUtil.hasMetaAttribute(property, META_INDEX)) {
            index = Integer.valueOf(MetaUtil.getMetaAttribute(property, META_INDEX));
        } else {
            index = MetaUtil.getHibernatePropertyIndex(entityMetadata.getName(), property.getName());
        }
        if (MetaUtil.hasMetaAttribute(property, META_DISABLE_USER_INSERT)) {
            disableUserInsert = Boolean.valueOf(MetaUtil.getMetaAttribute(property, META_DISABLE_USER_INSERT));
        } else {
            disableUserInsert = false;
        }
        if (MetaUtil.hasMetaAttribute(property, META_DISABLE_USER_UPDATE)) {
            disableUserUpdate = Boolean.valueOf(MetaUtil.getMetaAttribute(property, META_DISABLE_USER_UPDATE));
        } else {
            disableUserUpdate = false;
        }
        if (MetaUtil.hasMetaAttribute(property, META_DEFAULT_FORMAT)) {
            format = MetaUtil.getMetaAttribute(property, META_DEFAULT_FORMAT);
        } else {
            format = buildDefaultPattern();
        }
        if (MetaUtil.hasMetaAttribute(property, META_DEFAULT_STYLE)) {
            style = MetaUtil.getMetaAttribute(property, META_DEFAULT_STYLE);
        } else {
            style = null;
        }
        if (MetaUtil.hasMetaAttribute(property, META_DEFAULT_ALIGN)) {
            align = MetaUtil.getMetaAttribute(property, META_DEFAULT_ALIGN);
        } else {
            align = buildDefaultAlign();
        }
        if (MetaUtil.hasMetaAttribute(property, META_DEFAULT_WIDTH)) {
            width = MetaUtil.getMetaAttribute(property, META_DEFAULT_WIDTH);
        } else {
            width = null;
        }
        if (MetaUtil.hasMetaAttribute(property, META_DEFAULT_HEIGHT)) {
            height = MetaUtil.getMetaAttribute(property, META_DEFAULT_HEIGHT);
        } else {
            height = null;
        }
    }

    private Map<String, AnnotationMetadata<? extends Annotation>> getAnnotations() {
        final Map<String, AnnotationMetadata<? extends Annotation>> propAnnotations = new HashMap<String, AnnotationMetadata<? extends Annotation>>();
        if (MetaReaderServiceImpl.annoCache.containsKey(entityMetadata.getName())) {
            for (final String annotation : MetaReaderServiceImpl.annoCache.get(entityMetadata.getName()).keySet()) {
                if (MetaReaderServiceImpl.annoCache.get(entityMetadata.getName()).get(annotation)
                        .containsKey(getName())) {
                    propAnnotations.put(annotation, MetaReaderServiceImpl.annoCache.get(entityMetadata.getName())
                            .get(annotation).get(getName()));
                }
            }
        }

        if (propAnnotations.size() > 0) {
            return propAnnotations;
        } else {
            return Collections.emptyMap();
        }
    }

    @Override
    public String getName() {
        return property.getName();
    }

    private String buildDefaultPattern() {
        if (property.getType() instanceof LongType || property.getType() instanceof IntegerType
                || property.getType() instanceof ShortType || property.getType() instanceof BigIntegerType) {
            return "number,integer";
        } else if (property.getType() instanceof DoubleType || property.getType() instanceof FloatType
                || property.getType() instanceof BigDecimalType) {
            return "number,#,##0.00";
        } else if (property.getType() instanceof CurrencyType) {
            return "number,currency";
        } else if (property.getType() instanceof DateType) {
            return "date";
        } else if (property.getType() instanceof TimeType) {
            return "time";
        } else if (property.getType() instanceof TimestampType) {
            return "date,long";
        } else {
            return null;
        }
    }

    private String buildDefaultAlign() {
        if (Number.class.isAssignableFrom(getJavaType())) {
            return "right";
        } else if (Date.class.isAssignableFrom(getJavaType())) {
            return "center";
        } else if (String.class.isAssignableFrom(getJavaType()) && getMaxLength() > 0 && getMaxLength() <= 5) {
            return "center";
        }
        return null;
    }

    @Override
    public Class<?> getJavaType() {
        return property.getType().getReturnedClass();
    }

    // --------------------- GETTER / SETTER METHODS ---------------------

    @Override
    public String getAlign() {
        return align;
    }

    @Override
    public EntityMetadata getEntityMetadata() {
        return entityMetadata;
    }

    @Override
    public String getFormat() {
        return format;
    }

    @Override
    public String getFriendlyName() {
        return friendlyName;
    }

    @Override
    public int getIndex() {
        return index;
    }

    @Override
    public int getMaxLength() {
        return maxLength;
    }

    @Override
    public String getStyle() {
        return style;
    }

    @Override
    public boolean isIdentifier() {
        return identifier;
    }

    @Override
    public boolean isIdentityIdentifier() {
        return identityIdentifier;
    }

    // ------------------------ CANONICAL METHODS ------------------------

    @Override
    public String toString() {
        final StringBuilder sb = new StringBuilder();
        sb.append(entityMetadata.getName()).append("|").append(getName());
        return sb.toString();
    }

    // ------------------------ INTERFACE METHODS ------------------------

    // --------------------- Interface Comparable ---------------------

    @Override
    public int compareTo(PropertyMetadata other) {
        final Integer i1 = getIndex();
        final Integer i2 = other.getIndex();

        final int res = i1.compareTo(i2);
        if (res != 0) {
            return res;
        } else {
            return Integer.valueOf(hashCode()).compareTo(other.hashCode());
        }
    }

    // --------------------- Interface PropertyMetadata ---------------------

    @Override
    public <E extends Entity> void deproxyValue(E entity) {
        if (isProxyValue(entity)) {
            setValue(entity, getValue(entity));
        }
    }

    @Override
    @SuppressWarnings("unchecked")
    public <A extends Annotation> A getAnnotation(Class<A> annotationType) {
        if (isAnnotatedWith(annotationType)) {
            return (A) annotations.get(annotationType.getName()).getAnnotation();
        }
        return null;
    }

    @Override
    public EntityMetadata getAssociatedEntityMetadata() {
        if (associatedEntityName != null) {
            return ContextUtil.getMRS().getEntityMetadata(associatedEntityName);
        }
        return null;
    }

    @Override
    @SuppressWarnings("unchecked")
    public <T, E extends Entity> T getValue(E entity) {
        if (isOneToManyType()) {
            throw new RuntimeException("cannot call getValue for a Collection property");
        }

        if (entity instanceof HibernateProxy) {
            entity = ContextUtil.getMRS().deproxyEntity(entity);
        }

        T value;
        if (!isIdentifier()) {
            value = (T) entityMetadata.getClassMetadata().getPropertyValue(entity, getName());
        } else {
            value = (T) PropertyAccessorFactory.forDirectFieldAccess(entity).getPropertyValue(getName());
        }

        if (value != null && value instanceof HibernateProxy) {
            value = (T) ContextUtil.getMRS().deproxyEntity((Entity) value);
        }

        return value;
    }

    @Override
    public boolean hasColumn(String name) {
        for (final Iterator<?> iter = property.getColumnIterator(); iter.hasNext();) {
            final Object item = iter.next();
            if (item instanceof Selectable) {
                if (((Selectable) item).getText().equals(name)) {
                    return true;
                }
            }
        }

        return false;
    }

    @Override
    public boolean hasColumns() {
        return property.getColumnSpan() > 0;
    }

    @Override
    public <A extends Annotation> boolean isAnnotatedWith(Class<A> annotationType) {
        return annotations.containsKey(annotationType.getName());
    }

    @Override
    public boolean isAssociatedWith(PropertyMetadata propertyMetadata) {
        if (!(isAssociationType() && propertyMetadata.isAssociationType())) {
            return false;
        } else if (!getAssociatedEntityMetadata().getEntityType()
                .isAssignableFrom(propertyMetadata.getEntityMetadata().getEntityType())) {
            return false;
        } else if (isOneToOneType() && propertyMetadata.isOneToOneType()) {
            // OneToOne has no columns so since the entity type is the same
            // (checked above) it is safe to assume that properties are
            // associated.
            return true;
        } else if (!propertyMetadata.hasColumns()) {
            return propertyMetadata.isAssociatedWith(this);
        }

        final Joinable j = ((AssociationType) property.getType())
                .getAssociatedJoinable(ContextUtil.getBean(SessionFactoryImplementor.class));

        for (final String p : j.getKeyColumnNames()) {
            String p2 = p.replaceAll("\"", "");
            p2 = p2.replaceAll("`", "");
            if (propertyMetadata.hasColumn(p2)) {
                return true;
            }
        }

        return false;
    }

    @Override
    public boolean isAssociationType() {
        return property.getType() instanceof AssociationType;
    }

    @Override
    public boolean isSubclassType() {
        return false;
    }

    @Override
    public boolean isBlobType() {
        return property.getType() instanceof MaterializedBlobType;
    }

    @Override
    public boolean isClobType() {
        return property.getType() instanceof MaterializedClobType;
    }

    @Override
    public boolean isTextType() {
        return property.getType() instanceof TextType;
    }

    @Override
    public PropertyMetadata castForSubclass(Class<? extends Entity> subclass) {
        if (isAssociationType()) {
            for (Class<? extends Entity> subtype : getAssociatedEntityMetadata().getSubclasses()) {
                if (subtype.equals(subclass)) {
                    return new SubclassPropertyMetadataImpl(entityMetadata, subtype, getName());
                }
            }
        }
        return null;
    }

    @Override
    public boolean isCompositeType() {
        return property.getType() instanceof CompositeType;
    }

    @Override
    public boolean isInsertable() {
        return !entityMetadata.isReadOnly() && property.isInsertable() && !disableUserInsert && !formula
                && !isIdentityIdentifier() && (GenerationTiming.NEVER == getGenerationTiming());
    }

    @Override
    public boolean isManyToOneType() {
        return property.getType() instanceof ManyToOneType;
    }

    @Override
    public boolean isUniqueKeyWith(String propertyName) {

        if (getName().equals(propertyName)) {
            return false;
        }

        PersistentClass pc = entityMetadata.getPersistentClass();
        while (pc != null) {
            Table table = pc.getTable();
            for (Iterator<?> iterUK = table.getUniqueKeyIterator(); iterUK.hasNext();) {
                UniqueKey uniqueKey = (UniqueKey) iterUK.next();

                if (!(uniqueKey.getColumnSpan() > 1)) {
                    continue;
                }

                int totcols = 0;
                boolean keyMembers = true;

                for (PropertyMetadataImpl propertyMetadata : Arrays.asList(this,
                        (PropertyMetadataImpl) entityMetadata.getPropertyMetadata(propertyName))) {
                    if (!propertyMetadata.hasColumns()) {
                        return false;
                    }

                    for (final Iterator<?> iterCol = propertyMetadata.property.getColumnIterator(); iterCol
                            .hasNext();) {
                        Column column = (Column) iterCol.next();
                        keyMembers &= column != null && uniqueKey.containsColumn(column);
                    }
                    totcols += propertyMetadata.property.getColumnSpan();
                }

                if (keyMembers && totcols == uniqueKey.getColumnSpan()) {
                    return true;
                }

            }

            pc = pc.getSuperclass();

        }

        return false;
    }

    @Override
    public boolean isMemberOfUniqueKey() {
        if (!hasColumns()) {
            return false;
        }

        PersistentClass pc = entityMetadata.getPersistentClass();
        while (pc != null) {
            Table table = pc.getTable();
            for (Iterator<?> iterUK = table.getUniqueKeyIterator(); iterUK.hasNext();) {
                UniqueKey uniqueKey = (UniqueKey) iterUK.next();
                boolean member = true;
                for (final Iterator<?> iterCol = property.getColumnIterator(); iterCol.hasNext();) {
                    Column column = (Column) iterCol.next();
                    member &= column != null && uniqueKey.containsColumn(column);
                }

                if (member) {
                    return true;
                }
            }
            pc = pc.getSuperclass();
        }

        return false;
    }

    @Override
    @SuppressWarnings("unchecked")
    public boolean isOfJavaType(Class<?> type) {
        return property.getType().getReturnedClass().isAssignableFrom(type);
    }

    @Override
    public boolean isOneToManyType() {
        return property.getType() instanceof CollectionType;
    }

    @Override
    public boolean isOneToOneType() {
        return property.getType() instanceof OneToOneType;
    }

    @Override
    public boolean isNumericType() {
        return Number.class.isAssignableFrom(property.getType().getReturnedClass());
    }

    @Override
    public boolean isOptional() {
        return property.isOptional() && !isMandatory();
    }

    protected boolean isMandatory() {
        final Validator validator = ContextUtil.getBean(Validator.class);
        final BeanDescriptor beanDescriptor = validator
                .getConstraintsForClass(getEntityMetadata().getMappedClass());

        if (!beanDescriptor.isBeanConstrained()) {
            return false;
        }
        PropertyDescriptor propertyDescriptor = beanDescriptor.getConstraintsForProperty(property.getName());
        if (propertyDescriptor == null) {
            return false;
        }
        Set<ConstraintDescriptor<?>> constraints = propertyDescriptor.findConstraints()
                .declaredOn(ElementType.FIELD).unorderedAndMatchingGroups(Default.class).getConstraintDescriptors();

        for (ConstraintDescriptor<?> constraint : constraints) {
            if (NotNull.class.isInstance(constraint.getAnnotation())
                    || NotBlank.class.isInstance(constraint.getAnnotation())
                    || NotEmpty.class.isInstance(constraint.getAnnotation())) {
                return true;
            }
        }

        return false;
    }

    @Override
    public boolean isTimestampType() {
        return property.getType() instanceof TimestampType;
    }

    @Override
    public <E extends Entity> boolean isProxyValue(E entity) {
        if (!isIdentifier()) {
            if (entity instanceof HibernateProxy) {
                entity = ContextUtil.getMRS().deproxyEntity(entity);
            }

            Object value = entityMetadata.getClassMetadata().getPropertyValue(entity, getName());
            return value != null && value instanceof HibernateProxy;
        } else {
            return false;
        }
    }

    @Override
    public boolean isUpdateable() {
        return !entityMetadata.isReadOnly() && property.isUpdateable() && !disableUserUpdate && !formula
                && !isIdentityIdentifier() && (GenerationTiming.NEVER == getGenerationTiming()
                        || GenerationTiming.INSERT == getGenerationTiming());
    }

    private GenerationTiming getGenerationTiming() {
        if (property.getValueGenerationStrategy() != null) {
            return property.getValueGenerationStrategy().getGenerationTiming();
        }
        return GenerationTiming.NEVER;
    }

    @Override
    public <E extends Entity> void setValue(E entity, Object value) {
        if (!identifier) {
            entityMetadata.getClassMetadata().setPropertyValue(entity, getName(), value);
        } else {
            PropertyAccessorFactory.forDirectFieldAccess(entity).setPropertyValue(getName(), value);
        }
    }

    // -------------------------- OTHER METHODS --------------------------

    @Override
    public boolean equals(Object obj) {
        if (obj == null) {
            return false;
        } else if (!PropertyMetadata.class.isInstance(obj)) {
            return false;
        } else {
            final PropertyMetadata other = (PropertyMetadata) obj;
            return entityMetadata.equals(other.getEntityMetadata()) && getName().equals(other.getName());
        }
    }

    @Override
    public String getWidth() {
        return width;
    }

    @Override
    public String getHeight() {
        return height;
    }
}