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

io.trino.tests.product.jdbc.TestPreparedStatements Maven / Gradle / Ivy

The newest version!
/*
 * 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 io.trino.tests.product.jdbc;

import io.trino.tempto.ProductTest;
import io.trino.tempto.Requirement;
import io.trino.tempto.RequirementsProvider;
import io.trino.tempto.Requires;
import io.trino.tempto.configuration.Configuration;
import io.trino.tempto.query.QueryResult;
import org.testng.annotations.Test;

import java.math.BigDecimal;
import java.math.BigInteger;
import java.sql.Connection;
import java.sql.Date;
import java.sql.SQLException;
import java.sql.Statement;
import java.sql.Timestamp;

import static io.trino.tempto.assertions.QueryAssert.Row.row;
import static io.trino.tempto.fulfillment.table.MutableTableRequirement.State.CREATED;
import static io.trino.tempto.fulfillment.table.MutableTablesState.mutableTablesState;
import static io.trino.tempto.fulfillment.table.TableRequirements.immutableTable;
import static io.trino.tempto.fulfillment.table.TableRequirements.mutableTable;
import static io.trino.tempto.query.QueryExecutor.param;
import static io.trino.tests.product.TestGroups.JDBC;
import static io.trino.tests.product.hive.AllSimpleTypesTableDefinitions.ALL_HIVE_SIMPLE_TYPES_TEXTFILE;
import static io.trino.tests.product.utils.QueryExecutors.onTrino;
import static java.lang.String.format;
import static java.nio.charset.StandardCharsets.UTF_8;
import static java.sql.JDBCType.BIGINT;
import static java.sql.JDBCType.BOOLEAN;
import static java.sql.JDBCType.CHAR;
import static java.sql.JDBCType.DATE;
import static java.sql.JDBCType.DECIMAL;
import static java.sql.JDBCType.DOUBLE;
import static java.sql.JDBCType.FLOAT;
import static java.sql.JDBCType.INTEGER;
import static java.sql.JDBCType.REAL;
import static java.sql.JDBCType.SMALLINT;
import static java.sql.JDBCType.TIMESTAMP;
import static java.sql.JDBCType.TINYINT;
import static java.sql.JDBCType.VARBINARY;
import static java.sql.JDBCType.VARCHAR;
import static org.assertj.core.api.Assertions.assertThat;

// TODO Consider merging this class with TestJdbcPreparedStatement
public class TestPreparedStatements
        extends ProductTest
{
    private static final String TABLE_NAME = "textfile_all_types";
    private static final String TABLE_NAME_MUTABLE = "all_types_table_name";
    private static final String INSERT_SQL = "INSERT INTO %s VALUES(?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)";
    private static final String SELECT_STAR_SQL = "SELECT * FROM %s";

    private static class ImmutableAllTypesTable
            implements RequirementsProvider
    {
        @Override
        public Requirement getRequirements(Configuration configuration)
        {
            return immutableTable(ALL_HIVE_SIMPLE_TYPES_TEXTFILE);
        }
    }

    private static class MutableAllTypesTable
            implements RequirementsProvider
    {
        @Override
        public Requirement getRequirements(Configuration configuration)
        {
            return mutableTable(ALL_HIVE_SIMPLE_TYPES_TEXTFILE, TABLE_NAME_MUTABLE, CREATED);
        }
    }

    @Test(groups = JDBC)
    @Requires(ImmutableAllTypesTable.class)
    public void preparedSelectApi()
    {
        String selectSql = "SELECT c_int FROM " + TABLE_NAME + " WHERE c_int = ?";
        final int testValue = 2147483647;

        assertThat(onTrino().executeQuery(selectSql, param(INTEGER, testValue))).containsOnly(row(testValue));

        assertThat(onTrino().executeQuery(selectSql, param(INTEGER, null))).hasNoRows();

        assertThat(onTrino().executeQuery(selectSql, param(INTEGER, 2))).hasNoRows();
    }

    @Test(groups = JDBC)
    @Requires(ImmutableAllTypesTable.class)
    public void preparedSelectSql()
            throws SQLException
    {
        String prepareSql = "PREPARE ps1 from SELECT c_int FROM " + TABLE_NAME + " WHERE c_int = ?";
        int testValue = 2147483647;
        String executeSql = "EXECUTE ps1 using ";

        try (Statement statement = connection().createStatement()) {
            statement.execute(prepareSql);

            assertThat(QueryResult.forResultSet(statement.executeQuery(executeSql + testValue)))
                    .containsOnly(row(testValue));
            assertThat(QueryResult.forResultSet(statement.executeQuery(executeSql + "NULL")))
                    .hasNoRows();
            assertThat(QueryResult.forResultSet(statement.executeQuery(executeSql + 2)))
                    .hasNoRows();
        }
    }

    @Test(groups = JDBC)
    @Requires(ImmutableAllTypesTable.class)
    public void executeImmediateSelectSql()
            throws SQLException
    {
        int testValue = 2147483647;
        String executeSql = "EXECUTE IMMEDIATE 'SELECT c_int FROM " + TABLE_NAME + " WHERE c_int = ?' USING ";

        try (Connection conn = connection(); Statement statement = conn.createStatement()) {
            assertThat(QueryResult.forResultSet(statement.executeQuery(executeSql + testValue)))
                    .containsOnly(row(testValue));
            assertThat(QueryResult.forResultSet(statement.executeQuery(executeSql + "NULL")))
                    .hasNoRows();
            assertThat(QueryResult.forResultSet(statement.executeQuery(executeSql + 2)))
                    .hasNoRows();
        }
    }

    @Test(groups = JDBC)
    @Requires(MutableAllTypesTable.class)
    public void preparedInsertVarbinaryApi()
    {
        String tableNameInDatabase = mutableTablesState().get(TABLE_NAME_MUTABLE).getNameInDatabase();
        String insertSqlWithTable = format(INSERT_SQL, tableNameInDatabase);
        String selectSqlWithTable = format(SELECT_STAR_SQL, tableNameInDatabase);

        onTrino().executeQuery(
                insertSqlWithTable,
                param(TINYINT, null),
                param(SMALLINT, null),
                param(INTEGER, null),
                param(BIGINT, null),
                param(FLOAT, null),
                param(DOUBLE, null),
                param(DECIMAL, null),
                param(DECIMAL, null),
                param(TIMESTAMP, null),
                param(DATE, null),
                param(VARCHAR, null),
                param(VARCHAR, null),
                param(CHAR, null),
                param(BOOLEAN, null),
                param(VARBINARY, new byte[] {0, 1, 2, 3, 0, 42, -7}));

        QueryResult result = onTrino().executeQuery(selectSqlWithTable);
        assertColumnTypes(result);
        assertThat(result).containsOnly(
                row(null, null, null, null, null, null, null, null, null, null, null, null, null, null,
                        new byte[] {0, 1, 2, 3, 0, 42, -7}));
    }

    @Test(groups = JDBC)
    @Requires(MutableAllTypesTable.class)
    public void preparedInsertApi()
    {
        String tableNameInDatabase = mutableTablesState().get(TABLE_NAME_MUTABLE).getNameInDatabase();
        String insertSqlWithTable = format(INSERT_SQL, tableNameInDatabase);
        String selectSqlWithTable = format(SELECT_STAR_SQL, tableNameInDatabase);

        onTrino().executeQuery(
                insertSqlWithTable,
                param(TINYINT, 127),
                param(SMALLINT, 32767),
                param(INTEGER, 2147483647),
                param(BIGINT, new BigInteger("9223372036854775807")),
                param(FLOAT, Float.valueOf("123.345")),
                param(DOUBLE, 234.567),
                param(DECIMAL, BigDecimal.valueOf(345)),
                param(DECIMAL, BigDecimal.valueOf(345.678)),
                param(TIMESTAMP, Timestamp.valueOf("2015-05-10 12:15:35")),
                param(DATE, Date.valueOf("2015-05-10")),
                param(VARCHAR, "ala ma kota"),
                param(VARCHAR, "ala ma kot"),
                param(CHAR, "    ala ma"),
                param(BOOLEAN, Boolean.TRUE),
                param(VARBINARY, new byte[] {0, 1, 2, 3, 0, 42, -7}));

        onTrino().executeQuery(
                insertSqlWithTable,
                param(TINYINT, 1),
                param(SMALLINT, 2),
                param(INTEGER, 3),
                param(BIGINT, 4),
                param(FLOAT, Float.valueOf("5.6")),
                param(DOUBLE, 7.8),
                param(DECIMAL, BigDecimal.valueOf(91)),
                param(DECIMAL, BigDecimal.valueOf(2.3)),
                param(TIMESTAMP, Timestamp.valueOf("2012-05-10 1:35:15")),
                param(DATE, Date.valueOf("2014-03-10")),
                param(VARCHAR, "abc"),
                param(VARCHAR, "def"),
                param(CHAR, "       ghi"),
                param(BOOLEAN, Boolean.FALSE),
                param(VARBINARY, new byte[] {0, 1, 2, 3, 0, 42, -7}));

        onTrino().executeQuery(
                insertSqlWithTable,
                param(TINYINT, null),
                param(SMALLINT, null),
                param(INTEGER, null),
                param(BIGINT, null),
                param(FLOAT, null),
                param(DOUBLE, null),
                param(DECIMAL, null),
                param(DECIMAL, null),
                param(TIMESTAMP, null),
                param(DATE, null),
                param(VARCHAR, null),
                param(VARCHAR, null),
                param(CHAR, null),
                param(BOOLEAN, null),
                param(VARBINARY, null));

        QueryResult result = onTrino().executeQuery(selectSqlWithTable);
        assertColumnTypes(result);
        assertThat(result).containsOnly(
                row(
                        127,
                        32767,
                        2147483647,
                        Long.valueOf("9223372036854775807"),
                        Float.valueOf("123.345"),
                        234.567,
                        BigDecimal.valueOf(345),
                        new BigDecimal("345.67800"),
                        Timestamp.valueOf("2015-05-10 12:15:35"),
                        Date.valueOf("2015-05-10"),
                        "ala ma kota",
                        "ala ma kot",
                        "    ala ma",
                        Boolean.TRUE,
                        new byte[] {0, 1, 2, 3, 0, 42, -7}),
                row(
                        1,
                        2,
                        3,
                        4L,
                        Float.valueOf("5.6"),
                        7.8,
                        BigDecimal.valueOf(91),
                        BigDecimal.valueOf(2.3),
                        Timestamp.valueOf("2012-05-10 1:35:15"),
                        Date.valueOf("2014-03-10"),
                        "abc",
                        "def",
                        "       ghi",
                        Boolean.FALSE,
                        new byte[] {0, 1, 2, 3, 0, 42, -7}),
                row(null, null, null, null, null, null, null, null, null, null, null, null, null, null, null));
    }

    @Test(groups = JDBC)
    @Requires(MutableAllTypesTable.class)
    public void preparedInsertSql()
            throws SQLException
    {
        String tableNameInDatabase = mutableTablesState().get(TABLE_NAME_MUTABLE).getNameInDatabase();
        String insertSqlWithTable = "PREPARE ps1 from " + format(INSERT_SQL, tableNameInDatabase);
        String selectSqlWithTable = format(SELECT_STAR_SQL, tableNameInDatabase);
        String executeSql = "EXECUTE ps1 using ";

        try (Connection conn = connection(); Statement statement = conn.createStatement()) {
            statement.execute(insertSqlWithTable);
            insertAndVerify(statement, executeSql, selectSqlWithTable);
        }
    }

    @Test(groups = JDBC)
    @Requires(MutableAllTypesTable.class)
    public void executeImmediateInsertSql()
            throws SQLException
    {
        String tableNameInDatabase = mutableTablesState().get(TABLE_NAME_MUTABLE).getNameInDatabase();
        String selectSqlWithTable = format(SELECT_STAR_SQL, tableNameInDatabase);
        String executeInsertSql = format("EXECUTE IMMEDIATE '%s' using ", format(INSERT_SQL, tableNameInDatabase));

        try (Connection conn = connection(); Statement statement = conn.createStatement()) {
            insertAndVerify(statement, executeInsertSql, selectSqlWithTable);
        }
    }

    private void insertAndVerify(Statement statement, String insertSql, String verifySql)
            throws SQLException
    {
        statement.execute(insertSql +
                "cast(127 as tinyint), " +
                "cast(32767 as smallint), " +
                "2147483647, " +
                "9223372036854775807, " +
                "cast(123.345 as real), " +
                "cast(234.567 as double), " +
                "cast(345 as decimal(10)), " +
                "cast(345.678 as decimal(10,5)), " +
                "timestamp '2015-05-10 12:15:35', " +
                "date '2015-05-10', " +
                "'ala ma kota', " +
                "'ala ma kot', " +
                "cast('ala ma' as char(10)), " +
                "true, " +
                "X'00010203002AF9'");

        statement.execute(insertSql +
                "cast(1 as tinyint), " +
                "cast(2 as smallint), " +
                "3, " +
                "4, " +
                "cast(5.6 as real), " +
                "cast(7.8 as double), " +
                "cast(9 as decimal(10)), " +
                "cast(2.3 as decimal(10,5)), " +
                "timestamp '2012-05-10 1:35:15', " +
                "date '2014-03-10', " +
                "'abc', " +
                "'def', " +
                "cast('ghi' as char(10)), " +
                "false, " +
                "varbinary 'jkl'");

        statement.execute(insertSql +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null");

        QueryResult result = onTrino().executeQuery(verifySql);
        assertColumnTypes(result);
        assertThat(result).containsOnly(
                row(
                        127,
                        32767,
                        2147483647,
                        Long.valueOf("9223372036854775807"),
                        Float.valueOf("123.345"),
                        234.567,
                        BigDecimal.valueOf(345),
                        new BigDecimal("345.67800"),
                        Timestamp.valueOf("2015-05-10 12:15:35"),
                        Date.valueOf("2015-05-10"),
                        "ala ma kota",
                        "ala ma kot",
                        "ala ma    ",
                        Boolean.TRUE,
                        new byte[] {0, 1, 2, 3, 0, 42, -7}),
                row(
                        1,
                        2,
                        3,
                        4,
                        Float.valueOf("5.6"),
                        7.8,
                        BigDecimal.valueOf(9),
                        new BigDecimal("2.30000"),
                        Timestamp.valueOf("2012-05-10 1:35:15"),
                        Date.valueOf("2014-03-10"),
                        "abc",
                        "def",
                        "ghi       ",
                        Boolean.FALSE,
                        "jkl".getBytes(UTF_8)),
                row(null, null, null, null, null, null, null, null, null, null, null, null, null, null, null));
    }

    @Test(groups = JDBC)
    @Requires(MutableAllTypesTable.class)
    public void preparedInsertVarbinarySql()
            throws SQLException
    {
        String tableNameInDatabase = mutableTablesState().get(TABLE_NAME_MUTABLE).getNameInDatabase();
        String insertSqlWithTable = "PREPARE ps1 from " + format(INSERT_SQL, tableNameInDatabase);
        String selectSqlWithTable = format(SELECT_STAR_SQL, tableNameInDatabase);
        String executeSql = "EXECUTE ps1 using ";

        try (Statement statement = connection().createStatement()) {
            statement.execute(insertSqlWithTable);
            insertAndVerifyVarbinary(statement, executeSql, selectSqlWithTable);
        }
    }

    @Test(groups = JDBC)
    @Requires(MutableAllTypesTable.class)
    public void executeImmediateVarbinarySql()
            throws SQLException
    {
        String tableNameInDatabase = mutableTablesState().get(TABLE_NAME_MUTABLE).getNameInDatabase();
        String insertSqlWithTable = "PREPARE ps1 from " + format(INSERT_SQL, tableNameInDatabase);
        String selectSqlWithTable = format(SELECT_STAR_SQL, tableNameInDatabase);
        String executeSql = format("EXECUTE IMMEDIATE '%s' using ", format(INSERT_SQL, tableNameInDatabase));

        try (Connection conn = connection(); Statement statement = conn.createStatement()) {
            statement.execute(insertSqlWithTable);
            insertAndVerifyVarbinary(statement, executeSql, selectSqlWithTable);
        }
    }

    private void insertAndVerifyVarbinary(Statement statement, String insertSql, String verifySql)
            throws SQLException
    {
        statement.execute(insertSql +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "null, " +
                "X'00010203002AF9'");

        QueryResult result = onTrino().executeQuery(verifySql);
        assertColumnTypes(result);
        assertThat(result).containsOnly(
                row(null, null, null, null, null, null, null, null, null, null, null, null, null, null,
                        new byte[] {0, 1, 2, 3, 0, 42, -7}));
    }

    private void assertColumnTypes(QueryResult queryResult)
    {
        assertThat(queryResult).hasColumns(
                TINYINT,
                SMALLINT,
                INTEGER,
                BIGINT,
                REAL,
                DOUBLE,
                DECIMAL,
                DECIMAL,
                TIMESTAMP,
                DATE,
                VARCHAR,
                VARCHAR,
                CHAR,
                BOOLEAN,
                VARBINARY);
    }

    private Connection connection()
    {
        return onTrino().getConnection();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy