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

com.landawn.abacus.util.QueryUtil Maven / Gradle / Ivy

/*
 * Copyright (C) 2021 HaiYang Li
 *
 * 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.landawn.abacus.util;

import java.util.ArrayList;
import java.util.Collection;
import java.util.EnumMap;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;

import com.landawn.abacus.annotation.Beta;
import com.landawn.abacus.annotation.Immutable;
import com.landawn.abacus.annotation.Internal;
import com.landawn.abacus.annotation.NotColumn;
import com.landawn.abacus.annotation.Table;
import com.landawn.abacus.parser.ParserUtil;
import com.landawn.abacus.parser.ParserUtil.BeanInfo;
import com.landawn.abacus.parser.ParserUtil.PropInfo;
import com.landawn.abacus.type.Type;
import com.landawn.abacus.util.Tuple.Tuple2;

public final class QueryUtil {

    private QueryUtil() {
        // singleton
    }

    private static final Map, ImmutableMap> column2PropNameNameMapPool = new ConcurrentHashMap<>();

    private static final Map, Map>> entityTablePropColumnNameMap = new ObjectPool<>(N.POOL_SIZE);

    private static final Map, Map>>> entityTablePropColumnNameMap2 = new ObjectPool<>(
            N.POOL_SIZE);

    /**
     *
     * @param entityClass
     * @param namingPolicy
     * @return
     * @deprecated for internal use only.
     */
    @Deprecated
    @Beta
    public static ImmutableMap> prop2ColumnNameMap(final Class entityClass, final NamingPolicy namingPolicy) {
        Map>> namingPropColumnNameMap = entityTablePropColumnNameMap2.get(entityClass);
        ImmutableMap> result = null;

        if (namingPropColumnNameMap == null || (result = namingPropColumnNameMap.get(namingPolicy)) == null) {
            final ImmutableMap prop2ColumnNameMap = getProp2ColumnNameMap(entityClass, namingPolicy);
            final Map> newProp2ColumnNameMap = N.newHashMap(prop2ColumnNameMap.size() * 2);

            for (final Map.Entry entry : prop2ColumnNameMap.entrySet()) {
                newProp2ColumnNameMap.put(entry.getKey(), Tuple.of(entry.getValue(), entry.getKey().indexOf('.') < 0));

                if (!prop2ColumnNameMap.containsKey(entry.getValue())) {
                    newProp2ColumnNameMap.put(entry.getValue(), Tuple.of(entry.getValue(), entry.getValue().indexOf('.') < 0));
                }
            }

            result = ImmutableMap.wrap(newProp2ColumnNameMap);

            if (namingPropColumnNameMap == null) {
                namingPropColumnNameMap = new EnumMap<>(NamingPolicy.class);
                // TODO not necessary?
                // namingPropColumnNameMap = Collections.synchronizedMap(namingPropColumnNameMap)
                entityTablePropColumnNameMap2.put(entityClass, namingPropColumnNameMap);
            }

            namingPropColumnNameMap.put(namingPolicy, result);
        }

        return result;
    }

    /**
     * Gets the column 2 field name map.
     *
     * @param entityClass
     * @return
     */
    public static ImmutableMap getColumn2PropNameMap(final Class entityClass) {
        ImmutableMap result = column2PropNameNameMapPool.get(entityClass);

        if (result == null) {
            final BeanInfo entityInfo = ParserUtil.getBeanInfo(entityClass);
            final Map map = N.newHashMap(entityInfo.propInfoList.size() * 3);

            for (final PropInfo propInfo : entityInfo.propInfoList) {
                if (propInfo.columnName.isPresent()) {
                    map.put(propInfo.columnName.get(), propInfo.name);
                    map.put(propInfo.columnName.get().toLowerCase(), propInfo.name);
                    map.put(propInfo.columnName.get().toUpperCase(), propInfo.name);
                }
            }

            result = ImmutableMap.copyOf(map);

            column2PropNameNameMapPool.put(entityClass, result);
        }

        return result;
    }

    /**
     * Gets the prop column name map.
     *
     * @param entityClass
     * @param namingPolicy
     * @return
     */
    public static ImmutableMap getProp2ColumnNameMap(final Class entityClass, final NamingPolicy namingPolicy) {
        if (entityClass == null || Map.class.isAssignableFrom(entityClass)) {
            return ImmutableMap.empty();
        }

        final Map> namingColumnNameMap = entityTablePropColumnNameMap.get(entityClass);
        ImmutableMap result = null;

        if (namingColumnNameMap == null || (result = namingColumnNameMap.get(namingPolicy)) == null) {
            result = registerEntityPropColumnNameMap(entityClass, namingPolicy, null);
        }

        return result;
    }

    static ImmutableMap registerEntityPropColumnNameMap(final Class entityClass, final NamingPolicy namingPolicy,
            final Set> registeringClasses) {
        N.checkArgNotNull(entityClass);

        if (registeringClasses != null) {
            if (registeringClasses.contains(entityClass)) {
                throw new RuntimeException("Cycling references found among: " + registeringClasses);
            }

            registeringClasses.add(entityClass);
        }

        final Table tableAnno = entityClass.getAnnotation(Table.class);
        final Set columnFields = tableAnno == null ? N.emptySet() : N.asSet(tableAnno.columnFields());
        final Set nonColumnFields = tableAnno == null ? N.emptySet() : N.asSet(tableAnno.nonColumnFields());
        final BeanInfo entityInfo = ParserUtil.getBeanInfo(entityClass);
        Map propColumnNameMap = N.newHashMap(entityInfo.propInfoList.size() * 2);

        for (final PropInfo propInfo : entityInfo.propInfoList) {
            if (isNotColumn(columnFields, nonColumnFields, propInfo)) {
                continue;
            }

            if (propInfo.columnName.isPresent()) {
                propColumnNameMap.put(propInfo.name, propInfo.columnName.get());
            } else {
                propColumnNameMap.put(propInfo.name, SQLBuilder.formalizeColumnName(propInfo.name, namingPolicy));

                final Type propType = propInfo.type.isCollection() ? propInfo.type.getElementType() : propInfo.type;

                if (propType.isBean() && (registeringClasses == null || !registeringClasses.contains(propType.clazz()))) {
                    final Set> newRegisteringClasses = registeringClasses == null ? N.newLinkedHashSet() : registeringClasses;
                    newRegisteringClasses.add(entityClass);

                    final Map subPropColumnNameMap = registerEntityPropColumnNameMap(propType.clazz(), namingPolicy, newRegisteringClasses);

                    if (N.notEmpty(subPropColumnNameMap)) {
                        final String subTableAliasOrName = SQLBuilder.getTableAliasOrName(propType.clazz(), namingPolicy);

                        for (final Map.Entry entry : subPropColumnNameMap.entrySet()) {
                            propColumnNameMap.put(propInfo.name + WD.PERIOD + entry.getKey(), subTableAliasOrName + WD.PERIOD + entry.getValue());
                        }

                        propColumnNameMap.remove(propInfo.name); // remove sub entity prop.
                    }
                }
            }
        }

        //    final Map tmp = entityTablePropColumnNameMap.get(entityClass);
        //
        //    if (N.notEmpty(tmp)) {
        //        propColumnNameMap.putAll(tmp);
        //    }

        if (N.isEmpty(propColumnNameMap)) {
            propColumnNameMap = N.emptyMap();
        }

        final ImmutableMap result = ImmutableMap.wrap(propColumnNameMap);

        Map> namingPropColumnMap = entityTablePropColumnNameMap.get(entityClass);

        if (namingPropColumnMap == null) {
            namingPropColumnMap = new EnumMap<>(NamingPolicy.class);
            // TODO not necessary?
            // namingPropColumnMap = Collections.synchronizedMap(namingPropColumnMap)
            entityTablePropColumnNameMap.put(entityClass, namingPropColumnMap);
        }

        namingPropColumnMap.put(namingPolicy, result);

        return result;
    }

    /**
     * Gets the insert prop names by class.
     *
     * @param entity
     * @param excludedPropNames
     * @return
     */
    @Internal
    public static Collection getInsertPropNames(final Object entity, final Set excludedPropNames) {
        final Class entityClass = entity.getClass();

        final Collection[] val = SQLBuilder.loadPropNamesByClass(entityClass);

        if (!N.isEmpty(excludedPropNames)) {
            final List tmp = new ArrayList<>(val[2]);
            tmp.removeAll(excludedPropNames);
            return tmp;
        }

        final Collection idPropNames = getIdFieldNames(entityClass);

        if (N.isEmpty(idPropNames)) {
            return val[2];
        }

        final BeanInfo entityInfo = ParserUtil.getBeanInfo(entityClass);

        for (final String idPropName : idPropNames) {
            if (!SQLBuilder.isDefaultIdPropValue(entityInfo.getPropInfo(idPropName))) {
                return val[2];
            }
        }

        return val[3];
    }

    /**
     * Gets the insert prop names by class.
     *
     * @param entityClass
     * @param excludedPropNames
     * @return
     */
    @Internal
    public static Collection getInsertPropNames(final Class entityClass, final Set excludedPropNames) {
        final Collection[] val = SQLBuilder.loadPropNamesByClass(entityClass);
        final Collection propNames = val[2];

        if (N.isEmpty(excludedPropNames)) {
            return propNames;
        }
        final List tmp = new ArrayList<>(propNames);
        tmp.removeAll(excludedPropNames);
        return tmp;
    }

    /**
     * Gets the select prop names by class.
     *
     * @param entityClass
     * @param includeSubEntityProperties
     * @param excludedPropNames
     * @return
     */
    @Internal
    public static Collection getSelectPropNames(final Class entityClass, final boolean includeSubEntityProperties,
            final Set excludedPropNames) {
        final Collection[] val = SQLBuilder.loadPropNamesByClass(entityClass);
        final Collection propNames = includeSubEntityProperties ? val[0] : val[1];

        if (N.isEmpty(excludedPropNames)) {
            return propNames;
        }

        return N.excludeAll(propNames, excludedPropNames);
    }

    /**
     * Gets the update prop names by class.
     *
     * @param entityClass
     * @param excludedPropNames
     * @return
     */
    @Internal
    public static Collection getUpdatePropNames(final Class entityClass, final Set excludedPropNames) {
        final Collection[] val = SQLBuilder.loadPropNamesByClass(entityClass);
        final Collection propNames = val[4];

        if (N.isEmpty(excludedPropNames)) {
            return propNames;
        }
        final List tmp = new ArrayList<>(propNames);
        tmp.removeAll(excludedPropNames);
        return tmp;
    }

    /**
     * Gets the id field names.
     *
     * @param targetClass
     * @return an immutable List.
     * @deprecated for internal only.
     */
    @Deprecated
    @Internal
    @Immutable
    public static List getIdFieldNames(final Class targetClass) {
        return getIdFieldNames(targetClass, false);
    }

    /**
     * Gets the id field names.
     *
     * @param targetClass
     * @param fakeIdForEmpty
     * @return an immutable List.
     * @deprecated for internal only.
     */
    @Deprecated
    @Internal
    @Immutable
    public static List getIdFieldNames(final Class targetClass, final boolean fakeIdForEmpty) {
        final ImmutableList idPropNames = ParserUtil.getBeanInfo(targetClass).idPropNameList;

        return N.isEmpty(idPropNames) && fakeIdForEmpty ? fakeIds : idPropNames;
    }

    /**
     *
     *
     * @param columnFields
     * @param nonColumnFields
     * @param propInfo
     * @return
     */
    public static boolean isNotColumn(final Set columnFields, final Set nonColumnFields, final PropInfo propInfo) {
        return propInfo.isTransient || propInfo.isAnnotationPresent(NotColumn.class) || (N.notEmpty(columnFields) && !columnFields.contains(propInfo.name))
                || (N.notEmpty(nonColumnFields) && nonColumnFields.contains(propInfo.name));
    }

    private static final ImmutableList fakeIds = ImmutableList.of("not_defined_fake_id_in_abacus_" + Strings.uuid());

    /**
     * Checks if is fake id.
     *
     * @param idPropNames
     * @return true, if is fake id
     * @deprecated for internal only.
     */
    @Deprecated
    @Internal
    public static boolean isFakeId(final List idPropNames) {
        return idPropNames != null && idPropNames.size() == 1 && fakeIds.get(0).equals(idPropNames.get(0));
    }

    private static final Map QM_CACHE = new HashMap<>();

    static {
        for (int i = 0; i <= 30; i++) {
            QM_CACHE.put(i, Strings.repeat("?", i, ", "));
        }

        QM_CACHE.put(100, Strings.repeat("?", 100, ", "));
        QM_CACHE.put(200, Strings.repeat("?", 200, ", "));
        QM_CACHE.put(300, Strings.repeat("?", 300, ", "));
        QM_CACHE.put(500, Strings.repeat("?", 500, ", "));
        QM_CACHE.put(1000, Strings.repeat("?", 1000, ", "));
    }

    /**
     * Repeat question mark({@code ?}) {@code n} times with delimiter {@code ", "}.
     * 
* It's designed for batch SQL builder. * * @param n * @return */ public static String repeatQM(final int n) { N.checkArgNotNegative(n, "count"); String result = QM_CACHE.get(n); if (result == null) { result = Strings.repeat("?", n, ", "); } return result; } public static String getTableAlias(final Class entityClass) { final Table anno = entityClass.getAnnotation(Table.class); return anno == null ? null : anno.alias(); } public static String getTableNameAndAlias(final Class entityClass) { return getTableNameAndAlias(entityClass, NamingPolicy.LOWER_CASE_WITH_UNDERSCORE); } public static String getTableNameAndAlias(final Class entityClass, final NamingPolicy namingPolicy) { final Table anno = entityClass.getAnnotation(Table.class); if (anno == null) { return namingPolicy.convert(ClassUtil.getSimpleClassName(entityClass)); } else { final String tableName = anno.name(); final String alias = anno.alias(); return Strings.isEmpty(alias) ? tableName : Strings.concat(tableName, WD.SPACE, alias); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy