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

org.dbflute.s2dao.extension.TnRelationRowCreatorExtension Maven / Gradle / Ivy

/*
 * Copyright 2014-2017 the original author or 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
 *
 *     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 org.dbflute.s2dao.extension;

import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;

import org.dbflute.Entity;
import org.dbflute.bhv.core.context.ResourceContext;
import org.dbflute.dbmeta.DBMeta;
import org.dbflute.dbmeta.accessory.ColumnNullObjectable;
import org.dbflute.dbmeta.info.ColumnInfo;
import org.dbflute.jdbc.ValueType;
import org.dbflute.s2dao.metadata.TnBeanMetaData;
import org.dbflute.s2dao.metadata.TnPropertyMapping;
import org.dbflute.s2dao.metadata.TnPropertyType;
import org.dbflute.s2dao.metadata.TnRelationPropertyType;
import org.dbflute.s2dao.rowcreator.TnRelationKey;
import org.dbflute.s2dao.rowcreator.TnRelationRowCache;
import org.dbflute.s2dao.rowcreator.TnRelationRowCreationResource;
import org.dbflute.s2dao.rowcreator.TnRelationSelector;
import org.dbflute.s2dao.rowcreator.impl.TnRelationRowCreatorImpl;
import org.dbflute.s2dao.rshandler.TnBeanListResultSetHandler;
import org.dbflute.util.DfReflectionUtil;

/**
 * The DBFlute extension of relation row creator.
 * @author jflute
 */
public class TnRelationRowCreatorExtension extends TnRelationRowCreatorImpl {

    // ===================================================================================
    //                                                                           Attribute
    //                                                                           =========
    protected final TnRelationRowOptionalHandler _optionalHandler;

    // ===================================================================================
    //                                                                         Constructor
    //                                                                         ===========
    public TnRelationRowCreatorExtension(TnRelationRowOptionalHandler optionalHandler) {
        _optionalHandler = optionalHandler;
    }

    public static TnRelationRowCreatorExtension createRelationRowCreator(TnRelationRowOptionalHandler optionalHandler) {
        return new TnRelationRowCreatorExtension(optionalHandler);
    }

    // ===================================================================================
    //                                                             Relation KeyValue Setup
    //                                                             =======================
    @Override
    protected void setupRelationKeyValue(TnRelationRowCreationResource res) {
        // /= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
        // setup of relation key is handled at all-value setup marked as '#RELKEY'
        // so only entity instance creation exists in this method
        // = = = = = = = = = =/
        final TnRelationPropertyType rpt = res.getRelationPropertyType();
        final TnBeanMetaData yourBmd = rpt.getYourBeanMetaData();
        if (!res.hasRowInstance()) { // always no instance here (check just in case)
            final DBMeta dbmeta = yourBmd.getDBMeta();
            final Object row = newRelationRow(rpt, res.getRelationSelector(), res.getRelationNoSuffix(), dbmeta);
            res.setRow(row);
        }
    }

    protected Object newRelationRow(TnRelationPropertyType rpt, TnRelationSelector selector, String relationNoSuffix, DBMeta dbmeta) {
        final Object row;
        if (dbmeta != null) {
            final Entity entity = dbmeta.newEntity();
            reflectConditionBeanOptionToEntity(selector, relationNoSuffix, entity);
            row = entity;
        } else { // no way (relation of DBFlute entity is only supported)
            row = newNonEntityRelationRow(rpt);
        }
        return row;
    }

    protected void reflectConditionBeanOptionToEntity(TnRelationSelector selector, String relationNoSuffix, Entity entity) {
        // unlock access to undefined classification if allowed in condition-bean
        // this should be set before mapping values (and also base-point table's creator)
        if (selector.isUndefinedClassificationSelectAllowed(relationNoSuffix)) {
            entity.myunlockUndefinedClassificationAccess();
        }
    }

    protected Object newNonEntityRelationRow(TnRelationPropertyType rpt) { // for non DBFlute entity
        return DfReflectionUtil.newInstance(rpt.getPropertyDesc().getPropertyType());
    }

    // ===================================================================================
    //                                                             Relation AllValue Setup
    //                                                             =======================
    @Override
    protected void setupRelationAllValue(TnRelationRowCreationResource res) throws SQLException {
        final Map propertyCacheElement = res.extractPropertyCacheElement();
        for (Entry entry : propertyCacheElement.entrySet()) {
            final TnPropertyMapping pt = entry.getValue();
            res.setCurrentPropertyType(pt);
            if (!isValidRelationPerPropertyLoop(res)) { // no way unless the method is overridden
                res.clearRowInstance();
                return;
            }
            setupRelationProperty(res);
        }
        if (!isValidRelationAfterPropertyLoop(res)) { // e.g. when all values are null
            res.clearRowInstance();
            return;
        }
        res.clearValidValueCount();
        if (res.isStopNextRelationMapping()) {
            return;
        }
        setupNextRelationRow(res);
    }

    protected void setupRelationProperty(TnRelationRowCreationResource res) throws SQLException {
        final String columnName = res.buildRelationColumnName();
        // already created here, this is old S2Dao logic 
        //if (!res.hasRowInstance()) {
        //    res.setRow(newRelationRow(res));
        //}
        registerRelationValue(res, columnName);
    }

    protected void registerRelationValue(TnRelationRowCreationResource res, String columnName) throws SQLException {
        final TnPropertyMapping mapping = res.getCurrentPropertyMapping();
        Object value = null;
        if (res.containsRelationKeyColumn(columnName)) { // #RELKEY
            // if this column is relation key, it gets the value from relation key values
            // for performance and avoiding twice getting same column value
            value = res.extractRelationKeyValue(columnName);
        } else {
            final ValueType valueType = mapping.getValueType();
            final Map> selectIndexMap = res.getSelectIndexMap();
            final ResultSet rs = res.getResultSet();
            if (selectIndexMap != null) {
                final String relationNoSuffix = res.getRelationNoSuffix();
                value = ResourceContext.getRelationValue(rs, relationNoSuffix, columnName, valueType, selectIndexMap);
            } else {
                value = valueType.getValue(rs, columnName);
            }
        }
        handleRelationValueRegistration(res, mapping, value);
    }

    protected void handleRelationValueRegistration(TnRelationRowCreationResource res, TnPropertyMapping mapping, Object value) {
        if (value != null) {
            res.incrementValidValueCount();
        }
        // null is also set to trace modified properties for specified properties
        // little performance cost because only setupSelect and specified columns are here
        // (no setupSelect relation does not come here: old days, S2Dao might be possible)
        doRegisterRelationValue(res, mapping, value);
    }

    protected void doRegisterRelationValue(TnRelationRowCreationResource res, TnPropertyMapping mapping, Object value) {
        final ColumnInfo columnInfo = mapping.getEntityColumnInfo();
        if (columnInfo != null) {
            columnInfo.write((Entity) res.getRow(), value);
        } else {
            mapping.getPropertyAccessor().setValue(res.getRow(), value);
        }
    }

    // -----------------------------------------------------
    //                                         Next Relation
    //                                         -------------
    protected void setupNextRelationRow(TnRelationRowCreationResource res) throws SQLException {
        final TnBeanMetaData nextBmd = res.getRelationBeanMetaData();
        final Object row = res.getRow();
        res.prepareNextLevelMapping();
        try {
            final List nextRptList = nextBmd.getRelationPropertyTypeList();
            for (TnRelationPropertyType nextRpt : nextRptList) {
                setupNextRelationRowElement(res, row, nextRpt);
            }
        } finally {
            res.setRow(row);
            res.closeNextLevelMapping();
        }
    }

    protected void setupNextRelationRowElement(TnRelationRowCreationResource res, Object row, TnRelationPropertyType nextRpt)
            throws SQLException {
        res.prepareNextRelationProperty(nextRpt);
        try {
            mappingNextRelation(res, row);
        } finally {
            res.closeNextRelationProperty();
        }
    }

    /**
     * Do mapping next relation row. 
* This logic is similar to first relation mapping in {@link TnBeanListResultSetHandler}.
* So you should check it when this logic has modification. * @param res The resource of relation row creation. (NotNull) * @param row The base point row, which is previous relation row. (NotNull) * @throws SQLException When it fails to handle the SQL. */ protected void mappingNextRelation(TnRelationRowCreationResource res, Object row) throws SQLException { if (res.isStopCurrentRelationMapping()) { return; } final TnRelationKey relKey = res.prepareRelationKey(); // also saves it in resource final TnRelationPropertyType rpt = res.getRelationPropertyType(); Object relationRow = null; if (relKey != null) { final String relationNoSuffix = res.getRelationNoSuffix(); final boolean canUseRelationCache = res.canUseRelationCache(); TnRelationRowCache relRowCache = null; if (canUseRelationCache) { relRowCache = res.getRelRowCache(); relationRow = relRowCache.getRelationRow(relationNoSuffix, relKey); } if (relationRow == null) { // when no cache relationRow = createRelationRow(res); if (relationRow != null) { // is new created relation row adjustCreatedRelationRow(relationRow, res.getRelationNoSuffix(), res.getRelationSelector(), rpt); if (canUseRelationCache) { relRowCache.addRelationRow(relationNoSuffix, relKey, relationRow); } } } } // if exists, optional or plain value // if null, empty optional or nothing relationRow = filterOptionalRelationRowIfNeeds(row, rpt, relationRow); if (relationRow != null) { // exists or empty optional rpt.getPropertyAccessor().setValue(row, relationRow); } } /** * Adjust created row for relation tables. * @param relationRow The relation row of tables related to the base-point table. (NotNull) * @param relationNoSuffix The suffix of relation no, e.g. _0, _1_3. (NotNull) * @param relSelector The selector of relation, which has various determination. (NotNull) * @param rpt The property type of the relation. (NotNull) */ public static void adjustCreatedRelationRow(Object relationRow, String relationNoSuffix, TnRelationSelector relSelector, TnRelationPropertyType rpt) { // static for also handler calling // *similar implementation for base-point row exists, see it for the details if (relationRow instanceof Entity) { final Entity entity = (Entity) relationRow; // check access to non-specified-column if (!relSelector.isNonSpecifiedColumnAccessAllowed(relationNoSuffix) // not allowed && relSelector.isUsingSpecifyColumnInRelation(relationNoSuffix)) { // and use SpecifyColumn entity.modifiedToSpecified(); // so check it // adjust specification for column null object handling final Set nullObjectColumnSet = relSelector.getRelationSpecifiedNullObjectColumnSet(relationNoSuffix); for (ColumnInfo columnInfo : nullObjectColumnSet) { // might be empty loop if no null object entity.myspecifyProperty(columnInfo.getPropertyName()); } } // enable the handling of column null object if allowed and object-able if (relSelector.isColumnNullObjectEnabled(relationNoSuffix) && entity instanceof ColumnNullObjectable) { ((ColumnNullObjectable) entity).enableColumnNullObject(); } // clear modified properties for update process using selected entity entity.clearModifiedInfo(); // mark as select to determine the entity is selected or user-created // basically for e.g. determine columns of batch insert entity.markAsSelect(); } else { // not DBFlute entity // actually any bean meta data can be accepted // because only it gets modified properties rpt.getYourBeanMetaData().getModifiedPropertyNames(relationRow).clear(); } } // =================================================================================== // Property Cache Setup // ==================== @Override protected void setupPropertyCache(TnRelationRowCreationResource res) throws SQLException { // - - - - - - - - - - - // Recursive Call Point! // - - - - - - - - - - - if (res.isStopCurrentRelationMapping()) { return; } // set up property cache about current bean meta data final TnBeanMetaData nextBmd = res.getRelationBeanMetaData(); final List ptList = nextBmd.getPropertyTypeList(); for (TnPropertyType pt : ptList) { // already been filtered as target only res.setCurrentPropertyType(pt); setupPropertyCacheElement(res); } // set up next level relation's property cache if (res.isStopNextRelationMapping()) { return; } res.prepareNextLevelMapping(); try { setupNextPropertyCache(res, nextBmd); } finally { res.closeNextLevelMapping(); } } // ----------------------------------------------------- // Next Relation // ------------- protected void setupNextPropertyCache(TnRelationRowCreationResource res, TnBeanMetaData nextBmd) throws SQLException { final List nextRptList = nextBmd.getRelationPropertyTypeList(); for (TnRelationPropertyType nextRpt : nextRptList) { setupNextPropertyCacheElement(res, nextRpt); } } protected void setupNextPropertyCacheElement(TnRelationRowCreationResource res, TnRelationPropertyType nextRpt) throws SQLException { res.prepareNextRelationProperty(nextRpt); try { setupPropertyCache(res); // recursive call } finally { res.closeNextRelationProperty(); } } // =================================================================================== // Option Override // =============== @Override protected boolean isCreateDeadLink() { return false; // DBFlute does not create dead-link relation, treated as null } @Override protected int getLimitRelationNestLevel() { // basically unused on DBFlute because only ConditionBean uses relation row, // and ConditionBean supports unlimited relation nest level // so this limit size is always used after hasConditionBean() return 2; // for Compatible (old parameter) } // =================================================================================== // Optional Handling // ================= public Object filterOptionalRelationRowIfNeeds(Object row, TnRelationPropertyType rpt, Object relationRow) { return _optionalHandler.filterOptionalRelationRowIfNeeds(row, rpt, relationRow); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy