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

com.hazelcast.jet.sql.impl.schema.TableResolverImpl Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2024 Hazelcast Inc.
 *
 * Licensed under the Hazelcast Community License (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://hazelcast.com/hazelcast-community-license
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.hazelcast.jet.sql.impl.schema;

import com.hazelcast.core.EntryEvent;
import com.hazelcast.core.LifecycleEvent;
import com.hazelcast.dataconnection.impl.InternalDataConnectionService;
import com.hazelcast.jet.function.PentaFunction;
import com.hazelcast.jet.sql.impl.connector.SqlConnector;
import com.hazelcast.jet.sql.impl.connector.SqlConnector.SqlExternalResource;
import com.hazelcast.jet.sql.impl.connector.SqlConnectorCache;
import com.hazelcast.jet.sql.impl.connector.infoschema.MappingColumnsTable;
import com.hazelcast.jet.sql.impl.connector.infoschema.MappingsTable;
import com.hazelcast.jet.sql.impl.connector.infoschema.TablesTable;
import com.hazelcast.jet.sql.impl.connector.infoschema.UDTAttributesTable;
import com.hazelcast.jet.sql.impl.connector.infoschema.UserDefinedTypesTable;
import com.hazelcast.jet.sql.impl.connector.infoschema.ViewsTable;
import com.hazelcast.jet.sql.impl.connector.virtual.ViewTable;
import com.hazelcast.spi.impl.NodeEngine;
import com.hazelcast.sql.impl.QueryException;
import com.hazelcast.sql.impl.schema.BadTable;
import com.hazelcast.sql.impl.schema.ConstantTableStatistics;
import com.hazelcast.sql.impl.schema.Mapping;
import com.hazelcast.sql.impl.schema.MappingField;
import com.hazelcast.sql.impl.schema.Table;
import com.hazelcast.sql.impl.schema.TableResolver;
import com.hazelcast.sql.impl.schema.dataconnection.DataConnectionCatalogEntry;
import com.hazelcast.sql.impl.schema.type.Type;
import com.hazelcast.sql.impl.schema.view.View;
import com.hazelcast.sql.impl.security.SqlSecurityContext;

import javax.annotation.Nonnull;
import java.security.Permission;
import java.util.ArrayList;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CopyOnWriteArrayList;

import static com.hazelcast.internal.util.Preconditions.checkNotNull;
import static com.hazelcast.sql.impl.QueryUtils.CATALOG;
import static com.hazelcast.sql.impl.QueryUtils.SCHEMA_NAME_INFORMATION_SCHEMA;
import static com.hazelcast.sql.impl.QueryUtils.SCHEMA_NAME_PUBLIC;
import static java.util.Arrays.asList;
import static java.util.Collections.singletonList;

/**
 * A table resolver for DDL-created mappings and for the {@code
 * information_schema}.
 */
public class TableResolverImpl implements TableResolver {
    private static final List> SEARCH_PATHS = singletonList(
            asList(CATALOG, SCHEMA_NAME_PUBLIC)
    );

    private static final List, List, List, SqlConnectorCache, NodeEngine, Table>>
            ADDITIONAL_TABLE_PRODUCERS = asList(
            (m, v, t, cc, hz) -> new TablesTable(CATALOG, SCHEMA_NAME_INFORMATION_SCHEMA, SCHEMA_NAME_PUBLIC, m, v),
            (m, v, t, cc, hz) -> new MappingsTable(CATALOG, SCHEMA_NAME_INFORMATION_SCHEMA, SCHEMA_NAME_PUBLIC, m,
                    cc,
                    hz.getDataConnectionService()::typeForDataConnection,
                    hz.getConfig().getSecurityConfig().isEnabled()),
            (m, v, t, cc, hz) -> new MappingColumnsTable(CATALOG, SCHEMA_NAME_INFORMATION_SCHEMA, SCHEMA_NAME_PUBLIC, m, v),
            (m, v, t, cc, hz) -> new ViewsTable(CATALOG, SCHEMA_NAME_INFORMATION_SCHEMA, SCHEMA_NAME_PUBLIC, v),
            (m, v, t, cc, hz) -> new UserDefinedTypesTable(CATALOG, SCHEMA_NAME_INFORMATION_SCHEMA, SCHEMA_NAME_PUBLIC, t),
            (m, v, t, cc, hz) -> new UDTAttributesTable(CATALOG, SCHEMA_NAME_INFORMATION_SCHEMA, SCHEMA_NAME_PUBLIC, t)
    );

    private final NodeEngine nodeEngine;
    private final RelationsStorage relationsStorage;
    private final SqlConnectorCache connectorCache;
    private final List listeners;

    // These fields should normally be volatile because we're accessing them from multiple threads. But we
    // don't care if some thread doesn't see a newer value written by another thread. Each thread will write
    // the same value (we assume that the number of mappings and views doesn't change much), so we
    // shave a tiny bit of performance from not synchronizing :)
    private int lastViewsSize;
    private int lastMappingsSize;
    private int lastTypesSize;

    public TableResolverImpl(
            NodeEngine nodeEngine,
            RelationsStorage relationsStorage,
            SqlConnectorCache connectorCache
    ) {
        this.nodeEngine = nodeEngine;
        this.relationsStorage = relationsStorage;
        this.connectorCache = connectorCache;
        this.listeners = new CopyOnWriteArrayList<>();

        // because listeners are invoked asynchronously from the calling thread,
        // local changes are handled in createMapping() & removeMapping(), thus
        // we skip events originating from local member to avoid double processing
        nodeEngine.getHazelcastInstance().getLifecycleService().addLifecycleListener(event -> {
            if (event.getState() == LifecycleEvent.LifecycleState.STARTED) {
                this.relationsStorage.initializeWithListener(new AbstractSchemaStorage.EntryListenerAdapter() {
                    @Override
                    public void entryUpdated(EntryEvent event) {
                        if (!event.getMember().localMember()) {
                            listeners.forEach(TableListener::onTableChanged);
                        }
                    }

                    @Override
                    public void entryRemoved(EntryEvent event) {
                        if (!event.getMember().localMember()) {
                            listeners.forEach(TableListener::onTableChanged);
                        }
                    }
                });
            }
        });
    }

    // region mapping

    public void createMapping(Mapping mapping, boolean replace, boolean ifNotExists, SqlSecurityContext securityContext) {
        Mapping resolved = resolveMapping(mapping, securityContext);

        String name = resolved.name();
        if (ifNotExists) {
            relationsStorage.putIfAbsent(name, resolved);
        } else if (replace) {
            relationsStorage.put(name, resolved);
            listeners.forEach(TableListener::onTableChanged);
        } else if (!relationsStorage.putIfAbsent(name, resolved)) {
            throw QueryException.error("Mapping or view already exists: " + name);
        }
    }

    private Mapping resolveMapping(Mapping mapping, SqlSecurityContext securityContext) {
        Map options = mapping.options();
        String type = mapping.connectorType();
        String dataConnection = mapping.dataConnection();
        List resolvedFields;
        SqlConnector connector;

        if (type == null) {
            connector = extractConnector(dataConnection);
        } else {
            connector = connectorCache.forType(type);
        }
        String objectType = mapping.objectType() == null
                ? connector.defaultObjectType()
                : mapping.objectType();
        checkNotNull(objectType, "objectType cannot be null");

        SqlExternalResource externalResource = new SqlExternalResource(
                mapping.externalName(),
                mapping.dataConnection(),
                connector.typeName(),
                objectType,
                options
        );

        List permissions = connector.permissionsForResolve(externalResource, nodeEngine);
        for (Permission permission : permissions) {
            securityContext.checkPermission(permission);
        }

        resolvedFields = connector.resolveAndValidateFields(
                nodeEngine,
                externalResource,
                mapping.fields()
        );

        return new Mapping(
                mapping.name(),
                mapping.externalName(),
                mapping.dataConnection(), type,
                objectType,
                new ArrayList<>(resolvedFields),
                new LinkedHashMap<>(options)
        );
    }

    private SqlConnector extractConnector(@Nonnull String dataConnection) {
        InternalDataConnectionService dataConnectionService = nodeEngine.getDataConnectionService();
        // TODO atm data connection and connector types match, but that's
        // not going to be universally true in the future
        String type = dataConnectionService.typeForDataConnection(dataConnection);
        return connectorCache.forType(type);
    }

    public void removeMapping(String name, boolean ifExists) {
        if (relationsStorage.removeMapping(name) != null) {
            listeners.forEach(TableListener::onTableChanged);
        } else if (!ifExists) {
            throw QueryException.error("Mapping does not exist: " + name);
        }
    }

    @Nonnull
    public Collection getMappingNames() {
        return relationsStorage.mappingNames();
    }
    // endregion

    // region view

    public void createView(View view, boolean replace, boolean ifNotExists) {
        if (ifNotExists) {
            relationsStorage.putIfAbsent(view.name(), view);
        } else if (replace) {
            relationsStorage.put(view.name(), view);
        } else if (!relationsStorage.putIfAbsent(view.name(), view)) {
            throw QueryException.error("Mapping or view already exists: " + view.name());
        }
    }

    public void removeView(String name, boolean ifExists) {
        if (relationsStorage.removeView(name) == null && !ifExists) {
            throw QueryException.error("View does not exist: " + name);
        }
    }

    @Nonnull
    public Collection getViewNames() {
        return relationsStorage.viewNames();
    }

    // endregion

    // region type

    public Collection getTypeNames() {
        return relationsStorage.typeNames();
    }

    public Collection getTypes() {
        return relationsStorage.getAllTypes();
    }

    public void createType(Type type, boolean replace, boolean ifNotExists) {
        if (ifNotExists) {
            relationsStorage.putIfAbsent(type.name(), type);
        } else if (replace) {
            relationsStorage.put(type.name(), type);
        } else if (!relationsStorage.putIfAbsent(type.name(), type)) {
            throw QueryException.error("Type already exists: " + type.name());
        }
    }

    public void removeType(String name, boolean ifExists) {
        if (relationsStorage.removeType(name) == null && !ifExists) {
            throw QueryException.error("Type does not exist: " + name);
        }
    }

    // endregion

    @Nonnull
    @Override
    public List> getDefaultSearchPaths() {
        return SEARCH_PATHS;
    }

    @Nonnull
    @Override
    public List getTables() {
        Collection objects = relationsStorage.allObjects();
        List
tables = new ArrayList<>(objects.size() + ADDITIONAL_TABLE_PRODUCERS.size()); int lastMappingsSize = this.lastMappingsSize; int lastViewsSize = this.lastViewsSize; int lastTypesSize = this.lastTypesSize; // Trying to avoid list growing. List mappings = lastMappingsSize == 0 ? new ArrayList<>() : new ArrayList<>(lastMappingsSize); List views = lastViewsSize == 0 ? new ArrayList<>() : new ArrayList<>(lastViewsSize); List types = lastTypesSize == 0 ? new ArrayList<>() : new ArrayList<>(lastTypesSize); for (Object o : objects) { if (o instanceof Mapping) { tables.add(toTable((Mapping) o)); mappings.add((Mapping) o); } else if (o instanceof View) { tables.add(toTable((View) o)); views.add((View) o); } else if (o instanceof Type) { types.add((Type) o); } else if (o instanceof DataConnectionCatalogEntry) { // Note: data connection is not a 'table' or 'relation', // It's stored in a separate namespace. continue; } else { throw new RuntimeException("Unexpected: " + o); } } ADDITIONAL_TABLE_PRODUCERS.forEach(producer -> tables.add(producer.apply(mappings, views, types, connectorCache, nodeEngine))); this.lastViewsSize = views.size(); this.lastMappingsSize = mappings.size(); this.lastTypesSize = types.size(); return tables; } private Table toTable(Mapping mapping) { try { SqlConnector connector; if (mapping.connectorType() == null) { connector = extractConnector(mapping.dataConnection()); } else { connector = connectorCache.forType((mapping.connectorType())); } assert connector != null; return connector.createTable( nodeEngine, SCHEMA_NAME_PUBLIC, mapping.name(), sqlExternalResourceFrom(mapping, connector), mapping.fields()); } catch (Throwable e) { // will fail later if invalid table is actually used in a query return new BadTable(SCHEMA_NAME_PUBLIC, mapping.name(), mapping.objectType(), e); } } private static SqlExternalResource sqlExternalResourceFrom(Mapping internalMapping, SqlConnector connector) { String internalObjType = internalMapping.objectType() == null ? connector.defaultObjectType() : internalMapping.objectType(); checkNotNull(internalObjType, "objectType cannot be null"); String connectorType = connector.typeName(); return new SqlExternalResource(internalMapping.externalName(), internalMapping.dataConnection(), connectorType, internalObjType, internalMapping.options()); } private Table toTable(View view) { return new ViewTable(SCHEMA_NAME_PUBLIC, view.name(), view.query(), new ConstantTableStatistics(0L)); } @Override public void registerListener(TableListener listener) { listeners.add(listener); } }