org.dbflute.s2dao.sqlcommand.TnInsertEntityDynamicCommand Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of dbflute-runtime Show documentation
Show all versions of dbflute-runtime Show documentation
The runtime library of DBFlute
/*
* Copyright 2014-2023 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.sqlcommand;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;
import javax.sql.DataSource;
import org.dbflute.Entity;
import org.dbflute.bhv.core.context.InternalMapContext;
import org.dbflute.bhv.writable.InsertOption;
import org.dbflute.cbean.ConditionBean;
import org.dbflute.dbmeta.name.ColumnSqlName;
import org.dbflute.helper.message.ExceptionMessageBuilder;
import org.dbflute.jdbc.StatementConfig;
import org.dbflute.jdbc.StatementFactory;
import org.dbflute.s2dao.identity.TnIdentifierGenerator;
import org.dbflute.s2dao.metadata.TnBeanMetaData;
import org.dbflute.s2dao.metadata.TnPropertyType;
import org.dbflute.s2dao.sqlhandler.TnInsertEntityHandler;
/**
* @author modified by jflute (originated in S2Dao)
*/
public class TnInsertEntityDynamicCommand extends TnAbstractEntityDynamicCommand {
// ===================================================================================
// Constructor
// ===========
public TnInsertEntityDynamicCommand(DataSource dataSource, StatementFactory statementFactory) {
super(dataSource, statementFactory);
}
// ===================================================================================
// Execute
// =======
public Object execute(Object[] args) {
final Object bean = extractBeanFromArgsChecked(args);
final InsertOption option = extractInsertOptionChecked(args);
prepareStatementConfigOnThreadIfExists(option);
final TnBeanMetaData bmd = _beanMetaData;
final TnPropertyType[] propertyTypes = createInsertPropertyTypes(bmd, bean, _propertyNames, option);
final String sql = filterExecutedSql(createInsertSql(bmd, propertyTypes, option));
return doExecute(bean, propertyTypes, sql, option);
}
protected InsertOption extractInsertOptionChecked(Object[] args) {
if (args.length < 2 || args[1] == null) {
return null;
}
@SuppressWarnings("unchecked")
final InsertOption option = (InsertOption) args[1];
return option;
}
protected void prepareStatementConfigOnThreadIfExists(InsertOption option) {
final StatementConfig config = option != null ? option.getInsertStatementConfig() : null;
if (config != null) {
InternalMapContext.setUpdateStatementConfig(config);
}
}
protected Object doExecute(Object bean, TnPropertyType[] propertyTypes, String sql, InsertOption option) {
final TnInsertEntityHandler handler = createInsertEntityHandler(propertyTypes, sql, option);
final Object[] realArgs = new Object[] { bean };
handler.setExceptionMessageSqlArgs(realArgs);
final int rows = handler.execute(realArgs);
return Integer.valueOf(rows);
}
// ===================================================================================
// Insert Column
// =============
protected TnPropertyType[] createInsertPropertyTypes(TnBeanMetaData bmd, Object bean, String[] propertyNames,
InsertOption option) {
if (0 == propertyNames.length) {
String msg = "The property name was not found in the bean: " + bean;
throw new IllegalStateException(msg);
}
final List typeList = new ArrayList();
final Set> modifiedSet = getModifiedPropertyNames(bean);
final String timestampProp = bmd.getTimestampPropertyName();
final String versionNoProp = bmd.getVersionNoPropertyName();
for (int i = 0; i < propertyNames.length; ++i) {
final TnPropertyType pt = bmd.getPropertyType(propertyNames[i]);
if (pt.isPrimaryKey()) {
if (option == null || !option.isPrimaryKeyIdentityDisabled()) {
final TnIdentifierGenerator generator = bmd.getIdentifierGenerator(pt.getPropertyName());
if (!generator.isSelfGenerate()) {
continue;
}
}
typeList.add(pt);
} else {
if (isOptimisticLockProperty(timestampProp, versionNoProp, pt) // OptimisticLock
|| isSpecifiedProperty(bean, option, modifiedSet, pt)) { // Specified
typeList.add(pt);
}
}
}
if (typeList.isEmpty()) {
throwEntityInsertPropertyNotFoundException(bmd, bean);
}
return (TnPropertyType[]) typeList.toArray(new TnPropertyType[typeList.size()]);
}
protected Set> getModifiedPropertyNames(Object bean) {
return _beanMetaData.getModifiedPropertyNames(bean);
}
protected boolean isOptimisticLockProperty(String timestampProp, String versionNoProp, TnPropertyType pt) {
final String propertyName = pt.getPropertyName();
return propertyName.equalsIgnoreCase(timestampProp) || propertyName.equalsIgnoreCase(versionNoProp);
}
protected boolean isSpecifiedProperty(Object bean, InsertOption option, Set> modifiedSet, TnPropertyType pt) {
if (option != null && option.hasSpecifiedInsertColumn()) { // basically BatchUpdate
// BatchUpdate's modified properties are translated to specified columns
// so all BatchUpdate commands are here
return option.isSpecifiedInsertColumn(pt.getColumnDbName());
} else { // basically EntityInsert
if (isEntityCreatedBySelect(bean)) { // e.g. copy insert
return true; // every column
} else { // new-created entity: mainly here
if (option != null && option.xisCompatibleInsertColumnNotNullOnly()) { // for compatible
return isNotNullProperty(bean, pt);
} else { // mainly here
return isModifiedProperty(modifiedSet, pt); // process for ModifiedColumnInsert
}
}
}
}
protected boolean isEntityCreatedBySelect(Object bean) {
if (bean instanceof Entity) {
Entity entity = (Entity) bean;
return entity.createdBySelect();
}
return false;
}
protected boolean isNotNullProperty(Object bean, TnPropertyType pt) {
return pt.getPropertyAccessor().getValue(bean) != null;
}
protected boolean isModifiedProperty(Set> modifiedSet, TnPropertyType pt) {
return modifiedSet.contains(pt.getPropertyName());
}
protected void throwEntityInsertPropertyNotFoundException(TnBeanMetaData bmd, Object bean) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("The insert property of the entity was not found.");
br.addItem("Advice");
br.addElement("The entity should have one or more insert properties.");
br.addElement("For example, an identity-column-only table is unsupported.");
br.addItem("Table");
br.addElement(bmd.getTableName());
br.addItem("Entity");
br.addElement(bean != null ? bean.getClass() : null);
final String msg = br.buildExceptionMessage();
throw new IllegalStateException(msg);
}
// ===================================================================================
// Insert SQL
// ==========
protected String createInsertSql(TnBeanMetaData bmd, TnPropertyType[] propertyTypes, InsertOption option) {
final String tableDbName = _targetDBMeta.getTableDbName();
final StringBuilder columnSb = new StringBuilder(48);
final StringBuilder valuesSb = new StringBuilder(48);
for (int i = 0; i < propertyTypes.length; ++i) {
final TnPropertyType pt = propertyTypes[i];
final ColumnSqlName columnSqlName = pt.getColumnSqlName();
if (i > 0) {
columnSb.append(", ");
valuesSb.append(", ");
}
columnSb.append(columnSqlName);
final String columnDbName = pt.getColumnDbName();
valuesSb.append(encryptIfNeeds(tableDbName, columnDbName, "?"));
}
final StringBuilder sb = new StringBuilder(128);
sb.append("insert into ").append(_targetDBMeta.getTableSqlName());
sb.append(" (").append(columnSb).append(")");
sb.append(ln()).append(" values (").append(valuesSb).append(")");
return sb.toString();
}
// ===================================================================================
// Handler
// =======
protected TnInsertEntityHandler createInsertEntityHandler(TnPropertyType[] boundPropTypes, String sql,
InsertOption option) {
final TnInsertEntityHandler handler = new TnInsertEntityHandler(_dataSource, _statementFactory, sql, _beanMetaData, boundPropTypes);
handler.setInsertOption(option);
return handler;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy