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

io.micronaut.data.model.naming.NamingStrategy Maven / Gradle / Ivy

There is a newer version: 4.11.0
Show newest version
/*
 * Copyright 2017-2020 original authors
 *
 * 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
 *
 * https://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 io.micronaut.data.model.naming;

import io.micronaut.core.annotation.NonNull;
import io.micronaut.core.annotation.Introspected;
import io.micronaut.core.naming.NameUtils;
import io.micronaut.core.util.ArgumentUtils;
import io.micronaut.core.util.StringUtils;
import io.micronaut.data.annotation.MappedEntity;
import io.micronaut.data.annotation.MappedProperty;
import io.micronaut.data.annotation.Relation;
import io.micronaut.data.model.Association;
import io.micronaut.data.model.Embedded;
import io.micronaut.data.model.PersistentEntity;
import io.micronaut.data.model.PersistentProperty;

import java.util.List;
import java.util.Optional;
import java.util.function.Function;


/**
 * A strategy interface for resolving the mapped name of an entity or property.
 *
 * @author graemerocher
 * @since 1.0
 */
@FunctionalInterface
@Introspected
public interface NamingStrategy {

    /**
     * Constant for the default underscore separated lower case strategy.
     */
    NamingStrategy DEFAULT = new NamingStrategies.UnderScoreSeparatedLowerCase();

    /**
     * Return the mapped name for the given name.
     * @param name The name
     * @return The mapped name
     */
    @NonNull
    String mappedName(@NonNull String name);

    /**
     * Return the mapped name for the given entity.
     * @param entity The entity
     * @return The mapped name
     */
    default @NonNull String mappedName(@NonNull PersistentEntity entity) {
        ArgumentUtils.requireNonNull("entity", entity);
        return entity.getAnnotationMetadata().stringValue(MappedEntity.class)
                .filter(StringUtils::isNotEmpty)
                .orElseGet(() -> mappedName(entity.getSimpleName()));
    }

    /**
     * Return the mapped name given an {@link Embedded} association and the property of the association. The
     * default strategy takes the parent embedded property name and combines it underscore separated with the child parent property name.
     *
     * 

For example given:

* *

     * {@literal @}Embedded Address address;
     * 
* *

Where the {@code Address} type has a property called {@code street} then a name of {@code address_street} will be returned

* @param embedded The embedded parent * @param property The embedded property * @return The mapped name */ default @NonNull String mappedName(Embedded embedded, PersistentProperty property) { return mappedName(embedded.getName() + mappedAssociatedName(property.getPersistedName())); } /** * Return the mapped name for the given property. * @param property The property * @return The mapped name */ default @NonNull String mappedName(@NonNull PersistentProperty property) { ArgumentUtils.requireNonNull("property", property); if (property instanceof Association association) { return mappedName(association); } else { return property.getAnnotationMetadata() .stringValue(MappedProperty.class) .filter(StringUtils::isNotEmpty) .orElseGet(() -> mappedName(property.getName())); } } /** * Return the mapped name for the given association. * @param association The association * @return The mapped name */ default @NonNull String mappedName(Association association) { String providedName = association.getAnnotationMetadata().stringValue(MappedProperty.class).orElse(null); if (providedName != null) { return providedName; } if (association.isForeignKey()) { Optional inverseSide = association.getInverseSide().map(Function.identity()); Association owningAssociation = inverseSide.orElse(association); return mappedName(owningAssociation.getOwner().getDecapitalizedName() + owningAssociation.getAssociatedEntity().getSimpleName()); } else { return switch (association.getKind()) { case ONE_TO_ONE, MANY_TO_ONE -> mappedName(association.getName() + getForeignKeySuffix()); default -> mappedName(association.getName()); }; } } /** * Convert the associated name to a proper format to be appended to the path. * @param associatedName The associated name * @return the name in a proper format * @since 4.2.0 */ @NonNull default String mappedAssociatedName(@NonNull String associatedName) { return NameUtils.capitalize(associatedName); } default @NonNull String mappedName(@NonNull List associations, @NonNull PersistentProperty property) { if (associations.isEmpty()) { return mappedName(property); } StringBuilder sb = new StringBuilder(); Association foreignAssociation = null; for (Association association : associations) { if (association.getKind() != Relation.Kind.EMBEDDED && foreignAssociation == null) { foreignAssociation = association; } final String originalAssocName = association.getName(); String assocName = association.getKind() == Relation.Kind.EMBEDDED ? association.getAnnotationMetadata().stringValue(MappedProperty.class).orElse(originalAssocName) : originalAssocName; if (!sb.isEmpty()) { sb.append(mappedAssociatedName(assocName)); } else { sb.append(assocName); } } if (foreignAssociation != null) { PersistentEntity associatedEntity = foreignAssociation.getAssociatedEntity(); PersistentProperty associatedEntityIdentity = associatedEntity != null ? associatedEntity.getIdentity() : null; if (associatedEntity != null && associatedEntity.equals(property.getOwner()) && associatedEntityIdentity != null && associatedEntityIdentity.equals(property)) { String providedName = foreignAssociation.getAnnotationMetadata().stringValue(MappedProperty.class).orElse(null); if (providedName != null) { return providedName; } sb.append(getForeignKeySuffix()); return mappedName(sb.toString()); } else if (foreignAssociation.isForeignKey()) { throw new IllegalStateException("Foreign association cannot be mapped!"); } } else { String providedName = property.getAnnotationMetadata().stringValue(MappedProperty.class).orElse(null); if (providedName != null) { return providedName; } } if (!sb.isEmpty()) { sb.append(mappedAssociatedName(property.getName())); } else { sb.append(property.getName()); } return mappedName(sb.toString()); } default String mappedJoinTableColumn(PersistentEntity associated, List associations, PersistentProperty property) { StringBuilder sb = new StringBuilder(); sb.append(associated.getDecapitalizedName()); for (Association association : associations) { sb.append(mappedAssociatedName(association.getName())); } if (associations.isEmpty()) { sb.append(getForeignKeySuffix()); } else { sb.append(mappedAssociatedName(property.getName())); } return mappedName(sb.toString()); } /** * The default foreign key suffix for property names. * @return The suffix */ default @NonNull String getForeignKeySuffix() { return "Id"; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy