Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
io.trino.spi.connector.ConnectorMetadata 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.spi.connector;
import io.airlift.slice.Slice;
import io.trino.spi.ErrorCode;
import io.trino.spi.Experimental;
import io.trino.spi.TrinoException;
import io.trino.spi.expression.Call;
import io.trino.spi.expression.ConnectorExpression;
import io.trino.spi.expression.Constant;
import io.trino.spi.expression.Variable;
import io.trino.spi.function.AggregationFunctionMetadata;
import io.trino.spi.function.BoundSignature;
import io.trino.spi.function.FunctionDependencyDeclaration;
import io.trino.spi.function.FunctionId;
import io.trino.spi.function.FunctionMetadata;
import io.trino.spi.function.LanguageFunction;
import io.trino.spi.function.SchemaFunctionName;
import io.trino.spi.function.table.ConnectorTableFunctionHandle;
import io.trino.spi.predicate.TupleDomain;
import io.trino.spi.security.GrantInfo;
import io.trino.spi.security.Privilege;
import io.trino.spi.security.RoleGrant;
import io.trino.spi.security.TrinoPrincipal;
import io.trino.spi.statistics.ComputedStatistics;
import io.trino.spi.statistics.TableStatistics;
import io.trino.spi.statistics.TableStatisticsMetadata;
import io.trino.spi.type.Type;
import jakarta.annotation.Nullable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.OptionalInt;
import java.util.OptionalLong;
import java.util.Set;
import java.util.concurrent.CompletableFuture;
import java.util.function.UnaryOperator;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.stream.Stream;
import static io.trino.spi.ErrorType.EXTERNAL;
import static io.trino.spi.StandardErrorCode.FUNCTION_IMPLEMENTATION_ERROR;
import static io.trino.spi.StandardErrorCode.GENERIC_INTERNAL_ERROR;
import static io.trino.spi.StandardErrorCode.NOT_FOUND;
import static io.trino.spi.StandardErrorCode.NOT_SUPPORTED;
import static io.trino.spi.StandardErrorCode.TABLE_NOT_FOUND;
import static io.trino.spi.StandardErrorCode.UNSUPPORTED_TABLE_TYPE;
import static io.trino.spi.connector.SaveMode.IGNORE;
import static io.trino.spi.connector.SaveMode.REPLACE;
import static io.trino.spi.expression.Constant.FALSE;
import static io.trino.spi.expression.StandardFunctions.AND_FUNCTION_NAME;
import static java.util.Collections.emptyList;
import static java.util.Collections.emptyMap;
import static java.util.Locale.ENGLISH;
import static java.util.function.Function.identity;
import static java.util.stream.Collectors.collectingAndThen;
import static java.util.stream.Collectors.toMap;
import static java.util.stream.Collectors.toUnmodifiableList;
import static java.util.stream.Collectors.toUnmodifiableSet;
public interface ConnectorMetadata
{
String MODIFYING_ROWS_MESSAGE = "This connector does not support modifying table rows";
/**
* Checks if a schema exists. The connector may have schemas that exist
* but are not enumerable via {@link #listSchemaNames}.
*/
default boolean schemaExists(ConnectorSession session, String schemaName)
{
if (!schemaName.equals(schemaName.toLowerCase(ENGLISH))) {
// Currently, Trino schemas are always lowercase, so this one cannot exist (https://github.com/trinodb/trino/issues/17)
return false;
}
return listSchemaNames(session).stream()
// Lower-casing is done by callers of listSchemaNames (see MetadataManager)
.map(schema -> schema.toLowerCase(ENGLISH))
.anyMatch(schemaName::equals);
}
/**
* Returns the schemas provided by this connector.
*/
default List listSchemaNames(ConnectorSession session)
{
return emptyList();
}
/**
* Returns a table handle for the specified table name, or {@code null} if {@code tableName} relation does not exist
* or is not a table (e.g. is a view, or a materialized view).
*
* @throws TrinoException implementation can throw this exception when {@code tableName} refers to a table that
* cannot be queried.
* @see #getView(ConnectorSession, SchemaTableName)
* @see #getMaterializedView(ConnectorSession, SchemaTableName)
* @deprecated Implement {@link #getTableHandle(ConnectorSession, SchemaTableName, Optional, Optional)}.
*/
@Nullable
@Deprecated
default ConnectorTableHandle getTableHandle(ConnectorSession session, SchemaTableName tableName)
{
return null;
}
/**
* Returns a table handle for the specified table name and version, or {@code null} if {@code tableName} relation does not exist
* or is not a table (e.g. is a view, or a materialized view).
*
* @throws TrinoException implementation can throw this exception when {@code tableName} refers to a table that
* cannot be queried.
* @see #getView(ConnectorSession, SchemaTableName)
* @see #getMaterializedView(ConnectorSession, SchemaTableName)
*/
@Nullable
default ConnectorTableHandle getTableHandle(
ConnectorSession session,
SchemaTableName tableName,
Optional startVersion,
Optional endVersion)
{
ConnectorTableHandle tableHandle = getTableHandle(session, tableName);
if (tableHandle == null) {
// Not found
return null;
}
if (startVersion.isEmpty() && endVersion.isEmpty()) {
return tableHandle;
}
throw new TrinoException(NOT_SUPPORTED, "This connector does not support versioned tables");
}
/**
* Create initial handle for execution of table procedure. The handle will be used through planning process. It will be converted to final
* handle used for execution via @{link {@link ConnectorMetadata#beginTableExecute}
*
* If connector does not support execution with retries, the method should throw:
*
* new TrinoException(NOT_SUPPORTED, "This connector does not support query retries")
*
* unless {@code retryMode} is set to {@code NO_RETRIES}.
*/
default Optional getTableHandleForExecute(
ConnectorSession session,
ConnectorTableHandle tableHandle,
String procedureName,
Map executeProperties,
RetryMode retryMode)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support table procedures");
}
default Optional getLayoutForTableExecute(ConnectorSession session, ConnectorTableExecuteHandle tableExecuteHandle)
{
return Optional.empty();
}
/**
* Begin execution of table procedure
*/
default BeginTableExecuteResult beginTableExecute(ConnectorSession session, ConnectorTableExecuteHandle tableExecuteHandle, ConnectorTableHandle updatedSourceTableHandle)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata getTableHandleForExecute() is implemented without beginTableExecute()");
}
/**
* Finish table execute
*
* @param fragments all fragments returned by {@link ConnectorPageSink#finish()}
*/
default void finishTableExecute(ConnectorSession session, ConnectorTableExecuteHandle tableExecuteHandle, Collection fragments, List tableExecuteState)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata getTableHandleForExecute() is implemented without finishTableExecute()");
}
/**
* Execute a {@link TableProcedureExecutionMode#coordinatorOnly() coordinator-only} table procedure.
*/
default void executeTableExecute(ConnectorSession session, ConnectorTableExecuteHandle tableExecuteHandle)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata executeTableExecute() is not implemented");
}
/**
* Returns the system table for the specified table name, if one exists.
* The system tables handled via this method differ form those returned by {@link Connector#getSystemTables()}.
* The former mechanism allows dynamic resolution of system tables, while the latter is
* based on static list of system tables built during startup.
*/
default Optional getSystemTable(ConnectorSession session, SchemaTableName tableName)
{
return Optional.empty();
}
/**
* Return a table handle whose partitioning is converted to the provided partitioning handle,
* but otherwise identical to the provided table handle.
* The provided table handle must be one that the connector can transparently convert to from
* the original partitioning handle associated with the provided table handle,
* as promised by {@link #getCommonPartitioningHandle}.
*/
default ConnectorTableHandle makeCompatiblePartitioning(ConnectorSession session, ConnectorTableHandle tableHandle, ConnectorPartitioningHandle partitioningHandle)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata getCommonPartitioningHandle() is implemented without makeCompatiblePartitioning()");
}
/**
* Return a partitioning handle which the connector can transparently convert both {@code left} and {@code right} into.
*/
default Optional getCommonPartitioningHandle(ConnectorSession session, ConnectorPartitioningHandle left, ConnectorPartitioningHandle right)
{
if (left.equals(right)) {
return Optional.of(left);
}
return Optional.empty();
}
/**
* Return schema table name for the specified table handle.
* This method is useful when requiring only {@link SchemaTableName} without other objects.
*
* @throws RuntimeException if table handle is no longer valid
*/
default SchemaTableName getTableName(ConnectorSession session, ConnectorTableHandle table)
{
return getTableSchema(session, table).getTable();
}
/**
* Return table schema definition for the specified table handle.
* This method is useful when getting full table metadata is expensive.
*
* @throws RuntimeException if table handle is no longer valid
*/
default ConnectorTableSchema getTableSchema(ConnectorSession session, ConnectorTableHandle table)
{
return getTableMetadata(session, table).getTableSchema();
}
/**
* Return the metadata for the specified table handle.
*
* @throws RuntimeException if table handle is no longer valid
*/
default ConnectorTableMetadata getTableMetadata(ConnectorSession session, ConnectorTableHandle table)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata getTableHandle() is implemented without getTableMetadata()");
}
default Optional getInfo(ConnectorTableHandle table)
{
return Optional.empty();
}
/**
* List table, view and materialized view names, possibly filtered by schema. An empty list is returned if none match.
* An empty list is returned also when schema name does not refer to an existing schema.
*
* @see #listViews(ConnectorSession, Optional)
* @see #listMaterializedViews(ConnectorSession, Optional)
*/
default List listTables(ConnectorSession session, Optional schemaName)
{
return emptyList();
}
/**
* List table, view and materialized view names, possibly filtered by schema.
*/
default Map getRelationTypes(ConnectorSession session, Optional schemaName)
{
Set materializedViews = Set.copyOf(listMaterializedViews(session, schemaName));
Set views = Set.copyOf(listViews(session, schemaName));
return listTables(session, schemaName).stream()
.collect(toMap(
identity(),
relation -> {
if (materializedViews.contains(relation)) {
return RelationType.MATERIALIZED_VIEW;
}
if (views.contains(relation)) {
return RelationType.VIEW;
}
return RelationType.TABLE;
}));
}
/**
* Gets all of the columns on the specified table, or an empty map if the columns cannot be enumerated.
*
* @throws RuntimeException if table handle is no longer valid
*/
default Map getColumnHandles(ConnectorSession session, ConnectorTableHandle tableHandle)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata getTableHandle() is implemented without getColumnHandles()");
}
/**
* Gets the metadata for the specified table column.
*
* @throws RuntimeException if table or column handles are no longer valid
*/
default ColumnMetadata getColumnMetadata(ConnectorSession session, ConnectorTableHandle tableHandle, ColumnHandle columnHandle)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata getTableHandle() is implemented without getColumnMetadata()");
}
/**
* Gets the metadata for all columns that match the specified table prefix. Columns of views and materialized views are not included.
*
* @deprecated use {@link #streamTableColumns} which handles redirected tables
*/
@Deprecated
default Map> listTableColumns(ConnectorSession session, SchemaTablePrefix prefix)
{
return emptyMap();
}
/**
* Gets the metadata for all columns that match the specified table prefix. Redirected table names are included, but
* the column metadata for them is not. Views and materialized views are not included.
*
* @deprecated Implement {@link #streamRelationColumns}.
*/
@Deprecated
default Iterator streamTableColumns(ConnectorSession session, SchemaTablePrefix prefix)
{
return listTableColumns(session, prefix).entrySet().stream()
.map(entry -> TableColumnsMetadata.forTable(entry.getKey(), entry.getValue()))
.iterator();
}
/**
* Gets columns for all relations (tables, views, materialized views), possibly filtered by schemaName.
* (e.g. for all relations that would be returned by {@link #listTables(ConnectorSession, Optional)}).
* Redirected table names are included, but the comment for them is not.
*/
@Experimental(eta = "2024-01-01")
default Iterator streamRelationColumns(
ConnectorSession session,
Optional schemaName,
UnaryOperator> relationFilter)
{
Map relationColumns = new HashMap<>();
// Collect column metadata from tables
SchemaTablePrefix prefix = schemaName.map(SchemaTablePrefix::new)
.orElseGet(SchemaTablePrefix::new);
streamTableColumns(session, prefix)
.forEachRemaining(columnsMetadata -> {
SchemaTableName name = columnsMetadata.getTable();
relationColumns.put(name, columnsMetadata.getColumns()
.map(columns -> RelationColumnsMetadata.forTable(name, columns))
.orElseGet(() -> RelationColumnsMetadata.forRedirectedTable(name)));
});
// Collect column metadata from views. if table and view names overlap, the view wins
for (Map.Entry entry : getViews(session, schemaName).entrySet()) {
relationColumns.put(entry.getKey(), RelationColumnsMetadata.forView(entry.getKey(), entry.getValue().getColumns()));
}
// if view and materialized view names overlap, the materialized view wins
for (Map.Entry entry : getMaterializedViews(session, schemaName).entrySet()) {
relationColumns.put(entry.getKey(), RelationColumnsMetadata.forMaterializedView(entry.getKey(), entry.getValue().getColumns()));
}
return relationFilter.apply(relationColumns.keySet()).stream()
.map(relationColumns::get)
.iterator();
}
/**
* Gets comments for all relations (tables, views, materialized views), possibly filtered by schemaName.
* (e.g. for all relations that would be returned by {@link #listTables(ConnectorSession, Optional)}).
* Redirected table names are included, but the comment for them is not.
*/
@Experimental(eta = "2024-01-01")
default Iterator streamRelationComments(ConnectorSession session, Optional schemaName, UnaryOperator> relationFilter)
{
List materializedViews = getMaterializedViews(session, schemaName).entrySet().stream()
.map(entry -> RelationCommentMetadata.forRelation(entry.getKey(), entry.getValue().getComment()))
.toList();
Set mvNames = materializedViews.stream()
.map(RelationCommentMetadata::name)
.collect(toUnmodifiableSet());
List views = getViews(session, schemaName).entrySet().stream()
.map(entry -> RelationCommentMetadata.forRelation(entry.getKey(), entry.getValue().getComment()))
.filter(commentMetadata -> !mvNames.contains(commentMetadata.name()))
.toList();
Set mvAndViewNames = Stream.concat(mvNames.stream(), views.stream().map(RelationCommentMetadata::name))
.collect(toUnmodifiableSet());
List tables = listTables(session, schemaName).stream()
.filter(tableName -> !mvAndViewNames.contains(tableName))
.collect(collectingAndThen(toUnmodifiableSet(), relationFilter)).stream()
.map(tableName -> {
if (redirectTable(session, tableName).isPresent()) {
return RelationCommentMetadata.forRedirectedTable(tableName);
}
try {
ConnectorTableHandle tableHandle = getTableHandle(session, tableName, Optional.empty(), Optional.empty());
if (tableHandle == null) {
// disappeared during listing
return null;
}
return RelationCommentMetadata.forRelation(tableName, getTableMetadata(session, tableHandle).getComment());
}
catch (RuntimeException e) {
boolean silent = false;
if (e instanceof TrinoException trinoException) {
ErrorCode errorCode = trinoException.getErrorCode();
silent = errorCode.equals(UNSUPPORTED_TABLE_TYPE.toErrorCode()) ||
// e.g. table deleted concurrently
errorCode.equals(TABLE_NOT_FOUND.toErrorCode()) ||
errorCode.equals(NOT_FOUND.toErrorCode()) ||
// e.g. Iceberg/Delta table being deleted concurrently resulting in failure to load metadata from filesystem
errorCode.getType() == EXTERNAL;
}
if (silent) {
Helper.juliLogger.log(Level.FINE, e, () -> "Failed to get metadata for table: " + tableName);
}
else {
// getTableHandle or getTableMetadata failed call may fail if table disappeared during listing or is unsupported.
Helper.juliLogger.log(Level.WARNING, e, () -> "Failed to get metadata for table: " + tableName);
}
// Since the getTableHandle did not return null (i.e. succeeded or failed), we assume the table would be returned by listTables
return RelationCommentMetadata.forRelation(tableName, Optional.empty());
}
})
.filter(Objects::nonNull)
.toList();
Set availableMvAndViews = relationFilter.apply(mvAndViewNames);
return Stream.of(
materializedViews.stream()
.filter(commentMetadata -> availableMvAndViews.contains(commentMetadata.name())),
views.stream()
.filter(commentMetadata -> availableMvAndViews.contains(commentMetadata.name())),
tables.stream())
.flatMap(identity())
.iterator();
}
/**
* Get statistics for table.
*/
default TableStatistics getTableStatistics(ConnectorSession session, ConnectorTableHandle tableHandle)
{
return TableStatistics.empty();
}
/**
* Creates a schema.
*/
default void createSchema(ConnectorSession session, String schemaName, Map properties, TrinoPrincipal owner)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support creating schemas");
}
/**
* Drops the specified schema.
*
* @throws TrinoException with {@code SCHEMA_NOT_EMPTY} if {@code cascade} is false and the schema is not empty
*/
default void dropSchema(ConnectorSession session, String schemaName, boolean cascade)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support dropping schemas");
}
/**
* Renames the specified schema.
*/
default void renameSchema(ConnectorSession session, String source, String target)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support renaming schemas");
}
/**
* Sets the user/role on the specified schema.
*/
default void setSchemaAuthorization(ConnectorSession session, String schemaName, TrinoPrincipal principal)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting an owner on a schema");
}
/**
* Creates a table using the specified table metadata.
*
* @throws TrinoException with {@code ALREADY_EXISTS} if the table already exists and {@code ignoreExisting} is not set
* @deprecated use {@link #createTable(ConnectorSession session, ConnectorTableMetadata tableMetadata, SaveMode saveMode)}
*/
@Deprecated
default void createTable(ConnectorSession session, ConnectorTableMetadata tableMetadata, boolean ignoreExisting)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support creating tables");
}
/**
* Creates a table using the specified table metadata.
* IGNORE means the table is created using CREATE ... IF NOT EXISTS syntax.
* REPLACE means the table is created using CREATE OR REPLACE syntax.
*
* @throws TrinoException with {@code ALREADY_EXISTS} if the table already exists and {@param savemode} is FAIL.
*/
default void createTable(ConnectorSession session, ConnectorTableMetadata tableMetadata, SaveMode saveMode)
{
if (saveMode == REPLACE) {
throw new TrinoException(NOT_SUPPORTED, "This connector does not support replacing tables");
}
// Delegate to deprecated SPI to not break existing connectors
createTable(session, tableMetadata, saveMode == IGNORE);
}
/**
* Drops the specified table
*
* @throws RuntimeException if the table cannot be dropped or table handle is no longer valid
*/
default void dropTable(ConnectorSession session, ConnectorTableHandle tableHandle)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support dropping tables");
}
/**
* Truncates the specified table
*
* @throws RuntimeException if the table cannot be dropped or table handle is no longer valid
*/
default void truncateTable(ConnectorSession session, ConnectorTableHandle tableHandle)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support truncating tables");
}
/**
* Rename the specified table
*/
default void renameTable(ConnectorSession session, ConnectorTableHandle tableHandle, SchemaTableName newTableName)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support renaming tables");
}
/**
* Set properties to the specified table
*/
default void setTableProperties(ConnectorSession session, ConnectorTableHandle tableHandle, Map> properties)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting table properties");
}
/**
* Comments to the specified table
*/
default void setTableComment(ConnectorSession session, ConnectorTableHandle tableHandle, Optional comment)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting table comments");
}
/**
* Comments to the specified view
*/
default void setViewComment(ConnectorSession session, SchemaTableName viewName, Optional comment)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting view comments");
}
/**
* Comments to the specified view column.
*/
default void setViewColumnComment(ConnectorSession session, SchemaTableName viewName, String columnName, Optional comment)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting view column comments");
}
/**
* Comments to the specified materialized view column.
*/
default void setMaterializedViewColumnComment(ConnectorSession session, SchemaTableName viewName, String columnName, Optional comment)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting materialized view column comments");
}
/**
* Comments to the specified column
*/
default void setColumnComment(ConnectorSession session, ConnectorTableHandle tableHandle, ColumnHandle column, Optional comment)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting column comments");
}
/**
* Add the specified column
*/
default void addColumn(ConnectorSession session, ConnectorTableHandle tableHandle, ColumnMetadata column)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support adding columns");
}
/**
* Add the specified field, potentially nested, to a row.
*
* @param parentPath path to a field within the column, without leaf field name.
*/
@Experimental(eta = "2023-06-01") // TODO add support for rows inside arrays and maps and for anonymous row fields
default void addField(ConnectorSession session, ConnectorTableHandle tableHandle, List parentPath, String fieldName, Type type, boolean ignoreExisting)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support adding fields");
}
/**
* Set the specified column type
*/
@Experimental(eta = "2023-04-01")
default void setColumnType(ConnectorSession session, ConnectorTableHandle tableHandle, ColumnHandle column, Type type)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting column types");
}
/**
* Set the specified field type
*
* @param fieldPath path starting with column name. The path is always lower-cased. It cannot be an empty or a single element.
*/
@Experimental(eta = "2023-09-01")
default void setFieldType(ConnectorSession session, ConnectorTableHandle tableHandle, List fieldPath, Type type)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting field types");
}
/**
* Drop a not null constraint on the specified column
*/
default void dropNotNullConstraint(ConnectorSession session, ConnectorTableHandle tableHandle, ColumnHandle column)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support dropping a not null constraint");
}
/**
* Sets the user/role on the specified table.
*/
default void setTableAuthorization(ConnectorSession session, SchemaTableName tableName, TrinoPrincipal principal)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting an owner on a table");
}
/**
* Rename the specified column
*/
default void renameColumn(ConnectorSession session, ConnectorTableHandle tableHandle, ColumnHandle source, String target)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support renaming columns");
}
/**
* Rename the specified field, potentially nested, to a row.
*
* @param fieldPath path starting with column name.
* @param target the new field name. The field position and nested level shouldn't be changed.
*/
@Experimental(eta = "2023-09-01") // TODO add support for rows inside arrays and maps and for anonymous row fields
default void renameField(ConnectorSession session, ConnectorTableHandle tableHandle, List fieldPath, String target)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support renaming fields");
}
/**
* Drop the specified column
*/
default void dropColumn(ConnectorSession session, ConnectorTableHandle tableHandle, ColumnHandle column)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support dropping columns");
}
/**
* Drop the specified field, potentially nested, from a row.
*
* @param fieldPath path to a field within the column, without leading column name.
*/
@Experimental(eta = "2023-05-01") // TODO add support for rows inside arrays and maps and for anonymous row fields
default void dropField(ConnectorSession session, ConnectorTableHandle tableHandle, ColumnHandle column, List fieldPath)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support dropping fields");
}
/**
* Get the physical layout for a new table.
*/
default Optional getNewTableLayout(ConnectorSession session, ConnectorTableMetadata tableMetadata)
{
return Optional.empty();
}
/**
* Return the effective {@link io.trino.spi.type.Type} that is supported by the connector for the given type.
* If {@link Optional#empty()} is returned, the type will be used as is during table creation which may or may not be supported by the connector.
* The effective type shall be a type that is cast-compatible with the input type.
*/
@Experimental(eta = "2024-01-31")
default Optional getSupportedType(ConnectorSession session, Map tableProperties, Type type)
{
return Optional.empty();
}
/**
* Get the physical layout for inserting into an existing table.
*/
default Optional getInsertLayout(ConnectorSession session, ConnectorTableHandle tableHandle)
{
ConnectorTableProperties properties = getTableProperties(session, tableHandle);
return properties.getTablePartitioning()
.map(partitioning -> {
Map columnNamesByHandle = getColumnHandles(session, tableHandle).entrySet().stream()
.collect(toMap(Map.Entry::getValue, Map.Entry::getKey));
List partitionColumns = partitioning.getPartitioningColumns().stream()
.map(columnNamesByHandle::get)
.collect(toUnmodifiableList());
return new ConnectorTableLayout(partitioning.getPartitioningHandle(), partitionColumns);
});
}
/**
* Describes statistics that must be collected during a write.
*/
default TableStatisticsMetadata getStatisticsCollectionMetadataForWrite(ConnectorSession session, ConnectorTableMetadata tableMetadata)
{
return TableStatisticsMetadata.empty();
}
/**
* Describe statistics that must be collected during a statistics collection
*/
default ConnectorAnalyzeMetadata getStatisticsCollectionMetadata(ConnectorSession session, ConnectorTableHandle tableHandle, Map analyzeProperties)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support analyze");
}
/**
* Begin statistics collection
*/
default ConnectorTableHandle beginStatisticsCollection(ConnectorSession session, ConnectorTableHandle tableHandle)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata getStatisticsCollectionMetadata() is implemented without beginStatisticsCollection()");
}
/**
* Finish statistics collection
*/
default void finishStatisticsCollection(ConnectorSession session, ConnectorTableHandle tableHandle, Collection computedStatistics)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata beginStatisticsCollection() is implemented without finishStatisticsCollection()");
}
/**
* Begin the atomic creation of a table with data.
*
* If connector does not support execution with retries, the method should throw:
*
* new TrinoException(NOT_SUPPORTED, "This connector does not support query retries")
*
* unless {@code retryMode} is set to {@code NO_RETRIES}.
*
* @deprecated use {@link #beginCreateTable(ConnectorSession session, ConnectorTableMetadata tableMetadata, Optional layout, RetryMode retryMode, boolean replace)}
*/
@Deprecated
default ConnectorOutputTableHandle beginCreateTable(ConnectorSession session, ConnectorTableMetadata tableMetadata, Optional layout, RetryMode retryMode)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support creating tables with data");
}
/**
* Begin the atomic creation of a table with data.
*
*
* If connector does not support execution with retries, the method should throw:
*
* new TrinoException(NOT_SUPPORTED, "This connector does not support query retries")
*
* unless {@code retryMode} is set to {@code NO_RETRIES}.
*/
default ConnectorOutputTableHandle beginCreateTable(ConnectorSession session, ConnectorTableMetadata tableMetadata, Optional layout, RetryMode retryMode, boolean replace)
{
// Redirect to deprecated SPI to not break existing connectors
if (!replace) {
return beginCreateTable(session, tableMetadata, layout, retryMode);
}
throw new TrinoException(NOT_SUPPORTED, "This connector does not support replacing tables");
}
/**
* Finish a table creation with data after the data is written.
*/
default Optional finishCreateTable(ConnectorSession session, ConnectorOutputTableHandle tableHandle, Collection fragments, Collection computedStatistics)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata beginCreateTable() is implemented without finishCreateTable()");
}
/**
* Start a query. This notification is triggered before any other metadata access.
*/
default void beginQuery(ConnectorSession session) {}
/**
* Cleanup after a query. This is the very last notification after the query finishes, whether it succeeds or fails.
* An exception thrown in this method will not affect the result of the query.
*/
default void cleanupQuery(ConnectorSession session) {}
/**
* Begin insert query.
*
* If connector does not support execution with retries, the method should throw:
*
* new TrinoException(NOT_SUPPORTED, "This connector does not support query retries")
*
* unless {@code retryMode} is set to {@code NO_RETRIES}.
*/
default ConnectorInsertTableHandle beginInsert(ConnectorSession session, ConnectorTableHandle tableHandle, List columns, RetryMode retryMode)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support inserts");
}
/**
* @return whether connector handles missing columns during insert
*/
default boolean supportsMissingColumnsOnInsert()
{
return false;
}
/**
* Finish insert query
*
* @deprecated use {@link #finishInsert(ConnectorSession, ConnectorInsertTableHandle, List, Collection, Collection)}
*/
@Deprecated
default Optional finishInsert(ConnectorSession session, ConnectorInsertTableHandle insertHandle, Collection fragments, Collection computedStatistics)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata beginInsert() is implemented without finishInsert()");
}
/**
* Finish insert query
*/
default Optional finishInsert(
ConnectorSession session,
ConnectorInsertTableHandle insertHandle,
List sourceTableHandles,
Collection fragments,
Collection computedStatistics)
{
// Delegate to deprecated SPI to not break existing connectors
return finishInsert(session, insertHandle, fragments, computedStatistics);
}
/**
* Returns true if materialized view refresh should be delegated to connector using {@link ConnectorMetadata#refreshMaterializedView}
*/
default boolean delegateMaterializedViewRefreshToConnector(ConnectorSession session, SchemaTableName viewName)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support materialized views");
}
/**
* Refresh materialized view
*/
default CompletableFuture> refreshMaterializedView(ConnectorSession session, SchemaTableName viewName)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support materialized views");
}
/**
* Begin materialized view query.
*
* If connector does not support execution with retries, the method should throw:
*
* new TrinoException(NOT_SUPPORTED, "This connector does not support query retries")
*
* unless {@code retryMode} is set to {@code NO_RETRIES}.
*/
default ConnectorInsertTableHandle beginRefreshMaterializedView(ConnectorSession session, ConnectorTableHandle tableHandle, List sourceTableHandles, RetryMode retryMode)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support materialized views");
}
/**
* Finish materialized view query
*/
default Optional finishRefreshMaterializedView(
ConnectorSession session,
ConnectorTableHandle tableHandle,
ConnectorInsertTableHandle insertHandle,
Collection fragments,
Collection computedStatistics,
List sourceTableHandles,
List sourceTableFunctions)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata beginRefreshMaterializedView() is implemented without finishRefreshMaterializedView()");
}
/**
* Return the row change paradigm supported by the connector on the table.
*/
default RowChangeParadigm getRowChangeParadigm(ConnectorSession session, ConnectorTableHandle tableHandle)
{
throw new TrinoException(NOT_SUPPORTED, MODIFYING_ROWS_MESSAGE);
}
/**
* Get the column handle that will generate row IDs for the merge operation.
* These IDs will be passed to the {@link ConnectorMergeSink#storeMergedRows}
* method of the {@link ConnectorMergeSink} that created them.
*/
default ColumnHandle getMergeRowIdColumnHandle(ConnectorSession session, ConnectorTableHandle tableHandle)
{
throw new TrinoException(NOT_SUPPORTED, MODIFYING_ROWS_MESSAGE);
}
/**
* Get the physical layout for updated or deleted rows of a MERGE operation.
* Inserted rows are handled by {@link #getInsertLayout}.
* This layout always uses the {@link #getMergeRowIdColumnHandle merge row ID column}.
*/
default Optional getUpdateLayout(ConnectorSession session, ConnectorTableHandle tableHandle)
{
return Optional.empty();
}
/**
* Do whatever is necessary to start an MERGE query, returning the {@link ConnectorMergeTableHandle}
* instance that will be passed to the PageSink, and to the {@link #finishMerge} method.
*/
default ConnectorMergeTableHandle beginMerge(ConnectorSession session, ConnectorTableHandle tableHandle, RetryMode retryMode)
{
throw new TrinoException(NOT_SUPPORTED, MODIFYING_ROWS_MESSAGE);
}
/**
* Finish a merge query
*
* @param session The session
* @param mergeTableHandle A ConnectorMergeTableHandle for the table that is the target of the merge
* @param fragments All fragments returned by the merge plan
* @param computedStatistics Statistics for the table, meaningful only to the connector that produced them.
*/
default void finishMerge(ConnectorSession session, ConnectorMergeTableHandle mergeTableHandle, Collection fragments, Collection computedStatistics)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata beginMerge() is implemented without finishMerge()");
}
/**
* Create the specified view. The view definition is intended to
* be serialized by the connector for permanent storage.
*/
default void createView(ConnectorSession session, SchemaTableName viewName, ConnectorViewDefinition definition, boolean replace)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support creating views");
}
/**
* Rename the specified view
*/
default void renameView(ConnectorSession session, SchemaTableName source, SchemaTableName target)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support renaming views");
}
/**
* Sets the user/role on the specified view.
*/
default void setViewAuthorization(ConnectorSession session, SchemaTableName viewName, TrinoPrincipal principal)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting an owner on a view");
}
/**
* Drop the specified view.
*/
default void dropView(ConnectorSession session, SchemaTableName viewName)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support dropping views");
}
/**
* List view names (but not materialized views), possibly filtered by schema. An empty list is returned if none match.
* An empty list is returned also when schema name does not refer to an existing schema.
*
* @see #listMaterializedViews(ConnectorSession, Optional)
*/
default List listViews(ConnectorSession session, Optional schemaName)
{
return emptyList();
}
/**
* Gets the definitions of views (but not materialized views), possibly filtered by schema.
* This optional method may be implemented by connectors that can support fetching
* view data in bulk. It is used to implement {@code information_schema.views}.
*/
default Map getViews(ConnectorSession session, Optional schemaName)
{
Map views = new HashMap<>();
for (SchemaTableName name : listViews(session, schemaName)) {
getView(session, name).ifPresent(view -> views.put(name, view));
}
return views;
}
/**
* Gets the view data for the specified view name. Returns {@link Optional#empty()} if {@code viewName}
* relation does not or is not a view (e.g. is a table, or a materialized view).
*
* @see #getTableHandle(ConnectorSession, SchemaTableName)
* @see #getMaterializedView(ConnectorSession, SchemaTableName)
*/
default Optional getView(ConnectorSession session, SchemaTableName viewName)
{
return Optional.empty();
}
/**
* Gets the schema properties for the specified schema.
*/
default Map getSchemaProperties(ConnectorSession session, String schemaName)
{
return Map.of();
}
/**
* Get the schema properties for the specified schema.
*/
default Optional getSchemaOwner(ConnectorSession session, String schemaName)
{
return Optional.empty();
}
/**
* Attempt to push down an update operation into the connector. If a connector
* can execute an update for the table handle on its own, it should return a
* table handle, which will be passed back to {@link #executeUpdate} during
* query executing to actually execute the update.
*/
default Optional applyUpdate(ConnectorSession session, ConnectorTableHandle handle, Map assignments)
{
return Optional.empty();
}
/**
* Execute the update operation on the handle returned from {@link #applyUpdate}.
*/
default OptionalLong executeUpdate(ConnectorSession session, ConnectorTableHandle handle)
{
throw new TrinoException(FUNCTION_IMPLEMENTATION_ERROR, "ConnectorMetadata applyUpdate() is implemented without executeUpdate()");
}
/**
* Attempt to push down a delete operation into the connector. If a connector
* can execute a delete for the table handle on its own, it should return a
* table handle, which will be passed back to {@link #executeDelete} during
* query executing to actually execute the delete.
*/
default Optional applyDelete(ConnectorSession session, ConnectorTableHandle handle)
{
return Optional.empty();
}
/**
* Execute the delete operation on the handle returned from {@link #applyDelete}.
*/
default OptionalLong executeDelete(ConnectorSession session, ConnectorTableHandle handle)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata applyDelete() is implemented without executeDelete()");
}
/**
* Try to locate a table index that can lookup results by indexableColumns and provide the requested outputColumns.
*/
default Optional resolveIndex(ConnectorSession session, ConnectorTableHandle tableHandle, Set indexableColumns, Set outputColumns, TupleDomain tupleDomain)
{
return Optional.empty();
}
/**
* List available functions.
*/
default Collection listFunctions(ConnectorSession session, String schemaName)
{
return List.of();
}
/**
* Get all functions with specified name.
*/
default Collection getFunctions(ConnectorSession session, SchemaFunctionName name)
{
return List.of();
}
/**
* Return the function with the specified id.
*/
default FunctionMetadata getFunctionMetadata(ConnectorSession session, FunctionId functionId)
{
throw new IllegalArgumentException("Unknown function " + functionId);
}
/**
* Returns the aggregation metadata for the aggregation function with the specified id.
*/
default AggregationFunctionMetadata getAggregationFunctionMetadata(ConnectorSession session, FunctionId functionId)
{
throw new IllegalArgumentException("Unknown function " + functionId);
}
/**
* Returns the dependencies of the function with the specified id.
*/
default FunctionDependencyDeclaration getFunctionDependencies(ConnectorSession session, FunctionId functionId, BoundSignature boundSignature)
{
throw new IllegalArgumentException("Unknown function " + functionId);
}
/**
* List available language functions.
*/
default Collection listLanguageFunctions(ConnectorSession session, String schemaName)
{
return List.of();
}
/**
* Get all language functions with the specified name.
*/
default Collection getLanguageFunctions(ConnectorSession session, SchemaFunctionName name)
{
return List.of();
}
/**
* Check if a language function exists.
*/
default boolean languageFunctionExists(ConnectorSession session, SchemaFunctionName name, String signatureToken)
{
return getLanguageFunctions(session, name).stream()
.anyMatch(function -> function.signatureToken().equals(signatureToken));
}
/**
* Creates a language function with the specified name and signature token.
* The signature token is an opaque string that uniquely identifies the function signature.
* Multiple functions with the same name but with different signatures may exist.
* The signature token is used to identify the function when dropping it.
*
* @param replace if true, replace existing function with the same name and signature token
*/
default void createLanguageFunction(ConnectorSession session, SchemaFunctionName name, LanguageFunction function, boolean replace)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support creating functions");
}
/**
* Drops a language function with the specified name and signature token.
*/
default void dropLanguageFunction(ConnectorSession session, SchemaFunctionName name, String signatureToken)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support dropping functions");
}
/**
* Does the specified role exist.
*/
default boolean roleExists(ConnectorSession session, String role)
{
return listRoles(session).contains(role);
}
/**
* Creates the specified role.
*
* @param grantor represents the principal specified by WITH ADMIN statement
*/
default void createRole(ConnectorSession session, String role, Optional grantor)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support create role");
}
/**
* Drops the specified role.
*/
default void dropRole(ConnectorSession session, String role)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support drop role");
}
/**
* List available roles.
*/
default Set listRoles(ConnectorSession session)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support roles");
}
/**
* List role grants for a given principal, not recursively.
*/
default Set listRoleGrants(ConnectorSession session, TrinoPrincipal principal)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support roles");
}
/**
* Grants the specified roles to the specified grantees
*
* @param grantor represents the principal specified by GRANTED BY statement
*/
default void grantRoles(ConnectorSession connectorSession, Set roles, Set grantees, boolean adminOption, Optional grantor)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support roles");
}
/**
* Revokes the specified roles from the specified grantees
*
* @param grantor represents the principal specified by GRANTED BY statement
*/
default void revokeRoles(ConnectorSession connectorSession, Set roles, Set grantees, boolean adminOption, Optional grantor)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support roles");
}
/**
* List applicable roles, including the transitive grants, for the specified principal
*/
default Set listApplicableRoles(ConnectorSession session, TrinoPrincipal principal)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support roles");
}
/**
* List applicable roles, including the transitive grants, in given session
*/
default Set listEnabledRoles(ConnectorSession session)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support roles");
}
/**
* Grants the specified privilege to the specified user on the specified schema
*/
default void grantSchemaPrivileges(ConnectorSession session, String schemaName, Set privileges, TrinoPrincipal grantee, boolean grantOption)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support grants on schemas");
}
/**
* Denys the specified privilege to the specified user on the specified schema
*/
default void denySchemaPrivileges(ConnectorSession session, String schemaName, Set privileges, TrinoPrincipal grantee)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support denys on schemas");
}
/**
* Revokes the specified privilege on the specified schema from the specified user
*/
default void revokeSchemaPrivileges(ConnectorSession session, String schemaName, Set privileges, TrinoPrincipal grantee, boolean grantOption)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support revokes on schemas");
}
/**
* Grants the specified privilege to the specified user on the specified table
*/
default void grantTablePrivileges(ConnectorSession session, SchemaTableName tableName, Set privileges, TrinoPrincipal grantee, boolean grantOption)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support grants on tables");
}
/**
* Denys the specified privilege to the specified user on the specified table
*/
default void denyTablePrivileges(ConnectorSession session, SchemaTableName tableName, Set privileges, TrinoPrincipal grantee)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support denys on tables");
}
/**
* Revokes the specified privilege on the specified table from the specified user
*/
default void revokeTablePrivileges(ConnectorSession session, SchemaTableName tableName, Set privileges, TrinoPrincipal grantee, boolean grantOption)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support revokes on tables");
}
/**
* List the table privileges granted to the specified grantee for the tables that have the specified prefix considering the selected session role
*/
default List listTablePrivileges(ConnectorSession session, SchemaTablePrefix prefix)
{
return emptyList();
}
default ConnectorTableProperties getTableProperties(ConnectorSession session, ConnectorTableHandle table)
{
return new ConnectorTableProperties();
}
/**
* Attempt to push down the provided limit into the table.
*
* Connectors can indicate whether they don't support limit pushdown or that the action had no effect
* by returning {@link Optional#empty()}. Connectors should expect this method to be called multiple times
* during the optimization of a given query.
*
* Note : it's critical for connectors to return Optional.empty() if calling this method has no effect for that
* invocation, even if the connector generally supports limit pushdown. Doing otherwise can cause the optimizer
* to loop indefinitely.
*
*
* If the connector could benefit from the information but can't guarantee that it will be able to produce
* fewer rows than the provided limit, it should return a non-empty result containing a new handle for the
* derived table and the "limit guaranteed" flag set to false.
*
* If the connector can guarantee it will produce fewer rows than the provided limit, it should return a
* non-empty result with the "limit guaranteed" flag set to true.
*/
default Optional> applyLimit(ConnectorSession session, ConnectorTableHandle handle, long limit)
{
return Optional.empty();
}
/**
* Attempt to push down the provided constraint into the table.
*
* Connectors can indicate whether they don't support predicate pushdown or that the action had no effect
* by returning {@link Optional#empty()}. Connectors should expect this method to be called multiple times
* during the optimization of a given query.
*
* Note : it's critical for connectors to return Optional.empty() if calling this method has no effect for that
* invocation, even if the connector generally supports pushdown. Doing otherwise can cause the optimizer
* to loop indefinitely.
*
*
* Note : Implementation must not maintain reference to {@code constraint}'s {@link Constraint#predicate()} after the
* call returns.
*
*
* @param constraint constraint to be applied to the table. {@link Constraint#getSummary()} is guaranteed not to be {@link TupleDomain#isNone() none}.
*/
default Optional> applyFilter(ConnectorSession session, ConnectorTableHandle handle, Constraint constraint)
{
// applyFilter is expected not to be invoked with a "false" constraint
if (constraint.getSummary().isNone()) {
throw new IllegalArgumentException("constraint summary is NONE");
}
if (FALSE.equals(constraint.getExpression())) {
// DomainTranslator translates FALSE expressions into TupleDomain.none() (via Visitor#visitBooleanLiteral)
// so the remaining expression shouldn't be FALSE and therefore the translated connectorExpression shouldn't be FALSE either.
throw new IllegalArgumentException("constraint expression is FALSE");
}
return Optional.empty();
}
/**
* Attempt to push down the provided projections into the table.
*
* Connectors can indicate whether they don't support projection pushdown or that the action had no effect
* by returning {@link Optional#empty()}. Connectors should expect this method to be called multiple times
* during the optimization of a given query.
*
* Note : it's critical for connectors to return Optional.empty() if calling this method has no effect for that
* invocation, even if the connector generally supports pushdown. Doing otherwise can cause the optimizer
* to loop indefinitely.
*
*
* If the method returns a result, the list of projections in the result *replaces* the existing ones, and the
* list of assignments is the new set of columns exposed by the derived table.
*
* As an example, given the following plan:
*
*
* - project
* x = f1(a, b)
* y = f2(a, b)
* z = f3(a, b)
* - scan (TH0)
* a = CH0
* b = CH1
* c = CH2
*
*
* The optimizer would call this method with the following arguments:
*
*
* handle = TH0
* projections = [
* f1(a, b)
* f2(a, b)
* f3(a, b)
* ]
* assignments = [
* a = CH0
* b = CH1
* c = CH2
* ]
*
*
* Assuming the connector knows how to handle f1(...) and f2(...), it would return:
*
*
* handle = TH1
* projections = [
* v2
* v3
* f3(v0, v1)
* ]
* assignments = [
* v0 = CH0
* v1 = CH1
* v2 = CH3 (synthetic column for f1(CH0, CH1))
* v3 = CH4 (synthetic column for f2(CH0, CH1))
* ]
*
*/
default Optional> applyProjection(ConnectorSession session, ConnectorTableHandle handle, List projections, Map assignments)
{
return Optional.empty();
}
/**
* Attempt to push down the sampling into the table.
*
* Connectors can indicate whether they don't support sample pushdown or that the action had no effect
* by returning {@link Optional#empty()}. Connectors should expect this method to be called multiple times
* during the optimization of a given query.
*
* Note : it's critical for connectors to return Optional.empty() if calling this method has no effect for that
* invocation, even if the connector generally supports sample pushdown. Doing otherwise can cause the optimizer
* to loop indefinitely.
*
*/
default Optional> applySample(ConnectorSession session, ConnectorTableHandle handle, SampleType sampleType, double sampleRatio)
{
return Optional.empty();
}
/**
* Attempt to push down the aggregates into the table.
*
* Connectors can indicate whether they don't support aggregate pushdown or that the action had no effect
* by returning {@link Optional#empty()}. Connectors should expect this method may be called multiple times.
*
* Note : it's critical for connectors to return {@link Optional#empty()} if calling this method has no effect for that
* invocation, even if the connector generally supports pushdown. Doing otherwise can cause the optimizer
* to loop indefinitely.
*
* If the method returns a result, the list of assignments in the result will be merged with existing assignments. The projections
* returned by the method must have the same order as the given input list of aggregates.
*
* As an example, given the following plan:
*
*
* - aggregation (GROUP BY c)
* variable0 = agg_fn1(a)
* variable1 = agg_fn2(b, 2)
* variable2 = c
* - scan (TH0)
* a = CH0
* b = CH1
* c = CH2
*
*
* The optimizer would call this method with the following arguments:
*
*
* handle = TH0
* aggregates = [
* { functionName=agg_fn1, outputType = «some Trino type» inputs = [{@link Variable} a]} ,
* { functionName=agg_fn2, outputType = «some Trino type» inputs = [{@link Variable} b, {@link Constant} 2]}
* ]
* groupingSets=[[{@link ColumnHandle} CH2]]
* assignments = {a = CH0, b = CH1, c = CH2}
*
*
*
* Assuming the connector knows how to handle {@code agg_fn1(...)} and {@code agg_fn2(...)}, it would return:
*
*
* {@link AggregationApplicationResult} {
* handle = TH1
* projections = [{@link Variable} synthetic_name0, {@link Variable} synthetic_name1] -- The order in the list must be same as input list of aggregates
* assignments = {
* synthetic_name0 = CH3 (synthetic column for agg_fn1(a))
* synthetic_name1 = CH4 (synthetic column for agg_fn2(b,2))
* }
* }
*
*
* if the connector only knows how to handle {@code agg_fn1(...)}, but not {@code agg_fn2}, it should return {@link Optional#empty()}.
*
*
* Another example is where the connector wants to handle the aggregate function by pointing to a pre-materialized table.
* In this case the input can stay same as in the above example and the connector can return
*
* {@link AggregationApplicationResult} {
* handle = TH1 (could capture information about which pre-materialized table to use)
* projections = [{@link Variable} synthetic_name0, {@link Variable} synthetic_name1] -- The order in the list must be same as input list of aggregates
* assignments = {
* synthetic_name0 = CH3 (reference to the column in pre-materialized table that has agg_fn1(a) calculated)
* synthetic_name1 = CH4 (reference to the column in pre-materialized table that has agg_fn2(b,2) calculated)
* synthetic_name2 = CH5 (reference to the column in pre-materialized table that has the group by column c)
* }
* groupingColumnMapping = {
* CH2 -> CH5 (CH2 in the original assignment should now be replaced by CH5 in the new assignment)
* }
* }
*
*
*/
default Optional> applyAggregation(
ConnectorSession session,
ConnectorTableHandle handle,
List aggregates,
Map assignments,
List> groupingSets)
{
// Global aggregation is represented by [[]]
if (groupingSets.isEmpty()) {
throw new IllegalArgumentException("No grouping sets provided");
}
return Optional.empty();
}
/**
* Attempt to push down the join operation.
*
* Connectors can indicate whether they don't support join pushdown or that the action had no effect
* by returning {@link Optional#empty()}. Connectors should expect this method may be called multiple times.
*
* Warning : this is an experimental API and it will change in the future.
*
* Join condition conjuncts are passed via joinConditions list. For current implementation connector may
* assume that leftExpression and rightExpression in each of the conjucts are instances of {@link Variable}.
* This may be relaxed in the future.
*
*
* The leftAssignments and rightAssignments lists provide mappings from variable names, used in joinConditions to input tables column handles.
* It is guaranteed that all the required mappings will be provided but not necessarily *all* the column handles of tables which are join inputs.
*
*
* Table statistics for left, right table as well as estimated statistics for join are provided via statistics parameter.
* Those can be used by connector to assess if performing join pushdown is expected to improve query performance.
*
*
*
* If the method returns a result the returned table handle will be used in place of join and input table scans.
* Returned result must provide mapping from old column handles to new ones via leftColumnHandles and rightColumnHandles fields of the result.
* It is required that mapping is provided for *all* column handles exposed previously by both left and right join sources.
*
*/
default Optional> applyJoin(
ConnectorSession session,
JoinType joinType,
ConnectorTableHandle left,
ConnectorTableHandle right,
ConnectorExpression joinCondition,
Map leftAssignments,
Map rightAssignments,
JoinStatistics statistics)
{
List conditions;
if (joinCondition instanceof Call call && AND_FUNCTION_NAME.equals(call.getFunctionName())) {
conditions = new ArrayList<>(call.getArguments().size());
for (ConnectorExpression argument : call.getArguments()) {
if (Constant.TRUE.equals(argument)) {
continue;
}
Optional condition = JoinCondition.from(argument, leftAssignments.keySet(), rightAssignments.keySet());
if (condition.isEmpty()) {
// We would need to add a FilterNode on top of the result
return Optional.empty();
}
conditions.add(condition.get());
}
}
else {
Optional condition = JoinCondition.from(joinCondition, leftAssignments.keySet(), rightAssignments.keySet());
if (condition.isEmpty()) {
return Optional.empty();
}
conditions = List.of(condition.get());
}
return applyJoin(
session,
joinType,
left,
right,
conditions,
leftAssignments,
rightAssignments,
statistics);
}
@Deprecated
default Optional> applyJoin(
ConnectorSession session,
JoinType joinType,
ConnectorTableHandle left,
ConnectorTableHandle right,
List joinConditions,
Map leftAssignments,
Map rightAssignments,
JoinStatistics statistics)
{
return Optional.empty();
}
/**
* Attempt to push down the TopN into the table scan.
*
* Connectors can indicate whether they don't support topN pushdown or that the action had no effect
* by returning {@link Optional#empty()}. Connectors should expect this method may be called multiple times.
*
* Note : it's critical for connectors to return {@link Optional#empty()} if calling this method has no effect for that
* invocation, even if the connector generally supports topN pushdown. Doing otherwise can cause the optimizer
* to loop indefinitely.
*
* If the connector can handle TopN Pushdown and guarantee it will produce no more rows than requested then it should return a
* non-empty result with "topN guaranteed" flag set to true.
*/
default Optional> applyTopN(
ConnectorSession session,
ConnectorTableHandle handle,
long topNCount,
List sortItems,
Map assignments)
{
return Optional.empty();
}
/**
* Attempt to push down the table function invocation into the connector.
*
* Connectors can indicate whether they don't support table function invocation pushdown or that the action had no
* effect by returning {@link Optional#empty()}. Connectors should expect this method may be called multiple times.
*
* If the method returns a result, the returned table handle will be used in place of the table function invocation.
*/
default Optional> applyTableFunction(ConnectorSession session, ConnectorTableFunctionHandle handle)
{
return Optional.empty();
}
/**
* Allows the connector to reject the table scan produced by the planner.
*
* Connectors can choose to reject a query based on the table scan potentially being too expensive, for example
* if no filtering is done on a partition column.
*
*/
default void validateScan(ConnectorSession session, ConnectorTableHandle handle) {}
/**
* Create the specified materialized view. The view definition is intended to
* be serialized by the connector for permanent storage.
*
* @throws TrinoException with {@code ALREADY_EXISTS} if the object already exists and {@code ignoreExisting} is not set
*/
default void createMaterializedView(
ConnectorSession session,
SchemaTableName viewName,
ConnectorMaterializedViewDefinition definition,
Map properties,
boolean replace,
boolean ignoreExisting)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support creating materialized views");
}
/**
* Drop the specified materialized view.
*/
default void dropMaterializedView(ConnectorSession session, SchemaTableName viewName)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support dropping materialized views");
}
/**
* List materialized view names, possibly filtered by schema. An empty list is returned if none match.
* An empty list is returned also when schema name does not refer to an existing schema.
*/
default List listMaterializedViews(ConnectorSession session, Optional schemaName)
{
return List.of();
}
/**
* Gets the definitions of materialized views, possibly filtered by schema.
* This optional method may be implemented by connectors that can support fetching
* view data in bulk. It is used to populate {@code information_schema.columns}.
*/
default Map getMaterializedViews(ConnectorSession session, Optional schemaName)
{
Map materializedViews = new HashMap<>();
for (SchemaTableName name : listMaterializedViews(session, schemaName)) {
getMaterializedView(session, name).ifPresent(view -> materializedViews.put(name, view));
}
return materializedViews;
}
/**
* Gets the materialized view data for the specified materialized view name. Returns {@link Optional#empty()}
* if {@code viewName} relation does not or is not a materialized view (e.g. is a table, or a view).
*
* @see #getTableHandle(ConnectorSession, SchemaTableName)
* @see #getView(ConnectorSession, SchemaTableName)
*/
default Optional getMaterializedView(ConnectorSession session, SchemaTableName viewName)
{
return Optional.empty();
}
default Map getMaterializedViewProperties(ConnectorSession session, SchemaTableName viewName, ConnectorMaterializedViewDefinition materializedViewDefinition)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support materialized views");
}
/**
* The method is used by the engine to determine if a materialized view is current with respect to the tables it depends on.
*
* @throws MaterializedViewNotFoundException when materialized view is not found
*/
default MaterializedViewFreshness getMaterializedViewFreshness(ConnectorSession session, SchemaTableName name)
{
throw new TrinoException(GENERIC_INTERNAL_ERROR, "ConnectorMetadata getMaterializedView() is implemented without getMaterializedViewFreshness()");
}
/**
* Rename the specified materialized view
*/
default void renameMaterializedView(ConnectorSession session, SchemaTableName source, SchemaTableName target)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support renaming materialized views");
}
/**
* Sets the properties of the specified materialized view
*/
default void setMaterializedViewProperties(ConnectorSession session, SchemaTableName viewName, Map> properties)
{
throw new TrinoException(NOT_SUPPORTED, "This connector does not support setting materialized view properties");
}
default Optional applyTableScanRedirect(ConnectorSession session, ConnectorTableHandle tableHandle)
{
return Optional.empty();
}
/**
* Redirects table to other table which may or may not be in the same catalog.
* Currently the engine tries to do redirection only for table reads and metadata listing.
*
* Also consider implementing streamTableColumns to support redirection for listing.
*/
default Optional redirectTable(ConnectorSession session, SchemaTableName tableName)
{
return Optional.empty();
}
default OptionalInt getMaxWriterTasks(ConnectorSession session)
{
return OptionalInt.empty();
}
default WriterScalingOptions getNewTableWriterScalingOptions(ConnectorSession session, SchemaTableName tableName, Map tableProperties)
{
return WriterScalingOptions.DISABLED;
}
default WriterScalingOptions getInsertWriterScalingOptions(ConnectorSession session, ConnectorTableHandle tableHandle)
{
return WriterScalingOptions.DISABLED;
}
final class Helper
{
private Helper() {}
static final Logger juliLogger = Logger.getLogger(ConnectorMetadata.class.getName());
}
}