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

com.google.zetasql.Catalog Maven / Gradle / Ivy

/*
 * Copyright 2019 ZetaSQL Authors
 *
 * 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 com.google.zetasql;

import com.google.common.base.Joiner;
import com.google.common.base.Preconditions;
import java.io.Serializable;
import java.util.List;

/**
 * ZetaSQL uses the Catalog interface to look up names that are visible
 * at global scope inside a query.  Names include
 *   - tables (which could also be views or other table-like objects)
 *   - types (e.g. the named types defined in some database's schema)
 *   - functions (e.g. UDFs defined in a database's schema)
 *   - nested catalogs (which allow names like catalog1.catalog2.Table)
 */
public abstract class Catalog implements Serializable {

  /**
   * Get a fully-qualified description of this Catalog.
   * Suitable for log messages, but not necessarily a valid SQL path expression.
   */
  public abstract String getFullName();

  /**
   * Options for a LookupName call.
   */
  public static class FindOptions {
    FindOptions() {}

    // Possibly deadlines, Tasks for cancellation, etc.
  }

  /**
   * Looks up an object of Constant from this Catalog on {@code path}.
   *
   * 

Searches for the Constant by traversing nested Catalogs until it reaches the last level of * the path, and then calls getConstant. * * @throws NotFoundException if the Constant can not be found */ public final Constant findConstant(List path) throws NotFoundException { return findConstant(path, new FindOptions()); } public final Constant findConstant(List path, FindOptions options) throws NotFoundException { Preconditions.checkNotNull(path, "Invalid null Constant name path"); Preconditions.checkArgument(!path.isEmpty(), "Invalid empty Constant name path"); final String name = path.get(0); if (path.size() > 1) { Catalog catalog = getCatalog(name, options); if (catalog == null) { throw new NotFoundException( String.format( "Constant not found: Catalog %s not found in Catalog %s", ZetaSQLStrings.toIdentifierLiteral(name), getFullName())); } final List pathSuffix = path.subList(1, path.size()); return catalog.findConstant(pathSuffix, options); } else { Constant constant = getConstant(name, options); if (constant == null) { throw new NotFoundException( String.format( "Constant not found: %s not found in Catalog %s", ZetaSQLStrings.toIdentifierLiteral(name), getFullName())); } return constant; } } /** * Look up an object of Table from this Catalog on {@code path}. * *

If a Catalog implementation supports looking up Table by path, it * should implement the findTable method. Alternatively, a Catalog can * also contain nested catalogs, and implement FindTable method on * the inner-most Catalog. * *

The default findTable implementation traverses nested Catalogs until it reaches * a Catalog that overrides findTable, or until it gets to the last level of the * path and then calls getTable. * *

NOTE: The findTable methods take precedence over getTable methods and will always * be called first. So getTable method does not need to be implemented if findTable * method is implemented. If both getTable and findTable are implemented(though not * recommended), it is the implementation's responsibility to keep them * consistent. * * @throws NotFoundException if the Type can not be found */ public Table findTable(List path) throws NotFoundException { return findTable(path, new FindOptions()); } public Table findTable(List path, FindOptions options) throws NotFoundException { Preconditions.checkNotNull(path, "Invalid null Table name path"); Preconditions.checkArgument(!path.isEmpty(), "Invalid empty Table name path"); final String name = path.get(0); if (path.size() > 1) { Catalog catalog = getCatalog(name, options); if (catalog == null) { throw new NotFoundException(String.format( "Table not found: Catalog %s not found in Catalog %s", ZetaSQLStrings.toIdentifierLiteral(name), getFullName())); } final List pathSuffix = path.subList(1, path.size()); return catalog.findTable(pathSuffix, options); } else { Table table = getTable(name, options); if (table == null) { throw new NotFoundException(String.format( "Table not found: %s not found in Catalog %s", ZetaSQLStrings.toIdentifierLiteral(name), getFullName())); } return table; } } /** * Look up an object of Type from this Catalog on {@code path}. * *

If a Catalog implementation supports looking up Type by path, it * should implement the findType method. Alternatively, a Catalog can * also contain nested catalogs, and implement FindType method on * the inner-most Catalog. * *

The default findType implementation traverses nested Catalogs until it reaches * a Catalog that overrides findType, or until it gets to the last level of the * path and then calls getType. * *

NOTE: The findType methods take precedence over getType methods and will always * be called first. So getType method does not need to be implemented if findType * method is implemented. If both getType and findType are implemented(though not * recommended), it is the implementation's responsibility to keep them * consistent. * * @throws NotFoundException if the Type can not be found */ public Type findType(List path) throws NotFoundException { return findType(path, new FindOptions()); } public Type findType(List path, FindOptions options) throws NotFoundException { Type type = null; Preconditions.checkNotNull(path, "Invalid null Type name path"); Preconditions.checkArgument(!path.isEmpty(), "Invalid empty Table name path"); final String name = path.get(0); if (path.size() > 1) { Catalog catalog = getCatalog(name, options); if (catalog == null) { // FindType-specific behavior: If we have a path, and it looks like it // could be a valid unquoted multi-part proto name, try looking up that // proto name as a type. final String protoName = convertPathToProtoName(path); if (!protoName.isEmpty()) { type = getType(protoName, options); if (type != null) { return type; } else { throw new NotFoundException(String.format( "Type not found: %s is not a type and %s is not a nested Catalog in Catalog %s", ZetaSQLStrings.toIdentifierLiteral(protoName), ZetaSQLStrings.toIdentifierLiteral(name), getFullName())); } } throw new NotFoundException(String.format( "Type not found: Catalog %s not found in Catalog %s", ZetaSQLStrings.toIdentifierLiteral(name), getFullName())); } final List pathSuffix = path.subList(1, path.size()); return catalog.findType(pathSuffix, options); } else { type = getType(name, options); if (type == null) { throw new NotFoundException(String.format( "Type not found: %s not found in Catalog %s", ZetaSQLStrings.toIdentifierLiteral(name), getFullName())); } } return type; } /** * Look up an object of Procedure from this Catalog on {@code path}. * *

If a Catalog implementation supports looking up Procedure by path, it * should implement the findProcedure method. Alternatively, a Catalog can * also contain nested catalogs, and implement FindProcedure method on * the inner-most Catalog. * *

The default findProcedure implementation traverses nested Catalogs until it reaches * a Catalog that overrides findProcedure, or until it gets to the last level of the * path and then calls getProcedure. * *

NOTE: The findProcedure methods take precedence over getProcedure methods and will always * be called first. So getProcedure method does not need to be implemented if findProcedure * method is implemented. If both getProcedure and findProcedure are implemented(though not * recommended), it is the implementation's responsibility to keep them * consistent. * * @throws NotFoundException if the Procedure can not be found */ public Procedure findProcedure(List path) throws NotFoundException { return findProcedure(path, new FindOptions()); } public Procedure findProcedure(List path, FindOptions options) throws NotFoundException { Preconditions.checkNotNull(path, "Invalid null Procedure name path"); Preconditions.checkArgument(!path.isEmpty(), "Invalid empty Procedure name path"); final String name = path.get(0); if (path.size() > 1) { Catalog catalog = getCatalog(name, options); if (catalog == null) { throw new NotFoundException(String.format( "Procedure not found: Catalog %s not found in Catalog %s", ZetaSQLStrings.toIdentifierLiteral(name), getFullName())); } final List pathSuffix = path.subList(1, path.size()); return catalog.findProcedure(pathSuffix, options); } else { Procedure procedure = getProcedure(name, options); if (procedure == null) { throw new NotFoundException(String.format( "Procedure not found: %s not found in Catalog %s", ZetaSQLStrings.toIdentifierLiteral(name), getFullName())); } return procedure; } } /** * Given an identifier path, return the type name that results when combining * that path into a single protocol buffer type name, if applicable. * Returns empty string if {@code path} cannot form a valid proto-style type name. * * @return type name if applicable, empty string if not */ public static String convertPathToProtoName(List path) { if (path == null) { return ""; } for (String identifier : path) { if (!isIdentifier(identifier)) { return ""; } } // This also returns "" for empty paths. return Joiner.on(".").join(path); } /** * Get an object of Constant from this Catalog, without looking at any nested Catalogs. * *

Returns null if the object doesn't exist. * *

The default implementations always return null. * * @return Constant object if found, null if not found */ protected final Constant getConstant(String name) { return getConstant(name, new FindOptions()); } /** *

Get an object of Constant from this Catalog, without looking at any nested Catalogs. * *

Returns null if the object doesn't exist. * * @return Constant object if found, null if not found */ protected abstract Constant getConstant( @SuppressWarnings("unused") String name, @SuppressWarnings("unused") FindOptions options); /** * Get an object of Table from this Catalog, without * looking at any nested Catalogs. * *

A NULL pointer should be returned if the object doesn't exist. * *

The default implementations always return null. * * @return Table object if found, NULL if not found */ protected final Table getTable(String name) { return getTable(name, new FindOptions()); } /** * NOTE: If FindTable is implemented, there is no need to implement GetTable, as FindTable method * takes precedence over GetTable and is always called first. * *

Get an object of Table from this Catalog, without looking at any nested Catalogs. * *

A NULL pointer should be returned if the object doesn't exist. * *

This is normally overridden in subclasses. The default implementations always return null. * * @return Table object if found, NULL if not found */ protected Table getTable( @SuppressWarnings("unused") String name, @SuppressWarnings("unused") FindOptions options) { return null; } /** * Get an object of Procedure from this Catalog, without * looking at any nested Catalogs. * *

A NULL pointer should be returned if the object doesn't exist. * *

The default implementations always return null. * * @return Procedure object if found, NULL if not found */ protected final Procedure getProcedure(String name) { return getProcedure(name, new FindOptions()); } /** * NOTE: If findProcedure is implemented, there is no need to implement getProcedure, * as findProcedure method takes precedence over getProcedure and is always called first. * *

Get an object of Procedure from this Catalog, without * looking at any nested Catalogs. * *

A NULL pointer should be returned if the object doesn't exist. * *

This is normally overriden in subclasses. The default implementations * always return null. * * @return Procedure object if found, NULL if not found */ protected Procedure getProcedure( @SuppressWarnings("unused") String name, @SuppressWarnings("unused") FindOptions options) { return null; } /** * Get an object of Type from this Catalog, without * looking at any nested Catalogs. * *

A NULL pointer should be returned if the object doesn't exist. * *

The default implementations always return null. * * @return Type object if found, NULL if not found */ protected final Type getType(String name) { return getType(name, new FindOptions()); } /** * NOTE: If FindType is implemented, there is no need to implement GetType, * as FindType method takes precedence over GetType and is always called first. * *

Get an object of Type from this Catalog, without * looking at any nested Catalogs. * *

A NULL pointer should be returned if the object doesn't exist. * *

This is normally overriden in subclasses. The default implementations * always return null. * * @return Type object if found, NULL if not found */ protected Type getType( @SuppressWarnings("unused") String name, @SuppressWarnings("unused") FindOptions options) { return null; } /** * Get an object of Catalog from this Catalog, without * looking at any nested Catalogs. * *

A NULL pointer should be returned if the object doesn't exist. * *

The default implementations always return null. * * @return Catalog object if found, NULL if not found */ protected final Catalog getCatalog(String name) { return getCatalog(name, new FindOptions()); } /** * NOTE: If FindCatalog is implemented, there is no need to implement GetCatalog, * as FindCatalog method takes precedence over GetCatalog and is always called first. * *

Get an object of Catalog from this Catalog, without * looking at any nested Catalogs. * *

A NULL pointer should be returned if the object doesn't exist. * *

This is normally overriden in subclasses. The default implementations * always return null. * * @return Catalog object if found, NULL if not found */ protected Catalog getCatalog( @SuppressWarnings("unused") String name, @SuppressWarnings("unused") FindOptions options) { return null; } private static boolean isIdentifier(String text) { if (text.isEmpty()) { return false; } if (!isLetter(text.charAt(0))) { return false; } for (char c : text.toCharArray()) { if (!isAlphanumeric(c)) { return false; } } return true; } private static boolean isLetter(char c) { return ('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || (c == '_'); } private static boolean isAlphanumeric(char c) { return ('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || ('0' <= c && c <= '9') || (c == '_'); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy