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

com.jaxio.celerio.factory.ProjectFactory Maven / Gradle / Ivy

There is a newer version: 4.0.23
Show newest version
/*
 * Copyright 2015 JAXIO http://www.jaxio.com
 *
 * 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.jaxio.celerio.factory;

import com.jaxio.celerio.Config;
import com.jaxio.celerio.ConfigurationCheck;
import com.jaxio.celerio.aspects.ForbiddenWhenBuildingAspect;
import com.jaxio.celerio.configuration.*;
import com.jaxio.celerio.configuration.convention.*;
import com.jaxio.celerio.configuration.database.Table;
import com.jaxio.celerio.configuration.entity.EntityConfig;
import com.jaxio.celerio.configuration.entity.EnumConfig;
import com.jaxio.celerio.configuration.entity.EnumType;
import com.jaxio.celerio.configuration.entity.EnumValue;
import com.jaxio.celerio.convention.GeneratedPackage;
import com.jaxio.celerio.convention.WellKnownFolder;
import com.jaxio.celerio.factory.conventions.AccountConvention;
import com.jaxio.celerio.model.Attribute;
import com.jaxio.celerio.model.Entity;
import com.jaxio.celerio.model.Project;
import com.jaxio.celerio.model.Relation;
import com.jaxio.celerio.model.relation.AbstractRelation;
import com.jaxio.celerio.model.support.ClassNamer2;
import com.jaxio.celerio.model.support.ValuesAttribute;
import com.jaxio.celerio.model.support.custom.CustomAttribute;
import com.jaxio.celerio.model.support.custom.CustomEntity;
import com.jaxio.celerio.model.support.formatter.FormatterAttribute;
import com.jaxio.celerio.model.support.h2.H2Attribute;
import com.jaxio.celerio.model.support.jpa.*;
import com.jaxio.celerio.model.support.jquery.JQueryAttribute;
import com.jaxio.celerio.model.support.jsf.JsfAttribute;
import com.jaxio.celerio.model.support.search.SearchAttribute;
import com.jaxio.celerio.model.support.search.SearchEntity;
import com.jaxio.celerio.model.support.springmvc.SpringMVCAttribute;
import com.jaxio.celerio.model.support.springmvc.SpringMVCEntity;
import com.jaxio.celerio.model.support.validation.ValidationAttribute;
import com.jaxio.celerio.model.support.validation.ValidationRelation;
import com.jaxio.celerio.spi.AttributeSpi;
import com.jaxio.celerio.spi.EntitySpi;
import com.jaxio.celerio.spi.ProjectSpi;
import com.jaxio.celerio.spi.RelationSpi;
import com.jaxio.celerio.template.pack.PackLoader;
import com.jaxio.celerio.template.pack.TemplatePack;
import com.jaxio.celerio.template.pack.TemplatePackInfo;
import com.jaxio.celerio.util.FallBackUtil;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang.math.NumberUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.util.Assert;

import java.util.*;

import static com.google.common.collect.Lists.newArrayList;
import static com.jaxio.celerio.util.FallBackUtil.fallBack;
import static com.jaxio.celerio.util.StringUtil.getFirstCharacterUppered;

@Service
@Slf4j
public class ProjectFactory {
    public static final String DEFAULT_ENTITY_ROOTPACKAGE = "com.jaxio.demo";

    @Autowired
    private Config config;

    @Autowired
    private EntityFactory entityFactory;

    @Autowired
    private RelationFactory relationFactory;

    @Autowired
    private EntityConfigFactory entityConfigFactory;

    @Autowired
    private PrimaryKeyFactory primaryKeyFactory;

    @Autowired
    private UniqueFactory uniqueFactory;

    @Autowired
    private BusinessKeyFactory businessKeyFactory;

    @Autowired
    private ForeignKeyHintsFactory foreignKeyHintsFactory;

    @Autowired
    private InheritanceFactory inheritanceFactory;

    @Autowired
    private ConfigurationCheck configurationCheck;

    @Autowired
    private ForbiddenWhenBuildingAspect forbiddenWhenBuildingAspect;

    @Autowired
    private BuildInfo buildInfo;

    @Autowired
    private PackLoader packLoader;
    // -----------------------------------------------
    // SPI - Service Provider Interface
    // -----------------------------------------------
    private List> projectSpis = newArrayList();
    private List> entitySpis = newArrayList();
    private List> attributeSpis = newArrayList();
    private List> relationSpis = newArrayList();
    private List defaultProjectSpis = newArrayList();
    private List defaultEntitySpis = newArrayList( //
            new JpaEntity(), //
            new CustomEntity(), //
            new SearchEntity(), //
            new SpringMVCEntity() //
    );
    private List defaultAttributeSpis = newArrayList( //
            new CustomAttribute(), //
            new FormatterAttribute(), //
            new SearchAttribute(), //
            new ValidationAttribute(), //
            new SpringMVCAttribute(), //
            new JsfAttribute(), //
            new H2Attribute(), //
            new JQueryAttribute(), //
            new ValuesAttribute()
    );
    private List defaultRelationSpis = newArrayList( //
            new ValidationRelation(), //
            new JpaToOneRelation(), //
            new JpaOneToManyRelation(), //
            new JpaManyToManyRelation(), //
            new JpaIntermediateManyToOneRelation(), //
            new JpaIntermediateOneToManyRelation() //
    );

    public void init() {
        Assert.notNull(config.getOutputResult(), "Output result is not defined");

        Project project = config.getProject();
        Celerio celerio = config.getCelerio();

        if (config.isSpringfuseMode()) {
            handleSpringfuseSpecificities();
        }

        setCelerioVersion(celerio, buildInfo);
        updateNamingConventions(celerio.getConfiguration());

        celerio.setEntityConfigs(entityConfigFactory.filterEntityConfigs(celerio.getEntityConfigs()));

        log.info("Checking configuration consistency...");
        entityConfigFactory.assertEntityConfigListIsConsistent(celerio.getEntityConfigs());

        log.info("Processing enums...");
        buildSharedEnums(celerio);

        log.info("Processing entities...");
        buildEntities(project, celerio.getEntityConfigs());

        log.info("Processing inheritance...");
        inheritanceFactory.wireEntityHierarchies();

        log.info("Processing uniques/pk/fk...");
        setupEntities();

        forbiddenWhenBuildingAspect.buildingDone();

        log.info("Processing relations...");
        setupRelations();

        log.info("Processing business keys...");
        setupBusinessKeys();

        log.info("Processing global validation...");
        globalValidation();

        log.info("Applying conventions...");
        conventions(project, config.getCelerio().getConfiguration());

        log.info("Loading Celerio SPIs...");
        loadAndApplySpis();
    }

    /**
     * @Todo allow packs to say which module they require Update: not a good idea as pack may be simple folder on disk... => no
     */
    private void handleSpringfuseSpecificities() {
        if (config.isSpringfuseMode()) {
            // FOLDER FOR JAVA GENERATED CODE
            WellKnownFolderOverride javaFolder = new WellKnownFolderOverride();
            javaFolder.setWellKnownFolder(WellKnownFolder.JAVA);
            javaFolder.setGeneratedFolder("src/main/java");

            WellKnownFolderOverride javaTestFolder = new WellKnownFolderOverride();
            javaTestFolder.setWellKnownFolder(WellKnownFolder.JAVA_TEST);
            javaTestFolder.setGeneratedFolder("src/test/java");

            List wellKnownFolders = newArrayList(javaFolder, javaTestFolder);
            config.getCelerio().getConfiguration().getConventions().setWellKnownFolders(wellKnownFolders);

            // MISC
            config.getCelerio().getConfiguration().getGeneration().setUseMavenCelerioPlugin(false); // todo: a bit redundant!

            // OVERRIDE COMMENTS with default ones
            config.getCelerio().getConfiguration().setHeaderComment(new HeaderComment());
        }
    }

    private void setCelerioVersion(Celerio celerio, BuildInfo buildInfo) {
        celerio.getConfiguration().getGeneration().setVersion(buildInfo.getPomVersion());
    }

    private void buildEntities(Project project, List entityConfigs) {
        Set tablesProcessed = new HashSet();
        addEntitiesDefinedInUserConfiguration(project, entityConfigs, tablesProcessed);
        addRemainingEntities(project, tablesProcessed);
    }

    private void addRemainingEntities(Project project, Set tableProcessed) {
        for (Table table : config.getMetadata().getTables()) {
            // Make sure we do not process tables twice
            if (tableProcessed.contains(table.getName().toUpperCase())) {
                // skip default entity creation
                continue;
            }

            if (!config.getCelerio().getConfiguration().hasTable(table.getName())) {
                log.info("Skipping table '" + table.getName() + "'. Reason: excluded in configuration");
                continue;
            }

            EntityConfig entityConfig = entityConfigFactory.buildEntityConfig(table);

            // Skip conflict cases
            if (project.hasEntityByName(entityConfig.getEntityName())) {
                log.warn("Skip default entity creation for table name=" + table.getName() + " entity name=" + entityConfig.getEntityName()
                        + " as it was already registered!");
                continue;
            }

            Entity entity = entityFactory.buildEntity(entityConfig);
            log.info("Adding entity " + entity.getName());
            project.addEntity(entity);
            project.putEntityByTableName(entity);
            tableProcessed.add(table.getName().toUpperCase()); // not really needed but to be consistent...
        }
    }

    private void addEntitiesDefinedInUserConfiguration(Project project, List entityConfigs, Set tableProcessed) {
        for (EntityConfig entityConfig : entityConfigs) {
            if (entityConfig.hasTableName() && !config.getCelerio().getConfiguration().hasTable(entityConfig.getTableName())) {
                log.info("Skipping table '" + entityConfig.getTableName() + "'. Reason: excluded in configuration");
                continue;
            }

            entityConfigFactory.applyFallBacks(entityConfig);
            Entity entity = entityFactory.buildEntity(entityConfig);
            log.info("Adding entity " + entity.getName());
            project.addEntity(entity);

            if (!entity.hasInheritance()) {
                project.putEntityByTableName(entity);
            }

            // note: we may add a table twice, this is OK, this code is just here to
            // avoid processing tables not present in the conf
            // please check addRemainingEntities
            tableProcessed.add(entity.getTableName().toUpperCase());
        }
    }

    private void setupEntities() {
        for (Entity entity : config.getProject().getCurrentEntities()) {
            entityFactory.setupAttributes(entity);
        }

        for (Entity entity : config.getProject().getCurrentEntities()) {
            uniqueFactory.setupUniques(entity);
        }

        for (Entity entity : config.getProject().getCurrentEntities()) {
            primaryKeyFactory.setupPrimaryKey(entity);
        }

        // Attention: FK hints must be processed after PK
        for (Entity entity : config.getProject().getCurrentEntities()) {
            foreignKeyHintsFactory.setupForeignKeyHints(entity);
        }
    }

    private void setupRelations() {
        for (Entity entity : config.getProject().getCurrentEntities()) {
            relationFactory.setupRelations(entity);
        }
    }

    private void setupBusinessKeys() {
        for (Entity entity : config.getProject().getCurrentEntities()) {
            businessKeyFactory.setupBusinessKey(entity);
        }
    }

    private void globalValidation() {
        if (!configurationCheck.check(config)) {
            throw new IllegalStateException("There are some errors, generation aborted.");
        }
    }

    private void updateNamingConventions(Configuration configuration) {
        configuration.setRootPackage(fallBack(configuration.getRootPackage(), DEFAULT_ENTITY_ROOTPACKAGE));
        if ("".equals(configuration.getRootPackage())) {
            configuration.setRootPackage(DEFAULT_ENTITY_ROOTPACKAGE);
        }

        Conventions conventions = configuration.getConventions();
        updateClassTypeConventions(conventions);
        updateMethodConventions(conventions);
        updateWellKnownFolderConventions(conventions);
        updateGeneratedPackageConventions(configuration, conventions);
    }

    private void updateWellKnownFolderConventions(Conventions conventions) {
        for (WellKnownFolderOverride convention : conventions.getWellKnownFolders()) {
            convention.apply();
        }

        if (!config.getOutputResult().sameDirectory()) {
            for (WellKnownFolder wkf : WellKnownFolder.values()) {
                wkf.setGeneratedFolder(wkf.getFolder());
            }
        }
    }

    private void updateMethodConventions(Conventions conventions) {
        for (MethodConventionOverride convention : conventions.getMethodConventions()) {
            convention.apply();
        }
    }

    private void updateClassTypeConventions(Conventions conventions) {
        for (ClassTypeOverride convention : conventions.getClassTypes()) {
            convention.apply();
        }
    }

    private void updateGeneratedPackageConventions(Configuration configuration, Conventions conventions) {
        // set the values from the config
        for (GeneratedPackageOverride convention : conventions.getGeneratedPackages()) {
            convention.apply();
        }
        // set default values for the root package
        for (GeneratedPackage generatedPackage : GeneratedPackage.values()) {
            if (generatedPackage.getRootPackage() == null) {
                generatedPackage.setRootPackage(configuration.getRootPackage());
            }
        }
    }

    private void conventions(Project project, Configuration configuration) {
        AccountConvention accountConvention = new AccountConvention();

        // try to detect an account entity
        if (!project.isAccountEntityPresent()) {
            for (Entity entity : project.getCurrentEntities()) {
                if (accountConvention.setupAccount(entity)) {
                    return;
                }
            }
        }
    }

    private void buildSharedEnums(Celerio celerio) {
        for (EnumConfig sharedEnum : celerio.getSharedEnumConfigs()) {
            sharedEnum.setRootPackage(FallBackUtil.fallBack(sharedEnum.getRootPackage(), config.getCelerio().getConfiguration().getRootPackage()));
            sharedEnum.setType(fallBack(sharedEnum.getType(), EnumType.ORDINAL));
            sharedEnum.setName(getFirstCharacterUppered(sharedEnum.getName()));

            for (EnumValue ev : sharedEnum.getEnumValues()) {
                Assert.isTrue(ev.hasValue(), "The 'value' attribute is mandatory for the sharedEnumConfig " + sharedEnum.getName());
                ev.setName(fallBack(ev.getName(), ev.getValue()));

                if (!sharedEnum.isCustomType()) {
                    ev.setValue(null);
                }

                // fix for evdev
                if (NumberUtils.isNumber(ev.getName().substring(0, 1))) {
                    ev.setName("TODO_" + ev.getName());
                    log.warn("Please review your enum " + sharedEnum.getName() + " configuration. Some constant are numeric!");
                }
            }
        }
    }

    private void loadAndApplySpis() {
        // 1- project SPI
        loadProjectSpis(defaultProjectSpis.iterator());
        loadProjectSpis(ServiceLoader.load(ProjectSpi.class).iterator());

        // 2- entity SPI
        loadEntitySpis(defaultEntitySpis.iterator());
        loadEntitySpis(ServiceLoader.load(EntitySpi.class).iterator());

        // 3- attribute SPI
        loadAttributeSpis(defaultAttributeSpis.iterator());
        loadAttributeSpis(ServiceLoader.load(AttributeSpi.class).iterator());

        // 4- relation SPI
        loadRelationSpis(defaultRelationSpis.iterator());
        loadRelationSpis(ServiceLoader.load(RelationSpi.class).iterator());

        // 5- real binding on project, entities, attributes, relations.
        bindAllSpis();

        // 6- now bind all namers found in various pack config
        // Note: namers found in main config are loaded/binded by the entityFactory.
        for (TemplatePack templatePack : packLoader.getTemplatePacks()) {

            List entityContextPropertyList = templatePack.getTemplatePackInfo().getEntityContextPropertyList();
            if (entityContextPropertyList == null || entityContextPropertyList.size() == 0) {
                continue;
            }

            for (Entity entity : config.getProject().getCurrentEntities()) {
                for (EntityContextProperty ecp : entityContextPropertyList) {
                    entity.put(ecp.getProperty(), new ClassNamer2(entity, ecp.getRootPackage(), ecp.getSubPackage(), ecp.getPrefix(), ecp.getSuffix()));
                }
            }
        }
    }

    private void loadProjectSpis(Iterator iterator) {
        while (iterator.hasNext()) {
            ProjectSpi spi = iterator.next();
            log.info("Load ProjectSpi: " + spi.velocityVar() + " => " + spi.getClass().getName());
            projectSpis.add(spi.getClass());
        }
    }

    private void loadEntitySpis(Iterator iterator) {
        while (iterator.hasNext()) {
            EntitySpi spi = iterator.next();
            log.info("Load EntitySpi: " + spi.velocityVar() + " => " + spi.getClass().getName());
            entitySpis.add(spi.getClass());
        }
    }

    private void loadAttributeSpis(Iterator iterator) {
        while (iterator.hasNext()) {
            AttributeSpi spi = iterator.next();
            log.info("Load AttributeSpi: " + spi.velocityVar() + " => " + spi.getClass().getName());
            attributeSpis.add(spi.getClass());
        }
    }

    private void loadRelationSpis(Iterator iterator) {
        while (iterator.hasNext()) {
            RelationSpi spi = iterator.next();
            log.info("Load RelationSpi: " + spi.velocityVar() + " => " + spi.getClass().getName());
            relationSpis.add(spi.getClass());
        }
    }

    private void bindAllSpis() {
        for (Class projectSpiClass : projectSpis) {
            bindProjectSpi(config.getProject(), projectSpiClass);
        }

        for (Entity entity : config.getProject().getCurrentEntities()) {
            for (Class entitySpiClass : entitySpis) {
                bindEntitySpi(entity, entitySpiClass);
            }

            for (Attribute attribute : entity.getCurrentAttributes()) {
                for (Class attributeSpiClass : attributeSpis) {
                    bindAttributeSpi(attribute, attributeSpiClass);
                }
            }

            for (Relation relation : entity.getRelations().getList()) {
                for (Class relationSpiClass : relationSpis) {
                    bindRelationSpi(relation, relationSpiClass);
                }
            }
        }
    }

    private void bindProjectSpi(Project project, Class spiClass) {
        try {
            ProjectSpi spi = spiClass.newInstance();
            spi.init(project);
            project.put(spi.velocityVar(), spi.getTarget());
        } catch (Exception e) {
            log.error("Could not instantiate ProjectSpi", e);
        }
    }

    private void bindEntitySpi(Entity entity, Class spiClass) {
        try {
            EntitySpi spi = spiClass.newInstance();
            spi.init(entity);
            entity.put(spi.velocityVar(), spi.getTarget());
        } catch (Exception e) {
            log.error("Could not instantiate EntitySpi", e);
        }
    }

    private void bindAttributeSpi(Attribute attribute, Class spiClass) {
        try {
            AttributeSpi spi = spiClass.newInstance();
            spi.init(attribute);
            attribute.put(spi.velocityVar(), spi.getTarget());
        } catch (Exception e) {
            log.error("Could not instantiate AttributeSpi", e);
        }
    }

    private void bindRelationSpi(Relation relation, Class spiClass) {
        try {
            RelationSpi spi = spiClass.newInstance();
            if (spi.compatibleWith(relation)) {
                spi.init(relation);
                ((AbstractRelation) relation).put(spi.velocityVar(), spi.getTarget());
            }
        } catch (Exception e) {
            log.error("Could not instantiate RelationSpi", e);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy