com.haulmont.cuba.core.app.CrossDataStoreReferenceLoader.java Source code

Java tutorial

Introduction

Here is the source code for com.haulmont.cuba.core.app.CrossDataStoreReferenceLoader.java

Source

/*
 * Copyright (c) 2008-2016 Haulmont.
 *
 * 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 com.haulmont.cuba.core.app;

import com.google.common.collect.Sets;
import com.haulmont.bali.util.Preconditions;
import com.haulmont.chile.core.model.MetaClass;
import com.haulmont.chile.core.model.MetaProperty;
import com.haulmont.cuba.core.entity.Entity;
import com.haulmont.cuba.core.entity.IdProxy;
import com.haulmont.cuba.core.global.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;

import javax.inject.Inject;
import java.util.*;
import java.util.stream.Collectors;

/**
 * INTERNAL.
 * Populates references to entities from different data stores.
 */
@Component(CrossDataStoreReferenceLoader.NAME)
@Scope("prototype")
public class CrossDataStoreReferenceLoader {

    public static final String NAME = "cuba_CrossDataStoreReferenceLoader";

    private static final Logger log = LoggerFactory.getLogger(CrossDataStoreReferenceLoader.class);

    @Inject
    private Metadata metadata;

    @Inject
    private MetadataTools metadataTools;

    @Inject
    private DataManager dataManager;

    @Inject
    private ServerConfig serverConfig;

    private MetaClass metaClass;

    private View view;

    public CrossDataStoreReferenceLoader(MetaClass metaClass, View view) {
        Preconditions.checkNotNullArgument(metaClass, "metaClass is null");
        Preconditions.checkNotNullArgument(view, "view is null");
        this.metaClass = metaClass;
        this.view = view;
    }

    public Map<Class<? extends Entity>, List<CrossDataStoreProperty>> getCrossPropertiesMap() {
        Map<Class<? extends Entity>, List<CrossDataStoreProperty>> crossPropertiesMap = new HashMap<>();
        traverseView(view, crossPropertiesMap, Sets.newIdentityHashSet());
        return crossPropertiesMap;
    }

    private void traverseView(View view,
            Map<Class<? extends Entity>, List<CrossDataStoreProperty>> crossPropertiesMap, Set<View> visited) {
        if (visited.contains(view))
            return;
        visited.add(view);

        String storeName = metadataTools.getStoreName(metaClass);

        Class<? extends Entity> entityClass = view.getEntityClass();
        for (ViewProperty viewProperty : view.getProperties()) {
            MetaProperty metaProperty = metadata.getClassNN(entityClass).getPropertyNN(viewProperty.getName());
            if (metaProperty.getRange().isClass()) {
                MetaClass propertyMetaClass = metaProperty.getRange().asClass();
                if (!Objects.equals(metadataTools.getStoreName(propertyMetaClass), storeName)) {
                    List<String> relatedProperties = metadataTools.getRelatedProperties(metaProperty);
                    if (relatedProperties.size() == 0) {
                        continue;
                    }
                    if (relatedProperties.size() > 1) {
                        log.warn(
                                "More than 1 related property is defined for attribute {}, skip handling cross-datastore reference",
                                metaProperty);
                        continue;
                    }
                    List<CrossDataStoreProperty> crossProperties = crossPropertiesMap.computeIfAbsent(entityClass,
                            k -> new ArrayList<>());
                    if (crossProperties.stream().noneMatch(aProp -> aProp.property == metaProperty))
                        crossProperties.add(new CrossDataStoreProperty(metaProperty, viewProperty));
                }
                View propertyView = viewProperty.getView();
                if (propertyView != null) {
                    traverseView(propertyView, crossPropertiesMap, visited);
                }
            }
        }
    }

    public void processEntities(Collection<? extends Entity> entities) {
        Map<Class<? extends Entity>, List<CrossDataStoreProperty>> crossPropertiesMap = getCrossPropertiesMap();
        if (crossPropertiesMap.isEmpty())
            return;

        Set<Entity> affectedEntities = getAffectedEntities(entities, crossPropertiesMap);
        if (affectedEntities.isEmpty())
            return;

        List<EntityCrossDataStoreProperty> entityCrossDataStorePropertyList = new ArrayList<>();
        for (Entity affectedEntity : affectedEntities) {
            for (CrossDataStoreProperty crossDataStoreProperty : crossPropertiesMap
                    .get(affectedEntity.getClass())) {
                entityCrossDataStorePropertyList
                        .add(new EntityCrossDataStoreProperty(affectedEntity, crossDataStoreProperty));
            }
        }
        if (entityCrossDataStorePropertyList.size() == 1) {
            loadOne(entityCrossDataStorePropertyList.get(0));
        } else {
            entityCrossDataStorePropertyList.stream()
                    .collect(Collectors.groupingBy(EntityCrossDataStoreProperty::getCrossProp))
                    .forEach((ap, eapList) -> loadMany(ap,
                            eapList.stream().map(eap -> eap.entity).collect(Collectors.toList())));
        }
    }

    private Set<Entity> getAffectedEntities(Collection<? extends Entity> entities,
            Map<Class<? extends Entity>, List<CrossDataStoreProperty>> crossPropertiesMap) {
        Set<Entity> resultSet = new HashSet<>();
        for (Entity entity : entities) {
            metadataTools.traverseAttributesByView(view, entity, new EntityAttributeVisitor() {
                @Override
                public void visit(Entity entity, MetaProperty property) {
                    List<CrossDataStoreProperty> crossProperties = crossPropertiesMap.get(entity.getClass());
                    if (crossProperties != null) {
                        crossProperties.stream().filter(ap -> ap.property == property).forEach(ap -> {
                            if (entity.getValue(ap.relatedPropertyName) != null) {
                                resultSet.add(entity);
                            }
                        });
                    }
                }

                @Override
                public boolean skip(MetaProperty property) {
                    return !property.getRange().isClass();
                }
            });
        }
        return resultSet;
    }

    private void loadOne(EntityCrossDataStoreProperty entityCrossDataStoreProperty) {
        Entity entity = entityCrossDataStoreProperty.entity;
        CrossDataStoreProperty aProp = entityCrossDataStoreProperty.crossProp;
        Object id = entity.getValue(aProp.relatedPropertyName);

        LoadContext<Entity> loadContext = new LoadContext<>(aProp.property.getRange().asClass());
        loadContext.setId(id);
        if (aProp.viewProperty.getView() != null)
            loadContext.setView(aProp.viewProperty.getView());
        Entity relatedEntity = dataManager.load(loadContext);
        entity.setValue(aProp.property.getName(), relatedEntity);
    }

    private void loadMany(CrossDataStoreProperty crossDataStoreProperty, List<Entity> entities) {
        int offset = 0, limit = serverConfig.getCrossDataStoreReferenceLoadingBatchSize();
        while (true) {
            int end = offset + limit;
            List<Entity> batch = entities.subList(offset, Math.min(end, entities.size()));
            loadBatch(crossDataStoreProperty, batch);
            if (end >= entities.size())
                break;
            else
                offset += limit;
        }
    }

    private void loadBatch(CrossDataStoreProperty crossDataStoreProperty, List<Entity> entities) {
        List<Object> idList = entities.stream().map(e -> e.getValue(crossDataStoreProperty.relatedPropertyName))
                .filter(Objects::nonNull).distinct().collect(Collectors.toList());

        if (idList.isEmpty())
            return;

        MetaClass cdsrMetaClass = crossDataStoreProperty.property.getRange().asClass();
        LoadContext<Entity> loadContext = new LoadContext<>(cdsrMetaClass);

        MetaProperty primaryKeyProperty = metadataTools.getPrimaryKeyProperty(cdsrMetaClass);
        if (primaryKeyProperty == null || !primaryKeyProperty.getRange().isClass()) {
            String queryString = String.format("select e from %s e where e.%s in :idList", cdsrMetaClass,
                    crossDataStoreProperty.primaryKeyName);
            loadContext.setQuery(LoadContext.createQuery(queryString).setParameter("idList", idList));
        } else {
            // composite key entity
            StringBuilder sb = new StringBuilder("select e from ");
            sb.append(cdsrMetaClass).append(" e where ");

            MetaClass idMetaClass = primaryKeyProperty.getRange().asClass();
            for (Iterator<MetaProperty> it = idMetaClass.getProperties().iterator(); it.hasNext();) {
                MetaProperty property = it.next();
                sb.append("e.").append(crossDataStoreProperty.primaryKeyName).append(".")
                        .append(property.getName());
                sb.append(" in :list_").append(property.getName());
                if (it.hasNext())
                    sb.append(" and ");
            }
            LoadContext.Query query = LoadContext.createQuery(sb.toString());
            for (MetaProperty property : idMetaClass.getProperties()) {
                List<Object> propList = idList.stream().map(o -> ((Entity) o).getValue(property.getName()))
                        .collect(Collectors.toList());
                query.setParameter("list_" + property.getName(), propList);
            }
            loadContext.setQuery(query);
        }

        loadContext.setView(crossDataStoreProperty.viewProperty.getView());

        List<Entity> loadedEntities = dataManager.loadList(loadContext);

        for (Entity entity : entities) {
            Object relatedPropertyValue = entity.getValue(crossDataStoreProperty.relatedPropertyName);
            loadedEntities.stream().filter(e -> {
                Object id = e.getId() instanceof IdProxy ? ((IdProxy) e.getId()).getNN() : e.getId();
                return id.equals(relatedPropertyValue);
            }).findAny().ifPresent(e -> entity.setValue(crossDataStoreProperty.property.getName(), e));
        }
    }

    private static class EntityCrossDataStoreProperty {

        private final Entity entity;
        public final CrossDataStoreProperty crossProp;

        public EntityCrossDataStoreProperty(Entity entity, CrossDataStoreProperty crossDataStoreProperty) {
            this.entity = entity;
            this.crossProp = crossDataStoreProperty;
        }

        public CrossDataStoreProperty getCrossProp() {
            return crossProp;
        }

        @Override
        public String toString() {
            return entity + " -> " + crossProp;
        }
    }

    public class CrossDataStoreProperty {

        public final MetaProperty property;
        public final ViewProperty viewProperty;
        public final String relatedPropertyName;
        public final String primaryKeyName;

        public CrossDataStoreProperty(MetaProperty metaProperty, ViewProperty viewProperty) {
            this.property = metaProperty;
            this.viewProperty = viewProperty;

            List<String> relatedProperties = metadataTools.getRelatedProperties(property);
            relatedPropertyName = relatedProperties.get(0);

            String pkName = metadataTools.getPrimaryKeyName(property.getRange().asClass());
            primaryKeyName = pkName != null ? pkName : "id"; // sensible default for non-persistent entities
        }

        @Override
        public String toString() {
            return "CrossDataStoreProperty{" + property + "}";
        }
    }
}