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

org.dbflute.bhv.AbstractBehaviorWritable Maven / Gradle / Ivy

There is a newer version: 1.2.8
Show newest version
/*
 * Copyright 2014-2015 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.bhv;

import java.util.ArrayList;
import java.util.List;
import java.util.Set;

import org.dbflute.Entity;
import org.dbflute.bhv.core.command.AbstractListEntityCommand;
import org.dbflute.bhv.core.command.BatchDeleteCommand;
import org.dbflute.bhv.core.command.BatchDeleteNonstrictCommand;
import org.dbflute.bhv.core.command.BatchInsertCommand;
import org.dbflute.bhv.core.command.BatchUpdateCommand;
import org.dbflute.bhv.core.command.BatchUpdateNonstrictCommand;
import org.dbflute.bhv.core.command.DeleteEntityCommand;
import org.dbflute.bhv.core.command.DeleteNonstrictEntityCommand;
import org.dbflute.bhv.core.command.QueryDeleteCBCommand;
import org.dbflute.bhv.core.command.QueryInsertCBCommand;
import org.dbflute.bhv.core.command.QueryUpdateCBCommand;
import org.dbflute.bhv.core.command.UpdateEntityCommand;
import org.dbflute.bhv.core.command.UpdateNonstrictEntityCommand;
import org.dbflute.bhv.core.context.ResourceContext;
import org.dbflute.bhv.writable.DeleteOption;
import org.dbflute.bhv.writable.InsertOption;
import org.dbflute.bhv.writable.QueryInsertSetupper;
import org.dbflute.bhv.writable.UpdateOption;
import org.dbflute.bhv.writable.WritableOptionCall;
import org.dbflute.cbean.ConditionBean;
import org.dbflute.cbean.scoping.SpecifyQuery;
import org.dbflute.dbmeta.DBMeta;
import org.dbflute.dbmeta.info.ColumnInfo;
import org.dbflute.exception.EntityAlreadyDeletedException;
import org.dbflute.exception.EntityAlreadyUpdatedException;
import org.dbflute.exception.IllegalBehaviorStateException;
import org.dbflute.exception.IllegalConditionBeanOperationException;
import org.dbflute.exception.OptimisticLockColumnValueNullException;
import org.dbflute.helper.message.ExceptionMessageBuilder;
import org.dbflute.hook.CommonColumnAutoSetupper;

/**
 * The abstract class of writable behavior.
 * @param  The type of entity handled by this behavior.
 * @param  The type of condition-bean handled by this behavior.
 * @author jflute
 */
public abstract class AbstractBehaviorWritable extends
        AbstractBehaviorReadable implements BehaviorWritable {

    // ===================================================================================
    //                                                                           Attribute
    //                                                                           =========
    /** The auto-set-upper of common column. (NotNull) */
    protected CommonColumnAutoSetupper _commonColumnAutoSetupper;

    // ===================================================================================
    //                                                                       Entity Update
    //                                                                       =============
    // -----------------------------------------------------
    //                                                Insert
    //                                                ------
    protected void doInsert(ENTITY entity, InsertOption option) {
        assertEntityNotNull(entity);
        prepareInsertOption(option);
        delegateInsert(entity, option);
    }

    protected void prepareInsertOption(InsertOption option) {
        if (option == null) {
            return;
        }
        assertInsertOptionStatus(option);
        if (option.hasSpecifiedInsertColumn()) {
            final CB cb = createCBForSpecifiedUpdate();
            option.resolveInsertColumnSpecification(cb);
        }
    }

    protected void assertInsertOptionStatus(InsertOption option) {
        if (option.isCommonColumnAutoSetupDisabled() && !asDBMeta().hasCommonColumn()) {
            String msg = "The common column auto-setup disabling was set to the table not defined common columns:";
            msg = msg + " table=" + asTableDbName() + " option=" + option;
            throw new IllegalStateException(msg);
        }
        if (option.isPrimaryKeyIdentityDisabled() && !asDBMeta().hasIdentity()) {
            String msg = "The identity disabling was set to the table not defined identity:";
            msg = msg + " table=" + asTableDbName() + " option=" + option;
            throw new IllegalStateException(msg);
        }
    }

    protected CB createCBForSpecifiedUpdate() {
        final CB cb = newConditionBean();
        cb.xsetupForSpecifiedUpdate();
        return cb;
    }

    protected InsertOption createInsertOption(WritableOptionCall> opCall) {
        assertInsertOpCallNotNull(opCall);
        final InsertOption op = newInsertOption();
        opCall.callback(op);
        return op;
    }

    protected InsertOption newInsertOption() {
        return new InsertOption();
    }

    protected void assertInsertOpCallNotNull(WritableOptionCall> opCall) { // for varyingInsert()
        assertObjectNotNull("opLambda (for insert)", opCall);
    }

    // -----------------------------------------------------
    //                                                Create
    //                                                ------
    /**
     * {@inheritDoc}
     */
    public void create(Entity entity, InsertOption option) {
        doCreate(entity, option);
    }

    protected void doCreate(Entity entity, InsertOption option) {
        doInsert(downcast(entity), downcast(option));
    }

    // -----------------------------------------------------
    //                                                Update
    //                                                ------
    protected void doUpdate(ENTITY entity, UpdateOption option) {
        assertEntityNotNull(entity);
        prepareUpdateOption(option);
        helpUpdateInternally(entity, option);
    }

    protected void doUpdateNonstrict(ENTITY entity, UpdateOption option) {
        assertEntityNotNull(entity);
        prepareUpdateOption(option);
        helpUpdateNonstrictInternally(entity, option);
    }

    protected void prepareUpdateOption(UpdateOption option) {
        if (option == null) {
            return;
        }
        assertUpdateOptionStatus(option);
        if (option.hasSelfSpecification()) {
            option.resolveSelfSpecification(() -> createCBForVaryingUpdate());
        }
        if (option.hasSpecifiedUpdateColumn()) {
            final CB cb = createCBForSpecifiedUpdate();
            option.resolveUpdateColumnSpecification(cb);
        }
    }

    protected CB createCBForVaryingUpdate() {
        final CB cb = newConditionBean();
        cb.xsetupForVaryingUpdate();
        return cb;
    }

    protected  void helpUpdateInternally(RESULT entity, UpdateOption option) {
        assertEntityNotNull(entity);
        assertEntityHasOptimisticLockValue(entity);
        final int updatedCount = delegateUpdate(entity, option);
        if (updatedCount == 0) {
            throwUpdateEntityAlreadyDeletedException(entity);
        } else if (updatedCount > 1) {
            throwUpdateEntityDuplicatedException(entity, updatedCount);
        }
    }

    protected  void helpUpdateNonstrictInternally(RESULT entity, UpdateOption option) {
        assertEntityNotNull(entity);
        final int updatedCount = delegateUpdateNonstrict(entity, option);
        if (updatedCount == 0) {
            throwUpdateEntityAlreadyDeletedException(entity);
        } else if (updatedCount > 1) {
            throwUpdateEntityDuplicatedException(entity, updatedCount);
        }
    }

    protected void throwUpdateEntityAlreadyDeletedException(ENTITY entity) {
        createBhvExThrower().throwUpdateEntityAlreadyDeletedException(entity);
    }

    protected void throwUpdateEntityDuplicatedException(ENTITY entity, int count) {
        createBhvExThrower().throwUpdateEntityDuplicatedException(entity, count);
    }

    protected void assertUpdateOptionStatus(UpdateOption option) {
        if (option.isCommonColumnAutoSetupDisabled() && !asDBMeta().hasCommonColumn()) {
            String msg = "The common column auto-setup disabling was set to the table not defined common columns:";
            msg = msg + " table=" + asTableDbName() + " option=" + option;
            throw new IllegalStateException(msg);
        }
    }

    protected UpdateOption createUpdateOption(WritableOptionCall> opCall) {
        assertUpdateOpCallNotNull(opCall);
        final UpdateOption op = newUpdateOption();
        opCall.callback(op);
        return op;
    }

    protected UpdateOption newUpdateOption() {
        return new UpdateOption();
    }

    protected void assertUpdateOpCallNotNull(WritableOptionCall> opCall) { // for varyingUpdate()
        assertObjectNotNull("opLambda (for update)", opCall);
    }

    // -----------------------------------------------------
    //                                                Modify
    //                                                ------
    /**
     * {@inheritDoc}
     */
    public void modify(Entity entity, UpdateOption option) {
        doModify(entity, option);
    }

    protected void doModify(Entity entity, UpdateOption option) {
        doUpdate(downcast(entity), downcast(option));
    }

    /**
     * {@inheritDoc}
     */
    public void modifyNonstrict(Entity entity, UpdateOption option) {
        doModifyNonstrict(entity, option);
    }

    protected void doModifyNonstrict(Entity entity, UpdateOption option) {
        if (asDBMeta().hasOptimisticLock()) {
            doUpdateNonstrict(downcast(entity), downcast(option));
        } else {
            doUpdate(downcast(entity), downcast(option));
        }
    }

    // -----------------------------------------------------
    //                                      Insert or Update
    //                                      ----------------
    protected void doInsertOrUpdate(ENTITY entity, InsertOption insertOption, UpdateOption updateOption) {
        assertEntityNotNull(entity);
        helpInsertOrUpdateInternally(entity, insertOption, updateOption);
    }

    protected void doInsertOrUpdateNonstrict(ENTITY entity, InsertOption insertOption, UpdateOption updateOption) {
        assertEntityNotNull(entity);
        helpInsertOrUpdateNonstrictInternally(entity, insertOption, updateOption);
    }

    protected  void helpInsertOrUpdateInternally(RESULT entity, InsertOption insOption,
            UpdateOption updOption) {
        assertEntityNotNull(entity);
        if (helpDetermineInsertOrUpdateDirectInsert(entity)) {
            doCreate(entity, insOption);
            return;
        }
        RuntimeException updateException = null;
        try {
            doModify(entity, updOption);
        } catch (EntityAlreadyUpdatedException e) { // already updated (or means not found)
            updateException = e;
        } catch (EntityAlreadyDeletedException e) { // means not found
            updateException = e;
        } catch (OptimisticLockColumnValueNullException e) { // means insert?
            updateException = e;
        }
        if (updateException == null) {
            return;
        }
        final CB cb = newConditionBean();
        final Set uniqueDrivenProperties = entity.myuniqueDrivenProperties();
        if (uniqueDrivenProperties != null && !uniqueDrivenProperties.isEmpty()) {
            for (String prop : uniqueDrivenProperties) {
                final DBMeta dbmeta = entity.asDBMeta();
                final ColumnInfo columnInfo = dbmeta.findColumnInfo(prop);
                final Object value = columnInfo.read(entity); // already checked in update process
                cb.localCQ().invokeQueryEqual(columnInfo.getColumnDbName(), value);
            }
        } else {
            cb.acceptPrimaryKeyMap(asDBMeta().extractPrimaryKeyMap(entity));
        }
        if (readCount(cb) == 0) { // anyway if not found, insert
            doCreate(entity, insOption);
        } else {
            throw updateException;
        }
    }

    protected  void helpInsertOrUpdateNonstrictInternally(RESULT entity,
            InsertOption insOption, UpdateOption updOption) {
        assertEntityNotNull(entity);
        if (helpDetermineInsertOrUpdateDirectInsert(entity)) {
            doCreate(entity, insOption);
        } else {
            try {
                doModifyNonstrict(entity, updOption);
            } catch (EntityAlreadyDeletedException ignored) { // means not found
                doCreate(entity, insOption);
            }
        }
    }

    protected boolean helpDetermineInsertOrUpdateDirectInsert(Entity entity) {
        final Set uniqueDrivenProperties = entity.myuniqueDrivenProperties();
        if (uniqueDrivenProperties != null && !uniqueDrivenProperties.isEmpty()) {
            return false;
        }
        return !entity.hasPrimaryKeyValue();
    }

    // -----------------------------------------------------
    //                                      Create or Modify
    //                                      ----------------
    /**
     * {@inheritDoc}
     */
    public void createOrModify(Entity entity, InsertOption insertOption,
            UpdateOption updateOption) {
        doCreateOrModify(entity, insertOption, updateOption);
    }

    protected void doCreateOrModify(Entity entity, InsertOption insertOption,
            UpdateOption updateOption) {
        doInsertOrUpdate(downcast(entity), downcast(insertOption), downcast(updateOption));
    }

    /**
     * {@inheritDoc}
     */
    public void createOrModifyNonstrict(Entity entity, InsertOption insertOption,
            UpdateOption updateOption) {
        doCreateOrModifyNonstrict(entity, insertOption, updateOption);
    }

    protected void doCreateOrModifyNonstrict(Entity entity, InsertOption insertOption,
            UpdateOption updateOption) {
        if (asDBMeta().hasOptimisticLock()) {
            doInsertOrUpdateNonstrict(downcast(entity), downcast(insertOption), downcast(updateOption));
        } else {
            doInsertOrUpdate(downcast(entity), downcast(insertOption), downcast(updateOption));
        }
    }

    // -----------------------------------------------------
    //                                                Delete
    //                                                ------
    protected void doDelete(ENTITY entity, final DeleteOption option) {
        assertEntityNotNull(entity);
        prepareDeleteOption(option);
        helpDeleteInternally(entity, option);
    }

    protected void doDeleteNonstrict(ENTITY entity, final DeleteOption option) {
        assertEntityNotNull(entity);
        prepareDeleteOption(option);
        helpDeleteNonstrictInternally(entity, option);
    }

    protected void prepareDeleteOption(DeleteOption option) {
        if (option != null) {
            assertDeleteOptionStatus(option);
        }
    }

    protected  void helpDeleteInternally(RESULT entity, DeleteOption option) {
        assertEntityNotNull(entity);
        assertEntityHasOptimisticLockValue(entity);
        final int deletedCount = delegateDelete(entity, option);
        if (deletedCount == 0) {
            throwUpdateEntityAlreadyDeletedException(entity);
        } else if (deletedCount > 1) {
            throwUpdateEntityDuplicatedException(entity, deletedCount);
        }
    }

    protected  void helpDeleteNonstrictInternally(RESULT entity, DeleteOption option) {
        assertEntityNotNull(entity);
        final int deletedCount = delegateDeleteNonstrict(entity, option);
        if (deletedCount == 0) {
            throwUpdateEntityAlreadyDeletedException(entity);
        } else if (deletedCount > 1) {
            throwUpdateEntityDuplicatedException(entity, deletedCount);
        }
    }

    protected  void helpDeleteNonstrictIgnoreDeletedInternally(RESULT entity,
            DeleteOption option) {
        assertEntityNotNull(entity);
        final int deletedCount = delegateDeleteNonstrict(entity, option);
        if (deletedCount == 0) {
            return;
        } else if (deletedCount > 1) {
            throwUpdateEntityDuplicatedException(entity, deletedCount);
        }
    }

    protected void assertDeleteOptionStatus(DeleteOption option) {
    }

    protected DeleteOption createDeleteOption(WritableOptionCall> opCall) {
        assertDeleteOpCallNotNull(opCall);
        final DeleteOption op = newDeleteOption();
        opCall.callback(op);
        return op;
    }

    protected DeleteOption newDeleteOption() {
        return new DeleteOption();
    }

    protected void assertDeleteOpCallNotNull(WritableOptionCall> opCall) { // for varyingDelete()
        assertObjectNotNull("opLambda (for delete)", opCall);
    }

    // -----------------------------------------------------
    //                                                Remove
    //                                                ------
    /**
     * {@inheritDoc}
     */
    public void remove(Entity entity, DeleteOption option) {
        doRemove(entity, option);
    }

    protected void doRemove(Entity entity, DeleteOption option) {
        doDelete(downcast(entity), downcast(option));
    }

    /**
     * {@inheritDoc}
     */
    public void removeNonstrict(Entity entity, DeleteOption option) {
        doRemoveNonstrict(entity, option);
    }

    protected void doRemoveNonstrict(Entity entity, DeleteOption option) {
        if (asDBMeta().hasOptimisticLock()) {
            doDeleteNonstrict(downcast(entity), downcast(option));
        } else {
            doDelete(downcast(entity), downcast(option));
        }
    }

    // ===================================================================================
    //                                                                        Batch Update
    //                                                                        ============
    // -----------------------------------------------------
    //                                          Batch Insert
    //                                          ------------
    protected int[] doBatchInsert(List entityList, InsertOption option) {
        assertEntityListNotNull(entityList);
        final InsertOption rlop;
        if (option != null) {
            rlop = option;
        } else {
            rlop = createPlainInsertOption();
        }
        prepareBatchInsertOption(entityList, rlop); // required
        return delegateBatchInsert(entityList, rlop);
    }

    protected InsertOption createPlainInsertOption() {
        return newInsertOption();
    }

    protected  void prepareBatchInsertOption(List entityList, InsertOption option) { // might be overridden to set option
        if (isBatchInsertColumnModifiedPropertiesFragmentedDisallowed()) {
            option.xdisallowInsertColumnModifiedPropertiesFragmented(); // default is allowed so use 'disallow' as option
        }
        if (isCompatibleBatchInsertDefaultEveryColumn()) {
            option.xtoBeCompatibleBatchInsertDefaultEveryColumn(); // old style (basically no more use)
        }
        option.xacceptInsertColumnModifiedPropertiesIfNeeds(entityList);
        prepareInsertOption(option);
    }

    protected boolean isBatchInsertColumnModifiedPropertiesFragmentedDisallowed() {
        return false; // might be overridden by generator option 
    }

    protected boolean isCompatibleBatchInsertDefaultEveryColumn() {
        return false; // might be overridden by generator option
    }

    // -----------------------------------------------------
    //                                           Lump Create
    //                                           -----------
    /**
     * {@inheritDoc}
     */
    public int[] lumpCreate(List entityList, InsertOption option) {
        @SuppressWarnings("unchecked")
        final List castList = (List) entityList;
        return doLumpCreate(castList, option);
    }

    protected int[] doLumpCreate(List entityList, InsertOption option) {
        return doBatchInsert(downcast(entityList), downcast(option));
    }

    // -----------------------------------------------------
    //                                          Batch Update
    //                                          ------------
    protected int[] doBatchUpdate(List entityList, UpdateOption option) {
        assertEntityListNotNull(entityList);
        final UpdateOption rlop;
        if (option != null) {
            rlop = option;
        } else {
            rlop = createPlainUpdateOption();
        }
        prepareBatchUpdateOption(entityList, rlop); // required
        return delegateBatchUpdate(entityList, rlop);
    }

    protected int[] doBatchUpdateNonstrict(List entityList, UpdateOption option) {
        assertEntityListNotNull(entityList);
        final UpdateOption rlop;
        if (option != null) {
            rlop = option;
        } else {
            rlop = createPlainUpdateOption();
        }
        prepareBatchUpdateOption(entityList, rlop);
        return delegateBatchUpdateNonstrict(entityList, rlop);
    }

    protected UpdateOption createPlainUpdateOption() {
        return new UpdateOption();
    }

    protected UpdateOption createSpecifiedUpdateOption(SpecifyQuery updateColumnSpec) {
        assertUpdateColumnSpecificationNotNull(updateColumnSpec);
        final UpdateOption option = createPlainUpdateOption();
        option.specify(updateColumnSpec);
        return option;
    }

    protected void assertUpdateColumnSpecificationNotNull(SpecifyQuery updateColumnSpec) {
        assertObjectNotNull("updateColumnSpec", updateColumnSpec);
    }

    protected  void prepareBatchUpdateOption(List entityList, UpdateOption option) {
        if (isBatchUpdateColumnModifiedPropertiesFragmentedAllowed()) {
            option.xallowUpdateColumnModifiedPropertiesFragmented(); // default is disallowed so use 'allow' as option
        }
        if (isCompatibleBatchUpdateDefaultEveryColumn()) {
            option.xtoBeCompatibleBatchUpdateDefaultEveryColumn(); // old style (basically no more use)
        }
        option.xacceptUpdateColumnModifiedPropertiesIfNeeds(entityList);
        prepareUpdateOption(option);
    }

    protected boolean isBatchUpdateColumnModifiedPropertiesFragmentedAllowed() {
        return false; // might be overridden by generator option
    }

    protected boolean isCompatibleBatchUpdateDefaultEveryColumn() {
        return false; // might be overridden by generator option
    }

    // -----------------------------------------------------
    //                                           Lump Modify
    //                                           -----------
    /**
     * {@inheritDoc}
     */
    public int[] lumpModify(List entityList, UpdateOption option) {
        @SuppressWarnings("unchecked")
        final List castList = (List) entityList;
        return doLumpModify(castList, option);
    }

    protected int[] doLumpModify(List entityList, UpdateOption option) {
        return doBatchUpdate(downcast(entityList), downcast(option));
    }

    /**
     * {@inheritDoc}
     */
    public int[] lumpModifyNonstrict(List entityList, UpdateOption option) {
        @SuppressWarnings("unchecked")
        final List castList = (List) entityList;
        return doLumpModifyNonstrict(castList, option);
    }

    protected int[] doLumpModifyNonstrict(List entityList, UpdateOption option) {
        if (asDBMeta().hasOptimisticLock()) {
            return doBatchUpdateNonstrict(downcast(entityList), downcast(option));
        } else {
            return doBatchUpdate(downcast(entityList), downcast(option));
        }
    }

    // -----------------------------------------------------
    //                                          Batch Delete
    //                                          ------------
    protected int[] doBatchDelete(List entityList, DeleteOption option) {
        assertEntityListNotNull(entityList);
        prepareDeleteOption(option);
        return delegateBatchDelete(entityList, option);
    }

    protected int[] doBatchDeleteNonstrict(List entityList, DeleteOption option) {
        assertEntityListNotNull(entityList);
        prepareDeleteOption(option);
        return delegateBatchDeleteNonstrict(entityList, option);
    }

    // -----------------------------------------------------
    //                                           Lump Remove
    //                                           -----------
    /**
     * {@inheritDoc}
     */
    public int[] lumpRemove(List entityList, DeleteOption option) {
        @SuppressWarnings("unchecked")
        final List castList = (List) entityList;
        return doLumpRemove(castList, option);
    }

    protected int[] doLumpRemove(List entityList, DeleteOption option) {
        return doBatchDelete(downcast(entityList), downcast(option));
    }

    /**
     * {@inheritDoc}
     */
    public int[] lumpRemoveNonstrict(List entityList, DeleteOption option) {
        @SuppressWarnings("unchecked")
        final List castList = (List) entityList;
        return doLumpRemoveNonstrict(castList, option);
    }

    protected int[] doLumpRemoveNonstrict(List entityList, DeleteOption option) {
        if (asDBMeta().hasOptimisticLock()) {
            return doBatchDeleteNonstrict(downcast(entityList), downcast(option));
        } else {
            return doBatchDelete(downcast(entityList), downcast(option));
        }
    }

    // =====================================================================================
    //                                                                          Query Update
    //                                                                          ============
    // -----------------------------------------------------
    //                                          Query Insert
    //                                          ------------
    protected int doQueryInsert(QueryInsertSetupper setupper, InsertOption option) {
        assertObjectNotNull("setupper", setupper);
        prepareInsertOption(option);
        final ENTITY et = newEntity();
        final CB cb = createCBForQueryInsert();
        return delegateQueryInsert(et, cb, setupper.setup(et, cb), option);
    }

    protected CB createCBForQueryInsert() {
        final CB cb = newConditionBean();
        cb.xsetupForQueryInsert();
        return cb;
    }

    // -----------------------------------------------------
    //                                          Range Create
    //                                          ------------
    /**
     * {@inheritDoc}
     */
    public int rangeCreate(QueryInsertSetupper setupper,
            InsertOption option) {
        return doRangeCreate(setupper, option);
    }

    protected int doRangeCreate(QueryInsertSetupper setupper,
            InsertOption option) {
        return doQueryInsert(downcast(setupper), downcast(option));
    }

    // -----------------------------------------------------
    //                                          Query Update
    //                                          ------------
    protected int doQueryUpdate(ENTITY entity, CB cb, UpdateOption option) {
        assertObjectNotNull("${myEntityVariableName}", entity);
        assertCBStateValid(cb);
        prepareUpdateOption(option);
        return checkCountBeforeQueryUpdateIfNeeds(cb) ? delegateQueryUpdate(entity, cb, option) : 0;
    }

    /**
     * Check record count before QueryUpdate if it needs. (against MySQL's deadlock of next-key lock)
     * @param cb The condition-bean for QueryUpdate. (NotNull)
     * @return true if record count exists or no check.
     */
    protected boolean checkCountBeforeQueryUpdateIfNeeds(ConditionBean cb) {
        final boolean countExists;
        if (cb.isQueryUpdateCountPreCheck()) {
            countExists = readCount(cb) > 0;
        } else {
            countExists = true; // means no check
        }
        return countExists;
    }

    // -----------------------------------------------------
    //                                          Range Modify
    //                                          ------------
    /**
     * {@inheritDoc}
     */
    public int rangeModify(Entity entity, ConditionBean cb, UpdateOption option) {
        return doRangeModify(entity, cb, option);
    }

    protected int doRangeModify(Entity entity, ConditionBean cb, UpdateOption option) {
        return doQueryUpdate(downcast(entity), downcast(cb), downcast(option));
    }

    // -----------------------------------------------------
    //                                          Query Delete
    //                                          ------------
    protected int doQueryDelete(CB cb, DeleteOption option) {
        assertCBStateValid(cb);
        prepareDeleteOption(option);
        return checkCountBeforeQueryUpdateIfNeeds(cb) ? delegateQueryDelete(cb, option) : 0;
    }

    // -----------------------------------------------------
    //                                          Range Remove
    //                                          ------------
    /**
     * {@inheritDoc}
     */
    public int rangeRemove(ConditionBean cb, DeleteOption option) {
        return doRangeRemove(cb, option);
    }

    protected int doRangeRemove(ConditionBean cb, DeleteOption option) {
        return doQueryDelete(downcast(cb), downcast(option));
    }

    // ===================================================================================
    //                                                                      Delegate Entry
    //                                                                      ==============
    // -----------------------------------------------------
    //                                         Entity Update
    //                                         -------------
    protected int delegateInsert(Entity entity, InsertOption option) {
        if (!processBeforeInsert(entity, option)) {
            return 0;
        }
        return invoke(createInsertEntityCommand(entity, option));
    }

    protected int delegateUpdate(Entity entity, UpdateOption option) {
        if (!processBeforeUpdate(entity, option)) {
            return 0;
        }
        if (asDBMeta().hasOptimisticLock()) {
            return invoke(createUpdateEntityCommand(entity, option));
        } else {
            return delegateUpdateNonstrict(entity, option);
        }
    }

    protected int delegateUpdateNonstrict(Entity entity, UpdateOption option) {
        if (!processBeforeUpdate(entity, option)) {
            return 0;
        }
        return invoke(createUpdateNonstrictEntityCommand(entity, option));
    }

    protected int delegateDelete(Entity entity, DeleteOption option) {
        if (!processBeforeDelete(entity, option)) {
            return 0;
        }
        if (asDBMeta().hasOptimisticLock()) {
            return invoke(createDeleteEntityCommand(entity, option));
        } else {
            return delegateDeleteNonstrict(entity, option);
        }
    }

    protected int delegateDeleteNonstrict(Entity entity, DeleteOption option) {
        if (!processBeforeDelete(entity, option)) {
            return 0;
        }
        return invoke(createDeleteNonstrictEntityCommand(entity, option));
    }

    // -----------------------------------------------------
    //                                          Batch Update
    //                                          ------------
    protected int[] delegateBatchInsert(List entityList, InsertOption option) {
        if (entityList.isEmpty()) {
            return new int[] {};
        }
        return invoke(createBatchInsertCommand(processBatchInternally(entityList, option), option));
    }

    protected int[] delegateBatchUpdate(List entityList, UpdateOption option) {
        if (entityList.isEmpty()) {
            return new int[] {};
        }
        if (asDBMeta().hasOptimisticLock()) {
            return invoke(createBatchUpdateCommand(processBatchInternally(entityList, option, false), option));
        } else {
            return delegateBatchUpdateNonstrict(entityList, option);
        }
    }

    protected int[] delegateBatchUpdateNonstrict(List entityList, UpdateOption option) {
        if (entityList.isEmpty()) {
            return new int[] {};
        }
        return invoke(createBatchUpdateNonstrictCommand(processBatchInternally(entityList, option, true), option));
    }

    protected int[] delegateBatchDelete(List entityList, DeleteOption option) {
        if (entityList.isEmpty()) {
            return new int[] {};
        }
        if (asDBMeta().hasOptimisticLock()) {
            return invoke(createBatchDeleteCommand(processBatchInternally(entityList, option, false), option));
        } else {
            return delegateBatchDeleteNonstrict(entityList, option);
        }
    }

    protected int[] delegateBatchDeleteNonstrict(List entityList, DeleteOption option) {
        if (entityList.isEmpty()) {
            return new int[] {};
        }
        return invoke(createBatchDeleteNonstrictCommand(processBatchInternally(entityList, option, true), option));
    }

    // -----------------------------------------------------
    //                                          Query Update
    //                                          ------------
    protected int delegateQueryInsert(Entity entity, ConditionBean inCB, ConditionBean resCB, InsertOption option) {
        if (!processBeforeQueryInsert(entity, inCB, resCB, option)) {
            return 0;
        }
        return invoke(createQueryInsertCBCommand(entity, inCB, resCB, option));
    }

    protected int delegateQueryUpdate(Entity entity, ConditionBean cb, UpdateOption option) {
        if (!processBeforeQueryUpdate(entity, cb, option)) {
            return 0;
        }
        return invoke(createQueryUpdateCBCommand(entity, cb, option));
    }

    protected int delegateQueryDelete(ConditionBean cb, DeleteOption option) {
        if (!processBeforeQueryDelete(cb, option)) {
            return 0;
        }
        return invoke(createQueryDeleteCBCommand(cb, option));
    }

    // =====================================================================================
    //                                                                        Process Method
    //                                                                        ==============
    // -----------------------------------------------------
    //                                                Insert
    //                                                ------
    /**
     * Process before insert. 
* You can stop the process by your extension. * @param entity The entity for insert. (NotNull) * @param option The option of insert. (NullAllowed) * @return Execution Determination. (true: execute / false: non) */ protected boolean processBeforeInsert(Entity entity, InsertOption option) { assertEntityNotNull(entity); // primary key is checked later frameworkFilterEntityOfInsert(entity, option); filterEntityOfInsert(entity, option); assertEntityOfInsert(entity, option); // check primary key after filtering at an insert process // because a primary key value may be set in filtering process // (for example, sequence) if (!entity.asDBMeta().hasIdentity()) { // identity does not need primary key value here assertEntityNotNullAndHasPrimaryKeyValue(entity); } return true; } /** * Process before query-insert.
* You can stop the process by your extension. * @param entity The entity for query-insert. (NotNull) * @param intoCB The condition-bean for inserted table. (NotNull) * @param resourceCB The condition-bean for resource table. (NotNull) * @param option The option of insert. (NullAllowed) * @return Execution Determination. (true: execute / false: non) */ protected boolean processBeforeQueryInsert(Entity entity, ConditionBean intoCB, ConditionBean resourceCB, InsertOption option) { assertEntityNotNull(entity); // query-insert doesn't need to check primary key assertObjectNotNull("intoCB", intoCB); if (resourceCB == null) { String msg = "The set-upper of query-insert should return a condition-bean for resource table:"; msg = msg + " inserted=" + entity.asTableDbName(); throw new IllegalConditionBeanOperationException(msg); } frameworkFilterEntityOfInsert(entity, option); setupExclusiveControlColumnOfQueryInsert(entity); filterEntityOfInsert(entity, option); assertEntityOfInsert(entity, option); return true; } protected void setupExclusiveControlColumnOfQueryInsert(Entity entity) { final DBMeta dbmeta = asDBMeta(); if (dbmeta.hasVersionNo()) { final ColumnInfo columnInfo = dbmeta.getVersionNoColumnInfo(); columnInfo.write(entity, InsertOption.VERSION_NO_FIRST_VALUE); } if (dbmeta.hasUpdateDate()) { final ColumnInfo columnInfo = dbmeta.getUpdateDateColumnInfo(); columnInfo.write(entity, ResourceContext.getAccessTimestamp()); } } /** * {Framework Method} Filter the entity of insert. * @param entity The entity for insert. (NotNull) * @param option The option of insert. (NullAllowed) */ protected void frameworkFilterEntityOfInsert(Entity entity, InsertOption option) { injectSequenceToPrimaryKeyIfNeeds(entity); setupCommonColumnOfInsertIfNeeds(entity, option); } /** * Set up common columns of insert if it needs. * @param entity The entity for insert. (NotNull) * @param option The option of insert. (NullAllowed) */ protected void setupCommonColumnOfInsertIfNeeds(Entity entity, InsertOption option) { if (option != null && option.isCommonColumnAutoSetupDisabled()) { return; } final CommonColumnAutoSetupper setupper = getCommonColumnAutoSetupper(); assertCommonColumnAutoSetupperNotNull(); setupper.handleCommonColumnOfInsertIfNeeds(entity); } private void assertCommonColumnAutoSetupperNotNull() { if (_commonColumnAutoSetupper != null) { return; } final ExceptionMessageBuilder br = createExceptionMessageBuilder(); br.addNotice("Not found the auto set-upper of common column in the behavior!"); br.addItem("Advice"); br.addElement("Please confirm the definition of the set-upper at your component configuration of DBFlute."); br.addItem("Behavior"); br.addElement("Behavior for " + asTableDbName()); br.addItem("Attribute"); br.addElement("behaviorCommandInvoker : " + _behaviorCommandInvoker); br.addElement("behaviorSelector : " + _behaviorSelector); br.addElement("commonColumnAutoSetupper : " + _commonColumnAutoSetupper); final String msg = br.buildExceptionMessage(); throw new IllegalBehaviorStateException(msg); } /** * Filter the entity of insert. (for extension) * @param entity The entity for insert. (NotNull) * @param option The option of insert. (NullAllowed) */ protected void filterEntityOfInsert(Entity entity, InsertOption option) { } /** * Assert the entity of insert. (for extension) * @param entity The entity for insert. (NotNull) * @param option The option of insert. (NullAllowed) */ protected void assertEntityOfInsert(Entity entity, InsertOption option) { } // ----------------------------------------------------- // Update // ------ /** * Process before update.
* You can stop the process by your extension. * @param entity The entity for update that has primary key. (NotNull) * @param option The option of update. (NullAllowed) * @return Execution Determination. (true: execute / false: non) */ protected boolean processBeforeUpdate(Entity entity, UpdateOption option) { assertEntityNotNullAndHasPrimaryKeyValue(entity); frameworkFilterEntityOfUpdate(entity, option); filterEntityOfUpdate(entity, option); assertEntityOfUpdate(entity, option); return true; } /** * Process before query-update.
* You can stop the process by your extension. * @param entity The entity for update that is not needed primary key. (NotNull) * @param cb The condition-bean for query. (NotNull) * @param option The option of update. (NullAllowed) * @return Execution Determination. (true: execute / false: non) */ protected boolean processBeforeQueryUpdate(Entity entity, ConditionBean cb, UpdateOption option) { assertEntityNotNull(entity); // query-update doesn't need to check primary key assertCBStateValid(cb); frameworkFilterEntityOfUpdate(entity, option); filterEntityOfUpdate(entity, option); assertEntityOfUpdate(entity, option); assertQueryUpdateStatus(entity, cb, option); return true; } /** * {Framework Method} Filter the entity of update. * @param entity The entity for update. (NotNull) * @param option The option of update. (NullAllowed) */ protected void frameworkFilterEntityOfUpdate(Entity entity, UpdateOption option) { setupCommonColumnOfUpdateIfNeeds(entity, option); } /** * Set up common columns of update if it needs. * @param entity The entity for update. (NotNull) * @param option The option of update. (NullAllowed) */ protected void setupCommonColumnOfUpdateIfNeeds(Entity entity, UpdateOption option) { if (option != null && option.isCommonColumnAutoSetupDisabled()) { return; } final CommonColumnAutoSetupper setupper = getCommonColumnAutoSetupper(); assertCommonColumnAutoSetupperNotNull(); setupper.handleCommonColumnOfUpdateIfNeeds(entity); } /** * Filter the entity of update. (for extension) * @param entity The entity for update. (NotNull) * @param option The option of update. (NullAllowed) */ protected void filterEntityOfUpdate(Entity entity, UpdateOption option) { } /** * Assert the entity of update. (for extension) * @param entity The entity for update. (NotNull) * @param option The option of update. (NullAllowed) */ protected void assertEntityOfUpdate(Entity entity, UpdateOption option) { } /** * Assert that the query-update is legal status. * @param entity The entity for query-update. (NotNull) * @param cb The condition-bean for query-update. (NotNull) * @param option The option of update. (NullAllowed) */ protected void assertQueryUpdateStatus(Entity entity, ConditionBean cb, UpdateOption option) { if (option != null && option.isNonQueryUpdateAllowed()) { return; } if (cb.hasSelectAllPossible()) { createBhvExThrower().throwNonQueryUpdateNotAllowedException(entity, cb, option); } } // ----------------------------------------------------- // Delete // ------ /** * Process before delete.
* You can stop the process by your extension. * @param entity The entity for delete that has primary key. (NotNull) * @param option The option of delete. (NullAllowed) * @return Execution Determination. (true: execute / false: non) */ protected boolean processBeforeDelete(Entity entity, DeleteOption option) { assertEntityNotNullAndHasPrimaryKeyValue(entity); frameworkFilterEntityOfDelete(entity, option); filterEntityOfDelete(entity, option); assertEntityOfDelete(entity, option); return true; } /** * Process before query-delete.
* You can stop the process by your extension. * @param cb The condition-bean for query. (NotNull) * @param option The option of delete. (NullAllowed) * @return Execution Determination. (true: execute / false: non) */ protected boolean processBeforeQueryDelete(ConditionBean cb, DeleteOption option) { assertCBStateValid(cb); assertQueryDeleteStatus(cb, option); return true; } /** * {Framework Method} Filter the entity of delete. {not called if query-delete} * @param entity The entity for delete that has primary key. (NotNull) * @param option The option of delete. (NullAllowed) */ protected void frameworkFilterEntityOfDelete(Entity entity, DeleteOption option) { } /** * Filter the entity of delete. (for extension) {not called if query-delete} * @param entity The entity for delete that has primary key. (NotNull) * @param option The option of delete. (NullAllowed) */ protected void filterEntityOfDelete(Entity entity, DeleteOption option) { } /** * Assert the entity of delete. (for extension) {not called if query-delete} * @param entity The entity for delete that has primary key. (NotNull) * @param option The option of delete. (NullAllowed) */ protected void assertEntityOfDelete(Entity entity, DeleteOption option) { } /** * Assert that the query-delete is legal status. * @param cb The condition-bean for query-delete. (NotNull) * @param option The option of delete. (NullAllowed) */ protected void assertQueryDeleteStatus(ConditionBean cb, DeleteOption option) { if (option != null && option.isNonQueryDeleteAllowed()) { return; } if (cb.hasSelectAllPossible()) { createBhvExThrower().throwNonQueryDeleteNotAllowedException(cb, option); } } // ----------------------------------------------------- // Common // ------ protected void injectSequenceToPrimaryKeyIfNeeds(Entity entity) { final DBMeta dbmeta = entity.asDBMeta(); if (!dbmeta.hasSequence() || dbmeta.hasCompoundPrimaryKey() || entity.hasPrimaryKeyValue()) { return; } // basically property(column) type is same as next value type // so there is NOT type conversion cost when writing to the entity dbmeta.getPrimaryUniqueInfo().getFirstColumn().write(entity, readNextVal()); } protected void assertEntityHasOptimisticLockValue(Entity entity) { assertEntityHasVersionNoValue(entity); assertEntityHasUpdateDateValue(entity); } protected void assertEntityHasVersionNoValue(Entity entity) { if (!asDBMeta().hasVersionNo()) { return; } if (hasVersionNoValue(entity)) { return; } throwVersionNoValueNullException(entity); } protected void throwVersionNoValueNullException(Entity entity) { createBhvExThrower().throwVersionNoValueNullException(entity); } protected void assertEntityHasUpdateDateValue(Entity entity) { if (!asDBMeta().hasUpdateDate()) { return; } if (hasUpdateDateValue(entity)) { return; } throwUpdateDateValueNullException(entity); } protected void throwUpdateDateValueNullException(Entity entity) { createBhvExThrower().throwUpdateDateValueNullException(entity); } // ----------------------------------------------------- // Batch // ----- protected List processBatchInternally(List entityList, InsertOption option) { assertObjectNotNull("entityList", entityList); final List filteredList = new ArrayList(); for (ELEMENT entity : entityList) { if (!processBeforeInsert(entity, option)) { continue; } filteredList.add(entity); } return filteredList; } protected List processBatchInternally(List entityList, UpdateOption option, boolean nonstrict) { assertObjectNotNull("entityList", entityList); final List filteredList = new ArrayList(); for (ELEMENT entity : entityList) { if (!processBeforeUpdate(entity, option)) { continue; } if (!nonstrict) { assertEntityHasOptimisticLockValue(entity); } filteredList.add(entity); } return filteredList; } protected List processBatchInternally(List entityList, DeleteOption option, boolean nonstrict) { assertObjectNotNull("entityList", entityList); final List filteredList = new ArrayList(); for (ELEMENT entity : entityList) { if (!processBeforeDelete(entity, option)) { continue; } if (!nonstrict) { assertEntityHasOptimisticLockValue(entity); } filteredList.add(entity); } return filteredList; } // =================================================================================== // Behavior Command // ================ // ----------------------------------------------------- // Basic // ----- // an insert command creation is defined on the readable interface for non-primary key value protected UpdateEntityCommand createUpdateEntityCommand(Entity entity, UpdateOption option) { assertBehaviorCommandInvoker("createUpdateEntityCommand"); final UpdateEntityCommand cmd = newUpdateEntityCommand(); xsetupEntityCommand(cmd, entity); cmd.setUpdateOption(option); return cmd; } protected UpdateEntityCommand newUpdateEntityCommand() { return new UpdateEntityCommand(); } protected UpdateNonstrictEntityCommand createUpdateNonstrictEntityCommand(Entity entity, UpdateOption option) { assertBehaviorCommandInvoker("createUpdateNonstrictEntityCommand"); final UpdateNonstrictEntityCommand cmd = newUpdateNonstrictEntityCommand(); xsetupEntityCommand(cmd, entity); cmd.setUpdateOption(option); return cmd; } protected UpdateNonstrictEntityCommand newUpdateNonstrictEntityCommand() { return new UpdateNonstrictEntityCommand(); } protected DeleteEntityCommand createDeleteEntityCommand(Entity entity, DeleteOption option) { assertBehaviorCommandInvoker("createDeleteEntityCommand"); final DeleteEntityCommand cmd = newDeleteEntityCommand(); xsetupEntityCommand(cmd, entity); cmd.setDeleteOption(option); return cmd; } protected DeleteEntityCommand newDeleteEntityCommand() { return new DeleteEntityCommand(); } protected DeleteNonstrictEntityCommand createDeleteNonstrictEntityCommand(Entity entity, DeleteOption option) { assertBehaviorCommandInvoker("createDeleteNonstrictEntityCommand"); final DeleteNonstrictEntityCommand cmd = newDeleteNonstrictEntityCommand(); xsetupEntityCommand(cmd, entity); cmd.setDeleteOption(option); return cmd; } protected DeleteNonstrictEntityCommand newDeleteNonstrictEntityCommand() { return new DeleteNonstrictEntityCommand(); } // ----------------------------------------------------- // Batch // ----- protected BatchInsertCommand createBatchInsertCommand(List entityList, InsertOption option) { assertBehaviorCommandInvoker("createBatchInsertCommand"); final BatchInsertCommand cmd = newBatchInsertCommand(); xsetupListEntityCommand(cmd, entityList); cmd.setInsertOption(option); return cmd; } protected BatchInsertCommand newBatchInsertCommand() { return new BatchInsertCommand(); } protected BatchUpdateCommand createBatchUpdateCommand(List entityList, UpdateOption option) { assertBehaviorCommandInvoker("createBatchUpdateCommand"); final BatchUpdateCommand cmd = newBatchUpdateCommand(); xsetupListEntityCommand(cmd, entityList); cmd.setUpdateOption(option); return cmd; } protected BatchUpdateCommand newBatchUpdateCommand() { return new BatchUpdateCommand(); } protected BatchUpdateNonstrictCommand createBatchUpdateNonstrictCommand(List entityList, UpdateOption option) { assertBehaviorCommandInvoker("createBatchUpdateNonstrictCommand"); final BatchUpdateNonstrictCommand cmd = newBatchUpdateNonstrictCommand(); xsetupListEntityCommand(cmd, entityList); cmd.setUpdateOption(option); return cmd; } protected BatchUpdateNonstrictCommand newBatchUpdateNonstrictCommand() { return new BatchUpdateNonstrictCommand(); } protected BatchDeleteCommand createBatchDeleteCommand(List entityList, DeleteOption option) { assertBehaviorCommandInvoker("createBatchDeleteCommand"); final BatchDeleteCommand cmd = newBatchDeleteCommand(); xsetupListEntityCommand(cmd, entityList); cmd.setDeleteOption(option); return cmd; } protected BatchDeleteCommand newBatchDeleteCommand() { return new BatchDeleteCommand(); } protected BatchDeleteNonstrictCommand createBatchDeleteNonstrictCommand(List entityList, DeleteOption option) { assertBehaviorCommandInvoker("createBatchDeleteNonstrictCommand"); final BatchDeleteNonstrictCommand cmd = newBatchDeleteNonstrictCommand(); xsetupListEntityCommand(cmd, entityList); cmd.setDeleteOption(option); return cmd; } protected BatchDeleteNonstrictCommand newBatchDeleteNonstrictCommand() { return new BatchDeleteNonstrictCommand(); } /** * @param command The command of behavior. (NotNull) * @param entityList The list of entity. (NotNull, NotEmpty) */ protected void xsetupListEntityCommand(AbstractListEntityCommand command, List entityList) { if (entityList.isEmpty()) { String msg = "The argument 'entityList' should not be empty: " + entityList; throw new IllegalStateException(msg); } command.setTableDbName(asTableDbName()); _behaviorCommandInvoker.injectComponentProperty(command); command.setEntityType(entityList.get(0).getClass()); // *The list should not be empty! command.setEntityList(entityList); } // ----------------------------------------------------- // Query // ----- protected QueryInsertCBCommand createQueryInsertCBCommand(Entity entity, ConditionBean intoCB, ConditionBean resourceCB, InsertOption option) { assertBehaviorCommandInvoker("createQueryInsertCBCommand"); final QueryInsertCBCommand cmd = new QueryInsertCBCommand(); cmd.setTableDbName(asTableDbName()); _behaviorCommandInvoker.injectComponentProperty(cmd); cmd.setEntity(entity); cmd.setIntoConditionBean(intoCB); cmd.setConditionBean(resourceCB); cmd.setInsertOption(option); return cmd; } protected QueryUpdateCBCommand createQueryUpdateCBCommand(Entity entity, ConditionBean cb, UpdateOption option) { assertBehaviorCommandInvoker("createQueryUpdateCBCommand"); final QueryUpdateCBCommand cmd = new QueryUpdateCBCommand(); cmd.setTableDbName(asTableDbName()); _behaviorCommandInvoker.injectComponentProperty(cmd); cmd.setEntity(entity); cmd.setConditionBean(cb); cmd.setUpdateOption(option); return cmd; } protected QueryDeleteCBCommand createQueryDeleteCBCommand(ConditionBean cb, DeleteOption option) { assertBehaviorCommandInvoker("createQueryDeleteCBCommand"); final QueryDeleteCBCommand cmd = new QueryDeleteCBCommand(); cmd.setTableDbName(asTableDbName()); _behaviorCommandInvoker.injectComponentProperty(cmd); cmd.setConditionBean(cb); cmd.setDeleteOption(option); return cmd; } // =================================================================================== // Type Helper // =========== @SuppressWarnings("unchecked") protected InsertOption downcast(InsertOption option) { return (InsertOption) option; } @SuppressWarnings("unchecked") protected UpdateOption downcast(UpdateOption option) { return (UpdateOption) option; } @SuppressWarnings("unchecked") protected DeleteOption downcast(DeleteOption option) { return (DeleteOption) option; } @SuppressWarnings("unchecked") protected QueryInsertSetupper downcast(QueryInsertSetupper setupper) { return (QueryInsertSetupper) setupper; } // =================================================================================== // Accessor // ======== /** * Get the auto set-upper of common column. * @return The auto set-upper of common column. (NullAllowed: But normally NotNull) */ protected CommonColumnAutoSetupper getCommonColumnAutoSetupper() { return _commonColumnAutoSetupper; } /** * Set the auto set-upper of common column. * @param commonColumnAutoSetupper The auto set-upper of common column. (NotNull) */ public void setCommonColumnAutoSetupper(CommonColumnAutoSetupper commonColumnAutoSetupper) { this._commonColumnAutoSetupper = commonColumnAutoSetupper; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy