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

com.landawn.abacus.jdbc.dao.Dao Maven / Gradle / Ivy

/*
 * Copyright (c) 2021, Haiyang Li.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.landawn.abacus.jdbc.dao;

import java.sql.CallableStatement;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.Executor;
import java.util.function.Consumer;

import com.landawn.abacus.annotation.Beta;
import com.landawn.abacus.annotation.Internal;
import com.landawn.abacus.annotation.LazyEvaluation;
import com.landawn.abacus.condition.Condition;
import com.landawn.abacus.condition.ConditionFactory;
import com.landawn.abacus.condition.ConditionFactory.CF;
import com.landawn.abacus.exception.DuplicatedResultException;
import com.landawn.abacus.jdbc.AbstractQuery;
import com.landawn.abacus.jdbc.CallableQuery;
import com.landawn.abacus.jdbc.IsolationLevel;
import com.landawn.abacus.jdbc.Jdbc;
import com.landawn.abacus.jdbc.Jdbc.Columns.ColumnOne;
import com.landawn.abacus.jdbc.JdbcUtil;
import com.landawn.abacus.jdbc.NamedQuery;
import com.landawn.abacus.jdbc.PreparedQuery;
import com.landawn.abacus.jdbc.s;
import com.landawn.abacus.jdbc.annotation.NonDBOperation;
import com.landawn.abacus.parser.ParserUtil;
import com.landawn.abacus.parser.ParserUtil.PropInfo;
import com.landawn.abacus.type.Type;
import com.landawn.abacus.util.AsyncExecutor;
import com.landawn.abacus.util.ContinuableFuture;
import com.landawn.abacus.util.DataSet;
import com.landawn.abacus.util.N;
import com.landawn.abacus.util.NoCachingNoUpdating.DisposableObjArray;
import com.landawn.abacus.util.ParsedSql;
import com.landawn.abacus.util.QueryUtil;
import com.landawn.abacus.util.SQLBuilder;
import com.landawn.abacus.util.SQLMapper;
import com.landawn.abacus.util.Throwables;
import com.landawn.abacus.util.u.Nullable;
import com.landawn.abacus.util.u.Optional;
import com.landawn.abacus.util.u.OptionalBoolean;
import com.landawn.abacus.util.u.OptionalByte;
import com.landawn.abacus.util.u.OptionalChar;
import com.landawn.abacus.util.u.OptionalDouble;
import com.landawn.abacus.util.u.OptionalFloat;
import com.landawn.abacus.util.u.OptionalInt;
import com.landawn.abacus.util.u.OptionalLong;
import com.landawn.abacus.util.u.OptionalShort;
import com.landawn.abacus.util.stream.Stream;

/**
 * Performance Tips:
 * 
  • Avoid unnecessary/repeated database calls.
  • *
  • Only fetch the columns you need or update the columns you want.
  • *
  • Index is the key point in a lot of database performance issues.
  • * *
    * * This interface is designed to share/manager SQL queries by Java APIs/methods with static parameter types and return type, while hiding the SQL scripts. * It's a gift from nature and created by thoughts. * *
    * Note: Setting parameters by 'ParametersSetter' or Retrieving result/record by 'ResultExtractor/BiResultExtractor/RowMapper/BiRowMapper' is not enabled at present. * *
    * *
  • The SQL operations/methods should be annotated with SQL scripts by {@code @Select/@Insert/@Update/@Delete/@NamedSelect/@NamedInsert/@NamedUpdate/@NamedDelete}.
  • * *
  • The Order of the parameters in the method should be consistent with parameter order in SQL scripts for parameterized SQL. * For named parameterized SQL, the parameters must be binded with names through {@code @Bind}, or {@code Map/Entity} with getter/setter methods.
  • * *
  • SQL parameters can be set through input method parameters(by multiple parameters or a {@code Collection}, or a {@code Map/Entity} for named sql), or by {@code JdbcUtil.ParametersSetter}.
  • * *
  • {@code ResultExtractor/BiResultExtractor/RowMapper/BiRowMapper} can be specified by the last parameter of the method.
  • * *
  • The return type of the method must be same as the return type of {@code ResultExtractor/BiResultExtractor} if it's specified by the last parameter of the method.
  • * *
  • The return type of update/delete operations only can int/Integer/long/Long/boolean/Boolean/void. If it's long/Long, {@code PreparedQuery#largeUpdate()} will be called, * otherwise, {@code PreparedQuery#update()} will be called.
  • * *
  • Remember declaring {@code throws SQLException} in the method.
  • *
    *
  • Which underline {@code PreparedQuery/CallableQuery} method to call for SQL methods/operations annotated with {@code @Select/@NamedSelect}: *
      *
    • If {@code ResultExtractor/BiResultExtractor} is specified by the last parameter of the method, {@code PreparedQuery#query(ResultExtractor/BiResultExtractor)} will be called.
    • *
    • Or else if {@code RowMapper/BiRowMapper} is specified by the last parameter of the method:
    • *
        *
      • If the return type of the method is {@code List} and one of below conditions is matched, {@code PreparedQuery#list(RowMapper/BiRowMapper)} will be called:
      • *
          *
        • The return type of the method is raw {@code List} without parameterized type, and the method name doesn't start with {@code "get"/"findFirst"/"findOne"/"findOnlyOne"}.
        • *
        *
          *
        • The last parameter type is raw {@code RowMapper/BiRowMapper} without parameterized type, and the method name doesn't start with {@code "get"/"findFirst"/"findOne"/"findOnlyOne"}.
        • *
        *
          *
        • The return type of the method is generic {@code List} with parameterized type and The last parameter type is generic {@code RowMapper/BiRowMapper} with parameterized types, but They're not same.
        • *
        *
      *
        *
      • Or else if the return type of the method is {@code Stream/Stream}, {@code PreparedQuery#stream(RowMapper/BiRowMapper)} will be called.
      • *
      *
        *
      • Or else if the return type of the method is {@code Optional}, {@code PreparedQuery#findFirst(RowMapper/BiRowMapper)} will be called.
      • *
      *
        *
      • Or else, {@code PreparedQuery#findFirst(RowMapper/BiRowMapper).orElse(N.defaultValueOf(returnType))} will be called.
      • *
      *
    • Or else:
    • *
        *
      • If the return type of the method is {@code DataSet}, {@code PreparedQuery#query()} will be called.
      • *
      *
        *
      • Or else if the return type of the method is {@code Stream/Stream}, {@code PreparedQuery#stream(Class)} will be called.
      • *
      *
        *
      • Or else if the return type of the method is {@code Map} or {@code Entity} class with {@code getter/setter} methods, {@code PreparedQuery#findFirst(Class).orElseNull()} will be called.
      • *
      *
        *
      • Or else if the return type of the method is {@code Optional}:
      • *
          *
        • If the value type of {@code Optional} is {@code Map}, or {@code Entity} class with {@code getter/setter} methods, or {@code List}, or {@code Object[]}, {@code PreparedQuery#findFirst(Class)} will be called.
        • *
        *
          *
        • Or else, {@code PreparedQuery#queryForSingleNonNull(Class)} will be called.
        • *
        *
      *
        *
      • Or else if the return type of the method is {@code Nullable}:
      • *
          *
        • If the value type of {@code Nullable} is {@code Map}, or {@code Entity} class with {@code getter/setter} methods, or {@code List}, or {@code Object[]}, {@code PreparedQuery#findFirst(Class)} will be called.
        • *
        *
          *
        • Or else, {@code PreparedQuery#queryForSingleResult(Class)} will be called.
        • *
        *
      *
        *
      • Or else if the return type of the method is {@code OptionalBoolean/Byte/.../Double}, {@code PreparedQuery#queryForBoolean/Byte/...Double()} will be called.
      • *
      *
        *
      • Or else if the return type of the method is {@code List}, and the method name doesn't start with {@code "get"/"findFirst"/"findOne"/"findOnlyOne"}, {@code PreparedQuery#list(Class)} will be called.
      • *
      *
        *
      • Or else if the return type of the method is {@code boolean/Boolean}, and the method name starts with {@code "exist"/"exists"/"notExist"/"notExists"}, {@code PreparedQuery#exists()} or {@code PreparedQuery#notExists()} will be called.
      • *
      *
        *
      • Or else, {@code PreparedQuery#queryForSingleResult(Class).orElse(N.defaultValueOf(returnType)} will be called.
      • *
      *
    * *
    *
    * * Here is a simple {@code UserDao} sample. * *
     * 
     * public static interface UserDao extends JdbcUtil.CrudDao {
     *     @NamedInsert("INSERT INTO user (id, first_name, last_name, email) VALUES (:id, :firstName, :lastName, :email)")
     *     void insertWithId(User user) throws SQLException;
     *
     *     @NamedUpdate("UPDATE user SET first_name = :firstName, last_name = :lastName WHERE id = :id")
     *     int updateFirstAndLastName(@Bind("firstName") String newFirstName, @Bind("lastName") String newLastName, @Bind("id") long id) throws SQLException;
     *
     *     @NamedSelect("SELECT first_name, last_name FROM user WHERE id = :id")
     *     User getFirstAndLastNameBy(@Bind("id") long id) throws SQLException;
     *
     *     @NamedSelect("SELECT id, first_name, last_name, email FROM user")
     *     Stream allUsers() throws SQLException;
     * }
     * 
     * 
    * * Here is the generate way to work with transaction started by {@code SQLExecutor}. * *
     * 
     * static final UserDao userDao = Dao.newInstance(UserDao.class, dataSource);
     * ...
     *
     * final SQLTransaction tran = JdbcUtil.beginTransaction(dataSource, IsolationLevel.READ_COMMITTED);
     *
     * try {
     *      userDao.getById(id);
     *      userDao.update(...);
     *      // more...
     *
     *      tran.commit();
     * } finally {
     *      // The connection will be automatically closed after the transaction is committed or rolled back.
     *      tran.rollbackIfNotCommitted();
     * }
     * 
     * 
    * * @param * @param {@code SQLBuilder} used to generate sql scripts. Only can be {@code SQLBuilder.PSC/PAC/PLC} * @param * @see {@link com.landawn.abacus.condition.ConditionFactory} * @see {@link com.landawn.abacus.condition.ConditionFactory.CF} * @see JdbcUtil#prepareQuery(javax.sql.DataSource, String) * @see JdbcUtil#prepareNamedQuery(javax.sql.DataSource, String) * @see JdbcUtil#beginTransaction(javax.sql.DataSource, IsolationLevel, boolean) * @see Dao * @See CrudDao * @see com.landawn.abacus.annotation.AccessFieldByMethod * @see com.landawn.abacus.annotation.JoinedBy * @see com.landawn.abacus.condition.ConditionFactory * @see com.landawn.abacus.condition.ConditionFactory.CF * * @see How to turn off the Eclipse code formatter for certain sections of Java code? */ @SuppressWarnings({ "RedundantThrows", "resource" }) public interface Dao> { /** * Retrieves the data source. * * @return the data source */ @NonDBOperation javax.sql.DataSource dataSource(); // SQLExecutor sqlExecutor(); /** * Retrieves the {@code SQLMapper} instance if it's configured. Otherwise, an empty {@code SQLMapper} will be returned. * * @return the SQLMapper instance */ @NonDBOperation SQLMapper sqlMapper(); /** * Retrieves the class of the target entity. * * @return the class of the target entity * @deprecated for internal use only. */ @Deprecated @NonDBOperation @Internal Class targetEntityClass(); /** * Retrieves the name of the target table. * * @return the name of the target table * @deprecated for internal use only. */ @Deprecated @NonDBOperation @Internal String targetTableName(); /** * Retrieves the executor for asynchronous operations. * * @return the executor for asynchronous operations * @deprecated for internal use only. */ @Deprecated @NonDBOperation @Internal Executor executor(); /** * Retrieves the asynchronous executor. * * @return the asynchronous executor * @deprecated for internal use only. */ @Deprecated @NonDBOperation @Internal AsyncExecutor asyncExecutor(); // @NonDBOperation // @Beta // void cacheSql(String key, String sql); // // @NonDBOperation // @Beta // void cacheSqls(String key, Collection sqls); // // @NonDBOperation // @Beta // String getCachedSql(String key); // // @NonDBOperation // @Beta // ImmutableList getCachedSqls(String key); // /** // * // * @param isolationLevel // * @return // * @throws UncheckedSQLException // */ // @NonDBOperation // default SQLTransaction beginTransaction(final IsolationLevel isolationLevel) throws UncheckedSQLException { // return beginTransaction(isolationLevel, false); // } // // /** // * The connection opened in the transaction will be automatically closed after the transaction is committed or rolled back. // * DON'T close it again by calling the close method. // *
    // *
    // * The transaction will be shared cross the instances of {@code SQLExecutor/Dao} by the methods called in the same thread with same {@code DataSource}. // * // *
    // *
    // * // * The general programming way with SQLExecutor/Dao is to execute sql scripts(generated by SQLBuilder) with array/list/map/entity by calling (batch)insert/update/delete/query/... methods. // * If Transaction is required, it can be started: // * // *
        //     * 
        //     *   final SQLTransaction tran = someDao.beginTransaction(IsolationLevel.READ_COMMITTED);
        //     *   try {
        //     *       // sqlExecutor.insert(...);
        //     *       // sqlExecutor.update(...);
        //     *       // sqlExecutor.query(...);
        //     *
        //     *       tran.commit();
        //     *   } finally {
        //     *       // The connection will be automatically closed after the transaction is committed or rolled back.
        //     *       tran.rollbackIfNotCommitted();
        //     *   }
        //     * 
        //     * 
    // * // * @param isolationLevel // * @param forUpdateOnly // * @return // * @throws UncheckedSQLException // * @see {@link SQLExecutor#beginTransaction(IsolationLevel, boolean)} // */ // @NonDBOperation // default SQLTransaction beginTransaction(final IsolationLevel isolationLevel, final boolean forUpdateOnly) throws UncheckedSQLException { // N.checkArgNotNull(isolationLevel, "isolationLevel"); // // final javax.sql.DataSource ds = dataSource(); // SQLTransaction tran = SQLTransaction.getTransaction(ds); // // if (tran == null) { // Connection conn = null; // boolean noException = false; // // try { // conn = getConnection(ds); // tran = new SQLTransaction(ds, conn, isolationLevel, true, true); // tran.incrementAndGetRef(isolationLevel, forUpdateOnly); // // noException = true; // } catch (SQLException e) { // throw new UncheckedSQLException(e); // } finally { // if (noException == false) { // JdbcUtil.releaseConnection(conn, ds); // } // } // // logger.info("Create a new SQLTransaction(id={})", tran.id()); // SQLTransaction.putTransaction(ds, tran); // } else { // logger.info("Reusing the existing SQLTransaction(id={})", tran.id()); // tran.incrementAndGetRef(isolationLevel, forUpdateOnly); // } // // return tran; // } /** * * @param query * @return * @throws SQLException */ @Beta @NonDBOperation default PreparedQuery prepareQuery(final String query) throws SQLException { return JdbcUtil.prepareQuery(dataSource(), query); } /** * * @param query * @param generateKeys * @return * @throws SQLException */ @Beta @NonDBOperation default PreparedQuery prepareQuery(final String query, final boolean generateKeys) throws SQLException { return JdbcUtil.prepareQuery(dataSource(), query, generateKeys); } /** * * @param query * @param returnColumnIndexes * @return * @throws SQLException */ @Beta @NonDBOperation default PreparedQuery prepareQuery(final String query, final int[] returnColumnIndexes) throws SQLException { return JdbcUtil.prepareQuery(dataSource(), query, returnColumnIndexes); } /** * * * @param query * @param returnColumnNames * @return * @throws SQLException */ @Beta @NonDBOperation default PreparedQuery prepareQuery(final String query, final String[] returnColumnNames) throws SQLException { return JdbcUtil.prepareQuery(dataSource(), query, returnColumnNames); } /** * * @param sql * @param stmtCreator * @return * @throws SQLException */ @Beta @NonDBOperation default PreparedQuery prepareQuery(final String sql, final Throwables.BiFunction stmtCreator) throws SQLException { return JdbcUtil.prepareQuery(dataSource(), sql, stmtCreator); } /** * Prepare a {@code select} query by specified {@code cond}. *
    * {@code query} could be a {@code select/insert/update/delete} or other sql statement. If it's {@code select} by default if not specified. * * @param cond * @return * @throws SQLException * @see {@link #prepareQuery(Collection, Condition)} */ @Beta @NonDBOperation default PreparedQuery prepareQuery(final Condition cond) throws SQLException { return prepareQuery(null, cond); } /** * Prepare a {@code select} query by specified {@code selectPropNames} and {@code cond}. *
    * * {@code query} could be a {@code select/insert/update/delete} or other sql statement. If it's {@code select} by default if not specified. * * @param selectPropNames * @param cond * @return * @throws SQLException */ @Beta @NonDBOperation default PreparedQuery prepareQuery(final Collection selectPropNames, final Condition cond) throws SQLException { return DaoUtil.getDaoPreparedQueryFunc(this)._1.apply(selectPropNames, cond); } /** * * @param query * @return * @throws SQLException */ @Beta @NonDBOperation default PreparedQuery prepareQueryForBigResult(final String query) throws SQLException { return JdbcUtil.prepareQueryForBigResult(dataSource(), query); } /** * Prepare a big result {@code select} query by specified {@code cond}. *
    * * {@code query} could be a {@code select/insert/update/delete} or other sql statement. If it's {@code select} by default if not specified. * * @param cond * @return * @throws SQLException * @see {@link #prepareQueryForBigResult(Collection, Condition)} */ @Beta @NonDBOperation default PreparedQuery prepareQueryForBigResult(final Condition cond) throws SQLException { return prepareQueryForBigResult(null, cond); } /** * Prepare a big result {@code select} query by specified {@code selectPropNames} and {@code cond}. *
    * * {@code query} could be a {@code select/insert/update/delete} or other sql statement. If it's {@code select} by default if not specified. * * @param selectPropNames * @param cond * @return * @throws SQLException */ @Beta @NonDBOperation default PreparedQuery prepareQueryForBigResult(final Collection selectPropNames, final Condition cond) throws SQLException { return prepareQuery(selectPropNames, cond).configStmt(DaoUtil.stmtSetterForBigQueryResult); } /** * * @param namedQuery * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final String namedQuery) throws SQLException { return JdbcUtil.prepareNamedQuery(dataSource(), namedQuery); } /** * * @param namedQuery * @param generateKeys * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final String namedQuery, final boolean generateKeys) throws SQLException { return JdbcUtil.prepareNamedQuery(dataSource(), namedQuery, generateKeys); } /** * * @param namedQuery * @param returnColumnIndexes * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final String namedQuery, final int[] returnColumnIndexes) throws SQLException { return JdbcUtil.prepareNamedQuery(dataSource(), namedQuery, returnColumnIndexes); } /** * * @param namedQuery * @param returnColumnNames * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final String namedQuery, final String[] returnColumnNames) throws SQLException { return JdbcUtil.prepareNamedQuery(dataSource(), namedQuery, returnColumnNames); } /** * * @param namedQuery * @param stmtCreator * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final String namedQuery, final Throwables.BiFunction stmtCreator) throws SQLException { return JdbcUtil.prepareNamedQuery(dataSource(), namedQuery, stmtCreator); } /** * * @param namedSql the named query * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final ParsedSql namedSql) throws SQLException { return JdbcUtil.prepareNamedQuery(dataSource(), namedSql); } /** * * @param namedSql the named query * @param generateKeys * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final ParsedSql namedSql, final boolean generateKeys) throws SQLException { return JdbcUtil.prepareNamedQuery(dataSource(), namedSql, generateKeys); } /** * * @param namedQuery * @param returnColumnIndexes * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final ParsedSql namedQuery, final int[] returnColumnIndexes) throws SQLException { return JdbcUtil.prepareNamedQuery(dataSource(), namedQuery, returnColumnIndexes); } /** * * @param namedQuery * @param returnColumnNames * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final ParsedSql namedQuery, final String[] returnColumnNames) throws SQLException { return JdbcUtil.prepareNamedQuery(dataSource(), namedQuery, returnColumnNames); } /** * * @param namedSql the named query * @param stmtCreator * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final ParsedSql namedSql, final Throwables.BiFunction stmtCreator) throws SQLException { return JdbcUtil.prepareNamedQuery(dataSource(), namedSql, stmtCreator); } /** * Prepare a {@code select} query by specified {@code cond}. *
    * {@code query} could be a {@code select/insert/update/delete} or other sql statement. If it's {@code select} by default if not specified. * * @param cond * @return * @throws SQLException * @see {@link #prepareNamedQuery(Collection, Condition)} */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final Condition cond) throws SQLException { return prepareNamedQuery(null, cond); } /** * Prepare a {@code select} query by specified {@code selectPropNames} and {@code cond}. *
    * * {@code query} could be a {@code select/insert/update/delete} or other sql statement. If it's {@code select} by default if not specified. * * @param selectPropNames * @param cond * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQuery(final Collection selectPropNames, final Condition cond) throws SQLException { return DaoUtil.getDaoPreparedQueryFunc(this)._2.apply(selectPropNames, cond); } /** * * @param namedQuery * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQueryForBigResult(final String namedQuery) throws SQLException { return JdbcUtil.prepareNamedQueryForBigResult(dataSource(), namedQuery); } /** * * @param namedSql the named query * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQueryForBigResult(final ParsedSql namedSql) throws SQLException { return JdbcUtil.prepareNamedQueryForBigResult(dataSource(), namedSql); } /** * Prepare a big result {@code select} query by specified {@code cond}. *
    * * {@code query} could be a {@code select/insert/update/delete} or other sql statement. If it's {@code select} by default if not specified. * * @param cond * @return * @throws SQLException * @see {@link #prepareNamedQueryForBigResult(Collection, Condition)} */ @Beta @NonDBOperation default NamedQuery prepareNamedQueryForBigResult(final Condition cond) throws SQLException { return prepareNamedQueryForBigResult(null, cond); } /** * Prepare a big result {@code select} query by specified {@code selectPropNames} and {@code cond}. *
    * * {@code query} could be a {@code select/insert/update/delete} or other sql statement. If it's {@code select} by default if not specified. * * @param selectPropNames * @param cond * @return * @throws SQLException */ @Beta @NonDBOperation default NamedQuery prepareNamedQueryForBigResult(final Collection selectPropNames, final Condition cond) throws SQLException { return prepareNamedQuery(selectPropNames, cond).configStmt(DaoUtil.stmtSetterForBigQueryResult); } /** * * @param query * @return * @throws SQLException */ @Beta @NonDBOperation default CallableQuery prepareCallableQuery(final String query) throws SQLException { return JdbcUtil.prepareCallableQuery(dataSource(), query); } /** * * @param sql * @param stmtCreator * @return * @throws SQLException */ @Beta @NonDBOperation default CallableQuery prepareCallableQuery(final String sql, final Throwables.BiFunction stmtCreator) throws SQLException { return JdbcUtil.prepareCallableQuery(dataSource(), sql, stmtCreator); } /** * * * @param entityToSave * @throws SQLException */ void save(final T entityToSave) throws SQLException; /** * * * @param entityToSave * @param propNamesToSave * @throws SQLException */ void save(final T entityToSave, final Collection propNamesToSave) throws SQLException; /** * * * @param namedInsertSQL * @param entityToSave * @throws SQLException */ void save(final String namedInsertSQL, final T entityToSave) throws SQLException; /** * Insert the specified entities to database by batch. * * @param entitiesToSave * @throws SQLException * @see CrudDao#batchInsert(Collection) */ default void batchSave(final Collection entitiesToSave) throws SQLException { batchSave(entitiesToSave, JdbcUtil.DEFAULT_BATCH_SIZE); } /** * Insert the specified entities to database by batch. * * @param entitiesToSave * @param batchSize * @throws SQLException * @see CrudDao#batchInsert(Collection) */ void batchSave(final Collection entitiesToSave, final int batchSize) throws SQLException; /** * Insert the specified entities to database by batch. * * @param entitiesToSave * @param propNamesToSave * @throws SQLException * @see CrudDao#batchInsert(Collection) */ default void batchSave(final Collection entitiesToSave, final Collection propNamesToSave) throws SQLException { batchSave(entitiesToSave, propNamesToSave, JdbcUtil.DEFAULT_BATCH_SIZE); } /** * Insert the specified entities to database by batch. * * @param entitiesToSave * @param propNamesToSave * @param batchSize * @throws SQLException * @see CrudDao#batchInsert(Collection) */ void batchSave(final Collection entitiesToSave, final Collection propNamesToSave, final int batchSize) throws SQLException; /** * Insert the specified entities to database by batch. * * @param namedInsertSQL * @param entitiesToSave * @throws SQLException * @see CrudDao#batchInsert(Collection) */ @Beta default void batchSave(final String namedInsertSQL, final Collection entitiesToSave) throws SQLException { batchSave(namedInsertSQL, entitiesToSave, JdbcUtil.DEFAULT_BATCH_SIZE); } /** * Insert the specified entities to database by batch. * * @param namedInsertSQL * @param entitiesToSave * @param batchSize * @throws SQLException * @see CrudDao#batchInsert(Collection) */ @Beta void batchSave(final String namedInsertSQL, final Collection entitiesToSave, final int batchSize) throws SQLException; /** * * @param cond * @return {@code true}, if there is at least one record found. * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#exists() */ boolean exists(final Condition cond) throws SQLException; /** * * @param cond * @return {@code true}, if there is no record found. * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#notExists() */ @Beta default boolean notExists(final Condition cond) throws SQLException { return !exists(cond); } /** * * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ int count(final Condition cond) throws SQLException; // TODO dao.anyMatch/allMatch/noneMatch(...) // /** // * // * @param selectPropNames // * @param cond // * @param rowFilter // * @return // * @throws SQLException // */ // @Beta // default boolean anyMatch(final Collection selectPropNames, final Condition cond, final Jdbc.RowFilter rowFilter) throws SQLException { // return prepareQuery(selectPropNames, cond).anyMatch(rowFilter); // } // // /** // * // * @param selectPropNames // * @param cond // * @param rowFilter // * @return // * @throws SQLException // */ // @Beta // default boolean anyMatch(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowFilter rowFilter) throws SQLException { // return prepareQuery(selectPropNames, cond).anyMatch(rowFilter); // } // // /** // * // * @param selectPropNames // * @param cond // * @param rowFilter // * @return // * @throws SQLException // */ // @Beta // default boolean allMatch(final Collection selectPropNames, final Condition cond, final Jdbc.RowFilter rowFilter) throws SQLException { // return prepareQuery(selectPropNames, cond).allMatch(rowFilter); // } // // /** // * // * @param selectPropNames // * @param cond // * @param rowFilter // * @return // * @throws SQLException // */ // @Beta // default boolean allMatch(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowFilter rowFilter) throws SQLException { // return prepareQuery(selectPropNames, cond).allMatch(rowFilter); // } // // /** // * // * @param selectPropNames // * @param cond // * @param rowFilter // * @return // * @throws SQLException // */ // @Beta // default boolean noneMatch(final Collection selectPropNames, final Condition cond, final Jdbc.RowFilter rowFilter) throws SQLException { // return prepareQuery(selectPropNames, cond).noneMatch(rowFilter); // } // // /** // * // * @param selectPropNames // * @param cond // * @param rowFilter // * @return // * @throws SQLException // */ // @Beta // default boolean noneMatch(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowFilter rowFilter) throws SQLException { // return prepareQuery(selectPropNames, cond).noneMatch(rowFilter); // } /** * * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ Optional findFirst(final Condition cond) throws SQLException; /** * * * @param * @param cond * @param rowMapper * @return * @throws SQLException * @throws IllegalArgumentException if {@code rowMapper} returns {@code null} for the found record. * @see ConditionFactory * @see ConditionFactory.CF */ Optional findFirst(final Condition cond, final Jdbc.RowMapper rowMapper) throws SQLException, IllegalArgumentException; /** * * * @param * @param cond * @param rowMapper * @return * @throws SQLException * @throws IllegalArgumentException if {@code rowMapper} returns {@code null} for the found record. * @see ConditionFactory * @see ConditionFactory.CF */ Optional findFirst(final Condition cond, final Jdbc.BiRowMapper rowMapper) throws SQLException, IllegalArgumentException; /** * * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ Optional findFirst(final Collection selectPropNames, final Condition cond) throws SQLException; /** * * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowMapper * @return * @throws SQLException * @throws IllegalArgumentException if {@code rowMapper} returns {@code null} for the found record. * @see ConditionFactory * @see ConditionFactory.CF */ Optional findFirst(final Collection selectPropNames, final Condition cond, final Jdbc.RowMapper rowMapper) throws SQLException, IllegalArgumentException; /** * * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowMapper * @return * @throws SQLException * @throws IllegalArgumentException if {@code rowMapper} returns {@code null} for the found record. * @see ConditionFactory * @see ConditionFactory.CF */ Optional findFirst(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowMapper rowMapper) throws SQLException, IllegalArgumentException; /** * * @param cond * @return * @throws DuplicatedResultException if more than one record found by the specified {@code id} (or {@code condition}). * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ Optional findOnlyOne(final Condition cond) throws DuplicatedResultException, SQLException; /** * * * @param * @param cond * @param rowMapper * @return * @throws DuplicatedResultException if more than one record found by the specified {@code id} (or {@code condition}). * @throws SQLException * @throws IllegalArgumentException if {@code rowMapper} returns {@code null} for the found record. * @see ConditionFactory * @see ConditionFactory.CF */ Optional findOnlyOne(final Condition cond, final Jdbc.RowMapper rowMapper) throws DuplicatedResultException, SQLException, IllegalArgumentException; /** * * * @param * @param cond * @param rowMapper * @return * @throws DuplicatedResultException if more than one record found by the specified {@code id} (or {@code condition}). * @throws SQLException * @throws IllegalArgumentException if {@code rowMapper} returns {@code null} for the found record. * @see ConditionFactory * @see ConditionFactory.CF */ Optional findOnlyOne(final Condition cond, final Jdbc.BiRowMapper rowMapper) throws DuplicatedResultException, SQLException, IllegalArgumentException; /** * * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @return * @throws DuplicatedResultException if more than one record found by the specified {@code id} (or {@code condition}). * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ Optional findOnlyOne(final Collection selectPropNames, final Condition cond) throws DuplicatedResultException, SQLException; /** * * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowMapper * @return * @throws DuplicatedResultException if more than one record found by the specified {@code id} (or {@code condition}). * @throws SQLException * @throws IllegalArgumentException if {@code rowMapper} returns {@code null} for the found record. */ Optional findOnlyOne(final Collection selectPropNames, final Condition cond, final Jdbc.RowMapper rowMapper) throws DuplicatedResultException, SQLException, IllegalArgumentException; /** * * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowMapper * @return * @throws DuplicatedResultException if more than one record found by the specified {@code id} (or {@code condition}). * @throws SQLException * @throws IllegalArgumentException if {@code rowMapper} returns {@code null} for the found record. * @see ConditionFactory * @see ConditionFactory.CF */ Optional findOnlyOne(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowMapper rowMapper) throws DuplicatedResultException, SQLException, IllegalArgumentException; /** * Returns an {@code OptionalBoolean} describing the value in the first row/column if it exists, otherwise return an empty {@code OptionalBoolean}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForBoolean() */ OptionalBoolean queryForBoolean(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns an {@code OptionalChar} describing the value in the first row/column if it exists, otherwise return an empty {@code OptionalChar}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForChar() */ OptionalChar queryForChar(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns an {@code OptionalByte} describing the value in the first row/column if it exists, otherwise return an empty {@code OptionalByte}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForByte() */ OptionalByte queryForByte(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns an {@code OptionalShort} describing the value in the first row/column if it exists, otherwise return an empty {@code OptionalShort}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForShort() */ OptionalShort queryForShort(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns an {@code OptionalInt} describing the value in the first row/column if it exists, otherwise return an empty {@code OptionalInt}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForInt() */ OptionalInt queryForInt(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns an {@code OptionalLong} describing the value in the first row/column if it exists, otherwise return an empty {@code OptionalLong}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForLong() */ OptionalLong queryForLong(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns an {@code OptionalFloat} describing the value in the first row/column if it exists, otherwise return an empty {@code OptionalFloat}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForFloat() */ OptionalFloat queryForFloat(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns an {@code OptionalDouble} describing the value in the first row/column if it exists, otherwise return an empty {@code OptionalDouble}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForDouble() */ OptionalDouble queryForDouble(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns a {@code Nullable} describing the value in the first row/column if it exists, otherwise return an empty {@code Nullable}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForString() */ Nullable queryForString(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns a {@code Nullable} describing the value in the first row/column if it exists, otherwise return an empty {@code Nullable}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForDate() */ Nullable queryForDate(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns a {@code Nullable} describing the value in the first row/column if it exists, otherwise return an empty {@code Nullable}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForTime() */ Nullable queryForTime(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns a {@code Nullable} describing the value in the first row/column if it exists, otherwise return an empty {@code Nullable}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForTimestamp() */ Nullable queryForTimestamp(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns a {@code Nullable} describing the value in the first row/column if it exists, otherwise return an empty {@code Nullable}. * * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForBytes() */ Nullable queryForBytes(final String singleSelectPropName, final Condition cond) throws SQLException; /** * Returns a {@code Nullable} describing the value in the first row/column if it exists, otherwise return an empty {@code Nullable}. * @param singleSelectPropName * @param cond * @param targetValueType * * @param * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForSingleResult(Class) */ Nullable queryForSingleResult(final String singleSelectPropName, final Condition cond, final Class targetValueType) throws SQLException; /** * Returns an {@code Optional} describing the value in the first row/column if it exists, otherwise return an empty {@code Optional}. * @param singleSelectPropName * @param cond * @param targetValueType * * @param the value type * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForSingleNonNull(Class) */ Optional queryForSingleNonNull(final String singleSelectPropName, final Condition cond, final Class targetValueType) throws SQLException; /** * Returns an {@code Optional} describing the value in the first row/column if it exists, otherwise return an empty {@code Optional}. * * @param the value type * @param singleSelectPropName * @param cond * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForSingleNonNull(Class) */ @Beta Optional queryForSingleNonNull(final String singleSelectPropName, final Condition cond, final Jdbc.RowMapper rowMapper) throws SQLException; /** * Returns a {@code Nullable} describing the value in the first row/column if it exists, otherwise return an empty {@code Nullable}. * And throws {@code DuplicatedResultException} if more than one record found. * @param singleSelectPropName * @param cond * @param targetValueType * * @param the value type * @return * @throws DuplicatedResultException if more than one record found by the specified {@code id} (or {@code condition}). * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForUniqueResult(Class) */ Nullable queryForUniqueResult(final String singleSelectPropName, final Condition cond, final Class targetValueType) throws DuplicatedResultException, SQLException; /** * Returns an {@code Optional} describing the value in the first row/column if it exists, otherwise return an empty {@code Optional}. * @param singleSelectPropName * @param cond * @param targetValueType * * @param the value type * @return * @throws DuplicatedResultException if more than one record found by the specified {@code id} (or {@code condition}). * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForUniqueNonNull(Class) */ Optional queryForUniqueNonNull(final String singleSelectPropName, final Condition cond, final Class targetValueType) throws DuplicatedResultException, SQLException; /** * Returns an {@code Optional} describing the value in the first row/column if it exists, otherwise return an empty {@code Optional}. * * @param the value type * @param singleSelectPropName * @param cond * @param rowMapper * @return * @throws DuplicatedResultException if more than one record found by the specified {@code id} (or {@code condition}). * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF * @see AbstractQuery#queryForUniqueNonNull(Class) */ @Beta Optional queryForUniqueNonNull(final String singleSelectPropName, final Condition cond, final Jdbc.RowMapper rowMapper) throws DuplicatedResultException, SQLException; /** * * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ DataSet query(final Condition cond) throws SQLException; /** * * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ DataSet query(final Collection selectPropNames, final Condition cond) throws SQLException; /** * * * @param * @param cond * @param resultExtractor Don't save/return {@code ResultSet}. It will be closed after this call. * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ R query(final Condition cond, final Jdbc.ResultExtractor resultExtractor) throws SQLException; /** * * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param resultExtractor Don't save/return {@code ResultSet}. It will be closed after this call. * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ R query(final Collection selectPropNames, final Condition cond, final Jdbc.ResultExtractor resultExtractor) throws SQLException; /** * * * @param * @param cond * @param resultExtractor Don't save/return {@code ResultSet}. It will be closed after this call. * @return * @throws SQLException */ R query(final Condition cond, final Jdbc.BiResultExtractor resultExtractor) throws SQLException; /** * * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param resultExtractor Don't save/return {@code ResultSet}. It will be closed after this call. * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ R query(final Collection selectPropNames, final Condition cond, final Jdbc.BiResultExtractor resultExtractor) throws SQLException; /** * * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ List list(final Condition cond) throws SQLException; /** * * * @param * @param cond * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ List list(final Condition cond, final Jdbc.RowMapper rowMapper) throws SQLException; /** * * * @param * @param cond * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ List list(final Condition cond, final Jdbc.BiRowMapper rowMapper) throws SQLException; /** * * * @param * @param cond * @param rowFilter * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ List list(final Condition cond, final Jdbc.RowFilter rowFilter, final Jdbc.RowMapper rowMapper) throws SQLException; /** * * * @param * @param cond * @param rowFilter * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ List list(final Condition cond, final Jdbc.BiRowFilter rowFilter, final Jdbc.BiRowMapper rowMapper) throws SQLException; /** * * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ List list(final Collection selectPropNames, final Condition cond) throws SQLException; /** * * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ List list(final Collection selectPropNames, final Condition cond, final Jdbc.RowMapper rowMapper) throws SQLException; /** * * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ List list(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowMapper rowMapper) throws SQLException; /** * * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowFilter * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ List list(final Collection selectPropNames, final Condition cond, final Jdbc.RowFilter rowFilter, final Jdbc.RowMapper rowMapper) throws SQLException; /** * * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowFilter * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ List list(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowFilter rowFilter, final Jdbc.BiRowMapper rowMapper) throws SQLException; /** * * * @param * @param singleSelectPropName * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ default List list(final String singleSelectPropName, final Condition cond) throws SQLException { final PropInfo propInfo = ParserUtil.getBeanInfo(targetEntityClass()).getPropInfo(singleSelectPropName); final Jdbc.RowMapper rowMapper = propInfo == null ? ColumnOne.getObject() : ColumnOne.get((Type) propInfo.dbType); return list(singleSelectPropName, cond, rowMapper); } /** * * * @param * @param singleSelectPropName * @param cond * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ default List list(final String singleSelectPropName, final Condition cond, final Jdbc.RowMapper rowMapper) throws SQLException { return list(N.asList(singleSelectPropName), cond, rowMapper); } /** * * * @param * @param singleSelectPropName * @param cond * @param rowFilter * @param rowMapper * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ default List list(final String singleSelectPropName, final Condition cond, final Jdbc.RowFilter rowFilter, final Jdbc.RowMapper rowMapper) throws SQLException { return list(N.asList(singleSelectPropName), cond, rowFilter, rowMapper); } // Will it cause confusion if it's called in transaction? /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param cond * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation Stream stream(final Condition cond); // Will it cause confusion if it's called in transaction? /** * * @param * @param cond * @param rowMapper * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation Stream stream(final Condition cond, final Jdbc.RowMapper rowMapper); // Will it cause confusion if it's called in transaction? /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param * @param cond * @param rowMapper * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation Stream stream(final Condition cond, final Jdbc.BiRowMapper rowMapper); /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param * @param cond * @param rowFilter * @param rowMapper * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation Stream stream(final Condition cond, final Jdbc.RowFilter rowFilter, final Jdbc.RowMapper rowMapper); /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param * @param cond * @param rowFilter * @param rowMapper * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation Stream stream(final Condition cond, final Jdbc.BiRowFilter rowFilter, final Jdbc.BiRowMapper rowMapper); // Will it cause confusion if it's called in transaction? /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation Stream stream(final Collection selectPropNames, final Condition cond); // Will it cause confusion if it's called in transaction? /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowMapper * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation Stream stream(final Collection selectPropNames, final Condition cond, final Jdbc.RowMapper rowMapper); // Will it cause confusion if it's called in transaction? /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowMapper * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation Stream stream(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowMapper rowMapper); // Will it cause confusion if it's called in transaction? /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowFilter * @param rowMapper * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation Stream stream(final Collection selectPropNames, final Condition cond, Jdbc.RowFilter rowFilter, final Jdbc.RowMapper rowMapper); // Will it cause confusion if it's called in transaction? /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond * @param rowFilter * @param rowMapper * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation Stream stream(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowFilter rowFilter, final Jdbc.BiRowMapper rowMapper); // Will it cause confusion if it's called in transaction? /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param * @param singleSelectPropName * @param cond * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation default Stream stream(final String singleSelectPropName, final Condition cond) { final PropInfo propInfo = ParserUtil.getBeanInfo(targetEntityClass()).getPropInfo(singleSelectPropName); final Jdbc.RowMapper rowMapper = propInfo == null ? ColumnOne.getObject() : ColumnOne.get((Type) propInfo.dbType); return stream(singleSelectPropName, cond, rowMapper); } // Will it cause confusion if it's called in transaction? /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param * @param singleSelectPropName * @param cond * @param rowMapper * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation default Stream stream(final String singleSelectPropName, final Condition cond, final Jdbc.RowMapper rowMapper) { return stream(N.asList(singleSelectPropName), cond, rowMapper); } // Will it cause confusion if it's called in transaction? /** * Lazy execution, lazy fetching. No connection fetching/creating, no statement preparing or execution, no result fetching until {@code @TerminalOp} or {@code @TerminalOpTriggered} stream operation is called. * * @param * @param singleSelectPropName * @param cond * @param rowFilter * @param rowMapper * @return * @see ConditionFactory * @see ConditionFactory.CF */ @LazyEvaluation default Stream stream(final String singleSelectPropName, final Condition cond, final Jdbc.RowFilter rowFilter, final Jdbc.RowMapper rowMapper) { return stream(N.asList(singleSelectPropName), cond, rowFilter, rowMapper); } /** * Paginates the results based on the given condition and page size. * * @param cond the condition to filter the results. It must contain {@code orderBy} * @param pageSize the number of records per page * @param paramSetter the parameter setter for the prepared query * @return a Stream of DataSet and SQLException */ @Beta @LazyEvaluation Stream paginate(final Condition cond, final int pageSize, final Jdbc.BiParametersSetter paramSetter); /** * Paginates the results based on the given condition and page size. * * @param the type of the result * @param cond the condition to filter the results. It must contain {@code orderBy} * @param pageSize the number of records per page * @param paramSetter the parameter setter for the prepared query * @param resultExtractor the result extractor to process the query results * @return a Stream of the result type and SQLException */ @Beta @LazyEvaluation Stream paginate(final Condition cond, final int pageSize, final Jdbc.BiParametersSetter paramSetter, final Jdbc.ResultExtractor resultExtractor); /** * Paginates the results based on the given condition and page size. * * @param the type of the result * @param cond the condition to filter the results. It must contain {@code orderBy} * @param pageSize the number of records per page * @param paramSetter the parameter setter for the prepared query * @param resultExtractor the result extractor to process the query results * @return a Stream of the result type and SQLException */ @Beta @LazyEvaluation Stream paginate(final Condition cond, final int pageSize, final Jdbc.BiParametersSetter paramSetter, final Jdbc.BiResultExtractor resultExtractor); /** * Paginates the results based on the given condition, selected properties, and page size. * * @param selectPropNames the properties to select in the query * @param cond the condition to filter the results. It must contain {@code orderBy} * @param pageSize the number of records per page * @param paramSetter the parameter setter for the prepared query * @return a Stream of DataSet and SQLException */ @Beta @LazyEvaluation Stream paginate(final Collection selectPropNames, final Condition cond, final int pageSize, final Jdbc.BiParametersSetter paramSetter); /** * Paginates the results based on the given condition, selected properties, and page size. * * @param the type of the result * @param selectPropNames the properties to select in the query * @param cond the condition to filter the results. It must contain {@code orderBy} * @param pageSize the number of records per page * @param paramSetter the parameter setter for the prepared query * @param resultExtractor the result extractor to process the query results * @return a Stream of the result type and SQLException */ @Beta @LazyEvaluation Stream paginate(final Collection selectPropNames, final Condition cond, final int pageSize, final Jdbc.BiParametersSetter paramSetter, final Jdbc.ResultExtractor resultExtractor); /** * Paginates the results based on the given condition, selected properties, and page size. * * @param the type of the result * @param selectPropNames the properties to select in the query * @param cond the condition to filter the results. It must contain {@code orderBy} * @param pageSize the number of records per page * @param paramSetter the parameter setter for the prepared query * @param resultExtractor the result extractor to process the query results * @return a Stream of the result type and SQLException */ @Beta @LazyEvaluation Stream paginate(final Collection selectPropNames, final Condition cond, final int pageSize, final Jdbc.BiParametersSetter paramSetter, final Jdbc.BiResultExtractor resultExtractor); /** * * * @param cond * @param rowConsumer * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ void forEach(final Condition cond, final Jdbc.RowConsumer rowConsumer) throws SQLException; /** * * * @param cond * @param rowConsumer * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ void forEach(final Condition cond, final Jdbc.BiRowConsumer rowConsumer) throws SQLException; /** * * * @param cond * @param rowFilter * @param rowConsumer * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ void forEach(final Condition cond, final Jdbc.RowFilter rowFilter, final Jdbc.RowConsumer rowConsumer) throws SQLException; /** * * * @param cond * @param rowFilter * @param rowConsumer * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ void forEach(final Condition cond, final Jdbc.BiRowFilter rowFilter, final Jdbc.BiRowConsumer rowConsumer) throws SQLException; /** * * * @param selectPropNames * @param cond * @param rowConsumer * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ void forEach(final Collection selectPropNames, final Condition cond, final Jdbc.RowConsumer rowConsumer) throws SQLException; /** * * * @param selectPropNames * @param cond * @param rowConsumer * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ void forEach(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowConsumer rowConsumer) throws SQLException; /** * * * @param selectPropNames * @param cond * @param rowFilter * @param rowConsumer * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ void forEach(final Collection selectPropNames, final Condition cond, final Jdbc.RowFilter rowFilter, final Jdbc.RowConsumer rowConsumer) throws SQLException; /** * * * @param selectPropNames * @param cond * @param rowFilter * @param rowConsumer * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ void forEach(final Collection selectPropNames, final Condition cond, final Jdbc.BiRowFilter rowFilter, final Jdbc.BiRowConsumer rowConsumer) throws SQLException; /** * Iterates over the results of a query and applies the specified consumer to each row. * * @param selectPropNames the properties (columns) to be selected, excluding the properties of joining entities. * All the properties (columns) will be selected if the specified {@code selectPropNames} is {@code null}. * @param cond the condition to filter the results. * @param rowConsumer the consumer to process each row. The consumer should NOT update or cache input {@code row} parameter * @throws SQLException if a database access error occurs. */ @Beta default void foreach(final Collection selectPropNames, final Condition cond, final Consumer rowConsumer) throws SQLException { //NOSONAR forEach(selectPropNames, cond, Jdbc.RowConsumer.oneOff(targetEntityClass(), rowConsumer)); } /** * Iterates over the results of a query and applies the specified consumer to each row. * * @param cond the condition to filter the results. * @param rowConsumer the consumer to process each row. The consumer should NOT update or cache input {@code row} parameter * @see ConditionFactory * @see ConditionFactory.CF */ @Beta default void foreach(final Condition cond, final Consumer rowConsumer) throws SQLException { //NOSONAR forEach(cond, Jdbc.RowConsumer.oneOff(targetEntityClass(), rowConsumer)); } /** * * @param propName * @param propValue * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ default int update(final String propName, final Object propValue, final Condition cond) throws SQLException { final Map updateProps = new HashMap<>(); updateProps.put(propName, propValue); return update(updateProps, cond); } /** * Update all the records found by specified {@code cond} with all the properties from specified {@code updateProps}. * * @param updateProps * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ int update(final Map updateProps, final Condition cond) throws SQLException; /** * Update all the records found by specified {@code cond} with the properties from specified {@code entity}. * * @param entity * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ default int update(final T entity, final Condition cond) throws SQLException { final Collection propNamesToUpdate = QueryUtil.getUpdatePropNames(targetEntityClass(), null); return update(entity, propNamesToUpdate, cond); } /** * Update all the records found by specified {@code cond} with specified {@code propNamesToUpdate} from specified {@code entity}. * * @param entity * @param propNamesToUpdate * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ int update(final T entity, final Collection propNamesToUpdate, final Condition cond) throws SQLException; /** * Executes {@code insertion} and return the added entity if the record doesn't, otherwise, {@code update} is executed and updated db record is returned. * * @param entity the entity to add or update. * @param uniquePropNamesForQuery the list of unique property names to use to verify if the record exists or not. * @return the added or updated db record. * @throws SQLException if a database access error occurs */ default T upsert(final T entity, final List uniquePropNamesForQuery) throws SQLException { N.checkArgNotNull(entity, s.entity); N.checkArgNotEmpty(uniquePropNamesForQuery, s.uniquePropNamesForQuery); final Condition cond = CF.eqAnd(entity, uniquePropNamesForQuery); return upsert(entity, cond); } /** * Executes {@code insertion} and return the added entity if the record doesn't, otherwise, {@code update} is executed and updated db record is returned. * * @param entity the entity to add or update. * @param cond to verify if the record exists or not. * @return the added or updated db record. * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ default T upsert(final T entity, final Condition cond) throws SQLException { N.checkArgNotNull(entity, s.entity); N.checkArgNotNull(cond, s.cond); final T dbEntity = findOnlyOne(cond).orElseNull(); if (dbEntity == null) { save(entity); return entity; } else { N.merge(entity, dbEntity); update(dbEntity, cond); return dbEntity; } } /** * * @param cond * @return * @throws SQLException * @see ConditionFactory * @see ConditionFactory.CF */ int delete(final Condition cond) throws SQLException; /** * Asynchronously executes the given SQL action using the default executor. *
    * Any transaction started in current thread won't be automatically applied to specified {@code sqlAction} which will be executed in another thread. * * @param the type of the result * @param sqlAction the SQL action to be executed asynchronously * @return a ContinuableFuture representing the result of the asynchronous computation */ @Beta @NonDBOperation default ContinuableFuture asyncCall(final Throwables.Function sqlAction) { return asyncCall(sqlAction, executor()); } /** * Asynchronously executes the given SQL action using the specified executor. *
    * Any transaction started in current thread won't be automatically applied to specified {@code sqlAction} which will be executed in another thread. * * @param the type of the result * @param sqlAction the SQL action to be executed asynchronously * @param executor the executor to use for asynchronous execution * @return a ContinuableFuture representing the result of the asynchronous computation */ @Beta @NonDBOperation default ContinuableFuture asyncCall(final Throwables.Function sqlAction, final Executor executor) { N.checkArgNotNull(sqlAction, s.func); N.checkArgNotNull(executor, s.executor); final TD tdao = (TD) this; return ContinuableFuture.call(() -> sqlAction.apply(tdao), executor); } /** * Asynchronously executes the given SQL action using the default executor. *
    * Any transaction started in the current thread won't be automatically applied to the specified {@code sqlAction} which will be executed in another thread. * * @param sqlAction the SQL action to be executed asynchronously * @return a ContinuableFuture representing the result of the asynchronous computation */ @Beta @NonDBOperation default ContinuableFuture asyncRun(final Throwables.Consumer sqlAction) { return asyncRun(sqlAction, executor()); } /** * Asynchronously executes the given SQL action using the specified executor. *
    * Any transaction started in the current thread won't be automatically applied to the specified {@code sqlAction} which will be executed in another thread. * * @param sqlAction the SQL action to be executed asynchronously * @param executor the executor to use for asynchronous execution * @return a ContinuableFuture representing the result of the asynchronous computation */ @Beta @NonDBOperation default ContinuableFuture asyncRun(final Throwables.Consumer sqlAction, final Executor executor) { N.checkArgNotNull(sqlAction, s.action); N.checkArgNotNull(executor, s.executor); final TD tdao = (TD) this; return ContinuableFuture.run(() -> sqlAction.accept(tdao), executor); } }




  • © 2015 - 2025 Weber Informatics LLC | Privacy Policy