org.nuiton.topia.templates.TopiaTemplateHelper Maven / Gradle / Ivy
package org.nuiton.topia.templates;
/*
* #%L
* Toolkit :: Templates
* %%
* Copyright (C) 2017 - 2024 Ultreia.io
* %%
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public
* License along with this program. If not, see
* .
* #L%
*/
import com.google.common.base.Preconditions;
import fr.ird.observe.toolkit.templates.entity.query.SqlQueryDefinition;
import fr.ird.observe.toolkit.templates.entity.query.SqlQueryDefinitions;
import org.apache.commons.collections4.CollectionUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.tuple.Pair;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.nuiton.eugene.AbstractGenerator;
import org.nuiton.eugene.EugeneCoreTagValues;
import org.nuiton.eugene.GeneratorUtil;
import org.nuiton.eugene.java.JavaGeneratorUtil;
import org.nuiton.eugene.java.ObjectModelTransformerToJava;
import org.nuiton.eugene.models.extension.tagvalue.TagValueUtil;
import org.nuiton.eugene.models.object.ObjectModel;
import org.nuiton.eugene.models.object.ObjectModelAssociationClass;
import org.nuiton.eugene.models.object.ObjectModelAttribute;
import org.nuiton.eugene.models.object.ObjectModelClass;
import org.nuiton.eugene.models.object.ObjectModelClassifier;
import org.nuiton.eugene.models.object.ObjectModelElement;
import org.nuiton.eugene.models.object.ObjectModelInterface;
import org.nuiton.eugene.models.object.ObjectModelOperation;
import org.nuiton.eugene.models.object.ObjectModelPackage;
import org.nuiton.eugene.models.object.ObjectModelParameter;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.TreeMap;
/**
* Classe regroupant divers méthodes utiles pour la génération des entités.
*
* Remplace {@code TopiageneratorUtil} car on ne veut plus utiliser de méthodes
* statiques (par exemple pour avoir les valeurs par défaut des tagValues,...).
*
* Created on 7/6/14.
*
* @author Tony Chemit - [email protected]
* @since 3.0
*/
public class TopiaTemplateHelper {
private static final Logger log = LogManager.getLogger(TopiaTemplateHelper.class);
/**
* Type de persistence Hibernate
*/
public final String PERSISTENCE_TYPE_HIBERNATE = "hibernate";
/**
* Type de persistence par défaut (si aucun précisé)
*/
public final String PERSISTENCE_TYPE_DEFAULT = PERSISTENCE_TYPE_HIBERNATE;
/**
* Propriété des générateurs indiquant le package par défaut
*/
public final String PROPERTY_DEFAULT_PACKAGE = "defaultPackage";
/**
* Le package par défaut si aucun n'est spécifié
*/
public final String DEFAULT_PACKAGE = "org.codelutin.malo";
/**
* Stratégie d'heritage par defaut.
*/
public final String DEFAULT_INHERITANCE_STRATEGY = "union-subclass";
private final Map indexes;
private final Map uniqueKeys;
private final Map foreignKeys;
private final TopiaExtensionTagValues extensionTagValues = new TopiaExtensionTagValues();
protected final ObjectModel model;
protected final EugeneCoreTagValues eugeneTagValues;
protected final TopiaCoreTagValues topiaCoreTagValues;
protected final TopiaHibernateTagValues topiaHibernateTagValues;
public TopiaTemplateHelper(ObjectModel model) {
this.model = model;
this.eugeneTagValues = new EugeneCoreTagValues();
this.topiaCoreTagValues = new TopiaCoreTagValues();
this.topiaHibernateTagValues = new TopiaHibernateTagValues();
indexes = new TreeMap<>();
foreignKeys = new TreeMap<>();
uniqueKeys = new TreeMap<>();
}
public Set queriesForType(String qualifiedName, String schemaName, String tableName) {
Set result = new LinkedHashSet<>(SqlQueryDefinitions.load(qualifiedName).map(SqlQueryDefinitions::getQueries).orElse(Set.of()));
result.add(new SqlQueryDefinition(qualifiedName,
"GetLastUpdateDate",
String.format("SELECT max(lastUpdateDate) FROM %s.%s;", schemaName, tableName),
"Get last update date for the given entity type",
false,
Map.of()));
return result;
}
@Deprecated
public TopiaCoreTagValues getTopiaCoreTagValues() {
return topiaCoreTagValues;
}
@Deprecated
public TopiaHibernateTagValues getTopiaHibernateTagValues() {
return topiaHibernateTagValues;
}
public TopiaExtensionTagValues extensionTagValues() {
return extensionTagValues;
}
/**
* Renvoie le type de persistence pour le classifier donné. Si aucun n'est
* trouvé, le type par défaut est utilisé
*
* @param classifier l'élément à tester
* @return le type de persitence pour l'élément donné.
* @since 2.5
*/
public String getPersistenceType(ObjectModelClassifier classifier) {
String tag = topiaHibernateTagValues.getPersistenceTypeTagValue(classifier);
//TODO Set default value in tag metadata
if (tag == null) {
tag = PERSISTENCE_TYPE_DEFAULT;
}
return tag;
}
/**
* Renvoie le package par défaut pour le générateur donné
*
* @param generator le générateur donné
* @return le package par défaut du générator donné
*/
public String getDefaultPackage(AbstractGenerator> generator) {
String packageName = generator.getProperty(PROPERTY_DEFAULT_PACKAGE);
if (StringUtils.isBlank(packageName)) {
packageName = DEFAULT_PACKAGE;
}
return packageName;
}
public String getApplicationContextPackage(ObjectModelTransformerToJava transformer,
ObjectModel model) {
return transformer.getDefaultPackageName();
}
public String getPersistenceContextPackage(ObjectModelTransformerToJava transformer,
ObjectModel model) {
return transformer.getDefaultPackageName();
}
public String getDaoPackage(ObjectModelTransformerToJava transformer, ObjectModel model) {
return transformer.getDefaultPackageName();
}
public String getParentDaoName(ObjectModel model) {
return "Abstract" + model.getName() + "TopiaDao";
}
public String getParentDaoFqn(ObjectModelTransformerToJava transformer, ObjectModel model) {
return getDaoPackage(transformer, model) + "." + getParentDaoName(model);
}
public String getApplicationContextInterfaceName(ObjectModel model) {
return model.getName() + "ApplicationContext";
}
public String getApplicationContextAbstractName(ObjectModel model) {
return "Abstract" + model.getName() + "TopiaApplicationContext";
}
public String getApplicationContextConcreteName(ObjectModel model) {
return model.getName() + "TopiaApplicationContext";
}
public String getPersistenceContextAbstractName(ObjectModel model) {
return "Abstract" + model.getName() + "TopiaPersistenceContext";
}
public String getPersistenceContextConcreteName(ObjectModel model) {
return model.getName() + "TopiaPersistenceContext";
}
public String getPersistenceContextInterfaceName(ObjectModel model) {
return model.getName() + "PersistenceContext";
}
public String getDaoSupplierInterfaceName(ObjectModel model) {
return model.getName() + "DaoSupplier";
}
public String getDaoSupplierName(ObjectModel model) {
return model.getName() + "TopiaDaoSupplier";
}
public String getEntityEnumName(ObjectModel model) {
return model.getName() + "EntityEnum";
}
public String getEntityAbstractName(ObjectModelClass input) {
return input.getName() + "Abstract";
}
public String getEntityConcreteName(ObjectModelClass input) {
return input.getName();
}
public String getAbstractDaoName(ObjectModelClass input) {
return "Abstract" + input.getName() + "TopiaDao";
}
public String getGeneratedDaoName(ObjectModelClass input) {
return "Generated" + input.getName() + "TopiaDao";
}
public String getConcreteDaoName(ObjectModelClass input) {
return input.getName() + "TopiaDao";
}
public String getContractDaoName(ObjectModelClass input) {
return input.getName() + "Dao";
}
public String getAbstractDaoFqn(ObjectModelClass input) {
return input.getPackageName() + "." + getAbstractDaoName(input);
}
public String getGeneratedDaoFqn(ObjectModelClass input) {
return input.getPackageName() + "." + getGeneratedDaoName(input);
}
public String getConcreteDaoFqn(ObjectModelClass input) {
return input.getPackageName() + "." + getConcreteDaoName(input);
}
public String getEntityPackage(ObjectModelTransformerToJava transformer,
ObjectModel model,
ObjectModelClassifier input) {
return input.getPackageName();
}
/**
* Obtain the reverse db name of an attribute.
*
* Try first to get the reverse db Name from the reverseDbname tag-value, then
* if attribute has a specific reverse attribute, use his db name, otherwise
* suffix the db name of the attribute by {@code _id}.
*
* @param attr the attribute to seek
* @return the value of the reverse name
* @since 2.5
*/
public String getReverseDbName(ObjectModelAttribute attr) {
String result = TagValueUtil.findDirectTagValue(TopiaHibernateTagValues.Store.reverseDbName, attr);
if (StringUtils.isEmpty(result)) {
if (attr.getReverseAttribute() != null) {
result = getDbName(attr.getReverseAttribute());
} else {
result = getDbName(attr) + "_id";
}
}
return result;
}
/**
* Renvoie le nom BD de l'élement passé en paramètre. Elle se base sur le
* tag associé si il existe, sinon sur le nom de l'élément
*
* @param element l'élément à tester
* @return le nom de table
*/
public String getDbName(ObjectModelElement element) {
if (element == null) {
return null;
}
String value = topiaHibernateTagValues.getDbNameTagValue(element);
if (value != null) {
return value;
}
return GeneratorUtil.toLowerCaseFirstLetter(element.getName());
}
/**
* Obtain the reverse db name of a reverse attribute.
*
* Note that the reverse attribute can't be null here.
*
* - Try first to get the reverse db Name from the ReverseDbname tag-value
* - If not found, try then the ReverseDbname tag-value on the same attribute but from this other side of the relation
* - If not found, try then just get the name of the reverse attribute
*
*
* @param attr the attribute to seek
* @return the value of the reverse db name on the reverse attribute
* @since 2.9.5.2
*/
public String getReverseDbNameOnReverseAttribute(ObjectModelAttribute attr) {
ObjectModelAttribute reverseAttribute = attr.getReverseAttribute();
if (reverseAttribute == null) {
throw new IllegalArgumentException("The reverse attribute can't be null, but was on " + attr);
}
if (!Objects.equals(reverseAttribute.getReverseAttribute(), attr)) {
//FIXME Bug in eugene the reverse attribute may not be the good one
return getDbName(attr);
}
if (reverseAttribute == null) {
throw new IllegalArgumentException("The reverse attribute can't be null, but was on " + attr);
}
String result = getReverseDbName(reverseAttribute);
if (StringUtils.isEmpty(result)) {
// Try to get it from the other site of the relation
ObjectModelAttribute reverseAttribute2 = reverseAttribute.getClassifier().getAttribute(attr.getName());
result = getReverseDbName(reverseAttribute2);
}
if (StringUtils.isEmpty(result)) {
result = GeneratorUtil.toLowerCaseFirstLetter(reverseAttribute.getName());
}
return result;
}
/**
* Cherche et renvoie la liste des attributs constituant la clef metier
* d'une classe.
*
* @param clazz la classe à tester
* @return la liste des attributs de la clef métier
*/
public Set getNaturalIdAttributes(
ObjectModelClass clazz) {
// use {@link LinkedHashSet} to keep order and prevent duplicate natural ids found
Set results = new LinkedHashSet<>();
for (ObjectModelAttribute attr : clazz.getAttributes()) {
if (topiaHibernateTagValues.getNaturalIdTagValue(attr)) {
results.add(attr);
}
}
// sletellier : #2050 Natural id and not null attributes are not propagated on generalized entities (http://nuiton.org/issues/2050)
Collection superclasses = clazz.getSuperclasses();
for (ObjectModelClass superClass : superclasses) {
Set naturalIdsOfSuperClass = getNaturalIdAttributes(superClass);
results.addAll(naturalIdsOfSuperClass);
}
return results;
}
/**
* Cherche et renvoie la liste des attributs qui ne doivent pas etre null dans
* une classe.
*
* @param clazz la classe à tester
* @return la liste des attributs qui ne doivent pas etre null
*/
public Set getNotNullAttributes(
ObjectModelClass clazz) {
// use {@link LinkedHashSet} to keep order and prevent duplicate not null found
Set results =
new LinkedHashSet<>();
for (ObjectModelAttribute attr : clazz.getAttributes()) {
if (isAttributeNotNull(attr)) {
results.add(attr);
}
}
Collection superclasses = clazz.getSuperclasses();
for (ObjectModelClass superClass : superclasses) {
Set notNullOfSuperClass = getNotNullAttributes(superClass);
results.addAll(notNullOfSuperClass);
}
// Association class participants are obviously not null
if (clazz instanceof ObjectModelAssociationClass) {
List participantsAttributes = ((ObjectModelAssociationClass) clazz).getParticipantsAttributes();
results.addAll(participantsAttributes);
}
return results;
}
/**
* Detecte si un attribut est marqué comme non null.
* Les naturalId sont not null par défaut
*
* @param attribute l'attribut à tester
* @return {@code true} si l'attribut doit être non null,
* par défaut pour les naturalId, {@code false} sinon..
* @since 2.6.9
*/
public boolean isAttributeNotNull(ObjectModelAttribute attribute) {
Boolean value = topiaHibernateTagValues.getNotNullTagValue(attribute);
if (value == null) {
// valeur null, donc pas positionnee
return topiaHibernateTagValues.getNaturalIdTagValue(attribute);
}
return value;
}
public String getDOType(ObjectModelElement elem, ObjectModel model) {
String type = elem.getName();
if (elem instanceof ObjectModelAttribute) {
type = ((ObjectModelAttribute) elem).getType();
}
if (elem instanceof ObjectModelClass) {
type = ((ObjectModelClass) elem).getQualifiedName();
}
return getDOType(type, model);
}
public String getDOType(String type, ObjectModel model) {
if (!model.hasClass(type)) {
return type;
}
ObjectModelClass clazz = model.getClass(type);
if (isEntity(clazz)) {
//tchemit-2011-09-12 What ever abstract or not, we always use an Impl
type += "Impl";
}
return type;
}
private static final Set numberTypes = new HashSet<>();
private static final Set textTypes = new HashSet<>();
private static final Set booleanTypes = new HashSet<>();
private static final Set primitiveTypes = new HashSet<>();
private static final Map primitiveTypeToClass = new HashMap<>();
private static final String VOID_TYPE = "void";
static {
primitiveTypeToClass.put("byte", "java.lang.Byte");
primitiveTypeToClass.put("short", "java.lang.Short");
primitiveTypeToClass.put("int", "java.lang.Integer");
primitiveTypeToClass.put("long", "java.lang.Long");
primitiveTypeToClass.put("float", "java.lang.Float");
primitiveTypeToClass.put("double", "java.lang.Double");
primitiveTypeToClass.put("char", "java.lang.Char");
primitiveTypeToClass.put("boolean", "java.lang.Boolean");
numberTypes.add("byte");
numberTypes.add("java.lang.Byte");
numberTypes.add("Byte");
numberTypes.add("short");
numberTypes.add("java.lang.Short");
numberTypes.add("Short");
numberTypes.add("int");
numberTypes.add("java.lang.Integer");
numberTypes.add("Integer");
numberTypes.add("long");
numberTypes.add("java.lang.Long");
numberTypes.add("Long");
numberTypes.add("float");
numberTypes.add("java.lang.Float");
numberTypes.add("Float");
numberTypes.add("double");
numberTypes.add("java.lang.Double");
numberTypes.add("Double");
textTypes.add("char");
textTypes.add("java.lang.Char");
textTypes.add("Char");
textTypes.add("java.lang.String");
textTypes.add("String");
booleanTypes.add("boolean");
booleanTypes.add("java.lang.Boolean");
booleanTypes.add("Boolean");
primitiveTypes.addAll(numberTypes);
primitiveTypes.addAll(textTypes);
primitiveTypes.addAll(booleanTypes);
}
public boolean isNumericType(ObjectModelAttribute attr) {
return numberTypes.contains(attr.getType());
}
public boolean isTextType(ObjectModelAttribute attr) {
return textTypes.contains(attr.getType());
}
public boolean isDateType(ObjectModelAttribute attr) {
return "java.util.Date".equals(attr.getType());
}
public boolean isBooleanType(ObjectModelAttribute attr) {
return booleanTypes.contains(attr.getType());
}
public boolean isPrimitiveType(ObjectModelAttribute attr) {
return primitiveTypes.contains(attr.getType());
}
public String getClassForPrimitiveType(ObjectModelAttribute attr) {
Preconditions.checkState(isPrimitiveType(attr));
String className = primitiveTypeToClass.get(attr.getType());
Preconditions.checkNotNull(className);
return className;
}
/**
*
* Cette méthode permet de détecter si
* - l'attribut représente une relation 1-n
* - cette relation est unidirectionnelle
* - le type de l'attribut représente un entité
* - cette entité a des sous-classes dans le modèle
*
* Ce cas correspond à une incompatibilité d'Hibernate qui nous oblige a
* adopter un comportement particulier.
*
*
* @param attr l'attribut a tester
* @param model le model
* @return true si et seulement si il s'agit bien de ce type de relation
*/
public boolean hasUnidirectionalRelationOnAbstractType(
ObjectModelAttribute attr,
ObjectModel model) {
ObjectModelAttribute reverse = attr.getReverseAttribute();
//relation 1-n
if (reverse != null && GeneratorUtil.isNMultiplicity(attr) &&
!GeneratorUtil.isNMultiplicity(reverse)) {
//Pas de navigabilité
if (!reverse.isNavigable()) {
//Il s'agit d'une entity
ObjectModelClass clazz = model.getClass(attr.getType());
if (clazz != null && isEntity(clazz)) {
//Cette classe a des sous-classes dans le modèle
for (ObjectModelClass subClass : model.getClasses()) {
if (subClass.getSuperclasses().contains(clazz)) {
return true;
}
}
}
}
}
return false;
}
/**
* Renvoie le nom unique de table pour une relation ManyToMany en fonction
* de l'attribut {@code attr}
*
* @param attr l'attribut servant de base au calcul du nom
* @return le nom de la table
*/
public String getManyToManyTableName(ObjectModelAttribute attr) {
String result;
if (attr.hasAssociationClass()) {
result = getDbName(attr.getAssociationClass());
} else {
result = topiaHibernateTagValues.getManytoManyTableNameTagValue(attr);
if (StringUtils.isEmpty(result)) {
String name = attr.getName();
String revers = attr.getReverseAttributeName();
// FIXME echatellier 20170414 in case of attribute with * multiplicity
// name is wrong. Should always be "parenttablename_attributename" and name sort compare
if (name.compareToIgnoreCase(revers) < 0) {
result = name + '_' + revers;
} else {
result = revers + '_' + name;
}
}
}
return result;
}
/**
* Renvoie le type d'interface à utiliser en fonction de l'attribut
*
* @param attr l'attribut a traiter
* @return String
*/
public String getNMultiplicityHibernateType(
ObjectModelAttribute attr) {
if (JavaGeneratorUtil.isOrdered(attr)) {
return "list";
} else if (EugeneCoreTagValues.isUnique(attr)) {
return "set";
}
//attr.isOrdered() - On génère le ordered en bag
return "bag";
}
/**
* Obtain the list of entities classes with the possibility to sort the
* result.
*
* @param model the current model to scan
* @param sort flag to allow sort the result
* @return the list of filtred classes by their stereotype
*/
public List getEntityClasses(ObjectModel model,
boolean sort) {
List classes = new ArrayList<>();
for (ObjectModelClass clazz : model.getClasses()) {
if (isEntity(clazz)) {
classes.add(clazz);
}
}
if (sort && !classes.isEmpty()) {
Collections.sort(classes, OBJECT_MODEL_CLASS_COMPARATOR);
}
return classes;
}
public final Comparator
OBJECT_MODEL_CLASS_COMPARATOR =
new Comparator() {
@Override
public int compare(ObjectModelClass o1,
ObjectModelClass o2) {
return o1.getQualifiedName().compareTo(
o2.getQualifiedName());
}
};
/**
* Obtain the list of fqn of object involed in the given class.
*
* @param aClass the clazz to inspect
* @param incomingFqns incoming fqns
* @return the list of fqn of attributes
*/
public List getImports(ObjectModelClass aClass,
String... incomingFqns) {
Set tmp = new HashSet<>();
tmp.addAll(Arrays.asList(incomingFqns));
getImports(aClass, tmp);
return cleanImports(aClass.getPackageName(), tmp);
}
/**
* Obtain the list of fqn of object involed in the given class.
*
* @param aClass the class to inspect
* @param fqns where to store found fqns
*/
public void getImports(ObjectModelClass aClass,
Set fqns) {
// scan attributes
for (ObjectModelAttribute attr : aClass.getAttributes()) {
fqns.add(attr.getType());
if (GeneratorUtil.isNMultiplicity(attr)) {
String collectionType = getCollectionType(attr).getName();
fqns.add(collectionType);
String collectionObject = getCollectionInstanceType(attr).getName();
fqns.add(collectionObject);
}
}
for (ObjectModelAttribute attribute : aClass.getAllOtherAttributes()) {
fqns.add(attribute.getType());
}
// scan associations
if (aClass instanceof ObjectModelAssociationClass) {
ObjectModelAssociationClass assoc =
(ObjectModelAssociationClass) aClass;
for (ObjectModelAttribute attr :
assoc.getParticipantsAttributes()) {
if (attr == null) {
continue;
}
fqns.add(attr.getType());
if (GeneratorUtil.isNMultiplicity(attr)) {
String collectionType = getCollectionType(attr).getName();
fqns.add(collectionType);
String collectionObject = getCollectionInstanceType(attr).getName();
fqns.add(collectionObject);
}
}
}
// scan operations
for (ObjectModelOperation operation : aClass.getOperations()) {
getImports(operation, fqns);
}
// scan super interfaces
for (ObjectModelInterface modelInterface : aClass.getInterfaces()) {
fqns.add(modelInterface.getQualifiedName());
getImports(modelInterface, fqns);
}
// scan super classes
for (ObjectModelClass modelClass : aClass.getSuperclasses()) {
fqns.add(modelClass.getQualifiedName());
getImports(modelClass);
}
}
/**
* Obtain the list of fqn of object involed in the given interface.
*
* @param anInterface the interface to inspect
* @param fqns where to store found fqns
*/
public void getImports(ObjectModelInterface anInterface,
Set fqns) {
// scan operations
for (ObjectModelOperation operation : anInterface.getOperations()) {
getImports(operation, fqns);
}
// scan super interfaces
for (ObjectModelInterface modelInterface : anInterface.getInterfaces()) {
fqns.add(modelInterface.getQualifiedName());
getImports(modelInterface, fqns);
}
}
/**
* Obtain the fqn's list of all involed type in a givne operation.
*
* @param operation operation to inspect
* @param fqns where to store found fqns
*/
public void getImports(ObjectModelOperation operation,
Set fqns) {
String fqn = operation.getReturnType();
fqns.add(fqn);
for (ObjectModelParameter parameter : operation.getParameters()) {
fqns.add(parameter.getType());
}
}
/**
* Clean a set of fqns, transform it into a {@link List} and sort it.
*
* @param packageName the current package name
* @param fqns the dirty set of fqns
* @return the sorted cleaned list of fqns.
*/
public List cleanImports(String packageName,
Set fqns) {
fqns.removeAll(primitiveTypes);
fqns.remove(VOID_TYPE);
int packageLength = packageName.length();
List genericType = new ArrayList<>();
for (Iterator it = fqns.iterator(); it.hasNext(); ) {
String fqn = it.next();
int lastIndex = fqn.lastIndexOf(".");
if (lastIndex == packageLength && fqn.startsWith(packageName)) {
// same package
it.remove();
continue;
}
int genericIndex = fqn.indexOf('<');
if (genericIndex != -1) {
genericType.add(fqn.substring(0, genericIndex));
it.remove();
}
}
fqns.addAll(genericType);
ArrayList result = new ArrayList<>(fqns);
Collections.sort(result);
return result;
}
public Map>
searchDirectUsages(ObjectModel model) {
List allEntities;
Map allEntitiesByFQN;
Map> usages;
allEntities = getEntityClasses(model, true);
allEntitiesByFQN = new TreeMap<>();
usages = new LinkedHashMap<>();
// prepare usages map and fill allEntitiesByFQN map
for (ObjectModelClass klass : allEntities) {
usages.put(klass, new HashSet());
allEntitiesByFQN.put(klass.getQualifiedName(), klass);
}
// first pass to detect direct usages
for (ObjectModelClass klass : allEntities) {
searchDirectUsages(klass, allEntitiesByFQN, usages);
}
allEntities.clear();
allEntitiesByFQN.clear();
return usages;
}
public void searchDirectUsages(
ObjectModelClass klass,
Map allEntitiesByFQN,
Map> usages) {
if (log.isDebugEnabled()) {
log.debug("for entity " + klass.getQualifiedName());
}
for (ObjectModelAttribute attr : klass.getAttributes()) {
if (!attr.isNavigable()) {
// skip this case
continue;
}
String type;
if (attr.hasAssociationClass()) {
type = attr.getAssociationClass().getQualifiedName();
} else {
type = attr.getType();
}
if (!allEntitiesByFQN.containsKey(type)) {
// not a entity, can skip for this attribute
continue;
}
if (log.isDebugEnabled()) {
log.debug(" uses " + type);
}
// register the klass as using the targetEntity
ObjectModelClass targetEntity = allEntitiesByFQN.get(type);
Set classes = usages.get(targetEntity);
classes.add(klass);
}
}
public boolean isImportNeeded(Collection operations,
String importName) {
if (CollectionUtils.isNotEmpty(operations)) {
for (ObjectModelOperation op : operations) {
if (op.getReturnType().contains(importName)) {
return true;
}
for (ObjectModelParameter param : op.getParameters()) {
if (param.getType().contains(importName)) {
return true;
}
}
}
}
return false;
}
public boolean isCollectionNeeded(
Collection operations) {
return isImportNeeded(operations,
Collection.class.getSimpleName());
}
public boolean isSetNeeded(Collection operations) {
return isImportNeeded(operations,
Set.class.getSimpleName());
}
/**
* Check if the given attribute type is an entity.
*
* @param attribute attribute to test
* @param model model containing the attribute
* @return {@code true} if type of attribute is an entity,
* {@code false} otherwise
* @see TopiaCoreTagValues.Store#entity
* @since 2.7
*/
public boolean isEntity(ObjectModelAttribute attribute,
ObjectModel model) {
if (isPrimitiveType(attribute)) {
return false;
}
String attributeType = attribute.getType();
ObjectModelClassifier typeclassifier =
model.getClassifier(attributeType);
return typeclassifier != null && isEntity(typeclassifier);
}
/**
* Check if the given classifier has the
* {@link TopiaCoreTagValues.Store#entity} and is not an enumeration
*
* @param classifier classifier to test
* @return {@code true} if stereotype was found and classifier is not
* enumeration, {@code false} otherwise
* @see TopiaCoreTagValues.Store#entity
* @since 2.5
*/
public boolean isEntity(ObjectModelClassifier classifier) {
ObjectModelPackage aPackage = model.getPackage(classifier);
return !classifier.isEnum() && topiaCoreTagValues.isEntity(classifier, aPackage);
}
/**
* Check if the given classifier is an abstract class
*
* @param classifier classifier to test
* @return {@code true} if the classifier is abstract, {@code false} otherwise
*/
public boolean isAbstract(ObjectModelClassifier classifier) {
ObjectModelClass aClass = model.getClass(classifier.getQualifiedName());
return aClass.isAbstract();
}
/**
* Obtain the value of the {@link TopiaHibernateTagValues.Store#inheritanceStrategy} tag value on the given classifier.
*
* @param classifier classifier to seek
* @return the none empty value of the found tag value or {@code null} if not found nor empty.
* @see TopiaHibernateTagValues.Store#inheritanceStrategy
* @since 3.0
*/
public String getInheritanceStrategy(ObjectModelClassifier classifier) {
ObjectModelPackage aPackage = model.getPackage(classifier);
String value = topiaHibernateTagValues.getInheritanceStrategyTagValue(classifier,
aPackage);
if (value == null) {
value = DEFAULT_INHERITANCE_STRATEGY;
}
return value;
}
public Class> getCollectionType(ObjectModelAttribute attribute) {
return JavaGeneratorUtil.getCollectionType(attribute);
}
public Class> getCollectionInstanceType(ObjectModelAttribute attribute) {
boolean unique = EugeneCoreTagValues.isUnique(attribute);
boolean ordered = EugeneCoreTagValues.isOrdered(attribute);
boolean orderBy = topiaHibernateTagValues.getOrderByTagValue(attribute) != null;
Class> result;
if (unique && orderBy && !ordered) {
// Special case, we want to keep the order coming from database
// This does not mean we keep order while saving a such relation
result = LinkedHashSet.class;
} else {
result = JavaGeneratorUtil.getCollectionInstanceType(attribute);
}
return result;
}
public Pair registerIndexKeyName(String schema, String tableName, String attrColumn) {
String result = "idx_";
if (schema != null) {
result += schema + "_";
}
result += tableName + "_" + attrColumn;
result = result.toLowerCase();
if (!indexes.containsKey(result)) {
String create = String.format("CREATE INDEX %s ON %s.%s(%s)", result, schema, tableName, attrColumn);
indexes.put(result, create);
return Pair.of(result, create);
}
return null;
}
public Pair registerUniqueKeyName(String schema, String tableName, String attrColumn) {
String result = "uk_";
if (schema != null) {
result += schema + "_";
}
result += tableName + "_" + attrColumn;
result = result.toLowerCase();
if (!uniqueKeys.containsKey(result)) {
String create = String.format("ALTER TABLE %s.%s ADD CONSTRAINT %s unique (%s)", schema, tableName, result, attrColumn);
uniqueKeys.put(result, create);
return Pair.of(result, create);
}
return null;
}
public void registerForeignKey(String result, String schema, String tableName, ObjectModelAttribute attr) {
String attrColumn = getDbName(attr);
if (!foreignKeys.containsKey(result)) {
ObjectModelClassifier attrClassifier = attr.getClassifier();
String attrSchema = attrClassifier == null ? schema : getSchema(attrClassifier);
String attrTableName = attrClassifier == null ? attrColumn : getDbName(attrClassifier);
String create = String.format("ALTER TABLE %s.%s ADD CONSTRAINT %s FOREIGN KEY(%s) REFERENCES %s.%s", schema, tableName, result, attrColumn, attrSchema, attrTableName);
foreignKeys.put(result, create);
}
}
public Map getIndexes() {
return indexes;
}
public Map getForeignKeys() {
return foreignKeys;
}
public Map getUniqueKeys() {
return uniqueKeys;
}
public String getSchema(ObjectModelClassifier classifier) {
return topiaHibernateTagValues.getDbSchemaNameTagValue(classifier, model.getPackage(classifier), model);
}
public String getEntityEnumLiteralName(ObjectModelClassifier clazz) {
String literalName = clazz.getName();
ObjectModelPackage aPackage = model.getPackage(clazz);
String dbSchema = topiaHibernateTagValues.getDbSchemaNameTagValue(clazz, aPackage, model);
if (dbSchema != null) {
literalName = dbSchema + "_" + literalName;
}
return literalName;
}
public TopiaHibernateTagValues topiaHibernateTagValues() {
return topiaHibernateTagValues;
}
//---------------------------------------------------------------------------------------------------------------
//-- DEPRECATED API TO REMOVE ---------------------------------------------------------------------------------
//---------------------------------------------------------------------------------------------------------------
/**
* dependency to add extra operations for entity dao.
*
* @param input FIXME
* @return FIXME
* @since 2.3.4
* @deprecated only used for warn to help migration to 3.0, should be deleted
*/
// TODO brendan 03/10/14 delete in 3.1
@Deprecated
public static String getLegacyDaoName(ObjectModelClass input) {
return input.getName() + "DAO";
}
}