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

io.trino.tests.product.TpcTestUtils 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;

import com.google.common.base.CharMatcher;
import org.assertj.core.data.Percentage;
import org.testng.annotations.DataProvider;

import java.math.BigDecimal;
import java.math.RoundingMode;
import java.util.List;
import java.util.Objects;
import java.util.stream.IntStream;
import java.util.stream.Stream;

import static io.trino.tests.product.utils.QueryExecutors.onTrino;
import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.data.Percentage.withPercentage;

public final class TpcTestUtils
{
    // Do not increase this value without examination and multiple local runs.
    // 1E-10 fails roughly once every 100 - 1000 runs.
    // 1.1E-10 should be orders of magnitude safer.
    private static final Percentage DOUBLE_COMPARISON_ACCURACY = withPercentage(1.1E-10);

    private static final String[] TPCDS_TABLES = {
            "call_center",
            "catalog_page",
            "catalog_returns",
            "catalog_sales",
            "customer",
            "customer_address",
            "customer_demographics",
            "date_dim",
            "household_demographics",
            "income_band",
            "inventory",
            "item",
            "promotion",
            "reason",
            "ship_mode",
            "store",
            "store_returns",
            "store_sales",
            "time_dim",
            "warehouse",
            "web_page",
            "web_returns",
            "web_sales",
            "web_site"};

    private static final String[] TPCH_TABLES = {
            "nation",
            "region",
            "part",
            "supplier",
            "partsupp",
            "customer",
            "orders",
            "lineitem"};

    private TpcTestUtils() {}

    public static void createTpchDataset(String catalog)
    {
        onTrino().executeQuery("CREATE SCHEMA IF NOT EXISTS " + catalog + ".tpch");
        for (String table : TPCH_TABLES) {
            onTrino().executeQuery("CREATE TABLE IF NOT EXISTS " + catalog + ".tpch." + table + " WITH (format='PARQUET') AS SELECT * FROM tpch.sf1." + table);
        }
    }

    public static void createTpcdsDataset(String catalog)
    {
        onTrino().executeQuery("CREATE SCHEMA IF NOT EXISTS " + catalog + ".tpcds");
        for (String table : TPCDS_TABLES) {
            onTrino().executeQuery("CREATE TABLE IF NOT EXISTS " + catalog + ".tpcds." + table + " WITH (format='PARQUET') AS SELECT * FROM tpcds.sf1." + table);
        }
    }

    public static void createTpcdsAndTpchDatasets(String catalog)
    {
        createTpcdsDataset(catalog);
        createTpchDataset(catalog);
    }

    @DataProvider
    public static Object[][] tpchQueries()
    {
        return IntStream.range(1, 23)
                .filter(i -> i != 15) // Query creates a view beforehand, not supported here
                .mapToObj(i -> String.format("%02d", i))
                .sorted()
                .map(value -> new Object[] {value})
                .toArray(Object[][]::new);
    }

    @DataProvider
    public static Object[][] tpcdsQueries()
    {
        return Stream.concat(
                        IntStream.range(1, 100)
                                .filter(i -> i != 14) // There are two queries with id 14, 23, 24 and 39
                                .filter(i -> i != 23)
                                .filter(i -> i != 24)
                                .filter(i -> i != 39)
                                .filter(i -> i != 72) // TODO Results for q72 need to be fixed. https://github.com/trinodb/trino/issues/4564
                                .mapToObj(i -> String.format("%02d", i)),
                        Stream.of("14_1", "14_2", "23_1", "23_2", "24_2", "39_1", "39_2")) // 24_1 is ignored as it relies on the order of the result
                .sorted()
                .map(value -> new Object[] {value})
                .toArray(Object[][]::new);
    }

    /**
     * We use hardcoded results from Trino tpch/tpcds suite stored as plain text.
     * Some minor conversions are needed:
     * Double values may different at last bits due to decimal conversion to text.
     * Big decimal values may have trailing zeros and dots
     */
    public static void assertResults(List expected, String query)
    {
        List> result = onTrino().executeQuery(query).rows();
        assertThat(result).hasSize(expected.size());

        for (int i = 0; i < expected.size(); i++) {
            String expectedRow = expected.get(i);
            String[] expectedValues = expectedRow.split("\\|");
            List resultRow = result.get(i);

            assertThat(expectedValues.length).isEqualTo(resultRow.size());
            for (int j = 0; j < expectedValues.length; j++) {
                String expectedValue = expectedValues[j];
                Object resultValue = resultRow.get(j);
                if (resultValue instanceof Double) {
                    expectedValue = trimIfNeeded(expectedValue);
                    BigDecimal expectedDecimal = new BigDecimal(expectedValue);
                    BigDecimal resultDecimal = BigDecimal.valueOf((Double) resultValue);
                    resultDecimal = resultDecimal.setScale(expectedDecimal.scale(), RoundingMode.HALF_DOWN);

                    assertThat(expectedDecimal).isCloseTo(resultDecimal, DOUBLE_COMPARISON_ACCURACY);
                }
                else if (resultValue instanceof BigDecimal) {
                    assertThat(trimIfNeeded(Objects.toString(resultValue))).isEqualTo(trimIfNeeded(expectedValue));
                }
                else {
                    assertThat(Objects.toString(resultValue)).isEqualTo(expectedValue);
                }
            }
        }
    }

    public static String trimIfNeeded(String value)
    {
        // Trim zeros and a trailing dot to match Big Decimals in different formats
        if (value.contains(".")) {
            return CharMatcher.is('.').trimTrailingFrom(
                    CharMatcher.is('0').trimTrailingFrom(value));
        }
        return value;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy