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

io.trino.plugin.hive.metastore.file.TableMetadata Maven / Gradle / Ivy

/*
 * 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.plugin.hive.metastore.file;

import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import io.trino.metastore.HiveBucketProperty;
import io.trino.metastore.Storage;
import io.trino.metastore.StorageFormat;
import io.trino.metastore.Table;
import io.trino.plugin.hive.HiveStorageFormat;

import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.OptionalLong;

import static com.google.common.base.Preconditions.checkArgument;
import static io.trino.metastore.StorageFormat.VIEW_STORAGE_FORMAT;
import static io.trino.plugin.hive.HiveSchemaProperties.LOCATION_PROPERTY;
import static io.trino.plugin.hive.TableType.EXTERNAL_TABLE;
import static java.util.Objects.requireNonNull;

public class TableMetadata
{
    private final Optional writerVersion;
    private final Optional owner;
    private final String tableType;
    private final List dataColumns;
    private final List partitionColumns;
    private final Map parameters;

    private final Optional storageFormat;
    private final Optional originalStorageFormat;
    private final Optional bucketProperty;
    private final Map serdeParameters;

    private final Optional externalLocation;

    private final Optional viewOriginalText;
    private final Optional viewExpandedText;

    private final Map columnStatistics;

    @JsonCreator
    public TableMetadata(
            @JsonProperty("writerVersion") Optional writerVersion,
            @JsonProperty("owner") Optional owner,
            @JsonProperty("tableType") String tableType,
            @JsonProperty("dataColumns") List dataColumns,
            @JsonProperty("partitionColumns") List partitionColumns,
            @JsonProperty("parameters") Map parameters,
            @JsonProperty("storageFormat") Optional storageFormat,
            @JsonProperty("originalStorageFormat") Optional originalStorageFormat,
            @JsonProperty("bucketProperty") Optional bucketProperty,
            @JsonProperty("serdeParameters") Map serdeParameters,
            @JsonProperty("externalLocation") Optional externalLocation,
            @JsonProperty("viewOriginalText") Optional viewOriginalText,
            @JsonProperty("viewExpandedText") Optional viewExpandedText,
            @JsonProperty("columnStatistics") Map columnStatistics)
    {
        this.writerVersion = requireNonNull(writerVersion, "writerVersion is null");
        this.owner = requireNonNull(owner, "owner is null");
        this.tableType = requireNonNull(tableType, "tableType is null");
        this.dataColumns = ImmutableList.copyOf(requireNonNull(dataColumns, "dataColumns is null"));
        this.partitionColumns = ImmutableList.copyOf(requireNonNull(partitionColumns, "partitionColumns is null"));
        this.parameters = ImmutableMap.copyOf(requireNonNull(parameters, "parameters is null"));

        this.storageFormat = requireNonNull(storageFormat, "storageFormat is null");
        this.originalStorageFormat = requireNonNull(originalStorageFormat, "originalStorageFormat is null");
        this.bucketProperty = requireNonNull(bucketProperty, "bucketProperty is null");
        this.serdeParameters = requireNonNull(serdeParameters, "serdeParameters is null");
        this.externalLocation = requireNonNull(externalLocation, "externalLocation is null");
        if (tableType.equals(EXTERNAL_TABLE.name())) {
            checkArgument(externalLocation.isPresent(), "External location is required for external tables");
        }
        else {
            checkArgument(externalLocation.isEmpty(), "External location is only allowed for external tables");
        }

        this.viewOriginalText = requireNonNull(viewOriginalText, "viewOriginalText is null");
        this.viewExpandedText = requireNonNull(viewExpandedText, "viewExpandedText is null");
        this.columnStatistics = ImmutableMap.copyOf(requireNonNull(columnStatistics, "columnStatistics is null"));
        checkArgument(partitionColumns.isEmpty() || columnStatistics.isEmpty(), "column statistics cannot be set for partitioned table");
    }

    public TableMetadata(String currentVersion, Table table)
    {
        writerVersion = Optional.of(requireNonNull(currentVersion, "currentVersion is null"));
        owner = table.getOwner();
        tableType = table.getTableType();
        dataColumns = Column.fromMetastoreModel(table.getDataColumns());
        partitionColumns = Column.fromMetastoreModel(table.getPartitionColumns());
        parameters = table.getParameters();

        StorageFormat tableFormat = table.getStorage().getStorageFormat();
        storageFormat = Arrays.stream(HiveStorageFormat.values())
                .filter(format -> tableFormat.equals(format.toStorageFormat()))
                .findFirst();
        if (storageFormat.isPresent()) {
            originalStorageFormat = Optional.empty();
        }
        else {
            originalStorageFormat = Optional.of(tableFormat);
        }
        bucketProperty = table.getStorage().getBucketProperty();
        serdeParameters = table.getStorage().getSerdeParameters();

        if (tableType.equals(EXTERNAL_TABLE.name())) {
            externalLocation = Optional.of(table.getStorage().getLocation());
        }
        else {
            externalLocation = Optional.empty();
        }

        viewOriginalText = table.getViewOriginalText();
        viewExpandedText = table.getViewExpandedText();
        columnStatistics = ImmutableMap.of();
    }

    @JsonProperty
    public Optional getWriterVersion()
    {
        return writerVersion;
    }

    @JsonProperty
    public Optional getOwner()
    {
        return owner;
    }

    @JsonProperty
    public String getTableType()
    {
        return tableType;
    }

    @JsonProperty
    public List getDataColumns()
    {
        return dataColumns;
    }

    @JsonProperty
    public List getPartitionColumns()
    {
        return partitionColumns;
    }

    public Optional getColumn(String name)
    {
        for (Column partitionColumn : partitionColumns) {
            if (partitionColumn.getName().equals(name)) {
                return Optional.of(partitionColumn);
            }
        }
        for (Column dataColumn : dataColumns) {
            if (dataColumn.getName().equals(name)) {
                return Optional.of(dataColumn);
            }
        }
        return Optional.empty();
    }

    @JsonProperty
    public Map getParameters()
    {
        return parameters;
    }

    @JsonProperty
    public Optional getStorageFormat()
    {
        return storageFormat;
    }

    @JsonProperty
    public Optional getOriginalStorageFormat()
    {
        return originalStorageFormat;
    }

    @JsonProperty
    public Optional getBucketProperty()
    {
        return bucketProperty;
    }

    @JsonProperty
    public Map getSerdeParameters()
    {
        return serdeParameters;
    }

    @JsonProperty
    public Optional getExternalLocation()
    {
        return externalLocation;
    }

    @JsonProperty
    public Optional getViewOriginalText()
    {
        return viewOriginalText;
    }

    @JsonProperty
    public Optional getViewExpandedText()
    {
        return viewExpandedText;
    }

    @JsonProperty
    public Map getColumnStatistics()
    {
        return columnStatistics;
    }

    public TableMetadata withDataColumns(String currentVersion, List dataColumns)
    {
        return new TableMetadata(
                Optional.of(requireNonNull(currentVersion, "currentVersion is null")),
                owner,
                tableType,
                dataColumns,
                partitionColumns,
                parameters,
                storageFormat,
                originalStorageFormat,
                bucketProperty,
                serdeParameters,
                externalLocation,
                viewOriginalText,
                viewExpandedText,
                columnStatistics);
    }

    public TableMetadata withPartitionColumns(String currentVersion, List partitionColumns)
    {
        return new TableMetadata(
                Optional.of(requireNonNull(currentVersion, "currentVersion is null")),
                owner,
                tableType,
                dataColumns,
                partitionColumns,
                parameters,
                storageFormat,
                originalStorageFormat,
                bucketProperty,
                serdeParameters,
                externalLocation,
                viewOriginalText,
                viewExpandedText,
                columnStatistics);
    }

    public TableMetadata withParameters(String currentVersion, Map parameters)
    {
        return new TableMetadata(
                Optional.of(requireNonNull(currentVersion, "currentVersion is null")),
                owner,
                tableType,
                dataColumns,
                partitionColumns,
                parameters,
                storageFormat,
                originalStorageFormat,
                bucketProperty,
                serdeParameters,
                externalLocation,
                viewOriginalText,
                viewExpandedText,
                columnStatistics);
    }

    public TableMetadata withColumnStatistics(String currentVersion, Map columnStatistics)
    {
        return new TableMetadata(
                Optional.of(requireNonNull(currentVersion, "currentVersion is null")),
                owner,
                tableType,
                dataColumns,
                partitionColumns,
                parameters,
                storageFormat,
                originalStorageFormat,
                bucketProperty,
                serdeParameters,
                externalLocation,
                viewOriginalText,
                viewExpandedText,
                columnStatistics);
    }

    public Table toTable(String databaseName, String tableName, String location)
    {
        return new Table(
                databaseName,
                tableName,
                owner,
                tableType,
                Storage.builder()
                        .setLocation(externalLocation.or(() -> Optional.ofNullable(parameters.get(LOCATION_PROPERTY))).orElse(location))
                        .setStorageFormat(storageFormat.map(HiveStorageFormat::toStorageFormat)
                                .or(() -> originalStorageFormat)
                                .orElse(VIEW_STORAGE_FORMAT))
                        .setBucketProperty(bucketProperty)
                        .setSerdeParameters(serdeParameters)
                        .build(),
                Column.toMetastoreModel(dataColumns),
                Column.toMetastoreModel(partitionColumns),
                parameters,
                viewOriginalText,
                viewExpandedText,
                OptionalLong.empty());
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy