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

io.trino.metadata.PropertyUtil Maven / Gradle / Ivy

There is a newer version: 465
Show 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.metadata;

import com.google.common.collect.ImmutableMap;
import io.trino.Session;
import io.trino.security.AccessControl;
import io.trino.spi.ErrorCodeSupplier;
import io.trino.spi.TrinoException;
import io.trino.spi.block.Block;
import io.trino.spi.session.PropertyMetadata;
import io.trino.spi.type.Type;
import io.trino.sql.PlannerContext;
import io.trino.sql.planner.ParameterRewriter;
import io.trino.sql.tree.Expression;
import io.trino.sql.tree.ExpressionTreeRewriter;
import io.trino.sql.tree.NodeRef;
import io.trino.sql.tree.Parameter;
import io.trino.sql.tree.Property;

import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Optional;

import static io.trino.spi.type.TypeUtils.writeNativeValue;
import static io.trino.sql.planner.ExpressionInterpreter.evaluateConstantExpression;
import static java.lang.String.format;
import static java.util.Locale.ENGLISH;

public final class PropertyUtil
{
    private PropertyUtil() {}

    public static Map> evaluateProperties(
            Iterable setProperties,
            Session session,
            PlannerContext plannerContext,
            AccessControl accessControl,
            Map, Expression> parameters,
            boolean includeAllProperties,
            Map> metadata,
            ErrorCodeSupplier errorCode,
            String propertyTypeDescription)
    {
        Map> propertyValues = new LinkedHashMap<>();

        // Fill in user-specified properties
        for (Property property : setProperties) {
            // property names are case-insensitive and normalized to lower case
            String propertyName = property.getName().getValue().toLowerCase(ENGLISH);
            PropertyMetadata propertyMetadata = metadata.get(propertyName);
            if (propertyMetadata == null) {
                throw new TrinoException(errorCode, format("%s '%s' does not exist", capitalize(propertyTypeDescription), propertyName));
            }

            Optional value;
            if (property.isSetToDefault()) {
                value = Optional.ofNullable(propertyMetadata.getDefaultValue());
            }
            else {
                value = Optional.of(evaluateProperty(
                        property.getNonDefaultValue(),
                        propertyMetadata,
                        session,
                        plannerContext,
                        accessControl,
                        parameters,
                        errorCode,
                        propertyTypeDescription));
            }

            propertyValues.put(propertyMetadata.getName(), value);
        }

        if (includeAllProperties) {
            for (PropertyMetadata propertyMetadata : metadata.values()) {
                if (!propertyValues.containsKey(propertyMetadata.getName())) {
                    propertyValues.put(propertyMetadata.getName(), Optional.ofNullable(propertyMetadata.getDefaultValue()));
                }
            }
        }
        return ImmutableMap.copyOf(propertyValues);
    }

    private static Object evaluateProperty(
            Expression expression,
            PropertyMetadata property,
            Session session,
            PlannerContext plannerContext,
            AccessControl accessControl,
            Map, Expression> parameters,
            ErrorCodeSupplier errorCode,
            String propertyTypeDescription)
    {
        Object sqlObjectValue = evaluateProperty(
                property.getName(),
                property.getSqlType(),
                expression,
                session,
                plannerContext,
                accessControl,
                parameters,
                errorCode,
                propertyTypeDescription);

        try {
            return property.decode(sqlObjectValue);
        }
        catch (Exception e) {
            throw new TrinoException(
                    errorCode,
                    format(
                            "Unable to set %s '%s' to [%s]: %s",
                            propertyTypeDescription,
                            property.getName(),
                            expression,
                            e.getMessage()),
                    e);
        }
    }

    public static Object evaluateProperty(
            String propertyName,
            Type propertyType,
            Expression expression,
            Session session,
            PlannerContext plannerContext,
            AccessControl accessControl,
            Map, Expression> parameters,
            ErrorCodeSupplier errorCode,
            String propertyTypeDescription)
    {
        Object sqlObjectValue;
        try {
            Expression rewritten = ExpressionTreeRewriter.rewriteWith(new ParameterRewriter(parameters), expression);
            Object value = evaluateConstantExpression(rewritten, propertyType, plannerContext, session, accessControl, parameters);

            // convert to object value type of SQL type
            Block block = writeNativeValue(propertyType, value);
            sqlObjectValue = propertyType.getObjectValue(session.toConnectorSession(), block, 0);
        }
        catch (TrinoException e) {
            throw new TrinoException(
                    errorCode,
                    format(
                            "Invalid value for %s '%s': Cannot convert [%s] to %s",
                            propertyTypeDescription,
                            propertyName,
                            expression,
                            propertyType),
                    e);
        }

        if (sqlObjectValue == null) {
            throw new TrinoException(
                    errorCode,
                    format(
                            "Invalid null value for %s '%s' from [%s]",
                            propertyTypeDescription,
                            propertyName,
                            expression));
        }
        return sqlObjectValue;
    }

    private static String capitalize(String value)
    {
        return Character.toUpperCase(value.charAt(0)) + value.substring(1);
    }
}