org.dbflute.s2dao.extension.TnRelationRowCreatorExtension Maven / Gradle / Ivy
/*
* Copyright 2014-2020 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